@xemahq/ui-kernel 0.1.6 → 0.1.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/biome-host/agent-validation.d.ts +22 -0
- package/dist/lib/biome-host/agent-validation.d.ts.map +1 -0
- package/dist/lib/biome-host/agent-validation.js +127 -0
- package/dist/lib/biome-host/agent-validation.js.map +1 -0
- package/dist/lib/biome-host/create-biome-orval-config.d.ts +14 -0
- package/dist/lib/biome-host/create-biome-orval-config.d.ts.map +1 -0
- package/dist/lib/biome-host/create-biome-orval-config.js +22 -0
- package/dist/lib/biome-host/create-biome-orval-config.js.map +1 -0
- package/dist/lib/biome-host/frontend-biome.d.ts +3 -1
- package/dist/lib/biome-host/frontend-biome.d.ts.map +1 -1
- package/dist/lib/biome-host/host-bridge.d.ts +2 -0
- package/dist/lib/biome-host/host-bridge.d.ts.map +1 -1
- package/dist/lib/biome-host/host-bridge.js.map +1 -1
- package/dist/lib/biome-host/host-sources.d.ts +2 -0
- package/dist/lib/biome-host/host-sources.d.ts.map +1 -1
- package/dist/lib/biome-host/index.d.ts +2 -1
- package/dist/lib/biome-host/index.d.ts.map +1 -1
- package/dist/lib/biome-host/index.js +2 -1
- package/dist/lib/biome-host/index.js.map +1 -1
- package/dist/registry/index.d.ts +1 -1
- package/dist/registry/index.d.ts.map +1 -1
- package/dist/registry/index.js +1 -1
- package/dist/registry/index.js.map +1 -1
- package/dist/registry/lib/agent-validation-host.d.ts +3 -0
- package/dist/registry/lib/agent-validation-host.d.ts.map +1 -0
- package/dist/registry/lib/agent-validation-host.js +10 -0
- package/dist/registry/lib/agent-validation-host.js.map +1 -0
- package/dist/session-kit/approvals/ApprovalButton.d.ts +14 -0
- package/dist/session-kit/approvals/ApprovalButton.d.ts.map +1 -0
- package/dist/session-kit/approvals/ApprovalButton.js +45 -0
- package/dist/session-kit/approvals/ApprovalButton.js.map +1 -0
- package/dist/session-kit/approvals/ApprovalCard.d.ts +12 -0
- package/dist/session-kit/approvals/ApprovalCard.d.ts.map +1 -0
- package/dist/session-kit/approvals/ApprovalCard.js +117 -0
- package/dist/session-kit/approvals/ApprovalCard.js.map +1 -0
- package/dist/session-kit/approvals/ApprovalsCenter.d.ts +11 -0
- package/dist/session-kit/approvals/ApprovalsCenter.d.ts.map +1 -0
- package/dist/session-kit/approvals/ApprovalsCenter.js +127 -0
- package/dist/session-kit/approvals/ApprovalsCenter.js.map +1 -0
- package/dist/session-kit/approvals/CapabilityApprovalStickyBar.d.ts +12 -0
- package/dist/session-kit/approvals/CapabilityApprovalStickyBar.d.ts.map +1 -0
- package/dist/session-kit/approvals/CapabilityApprovalStickyBar.js +36 -0
- package/dist/session-kit/approvals/CapabilityApprovalStickyBar.js.map +1 -0
- package/dist/session-kit/approvals/Obligation.d.ts +15 -0
- package/dist/session-kit/approvals/Obligation.d.ts.map +1 -0
- package/dist/session-kit/approvals/Obligation.js +42 -0
- package/dist/session-kit/approvals/Obligation.js.map +1 -0
- package/dist/session-kit/approvals/ScopedBody.d.ts +9 -0
- package/dist/session-kit/approvals/ScopedBody.d.ts.map +1 -0
- package/dist/session-kit/approvals/ScopedBody.js +145 -0
- package/dist/session-kit/approvals/ScopedBody.js.map +1 -0
- package/dist/session-kit/approvals/approval-icons.d.ts +15 -0
- package/dist/session-kit/approvals/approval-icons.d.ts.map +1 -0
- package/dist/session-kit/approvals/approval-icons.js +3 -0
- package/dist/session-kit/approvals/approval-icons.js.map +1 -0
- package/dist/session-kit/approvals/approval-model.d.ts +83 -0
- package/dist/session-kit/approvals/approval-model.d.ts.map +1 -0
- package/dist/session-kit/approvals/approval-model.js +25 -0
- package/dist/session-kit/approvals/approval-model.js.map +1 -0
- package/dist/session-kit/approvals/index.d.ts +12 -0
- package/dist/session-kit/approvals/index.d.ts.map +1 -0
- package/dist/session-kit/approvals/index.js +28 -0
- package/dist/session-kit/approvals/index.js.map +1 -0
- package/dist/session-kit/approvals/obligation-display.d.ts +17 -0
- package/dist/session-kit/approvals/obligation-display.d.ts.map +1 -0
- package/dist/session-kit/approvals/obligation-display.js +58 -0
- package/dist/session-kit/approvals/obligation-display.js.map +1 -0
- package/dist/session-kit/approvals/risk-accent.d.ts +8 -0
- package/dist/session-kit/approvals/risk-accent.d.ts.map +1 -0
- package/dist/session-kit/approvals/risk-accent.js +28 -0
- package/dist/session-kit/approvals/risk-accent.js.map +1 -0
- package/dist/session-kit/approvals/scope-icons.d.ts +12 -0
- package/dist/session-kit/approvals/scope-icons.d.ts.map +1 -0
- package/dist/session-kit/approvals/scope-icons.js +14 -0
- package/dist/session-kit/approvals/scope-icons.js.map +1 -0
- package/dist/session-kit/combobox/Combobox.d.ts +46 -0
- package/dist/session-kit/combobox/Combobox.d.ts.map +1 -0
- package/dist/session-kit/combobox/Combobox.js +113 -0
- package/dist/session-kit/combobox/Combobox.js.map +1 -0
- package/dist/session-kit/combobox/use-click-outside.d.ts +3 -0
- package/dist/session-kit/combobox/use-click-outside.d.ts.map +1 -0
- package/dist/session-kit/combobox/use-click-outside.js +18 -0
- package/dist/session-kit/combobox/use-click-outside.js.map +1 -0
- package/dist/session-kit/display/ContextHeader.d.ts +27 -0
- package/dist/session-kit/display/ContextHeader.d.ts.map +1 -0
- package/dist/session-kit/display/ContextHeader.js +47 -0
- package/dist/session-kit/display/ContextHeader.js.map +1 -0
- package/dist/session-kit/display/FileDiffCard.d.ts +18 -0
- package/dist/session-kit/display/FileDiffCard.d.ts.map +1 -0
- package/dist/session-kit/display/FileDiffCard.js +58 -0
- package/dist/session-kit/display/FileDiffCard.js.map +1 -0
- package/dist/session-kit/display/MD.d.ts +7 -0
- package/dist/session-kit/display/MD.d.ts.map +1 -0
- package/dist/session-kit/display/MD.js +89 -0
- package/dist/session-kit/display/MD.js.map +1 -0
- package/dist/session-kit/display/MessageTurn.d.ts +21 -0
- package/dist/session-kit/display/MessageTurn.d.ts.map +1 -0
- package/dist/session-kit/display/MessageTurn.js +62 -0
- package/dist/session-kit/display/MessageTurn.js.map +1 -0
- package/dist/session-kit/display/ThinkingPanel.d.ts +12 -0
- package/dist/session-kit/display/ThinkingPanel.d.ts.map +1 -0
- package/dist/session-kit/display/ThinkingPanel.js +30 -0
- package/dist/session-kit/display/ThinkingPanel.js.map +1 -0
- package/dist/session-kit/display/TodoChecklist.d.ts +17 -0
- package/dist/session-kit/display/TodoChecklist.d.ts.map +1 -0
- package/dist/session-kit/display/TodoChecklist.js +50 -0
- package/dist/session-kit/display/TodoChecklist.js.map +1 -0
- package/dist/session-kit/display/TokenMeter.d.ts +15 -0
- package/dist/session-kit/display/TokenMeter.d.ts.map +1 -0
- package/dist/session-kit/display/TokenMeter.js +35 -0
- package/dist/session-kit/display/TokenMeter.js.map +1 -0
- package/dist/session-kit/display/ToolStrip.d.ts +31 -0
- package/dist/session-kit/display/ToolStrip.d.ts.map +1 -0
- package/dist/session-kit/display/ToolStrip.js +99 -0
- package/dist/session-kit/display/ToolStrip.js.map +1 -0
- package/dist/session-kit/display/TypingDots.d.ts +3 -0
- package/dist/session-kit/display/TypingDots.d.ts.map +1 -0
- package/dist/session-kit/display/TypingDots.js +14 -0
- package/dist/session-kit/display/TypingDots.js.map +1 -0
- package/dist/session-kit/index.d.ts +21 -0
- package/dist/session-kit/index.d.ts.map +1 -0
- package/dist/session-kit/index.js +37 -0
- package/dist/session-kit/index.js.map +1 -0
- package/dist/session-kit/lib/enums.d.ts +34 -0
- package/dist/session-kit/lib/enums.d.ts.map +1 -0
- package/dist/session-kit/lib/enums.js +44 -0
- package/dist/session-kit/lib/enums.js.map +1 -0
- package/dist/session-kit/lib/portal-accent.d.ts +3 -0
- package/dist/session-kit/lib/portal-accent.d.ts.map +1 -0
- package/dist/session-kit/lib/portal-accent.js +9 -0
- package/dist/session-kit/lib/portal-accent.js.map +1 -0
- package/dist/session-kit/lib/status-dot.d.ts +10 -0
- package/dist/session-kit/lib/status-dot.d.ts.map +1 -0
- package/dist/session-kit/lib/status-dot.js +43 -0
- package/dist/session-kit/lib/status-dot.js.map +1 -0
- package/dist/session-kit/primitives/Avatar.d.ts +10 -0
- package/dist/session-kit/primitives/Avatar.d.ts.map +1 -0
- package/dist/session-kit/primitives/Avatar.js +21 -0
- package/dist/session-kit/primitives/Avatar.js.map +1 -0
- package/dist/session-kit/primitives/PortalGlyph.d.ts +12 -0
- package/dist/session-kit/primitives/PortalGlyph.d.ts.map +1 -0
- package/dist/session-kit/primitives/PortalGlyph.js +21 -0
- package/dist/session-kit/primitives/PortalGlyph.js.map +1 -0
- package/dist/session-kit/primitives/RiskPill.d.ts +12 -0
- package/dist/session-kit/primitives/RiskPill.d.ts.map +1 -0
- package/dist/session-kit/primitives/RiskPill.js +53 -0
- package/dist/session-kit/primitives/RiskPill.js.map +1 -0
- package/dist/session-kit/primitives/ScopeDot.d.ts +9 -0
- package/dist/session-kit/primitives/ScopeDot.d.ts.map +1 -0
- package/dist/session-kit/primitives/ScopeDot.js +23 -0
- package/dist/session-kit/primitives/ScopeDot.js.map +1 -0
- package/dist/session-kit/primitives/Segmented.d.ts +16 -0
- package/dist/session-kit/primitives/Segmented.d.ts.map +1 -0
- package/dist/session-kit/primitives/Segmented.js +31 -0
- package/dist/session-kit/primitives/Segmented.js.map +1 -0
- package/package.json +2 -2
- package/src/lib/biome-host/create-biome-orval-config.ts +76 -0
- package/src/lib/biome-host/frontend-biome.ts +22 -2
- package/src/lib/biome-host/host-bridge.ts +22 -0
- package/src/lib/biome-host/host-sources.ts +13 -0
- package/src/lib/biome-host/index.ts +2 -1
- package/src/registry/index.ts +1 -1
- package/src/registry/lib/extension-points.ts +1 -1
- package/src/session-kit/approvals/ApprovalButton.tsx +89 -0
- package/src/session-kit/approvals/ApprovalCard.tsx +336 -0
- package/src/session-kit/approvals/ApprovalsCenter.tsx +327 -0
- package/src/session-kit/approvals/CapabilityApprovalStickyBar.tsx +118 -0
- package/src/session-kit/approvals/Obligation.tsx +111 -0
- package/src/session-kit/approvals/ScopedBody.tsx +392 -0
- package/src/session-kit/approvals/approval-icons.ts +31 -0
- package/src/session-kit/approvals/approval-model.ts +205 -0
- package/src/session-kit/approvals/index.ts +22 -0
- package/src/session-kit/approvals/obligation-display.ts +100 -0
- package/src/session-kit/approvals/risk-accent.ts +47 -0
- package/src/session-kit/approvals/scope-icons.ts +19 -0
- package/src/session-kit/combobox/Combobox.tsx +327 -0
- package/src/session-kit/combobox/use-click-outside.ts +21 -0
- package/src/session-kit/display/ContextHeader.tsx +148 -0
- package/src/session-kit/display/FileDiffCard.tsx +140 -0
- package/src/session-kit/display/MD.tsx +153 -0
- package/src/session-kit/display/MessageTurn.tsx +157 -0
- package/src/session-kit/display/ThinkingPanel.tsx +78 -0
- package/src/session-kit/display/TodoChecklist.tsx +120 -0
- package/src/session-kit/display/TokenMeter.tsx +89 -0
- package/src/session-kit/display/ToolStrip.tsx +278 -0
- package/src/session-kit/display/TypingDots.tsx +24 -0
- package/src/session-kit/index.ts +44 -0
- package/src/session-kit/lib/enums.ts +66 -0
- package/src/session-kit/lib/portal-accent.ts +30 -0
- package/src/session-kit/lib/status-dot.ts +68 -0
- package/src/session-kit/primitives/Avatar.tsx +44 -0
- package/src/session-kit/primitives/PortalGlyph.tsx +51 -0
- package/src/session-kit/primitives/RiskPill.tsx +95 -0
- package/src/session-kit/primitives/ScopeDot.tsx +47 -0
- package/src/session-kit/primitives/Segmented.tsx +71 -0
- /package/src/lib/biome-host/{composition-validation.ts → agent-validation.ts} +0 -0
- /package/src/registry/lib/{composition-validation-host.ts → agent-validation-host.ts} +0 -0
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Obligation → display metadata map (HANDOFF §6).
|
|
3
|
+
*
|
|
4
|
+
* Covers ALL NINE obligation kinds in the kernel closed taxonomy —
|
|
5
|
+
* `audit`, `redact-secrets`, `require-runner-kind`,
|
|
6
|
+
* `require-human-approval`, `max-duration-seconds`, `max-cost-usd`,
|
|
7
|
+
* `restrict-output-classification`, `data-residency`, `egress-allowlist`.
|
|
8
|
+
*
|
|
9
|
+
* The map is typed `Record<PolicyObligationKind, …>` so it is EXHAUSTIVE
|
|
10
|
+
* over the installed kernel enum: adding a new `PolicyObligationKind`
|
|
11
|
+
* member is a COMPILE ERROR here until it gets a label + icon key. The
|
|
12
|
+
* published `@xemahq/kernel-contracts` enum now ships all nine members
|
|
13
|
+
* (incl. `egress-allowlist`), so every kind is keyed directly by its enum
|
|
14
|
+
* value — no literal-wire fallback is needed.
|
|
15
|
+
*
|
|
16
|
+
* The kit is icon-agnostic: this module names a stable `iconKey`
|
|
17
|
+
* (`ObligationIcon`) and the host translates it into a concrete icon node
|
|
18
|
+
* via the `Obligation`/`renderIcon` prop. No icon dependency leaks into
|
|
19
|
+
* the kernel package.
|
|
20
|
+
*/
|
|
21
|
+
import { PolicyObligationKind } from '@xemahq/kernel-contracts/policy';
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Stable, icon-set-agnostic glyph keys for obligation chips. The host
|
|
25
|
+
* maps each onto a concrete icon (lucide, etc.); the kit never imports an
|
|
26
|
+
* icon set. Closed set — one key per obligation family.
|
|
27
|
+
*/
|
|
28
|
+
export enum ObligationIcon {
|
|
29
|
+
Dollar = 'dollar',
|
|
30
|
+
ShieldCheck = 'shield-check',
|
|
31
|
+
Scroll = 'scroll',
|
|
32
|
+
Eye = 'eye',
|
|
33
|
+
Clock = 'clock',
|
|
34
|
+
Layers = 'layers',
|
|
35
|
+
Globe = 'globe',
|
|
36
|
+
Lock = 'lock',
|
|
37
|
+
Network = 'network',
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export interface ObligationDisplayMeta {
|
|
41
|
+
readonly iconKey: ObligationIcon;
|
|
42
|
+
readonly label: string;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Chip metadata for every member of the kernel `PolicyObligationKind`
|
|
47
|
+
* enum. Typed as `Record<PolicyObligationKind, …>` so adding a new kernel
|
|
48
|
+
* enum member is a COMPILE ERROR here until it gets a label + icon key —
|
|
49
|
+
* the exhaustiveness guarantee.
|
|
50
|
+
*/
|
|
51
|
+
const ENUM_OBLIGATION_DISPLAY: Record<
|
|
52
|
+
PolicyObligationKind,
|
|
53
|
+
ObligationDisplayMeta
|
|
54
|
+
> = {
|
|
55
|
+
[PolicyObligationKind.MaxCostUsd]: {
|
|
56
|
+
iconKey: ObligationIcon.Dollar,
|
|
57
|
+
label: 'Spend cap',
|
|
58
|
+
},
|
|
59
|
+
[PolicyObligationKind.RequireHumanApproval]: {
|
|
60
|
+
iconKey: ObligationIcon.ShieldCheck,
|
|
61
|
+
label: 'Human approval',
|
|
62
|
+
},
|
|
63
|
+
[PolicyObligationKind.Audit]: {
|
|
64
|
+
iconKey: ObligationIcon.Scroll,
|
|
65
|
+
label: 'Audited',
|
|
66
|
+
},
|
|
67
|
+
[PolicyObligationKind.RestrictOutputClassification]: {
|
|
68
|
+
iconKey: ObligationIcon.Eye,
|
|
69
|
+
label: 'Output limited',
|
|
70
|
+
},
|
|
71
|
+
[PolicyObligationKind.MaxDurationSeconds]: {
|
|
72
|
+
iconKey: ObligationIcon.Clock,
|
|
73
|
+
label: 'Time cap',
|
|
74
|
+
},
|
|
75
|
+
[PolicyObligationKind.RequireRunnerKind]: {
|
|
76
|
+
iconKey: ObligationIcon.Layers,
|
|
77
|
+
label: 'Runner',
|
|
78
|
+
},
|
|
79
|
+
[PolicyObligationKind.DataResidency]: {
|
|
80
|
+
iconKey: ObligationIcon.Globe,
|
|
81
|
+
label: 'Residency',
|
|
82
|
+
},
|
|
83
|
+
[PolicyObligationKind.RedactSecrets]: {
|
|
84
|
+
iconKey: ObligationIcon.Lock,
|
|
85
|
+
label: 'Redacted',
|
|
86
|
+
},
|
|
87
|
+
[PolicyObligationKind.EgressAllowlist]: {
|
|
88
|
+
iconKey: ObligationIcon.Network,
|
|
89
|
+
label: 'Egress',
|
|
90
|
+
},
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* The authoritative obligation-kind → chip metadata map, keyed by the
|
|
95
|
+
* obligation wire-string. Exhaustive over the installed kernel enum via
|
|
96
|
+
* `ENUM_OBLIGATION_DISPLAY` (compile-checked).
|
|
97
|
+
*/
|
|
98
|
+
export const OBLIGATION_DISPLAY: Record<string, ObligationDisplayMeta> = {
|
|
99
|
+
...ENUM_OBLIGATION_DISPLAY,
|
|
100
|
+
};
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Risk-tier → colour expressions for the approval frame's accent rail and
|
|
3
|
+
* the sticky bar (HANDOFF §6). RiskPill owns the PILL styling; this map is
|
|
4
|
+
* the matching surface accent (foreground / soft background / border) the
|
|
5
|
+
* frame and bar paint with. Both read the SAME kernel `CapabilityRiskTier`
|
|
6
|
+
* enum so the pill and the rail can never disagree on a tier's colour.
|
|
7
|
+
*
|
|
8
|
+
* low → ink / muted (quiet)
|
|
9
|
+
* medium → --info
|
|
10
|
+
* high → --warning
|
|
11
|
+
* critical → --destructive / --danger
|
|
12
|
+
*/
|
|
13
|
+
import { CapabilityRiskTier } from '@xemahq/kernel-contracts/capability';
|
|
14
|
+
|
|
15
|
+
export interface RiskAccent {
|
|
16
|
+
/** Solid accent (rail, icon, critical button). */
|
|
17
|
+
readonly fg: string;
|
|
18
|
+
/** Soft tinted background (sticky bar surface). */
|
|
19
|
+
readonly bg: string;
|
|
20
|
+
/** Border (sticky bar, critical card outline). */
|
|
21
|
+
readonly bd: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const DANGER = 'var(--destructive, var(--danger))';
|
|
25
|
+
|
|
26
|
+
export const RISK_ACCENT: Record<CapabilityRiskTier, RiskAccent> = {
|
|
27
|
+
[CapabilityRiskTier.Low]: {
|
|
28
|
+
fg: 'hsl(var(--ink-3))',
|
|
29
|
+
bg: 'hsl(var(--muted))',
|
|
30
|
+
bd: 'hsl(var(--rule))',
|
|
31
|
+
},
|
|
32
|
+
[CapabilityRiskTier.Medium]: {
|
|
33
|
+
fg: 'hsl(var(--info))',
|
|
34
|
+
bg: 'hsl(var(--info) / 0.07)',
|
|
35
|
+
bd: 'hsl(var(--info) / 0.28)',
|
|
36
|
+
},
|
|
37
|
+
[CapabilityRiskTier.High]: {
|
|
38
|
+
fg: 'hsl(var(--warning))',
|
|
39
|
+
bg: 'hsl(var(--warning) / 0.08)',
|
|
40
|
+
bd: 'hsl(var(--warning) / 0.3)',
|
|
41
|
+
},
|
|
42
|
+
[CapabilityRiskTier.Critical]: {
|
|
43
|
+
fg: `hsl(${DANGER})`,
|
|
44
|
+
bg: `hsl(${DANGER} / 0.07)`,
|
|
45
|
+
bd: `hsl(${DANGER} / 0.32)`,
|
|
46
|
+
},
|
|
47
|
+
};
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Stable, icon-set-agnostic glyph keys used by the scoped approval bodies
|
|
3
|
+
* (`ScopedBody`). Kept in its own module (not alongside the body
|
|
4
|
+
* components) so the icon bundle and the bodies can both import it without
|
|
5
|
+
* a component file also exporting a non-component value.
|
|
6
|
+
*/
|
|
7
|
+
import type { ReactNode } from 'react';
|
|
8
|
+
|
|
9
|
+
export enum ScopeIcon {
|
|
10
|
+
Database = 'database',
|
|
11
|
+
Layers = 'layers',
|
|
12
|
+
Alert = 'alert',
|
|
13
|
+
Send = 'send',
|
|
14
|
+
Globe = 'globe',
|
|
15
|
+
ArrowRight = 'arrow-right',
|
|
16
|
+
FileEdit = 'file-edit',
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export type ScopeIconRenderer = (key: ScopeIcon, size: number) => ReactNode;
|
|
@@ -0,0 +1,327 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generic searchable picker (HANDOFF §7). Use it wherever a closed set
|
|
3
|
+
* is selected (agents, models, skills/tools, …).
|
|
4
|
+
*
|
|
5
|
+
* - Shows a search field when `items.length > searchThreshold` (default
|
|
6
|
+
* 7) with a live `shown/total` count.
|
|
7
|
+
* - Scrolls a long list (max-height 240).
|
|
8
|
+
* - Single OR multi (checkbox) select via a discriminated `multi` flag.
|
|
9
|
+
* - Long labels truncate with ellipsis in the trigger; the list always
|
|
10
|
+
* shows the full label.
|
|
11
|
+
* - Opens up/down, left/right aligned, with an optional footer slot.
|
|
12
|
+
* - Click-outside closes (and clears the query).
|
|
13
|
+
*
|
|
14
|
+
* Standalone (not built on a host Popover): the design's behaviour is a
|
|
15
|
+
* small self-contained surface (click-outside + absolute positioning),
|
|
16
|
+
* and this package is layer-agnostic ui-kernel — it must not depend on
|
|
17
|
+
* host primitives. So we ship a minimal, faithful standalone impl. The
|
|
18
|
+
* search/check/checkbox icons are host-supplied (icon-agnostic kit).
|
|
19
|
+
*/
|
|
20
|
+
import {
|
|
21
|
+
useEffect,
|
|
22
|
+
useRef,
|
|
23
|
+
useState,
|
|
24
|
+
type CSSProperties,
|
|
25
|
+
type ReactElement,
|
|
26
|
+
type ReactNode,
|
|
27
|
+
} from 'react';
|
|
28
|
+
|
|
29
|
+
import { useClickOutside } from './use-click-outside';
|
|
30
|
+
|
|
31
|
+
export interface ComboboxItem<V extends string> {
|
|
32
|
+
readonly value: V;
|
|
33
|
+
readonly label: string;
|
|
34
|
+
/** Secondary muted line under the label. */
|
|
35
|
+
readonly hint?: string;
|
|
36
|
+
/** Leading icon node (host-supplied). */
|
|
37
|
+
readonly icon?: ReactNode;
|
|
38
|
+
/** Small uppercase kind tag on the right (`.cap`). */
|
|
39
|
+
readonly kind?: string;
|
|
40
|
+
/** Render the label in the mono face. */
|
|
41
|
+
readonly mono?: boolean;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Trigger render-prop state. The consumer renders its own interactive
|
|
46
|
+
* trigger element (typically a `<button>`) and wires `toggle` to its
|
|
47
|
+
* `onClick`, so the picker never nests interactive elements.
|
|
48
|
+
*/
|
|
49
|
+
export interface ComboboxTriggerState<V extends string> {
|
|
50
|
+
readonly open: boolean;
|
|
51
|
+
/** The selected item (single-select only; `undefined` in multi mode). */
|
|
52
|
+
readonly selected: ComboboxItem<V> | undefined;
|
|
53
|
+
/** Toggle the dropdown open/closed. */
|
|
54
|
+
readonly toggle: () => void;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
export type ComboboxTrigger<V extends string> = (
|
|
58
|
+
state: ComboboxTriggerState<V>,
|
|
59
|
+
) => ReactNode;
|
|
60
|
+
|
|
61
|
+
/** Host-supplied icons (kit carries no icon set). */
|
|
62
|
+
export interface ComboboxIcons {
|
|
63
|
+
readonly search?: ReactNode;
|
|
64
|
+
/** Check mark for the selected single-select row. */
|
|
65
|
+
readonly check?: ReactNode;
|
|
66
|
+
/** Check mark rendered inside a ticked multi-select box. */
|
|
67
|
+
readonly multiCheck?: ReactNode;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
export type ComboboxAlign = 'left' | 'right';
|
|
71
|
+
|
|
72
|
+
interface ComboboxBaseProps<V extends string> {
|
|
73
|
+
readonly items: ReadonlyArray<ComboboxItem<V>>;
|
|
74
|
+
readonly trigger: ComboboxTrigger<V>;
|
|
75
|
+
readonly icons?: ComboboxIcons;
|
|
76
|
+
readonly placeholder?: string;
|
|
77
|
+
readonly searchThreshold?: number;
|
|
78
|
+
readonly align?: ComboboxAlign;
|
|
79
|
+
/** Open upward instead of downward. */
|
|
80
|
+
readonly up?: boolean;
|
|
81
|
+
readonly width?: number;
|
|
82
|
+
readonly footer?: ReactNode;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
interface ComboboxSingleProps<V extends string> extends ComboboxBaseProps<V> {
|
|
86
|
+
readonly multi?: false;
|
|
87
|
+
readonly value: V | undefined;
|
|
88
|
+
readonly onChange: (value: V) => void;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
interface ComboboxMultiProps<V extends string> extends ComboboxBaseProps<V> {
|
|
92
|
+
readonly multi: true;
|
|
93
|
+
readonly value: ReadonlyArray<V>;
|
|
94
|
+
readonly onChange: (value: V) => void;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
export type ComboboxProps<V extends string> =
|
|
98
|
+
| ComboboxSingleProps<V>
|
|
99
|
+
| ComboboxMultiProps<V>;
|
|
100
|
+
|
|
101
|
+
const POP_STYLE: CSSProperties = {
|
|
102
|
+
position: 'absolute',
|
|
103
|
+
zIndex: 60,
|
|
104
|
+
borderRadius: 11,
|
|
105
|
+
background: 'hsl(var(--paper-elev))',
|
|
106
|
+
border: '1px solid hsl(var(--rule-2))',
|
|
107
|
+
boxShadow: 'var(--shadow-28, var(--shadow-pop))',
|
|
108
|
+
overflow: 'hidden',
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
export function Combobox<V extends string>(props: ComboboxProps<V>): ReactElement {
|
|
112
|
+
const {
|
|
113
|
+
items,
|
|
114
|
+
trigger,
|
|
115
|
+
icons,
|
|
116
|
+
placeholder = 'Search…',
|
|
117
|
+
searchThreshold = 7,
|
|
118
|
+
align = 'left',
|
|
119
|
+
up = false,
|
|
120
|
+
width = 240,
|
|
121
|
+
footer,
|
|
122
|
+
} = props;
|
|
123
|
+
|
|
124
|
+
const [open, setOpen] = useState(false);
|
|
125
|
+
const [query, setQuery] = useState('');
|
|
126
|
+
const ref = useClickOutside<HTMLDivElement>(() => {
|
|
127
|
+
setOpen(false);
|
|
128
|
+
setQuery('');
|
|
129
|
+
});
|
|
130
|
+
const searchRef = useRef<HTMLInputElement>(null);
|
|
131
|
+
|
|
132
|
+
const showSearch = items.length > searchThreshold;
|
|
133
|
+
|
|
134
|
+
// Move focus to the search field when the dropdown opens (replaces the
|
|
135
|
+
// a11y-discouraged `autoFocus` attribute).
|
|
136
|
+
useEffect(() => {
|
|
137
|
+
if (open && showSearch) searchRef.current?.focus();
|
|
138
|
+
}, [open, showSearch]);
|
|
139
|
+
|
|
140
|
+
const filtered = query
|
|
141
|
+
? items.filter((it) =>
|
|
142
|
+
`${it.label} ${it.hint ?? ''}`.toLowerCase().includes(query.toLowerCase()),
|
|
143
|
+
)
|
|
144
|
+
: items;
|
|
145
|
+
|
|
146
|
+
const isOn = (value: V): boolean =>
|
|
147
|
+
props.multi ? props.value.includes(value) : props.value === value;
|
|
148
|
+
|
|
149
|
+
const selected = props.multi
|
|
150
|
+
? undefined
|
|
151
|
+
: items.find((it) => it.value === props.value);
|
|
152
|
+
|
|
153
|
+
function pick(value: V): void {
|
|
154
|
+
props.onChange(value);
|
|
155
|
+
if (!props.multi) {
|
|
156
|
+
setOpen(false);
|
|
157
|
+
setQuery('');
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
const pos: CSSProperties = up
|
|
162
|
+
? { bottom: 'calc(100% + 6px)' }
|
|
163
|
+
: { top: 'calc(100% + 6px)' };
|
|
164
|
+
|
|
165
|
+
return (
|
|
166
|
+
<div
|
|
167
|
+
ref={ref}
|
|
168
|
+
style={{ position: 'relative', display: 'inline-flex', flexShrink: 0 }}
|
|
169
|
+
>
|
|
170
|
+
{trigger({ open, selected, toggle: () => setOpen((o) => !o) })}
|
|
171
|
+
{open && (
|
|
172
|
+
<div style={{ ...POP_STYLE, ...pos, [align]: 0, width }}>
|
|
173
|
+
{showSearch && (
|
|
174
|
+
<div
|
|
175
|
+
style={{
|
|
176
|
+
display: 'flex',
|
|
177
|
+
alignItems: 'center',
|
|
178
|
+
gap: 7,
|
|
179
|
+
padding: '9px 11px',
|
|
180
|
+
borderBottom: '1px solid hsl(var(--rule))',
|
|
181
|
+
}}
|
|
182
|
+
>
|
|
183
|
+
{icons?.search && (
|
|
184
|
+
<span style={{ display: 'inline-flex', color: 'hsl(var(--ink-3))' }}>
|
|
185
|
+
{icons.search}
|
|
186
|
+
</span>
|
|
187
|
+
)}
|
|
188
|
+
<input
|
|
189
|
+
ref={searchRef}
|
|
190
|
+
value={query}
|
|
191
|
+
onChange={(e) => setQuery(e.target.value)}
|
|
192
|
+
placeholder={placeholder}
|
|
193
|
+
style={{
|
|
194
|
+
flex: 1,
|
|
195
|
+
border: 'none',
|
|
196
|
+
outline: 'none',
|
|
197
|
+
background: 'transparent',
|
|
198
|
+
fontSize: 13,
|
|
199
|
+
color: 'hsl(var(--ink))',
|
|
200
|
+
}}
|
|
201
|
+
/>
|
|
202
|
+
{items.length > 0 && (
|
|
203
|
+
<span className="mono" style={{ fontSize: 10, color: 'hsl(var(--ink-4))' }}>
|
|
204
|
+
{filtered.length}/{items.length}
|
|
205
|
+
</span>
|
|
206
|
+
)}
|
|
207
|
+
</div>
|
|
208
|
+
)}
|
|
209
|
+
<div style={{ maxHeight: 240, overflowY: 'auto', padding: 5 }}>
|
|
210
|
+
{filtered.map((it) => {
|
|
211
|
+
const on = isOn(it.value);
|
|
212
|
+
const singleSelected = on && !props.multi;
|
|
213
|
+
return (
|
|
214
|
+
<button
|
|
215
|
+
type="button"
|
|
216
|
+
key={it.value}
|
|
217
|
+
onClick={() => pick(it.value)}
|
|
218
|
+
style={{
|
|
219
|
+
display: 'flex',
|
|
220
|
+
alignItems: 'center',
|
|
221
|
+
gap: 9,
|
|
222
|
+
width: '100%',
|
|
223
|
+
padding: '7px 9px',
|
|
224
|
+
borderRadius: 7,
|
|
225
|
+
textAlign: 'left',
|
|
226
|
+
background: singleSelected ? 'hsl(var(--accent) / 0.6)' : 'transparent',
|
|
227
|
+
transition: 'background 140ms ease',
|
|
228
|
+
}}
|
|
229
|
+
onMouseEnter={(e) => {
|
|
230
|
+
if (!singleSelected) {
|
|
231
|
+
e.currentTarget.style.background = 'hsl(var(--paper-sunk))';
|
|
232
|
+
}
|
|
233
|
+
}}
|
|
234
|
+
onMouseLeave={(e) => {
|
|
235
|
+
if (!singleSelected) {
|
|
236
|
+
e.currentTarget.style.background = 'transparent';
|
|
237
|
+
}
|
|
238
|
+
}}
|
|
239
|
+
>
|
|
240
|
+
{it.icon && (
|
|
241
|
+
<span style={{ display: 'inline-flex', color: 'hsl(var(--ink-3))', flexShrink: 0 }}>
|
|
242
|
+
{it.icon}
|
|
243
|
+
</span>
|
|
244
|
+
)}
|
|
245
|
+
<span style={{ flex: 1, minWidth: 0 }}>
|
|
246
|
+
<span
|
|
247
|
+
className={it.mono ? 'mono' : ''}
|
|
248
|
+
style={{
|
|
249
|
+
display: 'block',
|
|
250
|
+
fontSize: it.mono ? 12 : 12.5,
|
|
251
|
+
color: 'hsl(var(--ink))',
|
|
252
|
+
overflow: 'hidden',
|
|
253
|
+
textOverflow: 'ellipsis',
|
|
254
|
+
whiteSpace: 'nowrap',
|
|
255
|
+
}}
|
|
256
|
+
>
|
|
257
|
+
{it.label}
|
|
258
|
+
</span>
|
|
259
|
+
{it.hint && (
|
|
260
|
+
<span
|
|
261
|
+
style={{
|
|
262
|
+
display: 'block',
|
|
263
|
+
fontSize: 10.5,
|
|
264
|
+
color: 'hsl(var(--ink-4))',
|
|
265
|
+
overflow: 'hidden',
|
|
266
|
+
textOverflow: 'ellipsis',
|
|
267
|
+
whiteSpace: 'nowrap',
|
|
268
|
+
}}
|
|
269
|
+
>
|
|
270
|
+
{it.hint}
|
|
271
|
+
</span>
|
|
272
|
+
)}
|
|
273
|
+
</span>
|
|
274
|
+
{it.kind && (
|
|
275
|
+
<span className="cap" style={{ fontSize: 8.5, flexShrink: 0 }}>
|
|
276
|
+
{it.kind}
|
|
277
|
+
</span>
|
|
278
|
+
)}
|
|
279
|
+
{props.multi ? (
|
|
280
|
+
<span
|
|
281
|
+
style={{
|
|
282
|
+
width: 17,
|
|
283
|
+
height: 17,
|
|
284
|
+
borderRadius: 5,
|
|
285
|
+
flexShrink: 0,
|
|
286
|
+
display: 'grid',
|
|
287
|
+
placeItems: 'center',
|
|
288
|
+
background: on ? 'hsl(var(--primary))' : 'transparent',
|
|
289
|
+
border: on ? 'none' : '1.5px solid hsl(var(--rule-2))',
|
|
290
|
+
color: '#fff',
|
|
291
|
+
}}
|
|
292
|
+
>
|
|
293
|
+
{on && icons?.multiCheck}
|
|
294
|
+
</span>
|
|
295
|
+
) : (
|
|
296
|
+
on && (
|
|
297
|
+
<span style={{ display: 'inline-flex', color: 'hsl(var(--primary))', flexShrink: 0 }}>
|
|
298
|
+
{icons?.check}
|
|
299
|
+
</span>
|
|
300
|
+
)
|
|
301
|
+
)}
|
|
302
|
+
</button>
|
|
303
|
+
);
|
|
304
|
+
})}
|
|
305
|
+
{filtered.length === 0 && (
|
|
306
|
+
<div
|
|
307
|
+
style={{
|
|
308
|
+
padding: '18px 0',
|
|
309
|
+
textAlign: 'center',
|
|
310
|
+
fontSize: 12,
|
|
311
|
+
color: 'hsl(var(--ink-4))',
|
|
312
|
+
}}
|
|
313
|
+
>
|
|
314
|
+
No matches
|
|
315
|
+
</div>
|
|
316
|
+
)}
|
|
317
|
+
</div>
|
|
318
|
+
{footer && (
|
|
319
|
+
<div style={{ borderTop: '1px solid hsl(var(--rule))', padding: 6 }}>
|
|
320
|
+
{footer}
|
|
321
|
+
</div>
|
|
322
|
+
)}
|
|
323
|
+
</div>
|
|
324
|
+
)}
|
|
325
|
+
</div>
|
|
326
|
+
);
|
|
327
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Close-on-outside-click hook. Returns a ref to attach to the element
|
|
3
|
+
* whose interior clicks should NOT trigger `onClose`.
|
|
4
|
+
*/
|
|
5
|
+
import { useEffect, useRef, type RefObject } from 'react';
|
|
6
|
+
|
|
7
|
+
export function useClickOutside<T extends HTMLElement>(
|
|
8
|
+
onClose: () => void,
|
|
9
|
+
): RefObject<T> {
|
|
10
|
+
const ref = useRef<T>(null);
|
|
11
|
+
useEffect(() => {
|
|
12
|
+
function handle(event: MouseEvent): void {
|
|
13
|
+
if (ref.current && !ref.current.contains(event.target as Node)) {
|
|
14
|
+
onClose();
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
document.addEventListener('mousedown', handle);
|
|
18
|
+
return () => document.removeEventListener('mousedown', handle);
|
|
19
|
+
}, [onClose]);
|
|
20
|
+
return ref;
|
|
21
|
+
}
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Session context header — agent / model / environment / audience meta
|
|
3
|
+
* row with the token meter pinned to the right. Icons are host-supplied
|
|
4
|
+
* (`ReactNode`) so the kit stays icon-agnostic.
|
|
5
|
+
*/
|
|
6
|
+
import { TokenMeter, type TokenCounts } from './TokenMeter';
|
|
7
|
+
import { SessionSkin } from '../lib/enums';
|
|
8
|
+
|
|
9
|
+
import type { CSSProperties, ReactElement, ReactNode } from 'react';
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
export interface ContextHeaderSession {
|
|
13
|
+
readonly agent: string;
|
|
14
|
+
readonly model: string;
|
|
15
|
+
readonly environment: string;
|
|
16
|
+
readonly audience: string;
|
|
17
|
+
readonly tokens: TokenCounts;
|
|
18
|
+
readonly cost: number;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/** Host-supplied icon nodes for each meta row. All optional. */
|
|
22
|
+
export interface ContextHeaderIcons {
|
|
23
|
+
readonly agent?: ReactNode;
|
|
24
|
+
readonly model?: ReactNode;
|
|
25
|
+
readonly environment?: ReactNode;
|
|
26
|
+
readonly audience?: ReactNode;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export interface ContextHeaderProps {
|
|
30
|
+
readonly session: ContextHeaderSession;
|
|
31
|
+
readonly icons?: ContextHeaderIcons;
|
|
32
|
+
readonly skin?: SessionSkin;
|
|
33
|
+
/**
|
|
34
|
+
* Audience value treated as the default (un-flagged) audience. When the
|
|
35
|
+
* session's audience differs, the audience meta is shown with a warn tone.
|
|
36
|
+
*/
|
|
37
|
+
readonly defaultAudience?: string;
|
|
38
|
+
/**
|
|
39
|
+
* Environment value treated as production (shown with a warn tone).
|
|
40
|
+
*/
|
|
41
|
+
readonly productionEnvironment?: string;
|
|
42
|
+
readonly style?: CSSProperties;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
function MetaDot(): ReactElement {
|
|
46
|
+
return (
|
|
47
|
+
<span
|
|
48
|
+
style={{
|
|
49
|
+
width: 3,
|
|
50
|
+
height: 3,
|
|
51
|
+
borderRadius: '50%',
|
|
52
|
+
background: 'hsl(var(--ink-4) / 0.6)',
|
|
53
|
+
}}
|
|
54
|
+
/>
|
|
55
|
+
);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
type MetaTone = 'default' | 'strong' | 'warn';
|
|
59
|
+
|
|
60
|
+
function Meta({
|
|
61
|
+
icon,
|
|
62
|
+
label,
|
|
63
|
+
mono,
|
|
64
|
+
tone = 'default',
|
|
65
|
+
}: {
|
|
66
|
+
readonly icon?: ReactNode;
|
|
67
|
+
readonly label: string;
|
|
68
|
+
readonly mono?: boolean;
|
|
69
|
+
readonly tone?: MetaTone;
|
|
70
|
+
}): ReactElement {
|
|
71
|
+
const color =
|
|
72
|
+
tone === 'warn'
|
|
73
|
+
? 'hsl(var(--warning))'
|
|
74
|
+
: tone === 'strong'
|
|
75
|
+
? 'hsl(var(--ink))'
|
|
76
|
+
: 'hsl(var(--ink-3))';
|
|
77
|
+
return (
|
|
78
|
+
<span
|
|
79
|
+
style={{
|
|
80
|
+
display: 'inline-flex',
|
|
81
|
+
alignItems: 'center',
|
|
82
|
+
gap: 5,
|
|
83
|
+
color,
|
|
84
|
+
fontSize: 12,
|
|
85
|
+
fontWeight: tone === 'strong' ? 600 : 450,
|
|
86
|
+
whiteSpace: 'nowrap',
|
|
87
|
+
}}
|
|
88
|
+
>
|
|
89
|
+
{icon && (
|
|
90
|
+
<span style={{ display: 'inline-flex', opacity: 0.8 }}>{icon}</span>
|
|
91
|
+
)}
|
|
92
|
+
<span
|
|
93
|
+
className={mono ? 'mono' : ''}
|
|
94
|
+
style={{ fontSize: mono ? 11.5 : 12, whiteSpace: 'nowrap' }}
|
|
95
|
+
>
|
|
96
|
+
{label}
|
|
97
|
+
</span>
|
|
98
|
+
</span>
|
|
99
|
+
);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export function ContextHeader({
|
|
103
|
+
session,
|
|
104
|
+
icons,
|
|
105
|
+
skin = SessionSkin.Minimal,
|
|
106
|
+
defaultAudience = 'internal-org',
|
|
107
|
+
productionEnvironment = 'production',
|
|
108
|
+
style,
|
|
109
|
+
}: ContextHeaderProps): ReactElement {
|
|
110
|
+
const showAudience = session.audience !== defaultAudience;
|
|
111
|
+
return (
|
|
112
|
+
<div
|
|
113
|
+
style={{
|
|
114
|
+
display: 'flex',
|
|
115
|
+
alignItems: 'center',
|
|
116
|
+
gap: 10,
|
|
117
|
+
flexWrap: 'wrap',
|
|
118
|
+
padding: '8px 14px',
|
|
119
|
+
borderBottom: '1px solid hsl(var(--rule))',
|
|
120
|
+
background:
|
|
121
|
+
skin === SessionSkin.Functional
|
|
122
|
+
? 'hsl(var(--paper-elev))'
|
|
123
|
+
: 'hsl(var(--paper) / 0.6)',
|
|
124
|
+
backdropFilter: 'blur(6px)',
|
|
125
|
+
...style,
|
|
126
|
+
}}
|
|
127
|
+
>
|
|
128
|
+
<Meta icon={icons?.agent} label={session.agent} tone="strong" />
|
|
129
|
+
<MetaDot />
|
|
130
|
+
<Meta icon={icons?.model} label={session.model} mono />
|
|
131
|
+
<MetaDot />
|
|
132
|
+
<Meta
|
|
133
|
+
icon={icons?.environment}
|
|
134
|
+
label={session.environment}
|
|
135
|
+
mono
|
|
136
|
+
tone={session.environment === productionEnvironment ? 'warn' : 'default'}
|
|
137
|
+
/>
|
|
138
|
+
{showAudience && (
|
|
139
|
+
<>
|
|
140
|
+
<MetaDot />
|
|
141
|
+
<Meta icon={icons?.audience} label={session.audience} mono tone="warn" />
|
|
142
|
+
</>
|
|
143
|
+
)}
|
|
144
|
+
<div style={{ flex: 1 }} />
|
|
145
|
+
<TokenMeter tokens={session.tokens} cost={session.cost} skin={skin} />
|
|
146
|
+
</div>
|
|
147
|
+
);
|
|
148
|
+
}
|