liora-components 0.1.16

Enterprise-style native GPUI component library for Liora applications.
//! Layout Helpers module.
//!
//! This public module implements the Liora shared demo/documentation layout helpers built from public components. It keeps the reusable
//! component logic inside `liora-components` rather than Gallery or Docs so
//! downstream GPUI applications can compose the same behavior with their own
//! app state, assets, and release policy.
//!
//! ## Usage model
//!
//! Components in this module render native GPUI element trees. Stateless builder
//! values can be constructed inline, while controls with focus, selection,
//! popup, drag, or editing state should be stored as `gpui::Entity<T>` fields in
//! the parent view so state survives GPUI render passes.
//!
//! ## Design contract
//!
//! The implementation should use Liora theme tokens from `liora-core` and
//! `liora-theme`, keep accessibility-oriented keyboard/pointer behavior close to
//! the component, and avoid app-specific Gallery/Docs resources in this SDK
//! crate.

use crate::{Card, Paragraph, Space, Title};
use gpui::{AnyElement, IntoElement, px};

/// Performs the page operation used by this component.
pub fn page(
    title: &'static str,
    description: &'static str,
    body: impl IntoElement,
) -> impl IntoElement {
    Space::new()
        .vertical()
        .gap_xl()
        .child(header(title, description))
        .child(body)
}

/// Renders a consistent demo/documentation showcase card for compact component examples.
pub fn showcase_card(
    title: &'static str,
    description: &'static str,
    body: impl IntoElement,
) -> impl IntoElement {
    Card::new(
        Space::new()
            .vertical()
            .gap_md()
            .child(header(title, description))
            .child(body),
    )
    .width(px(360.0))
    .no_shadow()
}

/// Renders a consistent demo/documentation showcase card for wide component examples.
pub fn showcase_card_wide(
    title: &'static str,
    description: &'static str,
    body: impl IntoElement,
) -> impl IntoElement {
    Card::new(
        Space::new()
            .vertical()
            .gap_md()
            .child(header(title, description))
            .child(body),
    )
    .width(px(760.0))
    .no_shadow()
}

/// Renders a wrapping grid of demo/documentation showcase cards with stable spacing.
pub fn showcase_grid(children: Vec<AnyElement>) -> impl IntoElement {
    Space::new()
        .wrap()
        .gap_lg()
        .align_start()
        .children(children)
}

/// Renders a single-column stack of wide demo/documentation showcase cards.
pub fn showcase_stack(children: Vec<AnyElement>) -> impl IntoElement {
    Space::new()
        .vertical()
        .gap_lg()
        .align_start()
        .children(children)
}

/// Performs the section operation used by this component.
pub fn section(
    title: &'static str,
    description: &'static str,
    body: impl IntoElement,
) -> impl IntoElement {
    Space::new()
        .vertical()
        .gap_md()
        .child(header(title, description))
        .child(body)
}

/// Performs the header operation used by this component.
pub fn header(title: &'static str, description: &'static str) -> impl IntoElement {
    Space::new()
        .vertical()
        .gap_xs()
        .child(Title::new(title).h3())
        .child(Paragraph::with_text(description))
}

/// Performs the row operation used by this component.
pub fn row(children: Vec<impl IntoElement>) -> impl IntoElement {
    Space::new().wrap().gap_lg().children(children)
}

/// Applies the predefined row md sizing preset.
pub fn row_md(children: Vec<impl IntoElement>) -> impl IntoElement {
    Space::new().wrap().gap_md().children(children)
}

#[cfg(test)]
mod tests {
    #[test]
    fn showcase_helpers_lock_demo_card_rhythm() {
        let source = include_str!("layout_helpers.rs");

        assert!(source.contains("pub fn showcase_card("));
        assert!(source.contains("pub fn showcase_card_wide("));
        assert!(source.contains("pub fn showcase_grid("));
        assert!(source.contains("pub fn showcase_stack("));
        assert!(source.contains(".width(px(360.0))"));
        assert!(source.contains(".width(px(760.0))"));
        assert!(source.contains(".align_start()"));
    }
}