2026-04-19 10:48:56 -04:00
|
|
|
|
import { useGpuTier } from "@nous-research/ui/hooks/use-gpu-tier";
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Replicates the visual layer stack of `<Overlays dark />` from
|
|
|
|
|
|
* `@nous-research/ui` without pulling in its leva / gsap / three peer deps.
|
|
|
|
|
|
*
|
|
|
|
|
|
* See `design-language/src/ui/components/overlays/index.tsx` for the source of
|
|
|
|
|
|
* truth. Defaults match LENS_0 (the Hermes teal dark preset); the deep canvas
|
|
|
|
|
|
* and the warm vignette both read theme-switchable CSS custom properties so
|
|
|
|
|
|
* `ThemeProvider` can repaint the stack without remounting.
|
|
|
|
|
|
*
|
|
|
|
|
|
* z-1 bg = `var(--background-base)`, mix-blend-mode: difference
|
|
|
|
|
|
* z-2 filler-bg jpeg, inverted, opacity 0.033, difference
|
|
|
|
|
|
* z-99 warm top-left vignette (`var(--warm-glow)`), opacity 0.22, lighten
|
|
|
|
|
|
* z-101 noise grain (SVG, ~55% opacity × `--noise-opacity-mul`,
|
|
|
|
|
|
* color-dodge) — gated on GPU tier
|
|
|
|
|
|
*
|
|
|
|
|
|
* `useGpuTier` returns 0 when WebGL is unavailable, the renderer is a
|
|
|
|
|
|
* software rasterizer (SwiftShader/llvmpipe), or the user has
|
|
|
|
|
|
* `prefers-reduced-motion: reduce` set. We skip the animated noise layer
|
|
|
|
|
|
* in that case so low-power / accessibility-conscious sessions stay crisp,
|
|
|
|
|
|
* mirroring the DS `<Noise />` component's own opt-out.
|
|
|
|
|
|
*/
|
|
|
|
|
|
export function Backdrop() {
|
|
|
|
|
|
const gpuTier = useGpuTier();
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<div
|
|
|
|
|
|
aria-hidden
|
|
|
|
|
|
className="pointer-events-none fixed inset-0 z-[1]"
|
|
|
|
|
|
style={{
|
|
|
|
|
|
backgroundColor: "var(--background-base)",
|
|
|
|
|
|
mixBlendMode: "difference",
|
|
|
|
|
|
}}
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
<div
|
|
|
|
|
|
aria-hidden
|
|
|
|
|
|
className="pointer-events-none fixed inset-0 z-[2]"
|
feat(dashboard): reskin extension points for themes and plugins (#14776)
Themes and plugins can now pull off arbitrary dashboard reskins (cockpit
HUD, retro terminal, etc.) without touching core code.
Themes gain four new fields:
- layoutVariant: standard | cockpit | tiled — shell layout selector
- assets: {bg, hero, logo, crest, sidebar, header, custom: {...}} —
artwork URLs exposed as --theme-asset-* CSS vars
- customCSS: raw CSS injected as a scoped <style> tag on theme apply
(32 KiB cap, cleaned up on theme switch)
- componentStyles: per-component CSS-var overrides (clipPath,
borderImage, background, boxShadow, ...) for card/header/sidebar/
backdrop/tab/progress/badge/footer/page
Plugin manifests gain three new fields:
- tab.override: replaces a built-in route instead of adding a tab
- tab.hidden: register component + slots without adding a nav entry
- slots: declares shell slots the plugin populates
10 named shell slots: backdrop, header-left/right/banner, sidebar,
pre-main, post-main, footer-left/right, overlay. Plugins register via
window.__HERMES_PLUGINS__.registerSlot(name, slot, Component). A
<PluginSlot> React helper is exported on the plugin SDK.
Ships a full demo at plugins/strike-freedom-cockpit/ — theme YAML +
slot-only plugin that reproduces a Gundam cockpit dashboard: MS-STATUS
sidebar with live telemetry, COMPASS crest in header, notched card
corners via componentStyles, scanline overlay via customCSS, gold/cyan
palette, Orbitron typography.
Validation:
- 15 new tests in test_web_server.py covering every extended field
- tests/hermes_cli/: 2615 passed (3 pre-existing unrelated failures)
- tsc -b --noEmit: clean
- vite build: 418 kB bundle, ~2 kB delta for slots/theme extensions
Co-authored-by: Teknium <p@nousresearch.com>
2026-04-23 15:31:01 -07:00
|
|
|
|
style={
|
|
|
|
|
|
{
|
|
|
|
|
|
// Themes can override the filler background by setting
|
|
|
|
|
|
// `assets.bg` — the <img> hides itself when a CSS bg is set
|
|
|
|
|
|
// so the two don't double-darken. CSS var fallbacks keep the
|
|
|
|
|
|
// default behaviour unchanged when no theme customises these.
|
2026-04-28 12:28:08 -04:00
|
|
|
|
mixBlendMode:
|
|
|
|
|
|
"var(--component-backdrop-filler-blend-mode, difference)",
|
feat(dashboard): reskin extension points for themes and plugins (#14776)
Themes and plugins can now pull off arbitrary dashboard reskins (cockpit
HUD, retro terminal, etc.) without touching core code.
Themes gain four new fields:
- layoutVariant: standard | cockpit | tiled — shell layout selector
- assets: {bg, hero, logo, crest, sidebar, header, custom: {...}} —
artwork URLs exposed as --theme-asset-* CSS vars
- customCSS: raw CSS injected as a scoped <style> tag on theme apply
(32 KiB cap, cleaned up on theme switch)
- componentStyles: per-component CSS-var overrides (clipPath,
borderImage, background, boxShadow, ...) for card/header/sidebar/
backdrop/tab/progress/badge/footer/page
Plugin manifests gain three new fields:
- tab.override: replaces a built-in route instead of adding a tab
- tab.hidden: register component + slots without adding a nav entry
- slots: declares shell slots the plugin populates
10 named shell slots: backdrop, header-left/right/banner, sidebar,
pre-main, post-main, footer-left/right, overlay. Plugins register via
window.__HERMES_PLUGINS__.registerSlot(name, slot, Component). A
<PluginSlot> React helper is exported on the plugin SDK.
Ships a full demo at plugins/strike-freedom-cockpit/ — theme YAML +
slot-only plugin that reproduces a Gundam cockpit dashboard: MS-STATUS
sidebar with live telemetry, COMPASS crest in header, notched card
corners via componentStyles, scanline overlay via customCSS, gold/cyan
palette, Orbitron typography.
Validation:
- 15 new tests in test_web_server.py covering every extended field
- tests/hermes_cli/: 2615 passed (3 pre-existing unrelated failures)
- tsc -b --noEmit: clean
- vite build: 418 kB bundle, ~2 kB delta for slots/theme extensions
Co-authored-by: Teknium <p@nousresearch.com>
2026-04-23 15:31:01 -07:00
|
|
|
|
opacity: "var(--component-backdrop-filler-opacity, 0.033)",
|
|
|
|
|
|
backgroundImage: "var(--theme-asset-bg)",
|
|
|
|
|
|
backgroundSize: "var(--component-backdrop-background-size, cover)",
|
2026-04-28 12:28:08 -04:00
|
|
|
|
backgroundPosition:
|
|
|
|
|
|
"var(--component-backdrop-background-position, center)",
|
feat(dashboard): reskin extension points for themes and plugins (#14776)
Themes and plugins can now pull off arbitrary dashboard reskins (cockpit
HUD, retro terminal, etc.) without touching core code.
Themes gain four new fields:
- layoutVariant: standard | cockpit | tiled — shell layout selector
- assets: {bg, hero, logo, crest, sidebar, header, custom: {...}} —
artwork URLs exposed as --theme-asset-* CSS vars
- customCSS: raw CSS injected as a scoped <style> tag on theme apply
(32 KiB cap, cleaned up on theme switch)
- componentStyles: per-component CSS-var overrides (clipPath,
borderImage, background, boxShadow, ...) for card/header/sidebar/
backdrop/tab/progress/badge/footer/page
Plugin manifests gain three new fields:
- tab.override: replaces a built-in route instead of adding a tab
- tab.hidden: register component + slots without adding a nav entry
- slots: declares shell slots the plugin populates
10 named shell slots: backdrop, header-left/right/banner, sidebar,
pre-main, post-main, footer-left/right, overlay. Plugins register via
window.__HERMES_PLUGINS__.registerSlot(name, slot, Component). A
<PluginSlot> React helper is exported on the plugin SDK.
Ships a full demo at plugins/strike-freedom-cockpit/ — theme YAML +
slot-only plugin that reproduces a Gundam cockpit dashboard: MS-STATUS
sidebar with live telemetry, COMPASS crest in header, notched card
corners via componentStyles, scanline overlay via customCSS, gold/cyan
palette, Orbitron typography.
Validation:
- 15 new tests in test_web_server.py covering every extended field
- tests/hermes_cli/: 2615 passed (3 pre-existing unrelated failures)
- tsc -b --noEmit: clean
- vite build: 418 kB bundle, ~2 kB delta for slots/theme extensions
Co-authored-by: Teknium <p@nousresearch.com>
2026-04-23 15:31:01 -07:00
|
|
|
|
} as unknown as React.CSSProperties
|
|
|
|
|
|
}
|
2026-04-19 10:48:56 -04:00
|
|
|
|
>
|
|
|
|
|
|
<img
|
|
|
|
|
|
alt=""
|
feat(dashboard): reskin extension points for themes and plugins (#14776)
Themes and plugins can now pull off arbitrary dashboard reskins (cockpit
HUD, retro terminal, etc.) without touching core code.
Themes gain four new fields:
- layoutVariant: standard | cockpit | tiled — shell layout selector
- assets: {bg, hero, logo, crest, sidebar, header, custom: {...}} —
artwork URLs exposed as --theme-asset-* CSS vars
- customCSS: raw CSS injected as a scoped <style> tag on theme apply
(32 KiB cap, cleaned up on theme switch)
- componentStyles: per-component CSS-var overrides (clipPath,
borderImage, background, boxShadow, ...) for card/header/sidebar/
backdrop/tab/progress/badge/footer/page
Plugin manifests gain three new fields:
- tab.override: replaces a built-in route instead of adding a tab
- tab.hidden: register component + slots without adding a nav entry
- slots: declares shell slots the plugin populates
10 named shell slots: backdrop, header-left/right/banner, sidebar,
pre-main, post-main, footer-left/right, overlay. Plugins register via
window.__HERMES_PLUGINS__.registerSlot(name, slot, Component). A
<PluginSlot> React helper is exported on the plugin SDK.
Ships a full demo at plugins/strike-freedom-cockpit/ — theme YAML +
slot-only plugin that reproduces a Gundam cockpit dashboard: MS-STATUS
sidebar with live telemetry, COMPASS crest in header, notched card
corners via componentStyles, scanline overlay via customCSS, gold/cyan
palette, Orbitron typography.
Validation:
- 15 new tests in test_web_server.py covering every extended field
- tests/hermes_cli/: 2615 passed (3 pre-existing unrelated failures)
- tsc -b --noEmit: clean
- vite build: 418 kB bundle, ~2 kB delta for slots/theme extensions
Co-authored-by: Teknium <p@nousresearch.com>
2026-04-23 15:31:01 -07:00
|
|
|
|
className="h-[150dvh] w-auto min-w-[100dvw] object-cover object-top-left invert theme-default-filler"
|
2026-04-19 10:48:56 -04:00
|
|
|
|
fetchPriority="low"
|
|
|
|
|
|
src="/ds-assets/filler-bg0.jpg"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div
|
|
|
|
|
|
aria-hidden
|
|
|
|
|
|
className="pointer-events-none fixed inset-0 z-[99]"
|
|
|
|
|
|
style={{
|
|
|
|
|
|
background:
|
|
|
|
|
|
"radial-gradient(ellipse at 0% 0%, transparent 60%, var(--warm-glow) 100%)",
|
|
|
|
|
|
mixBlendMode: "lighten",
|
|
|
|
|
|
opacity: 0.22,
|
|
|
|
|
|
}}
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
{gpuTier > 0 && (
|
|
|
|
|
|
<div
|
|
|
|
|
|
aria-hidden
|
|
|
|
|
|
className="pointer-events-none fixed inset-0 z-[101]"
|
|
|
|
|
|
style={{
|
|
|
|
|
|
backgroundImage:
|
|
|
|
|
|
"url(\"data:image/svg+xml,%3Csvg viewBox='0 0 512 512' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.85' numOctaves='4' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' fill='%23eaeaea' filter='url(%23n)' opacity='0.6'/%3E%3C/svg%3E\")",
|
|
|
|
|
|
backgroundSize: "512px 512px",
|
|
|
|
|
|
mixBlendMode: "color-dodge",
|
|
|
|
|
|
opacity: "calc(0.55 * var(--noise-opacity-mul, 1))",
|
|
|
|
|
|
}}
|
|
|
|
|
|
/>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|