commit fcd4eb39f4cfa975b9b81245e85ee16852f7ac10 Author: Gitea Bootstrap Date: Wed May 6 09:21:40 2026 +0000 chore: initial template commit diff --git a/.agentsignore b/.agentsignore new file mode 100644 index 0000000..dc46c4a --- /dev/null +++ b/.agentsignore @@ -0,0 +1,5 @@ +package-lock.json +Cargo.lock +/node_modules +/target +/dist \ No newline at end of file diff --git a/.gitea/template b/.gitea/template new file mode 100644 index 0000000..9f20350 --- /dev/null +++ b/.gitea/template @@ -0,0 +1,4 @@ +# Replace in specific config files +agent.md +Trunk.toml +index.html \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..eab879a --- /dev/null +++ b/.gitignore @@ -0,0 +1,277 @@ +/target +### JetBrains template +# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio, WebStorm and Rider +# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839 + +# User-specific stuff +.idea/**/workspace.xml +.idea/**/tasks.xml +.idea/**/usage.statistics.xml +.idea/**/dictionaries +.idea/**/shelf + +# AWS User-specific +.idea/**/aws.xml + +# Generated files +.idea/**/contentModel.xml + +# Sensitive or high-churn files +.idea/**/dataSources/ +.idea/**/dataSources.ids +.idea/**/dataSources.local.xml +.idea/**/sqlDataSources.xml +.idea/**/dynamic.xml +.idea/**/uiDesigner.xml +.idea/**/dbnavigator.xml + +# Gradle +.idea/**/gradle.xml +.idea/**/libraries + +# Gradle and Maven with auto-import +# When using Gradle or Maven with auto-import, you should exclude module files, +# since they will be recreated, and may cause churn. Uncomment if using +# auto-import. +# .idea/artifacts +# .idea/compiler.xml +# .idea/jarRepositories.xml +# .idea/modules.xml +# .idea/*.iml +# .idea/modules +# *.iml +# *.ipr + +# CMake +cmake-build-*/ + +# Mongo Explorer plugin +.idea/**/mongoSettings.xml + +# File-based project format +*.iws + +# IntelliJ +out/ + +# mpeltonen/sbt-idea plugin +.idea_modules/ + +# JIRA plugin +atlassian-ide-plugin.xml + +# Cursive Clojure plugin +.idea/replstate.xml + +# SonarLint plugin +.idea/sonarlint/ + +# Crashlytics plugin (for Android Studio and IntelliJ) +com_crashlytics_export_strings.xml +crashlytics.properties +crashlytics-build.properties +fabric.properties + +# Editor-based Rest Client +.idea/httpRequests + +# Android studio 3.1+ serialized cache file +.idea/caches/build_file_checksums.ser + +### Rust template +# Generated by Cargo +# will have compiled files and executables +debug/ +target/ + +# Remove Cargo.lock from gitignore if creating an executable, leave it for libraries +# More information here https://doc.rust-lang.org/cargo/guide/cargo-toml-vs-cargo-lock.html +Cargo.lock + +# These are backup files generated by rustfmt +**/*.rs.bk + +# MSVC Windows builds of rustc generate these, which store debugging information +*.pdb + +# RustRover +# JetBrains specific template is maintained in a separate JetBrains.gitignore that can +# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore +# and can be added to the global gitignore or merged into this file. For a more nuclear +# option (not recommended) you can uncomment the following to ignore the entire idea folder. +#.idea/ +### Linux template +*~ + +# temporary files which can be created if a process still has a handle open of a deleted file +.fuse_hidden* + +# KDE directory preferences +.directory + +# Linux trash folder which might appear on any partition or disk +.Trash-* + +# .nfs files are created when an open file is removed but is still being accessed +.nfs* + +### Node template +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Diagnostic reports (https://nodejs.org/api/report.html) +report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json + +# Runtime data +pids +*.pid +*.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage +*.lcov + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# Bower dependency directory (https://bower.io/) +bower_components + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) +web_modules/ + +# TypeScript cache +*.tsbuildinfo + +# Optional npm cache directory +.npm + +# Optional eslint cache +.eslintcache + +# Optional stylelint cache +.stylelintcache + +# Microbundle cache +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history +.node_repl_history + +# Output of 'npm pack' +*.tgz + +# Yarn Integrity file +.yarn-integrity + +# dotenv environment variable files +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) +.cache +.parcel-cache + +# Next.js build output +.next +out + +# Nuxt.js build / generate output +.nuxt +dist + +# Gatsby files +.cache/ +# Comment in the public line in if your project uses Gatsby and not Next.js +# https://nextjs.org/blog/next-9-1#public-directory-support +# public + +# vuepress build output +.vuepress/dist + +# vuepress v2.x temp and cache directory +.temp +.cache + +# Docusaurus cache and generated files +.docusaurus + +# Serverless directories +.serverless/ + +# FuseBox cache +.fusebox/ + +# DynamoDB Local files +.dynamodb/ + +# TernJS port file +.tern-port + +# Stores VSCode versions used for testing VSCode extensions +.vscode-test + +# yarn v2 +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +### macOS template +# General +.DS_Store +.AppleDouble +.LSOverride + +# Icon must end with two \r +Icon + +# Thumbnails +._* + +# Files that might appear in the root of a volume +.DocumentRevisions-V100 +.fseventsd +.Spotlight-V100 +.TemporaryItems +.Trashes +.VolumeIcon.icns +.com.apple.timemachine.donotpresent + +# Directories potentially created on remote AFP share +.AppleDB +.AppleDesktop +Network Trash Folder +Temporary Items +.apdisk + diff --git a/Cargo.toml b/Cargo.toml new file mode 100644 index 0000000..cf1f165 --- /dev/null +++ b/Cargo.toml @@ -0,0 +1,29 @@ +[package] +name = "{{.Repository.Name}}" +version = "0.1.0" +edition = "2021" + +[features] +desktop = [] # enables tauri and keycloak + +[dependencies] +leptos = { version = "0.8.16", features = ["csr"] } +leptos_router = "0.8.12" +#leptos-keycloak-auth = "0.13.0" +console_error_panic_hook = "0.1.7" +wasm-bindgen = "0.2.108" +uuid = { version = "1", features = ["serde"] } +serde = { version = "1.0.228", features = ["derive"] } +regress = "0.10.5" +chrono = { version = "0.4.43", features = ["serde"] } +anyhow = "1.0.101" +serde_json = "1.0.149" +reqwest = { version = "0.13.2", features = ["json"] } +thiserror = "2.0.18" +once_cell = "1.21.3" +tokio = { version = "1.49.0", features = ["sync"] } +gloo-timers = { version = "0.3", features = ["futures"] } +base64 = "0.22.1" +web-sys = "0.3.85" +wasm-bindgen-futures = "0.4.58" +js-sys = "0.3.85" diff --git a/Trunk.toml b/Trunk.toml new file mode 100644 index 0000000..9055069 --- /dev/null +++ b/Trunk.toml @@ -0,0 +1,6 @@ +[build] +filehash = false +public_url = "/crafts/releases/mridge-mini-app/" + +[serve] +addresses = ["0.0.0.0"] diff --git a/agent.md b/agent.md new file mode 100644 index 0000000..87a016c --- /dev/null +++ b/agent.md @@ -0,0 +1,104 @@ +# Mountainridge Leptos + Matterhorn Template – AI Coding Agent Instructions + +**Purpose** +This is the **single source of truth**. +Every AI coding agent **must** follow these rules exactly. +Any deviation breaks the template contract, Matterhorn codegen pipeline, Keycloak integration, Tauri desktop build, or +embedding mechanism. + +## Core Overview + +- Rust + **Leptos** (latest stable) **CSR-only** reactive web application +- Data layer: **Matterhorn entities** (JSON-schema driven) +- Styling: **Standalone TailwindCSS** + **DaisyUI** components (no `tailwind.config.js`) +- Corporate theme: **"mountainridge"** (locked in `styles.css`) +- Authentication: **Keycloak OAuth2** +- Desktop variant: enabled with feature flag **`desktop`** (uses Tauri) + +## Agent Tool Usage (Mandatory) + +All agents have the following **injected tools** and **must** use them exclusively: + +- `mridge_craft_add_matterhorn_entity` – add new Matterhorn data entities +- `mridge_craft_build` – full build / serve / watch / desktop package (Cargo + Trunk + Tauri) +- Git commit & push +- Create issues for the craft + +**Never** execute raw shell commands (`cargo`, `trunk`, `git`, `tauri`, etc.). +All GitHub operations are fully wrapped — repo details are irrelevant. + +## Project Structure & Immutable Rules + +| Path | Purpose | Modification Rules | +|------------------------|------------------------------------------------------------------------|---------------------------------------------------------------------| +| `./assets/` | Extra CSS, fonts, images, static assets | Add only | +| `./schemas/` | JSON schemas for Matterhorn entities | **Never edit manually** – use add-entity tool | +| `./src/codegen/` | **Auto-generated only** (structs, API client, helpers) | **Never touch** | +| `./src/codegen/api.rs` | Generated type-safe Matterhorn client (list/read/create/update/delete) | Auto-regenerated | +| `./src/components/` | All custom Leptos components | Free (PascalCase.rs) | +| `./src/main.rs` | Router + Keycloak token loader + BaseLayout | **Only edit the router section**. Never remove auth/base wrappers | +| `./src/styles.css` | Global stylesheet + full "mountainridge" DaisyUI theme | Extend only – never overwrite theme block | +| `./index.html` | HTML entry point (Trunk) | **Immutable section** (see below) – only meta/title changes allowed | + +## Immutable / Protected Elements (Never Change) + +**`./index.html`** – these exact lines **must remain untouched** (required for Trunk, favicon, public URL, and config +injection): + +```html + + + + + +``` + +**`./src/styles.css`** – never delete or replace the `"daisyui/theme"` block. + +**`./src/main.rs`** – never remove ``, Keycloak init, or ``. + +**`./src/codegen/`** – generated; never manual edits. + +## Authentication – Keycloak OAuth2 + +`main.rs` contains the protected Keycloak token loader: + +- **Web/iframe mode** (default): token injected via `postMessage` (uses `window.__embeddedAppConfig`) +- **Desktop mode** (`--features desktop`): full Keycloak login flow + Tauri + +**Rule:** Never remove, refactor, or bypass any auth-related wrapper. + +## Critical Rules for Coding Agents + +1. **Data Dependencies** + Always use `mridge_craft_add_matterhorn_entity` for any new or modified Matterhorn model. + +2. **Routing** + Add new routes **only** inside the existing `` block in `main.rs`. + Preserve all auth/protected-route wrappers. + +3. **Styling** + Prefer DaisyUI classes everywhere. + Tailwind is standalone — no config file. + Never override the "mountainridge" theme. + +4. **Data Fetching (CSR-only)** + All operations **must** use the generated `codegen::api::*` client. + No Leptos server functions. + +5. **Building & Testing** + Every compile, dev server, release, or Tauri desktop build → use `mridge_craft_build`. + +## Standard Agent Workflow + +1. Need new data entity? → use 'mrdige_add-entity' skill +2. Create component(s) in `./src/components/` +3. Add route in `main.rs` (inside router, respect auth) +4. Commit/push via git tools or create issue if needed diff --git a/assets/mridge-craft-icon.webp b/assets/mridge-craft-icon.webp new file mode 100644 index 0000000..00ac850 Binary files /dev/null and b/assets/mridge-craft-icon.webp differ diff --git a/index.html b/index.html new file mode 100644 index 0000000..03d7e88 --- /dev/null +++ b/index.html @@ -0,0 +1,21 @@ + + + + + + + MountainRidge.xyz - Mont Fort App + + + + + + + \ No newline at end of file diff --git a/package.json b/package.json new file mode 100644 index 0000000..11060c8 --- /dev/null +++ b/package.json @@ -0,0 +1,8 @@ +{ + "dependencies": { + "daisyui": "^5.5.18" + }, + "scripts": { + "build-craft": "npm install && tailwindcss -i src/styles.css -o assets/styles.css --minify && trunk build --config Trunk.toml" + } +} diff --git a/src/components/hello_mridge.rs b/src/components/hello_mridge.rs new file mode 100644 index 0000000..6456921 --- /dev/null +++ b/src/components/hello_mridge.rs @@ -0,0 +1,29 @@ +use crate::codegen::apis::PresseEintragApi; +use crate::codegen::types::presse_eintrag::{PressEintrag, PressEintragSentiments}; +use crate::matterhorn::types::SortDirection; +use leptos::prelude::*; +use leptos::task::spawn_local; +use leptos_router::components::A; +use uuid::Uuid; + +const ICON_BYTES: &[u8] = include_bytes!("../../assets/mridge-craft-icon.webp"); + +fn icon_data_url() -> String { + use base64::{engine::general_purpose::STANDARD, Engine}; + format!("data:image/webp;base64,{}", STANDARD.encode(ICON_BYTES)) +} + +#[component] +pub fn HelloMRidge() -> impl IntoView { + view! { +
+
+ MountainRidge icon +

