Well-documented Go functions for building HTMX attributes.
htmx is a powerful and simple tool for building dynamic, server-rendered web applications. It also pairs particularly well with Templ (a JSX-like template language for Go) and Gomponents (a Go-native view library).
However, when using it I have to have the docs open, to look up the specifics of each modifier. I wanted the simplicity of HTMX, the editor support of Go, and beautiful integration with Templ, without sacrificing performance. I built typed-htmx-go.
hx.NewTempl()
provides an hx
struct that exposes all documented HTMX attributes as Go functions, and templ.Attributes to be spread into a Templ element. hx.NewGomponents()
returns an hx
struct that exposes attributes as functions that return g.Node
attributes instead.. You can also support other templating libraries by simply passing an attr
function to htmx.NewHX(attr)
.
Each function and option includes a Godoc comment copied from the extensive HTMX docs, so you can access that documentation right from the comfort of your editor.
go get github.com/will-wow/typed-htmx-go
import (
"time"
"github.com/will-wow/typed-htmx-go/htmx"
"github.com/will-wow/typed-htmx-go/htmx/trigger"
"github.com/will-wow/typed-htmx-go/htmx/swap"
)
var hx = htmx.NewTempl()
templ search() {
<h3>
Search Contacts
<span class="htmx-indicator">
<img src="/static/img/bars.svg"/> Searching...
</span>
</h3>
<input
type="search"
name="search"
placeholder="Begin Typing To Search Users..."
{ hx.Post("/examples/templ/active-search/search/")... }
{ hx.TriggerExtended(
trigger.On("input").Changed().Delay(time.Millisecond * 500),
trigger.On("search"),
)... }
{ hx.Target("#search-results")... }
{ hx.Swap(swap.OuterHTML)... }
{ hx.Indicator(".htmx-indicator")... }
/>
<table>
<thead>
<tr>
<th>First Name</th>
<th>Last Name</th>
<th>Email</th>
</tr>
</thead>
<tbody id="search-results"></tbody>
</table>
}
htmx includes a set of extensions out of the box that address common developer needs. These extensions are tested against htmx in each distribution.
While you can always use any extension by adding standard HTML attributes, typed-htmx-go
has typed support for some extensions.
These extensions each have their own package, and expose function that take a configured hx
as a first parameter, and return a full attribute.
hx
also includes an hx.Ext()
method to register an extension on an element (ie: { hx.Ext(classtools.Extension)... }
instead of hx-ext="class-tools"
).
var hx = htmx.NewTempl()
<head>
<script src="https://unpkg.com/[email protected]/dist/ext/class-tools.js"></script>
</head>
...
<div { hx.Ext(classtools.Extension)... }>
<div { classtools.Classes(hx, classtools.Add("foo", time.Millisecond*100))... } />
<div class="bar" { classtools.ClassesParallel(hx, []classtools.Run{
{classtools.Remove("bar", time.Second)},
{classtools.Add("foo", time.Second)},
})... } />
</div>
Some of the more complex extensions supported:
See htmx/ext for a full list of extensions.
Usage examples are in examples (hosted at typed-htmx-go.vercel.app)
These are mostly ported from the HTMX examples, but include a Templ and Gomponents implementation, and working server code to borrow from.
typed-htmx-go
strives to keep up with HTMX releases. It currently supports HTMX v1.9.10
.
The project has some specific goals that drive the API.
Every documented HTMX attribute and modifier should have a corresponding Go function. If it's missing something please submit an issue or a PR! And in the meantime, you can always drop back to a raw HTML attribute.
Many HTMX attributes (like hx-swap
and hx-trigger
) support a complex syntax of methods, modifiers, and selectors in the attribute string (like hx-trigger='click[isActive] consume from:(#parent > #child) queue:first target:#element'
).
That's necessary for a tool that embeds in standard HTML attributes, but it requires a lot of studying the docs to get exactly right.
hx
strives to provide typed builders that ensure you're passing the right options to the right modifiers.
For instance, many attributes (like hx-target and hx-include) support "extended selectors", which is either a standard CSS selector, or some non-standard keyword like this
or closest
. But different attributes support different non-standard selectors, so they have their own types; HX.Target()
takes an htmx.TargetSelector
that supports only this
, next
, or previous
, plus optionally any of the standard relative modifiers like closest
or find
, while HX.Include()
takes an htmx.IncludeSelector
that only allows the closest
modifier, and no non-standard selectors. Since they also take any arbitrary CSS selector, you can pass in any string, but sticking to the provided types when available makes it easier to make sure you've got valid selectors.
Example:
hx.Target("#element")
hx.Target(htmx.TargetRelative(htmx.Next, "#element"))
hx.Target(htmx.TargetNext)
hx.Include("#element")
hx.Include(IncludeThis)
hx.Include(htmx.IncludeRelative(htmx.Next, "#element"))
hx.Include(htmx.TargetNext) // Invalid: cannot use TargetNext (constant "next" of type TargetSelector) as IncludeSelector value in argument to hx.Include
The HTMX References are through and readable (otherwise this project wouldn't have been possible!) However, having those docs at your fingertips as you write, instead of in a separate tab, is even better.
hx
strives to have a Go-adjusted copy of every line of documentation from the HTMX References, including examples, included in the godocs of functions and options, and an example test.
Note: Documentation is in progress. If you see something missing, please submit a PR!
hx
is built to support any Go HTML templating library that can take attributes as some data type. The two supported out of the box are Templ and Gomponents, but it should be easy work with other libraries. And if you do use typed-htmx-go
with another library, please submit a PR to add official support!
To handle the different types (Templ expects a Templ.Attributes map[string]any
, and Gomponents wants a g.Node
with a Render
method), the htmx.NewHX
constructor takes an attr
function that takes an HTMX attribute and an any
value, and returns some type T. That means you can construct an hx
that returns Templ.Attributes
from every attribute function, one that returns a g.Node
, and another that returns whatever your library expects.
For ease of use, you should create a private var hx
in your template packages, like so:
var hx = htmx.NewTempl()
templ MyDiv() {
<button { hx.Get("/some/path")... } />
}
As much as possible, it should be the case that if you know HTMX, you can use hx
, and using hx
should prepare you to use raw HTMX. That means that attributes functions should match their HTMX counterparts, names should match terms in the docs, and arguments should occur in the order they are printed in the HTML.
This also means that written hx
attributes should look like HTMX attributes. So if in HTMX you would write:
<form
method="GET"
action="/page"
hx-get="/page"
hx-target="body"
hx-replace-url="true"
hx-swap="scroll:#search-results:top swap:1s"
></form>
The hx
equivalent should take the same names and values in the same order:
// templ
<form
method="GET"
action="/page"
{ hx.Get("/page")... }
{ hx.Target("body")... }
{ hx.ReplaceURL(true)... }
{ hx.Swap(
swap.New().
ScrollElement("#search-results", swap.Top).
Swap(time.Second),
)... }
>
// gomponents
Form(
Method("GET"),
Action("/page"),
hx.Get("/page"),
hx.Target("body"),
hx.ReplaceURL(true),
hx.Swap(
swap.New().
ScrollElement("#search-results", swap.Top).
Swap(time.Second),
),
)
Every attribute function should have a test to make sure it's printing valid HTMX. And every function and option should include an example test, to make it easy to see usage in the godocs. These are also a good opportunity to try out the API and make sure it's ergonomic in practice.
go install github.com/go-task/task/v3/cmd/task@latest
task tools
task ready
VERSION="0.0.0" task publish