2023-07-22 23:36:28 +02:00
|
|
|
use axum::{extract::FromRequestParts, http::request::Parts};
|
|
|
|
|
2023-07-23 00:20:29 +02:00
|
|
|
use crate::{
|
|
|
|
HX_BOOSTED, HX_CURRENT_URL, HX_HISTORY_RESTORE_REQUEST, HX_PROMPT, HX_TARGET, HX_TRIGGER,
|
|
|
|
HX_TRIGGER_NAME,
|
|
|
|
};
|
2023-07-22 23:36:28 +02:00
|
|
|
|
|
|
|
/// 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> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if parts.headers.contains_key(HX_BOOSTED) {
|
2023-07-22 23:36:28 +02:00
|
|
|
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> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if let Some(url) = parts.headers.get(HX_CURRENT_URL) {
|
2023-07-22 23:36:28 +02:00
|
|
|
if let Ok(url) = url.to_str() {
|
|
|
|
return Ok(HxCurrentUrl(url.to_string()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(HxCurrentUrl("".to_string()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-22 23:47:48 +02:00
|
|
|
pub struct HxHistoryRestoreRequest(pub bool);
|
|
|
|
|
|
|
|
#[axum::async_trait]
|
|
|
|
impl<S> FromRequestParts<S> for HxHistoryRestoreRequest
|
|
|
|
where
|
|
|
|
S: Send + Sync,
|
|
|
|
{
|
|
|
|
type Rejection = std::convert::Infallible;
|
|
|
|
|
|
|
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if parts.headers.contains_key(HX_HISTORY_RESTORE_REQUEST) {
|
2023-07-22 23:47:48 +02:00
|
|
|
return Ok(HxHistoryRestoreRequest(true));
|
|
|
|
} else {
|
|
|
|
return Ok(HxHistoryRestoreRequest(false));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct HxPrompt(pub Option<String>);
|
|
|
|
|
|
|
|
#[axum::async_trait]
|
|
|
|
impl<S> FromRequestParts<S> for HxPrompt
|
|
|
|
where
|
|
|
|
S: Send + Sync,
|
|
|
|
{
|
|
|
|
type Rejection = std::convert::Infallible;
|
|
|
|
|
|
|
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if let Some(prompt) = parts.headers.get(HX_PROMPT) {
|
2023-07-22 23:47:48 +02:00
|
|
|
if let Ok(prompt) = prompt.to_str() {
|
|
|
|
return Ok(HxPrompt(Some(prompt.to_string())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(HxPrompt(None));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-22 23:36:28 +02:00
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
|
|
pub struct HxRequest(pub bool);
|
|
|
|
|
|
|
|
#[axum::async_trait]
|
|
|
|
impl<S> FromRequestParts<S> for HxRequest
|
|
|
|
where
|
|
|
|
S: Send + Sync,
|
|
|
|
{
|
|
|
|
type Rejection = std::convert::Infallible;
|
|
|
|
|
|
|
|
async fn from_request_parts(_: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
|
|
|
return Ok(HxRequest(true));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct HxTarget(pub Option<String>);
|
|
|
|
|
|
|
|
#[axum::async_trait]
|
|
|
|
impl<S> FromRequestParts<S> for HxTarget
|
|
|
|
where
|
|
|
|
S: Send + Sync,
|
|
|
|
{
|
|
|
|
type Rejection = std::convert::Infallible;
|
|
|
|
|
|
|
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if let Some(target) = parts.headers.get(HX_TARGET) {
|
2023-07-22 23:36:28 +02:00
|
|
|
if let Ok(target) = target.to_str() {
|
|
|
|
return Ok(HxTarget(Some(target.to_string())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(HxTarget(None));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct HxTriggerName(pub Option<String>);
|
|
|
|
|
|
|
|
#[axum::async_trait]
|
|
|
|
impl<S> FromRequestParts<S> for HxTriggerName
|
|
|
|
where
|
|
|
|
S: Send + Sync,
|
|
|
|
{
|
|
|
|
type Rejection = std::convert::Infallible;
|
|
|
|
|
|
|
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if let Some(trigger_name) = parts.headers.get(HX_TRIGGER_NAME) {
|
2023-07-22 23:36:28 +02:00
|
|
|
if let Ok(trigger_name) = trigger_name.to_str() {
|
|
|
|
return Ok(HxTriggerName(Some(trigger_name.to_string())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(HxTriggerName(None));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct HxTrigger(pub Option<String>);
|
|
|
|
|
|
|
|
#[axum::async_trait]
|
|
|
|
impl<S> FromRequestParts<S> for HxTrigger
|
|
|
|
where
|
|
|
|
S: Send + Sync,
|
|
|
|
{
|
|
|
|
type Rejection = std::convert::Infallible;
|
|
|
|
|
|
|
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
2023-07-23 00:20:29 +02:00
|
|
|
if let Some(trigger) = parts.headers.get(HX_TRIGGER) {
|
2023-07-22 23:36:28 +02:00
|
|
|
if let Ok(trigger) = trigger.to_str() {
|
|
|
|
return Ok(HxTrigger(Some(trigger.to_string())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(HxTrigger(None));
|
|
|
|
}
|
|
|
|
}
|