handlebars-go/README.md

1429 lines
31 KiB
Markdown
Raw Normal View History

2023-09-10 10:55:27 +00:00
# handlebars
[![Go Reference](https://pkg.go.dev/badge/github.com/flowchartsman/handlebars.svg)](https://pkg.go.dev/git.reinaldyrafli.com/aldy505/handlebars-go)
This is yet another fork of [github.com/flowchartsman/handlebars](https://github.com/flowchartsman/handlebars).
Handlebars for [golang](https://golang.org) with the same features as [handlebars.js](http://handlebarsjs.com) `3.0`. Hard fork of [Raymond](https://github.com/aymerick/raymond) to modularize and keep up with handlebars development.
![Handlebars Logo](https://github.com/flowchartsman/handlebars/blob/main/handlebars-gopher.png?raw=true "Handlebars")
# Table of Contents
- [Quick Start](#quick-start)
- [Correct Usage](#correct-usage)
- [Context](#context)
- [HTML Escaping](#html-escaping)
- [Helpers](#helpers)
- [Template Helpers](#template-helpers)
- [Built-In Helpers](#built-in-helpers)
- [The `if` block helper](#the-if-block-helper)
- [The `unless` block helper](#the-unless-block-helper)
- [The `each` block helper](#the-each-block-helper)
- [The `with` block helper](#the-with-block-helper)
- [The `lookup` helper](#the-lookup-helper)
- [The `log` helper](#the-log-helper)
- [The `equal` helper](#the-equal-helper)
- [Block Helpers](#block-helpers)
- [Block Evaluation](#block-evaluation)
- [Conditional](#conditional)
- [Else Block Evaluation](#else-block-evaluation)
- [Block Parameters](#block-parameters)
- [Helper Parameters](#helper-parameters)
- [Automatic conversion](#automatic-conversion)
- [Options Argument](#options-argument)
- [Context Values](#context-values)
- [Helper Hash Arguments](#helper-hash-arguments)
- [Private Data](#private-data)
- [Utilites](#utilites)
- [`Str()`](#str)
- [`IsTrue()`](#istrue)
- [Context Functions](#context-functions)
- [Partials](#partials)
- [Template Partials](#template-partials)
- [Global Partials](#global-partials)
- [Dynamic Partials](#dynamic-partials)
- [Partial Contexts](#partial-contexts)
- [Partial Parameters](#partial-parameters)
- [Utility Functions](#utility-functions)
- [Mustache](#mustache)
- [Limitations](#limitations)
- [Handlebars Lexer](#handlebars-lexer)
- [Handlebars Parser](#handlebars-parser)
- [Test](#test)
- [References](#references)
- [Others Implementations](#others-implementations)
## Quick Start
$ go get git.reinaldyrafli.com/aldy505/handlebars-go
The quick and dirty way of rendering a handlebars template:
```go
package main
import (
"fmt"
"git.reinaldyrafli.com/aldy505/handlebars-go"
)
func main() {
tpl := `<div class="entry">
<h1>{{title}}</h1>
<div class="body">
{{body}}
</div>
</div>
`
ctx := map[string]string{
"title": "My New Post",
"body": "This is my first post!",
}
result, err := handlebars.Render(tpl, ctx)
if err != nil {
panic("Please report a bug :)")
}
fmt.Print(result)
}
```
Displays:
```html
<div class="entry">
<h1>My New Post</h1>
<div class="body">
This is my first post!
</div>
</div>
```
Please note that the template will be parsed everytime you call `Render()` function. So you probably want to read the next section.
## Correct Usage
To avoid parsing a template several times, use the `Parse()` and `Exec()` functions:
```go
package main
import (
"fmt"
"git.reinaldyrafli.com/aldy505/handlebars-go
)
func main() {
source := `<div class="entry">
<h1>{{title}}</h1>
<div class="body">
{{body}}
</div>
</div>
`
ctxList := []map[string]string{
{
"title": "My New Post",
"body": "This is my first post!",
},
{
"title": "Here is another post",
"body": "This is my second post!",
},
}
// parse template
tpl, err := handlebars.Parse(source)
if err != nil {
panic(err)
}
for _, ctx := range ctxList {
// render template
result, err := tpl.Exec(ctx)
if err != nil {
panic(err)
}
fmt.Print(result)
}
}
```
Displays:
```html
<div class="entry">
<h1>My New Post</h1>
<div class="body">
This is my first post!
</div>
</div>
<div class="entry">
<h1>Here is another post</h1>
<div class="body">
This is my second post!
</div>
</div>
```
You can use `MustParse()` and `MustExec()` functions if you don't want to deal with errors:
```go
// parse template
tpl := handlebars.MustParse(source)
// render template
result := tpl.MustExec(ctx)
```
## Context
The rendering context can contain any type of values, including `array`, `slice`, `map`, `struct` and `func`.
When using structs, be warned that only exported fields are accessible. However you can access exported fields in template with their lowercase names. For example, both `{{author.firstName}}` and `{{Author.FirstName}}` references give the same result, as long as `Author` and `FirstName` are exported struct fields.
More, you can use the `handlebars` struct tag to specify a template variable name different from the struct field name.
```go
package main
import (
"fmt"
"git.reinaldyrafli.com/aldy505/handlebars-go
)
func main() {
source := `<div class="post">
<h1>By {{author.firstName}} {{author.lastName}}</h1>
<div class="body">{{body}}</div>
<h1>Comments</h1>
{{#each comments}}
<h2>By {{author.firstName}} {{author.lastName}}</h2>
<div class="body">{{content}}</div>
{{/each}}
</div>`
type Person struct {
FirstName string
LastName string
}
type Comment struct {
Author Person
Body string `handlebars:"content"`
}
type Post struct {
Author Person
Body string
Comments []Comment
}
ctx := Post{
Person{"Jean", "Valjean"},
"Life is difficult",
[]Comment{
Comment{
Person{"Marcel", "Beliveau"},
"LOL!",
},
},
}
output := handlebars.MustRender(source, ctx)
fmt.Print(output)
}
```
Output:
```html
<div class="post">
<h1>By Jean Valjean</h1>
<div class="body">Life is difficult</div>
<h1>Comments</h1>
<h2>By Marcel Beliveau</h2>
<div class="body">LOL!</div>
</div>
```
## HTML Escaping
By default, the result of a mustache expression is HTML escaped. Use the triple mustache `{{{` to output unescaped values.
```go
source := `<div class="entry">
<h1>{{title}}</h1>
<div class="body">
{{{body}}}
</div>
</div>
`
ctx := map[string]string{
"title": "All about <p> Tags",
"body": "<p>This is a post about &lt;p&gt; tags</p>",
}
tpl := handlebars.MustParse(source)
result := tpl.MustExec(ctx)
fmt.Print(result)
```
Output:
```html
<div class="entry">
<h1>All about &lt;p&gt; Tags</h1>
<div class="body">
<p>This is a post about &lt;p&gt; tags</p>
</div>
</div>
```
When returning HTML from a helper, you should return a `SafeString` if you don't want it to be escaped by default. When using `SafeString` all unknown or unsafe data should be manually escaped with the `Escape` method.
```go
handlebars.RegisterHelper("link", func(url, text string) handlebars.SafeString {
return handlebars.SafeString("<a href='" + handlebars.Escape(url) + "'>" + handlebars.Escape(text) + "</a>")
})
tpl := handlebars.MustParse("{{link url text}}")
ctx := map[string]string{
"url": "http://www.aymerick.com/",
"text": "This is a <em>cool</em> website",
}
result := tpl.MustExec(ctx)
fmt.Print(result)
```
Output:
```html
<a href='http://www.aymerick.com/'>This is a &lt;em&gt;cool&lt;/em&gt; website</a>
```
## Helpers
Helpers can be accessed from any context in a template. You can register a helper with the `RegisterHelper` function.
For example:
```html
<div class="post">
<h1>By {{fullName author}}</h1>
<div class="body">{{body}}</div>
<h1>Comments</h1>
{{#each comments}}
<h2>By {{fullName author}}</h2>
<div class="body">{{body}}</div>
{{/each}}
</div>
```
With this context and helper:
```go
ctx := map[string]interface{}{
"author": map[string]string{"firstName": "Jean", "lastName": "Valjean"},
"body": "Life is difficult",
"comments": []map[string]interface{}{{
"author": map[string]string{"firstName": "Marcel", "lastName": "Beliveau"},
"body": "LOL!",
}},
}
handlebars.RegisterHelper("fullName", func(person map[string]string) string {
return person["firstName"] + " " + person["lastName"]
})
```
Outputs:
```html
<div class="post">
<h1>By Jean Valjean</h1>
<div class="body">Life is difficult</div>
<h1>Comments</h1>
<h2>By Marcel Beliveau</h2>
<div class="body">LOL!</div>
</div>
```
Helper arguments can be any type.
The following example uses structs instead of maps and produces the same output as the previous one:
```html
<div class="post">
<h1>By {{fullName author}}</h1>
<div class="body">{{body}}</div>
<h1>Comments</h1>
{{#each comments}}
<h2>By {{fullName author}}</h2>
<div class="body">{{body}}</div>
{{/each}}
</div>
```
With this context and helper:
```go
type Post struct {
Author Person
Body string
Comments []Comment
}
type Person struct {
FirstName string
LastName string
}
type Comment struct {
Author Person
Body string
}
ctx := Post{
Person{"Jean", "Valjean"},
"Life is difficult",
[]Comment{
Comment{
Person{"Marcel", "Beliveau"},
"LOL!",
},
},
}
handlebars.RegisterHelper("fullName", func(person Person) string {
return person.FirstName + " " + person.LastName
})
```
You can unregister global helpers with `RemoveHelper` and `RemoveAllHelpers` functions:
```go
handlebars.RemoveHelper("fullname")
```
```go
handlebars.RemoveAllHelpers()
```
### Template Helpers
You can register a helper on a specific template, and in that case that helper will be available to that template only:
```go
tpl := handlebars.MustParse("User: {{fullName user.firstName user.lastName}}")
tpl.RegisterHelper("fullName", func(firstName, lastName string) string {
return firstName + " " + lastName
})
```
### Built-In Helpers
Those built-in helpers are available to all templates.
#### The `if` block helper
You can use the `if` helper to conditionally render a block. If its argument returns `false`, `nil`, `0`, `""`, an empty array, an empty slice or an empty map, then handlebars will not render the block.
```html
<div class="entry">
{{#if author}}
<h1>{{firstName}} {{lastName}}</h1>
{{/if}}
</div>
```
When using a block expression, you can specify a template section to run if the expression returns a falsy value. That section, marked by `{{else}}` is called an "else section".
```html
<div class="entry">
{{#if author}}
<h1>{{firstName}} {{lastName}}</h1>
{{else}}
<h1>Unknown Author</h1>
{{/if}}
</div>
```
You can chain several blocks. For example that template:
```html
{{#if isActive}}
<img src="star.gif" alt="Active">
{{else if isInactive}}
<img src="cry.gif" alt="Inactive">
{{else}}
<img src="wat.gif" alt="Unknown">
{{/if}}
```
With that context:
```go
ctx := map[string]interface{}{
"isActive": false,
"isInactive": false,
}
```
Outputs:
```html
<img src="wat.gif" alt="Unknown">
```
#### The `unless` block helper
You can use the `unless` helper as the inverse of the `if` helper. Its block will be rendered if the expression returns a falsy value.
```html
<div class="entry">
{{#unless license}}
<h3 class="warning">WARNING: This entry does not have a license!</h3>
{{/unless}}
</div>
```
#### The `each` block helper
You can iterate over an array, a slice, a map or a struct instance using this built-in `each` helper. Inside the block, you can use `this` to reference the element being iterated over.
For example:
```html
<ul class="people">
{{#each people}}
<li>{{this}}</li>
{{/each}}
</ul>
```
With this context:
```go
map[string]interface{}{
"people": []string{
"Marcel", "Jean-Claude", "Yvette",
},
}
```
Outputs:
```html
<ul class="people">
<li>Marcel</li>
<li>Jean-Claude</li>
<li>Yvette</li>
</ul>
```
You can optionally provide an `{{else}}` section which will display only when the passed argument is an empty array, an empty slice or an empty map (a `struct` instance is never considered empty).
```html
{{#each paragraphs}}
<p>{{this}}</p>
{{else}}
<p class="empty">No content</p>
{{/each}}
```
When looping through items in `each`, you can optionally reference the current loop index via `{{@index}}`.
```html
{{#each array}}
{{@index}}: {{this}}
{{/each}}
```
Additionally for map and struct instance iteration, `{{@key}}` references the current map key or struct field name:
```html
{{#each map}}
{{@key}}: {{this}}
{{/each}}
```
The first and last steps of iteration are noted via the `@first` and `@last` variables.
#### The `with` block helper
You can shift the context for a section of a template by using the built-in `with` block helper.
```html
<div class="entry">
<h1>{{title}}</h1>
{{#with author}}
<h2>By {{firstName}} {{lastName}}</h2>
{{/with}}
</div>
```
With this context:
```go
map[string]interface{}{
"title": "My first post!",
"author": map[string]string{
"firstName": "Jean",
"lastName": "Valjean",
},
}
```
Outputs:
```html
<div class="entry">
<h1>My first post!</h1>
<h2>By Jean Valjean</h2>
</div>
```
You can optionally provide an `{{else}}` section which will display only when the passed argument is falsy.
```html
{{#with author}}
<p>{{name}}</p>
{{else}}
<p class="empty">No content</p>
{{/with}}
```
#### The `lookup` helper
The `lookup` helper allows for dynamic parameter resolution using handlebars variables.
```html
{{#each bar}}
{{lookup ../foo @index}}
{{/each}}
```
#### The `log` helper
The `log` helper allows for logging while rendering a template.
```html
{{log "Look at me!"}}
```
Note that the handlebars.js `@level` variable is not supported.
#### The `equal` helper
The `equal` helper renders a block if the string version of both arguments are equals.
For example that template:
```html
{{#equal foo "bar"}}foo is bar{{/equal}}
{{#equal foo baz}}foo is the same as baz{{/equal}}
{{#equal nb 0}}nothing{{/equal}}
{{#equal nb 1}}there is one{{/equal}}
{{#equal nb "1"}}everything is stringified before comparison{{/equal}}
```
With that context:
```go
ctx := map[string]interface{}{
"foo": "bar",
"baz": "bar",
"nb": 1,
}
```
Outputs:
```html
foo is bar
foo is the same as baz
there is one
everything is stringified before comparison
```
### Block Helpers
Block helpers make it possible to define custom iterators and other functionality that can invoke the passed block with a new context.
#### Block Evaluation
As an example, let's define a block helper that adds some markup to the wrapped text.
```html
<div class="entry">
<h1>{{title}}</h1>
<div class="body">
{{#bold}}{{body}}{{/bold}}
</div>
</div>
```
The `bold` helper will add markup to make its text bold.
```go
handlebars.RegisterHelper("bold", func(options *handlebars.Options) handlebars.SafeString {
return handlebars.SafeString(`<div class="mybold">` + options.Fn() + "</div>")
})
```
A helper evaluates the block content with current context by calling `options.Fn()`.
If you want to evaluate the block with another context, then use `options.FnWith(ctx)`, like this french version of built-in `with` block helper:
```go
handlebars.RegisterHelper("avec", func(context interface{}, options *handlebars.Options) string {
return options.FnWith(context)
})
```
With that template:
```html
{{#avec obj.text}}{{this}}{{/avec}}
```
#### Conditional
Let's write a french version of `if` block helper:
```go
source := `{{#si yep}}YEP !{{/si}}`
ctx := map[string]interface{}{"yep": true}
handlebars.RegisterHelper("si", func(conditional bool, options *handlebars.Options) string {
if conditional {
return options.Fn()
}
return ""
})
```
Note that as the first parameter of the helper is typed as `bool` an automatic conversion is made if corresponding context value is not a boolean. So this helper works with that context too:
```go
ctx := map[string]interface{}{"yep": "message"}
```
Here, `"message"` is converted to `true` because it is an non-empty string. See `IsTrue()` function for more informations on boolean conversion.
#### Else Block Evaluation
We can enhance the `si` block helper to evaluate the `else block` by calling `options.Inverse()` if conditional is false:
```go
source := `{{#si yep}}YEP !{{else}}NOP !{{/si}}`
ctx := map[string]interface{}{"yep": false}
handlebars.RegisterHelper("si", func(conditional bool, options *handlebars.Options) string {
if conditional {
return options.Fn()
}
return options.Inverse()
})
```
Outputs:
```
NOP !
```
#### Block Parameters
It's possible to receive named parameters from supporting helpers.
```html
{{#each users as |user userId|}}
Id: {{userId}} Name: {{user.name}}
{{/each}}
```
In this particular example, `user` will have the same value as the current context and `userId` will have the index/key value for the iteration.
This allows for nested helpers to avoid name conflicts.
For example:
```html
{{#each users as |user userId|}}
{{#each user.books as |book bookId|}}
User: {{userId}} Book: {{bookId}}
{{/each}}
{{/each}}
```
With this context:
```go
ctx := map[string]interface{}{
"users": map[string]interface{}{
"marcel": map[string]interface{}{
"books": map[string]interface{}{
"book1": "My first book",
"book2": "My second book",
},
},
"didier": map[string]interface{}{
"books": map[string]interface{}{
"bookA": "Good book",
"bookB": "Bad book",
},
},
},
}
```
Outputs:
```html
User: marcel Book: book1
User: marcel Book: book2
User: didier Book: bookA
User: didier Book: bookB
```
As you can see, the second block parameter is the map key. When using structs, it is the struct field name.
When using arrays and slices, the second parameter is element index:
```go
ctx := map[string]interface{}{
"users": []map[string]interface{}{
{
"id": "marcel",
"books": []map[string]interface{}{
{"id": "book1", "title": "My first book"},
{"id": "book2", "title": "My second book"},
},
},
{
"id": "didier",
"books": []map[string]interface{}{
{"id": "bookA", "title": "Good book"},
{"id": "bookB", "title": "Bad book"},
},
},
},
}
```
Outputs:
```html
User: 0 Book: 0
User: 0 Book: 1
User: 1 Book: 0
User: 1 Book: 1
```
### Helper Parameters
When calling a helper in a template, handlebars expects the same number of arguments as the number of helper function parameters.
So this template:
```html
{{add a}}
```
With this helper:
```go
handlebars.RegisterHelper("add", func(val1, val2 int) string {
return strconv.Itoa(val1 + val2)
})
```
Will simply panics, because we call the helper with one argument whereas it expects two.
#### Automatic conversion
Let's create a `concat` helper that expects two strings and concat them:
```go
source := `{{concat a b}}`
ctx := map[string]interface{}{
"a": "Jean",
"b": "Valjean",
}
handlebars.RegisterHelper("concat", func(val1, val2 string) string {
return val1 + " " + val2
})
```
Everything goes well, two strings are passed as arguments to the helper that outputs:
```html
Jean VALJEAN
```
But what happens if there is another type than `string` in the context ? For example:
```go
ctx := map[string]interface{}{
"a": 10,
"b": "Valjean",
}
```
Actually, handlebars perfoms automatic string conversion. So because the first parameter of the helper is typed as `string`, the first argument will be converted from the `10` integer to `"10"`, and the helper outputs:
```html
10 VALJEAN
```
Note that this kind of automatic conversion is done with `bool` type too, thanks to the `IsTrue()` function.
### Options Argument
If a helper needs the `Options` argument, just add it at the end of helper parameters:
```go
handlebars.RegisterHelper("add", func(val1, val2 int, options *handlebars.Options) string {
return strconv.Itoa(val1 + val2) + " " + options.ValueStr("bananas")
})
```
Thanks to the `options` argument, helpers have access to the current evaluation context, to the `Hash` arguments, and they can manipulate the private data variables.
The `Options` argument is even necessary for Block Helpers to evaluate block and "else block".
#### Context Values
Helpers fetch current context values with `options.Value()` and `options.ValuesStr()`.
`Value()` returns an `interface{}` and lets the helper do the type assertions whereas `ValueStr()` automatically converts the value to a `string`.
For example:
```go
source := `{{concat a b}}`
ctx := map[string]interface{}{
"a": "Marcel",
"b": "Beliveau",
"suffix": "FOREVER !",
}
handlebars.RegisterHelper("concat", func(val1, val2 string, options *handlebars.Options) string {
return val1 + " " + val2 + " " + options.ValueStr("suffix")
})
```
Outputs:
```html
Marcel Beliveau FOREVER !
```
Helpers can get the entire current context with `options.Ctx()` that returns an `interface{}`.
#### Helper Hash Arguments
Helpers access hash arguments with `options.HashProp()` and `options.HashStr()`.
`HashProp()` returns an `interface{}` and lets the helper do the type assertions whereas `HashStr()` automatically converts the value to a `string`.
For example:
```go
source := `{{concat suffix first=a second=b}}`
ctx := map[string]interface{}{
"a": "Marcel",
"b": "Beliveau",
"suffix": "FOREVER !",
}
handlebars.RegisterHelper("concat", func(suffix string, options *handlebars.Options) string {
return options.HashStr("first") + " " + options.HashStr("second") + " " + suffix
})
```
Outputs:
```html
Marcel Beliveau FOREVER !
```
Helpers can get the full hash with `options.Hash()` that returns a `map[string]interface{}`.
#### Private Data
Helpers access private data variables with `options.Data()` and `options.DataStr()`.
`Data()` returns an `interface{}` and lets the helper do the type assertions whereas `DataStr()` automatically converts the value to a `string`.
Helpers can get the entire current data frame with `options.DataFrame()` that returns a `*DataFrame`.
For helpers that need to inject their own private data frame, use `options.NewDataFrame()` to create the frame and `options.FnData()` to evaluate the block with that frame.
For example:
```go
source := `{{#voodoo kind=a}}Voodoo is {{@magix}}{{/voodoo}}`
ctx := map[string]interface{}{
"a": "awesome",
}
handlebars.RegisterHelper("voodoo", func(options *handlebars.Options) string {
// create data frame with @magix data
frame := options.NewDataFrame()
frame.Set("magix", options.HashProp("kind"))
// evaluates block with new data frame
return options.FnData(frame)
})
```
Helpers that need to evaluate the block with a private data frame and a new context can call `options.FnCtxData()`.
### Utilites
In addition to `Escape()`, handlebars provides utility functions that can be usefull for helpers.
#### `Str()`
`Str()` converts its parameter to a `string`.
Booleans:
```go
handlebars.Str(3) + " foos and " + handlebars.Str(-1.25) + " bars"
// Outputs: "3 foos and -1.25 bars"
```
Numbers:
``` go
"everything is " + handlebars.Str(true) + " and nothing is " + handlebars.Str(false)
// Outputs: "everything is true and nothing is false"
```
Maps:
```go
handlebars.Str(map[string]string{"foo": "bar"})
// Outputs: "map[foo:bar]"
```
Arrays and Slices:
```go
handlebars.Str([]interface{}{true, 10, "foo", 5, "bar"})
// Outputs: "true10foo5bar"
```
#### `IsTrue()`
`IsTrue()` returns the truthy version of its parameter.
It returns `false` when parameter is either:
- an empty array
- an empty slice
- an empty map
- `""`
- `nil`
- `0`
- `false`
For all others values, `IsTrue()` returns `true`.
## Context Functions
In addition to helpers, lambdas found in context are evaluated.
For example, that template and context:
```go
source := "I {{feeling}} you"
ctx := map[string]interface{}{
"feeling": func() string {
rand.Seed(time.Now().UTC().UnixNano())
feelings := []string{"hate", "love"}
return feelings[rand.Intn(len(feelings))]
},
}
```
Randomly renders `I hate you` or `I love you`.
Those context functions behave like helper functions: they can be called with parameters and they can have an `Options` argument.
## Partials
### Template Partials
You can register template partials before execution:
```go
tpl := handlebars.MustParse("{{> foo}} baz")
tpl.RegisterPartial("foo", "<span>bar</span>")
result := tpl.MustExec(nil)
fmt.Print(result)
```
Output:
```html
<span>bar</span> baz
```
You can register several partials at once:
```go
tpl := handlebars.MustParse("{{> foo}} and {{> baz}}")
tpl.RegisterPartials(map[string]string{
"foo": "<span>bar</span>",
"baz": "<span>bat</span>",
})
result := tpl.MustExec(nil)
fmt.Print(result)
```
Output:
```html
<span>bar</span> and <span>bat</span>
```
### Global Partials
You can registers global partials that will be accessible by all templates:
```go
handlebars.RegisterPartial("foo", "<span>bar</span>")
tpl := handlebars.MustParse("{{> foo}} baz")
result := tpl.MustExec(nil)
fmt.Print(result)
```
Or:
```go
handlebars.RegisterPartials(map[string]string{
"foo": "<span>bar</span>",
"baz": "<span>bat</span>",
})
tpl := handlebars.MustParse("{{> foo}} and {{> baz}}")
result := tpl.MustExec(nil)
fmt.Print(result)
```
### Dynamic Partials
It's possible to dynamically select the partial to be executed by using sub expression syntax.
For example, that template randomly evaluates the `foo` or `baz` partial:
```go
tpl := handlebars.MustParse("{{> (whichPartial) }}")
tpl.RegisterPartials(map[string]string{
"foo": "<span>bar</span>",
"baz": "<span>bat</span>",
})
ctx := map[string]interface{}{
"whichPartial": func() string {
rand.Seed(time.Now().UTC().UnixNano())
names := []string{"foo", "baz"}
return names[rand.Intn(len(names))]
},
}
result := tpl.MustExec(ctx)
fmt.Print(result)
```
### Partial Contexts
It's possible to execute partials on a custom context by passing in the context to the partial call.
For example:
```go
tpl := handlebars.MustParse("User: {{> userDetails user }}")
tpl.RegisterPartial("userDetails", "{{firstname}} {{lastname}}")
ctx := map[string]interface{}{
"user": map[string]string{
"firstname": "Jean",
"lastname": "Valjean",
},
}
result := tpl.MustExec(ctx)
fmt.Print(result)
```
Displays:
```html
User: Jean Valjean
```
### Partial Parameters
Custom data can be passed to partials through hash parameters.
For example:
```go
tpl := handlebars.MustParse("{{> myPartial name=hero }}")
tpl.RegisterPartial("myPartial", "My hero is {{name}}")
ctx := map[string]interface{}{
"hero": "Goldorak",
}
result := tpl.MustExec(ctx)
fmt.Print(result)
```
Displays:
```html
My hero is Goldorak
```
## Utility Functions
You can use following utility fuctions to parse and register partials from files:
- `ParseFile()` - reads a file and return parsed template
- `Template.RegisterPartialFile()` - reads a file and registers its content as a partial with given name
- `Template.RegisterPartialFiles()` - reads several files and registers them as partials, the filename base is used as the partial name
## Mustache
Handlebars is a superset of [mustache](https://mustache.github.io) but it differs on those points:
- Alternative delimiters are not supported
- There is no recursive lookup
## Limitations
These handlebars options are currently NOT implemented:
- `compat` - enables recursive field lookup
- `knownHelpers` - list of helpers that are known to exist (truthy) at template execution time
- `knownHelpersOnly` - allows further optimizations based on the known helpers list
- `trackIds` - include the id names used to resolve parameters for helpers
- `noEscape` - disables HTML escaping globally
- `strict` - templates will throw rather than silently ignore missing fields
- `assumeObjects` - removes object existence checks when traversing paths
- `preventIndent` - disables the auto-indententation of nested partials
- `stringParams` - resolves a parameter to it's name if the value isn't present in the context stack
These handlebars features are currently NOT implemented:
- raw block content is not passed as a parameter to helper
- `blockHelperMissing` - helper called when a helper can not be directly resolved
- `helperMissing` - helper called when a potential helper expression was not found
- `@contextPath` - value set in `trackIds` mode that records the lookup path for the current context
- `@level` - log level
## Handlebars Lexer
You should not use the lexer directly, but for your information here is an example:
```go
package main
import (
"fmt"
"git.reinaldyrafli.com/aldy505/handlebars-go/lexer"
)
func main() {
source := "You know {{nothing}} John Snow"
output := ""
lex := lexer.Scan(source)
for {
// consume next token
token := lex.NextToken()
output += fmt.Sprintf(" %s", token)
// stops when all tokens have been consumed, or on error
if token.Kind == lexer.TokenEOF || token.Kind == lexer.TokenError {
break
}
}
fmt.Print(output)
}
```
Outputs:
```
Content{"You know "} Open{"{{"} ID{"nothing"} Close{"}}"} Content{" John Snow"} EOF
```
## Handlebars Parser
You should not use the parser directly, but for your information here is an example:
```go
package main
import (
"fmt"
"git.reinaldyrafli.com/aldy505/handlebars-go/ast"
"git.reinaldyrafli.com/aldy505/handlebars-go/parser"
)
fu nc main() {
source := "You know {{nothing}} John Snow"
// parse template
program, err := parser.Parse(source)
if err != nil {
panic(err)
}
// print AST
output := ast.Print(program)
fmt.Print(output)
}
```
Outputs:
```
CONTENT[ 'You know ' ]
{{ PATH:nothing [] }}
CONTENT[ ' John Snow' ]
```
## Test
First, fetch mustache tests:
$ git submodule update --init
To run all tests:
$ go test ./...
To filter tests:
$ go test -run="Partials"
To run all test and all benchmarks:
$ go test -bench . ./...
To test with race detection:
$ go test -race ./...
## References
- <http://handlebarsjs.com/>
- <https://mustache.github.io/mustache.5.html>
- <https://github.com/golang/go/tree/master/src/text/template>
- <https://www.youtube.com/watch?v=HxaD_trXwRE>
## Others Implementations
- [handlebars.js](http://handlebarsjs.com) - javascript
- [handlebars.java](https://github.com/jknack/handlebars.java) - java
- [handlebars.rb](https://github.com/cowboyd/handlebars.rb) - ruby
- [handlebars.php](https://github.com/XaminProject/handlebars.php) - php
- [handlebars-objc](https://github.com/Bertrand/handlebars-objc) - Objective C
- [rumblebars](https://github.com/nicolas-cherel/rumblebars) - rust