"Mont Fort Craft"

+
+
+

Hello MountainRidge!

+
+
+ } +} diff --git a/src/components/mod.rs b/src/components/mod.rs new file mode 100644 index 0000000..b4ff7e0 --- /dev/null +++ b/src/components/mod.rs @@ -0,0 +1,4 @@ +mod hello_mridge; +pub mod views { + pub use super::hello_mridge::*; +} diff --git a/src/main.rs b/src/main.rs new file mode 100644 index 0000000..3f6173b --- /dev/null +++ b/src/main.rs @@ -0,0 +1,64 @@ +mod components; +mod matterhorn; + +use crate::components::views::{HelloMRidge}; +use leptos::prelude::*; +use leptos::reactive::owner::Owner; +use leptos_router::components::{Route, Router, Routes}; +use leptos_router::path; +use wasm_bindgen::prelude::*; +use wasm_bindgen::JsCast; + +fn main() { + console_error_panic_hook::set_once(); + mount_to_body(App); +} + +/* +* App entrypoint; define new routes and add new components here. +* Hint: do not change the Router base and always ever use relative routes +*/ +fn App() -> impl IntoView { + view! { + + "Not found"

}> + +
+
+ } +} + +/** +* DO NOT change this config retrieval method +*/ +fn get_config() -> Option<(String, String)> { + let window = web_sys::window()?; + let config = js_sys::Reflect::get(&window, &"__appConfig".into()).ok()?; + let token = js_sys::Reflect::get(&config, &"token".into()) + .ok()? + .as_string()?; + let user = js_sys::Reflect::get(&config, &"user".into()) + .ok()? + .as_string()?; + Some((token, user)) +} + +/* +* Gets the base path for the web application +*/ +pub fn get_base_path() -> String { + web_sys::window() + .and_then(|w| w.document()) + .and_then(|d| d.query_selector("base").ok().flatten()) + .and_then(|el| el.get_attribute("href")) + .unwrap_or_else(|| "/".to_string()) +} + +/* +* Gets the +*/ +pub fn asset_path(file: &str) -> String { + format!("{}assets/{}", get_base_path(), file.trim_start_matches('/')) +} + + diff --git a/src/matterhorn/api.rs b/src/matterhorn/api.rs new file mode 100644 index 0000000..0f27f45 --- /dev/null +++ b/src/matterhorn/api.rs @@ -0,0 +1,109 @@ +use crate::codegen::types::PressEintrag; +use crate::get_config; +use crate::matterhorn::types; +use crate::matterhorn::types::{FetchInstanceRequest, PostInstancesRequest}; +use leptos::prelude::GetValue; +use reqwest::{Client, StatusCode}; +use serde_json::Value; +use thiserror::Error; +use uuid::Uuid; + +#[derive(Debug, Error)] +pub enum ApiError { + #[error("network error: {0}")] + Network(#[from] reqwest::Error), + #[error("json error: {0}")] + Json(#[from] serde_json::Error), + #[error("http {0}")] + Http(u16, String), + #[error("missing 'records' field")] + MissingRecords, + #[error("records is not an array")] + RecordsNotArray, + #[error("other: {0}")] + Other(String), +} +#[derive(Clone)] +pub struct MatterhornApi { + client: Client, + base_url: String, + dataset_id: Option, + token: String, +} + +impl MatterhornApi { + pub fn new() -> Self { + // todo base_url and dataset should be injected + let (token, _user) = get_config().expect("__embeddedAppConfig must be set"); + Self { + client: Client::new(), + base_url: "http://127.0.0.1:8085/matterhorn/api/v0/ontology/2163f231-2e7c-44cf-bc14-fbde9edea884/data".into(), + dataset_id: None, + token, + } + } + + pub async fn fetch_records( + &self, + matterhorn_entity_id: &u64, + projections: Vec, + filters: Vec, + sorters: std::collections::HashMap, + limit: u32, + ) -> Result, ApiError> { + let url = format!("{:1}/{:2}", self.base_url, matterhorn_entity_id); + let request_body = FetchInstanceRequest { + projections, + filters, + sorters, + limit, + }; + + let resp = self + .client + .post(url) + .bearer_auth(self.token.as_str()) + .json(&request_body) + .header("Accept", "application/json") + .send() + .await + .map_err(ApiError::Network)?; + // ── Step 1: Get generic map-like response ─────────────────────────────── + let raw: Value = resp + .json::() + .await + .map_err(|e| ApiError::Other(format!("Deserialization failed: {}", e)))?; + let graph = raw.get("@graph").ok_or(ApiError::MissingRecords)?; + // Check it's an array and convert to Vec + let array = graph.as_array().ok_or_else(|| ApiError::RecordsNotArray)?; + // If you want owned values (most common when returning Vec) + Ok(array.to_vec()) + } + + pub async fn store_records( + &self, + matterhorn_entity_id: &u64, + records: Vec, + ) -> Result, ApiError> { + let url = format!("{:1}/{:2}", self.base_url, matterhorn_entity_id); + let (token, _user) = get_config().expect("__embeddedAppConfig must be set"); + let request_body = PostInstancesRequest { records }; + let resp = self + .client + .post(url) + .bearer_auth(token) + .json(&request_body) + .header("Accept", "application/json") + .send() + .await + .map_err(ApiError::Network)?; + + web_sys::console::info_1(&format!("{:#?}", "response send").into()); + let status = resp.status(); + if !status.is_success() { + let text = resp.text().await.unwrap_or_default(); + return Err(ApiError::Http(status.as_u16(), text)); + } + Ok(resp.json().await.unwrap_or_default()) + } +} diff --git a/src/matterhorn/mod.rs b/src/matterhorn/mod.rs new file mode 100644 index 0000000..8389f11 --- /dev/null +++ b/src/matterhorn/mod.rs @@ -0,0 +1,2 @@ +pub mod api; +pub mod types; diff --git a/src/matterhorn/types.rs b/src/matterhorn/types.rs new file mode 100644 index 0000000..4b065b4 --- /dev/null +++ b/src/matterhorn/types.rs @@ -0,0 +1,30 @@ +use serde::{Deserialize, Serialize}; +use serde_json::Value; +use std::collections::HashMap; + +#[derive(Serialize, Default)] +pub struct FetchInstanceRequest { + pub projections: Vec, + pub filters: Vec, + pub sorters: HashMap, + pub limit: u32, +} + +#[derive(Serialize, Default)] +pub struct PostInstancesRequest { + pub records: Vec, +} +#[derive(Serialize, Deserialize, Clone, Debug)] +pub enum SortDirection { + #[serde(rename = "ASC")] + Asc, + #[serde(rename = "DESC")] + Desc, +} + +#[derive(Serialize, Deserialize, Clone, Debug)] +pub struct DataGraphFilter { + pub field: String, + pub operator: String, // e.g. "EQ", "CONTAINS" + pub value: Value, +} diff --git a/src/styles.css b/src/styles.css new file mode 100644 index 0000000..3d8a07a --- /dev/null +++ b/src/styles.css @@ -0,0 +1,254 @@ +@import "tailwindcss"; + + +@plugin "daisyui" { + themes: mountainridge --default; + darkTheme: mountainridge; + base: true; + styled: true; + utils: true; + logs: false; +} + +@plugin "daisyui/theme" { + name: "mountainridge"; + default: true; + prefersdark: true; + color-scheme: "dark"; + font-family: "Geist", system-ui, sans-serif; + /* Backgrounds — charcoal scale */ + --color-base-100: oklch(16% 0.02 214); + --color-base-200: oklch(13% 0.02 214); + --color-base-300: oklch(10% 0.02 214); + --color-base-content: oklch(92% 0.005 214); + + /* Primary — sea-grass green rgb(88, 191, 177) */ + --color-primary: oklch(72% 0.097 183); + --color-primary-content: oklch(13% 0.02 214); + + /* Secondary — glacier light blue rgb(94, 150, 159) */ + --color-secondary: oklch(60% 0.055 205); + --color-secondary-content: oklch(13% 0.02 214); + + /* Accent — flamingo pink rgba(253, 117, 127) */ + --color-accent: oklch(72% 0.155 14); /* flamingo pink */ + --color-accent-content: oklch(13% 0.02 214); + + + /* Neutral — charcoal slate */ + --color-neutral: oklch(28% 0.02 214); + --color-neutral-content: oklch(80% 0.01 214); + + /* Semantic */ + --color-info: oklch(60% 0.055 205); + --color-info-content: oklch(13% 0.02 214); + --color-success: oklch(72% 0.097 183); + --color-success-content: oklch(13% 0.02 214); + --color-warning: oklch(68% 0.16 42); /* alpenglow orange */ + --color-warning-content: oklch(13% 0.02 214); + --color-error: oklch(72% 0.155 14); /* flamingo pink */ + --color-error-content: oklch(13% 0.02 214); + + /* Shape */ + --radius-selector: 1.5rem; + --radius-field: 0.48rem; + --radius-box: 0.48rem; + --size-selector: 0.25rem; + --size-field: 0.25rem; + --border: 1px; + --depth: 1; + --noise: 0; +} + +:root { + all: inherit; + color: var(--color-base-content); /* DaisyUI's base text color */ + font-family: 'Geist', Sans, serif; + + /* creating the perfect shadow according to twitter/X + - https://codepen.io/jh3y/pen/yLWgjpd + - https://x.com/jh3yy/status/1796208073830232574 + */ + --tint: 214; + --alpha: 4; + --base: hsl(var(--tint, 214) 80% 27% / calc(var(--alpha, 4) * 1%)); + --shade: hsl(from var(--base) calc(h + 8) 25 calc(l - 5)); + + + --perfect-shadow: 0 0 0 1px var(--base), + 0 1px 1px -0.5px var(--shade), + 0 3px 3px -1.5px var(--shade), + 0 6px 6px -3px var(--shade), + 0 12px 12px -6px var(--base), + 0 24px 24px -12px var(--base); + + + --shade-active: hsla(173, 41%, 53%, 0.66); + --base-active: hsl(var(--tint, 214) 80% 27% / calc(var(--alpha, 4) * 1%)); + + --perfect-shadow-active: 0 0 0 1px var(--base-active), + 0 1px 1px -0.5px var(--shade-active), + 0 3px 3px -1.5px var(--shade-active), + 0 6px 6px -3px var(--shade-active), + 0 12px 12px -6px var(--base-active), + 0 24px 24px -12px var(--base-active); + + --perfect-shadow-thin-active: 0 0 0 1px var(--base-active), + 0 1px 2px -0.5px var(--shade-active), + 0 2px 4px -1px var(--shade-active), + 0 4px 8px -2px var(--shade-active); + + --shade-active-warn: rgba(253, 117, 127, 0.66); + --perfect-shadow-warn: 0 0 0 1px var(--shade-active-warn), + 0 1px 1px -0.5px var(--shade-active-warn), + 0 3px 3px -1.5px var(--shade-active-warn), + 0 6px 6px -3px var(--shade-active-warn), + 0 12px 12px -6px var(--base-active), + 0 24px 24px -12px var(--base-active); + + + --base-thin: hsl(var(--tint, 214) 80% 27% / calc(var(--alpha, 4) * 1%)); + --shade-thin: hsl(from var(--base-thin) calc(h + 8) 25 calc(l - 5)); + + --perfect-shadow-thin: 0 0 0 1px var(--base-thin), + 0 1px 2px -0.5px var(--shade-thin), + 0 3px 6px -1px var(--shade-thin); + + + --perfect-shadow-light: 0px 6px 20px -4px rgba(0, 0, 0, 0.40), 0px 0px 30px -8px rgba(123, 94, 248, 0.10); + + + /* === DARK BACKGROUND SHADOWS === */ + + /* Tinted with glacier blue for atmosphere */ + --tint-dark: 205; + --alpha-dark: 30; + + --base-dark: hsl(var(--tint-dark) 60% 60% / calc(var(--alpha-dark) * 1%)); + --shade-dark: hsl(from var(--base-dark) calc(h + 8) 70 calc(l + 10)); + + /* Full shadow — for cards, modals, elevated surfaces */ + --perfect-shadow-dark: 0 0 0 1px hsl(205 40% 50% / 15%), + 0 1px 1px -0.5px var(--base-dark), + 0 3px 3px -1.5px var(--base-dark), + 0 6px 6px -3px var(--shade-dark), + 0 12px 12px -6px var(--shade-dark), + 0 24px 24px -12px var(--base-dark); + + /* Thin shadow — for buttons, inputs, small elements */ + --perfect-shadow-thin-dark: 0 0 0 1px hsl(205 40% 50% / 12%), + 0 1px 2px -0.5px var(--base-dark), + 0 3px 6px -1px var(--shade-dark); + + /* Active/focus state — sea-grass green glow */ + --shade-active-dark: hsla(183, 35%, 55%, 0.45); /* sea-grass green */ + + --perfect-shadow-thin-active-dark: 0 0 0 1px hsla(183, 35%, 55%, 0.3), + 0 1px 2px -0.5px var(--shade-active-dark), + 0 2px 4px -1px var(--shade-active-dark), + 0 4px 8px -2px var(--shade-active-dark); + + /* Warning/error active state — flamingo pink glow */ + --shade-active-warn-dark: rgba(253, 117, 127, 0.45); + + --perfect-shadow-warn-dark: 0 0 0 1px rgba(253, 117, 127, 0.3), + 0 1px 1px -0.5px var(--shade-active-warn-dark), + 0 3px 3px -1.5px var(--shade-active-warn-dark), + 0 6px 6px -3px var(--shade-active-warn-dark), + 0 12px 12px -6px var(--shade-active-warn-dark); + + + --mountain-white-dark: rgb(245, 242, 242); /* ≈ 80% – popular "dark mode off-white" */ + --mountain-white: rgb(255, 252, 252, 1); + --ghost-white: rgb(248, 248, 255, 1); + + /* Very subtle darkening – still very bright */ + --mountain-white-darker-1: rgb(245, 242, 242); /* ≈ 96% brightness */ + --mountain-white-darker-2: rgb(235, 232, 232); /* ≈ 92% */ + + /* Noticeably darker but still clearly a very light / "off-white" */ + --mountain-white-darker-3: rgb(230, 227, 227); /* ≈ 90% */ + --mountain-white-darker-4: rgb(220, 217, 217); /* ≈ 86% */ + --mountain-white-darker-5: rgb(204, 201, 201); /* ≈ 80% – popular "dark mode off-white" */ + + /* Deeper / moodier off-whites – start feeling more greyish */ + --mountain-white-darker-6: rgb(189, 186, 186); /* ≈ 74% */ + --mountain-white-darker-7: rgb(170, 167, 167); /* ≈ 67% – quite smoky now */ + --mountain-white-darker-8: rgb(150, 147, 147); /* ≈ 59% – very muted/dark off-white */ + + --charcoal-deep: rgb(15, 20, 25); + --charcoal-dark: rgb(25, 30, 35); /* +10 */ + --charcoal-medium: rgb(35, 40, 48); /* +20ish */ + --charcoal-soft: rgb(45, 50, 58); /* noticeably lighter but still moody */ + --charcoal-slate: rgb(55, 60, 68); /* good background / surface color */ + --charcoal-lighter: rgb(70, 75, 85); /* readable dark gray, modern UI feel */ + --deep-night-black: rgb(8, 18, 28); + --abyss-black: rgb(6, 15, 22); + + + --bs-dark: var(--charcoal-medium); /* changes the base dark color */ + --bs-dark-rgb: 35, 40, 48; + --bs-blue: var(--glacier-blue); + + /* custom colors */ + --flamingo-pink: rgba(253, 117, 127, 1); + --sea-grass-green: rgba(88, 191, 177, 1); + + --mountain-darkgray: rgb(12, 31, 49, 1); + --mountain-gray: rgb(102, 133, 139, 1); + --mountain-lightgray: rgb(121, 152, 155, 1); + + --glacier-blue: rgb(28, 80, 101); + --glacier-darkblue: rgb(12, 39, 49); + --glacier-lightblue: rgb(94, 150, 159); + --glacier-blue-rgb: 94, 150, 159; + + /* borders, corners, and shadows */ + --perfect-border-radius: 0.48em; + --backdrop-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.2), 0 1px 3px 0 rgba(0, 0, 0, 0.19); + --backdrop-transition: all 0.3s ease; + + /* global overwrites of bootstrap vars */ + --text-bg-primary-badge-color: var(--glacier-blue); + + /* White to Darker Blue diagonal gradient */ + --gradient-white-dark-blue: linear-gradient( + 142deg, + var(--ghost-white) 0%, + var(--ghost-white) 55%, + var(--glacier-lightblue) 67%, + var(--glacier-lightblue) 75%, + var(--glacier-blue) 100% + ); + + --gradient-dark: linear-gradient(135deg, var(--mountain-white) 0%, var(--mountain-darkgray) 100%); + +} + +* { + scrollbar-width: none; + +} + +h1, h2, h3 { + font-weight: 300; +} + +h4, h5, h6, p, body { + font-weight: 400; +} + +.btn { + box-shadow: var(--perfect-shadow-thin-dark); + transition: transform 0.3s ease-out, box-shadow 0.3s ease-out; +} + +.btn:hover { + transform: scale(1.1) translateY(-2px); + box-shadow: var(--perfect-shadow-thin-active-dark); +} + +.btn:not(.btn-ghost):not(.btn-outline):hover { + transform: scale(1.1) translateY(-2px); + box-shadow: var(--perfect-shadow-thin-active-dark); +} \ No newline at end of file