2023-07-22 22:37:15 +02:00
# axum-htmx
2023-07-23 01:17:07 +02:00
<!-- markdownlint - disable -->
< div align = "right" >
< a href = "https://crates.io/crates/axum-htmx" >
< img src = "https://img.shields.io/crates/v/axum-htmx?style=flat-square" alt = "crates.io badge" >
< / a >
2023-07-23 02:06:25 +02:00
< a href = "https://docs.rs/axum-htmx/latest/" >
2023-07-23 01:17:07 +02:00
< img src = "https://img.shields.io/docsrs/axum-htmx?style=flat-square" alt = "docs.rs badge" >
< / a >
< / div >
2023-07-23 01:21:58 +02:00
< br >
2023-07-23 01:17:07 +02:00
<!-- markdownlint - enable -->
2023-10-24 03:24:00 +02:00
`axum-htmx` is a small extension library providing extractors, responders, and
2023-12-03 21:55:56 +01:00
request guards for [htmx ](https://htmx.org/ ) headers within
[axum ](https://github.com/tokio-rs/axum ).
2023-07-22 22:37:15 +02:00
2023-10-24 05:58:41 +02:00
## Table of Contents
- [axum-htmx ](#axum-htmx )
- [Table of Contents ](#table-of-contents )
- [Getting Started ](#getting-started )
- [Extractors ](#extractors )
- [Responders ](#responders )
- [Request Guards ](#request-guards )
2024-04-06 09:39:00 +02:00
- [Macroses ](#macroses )
2023-10-24 05:58:41 +02:00
- [Examples ](#examples )
- [Example: Extractors ](#example-extractors )
- [Example: Responders ](#example-responders )
- [Example: Router Guard ](#example-router-guard )
- [Feature Flags ](#feature-flags )
- [Contributing ](#contributing )
- [License ](#license )
2023-07-23 01:21:58 +02:00
## Getting Started
2023-07-22 22:37:15 +02:00
2023-12-03 21:55:56 +01:00
Run `cargo add axum-htmx` to add the library to your project.
2023-07-22 22:37:15 +02:00
2023-07-23 00:50:23 +02:00
## Extractors
All of the [htmx request headers ](https://htmx.org/reference/#request_headers )
2023-08-09 02:21:36 +02:00
have a supported extractor. Extractors are infallible, meaning they will always
succeed and never return an error. In the case where a header is not present,
the extractor will return `None` or `false` dependant on the expected return
type.
2023-07-23 00:50:23 +02:00
2023-12-03 21:47:22 +01:00
| Header | Extractor | Value |
|------------------------------|---------------------------|---------------------------|
| `HX-Boosted` | `HxBoosted` | `bool` |
| `HX-Current-URL` | `HxCurrentUrl` | `Option<axum::http::Uri>` |
| `HX-History-Restore-Request` | `HxHistoryRestoreRequest` | `bool` |
| `HX-Prompt` | `HxPrompt` | `Option<String>` |
| `HX-Request` | `HxRequest` | `bool` |
| `HX-Target` | `HxTarget` | `Option<String>` |
| `HX-Trigger-Name` | `HxTriggerName` | `Option<String>` |
| `HX-Trigger` | `HxTrigger` | `Option<String>` |
2023-07-23 00:50:23 +02:00
2023-10-24 03:24:00 +02:00
## Responders
All of the [htmx response headers ](https://htmx.org/reference/#response_headers )
2023-10-24 04:06:43 +02:00
have a supported responder. A responder is a basic type that implements
2023-10-24 03:57:00 +02:00
`IntoResponseParts` , allowing you to simply and safely apply the HX-* headers to
any of your responses.
2023-10-24 03:24:00 +02:00
2023-12-03 21:47:22 +01:00
| Header | Responder | Value |
|---------------------------|---------------------|-------------------------------------|
| `HX-Location` | `HxLocation` | `axum::http::Uri` |
| `HX-Push-Url` | `HxPushUrl` | `axum::http::Uri` |
| `HX-Redirect` | `HxRedirect` | `axum::http::Uri` |
| `HX-Refresh` | `HxRefresh` | `bool` |
| `HX-Replace-Url` | `HxReplaceUrl` | `axum::http::Uri` |
| `HX-Reswap` | `HxReswap` | `axum_htmx::responders::SwapOption` |
| `HX-Retarget` | `HxRetarget` | `String` |
| `HX-Reselect` | `HxReselect` | `String` |
| `HX-Trigger` | `HxResponseTrigger` | `axum_htmx::serde::HxEvent` |
| `HX-Trigger-After-Settle` | `HxResponseTrigger` | `axum_htmx::serde::HxEvent` |
| `HX-Trigger-After-Swap` | `HxResponseTrigger` | `axum_htmx::serde::HxEvent` |
2023-10-24 03:24:00 +02:00
2023-07-28 03:58:53 +02:00
## Request Guards
__Requires features `guards` .__
In addition to the extractors, there is also a route-wide layer request guard
2023-07-29 23:05:05 +02:00
for the `HX-Request` header. This will redirect any requests without the header
to "/" by default.
2023-07-28 03:58:53 +02:00
2023-07-28 05:04:22 +02:00
_It should be noted that this is NOT a replacement for an auth guard. A user can
trivially set the `HX-Request` header themselves. This is merely a convenience
for preventing users from receiving partial responses without context. If you
2023-07-28 05:21:31 +02:00
need to secure an endpoint you should be using a proper auth system._
2023-07-28 05:03:16 +02:00
2024-04-06 09:39:00 +02:00
## Macroses
2024-04-06 09:42:05 +02:00
__Requires features `derive` .__
2024-04-06 09:39:00 +02:00
In addition to the HxBoosted extractor, the library provides macroses `hx_boosted_by` and it's async version `hx_boosted_by_async` for managing the response based on the presence of the `HX-Boosted` header.
2024-04-07 12:22:40 +02:00
The macro input should have a `layout_fn` , and can have arguments passed from annotated function into `layout_fn` . The macro will call the `layout_fn` if the `HX-Boosted` header is not present, otherwise it will return the response directly.
2024-04-07 12:19:54 +02:00
```rust
#[hx_boosted_by(layout_fn, [arg1, agr2, ...])]
```
If `layout_fn` is an async function, use `hx_boosted_by_async` instead.
2023-10-24 05:58:41 +02:00
## Examples
2023-10-24 03:55:47 +02:00
### Example: Extractors
2023-07-22 22:37:15 +02:00
2023-07-23 00:50:23 +02:00
In this example, we'll look for the `HX-Boosted` header, which is set when
applying the [hx-boost ](https://htmx.org/attributes/hx-boost/ ) attribute to an
element. In our case, we'll use it to determine what kind of response we send.
2023-07-22 22:37:15 +02:00
2023-07-23 00:50:23 +02:00
When is this useful? When using a templating engine, like
[minijinja ](https://github.com/mitsuhiko/minijinja ), it is common to extend
different templates from a `_base.html` template. However, htmx works by sending
partial responses, so extending our `_base.html` would result in lots of extra
data being sent over the wire.
2023-07-22 22:37:15 +02:00
2023-07-23 00:50:23 +02:00
If we wanted to swap between pages, we would need to support both full template
responses and partial responses _(as the page can be accessed directly or
through a boosted anchor)_, so we look for the `HX-Boosted` header and extend
from a `_partial.html` template instead.
2023-07-22 22:37:15 +02:00
2023-07-28 05:36:41 +02:00
```rust
2023-07-28 03:58:53 +02:00
use axum::response::IntoResponse;
use axum_htmx::HxBoosted;
2023-07-22 22:37:15 +02:00
async fn get_index(HxBoosted(boosted): HxBoosted) -> impl IntoResponse {
if boosted {
// Send a template extending from _partial.html
} else {
// Send a template extending from _base.html
}
}
```
2023-10-24 03:55:47 +02:00
### Example: Responders
2023-10-24 03:24:00 +02:00
2023-10-24 05:25:18 +02:00
We can trigger any event being listened to by the DOM using an [htmx
trigger](https://htmx.org/attributes/hx-trigger/) header.
2023-10-24 03:24:00 +02:00
```rust
use axum_htmx::HxResponseTrigger;
// When we load our page, we will trigger any event listeners for "my-event.
2023-12-15 02:44:50 +01:00
async fn index() -> (HxResponseTrigger, & 'static str) {
// Note: As HxResponseTrigger only implements `IntoResponseParts` , we must
// return our trigger first here.
2023-10-24 03:24:00 +02:00
(
2023-12-03 21:33:27 +01:00
HxResponseTrigger::normal(["my-event", "second-event"]),
2023-12-15 02:44:50 +01:00
"Hello, world!",
2023-10-24 03:24:00 +02:00
)
}
```
2023-12-03 21:55:56 +01:00
`htmx` also allows arbitrary data to be sent along with the event, which we can
use via the `serde` feature flag and the `HxEvent` type.
2023-10-24 05:25:18 +02:00
```rust
2023-11-28 01:26:50 +01:00
use serde_json::json;
2023-10-24 05:25:18 +02:00
// Note that we are using `HxResponseTrigger` from the `axum_htmx::serde` module
// instead of the root module.
2023-12-01 20:15:10 +01:00
use axum_htmx::{HxEvent, HxResponseTrigger};
2023-10-24 05:25:18 +02:00
2023-12-15 02:44:50 +01:00
async fn index() -> (HxResponseTrigger, & 'static str) {
2023-10-24 05:25:18 +02:00
let event = HxEvent::new_with_data(
"my-event",
2023-12-01 20:15:10 +01:00
// May be any object that implements `serde::Serialize`
2023-10-24 05:25:18 +02:00
json!({"level": "info", "message": {
"title": "Hello, world!",
"body": "This is a test message.",
}}),
)
.unwrap();
2023-12-15 02:44:50 +01:00
// Note: As HxResponseTrigger only implements `IntoResponseParts` , we must
// return our trigger first here.
(HxResponseTrigger::normal([event]), "Hello, world!")
2023-10-24 05:25:18 +02:00
}
```
2023-07-28 03:58:53 +02:00
### Example: Router Guard
2023-07-23 00:50:23 +02:00
2023-07-28 05:36:41 +02:00
```rust
2023-07-28 03:58:53 +02:00
use axum::Router;
use axum_htmx::HxRequestGuardLayer;
2023-07-29 23:10:02 +02:00
fn router_one() -> Router {
2023-07-28 03:58:53 +02:00
Router::new()
2023-07-29 23:06:49 +02:00
// Redirects to "/" if the HX-Request header is not present
2023-07-29 23:05:05 +02:00
.layer(HxRequestGuardLayer::default())
2023-07-28 03:58:53 +02:00
}
2023-07-29 23:06:49 +02:00
2023-07-29 23:10:02 +02:00
fn router_two() -> Router {
2023-07-29 23:06:49 +02:00
Router::new()
.layer(HxRequestGuardLayer::new("/redirect-to-this-route"))
}
2023-07-23 00:50:23 +02:00
```
2024-04-06 09:39:00 +02:00
### Example: Macros
```rust
use axum_htmx::hx_boosted_by;
2024-04-07 12:19:54 +02:00
#[hx_boosted_by(with_layout, page_title)]
2024-04-06 09:39:00 +02:00
async fn get_hello(Path(name): Path< String > ) -> Html< String > {
2024-04-07 12:19:54 +02:00
let page_title = "Hello Page";
2024-04-06 10:03:43 +02:00
Html(format!("Hello, {}!", name))
2024-04-06 09:39:00 +02:00
}
2024-04-07 12:19:54 +02:00
#[hx_boosted_by(with_layout, page_title)]
2024-04-06 09:39:00 +02:00
async fn get_bye(Path(name): Path< String > ) -> Html< String > {
2024-04-07 12:19:54 +02:00
let page_title = "Bye Page";
2024-04-06 10:03:43 +02:00
Html(format!("Bye, {}!", name))
2024-04-06 09:39:00 +02:00
}
2024-04-07 12:19:54 +02:00
fn with_layout(Html(partial): Html< String > , page_title: & str) -> Html< String > {
Html(format!("< html > < head > < title > {}< / title > < / head > < body > {}< / body > < / html > ", page_title, partial))
2024-04-06 09:39:00 +02:00
}
```
2023-10-24 05:58:41 +02:00
## Feature Flags
2023-07-28 03:58:53 +02:00
<!-- markdownlint - disable -->
2023-12-01 20:15:10 +01:00
| Flag | Default | Description | Dependencies |
|----------|----------|------------------------------------------------------------|---------------------------------------------|
| `guards` | Disabled | Adds request guard layers. | `tower` , `futures-core` , `pin-project-lite` |
| `serde` | Disabled | Adds serde support for the `HxEvent` and `LocationOptions` | `serde` , `serde_json` |
2024-04-06 09:42:05 +02:00
| `derive` | Disabled | Adds the `hx_boosted_by` and `hx_boosted_by_async` macros. | `proc-macro-error` , `proc-macro2` , `quote` , `syn` |
2023-07-28 03:58:53 +02:00
<!-- markdownlint - enable -->
2023-07-28 04:02:16 +02:00
## Contributing
Contributions are always welcome! If you have an idea for a feature or find a
bug, let me know. PR's are appreciated, but if it's not a small change, please
open an issue first so we're all on the same page!
2023-07-22 22:37:15 +02:00
## License
`axum-htmx` is dual-licensed under either
2023-07-28 01:03:22 +02:00
- **[MIT License](/LICENSE-MIT)**
- **[Apache License, Version 2.0](/LICENSE-APACHE)**
2023-07-22 22:37:15 +02:00
at your option.