# @shade/widgets Embeddable React widgets for live Shade observability. Drop them into any React dashboard (Nova, Orchestrator, your own apps) to show what's happening in your Shade deployment. ## Install ```bash bun add @shade/widgets react react-dom ``` ## Quick start ```tsx import { ShadeProvider, IdentityCard, SessionList, RecentActivity } from '@shade/widgets'; function MyDashboard() { return ( ); } ``` You need a running [`@shade/observer`](../shade-observer/README.md) endpoint for the widgets to talk to. ## Components | Component | Description | |-----------|-------------| | `` | Your fingerprint, registration ID, init/rotation timestamps | | `` | Active Shade sessions with per-session message counts and DH ratchet steps | | `` | Gauge of remaining one-time prekeys with low-stock warning | | `` | Live SSE feed of events flowing through the system | | `` | Prekey server stats (registered identities, fetches, replenishes, rate limits) | | `` | Paste a safety number to verify it matches your identity | | `` | Meta-widget letting users pick which widgets to display | ## Letting users pick widgets ```tsx ``` User selections persist to `localStorage`. Pass `onLayoutChange` to override with your own persistence. ## Theming ```tsx ``` Modes: `dark` (default), `light`, `auto` (matches `prefers-color-scheme`). Each widget renders self-contained CSS via inline styles — no Tailwind, no external CSS file, no conflicts with your host app. ## Hooks For custom layouts, use the underlying hooks directly: ```tsx import { useShadeState, useShadeEvents } from '@shade/widgets'; function CustomWidget() { const { state, loading } = useShadeState(); const { events, connected } = useShadeEvents(); // ... render whatever you want } ``` ## Polling interval Default poll for `/api/state` is 5 seconds. Override: ```tsx ``` The SSE event stream updates instantly regardless of poll interval.