axum-htmx/src/lib.rs

156 lines
5.3 KiB
Rust
Raw Normal View History

2023-07-22 22:37:15 +02:00
#![forbid(unsafe_code)]
use axum::{extract::FromRequestParts, http::request::Parts};
/// Represents all of the headers that can be sent in a request to the server.
///
/// See <https://htmx.org/reference/#request_headers> for more information.
#[derive(Debug, Clone, Copy, PartialEq)]
pub enum HxRequestHeader {
/// Indicates that the request is via an element using `hx-boost` attribute.
///
/// See <https://htmx.org/attributes/hx-boost/> for more information.
Boosted,
/// The current URL of the browser.
CurrentUrl,
/// `true` if the request is for history restoration after a miss in the
/// local history cache.
HistoryRestoreRequest,
/// The user response to an `hx-prompt`
///
/// See <https://htmx.org/attributes/hx-prompt/> for more information.
Prompt,
/// Always `true`.
Request,
/// The `id` of the target element, if it exists.
Target,
/// The `name` of the triggered element, if it exists.
TriggerName,
/// The `id` of the triggered element, if it exists.
Trigger,
}
impl HxRequestHeader {
pub fn as_str(&self) -> &'static str {
match self {
HxRequestHeader::Boosted => "HX-Boosted",
HxRequestHeader::CurrentUrl => "HX-Current-Url",
HxRequestHeader::HistoryRestoreRequest => "HX-History-Restore-Request",
HxRequestHeader::Prompt => "HX-Prompt",
HxRequestHeader::Request => "HX-Request",
HxRequestHeader::Target => "HX-Target",
HxRequestHeader::TriggerName => "HX-Trigger-Name",
HxRequestHeader::Trigger => "HX-Trigger",
}
}
}
/// Represents all of the headers that can be sent in a response to the client.
///
/// See <https://htmx.org/reference/#response_headers> for more information.
#[derive(Debug, Clone, Copy, PartialEq)]
pub enum HxResponseHeader {
/// Allows you to do a client-side redirect that does not do a full page
/// reload.
Location,
/// Pushes a new URL onto the history stack.
PushUrl,
/// Can be used to do a client-side redirect to a new location.
Redirect,
/// If set to `true`, the client will do a full refresh on the page.
Refresh,
/// Replaces the currelt URL in the location bar.
ReplaceUrl,
/// Allows you to specify how the response value will be swapped.
///
/// See <https://htmx.org/attributes/hx-swap/> for more information.
Reswap,
/// A CSS selector that update the target of the content update to a
/// different element on the page.
Retarget,
/// A CSS selector that allows you to choose which part of the response is
/// used to be swapped in. Overrides an existing `hx-select` on the
/// triggering element
Reselect,
/// Allows you to trigger client-side events.
///
/// See <https://htmx.org/headers/hx-trigger/> for more information.
Trigger,
/// Allows you to trigger client-side events.
///
/// See <https://htmx.org/headers/hx-trigger/> for more information.
TriggerAfterSettle,
/// Allows you to trigger client-side events.
///
/// See <https://htmx.org/headers/hx-trigger/> for more information.
TriggerAfterSwap,
}
impl HxResponseHeader {
pub fn as_str(&self) -> &'static str {
match self {
HxResponseHeader::Location => "HX-Location",
HxResponseHeader::PushUrl => "HX-Push-Url",
HxResponseHeader::Redirect => "HX-Redirect",
HxResponseHeader::Refresh => "HX-Refresh",
HxResponseHeader::ReplaceUrl => "HX-Replace-Url",
HxResponseHeader::Reswap => "HX-Reswap",
HxResponseHeader::Retarget => "HX-Retarget",
HxResponseHeader::Reselect => "HX-Reselect",
HxResponseHeader::Trigger => "HX-Trigger",
HxResponseHeader::TriggerAfterSettle => "HX-Trigger-After-Settle",
HxResponseHeader::TriggerAfterSwap => "HX-Trigger-After-Swap",
}
}
}
/// The `HX-Boosted` header. This header is set when a request is made with the
/// "hx-boost" attribute is set on an element.
///
/// This extractor does not fail if no header is present, instead returning a
/// `false` value.
///
/// See <https://htmx.org/attributes/hx-boost/> for more information.
#[derive(Debug, Clone, Copy)]
pub struct HxBoosted(pub bool);
#[axum::async_trait]
impl<S> FromRequestParts<S> for HxBoosted
where
S: Send + Sync,
{
type Rejection = std::convert::Infallible;
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
if parts
.headers
.contains_key(HxRequestHeader::Boosted.as_str())
{
return Ok(HxBoosted(true));
} else {
return Ok(HxBoosted(false));
}
}
}
#[derive(Debug, Clone)]
pub struct HxCurrentUrl(pub String);
#[axum::async_trait]
impl<S> FromRequestParts<S> for HxCurrentUrl
where
S: Send + Sync,
{
type Rejection = std::convert::Infallible;
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
if let Some(url) = parts.headers.get(HxRequestHeader::CurrentUrl.as_str()) {
if let Ok(url) = url.to_str() {
return Ok(HxCurrentUrl(url.to_string()));
}
}
return Ok(HxCurrentUrl("".to_string()));
}
}