WIP: Simplifica gestión de regiones y plantillas en los temas #9
7 changed files with 43 additions and 39 deletions
|
|
@ -82,13 +82,13 @@ async fn homepage(request: HttpRequest) -> ResultPage<Markup, ErrorPage> {
|
||||||
|
|
||||||
use pagetop::prelude::*;
|
use pagetop::prelude::*;
|
||||||
|
|
||||||
/// El tema usa las mismas regiones predefinidas por [`ThemeRegion`].
|
/// El tema usa las mismas regiones predefinidas por [`DefaultRegions`].
|
||||||
pub type AlinerRegion = ThemeRegion;
|
pub type AlinerRegions = DefaultRegions;
|
||||||
|
|
||||||
/// Implementa el tema para usar en pruebas que muestran el esquema de páginas HTML.
|
/// Implementa el tema para usar en pruebas que muestran el esquema de páginas HTML.
|
||||||
///
|
///
|
||||||
/// Tema mínimo ideal para **pruebas y demos** que renderiza el **esqueleto HTML** con las mismas
|
/// Tema mínimo ideal para **pruebas y demos** que renderiza el **esqueleto HTML** con las mismas
|
||||||
/// regiones básicas definidas por [`ThemeRegion`]. No pretende ser un tema para producción, está
|
/// regiones básicas definidas por [`DefaultRegions`]. No pretende ser un tema para producción, está
|
||||||
/// pensado para:
|
/// pensado para:
|
||||||
///
|
///
|
||||||
/// - Verificar integración de componentes y composiciones (*layouts*) sin estilos complejos.
|
/// - Verificar integración de componentes y composiciones (*layouts*) sin estilos complejos.
|
||||||
|
|
|
||||||
|
|
@ -101,8 +101,8 @@ pub mod prelude {
|
||||||
pub use crate::theme::*;
|
pub use crate::theme::*;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// El tema usa las mismas regiones predefinidas por [`ThemeRegion`].
|
/// El tema usa las mismas regiones predefinidas por [`DefaultRegions`].
|
||||||
pub type BootsierRegion = ThemeRegion;
|
pub type BootsierRegions = DefaultRegions;
|
||||||
|
|
||||||
/// Implementa el tema.
|
/// Implementa el tema.
|
||||||
pub struct Bootsier;
|
pub struct Bootsier;
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
//! Temas básicos soportados por PageTop.
|
//! Temas básicos soportados por PageTop.
|
||||||
|
|
||||||
mod basic;
|
mod basic;
|
||||||
pub use basic::{Basic, BasicRegion};
|
pub use basic::{Basic, BasicRegions};
|
||||||
|
|
|
||||||
|
|
@ -1,8 +1,8 @@
|
||||||
/// Es el tema básico que incluye PageTop por defecto.
|
/// Es el tema básico que incluye PageTop por defecto.
|
||||||
use crate::prelude::*;
|
use crate::prelude::*;
|
||||||
|
|
||||||
/// El tema básico usa las mismas regiones predefinidas por [`ThemeRegion`].
|
/// El tema básico usa las mismas regiones predefinidas por [`DefaultRegions`].
|
||||||
pub type BasicRegion = ThemeRegion;
|
pub type BasicRegions = DefaultRegions;
|
||||||
|
|
||||||
/// Tema básico por defecto que extiende el funcionamiento predeterminado de [`Theme`].
|
/// Tema básico por defecto que extiende el funcionamiento predeterminado de [`Theme`].
|
||||||
pub struct Basic;
|
pub struct Basic;
|
||||||
|
|
|
||||||
|
|
@ -9,13 +9,14 @@
|
||||||
//! tipografías, espaciados y cualquier otro detalle visual o de comportamiento (como animaciones,
|
//! tipografías, espaciados y cualquier otro detalle visual o de comportamiento (como animaciones,
|
||||||
//! scripts de interfaz, etc.).
|
//! scripts de interfaz, etc.).
|
||||||
//!
|
//!
|
||||||
//! Los temas son extensiones que implementan [`Extension`](crate::core::extension::Extension); por
|
//! Los temas son extensiones que implementan [`Extension`](crate::core::extension::Extension), por
|
||||||
//! lo que se instancian, declaran sus dependencias y se inician igual que el resto de extensiones;
|
//! lo que se instancian, declaran dependencias y se inician igual que cualquier otra extensión.
|
||||||
//! pero serán temas si además implementan [`theme()`](crate::core::extension::Extension::theme) y
|
//! También deben implementar [`Theme`] y sobrescribir el método
|
||||||
//! [`Theme`].
|
//! [`Extension::theme()`](crate::core::extension::Extension::theme) para que PageTop pueda
|
||||||
|
//! registrarlos como temas
|
||||||
|
|
||||||
mod definition;
|
mod definition;
|
||||||
pub use definition::{Theme, ThemePage, ThemeRef, ThemeRegion};
|
pub use definition::{Theme, ThemePage, ThemeRef, DefaultRegions};
|
||||||
|
|
||||||
mod regions;
|
mod regions;
|
||||||
pub(crate) use regions::{ChildrenInRegions, REGION_CONTENT};
|
pub(crate) use regions::{ChildrenInRegions, REGION_CONTENT};
|
||||||
|
|
|
||||||
|
|
@ -4,7 +4,7 @@ use crate::core::theme::{Region, RegionRef, REGION_CONTENT};
|
||||||
use crate::html::{html, Markup, StyleSheet};
|
use crate::html::{html, Markup, StyleSheet};
|
||||||
use crate::locale::L10n;
|
use crate::locale::L10n;
|
||||||
use crate::response::page::Page;
|
use crate::response::page::Page;
|
||||||
use crate::{global, join};
|
use crate::{global, join, AutoDefault};
|
||||||
|
|
||||||
use std::sync::LazyLock;
|
use std::sync::LazyLock;
|
||||||
|
|
||||||
|
|
@ -14,16 +14,17 @@ use std::sync::LazyLock;
|
||||||
/// implementen [`Theme`] y, a su vez, [`Extension`].
|
/// implementen [`Theme`] y, a su vez, [`Extension`].
|
||||||
pub type ThemeRef = &'static dyn Theme;
|
pub type ThemeRef = &'static dyn Theme;
|
||||||
|
|
||||||
/// Conjunto de regiones que los temas pueden exponer para el renderizado.
|
/// Conjunto de regiones predefinidas que los temas pueden exponer para el renderizado.
|
||||||
///
|
///
|
||||||
/// `ThemeRegion` define un conjunto de regiones predefinidas para estructurar un documento HTML.
|
/// `DefaultRegions` define un conjunto de regiones predefinidas para estructurar un documento HTML.
|
||||||
/// Proporciona **identificadores estables** (vía [`Region::key()`]) y **etiquetas localizables**
|
/// Proporciona **identificadores estables** (vía [`Region::key()`]) y **etiquetas localizables**
|
||||||
/// (vía [`Region::label()`]) a las regiones donde se añadirán los componentes.
|
/// (vía [`Region::label()`]) a las regiones donde se añadirán los componentes.
|
||||||
///
|
///
|
||||||
/// Se usa por defecto en [`Theme::page_regions()`](crate::core::theme::Theme::page_regions) y sus
|
/// Se usa por defecto en [`Theme::page_regions()`](crate::core::theme::Theme::page_regions) y sus
|
||||||
/// variantes representan el conjunto mínimo recomendado para cualquier tema. Sin embargo, cada tema
|
/// variantes representan el conjunto mínimo recomendado para cualquier tema. Sin embargo, cada tema
|
||||||
/// podría exponer su propio conjunto de regiones.
|
/// podría exponer su propio conjunto de regiones.
|
||||||
pub enum ThemeRegion {
|
#[derive(AutoDefault)]
|
||||||
|
pub enum DefaultRegions {
|
||||||
/// Cabecera de la página.
|
/// Cabecera de la página.
|
||||||
///
|
///
|
||||||
/// Clave: `"header"`. Suele contener *branding*, navegación principal o avisos globales.
|
/// Clave: `"header"`. Suele contener *branding*, navegación principal o avisos globales.
|
||||||
|
|
@ -32,6 +33,7 @@ pub enum ThemeRegion {
|
||||||
/// Contenido principal de la página (**obligatoria**).
|
/// Contenido principal de la página (**obligatoria**).
|
||||||
///
|
///
|
||||||
/// Clave: `"content"`. Es el destino por defecto para insertar componentes a nivel de página.
|
/// Clave: `"content"`. Es el destino por defecto para insertar componentes a nivel de página.
|
||||||
|
#[default]
|
||||||
Content,
|
Content,
|
||||||
|
|
||||||
/// Pie de página.
|
/// Pie de página.
|
||||||
|
|
@ -40,12 +42,12 @@ pub enum ThemeRegion {
|
||||||
Footer,
|
Footer,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Region for ThemeRegion {
|
impl Region for DefaultRegions {
|
||||||
fn key(&self) -> &str {
|
fn key(&self) -> &str {
|
||||||
match self {
|
match self {
|
||||||
ThemeRegion::Header => "header",
|
Self::Header => "header",
|
||||||
ThemeRegion::Content => REGION_CONTENT,
|
Self::Content => REGION_CONTENT,
|
||||||
ThemeRegion::Footer => "footer",
|
Self::Footer => "footer",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -60,16 +62,17 @@ impl Region for ThemeRegion {
|
||||||
/// implementa automáticamente para cualquier tipo que implemente [`Theme`], por lo que normalmente
|
/// implementa automáticamente para cualquier tipo que implemente [`Theme`], por lo que normalmente
|
||||||
/// no requiere implementación explícita.
|
/// no requiere implementación explícita.
|
||||||
///
|
///
|
||||||
/// Si un tema **sobrescribe** uno o más de estos métodos de [`Theme`]:
|
/// Si un tema **sobrescribe** uno o más de los siguientes métodos de [`Theme`]:
|
||||||
///
|
///
|
||||||
/// - [`render_page_region()`](Theme::render_page_region),
|
/// - [`render_page_region()`](Theme::render_page_region),
|
||||||
/// - [`render_page_head()`](Theme::render_page_head), o
|
/// - [`render_page_head()`](Theme::render_page_head), o
|
||||||
/// - [`render_page_body()`](Theme::render_page_body);
|
/// - [`render_page_body()`](Theme::render_page_body);
|
||||||
///
|
///
|
||||||
/// es posible volver al comportamiento por defecto usando FQS (*Fully Qualified Syntax*):
|
/// puede volver al comportamiento por defecto con una llamada FQS (*Fully Qualified Syntax*) a:
|
||||||
///
|
///
|
||||||
/// - `<Self as ThemePage>::render_body(self, page, self.page_regions())`
|
/// - `<Self as ThemePage>::render_region(self, page, region)`,
|
||||||
/// - `<Self as ThemePage>::render_head(self, page)`
|
/// - `<Self as ThemePage>::render_body(self, page, self.page_regions())`, o
|
||||||
|
/// - `<Self as ThemePage>::render_head(self, page)`.
|
||||||
pub trait ThemePage {
|
pub trait ThemePage {
|
||||||
/// Renderiza el **contenedor** de una región concreta del `<body>` de la página.
|
/// Renderiza el **contenedor** de una región concreta del `<body>` de la página.
|
||||||
///
|
///
|
||||||
|
|
@ -206,9 +209,9 @@ pub trait Theme: Extension + ThemePage + Send + Sync {
|
||||||
/// fn page_regions(&self) -> &'static [RegionRef] {
|
/// fn page_regions(&self) -> &'static [RegionRef] {
|
||||||
/// static REGIONS: LazyLock<[RegionRef; 4]> = LazyLock::new(|| {
|
/// static REGIONS: LazyLock<[RegionRef; 4]> = LazyLock::new(|| {
|
||||||
/// [
|
/// [
|
||||||
/// &ThemeRegion::Header,
|
/// &DefaultRegions::Header,
|
||||||
/// &ThemeRegion::Content,
|
/// &DefaultRegions::Content,
|
||||||
/// &ThemeRegion::Footer,
|
/// &DefaultRegions::Footer,
|
||||||
/// ]
|
/// ]
|
||||||
/// });
|
/// });
|
||||||
/// &*REGIONS
|
/// &*REGIONS
|
||||||
|
|
@ -217,9 +220,9 @@ pub trait Theme: Extension + ThemePage + Send + Sync {
|
||||||
fn page_regions(&self) -> &'static [RegionRef] {
|
fn page_regions(&self) -> &'static [RegionRef] {
|
||||||
static REGIONS: LazyLock<[RegionRef; 3]> = LazyLock::new(|| {
|
static REGIONS: LazyLock<[RegionRef; 3]> = LazyLock::new(|| {
|
||||||
[
|
[
|
||||||
&ThemeRegion::Header,
|
&DefaultRegions::Header,
|
||||||
&ThemeRegion::Content,
|
&DefaultRegions::Content,
|
||||||
&ThemeRegion::Footer,
|
&DefaultRegions::Footer,
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
&*REGIONS
|
&*REGIONS
|
||||||
|
|
|
||||||
|
|
@ -31,25 +31,25 @@ pub const REGION_CONTENT: &str = "content";
|
||||||
/// `aria-label` o en descripciones semánticas del contenedor).
|
/// `aria-label` o en descripciones semánticas del contenedor).
|
||||||
///
|
///
|
||||||
/// Las implementaciones típicas son *enumeraciones estáticas* declaradas por cada tema (ver como
|
/// Las implementaciones típicas son *enumeraciones estáticas* declaradas por cada tema (ver como
|
||||||
/// ejemplo [`ThemeRegion`](crate::core::theme::ThemeRegion)), de modo que las claves y etiquetas
|
/// ejemplo [`DefaultRegions`](crate::core::theme::DefaultRegions)), de modo que las claves y
|
||||||
/// permanecen inmutables y fácilmente referenciables.
|
/// etiquetas permanecen inmutables y fácilmente referenciables.
|
||||||
///
|
///
|
||||||
/// # Ejemplo
|
/// # Ejemplo
|
||||||
///
|
///
|
||||||
/// ```rust
|
/// ```rust
|
||||||
/// # use pagetop::prelude::*;
|
/// # use pagetop::prelude::*;
|
||||||
/// pub enum MyThemeRegion {
|
/// pub enum MyThemeRegions {
|
||||||
/// Header,
|
/// Header,
|
||||||
/// Content,
|
/// Content,
|
||||||
/// Footer,
|
/// Footer,
|
||||||
/// }
|
/// }
|
||||||
///
|
///
|
||||||
/// impl Region for MyThemeRegion {
|
/// impl Region for MyThemeRegions {
|
||||||
/// fn key(&self) -> &str {
|
/// fn key(&self) -> &str {
|
||||||
/// match self {
|
/// match self {
|
||||||
/// MyThemeRegion::Header => "header",
|
/// Self::Header => "header",
|
||||||
/// MyThemeRegion::Content => "content",
|
/// Self::Content => "content",
|
||||||
/// MyThemeRegion::Footer => "footer",
|
/// Self::Footer => "footer",
|
||||||
/// }
|
/// }
|
||||||
/// }
|
/// }
|
||||||
///
|
///
|
||||||
|
|
@ -111,7 +111,7 @@ impl ChildrenInRegions {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Punto de acceso para añadir componentes a regiones globales o específicas de un tema.
|
/// Permite añadir componentes a regiones globales o específicas de un tema.
|
||||||
///
|
///
|
||||||
/// Según la variante, se pueden añadir componentes ([`add()`](Self::add)) que permanecerán
|
/// Según la variante, se pueden añadir componentes ([`add()`](Self::add)) que permanecerán
|
||||||
/// disponibles durante toda la ejecución.
|
/// disponibles durante toda la ejecución.
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue