add initial crates and apps

This commit is contained in:
Evan Carroll 2026-01-12 15:34:40 -06:00
parent 5c87ba3519
commit 1ca300098f
113 changed files with 28169 additions and 0 deletions

View file

@ -0,0 +1,78 @@
//! Admin Leptos application root and router.
use leptos::prelude::*;
use leptos_meta::{provide_meta_context, MetaTags, Stylesheet, Title};
use leptos_router::components::Router;
use crate::routes::AdminRoutes;
/// Application state for the admin app.
///
/// Note: We intentionally don't derive `FromRef` because both pools are
/// the same type (`PgPool`), which would cause a conflicting implementation.
/// Instead, handlers should use Extension extractors for the pools.
#[cfg(feature = "ssr")]
#[derive(Clone)]
pub struct AdminAppState {
/// The primary database pool for this admin instance.
/// For Owner App: chattyness_owner pool (no RLS)
/// For Admin App: chattyness_app pool (RLS enforced)
pub pool: sqlx::PgPool,
pub leptos_options: LeptosOptions,
}
#[cfg(feature = "ssr")]
impl axum::extract::FromRef<AdminAppState> for LeptosOptions {
fn from_ref(state: &AdminAppState) -> Self {
state.leptos_options.clone()
}
}
#[cfg(feature = "ssr")]
impl axum::extract::FromRef<AdminAppState> for sqlx::PgPool {
fn from_ref(state: &AdminAppState) -> Self {
state.pool.clone()
}
}
/// Shell component for SSR.
///
/// The `data-app="admin"` attribute tells the WASM hydration script to mount
/// AdminApp.
pub fn admin_shell(options: LeptosOptions) -> impl IntoView {
view! {
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<AutoReload options=options.clone() />
<HydrationScripts options />
<MetaTags />
</head>
<body class="admin-app" data-app="admin">
<AdminApp />
</body>
</html>
}
}
/// Main admin application component.
///
/// This wraps `AdminRoutes` with a `Router` for standalone use (e.g., chattyness-owner).
/// Routes are nested under `/admin` to match the link paths used in page components.
/// For embedding in a combined app (e.g., chattyness-app), use `AdminRoutes` directly.
#[component]
pub fn AdminApp() -> impl IntoView {
// Provide meta context for title and meta tags
provide_meta_context();
view! {
<Stylesheet id="admin-styles" href="/static/css/admin.css" />
<Title text="Chattyness Admin Panel" />
<Router base="/admin">
<AdminRoutes />
</Router>
}
}