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.
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.
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>
}
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.