mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-10 08:32:09 +00:00
* feat(dashboard): nous-blue theme, bulk sessions, schedule picker
Batch of related dashboard improvements gathered on
austin/fix/dashboard-changes:
* Nous Blue theme — faithful port of the LENS_5I overlay system onto
the existing DashboardTheme. Lifts the foreground inversion layer to
z-index 200 to fix the long-standing hover / loading visual artifact,
adds an explicit swatchColors slot so the theme picker shows the
post-inversion preview, and migrates the legacy "lens-5i" theme key
from localStorage / API to "nous-blue" on first read.
* Theme-aware series colors: new --series-input-token /
--series-output-token CSS vars consumed by Analytics + Models
charts; ToolCall + ModelInfoCard switched to semantic
--color-success for diff lines and the Tools capability badge.
* Analytics + Models headers: consolidate period selector + refresh
next to the page title and drop the redundant period badge.
* Bulk session management — "Delete empty (N)" button + per-row
checkboxes with shift-click range select and a bulk-delete action
bar. Backed by SessionDB.delete_sessions() /
delete_empty_sessions() plus POST /api/sessions/bulk-delete and
DELETE /api/sessions/empty (registered before the templated
/api/sessions/{session_id} family so they don't get shadowed).
Hard cap of 500 IDs per bulk request. Full pytest coverage.
* Cron page — human-readable schedule picker (every-interval / daily
/ weekly / monthly / once / custom) replaces the raw cron
expression input; the job list now renders "Weekly on Mon, Wed,
Fri at 14:30" instead of "30 14 * * 1,3,5". English-only ordinals
for monthly schedules so non-English locales don't get incorrect
suffixes.
* example-dashboard plugin moved from plugins/ to tests/fixtures/ so
stock installs no longer ship the demo. Tests install it
dynamically via a pytest fixture that also reorders the FastAPI
routes.
* i18n: 40+ new keys for the bulk-select UI and schedule
picker/describer translated across all 16 locales.
Co-authored-by: Cursor <cursoragent@cursor.com>
* refactor(dashboard): dedupe memory provider picker
The memory provider <Select> lived on both /system and /plugins,
writing the same config.yaml field through two different endpoints
with no cross-page refresh. Remove the picker from /system in favor
of a read-only status row + link to /plugins, where it pairs with
the context-engine picker under "Plugin providers".
/system retains the destructive admin controls (file sizes, Reset
MEMORY.md / USER.md / all). The api.setMemoryProvider client and
PUT /api/memory/provider backend endpoint are left in place for
CLI / script callers.
Co-authored-by: Cursor <cursoragent@cursor.com>
* docs(dashboard): address Copilot review on PR #37383
- Backdrop layer-stack comment claimed LENS_5I-style themes override
--component-backdrop-bg-blend-mode to multiply, but our only
LENS_5I-style theme (nous-blue) keeps the default difference.
Reword to describe what the code actually does and present the
var as a forward-looking extension hook.
- /api/sessions/bulk-delete docstring promised the response would
echo back the list of deleted IDs, but the implementation only
returns {ok, deleted}. Tighten the docstring to match the wire
format; the client already knows what it asked to delete, so the
IDs aren't needed.
Co-authored-by: Cursor <cursoragent@cursor.com>
* fix(dashboard): address copilot review on cron describe + bulk-select checkbox
- schedule.ts: restrict `describeCronExpression` to strictly 5-field cron
expressions. The backend `parse_schedule` also accepts the 6-field
`min hour dom month dow year` form, and humanising those by
destructuring only the first five fields would silently drop the year
(e.g. ``0 9 * * * 2099`` rendered as "Daily at 09:00"). 6+ field
expressions now fall through to the raw-string fallback so the user
sees what's actually scheduled.
- SessionsPage.tsx (SessionRow): wire the bulk-select Checkbox's
``onClick`` directly instead of attaching it to a parent ``<span>``
with a no-op ``onCheckedChange``. Radix forwards onClick to the
underlying ``<button role=checkbox>``, so the same handler now drives
both mouse clicks (preserving shift-key state for range select) and
keyboard activation (Space on the focused checkbox, which the browser
synthesises as a click on the <button>). Improves a11y / keyboard UX
without changing the controlled-selection model.
- SessionsPage.tsx: also extend ``SessionRowProps`` with the new
``onRename`` / ``onExport`` props introduced on main so the row's
destructured prop types resolve after the merge.
Co-authored-by: Cursor <cursoragent@cursor.com>
---------
Co-authored-by: Cursor <cursoragent@cursor.com>
137 lines
6 KiB
TypeScript
137 lines
6 KiB
TypeScript
import { useGpuTier } from "@nous-research/ui/hooks/use-gpu-tier";
|
||
|
||
import fillerBgUrl from "@nous-research/ui/assets/filler-bg0.webp";
|
||
|
||
/**
|
||
* 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 driven by
|
||
* `--component-backdrop-bg-blend-mode` (default `difference`).
|
||
* Both LENS_0-style dark themes and the LENS_5I-style Nous Blue
|
||
* light theme keep `difference` here — the canvas is flipped by
|
||
* the z-200 FG inversion layer, not by changing this blend mode.
|
||
* The CSS var is exposed as a hook so future presets can override
|
||
* it (e.g. `multiply` to paint the bg as-is before inversion)
|
||
* without touching this component.
|
||
* z-2 bundled filler-bg WebP, inverted, opacity 0.033, difference
|
||
* z-99 warm top-left vignette (`var(--warm-glow)`), opacity 0.22, lighten
|
||
* z-200 FG inversion = `var(--foreground)` (opaque white in LENS_5I,
|
||
* alpha-0 in LENS_0), mix-blend-mode: difference. This is the
|
||
* layer that flips the dashboard into "light mode" for inverted
|
||
* themes; for normal dark themes its alpha is 0 so it's a no-op.
|
||
* Deliberately placed above every UI overlay z-index (modals,
|
||
* tooltips, and dropUp dropdowns all sit at z-[100]) so portaled
|
||
* elements get inverted along with the rest of the page instead
|
||
* of painting with pre-inversion colors on top of the lens.
|
||
* z-201 noise grain (SVG, ~55% opacity × `--noise-opacity-mul`,
|
||
* color-dodge) — gated on GPU tier. Sits above the inversion
|
||
* layer by design so the grain is not flipped.
|
||
*
|
||
* `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:
|
||
"var(--component-backdrop-bg-blend-mode, difference)",
|
||
} as unknown as React.CSSProperties
|
||
}
|
||
/>
|
||
|
||
<div
|
||
aria-hidden
|
||
className="pointer-events-none fixed inset-0 z-[2]"
|
||
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.
|
||
mixBlendMode:
|
||
"var(--component-backdrop-filler-blend-mode, difference)",
|
||
opacity: "var(--component-backdrop-filler-opacity, 0.033)",
|
||
backgroundImage: "var(--theme-asset-bg)",
|
||
backgroundSize: "var(--component-backdrop-background-size, cover)",
|
||
backgroundPosition:
|
||
"var(--component-backdrop-background-position, center)",
|
||
} as unknown as React.CSSProperties
|
||
}
|
||
>
|
||
<img
|
||
alt=""
|
||
className="h-[150dvh] w-auto min-w-[100dvw] object-cover object-top-left invert theme-default-filler"
|
||
fetchPriority="low"
|
||
src={fillerBgUrl}
|
||
/>
|
||
</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,
|
||
}}
|
||
/>
|
||
|
||
{/* Foreground inversion layer. Source-of-truth: LENS_5I.Lens.fgOpacity
|
||
+ fgBlend: 'difference' in `design-language/src/ui/components/
|
||
overlays/lens.ts`. With `--foreground-alpha: 0` (LENS_0 dark default)
|
||
the layer is fully transparent and contributes nothing; with
|
||
alpha 1 + opaque white it inverts the entire stack below it,
|
||
producing the LENS_5I "light mode" look without altering any
|
||
downstream component code.
|
||
|
||
z-200 (not 100) so it sits above every portaled UI overlay —
|
||
sidebar tooltips, dropUp dropdowns, and modal dialogs all use
|
||
z-[100], which is what the DS Lens picks too; portals append
|
||
at the end of <body>, so equal z-index + later DOM order means
|
||
they'd paint on top of the inversion and skip the flip. Inlined
|
||
z-index for the same reason the DS does it — Tailwind's JIT
|
||
scan sometimes drops non-default z utilities. */}
|
||
<div
|
||
aria-hidden
|
||
className="pointer-events-none fixed inset-0"
|
||
style={{
|
||
backgroundColor: "var(--foreground)",
|
||
mixBlendMode: "difference",
|
||
zIndex: 200,
|
||
}}
|
||
/>
|
||
|
||
{gpuTier > 0 && (
|
||
<div
|
||
aria-hidden
|
||
className="pointer-events-none fixed inset-0 z-[201]"
|
||
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))",
|
||
}}
|
||
/>
|
||
)}
|
||
</>
|
||
);
|
||
}
|