From 4a4e97f7beeb54bcfbef2f82777391b7cb614acf Mon Sep 17 00:00:00 2001 From: jakubmanczak Date: Sun, 8 Mar 2026 23:50:06 +0100 Subject: [PATCH] users::page, users::created_at, nav gating, icons, misc --- src/users/mod.rs | 8 ++++ src/web/components/mod.rs | 1 + src/web/components/nav.rs | 35 +++++++++------- src/web/components/user_miniprofile.rs | 33 +++++++++++++++ src/web/icons/calendar-1.svg | 1 + src/web/icons/mod.rs | 3 ++ src/web/icons/server.svg | 1 + src/web/icons/shield-user.svg | 1 + src/web/pages/dashboard.rs | 2 +- src/web/pages/mod.rs | 2 + src/web/pages/users.rs | 58 ++++++++++++++++++++++++++ src/web/styles.css | 2 +- 12 files changed, 129 insertions(+), 18 deletions(-) create mode 100644 src/web/components/user_miniprofile.rs create mode 100644 src/web/icons/calendar-1.svg create mode 100644 src/web/icons/server.svg create mode 100644 src/web/icons/shield-user.svg create mode 100644 src/web/pages/users.rs diff --git a/src/users/mod.rs b/src/users/mod.rs index ccbfd11..4e0ad3a 100644 --- a/src/users/mod.rs +++ b/src/users/mod.rs @@ -2,6 +2,7 @@ use axum::{ http::StatusCode, response::{IntoResponse, Response}, }; +use chrono::{DateTime, NaiveDate}; use rusqlite::{OptionalExtension, ffi::SQLITE_CONSTRAINT_UNIQUE}; use serde::{Deserialize, Serialize}; use uuid::Uuid; @@ -101,6 +102,13 @@ impl User { self.handle = new_handle; Ok(()) } + + pub fn created_at(&self) -> Option { + self.id + .get_timestamp() + .and_then(|ts| DateTime::from_timestamp(ts.to_unix().0 as i64, 0)) + .map(|dt| dt.date_naive()) + } } // DANGEROUS: AUTH diff --git a/src/web/components/mod.rs b/src/web/components/mod.rs index a050ba9..385dd5d 100644 --- a/src/web/components/mod.rs +++ b/src/web/components/mod.rs @@ -1,2 +1,3 @@ pub mod marquee; pub mod nav; +pub mod user_miniprofile; diff --git a/src/web/components/nav.rs b/src/web/components/nav.rs index 856722a..92171b7 100644 --- a/src/web/components/nav.rs +++ b/src/web/components/nav.rs @@ -2,30 +2,33 @@ use maud::{Markup, PreEscaped, html}; use crate::{users::User, web::icons}; -const LINKS: &[(&str, &str, &str)] = &[ - ("Dashboard", "/dashboard", icons::LAYOUT_DASHBOARD), - ("Quotes", "#quotes", icons::SCROLL_TEXT), - ("Photos", "#photos", icons::FILE_IMAGE), - ("Persons", "#persons", icons::CONTACT), - ("Tags", "#tags", icons::TAG), - ("Users", "#users", icons::USERS), - ("Logs", "#logs", icons::CLIPBOARD_CLOCK), +// (SHOWTEXT, LINK, ICON, REQUIRES_LOG_IN) +const LINKS: &[(&str, &str, &str, bool)] = &[ + ("Dashboard", "/dashboard", icons::LAYOUT_DASHBOARD, false), + ("Quotes", "#quotes", icons::SCROLL_TEXT, false), + ("Photos", "#photos", icons::FILE_IMAGE, false), + ("Persons", "#persons", icons::CONTACT, false), + ("Tags", "#tags", icons::TAG, false), + ("Users", "/users", icons::USERS, true), + ("Logs", "#logs", icons::CLIPBOARD_CLOCK, true), ]; -pub fn nav(user: Option, uri: &str) -> Markup { +pub fn nav(user: Option<&User>, uri: &str) -> Markup { html!( div class="flex items-center text-sm gap-4 border-b border-neutral-200/25 bg-neutral-200/5 px-4 py-2" { a href="/dashboard" class="font-lora font-semibold hidden xs:block md:text-xl sm:mr-2" {"Mnemosyne"} div class="w-px h-5 bg-neutral-200/15 hidden sm:block" {} div class="flex flex-row" { @for link in LINKS { - a href={(link.1)} class="flex flex-row px-2 py-1 rounded items-center gap-2 hover:bg-neutral-200/5 border border-transparent hover:border-neutral-200/25" { - @if uri.starts_with(link.1) { - div class="scale-[.75] text-neutral-300" {(PreEscaped(link.2))} - span class="text-neutral-200 font-light hidden lg:block" { (link.0) } - } @else { - div class="scale-[.75] text-neutral-500" {(PreEscaped(link.2))} - span class="text-neutral-400 font-light hidden lg:block" { (link.0) } + @if !link.3 || user.is_some() { + a href={(link.1)} class="flex flex-row px-2 py-1 rounded items-center gap-2 hover:bg-neutral-200/5 border border-transparent hover:border-neutral-200/25" { + @if uri.starts_with(link.1) { + div class="scale-[.75] text-neutral-300" {(PreEscaped(link.2))} + span class="text-neutral-200 font-light hidden lg:block" { (link.0) } + } @else { + div class="scale-[.75] text-neutral-500" {(PreEscaped(link.2))} + span class="text-neutral-400 font-light hidden lg:block" { (link.0) } + } } } } diff --git a/src/web/components/user_miniprofile.rs b/src/web/components/user_miniprofile.rs new file mode 100644 index 0000000..b465642 --- /dev/null +++ b/src/web/components/user_miniprofile.rs @@ -0,0 +1,33 @@ +use maud::{Markup, PreEscaped, html}; + +use crate::{users::User, web::icons}; + +pub fn user_miniprofile(u: &User) -> Markup { + let show_shield = u.is_infradmin() || u.is_systemuser(); + html!( + a href=(format!("/users/{}", u.id)) + class="w-70 border border-neutral-200/25 hover:border-neutral-200/50 bg-neutral-200/5 hover:bg-neutral-200/10 transition-colors rounded flex" { + div class="bg-neutral-200/10 text-neutral-300 font-semibold aspect-square flex items-center justify-center" { + (u.handle.as_str().chars().next().unwrap_or('?').to_uppercase()) + } + div class="p-3" { + p class="text-semibold flex" { + (u.handle) + @if show_shield { + span class="scale-[.75] text-neutral-500" + title="This is a special internal user." {(PreEscaped(icons::SHIELD_USER))} + } + } + p class="text-xs text-neutral-500 flex items-center mt-1" { + @if show_shield { + span class="scale-[.5] -ml-1" {(PreEscaped(icons::SERVER))} + "System account" + } @else { + span class="scale-[.5] -ml-1" {(PreEscaped(icons::CALENDAR_1))} + (u.created_at().map_or("Unknown".into(), |d| d.to_string())) + } + } + } + } + ) +} diff --git a/src/web/icons/calendar-1.svg b/src/web/icons/calendar-1.svg new file mode 100644 index 0000000..e551984 --- /dev/null +++ b/src/web/icons/calendar-1.svg @@ -0,0 +1 @@ + diff --git a/src/web/icons/mod.rs b/src/web/icons/mod.rs index f11d9a3..6fda091 100644 --- a/src/web/icons/mod.rs +++ b/src/web/icons/mod.rs @@ -1,10 +1,13 @@ // Below icons sourced from https://lucide.dev pub const ARROW_RIGHT: &str = include_str!("arrow-right.svg"); +pub const CALENDAR_1: &str = include_str!("calendar-1.svg"); pub const CLIPBOARD_CLOCK: &str = include_str!("clipboard-clock.svg"); pub const CONTACT: &str = include_str!("contact.svg"); pub const FILE_IMAGE: &str = include_str!("file-image.svg"); pub const LAYOUT_DASHBOARD: &str = include_str!("layout-dashboard.svg"); pub const SCROLL_TEXT: &str = include_str!("scroll-text.svg"); +pub const SERVER: &str = include_str!("server.svg"); +pub const SHIELD_USER: &str = include_str!("shield-user.svg"); pub const TAG: &str = include_str!("tag.svg"); pub const USER: &str = include_str!("user.svg"); pub const USER_KEY: &str = include_str!("user-key.svg"); diff --git a/src/web/icons/server.svg b/src/web/icons/server.svg new file mode 100644 index 0000000..ef896e6 --- /dev/null +++ b/src/web/icons/server.svg @@ -0,0 +1 @@ + diff --git a/src/web/icons/shield-user.svg b/src/web/icons/shield-user.svg new file mode 100644 index 0000000..af35cba --- /dev/null +++ b/src/web/icons/shield-user.svg @@ -0,0 +1 @@ + diff --git a/src/web/pages/dashboard.rs b/src/web/pages/dashboard.rs index 9c2f580..08f0e33 100644 --- a/src/web/pages/dashboard.rs +++ b/src/web/pages/dashboard.rs @@ -11,7 +11,7 @@ pub async fn page(req: Request) -> Markup { base( "Dashboard | Mnemosyne", html!( - (nav(u, req.uri().path())) + (nav(u.as_ref(), req.uri().path())) div class="text-6xl sm:text-8xl text-neutral-800/25 mt-16 text-center font-semibold font-lora select-none overflow-hidden" {"Mnemosyne"} ), diff --git a/src/web/pages/mod.rs b/src/web/pages/mod.rs index dbeeda5..da2c150 100644 --- a/src/web/pages/mod.rs +++ b/src/web/pages/mod.rs @@ -4,12 +4,14 @@ use maud::{DOCTYPE, Markup, html}; pub mod dashboard; pub mod index; pub mod login; +pub mod users; pub fn pages() -> Router { Router::new() .route("/", get(index::page)) .route("/login", get(login::page)) .route("/dashboard", get(dashboard::page)) + .route("/users", get(users::page)) } pub fn base(title: &str, inner: Markup) -> Markup { diff --git a/src/web/pages/users.rs b/src/web/pages/users.rs new file mode 100644 index 0000000..91144ea --- /dev/null +++ b/src/web/pages/users.rs @@ -0,0 +1,58 @@ +use axum::{ + extract::Request, + response::{IntoResponse, Response}, +}; +use maud::{PreEscaped, html}; + +use crate::{ + users::{ + User, + auth::{AuthError, UserAuthenticate}, + }, + web::{ + components::{nav::nav, user_miniprofile::user_miniprofile}, + icons, + pages::base, + }, +}; + +pub async fn page(req: Request) -> Result { + let u = User::authenticate(req.headers())?; + let us = match u.is_some() { + true => User::get_all(), + false => Ok(vec![]), + }; + + Ok(base( + "Users | Mnemosyne", + html!( + (nav(u.as_ref(), req.uri().path())) + + @if let Some(_) = u { + div class="mx-auto max-w-4xl px-2 my-4" { + p class="flex items-center gap-2" { + span class="text-neutral-500" {(PreEscaped(icons::USERS))} + span class="text-2xl font-semibold font-lora" {"Users"} + } + p class="text-neutral-500 text-sm font-light" { + @if let Ok(v) = &us { + (v.len()) " users registered with Mnemosyne." + } + } + } + div class="mx-auto max-w-4xl flex flex-wrap gap-4" { + @if let Ok(vec) = &us { + @for user in vec { + (user_miniprofile(user)) + } + } @else { + p class="text-center py-4 text-light text-red-500" {"Failed to load users."} + } + } + } @else { + p class="text-center p-2" {"You must be logged in to view this page."} + } + ), + ) + .into_response()) +} diff --git a/src/web/styles.css b/src/web/styles.css index 4478aab..b3b6e21 100644 --- a/src/web/styles.css +++ b/src/web/styles.css @@ -1,2 +1,2 @@ /*! tailwindcss v4.2.1 | MIT License | https://tailwindcss.com */ -@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-border-style:solid;--tw-font-weight:initial;--tw-tracking:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-red-400:oklch(70.4% .191 22.216);--color-neutral-200:oklch(92.2% 0 0);--color-neutral-300:oklch(87% 0 0);--color-neutral-400:oklch(70.8% 0 0);--color-neutral-500:oklch(55.6% 0 0);--color-neutral-800:oklch(26.9% 0 0);--color-neutral-900:oklch(20.5% 0 0);--color-neutral-950:oklch(14.5% 0 0);--spacing:.25rem;--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--text-4xl:2.25rem;--text-4xl--line-height:calc(2.5 / 2.25);--text-6xl:3.75rem;--text-6xl--line-height:1;--text-8xl:6rem;--text-8xl--line-height:1;--font-weight-light:300;--font-weight-normal:400;--font-weight-semibold:600;--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono);--font-lora:"Lora", ui-serif, Georgia, Cambria, "Times New Roman", serif;--font-lexend:"Lexend", sans-serif;--animate-marquee:marquee 180s linear infinite}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab, red, red)){::placeholder{color:color-mix(in oklab, currentcolor 50%, transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.static{position:static}.mx-2{margin-inline:calc(var(--spacing) * 2)}.mx-auto{margin-inline:auto}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-8{margin-top:calc(var(--spacing) * 8)}.mt-16{margin-top:calc(var(--spacing) * 16)}.mt-24{margin-top:calc(var(--spacing) * 24)}.mt-auto{margin-top:auto}.mb-16{margin-bottom:calc(var(--spacing) * 16)}.ml-auto{margin-left:auto}.block{display:block}.flex{display:flex}.hidden{display:none}.h-5{height:calc(var(--spacing) * 5)}.min-h-screen{min-height:100vh}.w-4\/5{width:80%}.w-full{width:100%}.w-px{width:1px}.shrink-0{flex-shrink:0}.scale-\[\.75\]{scale:.75}.animate-marquee{animation:var(--animate-marquee)}.cursor-pointer{cursor:pointer}.flex-col{flex-direction:column}.flex-row{flex-direction:row}.items-center{align-items:center}.justify-between{justify-content:space-between}.gap-2{gap:calc(var(--spacing) * 2)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-8{gap:calc(var(--spacing) * 8)}.overflow-hidden{overflow:hidden}.overflow-x-hidden{overflow-x:hidden}.rounded{border-radius:.25rem}.border{border-style:var(--tw-border-style);border-width:1px}.border-y{border-block-style:var(--tw-border-style);border-block-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-neutral-200\/25{border-color:#e5e5e540}@supports (color:color-mix(in lab, red, red)){.border-neutral-200\/25{border-color:color-mix(in oklab, var(--color-neutral-200) 25%, transparent)}}.border-neutral-500\/20{border-color:#73737333}@supports (color:color-mix(in lab, red, red)){.border-neutral-500\/20{border-color:color-mix(in oklab, var(--color-neutral-500) 20%, transparent)}}.border-transparent{border-color:#0000}.bg-neutral-200\/5{background-color:#e5e5e50d}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/5{background-color:color-mix(in oklab, var(--color-neutral-200) 5%, transparent)}}.bg-neutral-200\/15{background-color:#e5e5e526}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/15{background-color:color-mix(in oklab, var(--color-neutral-200) 15%, transparent)}}.bg-neutral-900{background-color:var(--color-neutral-900)}.bg-neutral-950\/50{background-color:#0a0a0a80}@supports (color:color-mix(in lab, red, red)){.bg-neutral-950\/50{background-color:color-mix(in oklab, var(--color-neutral-950) 50%, transparent)}}.bg-transparent{background-color:#0000}.p-4{padding:calc(var(--spacing) * 4)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-4{padding-inline:calc(var(--spacing) * 4)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-3{padding-block:calc(var(--spacing) * 3)}.pr-1{padding-right:calc(var(--spacing) * 1)}.pr-8{padding-right:calc(var(--spacing) * 8)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-\[2px\]{padding-left:2px}.text-center{text-align:center}.font-lexend{font-family:var(--font-lexend)}.font-lora{font-family:var(--font-lora)}.text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-\[10px\]{font-size:10px}.font-light{--tw-font-weight:var(--font-weight-light);font-weight:var(--font-weight-light)}.font-normal{--tw-font-weight:var(--font-weight-normal);font-weight:var(--font-weight-normal)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.3em\]{--tw-tracking:.3em;letter-spacing:.3em}.text-neutral-200{color:var(--color-neutral-200)}.text-neutral-300{color:var(--color-neutral-300)}.text-neutral-400{color:var(--color-neutral-400)}.text-neutral-500{color:var(--color-neutral-500)}.text-neutral-500\/40{color:#73737366}@supports (color:color-mix(in lab, red, red)){.text-neutral-500\/40{color:color-mix(in oklab, var(--color-neutral-500) 40%, transparent)}}.text-neutral-800\/25{color:#26262640}@supports (color:color-mix(in lab, red, red)){.text-neutral-800\/25{color:color-mix(in oklab, var(--color-neutral-800) 25%, transparent)}}.text-red-400{color:var(--color-red-400)}.uppercase{text-transform:uppercase}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}@media (hover:hover){.hover\:border-neutral-200\/25:hover{border-color:#e5e5e540}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/25:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 25%, transparent)}}.hover\:bg-neutral-200\/5:hover{background-color:#e5e5e50d}@supports (color:color-mix(in lab, red, red)){.hover\:bg-neutral-200\/5:hover{background-color:color-mix(in oklab, var(--color-neutral-200) 5%, transparent)}}.hover\:underline:hover{text-decoration-line:underline}}@media (min-width:30rem){.xs\:block{display:block}}@media (min-width:40rem){.sm\:mx-0{margin-inline:calc(var(--spacing) * 0)}.sm\:mr-2{margin-right:calc(var(--spacing) * 2)}.sm\:block{display:block}.sm\:hidden{display:none}.sm\:w-fit{width:fit-content}.sm\:text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.sm\:text-8xl{font-size:var(--text-8xl);line-height:var(--tw-leading,var(--text-8xl--line-height))}}@media (min-width:48rem){.md\:text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}}@media (min-width:64rem){.lg\:block{display:block}}}@keyframes marquee{0%{transform:translate(0%)}to{transform:translate(-100%)}}@media (prefers-reduced-motion:reduce){.animate-marquee{animation-play-state:paused}}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false} \ No newline at end of file +@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-border-style:solid;--tw-font-weight:initial;--tw-tracking:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-red-400:oklch(70.4% .191 22.216);--color-red-500:oklch(63.7% .237 25.331);--color-neutral-200:oklch(92.2% 0 0);--color-neutral-300:oklch(87% 0 0);--color-neutral-400:oklch(70.8% 0 0);--color-neutral-500:oklch(55.6% 0 0);--color-neutral-800:oklch(26.9% 0 0);--color-neutral-900:oklch(20.5% 0 0);--color-neutral-950:oklch(14.5% 0 0);--spacing:.25rem;--container-4xl:56rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--text-2xl:1.5rem;--text-2xl--line-height:calc(2 / 1.5);--text-4xl:2.25rem;--text-4xl--line-height:calc(2.5 / 2.25);--text-6xl:3.75rem;--text-6xl--line-height:1;--text-8xl:6rem;--text-8xl--line-height:1;--font-weight-light:300;--font-weight-normal:400;--font-weight-semibold:600;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono);--font-lora:"Lora", ui-serif, Georgia, Cambria, "Times New Roman", serif;--font-lexend:"Lexend", sans-serif;--animate-marquee:marquee 180s linear infinite}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab, red, red)){::placeholder{color:color-mix(in oklab, currentcolor 50%, transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.static{position:static}.mx-2{margin-inline:calc(var(--spacing) * 2)}.mx-auto{margin-inline:auto}.my-4{margin-block:calc(var(--spacing) * 4)}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-8{margin-top:calc(var(--spacing) * 8)}.mt-16{margin-top:calc(var(--spacing) * 16)}.mt-24{margin-top:calc(var(--spacing) * 24)}.mt-auto{margin-top:auto}.mb-16{margin-bottom:calc(var(--spacing) * 16)}.-ml-1{margin-left:calc(var(--spacing) * -1)}.ml-auto{margin-left:auto}.block{display:block}.flex{display:flex}.hidden{display:none}.aspect-square{aspect-ratio:1}.h-5{height:calc(var(--spacing) * 5)}.min-h-screen{min-height:100vh}.w-4\/5{width:80%}.w-70{width:calc(var(--spacing) * 70)}.w-full{width:100%}.w-px{width:1px}.max-w-4xl{max-width:var(--container-4xl)}.shrink-0{flex-shrink:0}.scale-\[\.5\]{scale:.5}.scale-\[\.75\]{scale:.75}.animate-marquee{animation:var(--animate-marquee)}.cursor-pointer{cursor:pointer}.flex-col{flex-direction:column}.flex-row{flex-direction:row}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-2{gap:calc(var(--spacing) * 2)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-8{gap:calc(var(--spacing) * 8)}.overflow-hidden{overflow:hidden}.overflow-x-hidden{overflow-x:hidden}.rounded{border-radius:.25rem}.border{border-style:var(--tw-border-style);border-width:1px}.border-y{border-block-style:var(--tw-border-style);border-block-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-neutral-200\/25{border-color:#e5e5e540}@supports (color:color-mix(in lab, red, red)){.border-neutral-200\/25{border-color:color-mix(in oklab, var(--color-neutral-200) 25%, transparent)}}.border-neutral-500\/20{border-color:#73737333}@supports (color:color-mix(in lab, red, red)){.border-neutral-500\/20{border-color:color-mix(in oklab, var(--color-neutral-500) 20%, transparent)}}.border-transparent{border-color:#0000}.bg-neutral-200\/5{background-color:#e5e5e50d}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/5{background-color:color-mix(in oklab, var(--color-neutral-200) 5%, transparent)}}.bg-neutral-200\/10{background-color:#e5e5e51a}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/10{background-color:color-mix(in oklab, var(--color-neutral-200) 10%, transparent)}}.bg-neutral-200\/15{background-color:#e5e5e526}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/15{background-color:color-mix(in oklab, var(--color-neutral-200) 15%, transparent)}}.bg-neutral-900{background-color:var(--color-neutral-900)}.bg-neutral-950\/50{background-color:#0a0a0a80}@supports (color:color-mix(in lab, red, red)){.bg-neutral-950\/50{background-color:color-mix(in oklab, var(--color-neutral-950) 50%, transparent)}}.bg-transparent{background-color:#0000}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-4{padding-inline:calc(var(--spacing) * 4)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-4{padding-block:calc(var(--spacing) * 4)}.pr-1{padding-right:calc(var(--spacing) * 1)}.pr-8{padding-right:calc(var(--spacing) * 8)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-\[2px\]{padding-left:2px}.text-center{text-align:center}.font-lexend{font-family:var(--font-lexend)}.font-lora{font-family:var(--font-lora)}.text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[10px\]{font-size:10px}.font-light{--tw-font-weight:var(--font-weight-light);font-weight:var(--font-weight-light)}.font-normal{--tw-font-weight:var(--font-weight-normal);font-weight:var(--font-weight-normal)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.3em\]{--tw-tracking:.3em;letter-spacing:.3em}.text-neutral-200{color:var(--color-neutral-200)}.text-neutral-300{color:var(--color-neutral-300)}.text-neutral-400{color:var(--color-neutral-400)}.text-neutral-500{color:var(--color-neutral-500)}.text-neutral-500\/40{color:#73737366}@supports (color:color-mix(in lab, red, red)){.text-neutral-500\/40{color:color-mix(in oklab, var(--color-neutral-500) 40%, transparent)}}.text-neutral-800\/25{color:#26262640}@supports (color:color-mix(in lab, red, red)){.text-neutral-800\/25{color:color-mix(in oklab, var(--color-neutral-800) 25%, transparent)}}.text-red-400{color:var(--color-red-400)}.text-red-500{color:var(--color-red-500)}.uppercase{text-transform:uppercase}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}@media (hover:hover){.hover\:border-neutral-200\/25:hover{border-color:#e5e5e540}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/25:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 25%, transparent)}}.hover\:border-neutral-200\/50:hover{border-color:#e5e5e580}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/50:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 50%, transparent)}}.hover\:bg-neutral-200\/5:hover{background-color:#e5e5e50d}@supports (color:color-mix(in lab, red, red)){.hover\:bg-neutral-200\/5:hover{background-color:color-mix(in oklab, var(--color-neutral-200) 5%, transparent)}}.hover\:bg-neutral-200\/10:hover{background-color:#e5e5e51a}@supports (color:color-mix(in lab, red, red)){.hover\:bg-neutral-200\/10:hover{background-color:color-mix(in oklab, var(--color-neutral-200) 10%, transparent)}}.hover\:underline:hover{text-decoration-line:underline}}@media (min-width:30rem){.xs\:block{display:block}}@media (min-width:40rem){.sm\:mx-0{margin-inline:calc(var(--spacing) * 0)}.sm\:mr-2{margin-right:calc(var(--spacing) * 2)}.sm\:block{display:block}.sm\:hidden{display:none}.sm\:w-fit{width:fit-content}.sm\:text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.sm\:text-8xl{font-size:var(--text-8xl);line-height:var(--tw-leading,var(--text-8xl--line-height))}}@media (min-width:48rem){.md\:text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}}@media (min-width:64rem){.lg\:block{display:block}}}@keyframes marquee{0%{transform:translate(0%)}to{transform:translate(-100%)}}@media (prefers-reduced-motion:reduce){.animate-marquee{animation-play-state:paused}}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false} \ No newline at end of file