Interactive UI Components for Django using htmx
Add djhtmx
to your INSTALLED_APPS
and install the Middleware as the last one
of the list:
INSTALLED_APPS = [
...
"djhtmx",
...
]
MIDDLEWARE = [
...,
"djhtmx.Middleware",
]
Expose the HTTP endpoint in your urls.py
as you wish, you can use any path you want.
from django.urls import path, include
urlpatterns = [
# ...
path("_htmx/", include("djhtmx.urls")),
# ...
]
In your base template you need to load the necessary scripts to make this work
{% load htmx %}
<!doctype html>
<html>
<head>
{% htmx-headers %}
</head>
</html>
This app will look for htmx.py
files in your app and registers all components found there, but if you load any module where you have components manually when Django boots up, that also works.
from djhtmx.component import PydanticComponent
class Counter(PydanticComponent):
_template_name = "Counter.html"
counter: int = 0
def inc(self, amount: int = 1):
self.counter += amount
The inc
event handler is ready to be called from the front-end to respond to an event.
The counter.html
would be:
{% load htmx %}
<div {% hx-tag %}>
{{ counter }}
<button {% on "inc" %}>+</button>
<button {% on "inc" amount=2 %}>+2</button>
</div>
When the event is dispatched to the back-end the component state is reconstructed, the event handler called and then the full component is rendered back to the front-end.
Now use the component in any of your html templates, by passing attributes that are part of the component state:
{% load htmx %}
Counters: <br />
{% htmx "Counter" %} Counter with init value 3:<br />
{% htmx "Counter" counter=3 %}
All components have a self.user
representing the current logged in user or None
in case the user is anonymous. If you wanna make sure your user is properly validated and enforced. You need to create a base component and annotate the right user:
from typing import Annotated
from pydantic import Field
from djhtmx.component import PydanticComponent
class BaseComponent(PydanticComponent, public=False):
user: Annotated[User, Field(exclude=True)]
class Counter(BaseComponent):
_template_name = "Counter.html"
counter: int = 0
def inc(self, amount: int = 1):
self.counter += amount
These are components that can't be instantiated using {% htmx "ComponentName" %}
because they are used to create some abstraction and reuse code.
Pass public=False
in their declaration
class BaseComponent(PydanticComponent, public=False):
...
Components can contain components inside to decompose the behavior in more granular and specialized parts, for this you don't have to do anything but to a component inside the template of other component....
class Items(PydanticComponent):
_template_name = "Items.html"
def items(self):
return Item.objects.all()
class ItemEntry(PydanticComponent):
...
item: Item
is_open: bool = False
...
Items.html
:
{% load htmx %}
<ul {% hx-tag %}>
{% for item in items %}
{% htmx "ItemEntry" item=item %}
{% endfor %}
</ul>
In this case every time there is a render of the parent component all children components will also be re-rendered.
How can you preserve the state in the child components if there were some of them that were already had is_open = True
? The state that is not passed directly during instantiation to the component is retrieved from the session, but the component needs to have consistent id. To do this you have to pass an id
to the component.
Items.html
:
{% load htmx %}
<ul {% hx-tag %}>
{% for item in items %}
{% htmx "ItemEntry" id="item-"|add:item.id item=item %}
{% endfor %}
</ul>
If you want some component to load lazily, you pass lazy=True
where it is being instantiated.
Items.html
:
{% load htmx %}
<ul {% hx-tag %}>
{% for item in items %}
{% htmx "ItemEntry" id="item-"|add:item.id item=item lazy=True %}
{% endfor %}
</ul>
This makes the component to be initialized, but instead of rendering the template in _template_name
the template defined in _template_name_lazy
will be rendered (you can override this). When the component arrives to the front-end it will trigger an event to render it self.
When sending an event to the back-end sometimes you can pass the parameters explicitly to the event handler, and sometimes these are inputs the user is typing stuff on. The value of those inputs are passed implicitly if they nave a name="..."
attribute.
class Component(PydanticComponent):
...
def create(self, name: str, is_active: bool = False):
Item.objects.create(name=name, is_active=is_active)
{% load htmx %}
<form {% hx-tag %} {% on "submit" "create" %}>
<input type="text" name="name">
<input type="checkbox" name="is_active">
<button type="submit">Create!</button>
</form>
The parameters of any event handler are always converted by pydantic to the annotated types. It's suggested to properly annotate the event handler parameter with the more restrictive types you can.
Suppose that you have a multiple choice list and you want to select multiple options, you can do this by suffixing the name with []
as in choices[]
:
class DeleteSelection(PydanticComponent):
@property
def items(self):
return self.filter(owner=self.user)
def delete(self, selected: list[UUID] | None = None):
if selected:
self.items.filter(id__in=selected).delete()
{% load htmx %}
<form {% hx-tag %} {% on "submit" "delete" %}>
<h1>Select items to be deleted</h1>
{% for item in items %}
<p>
<input
type="checkbox"
name="selected[]"
value="{{ item.id }}"
id="checkbox-{{ item.id }}"
/>
<label for="checkbox-{{ item.id }}">{{ item.name}}</label>
</p>
{% endfor %}
<p><button type="submit">Delete selected</button></p>
</form>
Each event handler in a component can yield commands for the library to execute. These are useful for skipping the default component render, redirecting the user, remove the component from the front-end, and updating other components.
Wanna redirect the user to some object url:
-
If you have the url directly you can
yield Redirect(url)
. -
If you want Django to resolve the url automatically use:
yield Redirect.to(obj, *args, **kwargs)
as you would usedjango.shortcuts.resolve_url
.
from djhtmx.component import PydanticComponent, Redirect
class Component(PydanticComponent):
...
def create(self, name: str):
item = Item.objects.create(name=name)
yield Redirect.to(item)
If you want to open the url in a new url use the yield Open...
command with similar syntax to Redirect
.
Sometimes you want to remove the component when it responds to an event, for that you need to yield Destroy(component_id: str)
. You can also use this to remove any other component if you know their id.
from djhtmx.component import PydanticComponent, Destroy
class Notification(PydanticComponent):
...
def close(self):
yield Destroy(self.id)
Sometimes when reacting to a front-end event is handy to skip the default render of the current component, to achieve this do:
from djhtmx.component import PydanticComponent, Redirect
class Component(PydanticComponent):
...
def do_something(self):
...
yield SkipRender(self)
Sometimes you don't want to do a full component render, but a partial one. Specially if the user if typing somewhere to filter items and you don't wanna interfere with the user typing or focus. Here is the technique to do that:
from djhtmx.component import PydanticComponent, Render
class SmartFilter(PydanticComponent):
_template_name = "SmartFilter.html"
query: str = ""
@property
def items(self):
items = Item.objects.all()
if self.query:
items = items.filter(name__icontains=self.query)
return items
def filter(self, query: str):
self.query = query.trim()
yield Render(self, template="SmartFilter_list.html")
SmartFilter.html
:
{% load htmx %}
<div {% hx-tag %}>
<input type="text" name="query" value="{{ query }}">
{% include "SmartFilter_list.html" %}
</div>
SmartFilter_list.html
:
<ul {% oob "list" %}>
{% for item in items %}
<li><a href="{{ item.get_absolute_url }}">{{ item }}</a></li>
{% empty %}
<li>Nothing found!</li>
{% endfor %}
</ul>
- Split the component in multiple templates, the main one and the partial ones.
- For readability prefix the name of the partials with the name of the parent.
- The partials need a single root HTML Element with an id and the
{% oob %}
tag next to it. - When you wanna do the partial render you have to
yield Render(self, template=...)
with the name of the partial template, this will automatically skip the default full render and render the component with that partial template.
Coming back to the previous example let's say that we want to persist the state of the query
in the URL, so in case the user refreshes the page or shares the link the state of the component is partially restored. For do the following:
from typing import Annotated
from djhtmx.component import PydanticComponent
from djhtmx.query import Query
class SmartFilter(PydanticComponent):
...
query: Annotated[str, Query("query")] = ""
...
Annotating with Query causes that if the state of the query is not explicitly passed to the component during instantiation it is taken from the query string of the current URL.
There can be multiple components subscribed to the same query parameter or to individual ones.
If you want now you can split this component in two, each with their own template:
from typing import Annotated
from djhtmx.component import PydanticComponent, SkipRender
from djhtmx.query import Query
class SmartFilter(PydanticComponent):
_template_name = "SmartFilter.html"
query: Annotated[str, Query("query")] = ""
def filter(self, query: str):
self.query = query.trim()
yield SkipRender(self)
class SmartList(PydanticComponent):
_template_name = "SmartList.html"
query: Annotated[str, Query("query")] = ""
@property
def items(self):
items = Item.objects.all()
if self.query:
items = items.filter(name__icontains=self.query)
return items
Instantiate next to each other:
<div>
...
{% htmx "SmartFilter" %}
{% htmx "SmartList" %}
...
</div>
When the filter mutates the query
, the URL is updated and the SmartList
is awaken because the both point to the same query parameter, and will be re-rendered.
Sometimes you modify a model and you want not just the current component to react to this, but also trigger re-renders of other components that are not directly related to the current one. For this signals are very convenient. These are strings that represent topics you can subscribe a component to and make sure it is rendered in case any of the topics it subscribed to is triggered.
Signal formats:
app_label.modelname
: Some mutation happened to a model instance of this kindapp_label.modelname.instance_pk
: Some mutation happened to this precise instance of modelapp_label.modelname.instance_pk.created
: This instance was createdapp_label.modelname.instance_pk.updated
: This instance was updatedapp_label.modelname.instance_pk.deleted
: This instance was deleted
When an instance is modified the mode specific and not so specific signals are triggered. Together with them some other signals to related models are triggered.
Example: if we have a Todo list app with the models:
class TodoList(Model):
...
class Item(Model):
todo_list = ForeignKey(TodoList, related_name="items")
And from the list with id 932
you take a item with id 123
and update it all this signals will be triggered:
todoapp.item
todoapp.item.123
todoapp.item.123.updated
todoapp.todolist.932.items
todoapp.todolist.932.items.updated
Let's say you wanna count how many items there are in certain Todo list, but your component does not receive an update when the list is updated because it is out of it. You can do this.
from djhtmx.component import PydanticComponent
class ItemCounter(PydanticComponent):
todo_list: TodoList
def subscriptions(self):
return {
f"todoapp.todolist.{self.todo_list.id}.items.deleted",
f"todoapp.todolist.{self.todo_list.id}.items.created",
}
def count(self):
return self.todo_list.items.count()
This will make this component re-render every time an item is added or removed from the list todo_list
.
Sometimes is handy to notify components in the same session that something changed and they need to perform the corresponding update and Query()
nor Signals are very convenient for this. In this case you can Emit
events and listen to them.
Find here an implementation of SmartFilter
and SmartItem
using this mechanism:
from dataclasses import dataclass
from djhtmx.component import PydanticComponent, SkipRender, Emit
@dataclass(slots=True)
class QueryChanged:
query: str
class SmartFilter(PydanticComponent):
_template_name = "SmartFilter.html"
query: str = ""
def filter(self, query: str):
self.query = query.trim()
yield Emit(QueryChanged(query))
yield SkipRender(self)
class SmartList(PydanticComponent):
_template_name = "SmartList.html"
query: str = ""
def _handle_event(self, event: QueryChanged):
self.query = event.query
@property
def items(self):
items = Item.objects.all()
if self.query:
items = items.filter(name__icontains=self.query)
return items
The library will look in all components if they define _handle_event(event: ...)
and based on the annotation of event
subscribe them to those events. This annotation can be a single type or a Union
with multiple even types.
Let's say that we are making the TODO list app and we want that when a new item is added to the list there is not a full re-render of the whole list, just that the Component handling a single Item is added to the list.
from djhtmx.component import PydanticComponent, SkipRender, BuildAndRender
class TodoListComponent(PydanticComponent):
_template_name = "TodoListComponent.html"
todo_list: TodoList
def create(self, name: str):
item = self.todo_list.items.create(name=name)
yield BuildAndRender.prepend(
f"{self.id} .list",
ItemComponent,
id=f"item-{item.id}",
item=item,
)
yield SkipRender(self)
class ItemComponent(PydanticComponent):
...
item: Item
...
TodoListComponent.html
:
{% load htmx %}
<div {% hx-tag %}>
<form {% on "submit" "create" %}>
<input type="text" name="name">
</form>
<ul class="list">
{% for item in items %}
{% htmx "ItemComponent" id="item-"|add:item.id item=item %}
{% endfor %}
</ul>
</div>
Use the BuildAndRender.<helper>(target: str, ...)
to send a component to be inserted somewhere or updated.
Let's say we want to put the focus in an input that inside the new ItemComponent rendered, for this use yield Focus(target)
from djhtmx.component import PydanticComponent, SkipRender, BuildAndRender, Focus
class TodoListComponent(PydanticComponent):
_template_name = "TodoListComponent.html"
todo_list: TodoList
def create(self, name: str):
item = self.todo_list.items.create(name=name)
item_id = f"item-{item.id}"
yield BuildAndRender.prepend(
f"{self.id} .list",
ItemComponent,
id=item_id,
item=item,
)
yield Focus(f"#{item_id} input")
yield SkipRender(self)
Suppose you have a rich JavaScript library (graphs, maps, or anything...) in the front-end and you want to communicate something to it because it is subscribed to some dome event. For that you can use yield DispatchDOMEvent(target, event, detail, ....)
from djhtmx.component import PydanticComponent, DispatchDOMEvent
class TodoListComponent(PydanticComponent):
_template_name = "TodoListComponent.html"
todo_list: TodoList
def create(self, name: str):
item = self.todo_list.items.create(name=name)
yield DispatchDOMEvent(
"#leaflet-map",
"new-item",
{"id": item.id, "name": item.name, "geojson": item.geojson}
)
This will trigger that event in the front-end when the request arrives allowing rich JavaScript components to react accordingly without full re-render.
{% htmx-headers %}
: put it inside your<header></header>
to load the right scripts and configuration.
<header>
{% htmx-headers %}
</header>
{% htmx <ComponentName: str> **kwargs %}
: instantiates and inserts the result of rendering that component with those initialization parameters.
<div>
{% htmx 'Button' document=document name='Save Document' is_primary=True %}
</div>
{% hx-tag %}
: goes in the root HTML Element of a component template, it sets the componentid
and some other basic configuration details of the component.
<button {% hx-tag %}>
...
</button>
{% oob <LocalId: str> %}
: goes in the root HTML Element of an element that will used for partial render (swapped Out Of Band). It sets the id of the element to a concatenation of the current component id and whatever you pass to it, and sets the right hx-swap-oob strategy.
<div {% oob "dropdown" %} class="dropdown">
...
</div>
{% on <EventName: str> <EventHandler: str> **kwargs %}
binds the event handler using hx-trigger to an event handler in the component with certain explicit parameters. Implicit parameters are passed from anything that has a name attribute defined inside the component.
<button {% on "click" "save" %}>
...
</button>
{% class <ClassName: str>: <BooleanExpr: bool>[, ...] %}
used inside of any html tag to set the class attribute, activating certain classes when corresponding boolean expression isTrue
.
<button {% class "btn": True, "btn-primary": is_primary %}>
...
</button>
This library provides the class djhtmx.testing.Htmx
which implements a very basic a dumb runtime for testing components. How to use:
from django.test import Client, TestCase
from djhtmx.testing import Htmx
from .models import Item
class TestNormalRendering(TestCase):
def setUp(self):
Item.objects.create(text="First task")
Item.objects.create(text="Second task")
self.htmx = Htmx(Client())
def test_todo_app(self):
self.htmx.navigate_to("/todo")
[a, b] = self.htmx.select('[hx-name="TodoItem"] label')
self.assertEqual(a.text_content(), "First task")
self.assertEqual(b.text_content(), "Second task")
[count] = self.htmx.select(".todo-count")
self.assertEqual(count.text_content(), "2 items left")
# Add new item
self.htmx.type("input.new-todo", "3rd task")
self.htmx.trigger("input.new-todo")
[count] = self.htmx.select(".todo-count")
self.assertEqual(count.text_content(), "3 items left")
[a, b, c] = self.htmx.select('[hx-name="TodoItem"] label')
self.assertEqual(a.text_content(), "First task")
self.assertEqual(b.text_content(), "Second task")
self.assertEqual(c.text_content(), "3rd task")
Htmx(client: Client)
: pass a Django test client, that can be authenticated if that's required.
htmx.navigate_to(url, *args, **kwargs)
: This is used to navigate to some url. It is a wrapper of Client.get
that will retrieve the page and parse the HTML into htmx.dom: lxml.html.HtmlElement
and create a component repository in htmx.repo
.
htmx.select(css_selector: str) -> list[lxml.html.HtmlElement]
: Pass some CSS selector here to retrieve nodes from the DOM, so you can modify them or perform assertions over them.
htmx.find_by_text(text: str) -> lxml.html.HtmlElement
: Returns the first element that contains certain text.
htmx.get_component_by_type(component_type: type[TPydanticComponent]) -> TPydanticComponent
: Retrieves the only instance rendered of that component type in the current page. If there is more than one instance this fails.
htmx.get_components_by_type(component_type: type[TPydanticComponent]) -> list[TPydanticComponent]
: Retrieves all instances of this component type in the current page.
htmx.get_component_by_id(component_id: str) -> TPydanticComponent
: Retrieves a component by its id from the current page.
htmx.type(selector: str | html.HtmlElement, text: str, clear=False)
: This simulates typing in an input or text area. If clear=True
it clears it replaces the current text in it.
htmx.trigger(selector: str | html.HtmlElement)
: This triggers whatever event is bound in the selected element and returns after all side effects had been processed.
self.htmx.type("input.new-todo", "3rd task")
self.htmx.trigger("input.new-todo")
htmx.send(method: Callable[P, Any], *args: P.args, **kwargs: P.kwargs)
: This sends the runtime to execute that a bound method of a PydanticComponent and returns after all side effects had been processed. Use as in:
todo_list = htmx.get_component_by_type(TodoList)
htmx.send(todo_list.new_item, text="New todo item")
htmx.dispatch_event(self, component_id: str, event_handler: str, kwargs: dict[str, Any])
: Similar to htmx.send
, but you don't need the instance, you just need to know its id.
htmx.dispatch_event("#todo-list", "new_item", {"text": "New todo item"})