Skip to main content
Version: v2

Custom Functions

Introduction

Custom functions are user-defined functions in your Go code that extend Textwire’s capabilities beyond what built-in functions offer. They were introduced in Textwire v2.0.0 by suggestion from @joeyjurjens.

You can attach custom functions to any data type in Textwire and invoke them on a value using the dot operator . followed by the function name.

Custom functions can take any number of arguments and return a value of any type. They can be used to perform any operations and return any value of any type.

info

Custom function were introduced in Textwire v2.0.0 by suggestion from @joeyjurjens in the issue #48

Defining custom functions

To define a custom function, you need to create a function in your Go code. Here is an example of defining a upperLast function that converts the last character of a string to uppercase:

package main

import (
"fmt"
"log"
"unicode"

"github.com/textwire/textwire/v2"
)

func main() {
err := textwire.RegisterStrFunc("_upperLast", func(s string, args ...interface{}) string {
runes := []rune(s)

if len(runes) > 0 {
runes[len(runes)-1] = unicode.ToUpper(runes[len(runes)-1])
}

return string(runes)
})

if err != nil {
log.Fatal(err)
}
}

You can now use the _upperLast function anywhere in your Textwire code.

Prefix custom functions

To avoid conflicts with built-in functions, it’s recommended to prefix your custom functions with an underscore (_). Since built-in functions take precedence over custom ones, defining a custom function with the same name as a built-in function will cause the built-in function to be used. By adding an underscore prefix, you can prevent these conflicts. For example: {{ name._upperLast() }}

Using custom functions

Here is the example of using the upperLast function in Textwire after defining it:

package main

import (
"fmt"
"log"
"unicode"

"github.com/textwire/textwire/v2"
)

func main() {
err := textwire.RegisterStrFunc("upperLast", func(s string, args ...interface{}) string {
runes := []rune(s)

if len(runes) > 0 {
runes[len(runes)-1] = unicode.ToUpper(runes[len(runes)-1])
}

return string(runes)
})

if err != nil {
log.Fatal(err)
}

twCode := "<h1>{{ 'hello'.upperLast() }}</h1>"
output, err := textwire.EvaluateString(twCode, nil)

if err != nil {
log.Fatal(err)
}

fmt.Println(output) // Output: <h1>hellO</h1>
}
Performance warning

Custom functions are not as performant as built-in functions because they are defined in Go code. When you use a custom function, first it needs to be translated from Go to Textwire, which can be slower than built-in functions. Therefore, I would recommend to create an issue with your desired built-in function, so that it can be added to Textwire if you need 100% performance.

The performance difference is not noticeable, but if performance is critical for your application, you should consider creating an issue.