From 3ab3567ac31da5b80dd925bde75af65820cf4f4b Mon Sep 17 00:00:00 2001 From: jmanczak Date: Sun, 5 Apr 2026 15:48:30 +0200 Subject: [PATCH 1/7] Name::get_all, quotes page stub, quote add UI work, icons --- src/persons/mod.rs | 14 +++++ src/web/components/nav.rs | 2 +- src/web/icons/mod.rs | 1 + src/web/icons/plus.svg | 1 + src/web/pages/mod.rs | 3 ++ src/web/pages/quotes.rs | 43 +++++++++++++++ src/web/pages/quotes/add.rs | 102 ++++++++++++++++++++++++++++++++++++ src/web/styles.css | 2 +- 8 files changed, 166 insertions(+), 2 deletions(-) create mode 100644 src/web/icons/plus.svg create mode 100644 src/web/pages/quotes.rs create mode 100644 src/web/pages/quotes/add.rs diff --git a/src/persons/mod.rs b/src/persons/mod.rs index 906172c..4954e0f 100644 --- a/src/persons/mod.rs +++ b/src/persons/mod.rs @@ -157,6 +157,20 @@ impl Name { None => Err(PersonError::NoNameWithId(id)), } } + pub fn get_all(conn: &Connection) -> Result, PersonError> { + Ok(conn + .prepare("SELECT id, is_primary, person_id, created_by, name FROM names")? + .query_map((), |r| { + Ok(Name { + id: r.get(0)?, + is_primary: r.get(1)?, + person_id: r.get(2)?, + created_by: r.get(3)?, + name: r.get(4)?, + }) + })? + .collect::, _>>()?) + } pub fn set_primary(&mut self, conn: &Connection) -> Result<(), PersonError> { if self.is_primary { return Err(PersonError::AlreadyPrimary); diff --git a/src/web/components/nav.rs b/src/web/components/nav.rs index eaf3ef0..1f54fc8 100644 --- a/src/web/components/nav.rs +++ b/src/web/components/nav.rs @@ -5,7 +5,7 @@ use crate::{users::User, web::icons}; // (SHOWTEXT, LINK, ICON, REQUIRES_LOG_IN) const LINKS: &[(&str, &str, &str, bool)] = &[ ("Dashboard", "/dashboard", icons::LAYOUT_DASHBOARD, false), - ("Quotes", "#quotes", icons::SCROLL_TEXT, false), + ("Quotes", "/quotes", icons::SCROLL_TEXT, false), ("Photos", "#photos", icons::FILE_IMAGE, false), ("Persons", "/persons", icons::CONTACT, false), ("Tags", "/tags", icons::TAG, false), diff --git a/src/web/icons/mod.rs b/src/web/icons/mod.rs index 385e8be..5a8e1d8 100644 --- a/src/web/icons/mod.rs +++ b/src/web/icons/mod.rs @@ -11,6 +11,7 @@ pub const LAYOUT_DASHBOARD: &str = include_str!("layout-dashboard.svg"); pub const LOG_OUT: &str = include_str!("log-out.svg"); pub const MAP_PIN: &str = include_str!("map-pin.svg"); pub const PEN: &str = include_str!("pen.svg"); +pub const PLUS: &str = include_str!("plus.svg"); pub const QUOTE: &str = include_str!("quote.svg"); pub const SCROLL_TEXT: &str = include_str!("scroll-text.svg"); pub const SERVER: &str = include_str!("server.svg"); diff --git a/src/web/icons/plus.svg b/src/web/icons/plus.svg new file mode 100644 index 0000000..44adbb4 --- /dev/null +++ b/src/web/icons/plus.svg @@ -0,0 +1 @@ + diff --git a/src/web/pages/mod.rs b/src/web/pages/mod.rs index 4ff3aef..d947723 100644 --- a/src/web/pages/mod.rs +++ b/src/web/pages/mod.rs @@ -9,6 +9,7 @@ pub mod index; pub mod login; pub mod logs; pub mod persons; +pub mod quotes; pub mod tags; pub mod users; pub mod usersettings; @@ -30,6 +31,8 @@ pub fn pages() -> Router { .route("/persons", get(persons::page)) .route("/persons/create", post(persons::create)) .route("/logs", get(logs::page)) + .route("/quotes", get(quotes::page)) + .route("/quotes/add", get(quotes::add::page)) } pub fn base(title: &str, inner: Markup) -> Markup { diff --git a/src/web/pages/quotes.rs b/src/web/pages/quotes.rs new file mode 100644 index 0000000..6620ac9 --- /dev/null +++ b/src/web/pages/quotes.rs @@ -0,0 +1,43 @@ +use axum::{ + extract::Request, + response::{IntoResponse, Response}, +}; +use maud::{PreEscaped, html}; + +use crate::{ + error::CompositeError, + users::{User, auth::UserAuthenticate}, + web::{components::nav::nav, icons, pages::base}, +}; + +pub mod add; + +pub async fn page(req: Request) -> Result { + let u = User::authenticate(req.headers())?; + + Ok(base( + "Persons | Mnemosyne", + html!( + (nav(u.as_ref(), req.uri().path())) + + div class="max-w-4xl mx-auto px-2" { + div class="my-4 flex justify-between" { + p class="flex items-center gap-2" { + span class="text-neutral-500" {(PreEscaped(icons::SCROLL_TEXT))} + span class="text-2xl font-semibold font-lora" {"Quotes"} + } + @if let Some(_) = u { + a href="/quotes/add" class="group border rounded flex items-center gap-1 px-2 border-neutral-200/25 hover:border-neutral-200/45 bg-neutral-400/5 hover:bg-neutral-400/10" { + span class="text-neutral-300 group-hover:text-neutral-200" {(PreEscaped(icons::PLUS))} + span class="text-neutral-300 group-hover:text-neutral-200" {"Add quote"} + } + } + } + input class="border w-full border-neutral-200/25 hover:border-neutral-200/45 bg-neutral-950/50 p-2 rounded" + placeholder="Search for quotes..."; + div class="text-center p-4" {"Search not yet implemented."} + } + ), + ) + .into_response()) +} diff --git a/src/web/pages/quotes/add.rs b/src/web/pages/quotes/add.rs new file mode 100644 index 0000000..bfba9fc --- /dev/null +++ b/src/web/pages/quotes/add.rs @@ -0,0 +1,102 @@ +use axum::{ + extract::Request, + response::{IntoResponse, Response}, +}; +use maud::{PreEscaped, html}; + +use crate::{ + database, + error::CompositeError, + persons::Name, + users::{User, auth::UserAuthenticate}, + web::{components::nav::nav, icons, pages::base}, +}; + +pub async fn page(req: Request) -> Result { + let u = User::authenticate(req.headers())?; + let conn = database::conn()?; + let names = Name::get_all(&conn)?; + + Ok(base( + "Persons | Mnemosyne", + html!( + (nav(u.as_ref(), req.uri().path())) + + div class="max-w-4xl mx-auto px-2" { + div class="my-4 flex justify-between" { + p class="flex items-center gap-2" { + span class="text-neutral-500" {(PreEscaped(icons::SCROLL_TEXT))} + span class="text-2xl font-semibold font-lora" {"Quote Maker"} + } + } + div class="border border-neutral-200/25 bg-neutral-200/5 rounded-md p-4 flex flex-col" { + @for i in 1..=2 { + div class="flex justify-between gap-4" { + div class="flex flex-col flex-1" { + label class="w-full" { + p class="mb-1" {(format!("Quote Line #{i}"))} + input type="text" name="quoteline" placeholder="They said..." autocomplete="off" + class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; + } + // label for=(format!("line-{i}")) class="mb-1" {(format!("Quote Line #{i}"))} + // input type="text" id=(format!("line-{i}")) name=(format!("line-{i}")) + // placeholder=(format!("They said...")) autocomplete="off" + // class="px-2 py-1 mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; + } + div class="flex flex-col" { + label { + p class="mb-1" {(format!("Quote Author #{i}"))} + select name="quoteauthor" autocomplete="off" + class="px-2 py-1.5 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"{ + option {"--"} + @for name in &names { + option {(name.name)} + } + } + } + // label for=(format!("who-{i}")) class="mb-1" {(format!("Quote Author #{i}"))} + // select id=(format!("line-{i}")) name=(format!("line-{i}")) autocomplete="off" + // class="px-2 py-1.5 mb-2 bg-neutral-950/50 rounded border border-neutral-200/25" { + // option {"--"} + // @for name in &names { + // option {(name.name)} + // } + // } + } + } + } + hr class="border-neutral-200/25 my-4"; + div class="flex gap-4 justify-between" { + div class="flex flex-col flex-1" { + label class="w-full"{ + p class="mb-1" {"Location"} + input type="text" name="location" autocomplete="off" placeholder="Right there!" + class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; + } + } + div class="flex flex-col flex-1" { + label class="w-full" { + p class="mb-1" {"Time of utterance"} + input type="text" name="time" autocomplete="off" placeholder="2026-04-05T01:14:05+02:00" + class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; + } + } + } + div class="flex gap-4 justify-between" { + div class="flex flex-col flex-1" { + label class="w-full" { + p class="mb-1" {"Context"} + input type="text" name="context" autocomplete="off" placeholder="It was like this.." + class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; + } + } + button type="submit" class="border mt-auto mb-2 cursor-pointer rounded h-fit px-2 py-1 bg-neutral-200/5 border-neutral-200/25 hover:border-neutral-200/45 hover:bg-neutral-200/15" { + "Submit" + } + } + } + } + ), + ) + .into_response()) +} diff --git a/src/web/styles.css b/src/web/styles.css index ebe98ff..e64067b 100644 --- a/src/web/styles.css +++ b/src/web/styles.css @@ -1,2 +1,2 @@ /*! tailwindcss v4.2.2 | 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-scale-x:1;--tw-scale-y:1;--tw-scale-z:1;--tw-space-y-reverse:0;--tw-border-style:solid;--tw-gradient-position:initial;--tw-gradient-from:#0000;--tw-gradient-via:#0000;--tw-gradient-to:#0000;--tw-gradient-stops:initial;--tw-gradient-via-stops:initial;--tw-gradient-from-position:0%;--tw-gradient-via-position:50%;--tw-gradient-to-position:100%;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-duration: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-300:oklch(80.8% .114 19.571);--color-red-400:oklch(70.4% .191 22.216);--color-red-500:oklch(63.7% .237 25.331);--color-emerald-400:oklch(76.5% .177 163.223);--color-emerald-500:oklch(69.6% .17 162.48);--color-emerald-950:oklch(26.2% .051 172.552);--color-blue-400:oklch(70.7% .165 254.624);--color-blue-500:oklch(62.3% .214 259.815);--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-600:oklch(43.9% 0 0);--color-neutral-700:oklch(37.1% 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-lg:1.125rem;--text-lg--line-height:calc(1.75 / 1.125);--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-5xl:3rem;--text-5xl--line-height:1;--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;--tracking-wide:.025em;--tracking-wider:.05em;--leading-relaxed:1.625;--radius-md:.375rem;--radius-lg:.5rem;--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{.invisible{visibility:hidden}.absolute{position:absolute}.relative{position:relative}.static{position:static}.end{inset-inline-end:var(--spacing)}.top-1{top:calc(var(--spacing) * 1)}.top-4{top:calc(var(--spacing) * 4)}.top-full{top:100%}.right-0{right:calc(var(--spacing) * 0)}.right-1{right:calc(var(--spacing) * 1)}.right-6{right:calc(var(--spacing) * 6)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-1{bottom:calc(var(--spacing) * 1)}.left-0{left:calc(var(--spacing) * 0)}.left-1{left:calc(var(--spacing) * 1)}.z-50{z-index:50}.col-span-3{grid-column:span 3/span 3}.container{width:100%}@media (min-width:30rem){.container{max-width:30rem}}@media (min-width:40rem){.container{max-width:40rem}}@media (min-width:48rem){.container{max-width:48rem}}@media (min-width:64rem){.container{max-width:64rem}}@media (min-width:80rem){.container{max-width:80rem}}@media (min-width:96rem){.container{max-width:96rem}}.mx-2{margin-inline:calc(var(--spacing) * 2)}.mx-auto{margin-inline:auto}.my-4{margin-block:calc(var(--spacing) * 4)}.my-6{margin-block:calc(var(--spacing) * 6)}.my-auto{margin-block:auto}.-mt-16{margin-top:calc(var(--spacing) * -16)}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.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-6{margin-top:calc(var(--spacing) * 6)}.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}.mr-1{margin-right:calc(var(--spacing) * 1)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.mb-4{margin-bottom:calc(var(--spacing) * 4)}.mb-12{margin-bottom:calc(var(--spacing) * 12)}.mb-16{margin-bottom:calc(var(--spacing) * 16)}.-ml-1{margin-left:calc(var(--spacing) * -1)}.ml-1{margin-left:calc(var(--spacing) * 1)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-3{margin-left:calc(var(--spacing) * 3)}.ml-6{margin-left:calc(var(--spacing) * 6)}.ml-auto{margin-left:auto}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.aspect-square{aspect-ratio:1}.h-2\/3{height:66.6667%}.h-5{height:calc(var(--spacing) * 5)}.h-28{height:calc(var(--spacing) * 28)}.h-48{height:calc(var(--spacing) * 48)}.h-full{height:100%}.h-px{height:1px}.min-h-screen{min-height:100vh}.w-4\/5{width:80%}.w-28{width:calc(var(--spacing) * 28)}.w-40{width:calc(var(--spacing) * 40)}.w-64{width:calc(var(--spacing) * 64)}.w-70{width:calc(var(--spacing) * 70)}.w-fit{width:fit-content}.w-full{width:100%}.w-px{width:1px}.max-w-4xl{max-width:var(--container-4xl)}.flex-1{flex:1}.shrink-0{flex-shrink:0}.scale-125{--tw-scale-x:125%;--tw-scale-y:125%;--tw-scale-z:125%;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-x-\[\.65\]{--tw-scale-x:.65;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-x-\[4\.5\]{--tw-scale-x:4.5;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-y-\[\.5\]{--tw-scale-y:.5;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-y-\[4\]{--tw-scale-y:4;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-\[\.5\]{scale:.5}.scale-\[\.7\]{scale:.7}.scale-\[\.8\]{scale:.8}.scale-\[\.65\]{scale:.65}.scale-\[\.75\]{scale:.75}.scale-\[1\.5\]{scale:1.5}.scale-\[2\]{scale:2}.-rotate-12{rotate:-12deg}.animate-marquee{animation:var(--animate-marquee)}.cursor-pointer{cursor:pointer}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-\[auto_auto_1fr\]{grid-template-columns:auto auto 1fr}.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-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-6{gap:calc(var(--spacing) * 6)}.gap-8{gap:calc(var(--spacing) * 8)}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-clip{overflow:clip}.overflow-hidden{overflow:hidden}.overflow-x-hidden{overflow-x:hidden}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.border{border-style:var(--tw-border-style);border-width:1px}.border-4{border-style:var(--tw-border-style);border-width:4px}.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-dashed{--tw-border-style:dashed;border-style:dashed}.border-emerald-500\/30{border-color:#00bb7f4d}@supports (color:color-mix(in lab, red, red)){.border-emerald-500\/30{border-color:color-mix(in oklab, var(--color-emerald-500) 30%, transparent)}}.border-neutral-200\/10{border-color:#e5e5e51a}@supports (color:color-mix(in lab, red, red)){.border-neutral-200\/10{border-color:color-mix(in oklab, var(--color-neutral-200) 10%, transparent)}}.border-neutral-200\/15{border-color:#e5e5e526}@supports (color:color-mix(in lab, red, red)){.border-neutral-200\/15{border-color:color-mix(in oklab, var(--color-neutral-200) 15%, transparent)}}.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-neutral-600{border-color:var(--color-neutral-600)}.border-neutral-900{border-color:var(--color-neutral-900)}.border-transparent{border-color:#0000}.bg-emerald-500\/10{background-color:#00bb7f1a}@supports (color:color-mix(in lab, red, red)){.bg-emerald-500\/10{background-color:color-mix(in oklab, var(--color-emerald-500) 10%, transparent)}}.bg-neutral-200\/3{background-color:#e5e5e508}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/3{background-color:color-mix(in oklab, var(--color-neutral-200) 3%, transparent)}}.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-500\/50{background-color:#73737380}@supports (color:color-mix(in lab, red, red)){.bg-neutral-500\/50{background-color:color-mix(in oklab, var(--color-neutral-500) 50%, transparent)}}.bg-neutral-800{background-color:var(--color-neutral-800)}.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}.bg-linear-to-b{--tw-gradient-position:to bottom}@supports (background-image:linear-gradient(in lab, red, red)){.bg-linear-to-b{--tw-gradient-position:to bottom in oklab}}.bg-linear-to-b{background-image:linear-gradient(var(--tw-gradient-stops))}.from-neutral-800{--tw-gradient-from:var(--color-neutral-800);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-25\%{--tw-gradient-from-position:25%}.to-emerald-950{--tw-gradient-to:var(--color-emerald-950);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.p-8{padding:calc(var(--spacing) * 8)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.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)}.pt-1{padding-top:calc(var(--spacing) * 1)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pr-1{padding-right:calc(var(--spacing) * 1)}.pr-8{padding-right:calc(var(--spacing) * 8)}.pb-1{padding-bottom:calc(var(--spacing) * 1)}.pl-0\.5{padding-left:calc(var(--spacing) * .5)}.pl-2{padding-left:calc(var(--spacing) * 2)}.text-center{text-align:center}.text-left{text-align:left}.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-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[10px\]{font-size:10px}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.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}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.text-blue-500{color:var(--color-blue-500)}.text-emerald-400{color:var(--color-emerald-400)}.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-700{color:var(--color-neutral-700)}.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-300{color:var(--color-red-300)}.text-red-400{color:var(--color-red-400)}.text-red-500{color:var(--color-red-500)}.uppercase{text-transform:uppercase}.italic{font-style:italic}.opacity-0{opacity:0}.opacity-\[\.3\]{opacity:.3}.opacity-\[\.025\]{opacity:.025}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a), 0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a), 0 8px 10px -6px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring-2{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring-neutral-700\/50{--tw-ring-color:#40404080}@supports (color:color-mix(in lab, red, red)){.ring-neutral-700\/50{--tw-ring-color:color-mix(in oklab, var(--color-neutral-700) 50%, transparent)}}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.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))}.duration-100{--tw-duration:.1s;transition-duration:.1s}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}:is(.\*\:size-3>*){width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.group-focus-within\:visible:is(:where(.group):focus-within *){visibility:visible}.group-focus-within\:opacity-100:is(:where(.group):focus-within *){opacity:1}@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\/35:hover{border-color:#e5e5e559}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/35:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 35%, transparent)}}.hover\:border-neutral-200\/40:hover{border-color:#e5e5e566}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/40:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 40%, 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\:text-blue-400:hover{color:var(--color-blue-400)}.hover\:underline:hover{text-decoration-line:underline}}@media (min-width:30rem){.xs\:block{display:block}.xs\:text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}}@media (min-width:40rem){.sm\:mx-0{margin-inline:calc(var(--spacing) * 0)}.sm\:-mt-20{margin-top:calc(var(--spacing) * -20)}.sm\:mr-2{margin-right:calc(var(--spacing) * 2)}.sm\:block{display:block}.sm\:hidden{display:none}.sm\:h-36{height:calc(var(--spacing) * 36)}.sm\:h-56{height:calc(var(--spacing) * 56)}.sm\:w-36{width:calc(var(--spacing) * 36)}.sm\:w-fit{width:fit-content}.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:flex-row{flex-direction:row}.sm\:items-end{align-items:flex-end}.sm\:gap-6{gap:calc(var(--spacing) * 6)}.sm\:px-6{padding-inline:calc(var(--spacing) * 6)}.sm\:text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.sm\:text-5xl{font-size:var(--text-5xl);line-height:var(--tw-leading,var(--text-5xl--line-height))}.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\:col-span-1{grid-column:span 1/span 1}.md\:col-span-2{grid-column:span 2/span 2}.md\:h-64{height:calc(var(--spacing) * 64)}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.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}}.\[\&\>div\]\:h-full>div{height:100%}}@keyframes marquee{0%{transform:translate(0%)}to{transform:translate(-100%)}}@media (prefers-reduced-motion:reduce){.animate-marquee{animation-play-state:paused}}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-gradient-position{syntax:"*";inherits:false}@property --tw-gradient-from{syntax:"";inherits:false;initial-value:#0000}@property --tw-gradient-via{syntax:"";inherits:false;initial-value:#0000}@property --tw-gradient-to{syntax:"";inherits:false;initial-value:#0000}@property --tw-gradient-stops{syntax:"*";inherits:false}@property --tw-gradient-via-stops{syntax:"*";inherits:false}@property --tw-gradient-from-position{syntax:"";inherits:false;initial-value:0%}@property --tw-gradient-via-position{syntax:"";inherits:false;initial-value:50%}@property --tw-gradient-to-position{syntax:"";inherits:false;initial-value:100%}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-duration{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-scale-x:1;--tw-scale-y:1;--tw-scale-z:1;--tw-space-y-reverse:0;--tw-border-style:solid;--tw-gradient-position:initial;--tw-gradient-from:#0000;--tw-gradient-via:#0000;--tw-gradient-to:#0000;--tw-gradient-stops:initial;--tw-gradient-via-stops:initial;--tw-gradient-from-position:0%;--tw-gradient-via-position:50%;--tw-gradient-to-position:100%;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-duration: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-300:oklch(80.8% .114 19.571);--color-red-400:oklch(70.4% .191 22.216);--color-red-500:oklch(63.7% .237 25.331);--color-emerald-400:oklch(76.5% .177 163.223);--color-emerald-500:oklch(69.6% .17 162.48);--color-emerald-950:oklch(26.2% .051 172.552);--color-blue-400:oklch(70.7% .165 254.624);--color-blue-500:oklch(62.3% .214 259.815);--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-600:oklch(43.9% 0 0);--color-neutral-700:oklch(37.1% 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-lg:1.125rem;--text-lg--line-height:calc(1.75 / 1.125);--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-5xl:3rem;--text-5xl--line-height:1;--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;--tracking-wide:.025em;--tracking-wider:.05em;--leading-relaxed:1.625;--radius-md:.375rem;--radius-lg:.5rem;--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{.invisible{visibility:hidden}.absolute{position:absolute}.relative{position:relative}.static{position:static}.end{inset-inline-end:var(--spacing)}.top-1{top:calc(var(--spacing) * 1)}.top-4{top:calc(var(--spacing) * 4)}.top-full{top:100%}.right-0{right:calc(var(--spacing) * 0)}.right-1{right:calc(var(--spacing) * 1)}.right-6{right:calc(var(--spacing) * 6)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-1{bottom:calc(var(--spacing) * 1)}.left-0{left:calc(var(--spacing) * 0)}.left-1{left:calc(var(--spacing) * 1)}.z-50{z-index:50}.col-span-3{grid-column:span 3/span 3}.container{width:100%}@media (min-width:30rem){.container{max-width:30rem}}@media (min-width:40rem){.container{max-width:40rem}}@media (min-width:48rem){.container{max-width:48rem}}@media (min-width:64rem){.container{max-width:64rem}}@media (min-width:80rem){.container{max-width:80rem}}@media (min-width:96rem){.container{max-width:96rem}}.mx-2{margin-inline:calc(var(--spacing) * 2)}.mx-auto{margin-inline:auto}.my-4{margin-block:calc(var(--spacing) * 4)}.my-6{margin-block:calc(var(--spacing) * 6)}.my-auto{margin-block:auto}.-mt-16{margin-top:calc(var(--spacing) * -16)}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.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-6{margin-top:calc(var(--spacing) * 6)}.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}.mr-1{margin-right:calc(var(--spacing) * 1)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.mb-4{margin-bottom:calc(var(--spacing) * 4)}.mb-12{margin-bottom:calc(var(--spacing) * 12)}.mb-16{margin-bottom:calc(var(--spacing) * 16)}.-ml-1{margin-left:calc(var(--spacing) * -1)}.ml-1{margin-left:calc(var(--spacing) * 1)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-3{margin-left:calc(var(--spacing) * 3)}.ml-6{margin-left:calc(var(--spacing) * 6)}.ml-auto{margin-left:auto}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.aspect-square{aspect-ratio:1}.h-2\/3{height:66.6667%}.h-5{height:calc(var(--spacing) * 5)}.h-28{height:calc(var(--spacing) * 28)}.h-48{height:calc(var(--spacing) * 48)}.h-fit{height:fit-content}.h-full{height:100%}.h-px{height:1px}.min-h-screen{min-height:100vh}.w-4\/5{width:80%}.w-28{width:calc(var(--spacing) * 28)}.w-40{width:calc(var(--spacing) * 40)}.w-64{width:calc(var(--spacing) * 64)}.w-70{width:calc(var(--spacing) * 70)}.w-fit{width:fit-content}.w-full{width:100%}.w-px{width:1px}.max-w-4xl{max-width:var(--container-4xl)}.flex-1{flex:1}.shrink-0{flex-shrink:0}.scale-125{--tw-scale-x:125%;--tw-scale-y:125%;--tw-scale-z:125%;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-x-\[\.65\]{--tw-scale-x:.65;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-x-\[4\.5\]{--tw-scale-x:4.5;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-y-\[\.5\]{--tw-scale-y:.5;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-y-\[4\]{--tw-scale-y:4;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-\[\.5\]{scale:.5}.scale-\[\.7\]{scale:.7}.scale-\[\.8\]{scale:.8}.scale-\[\.65\]{scale:.65}.scale-\[\.75\]{scale:.75}.scale-\[1\.5\]{scale:1.5}.scale-\[2\]{scale:2}.-rotate-12{rotate:-12deg}.animate-marquee{animation:var(--animate-marquee)}.cursor-pointer{cursor:pointer}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-\[auto_auto_1fr\]{grid-template-columns:auto auto 1fr}.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-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-6{gap:calc(var(--spacing) * 6)}.gap-8{gap:calc(var(--spacing) * 8)}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-clip{overflow:clip}.overflow-hidden{overflow:hidden}.overflow-x-hidden{overflow-x:hidden}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.border{border-style:var(--tw-border-style);border-width:1px}.border-4{border-style:var(--tw-border-style);border-width:4px}.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-dashed{--tw-border-style:dashed;border-style:dashed}.border-emerald-500\/30{border-color:#00bb7f4d}@supports (color:color-mix(in lab, red, red)){.border-emerald-500\/30{border-color:color-mix(in oklab, var(--color-emerald-500) 30%, transparent)}}.border-neutral-200\/10{border-color:#e5e5e51a}@supports (color:color-mix(in lab, red, red)){.border-neutral-200\/10{border-color:color-mix(in oklab, var(--color-neutral-200) 10%, transparent)}}.border-neutral-200\/15{border-color:#e5e5e526}@supports (color:color-mix(in lab, red, red)){.border-neutral-200\/15{border-color:color-mix(in oklab, var(--color-neutral-200) 15%, transparent)}}.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-neutral-600{border-color:var(--color-neutral-600)}.border-neutral-900{border-color:var(--color-neutral-900)}.border-transparent{border-color:#0000}.bg-emerald-500\/10{background-color:#00bb7f1a}@supports (color:color-mix(in lab, red, red)){.bg-emerald-500\/10{background-color:color-mix(in oklab, var(--color-emerald-500) 10%, transparent)}}.bg-neutral-200\/3{background-color:#e5e5e508}@supports (color:color-mix(in lab, red, red)){.bg-neutral-200\/3{background-color:color-mix(in oklab, var(--color-neutral-200) 3%, transparent)}}.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-400\/5{background-color:#a1a1a10d}@supports (color:color-mix(in lab, red, red)){.bg-neutral-400\/5{background-color:color-mix(in oklab, var(--color-neutral-400) 5%, transparent)}}.bg-neutral-500\/50{background-color:#73737380}@supports (color:color-mix(in lab, red, red)){.bg-neutral-500\/50{background-color:color-mix(in oklab, var(--color-neutral-500) 50%, transparent)}}.bg-neutral-800{background-color:var(--color-neutral-800)}.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}.bg-linear-to-b{--tw-gradient-position:to bottom}@supports (background-image:linear-gradient(in lab, red, red)){.bg-linear-to-b{--tw-gradient-position:to bottom in oklab}}.bg-linear-to-b{background-image:linear-gradient(var(--tw-gradient-stops))}.from-neutral-800{--tw-gradient-from:var(--color-neutral-800);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-25\%{--tw-gradient-from-position:25%}.to-emerald-950{--tw-gradient-to:var(--color-emerald-950);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.p-8{padding:calc(var(--spacing) * 8)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-4{padding-block:calc(var(--spacing) * 4)}.pt-1{padding-top:calc(var(--spacing) * 1)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pr-1{padding-right:calc(var(--spacing) * 1)}.pr-8{padding-right:calc(var(--spacing) * 8)}.pb-1{padding-bottom:calc(var(--spacing) * 1)}.pl-0\.5{padding-left:calc(var(--spacing) * .5)}.pl-2{padding-left:calc(var(--spacing) * 2)}.text-center{text-align:center}.text-left{text-align:left}.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-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[10px\]{font-size:10px}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.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}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.text-blue-500{color:var(--color-blue-500)}.text-emerald-400{color:var(--color-emerald-400)}.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-700{color:var(--color-neutral-700)}.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-300{color:var(--color-red-300)}.text-red-400{color:var(--color-red-400)}.text-red-500{color:var(--color-red-500)}.uppercase{text-transform:uppercase}.italic{font-style:italic}.opacity-0{opacity:0}.opacity-\[\.3\]{opacity:.3}.opacity-\[\.025\]{opacity:.025}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a), 0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a), 0 8px 10px -6px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring-2{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring-neutral-700\/50{--tw-ring-color:#40404080}@supports (color:color-mix(in lab, red, red)){.ring-neutral-700\/50{--tw-ring-color:color-mix(in oklab, var(--color-neutral-700) 50%, transparent)}}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.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))}.duration-100{--tw-duration:.1s;transition-duration:.1s}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}:is(.\*\:size-3>*){width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.group-focus-within\:visible:is(:where(.group):focus-within *){visibility:visible}.group-focus-within\:opacity-100:is(:where(.group):focus-within *){opacity:1}@media (hover:hover){.group-hover\:text-neutral-200:is(:where(.group):hover *){color:var(--color-neutral-200)}.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\/35:hover{border-color:#e5e5e559}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/35:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 35%, transparent)}}.hover\:border-neutral-200\/40:hover{border-color:#e5e5e566}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/40:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 40%, transparent)}}.hover\:border-neutral-200\/45:hover{border-color:#e5e5e573}@supports (color:color-mix(in lab, red, red)){.hover\:border-neutral-200\/45:hover{border-color:color-mix(in oklab, var(--color-neutral-200) 45%, 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\:bg-neutral-200\/15:hover{background-color:#e5e5e526}@supports (color:color-mix(in lab, red, red)){.hover\:bg-neutral-200\/15:hover{background-color:color-mix(in oklab, var(--color-neutral-200) 15%, transparent)}}.hover\:bg-neutral-400\/10:hover{background-color:#a1a1a11a}@supports (color:color-mix(in lab, red, red)){.hover\:bg-neutral-400\/10:hover{background-color:color-mix(in oklab, var(--color-neutral-400) 10%, transparent)}}.hover\:text-blue-400:hover{color:var(--color-blue-400)}.hover\:underline:hover{text-decoration-line:underline}}@media (min-width:30rem){.xs\:block{display:block}.xs\:text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}}@media (min-width:40rem){.sm\:mx-0{margin-inline:calc(var(--spacing) * 0)}.sm\:-mt-20{margin-top:calc(var(--spacing) * -20)}.sm\:mr-2{margin-right:calc(var(--spacing) * 2)}.sm\:block{display:block}.sm\:hidden{display:none}.sm\:h-36{height:calc(var(--spacing) * 36)}.sm\:h-56{height:calc(var(--spacing) * 56)}.sm\:w-36{width:calc(var(--spacing) * 36)}.sm\:w-fit{width:fit-content}.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:flex-row{flex-direction:row}.sm\:items-end{align-items:flex-end}.sm\:gap-6{gap:calc(var(--spacing) * 6)}.sm\:px-6{padding-inline:calc(var(--spacing) * 6)}.sm\:text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.sm\:text-5xl{font-size:var(--text-5xl);line-height:var(--tw-leading,var(--text-5xl--line-height))}.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\:col-span-1{grid-column:span 1/span 1}.md\:col-span-2{grid-column:span 2/span 2}.md\:h-64{height:calc(var(--spacing) * 64)}.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.md\:text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.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}}.\[\&\>div\]\:h-full>div{height:100%}}@keyframes marquee{0%{transform:translate(0%)}to{transform:translate(-100%)}}@media (prefers-reduced-motion:reduce){.animate-marquee{animation-play-state:paused}}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-gradient-position{syntax:"*";inherits:false}@property --tw-gradient-from{syntax:"";inherits:false;initial-value:#0000}@property --tw-gradient-via{syntax:"";inherits:false;initial-value:#0000}@property --tw-gradient-to{syntax:"";inherits:false;initial-value:#0000}@property --tw-gradient-stops{syntax:"*";inherits:false}@property --tw-gradient-via-stops{syntax:"*";inherits:false}@property --tw-gradient-from-position{syntax:"";inherits:false;initial-value:0%}@property --tw-gradient-via-position{syntax:"";inherits:false;initial-value:50%}@property --tw-gradient-to-position{syntax:"";inherits:false;initial-value:100%}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-duration{syntax:"*";inherits:false} \ No newline at end of file From f6337104cf8fe119f2bae7740a10b48cda11e114 Mon Sep 17 00:00:00 2001 From: jmanczak Date: Sun, 5 Apr 2026 15:51:22 +0200 Subject: [PATCH 2/7] oops, forgot about page titles --- src/web/pages/logs.rs | 2 +- src/web/pages/quotes.rs | 2 +- src/web/pages/quotes/add.rs | 2 +- src/web/pages/users/create.rs | 2 +- src/web/pages/usersettings.rs | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/src/web/pages/logs.rs b/src/web/pages/logs.rs index e9e51d6..97ddb2e 100644 --- a/src/web/pages/logs.rs +++ b/src/web/pages/logs.rs @@ -20,7 +20,7 @@ pub async fn page(req: Request) -> Result { let logs = LogEntry::get_all(&tx)?; Ok(base( - "Persons | Mnemosyne", + "Logs | Mnemosyne", html!( (nav(Some(&u), req.uri().path())) diff --git a/src/web/pages/quotes.rs b/src/web/pages/quotes.rs index 6620ac9..b4b5da8 100644 --- a/src/web/pages/quotes.rs +++ b/src/web/pages/quotes.rs @@ -16,7 +16,7 @@ pub async fn page(req: Request) -> Result { let u = User::authenticate(req.headers())?; Ok(base( - "Persons | Mnemosyne", + "Quotes | Mnemosyne", html!( (nav(u.as_ref(), req.uri().path())) diff --git a/src/web/pages/quotes/add.rs b/src/web/pages/quotes/add.rs index bfba9fc..04ad6d4 100644 --- a/src/web/pages/quotes/add.rs +++ b/src/web/pages/quotes/add.rs @@ -18,7 +18,7 @@ pub async fn page(req: Request) -> Result { let names = Name::get_all(&conn)?; Ok(base( - "Persons | Mnemosyne", + "Add Quote | Mnemosyne", html!( (nav(u.as_ref(), req.uri().path())) diff --git a/src/web/pages/users/create.rs b/src/web/pages/users/create.rs index 595103c..68d530b 100644 --- a/src/web/pages/users/create.rs +++ b/src/web/pages/users/create.rs @@ -25,7 +25,7 @@ pub async fn page(req: Request) -> Result { let conn = database::conn()?; Ok(base( - "Users | Mnemosyne", + "Create User | Mnemosyne", html!( (nav(u.as_ref(), req.uri().path())) diff --git a/src/web/pages/usersettings.rs b/src/web/pages/usersettings.rs index 31a6668..ba5e584 100644 --- a/src/web/pages/usersettings.rs +++ b/src/web/pages/usersettings.rs @@ -23,7 +23,7 @@ pub async fn page(req: Request) -> Result { let u = User::authenticate(req.headers())?; Ok(base( - "Persons | Mnemosyne", + "User Settings | Mnemosyne", html!( (nav(u.as_ref(), req.uri().path())) From b0d86efae63b7e76d8dbc5ec93637b04eed4e9f4 Mon Sep 17 00:00:00 2001 From: jmanczak Date: Sun, 5 Apr 2026 18:12:51 +0200 Subject: [PATCH 3/7] why was readme not given a file extension --- readme => readme.txt | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename readme => readme.txt (100%) diff --git a/readme b/readme.txt similarity index 100% rename from readme rename to readme.txt From 26be03ba31a8046996fd1a6b65598c9c363a3143 Mon Sep 17 00:00:00 2001 From: jmanczak Date: Sun, 5 Apr 2026 22:11:43 +0200 Subject: [PATCH 4/7] working quote submission (with limits for now) - must have two lines - can only submit timestamp in system timezone --- Cargo.lock | 39 +++++++++++++++++++ Cargo.toml | 1 + src/web/pages/mod.rs | 2 + src/web/pages/quotes/add.rs | 76 +++++++++++++++++++++++++++++-------- 4 files changed, 102 insertions(+), 16 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index a396c2a..281d8db 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -185,6 +185,31 @@ dependencies = [ "tracing", ] +[[package]] +name = "axum-extra" +version = "0.12.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fef252edff26ddba56bbcdf2ee3307b8129acb86f5749b68990c168a6fcc9c76" +dependencies = [ + "axum", + "axum-core", + "bytes", + "form_urlencoded", + "futures-core", + "futures-util", + "http", + "http-body", + "http-body-util", + "mime", + "pin-project-lite", + "serde_core", + "serde_html_form", + "serde_path_to_error", + "tower-layer", + "tower-service", + "tracing", +] + [[package]] name = "base32" version = "0.5.1" @@ -934,6 +959,7 @@ version = "0.1.0" dependencies = [ "argon2", "axum", + "axum-extra", "base32", "base64", "chrono", @@ -1289,6 +1315,19 @@ dependencies = [ "syn", ] +[[package]] +name = "serde_html_form" +version = "0.2.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2f2d7ff8a2140333718bb329f5c40fc5f0865b84c426183ce14c97d2ab8154f" +dependencies = [ + "form_urlencoded", + "indexmap", + "itoa", + "ryu", + "serde_core", +] + [[package]] name = "serde_json" version = "1.0.149" diff --git a/Cargo.toml b/Cargo.toml index f721615..27a0eb3 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,6 +6,7 @@ edition = "2024" [dependencies] argon2 = "0.5.3" axum = "0.8.8" +axum-extra = { version = "0.12.5", features = ["form"] } base32 = "0.5.1" base64 = "0.22.1" chrono = { version = "0.4.43", features = ["serde"] } diff --git a/src/web/pages/mod.rs b/src/web/pages/mod.rs index d947723..1586a27 100644 --- a/src/web/pages/mod.rs +++ b/src/web/pages/mod.rs @@ -31,8 +31,10 @@ pub fn pages() -> Router { .route("/persons", get(persons::page)) .route("/persons/create", post(persons::create)) .route("/logs", get(logs::page)) + // .route("/quotes", get(quotes::page)) .route("/quotes/add", get(quotes::add::page)) + .route("/quotes/add-form", post(quotes::add::form)) } pub fn base(title: &str, inner: Markup) -> Markup { diff --git a/src/web/pages/quotes/add.rs b/src/web/pages/quotes/add.rs index 04ad6d4..0a34cd9 100644 --- a/src/web/pages/quotes/add.rs +++ b/src/web/pages/quotes/add.rs @@ -1,14 +1,26 @@ use axum::{ + Json, extract::Request, + http::HeaderMap, response::{IntoResponse, Response}, }; +use axum_extra::extract::Form; +use chrono::{DateTime, Utc}; +use chrono_tz::Europe::Warsaw; use maud::{PreEscaped, html}; +use serde::Deserialize; +use uuid::Uuid; use crate::{ database, error::CompositeError, + logs::{LogAction, LogEntry}, persons::Name, - users::{User, auth::UserAuthenticate}, + quotes::Quote, + users::{ + User, + auth::{UserAuthRequired, UserAuthenticate}, + }, web::{components::nav::nav, icons, pages::base}, }; @@ -29,7 +41,8 @@ pub async fn page(req: Request) -> Result { span class="text-2xl font-semibold font-lora" {"Quote Maker"} } } - div class="border border-neutral-200/25 bg-neutral-200/5 rounded-md p-4 flex flex-col" { + form method="post" action="/quotes/add-form" + class="border border-neutral-200/25 bg-neutral-200/5 rounded-md p-4 flex flex-col" { @for i in 1..=2 { div class="flex justify-between gap-4" { div class="flex flex-col flex-1" { @@ -38,10 +51,6 @@ pub async fn page(req: Request) -> Result { input type="text" name="quoteline" placeholder="They said..." autocomplete="off" class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; } - // label for=(format!("line-{i}")) class="mb-1" {(format!("Quote Line #{i}"))} - // input type="text" id=(format!("line-{i}")) name=(format!("line-{i}")) - // placeholder=(format!("They said...")) autocomplete="off" - // class="px-2 py-1 mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; } div class="flex flex-col" { label { @@ -50,18 +59,10 @@ pub async fn page(req: Request) -> Result { class="px-2 py-1.5 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"{ option {"--"} @for name in &names { - option {(name.name)} + option value=(name.id.to_string()) {(name.name)} } } } - // label for=(format!("who-{i}")) class="mb-1" {(format!("Quote Author #{i}"))} - // select id=(format!("line-{i}")) name=(format!("line-{i}")) autocomplete="off" - // class="px-2 py-1.5 mb-2 bg-neutral-950/50 rounded border border-neutral-200/25" { - // option {"--"} - // @for name in &names { - // option {(name.name)} - // } - // } } } } @@ -77,7 +78,9 @@ pub async fn page(req: Request) -> Result { div class="flex flex-col flex-1" { label class="w-full" { p class="mb-1" {"Time of utterance"} - input type="text" name="time" autocomplete="off" placeholder="2026-04-05T01:14:05+02:00" + input type="hidden" name="time" id="time_hidden"; + input type="datetime-local" autocomplete="off" + onchange="document.getElementById('time_hidden').value = new Date(this.value).toISOString()" class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; } } @@ -100,3 +103,44 @@ pub async fn page(req: Request) -> Result { ) .into_response()) } + +#[derive(Deserialize, Debug)] +pub struct IncomingQuote { + #[serde(rename = "quoteline")] + lines: Vec, + #[serde(rename = "quoteauthor")] + authors: Vec, + location: String, + time: String, + context: String, +} +pub async fn form( + headers: HeaderMap, + Form(form): Form, +) -> Result { + let u = User::authenticate(&headers)?.required()?; + let mut conn = database::conn()?; + let tx = conn.transaction()?; + + let authors = form + .authors + .into_iter() + .map(|nid| Name::get_by_id(&tx, nid).unwrap()); + let lines = form.lines.into_iter().zip(authors).collect(); + let timestamp = DateTime::parse_from_rfc3339(&form.time) + .unwrap_or_else(|_| Utc::now().with_timezone(&Warsaw).fixed_offset()); + let context = match form.context.trim() { + "" => None, + s => Some(s.to_string()), + }; + let location = match form.location.trim() { + "" => None, + s => Some(s.to_string()), + }; + + let q = Quote::create(&tx, lines, timestamp, context, location, u.id, false)?; + LogEntry::new(&tx, u, LogAction::CreateQuote { id: q.id })?; + tx.commit()?; + + Ok(Json(q).into_response()) +} From f09af791e261a530727dfc1605cabb196d28ca95 Mon Sep 17 00:00:00 2001 From: jmanczak Date: Sun, 5 Apr 2026 23:14:49 +0200 Subject: [PATCH 5/7] fetch newest quote for dashboard, helpers --- src/quotes/mod.rs | 23 ++++++++++++- src/web/pages/dashboard.rs | 69 ++++++++++++++------------------------ 2 files changed, 48 insertions(+), 44 deletions(-) diff --git a/src/quotes/mod.rs b/src/quotes/mod.rs index 604f60c..865219b 100644 --- a/src/quotes/mod.rs +++ b/src/quotes/mod.rs @@ -1,5 +1,5 @@ use axum::{http::StatusCode, response::IntoResponse}; -use chrono::{DateTime, FixedOffset}; +use chrono::{DateTime, FixedOffset, Utc}; use rusqlite::{Connection, OptionalExtension}; use serde::Serialize; use uuid::Uuid; @@ -34,6 +34,15 @@ pub enum QuoteError { DatabaseError(#[from] DatabaseError), } +impl Quote { + pub fn get_creation_timestamp(&self) -> DateTime { + // unwrap here because all IDs use UUIDv7 + let (s, n) = self.id.get_timestamp().unwrap().to_unix(); + // unwrap here because timestamps held by UUIDs are valid by spec + DateTime::from_timestamp(s as i64, n).unwrap() + } +} + impl Quote { pub fn total_count(conn: &Connection) -> Result { Ok(conn.query_row("SELECT COUNT(*) FROM quotes", (), |r| r.get(0))?) @@ -92,6 +101,18 @@ impl Quote { public, }) } + pub fn get_newest(conn: &Connection) -> Result, QuoteError> { + let id: Option = conn + .query_row("SELECT id FROM quotes ORDER BY id DESC LIMIT 1", (), |r| { + r.get(0) + }) + .optional()?; + + match id { + Some(id) => Ok(Some(Self::get_by_id(conn, id)?)), + None => Ok(None), + } + } pub fn create( conn: &Connection, lines: Vec<(String, Name)>, diff --git a/src/web/pages/dashboard.rs b/src/web/pages/dashboard.rs index 095b0fa..7267021 100644 --- a/src/web/pages/dashboard.rs +++ b/src/web/pages/dashboard.rs @@ -24,8 +24,9 @@ const LINKS: &[(&str, &str, &str)] = &[ pub async fn page(req: Request) -> Result { let u = User::authenticate(req.headers()).ok().flatten(); - let mut conn = database::conn()?; - let tx = conn.transaction()?; + let conn = database::conn()?; + + let newest_quote = Quote::get_newest(&conn)?; Ok(base( "Dashboard | Mnemosyne", @@ -35,8 +36,15 @@ pub async fn page(req: Request) -> Result { div class="mx-auto max-w-4xl mt-4 grid grid-cols-1 sm:grid-cols-2 gap-4" { div class="flex flex-col" { p {"Newest Quote"} - p class="text-neutral-500 font-light mb-4" {"This just in! This quote was added 15s ago."} - div class="flex-1 [&>div]:h-full" {(quote(&sample_quote_1()))} + @if let Some(q) = newest_quote { + p class="text-neutral-500 font-light mb-4" { + "This just in! This quote was added " + (format_time_ago(q.get_creation_timestamp())) " ago." + } + div class="flex-1 [&>div]:h-full" {(quote(&q))} + } @else { + p class="text-neutral-500 font-light mb-4" {"No quotes yet."} + } } div class="flex flex-col" { p {"Quote of the Day"} @@ -60,25 +68,25 @@ pub async fn page(req: Request) -> Result { } div class="mx-auto max-w-4xl mt-4 flex flex-row gap-2" { (chip(html!({ - @match Quote::total_count(&tx) { + @match Quote::total_count(&conn) { Ok(count) => {(count) " QUOTES TOTAL"}, Err(_) => span class="text-red-400" {"QUOTE COUNT ERR"}, } }))) (chip(html!({ - @match Person::total_count(&tx) { + @match Person::total_count(&conn) { Ok(count) => {(count) " PERSONS TOTAL"}, Err(_) => span class="text-red-400" {"PERSON COUNT ERR"}, } }))) (chip(html!({ - @match Tag::total_count(&tx) { + @match Tag::total_count(&conn) { Ok(count) => {(count) " TAGS TOTAL"}, Err(_) => span class="text-red-400" {"TAG COUNT ERR"} } }))) (chip(html!({ - @match User::total_count(&tx) { + @match User::total_count(&conn) { Ok(count) => {(count) " USERS TOTAL"}, Err(_) => span class="text-red-400" {"USER COUNT ERR"} } @@ -90,41 +98,6 @@ pub async fn page(req: Request) -> Result { )) } -fn sample_quote_1() -> Quote { - Quote { - id: Uuid::now_v7(), - public: true, - location: Some(String::from("Poznań")), - context: Some(String::from("Wykład z językoznawstwa")), - created_by: Uuid::max(), - timestamp: DateTime::from(Utc::now()), - lines: vec![ - QuoteLine { - id: Uuid::now_v7(), - content: String::from("Nie wiem, czy są tutaj osoby fanowskie zipline-ów?"), - attribution: Name { - id: Uuid::nil(), - created_by: Uuid::max(), - person_id: Uuid::now_v7(), - is_primary: true, - name: String::from("dr. Barbara Konat"), - }, - }, - QuoteLine { - id: Uuid::now_v7(), - content: String::from("Taka uprząż co robi pziuuum!"), - attribution: Name { - id: Uuid::nil(), - created_by: Uuid::max(), - person_id: Uuid::now_v7(), - is_primary: true, - name: String::from("dr. Barbara Konat"), - }, - }, - ], - } -} - fn sample_quote_2() -> Quote { Quote { id: Uuid::now_v7(), @@ -159,3 +132,13 @@ fn sample_quote_2() -> Quote { ], } } + +fn format_time_ago(dt: DateTime) -> String { + let secs = Utc::now().signed_duration_since(dt).num_seconds(); + match secs { + ..60 => format!("{}s", secs), + 60..3600 => format!("{}m", secs / 60), + 3600..86400 => format!("{}h", secs / 3600), + _ => format!("{}d", secs / 86400), + } +} From 777027d4718ab360dedd9c57761a5165936bc6c0 Mon Sep 17 00:00:00 2001 From: jmanczak Date: Sun, 5 Apr 2026 23:37:09 +0200 Subject: [PATCH 6/7] use YMD date display format so i don't have to fight americans --- src/web/components/quote.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/web/components/quote.rs b/src/web/components/quote.rs index ef41960..6f159a3 100644 --- a/src/web/components/quote.rs +++ b/src/web/components/quote.rs @@ -25,7 +25,7 @@ pub fn quote(quote: &Quote) -> Markup { } } div class="flex flex-row text-neutral-400 mt-auto pt-4 items-center font-light text-xs" { - p {(quote.timestamp.format("%d/%m/%Y %H:%M"))} + p {(quote.timestamp.format("%Y-%m-%d %H:%M"))} @if let Some(loc) = "e.location { span class="ml-3 scale-[.5]"{(PreEscaped(icons::MAP_PIN))} p { (loc) } } From 2dd4c8ac47313ac57d208ccc28f8bf29bdca7f04 Mon Sep 17 00:00:00 2001 From: jmanczak Date: Mon, 6 Apr 2026 00:24:41 +0200 Subject: [PATCH 7/7] don't normalize quote timestamp to UTC --- src/web/pages/quotes/add.rs | 17 ++++++++++++----- 1 file changed, 12 insertions(+), 5 deletions(-) diff --git a/src/web/pages/quotes/add.rs b/src/web/pages/quotes/add.rs index 0a34cd9..1d3cf89 100644 --- a/src/web/pages/quotes/add.rs +++ b/src/web/pages/quotes/add.rs @@ -5,7 +5,7 @@ use axum::{ response::{IntoResponse, Response}, }; use axum_extra::extract::Form; -use chrono::{DateTime, Utc}; +use chrono::{TimeZone, Utc}; use chrono_tz::Europe::Warsaw; use maud::{PreEscaped, html}; use serde::Deserialize; @@ -78,9 +78,9 @@ pub async fn page(req: Request) -> Result { div class="flex flex-col flex-1" { label class="w-full" { p class="mb-1" {"Time of utterance"} - input type="hidden" name="time" id="time_hidden"; - input type="datetime-local" autocomplete="off" - onchange="document.getElementById('time_hidden').value = new Date(this.value).toISOString()" + input type="hidden" name="tz_offset" id="tz_offset" value="0"; + script { (PreEscaped("document.getElementById('tz_offset').value = new Date().getTimezoneOffset();")) } + input type="datetime-local" name="time" autocomplete="off" class="px-2 py-1 w-full mb-2 bg-neutral-950/50 rounded border border-neutral-200/25"; } } @@ -112,6 +112,7 @@ pub struct IncomingQuote { authors: Vec, location: String, time: String, + tz_offset: Option, context: String, } pub async fn form( @@ -127,7 +128,13 @@ pub async fn form( .into_iter() .map(|nid| Name::get_by_id(&tx, nid).unwrap()); let lines = form.lines.into_iter().zip(authors).collect(); - let timestamp = DateTime::parse_from_rfc3339(&form.time) + let offset = form + .tz_offset + .and_then(|mins| chrono::FixedOffset::west_opt(mins * 60)) + .unwrap_or_else(|| chrono::FixedOffset::west_opt(0).unwrap()); + + let timestamp = chrono::NaiveDateTime::parse_from_str(&form.time, "%Y-%m-%dT%H:%M") + .map(|ndt| offset.from_local_datetime(&ndt).unwrap()) .unwrap_or_else(|_| Utc::now().with_timezone(&Warsaw).fixed_offset()); let context = match form.context.trim() { "" => None,