@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.
Files changed (197) hide show
  1. package/dist/lib/biome-host/agent-validation.d.ts +22 -0
  2. package/dist/lib/biome-host/agent-validation.d.ts.map +1 -0
  3. package/dist/lib/biome-host/agent-validation.js +127 -0
  4. package/dist/lib/biome-host/agent-validation.js.map +1 -0
  5. package/dist/lib/biome-host/create-biome-orval-config.d.ts +14 -0
  6. package/dist/lib/biome-host/create-biome-orval-config.d.ts.map +1 -0
  7. package/dist/lib/biome-host/create-biome-orval-config.js +22 -0
  8. package/dist/lib/biome-host/create-biome-orval-config.js.map +1 -0
  9. package/dist/lib/biome-host/frontend-biome.d.ts +3 -1
  10. package/dist/lib/biome-host/frontend-biome.d.ts.map +1 -1
  11. package/dist/lib/biome-host/host-bridge.d.ts +2 -0
  12. package/dist/lib/biome-host/host-bridge.d.ts.map +1 -1
  13. package/dist/lib/biome-host/host-bridge.js.map +1 -1
  14. package/dist/lib/biome-host/host-sources.d.ts +2 -0
  15. package/dist/lib/biome-host/host-sources.d.ts.map +1 -1
  16. package/dist/lib/biome-host/index.d.ts +2 -1
  17. package/dist/lib/biome-host/index.d.ts.map +1 -1
  18. package/dist/lib/biome-host/index.js +2 -1
  19. package/dist/lib/biome-host/index.js.map +1 -1
  20. package/dist/registry/index.d.ts +1 -1
  21. package/dist/registry/index.d.ts.map +1 -1
  22. package/dist/registry/index.js +1 -1
  23. package/dist/registry/index.js.map +1 -1
  24. package/dist/registry/lib/agent-validation-host.d.ts +3 -0
  25. package/dist/registry/lib/agent-validation-host.d.ts.map +1 -0
  26. package/dist/registry/lib/agent-validation-host.js +10 -0
  27. package/dist/registry/lib/agent-validation-host.js.map +1 -0
  28. package/dist/session-kit/approvals/ApprovalButton.d.ts +14 -0
  29. package/dist/session-kit/approvals/ApprovalButton.d.ts.map +1 -0
  30. package/dist/session-kit/approvals/ApprovalButton.js +45 -0
  31. package/dist/session-kit/approvals/ApprovalButton.js.map +1 -0
  32. package/dist/session-kit/approvals/ApprovalCard.d.ts +12 -0
  33. package/dist/session-kit/approvals/ApprovalCard.d.ts.map +1 -0
  34. package/dist/session-kit/approvals/ApprovalCard.js +117 -0
  35. package/dist/session-kit/approvals/ApprovalCard.js.map +1 -0
  36. package/dist/session-kit/approvals/ApprovalsCenter.d.ts +11 -0
  37. package/dist/session-kit/approvals/ApprovalsCenter.d.ts.map +1 -0
  38. package/dist/session-kit/approvals/ApprovalsCenter.js +127 -0
  39. package/dist/session-kit/approvals/ApprovalsCenter.js.map +1 -0
  40. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.d.ts +12 -0
  41. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.d.ts.map +1 -0
  42. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.js +36 -0
  43. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.js.map +1 -0
  44. package/dist/session-kit/approvals/Obligation.d.ts +15 -0
  45. package/dist/session-kit/approvals/Obligation.d.ts.map +1 -0
  46. package/dist/session-kit/approvals/Obligation.js +42 -0
  47. package/dist/session-kit/approvals/Obligation.js.map +1 -0
  48. package/dist/session-kit/approvals/ScopedBody.d.ts +9 -0
  49. package/dist/session-kit/approvals/ScopedBody.d.ts.map +1 -0
  50. package/dist/session-kit/approvals/ScopedBody.js +145 -0
  51. package/dist/session-kit/approvals/ScopedBody.js.map +1 -0
  52. package/dist/session-kit/approvals/approval-icons.d.ts +15 -0
  53. package/dist/session-kit/approvals/approval-icons.d.ts.map +1 -0
  54. package/dist/session-kit/approvals/approval-icons.js +3 -0
  55. package/dist/session-kit/approvals/approval-icons.js.map +1 -0
  56. package/dist/session-kit/approvals/approval-model.d.ts +83 -0
  57. package/dist/session-kit/approvals/approval-model.d.ts.map +1 -0
  58. package/dist/session-kit/approvals/approval-model.js +25 -0
  59. package/dist/session-kit/approvals/approval-model.js.map +1 -0
  60. package/dist/session-kit/approvals/index.d.ts +12 -0
  61. package/dist/session-kit/approvals/index.d.ts.map +1 -0
  62. package/dist/session-kit/approvals/index.js +28 -0
  63. package/dist/session-kit/approvals/index.js.map +1 -0
  64. package/dist/session-kit/approvals/obligation-display.d.ts +17 -0
  65. package/dist/session-kit/approvals/obligation-display.d.ts.map +1 -0
  66. package/dist/session-kit/approvals/obligation-display.js +58 -0
  67. package/dist/session-kit/approvals/obligation-display.js.map +1 -0
  68. package/dist/session-kit/approvals/risk-accent.d.ts +8 -0
  69. package/dist/session-kit/approvals/risk-accent.d.ts.map +1 -0
  70. package/dist/session-kit/approvals/risk-accent.js +28 -0
  71. package/dist/session-kit/approvals/risk-accent.js.map +1 -0
  72. package/dist/session-kit/approvals/scope-icons.d.ts +12 -0
  73. package/dist/session-kit/approvals/scope-icons.d.ts.map +1 -0
  74. package/dist/session-kit/approvals/scope-icons.js +14 -0
  75. package/dist/session-kit/approvals/scope-icons.js.map +1 -0
  76. package/dist/session-kit/combobox/Combobox.d.ts +46 -0
  77. package/dist/session-kit/combobox/Combobox.d.ts.map +1 -0
  78. package/dist/session-kit/combobox/Combobox.js +113 -0
  79. package/dist/session-kit/combobox/Combobox.js.map +1 -0
  80. package/dist/session-kit/combobox/use-click-outside.d.ts +3 -0
  81. package/dist/session-kit/combobox/use-click-outside.d.ts.map +1 -0
  82. package/dist/session-kit/combobox/use-click-outside.js +18 -0
  83. package/dist/session-kit/combobox/use-click-outside.js.map +1 -0
  84. package/dist/session-kit/display/ContextHeader.d.ts +27 -0
  85. package/dist/session-kit/display/ContextHeader.d.ts.map +1 -0
  86. package/dist/session-kit/display/ContextHeader.js +47 -0
  87. package/dist/session-kit/display/ContextHeader.js.map +1 -0
  88. package/dist/session-kit/display/FileDiffCard.d.ts +18 -0
  89. package/dist/session-kit/display/FileDiffCard.d.ts.map +1 -0
  90. package/dist/session-kit/display/FileDiffCard.js +58 -0
  91. package/dist/session-kit/display/FileDiffCard.js.map +1 -0
  92. package/dist/session-kit/display/MD.d.ts +7 -0
  93. package/dist/session-kit/display/MD.d.ts.map +1 -0
  94. package/dist/session-kit/display/MD.js +89 -0
  95. package/dist/session-kit/display/MD.js.map +1 -0
  96. package/dist/session-kit/display/MessageTurn.d.ts +21 -0
  97. package/dist/session-kit/display/MessageTurn.d.ts.map +1 -0
  98. package/dist/session-kit/display/MessageTurn.js +62 -0
  99. package/dist/session-kit/display/MessageTurn.js.map +1 -0
  100. package/dist/session-kit/display/ThinkingPanel.d.ts +12 -0
  101. package/dist/session-kit/display/ThinkingPanel.d.ts.map +1 -0
  102. package/dist/session-kit/display/ThinkingPanel.js +30 -0
  103. package/dist/session-kit/display/ThinkingPanel.js.map +1 -0
  104. package/dist/session-kit/display/TodoChecklist.d.ts +17 -0
  105. package/dist/session-kit/display/TodoChecklist.d.ts.map +1 -0
  106. package/dist/session-kit/display/TodoChecklist.js +50 -0
  107. package/dist/session-kit/display/TodoChecklist.js.map +1 -0
  108. package/dist/session-kit/display/TokenMeter.d.ts +15 -0
  109. package/dist/session-kit/display/TokenMeter.d.ts.map +1 -0
  110. package/dist/session-kit/display/TokenMeter.js +35 -0
  111. package/dist/session-kit/display/TokenMeter.js.map +1 -0
  112. package/dist/session-kit/display/ToolStrip.d.ts +31 -0
  113. package/dist/session-kit/display/ToolStrip.d.ts.map +1 -0
  114. package/dist/session-kit/display/ToolStrip.js +99 -0
  115. package/dist/session-kit/display/ToolStrip.js.map +1 -0
  116. package/dist/session-kit/display/TypingDots.d.ts +3 -0
  117. package/dist/session-kit/display/TypingDots.d.ts.map +1 -0
  118. package/dist/session-kit/display/TypingDots.js +14 -0
  119. package/dist/session-kit/display/TypingDots.js.map +1 -0
  120. package/dist/session-kit/index.d.ts +21 -0
  121. package/dist/session-kit/index.d.ts.map +1 -0
  122. package/dist/session-kit/index.js +37 -0
  123. package/dist/session-kit/index.js.map +1 -0
  124. package/dist/session-kit/lib/enums.d.ts +34 -0
  125. package/dist/session-kit/lib/enums.d.ts.map +1 -0
  126. package/dist/session-kit/lib/enums.js +44 -0
  127. package/dist/session-kit/lib/enums.js.map +1 -0
  128. package/dist/session-kit/lib/portal-accent.d.ts +3 -0
  129. package/dist/session-kit/lib/portal-accent.d.ts.map +1 -0
  130. package/dist/session-kit/lib/portal-accent.js +9 -0
  131. package/dist/session-kit/lib/portal-accent.js.map +1 -0
  132. package/dist/session-kit/lib/status-dot.d.ts +10 -0
  133. package/dist/session-kit/lib/status-dot.d.ts.map +1 -0
  134. package/dist/session-kit/lib/status-dot.js +43 -0
  135. package/dist/session-kit/lib/status-dot.js.map +1 -0
  136. package/dist/session-kit/primitives/Avatar.d.ts +10 -0
  137. package/dist/session-kit/primitives/Avatar.d.ts.map +1 -0
  138. package/dist/session-kit/primitives/Avatar.js +21 -0
  139. package/dist/session-kit/primitives/Avatar.js.map +1 -0
  140. package/dist/session-kit/primitives/PortalGlyph.d.ts +12 -0
  141. package/dist/session-kit/primitives/PortalGlyph.d.ts.map +1 -0
  142. package/dist/session-kit/primitives/PortalGlyph.js +21 -0
  143. package/dist/session-kit/primitives/PortalGlyph.js.map +1 -0
  144. package/dist/session-kit/primitives/RiskPill.d.ts +12 -0
  145. package/dist/session-kit/primitives/RiskPill.d.ts.map +1 -0
  146. package/dist/session-kit/primitives/RiskPill.js +53 -0
  147. package/dist/session-kit/primitives/RiskPill.js.map +1 -0
  148. package/dist/session-kit/primitives/ScopeDot.d.ts +9 -0
  149. package/dist/session-kit/primitives/ScopeDot.d.ts.map +1 -0
  150. package/dist/session-kit/primitives/ScopeDot.js +23 -0
  151. package/dist/session-kit/primitives/ScopeDot.js.map +1 -0
  152. package/dist/session-kit/primitives/Segmented.d.ts +16 -0
  153. package/dist/session-kit/primitives/Segmented.d.ts.map +1 -0
  154. package/dist/session-kit/primitives/Segmented.js +31 -0
  155. package/dist/session-kit/primitives/Segmented.js.map +1 -0
  156. package/package.json +2 -2
  157. package/src/lib/biome-host/create-biome-orval-config.ts +76 -0
  158. package/src/lib/biome-host/frontend-biome.ts +22 -2
  159. package/src/lib/biome-host/host-bridge.ts +22 -0
  160. package/src/lib/biome-host/host-sources.ts +13 -0
  161. package/src/lib/biome-host/index.ts +2 -1
  162. package/src/registry/index.ts +1 -1
  163. package/src/registry/lib/extension-points.ts +1 -1
  164. package/src/session-kit/approvals/ApprovalButton.tsx +89 -0
  165. package/src/session-kit/approvals/ApprovalCard.tsx +336 -0
  166. package/src/session-kit/approvals/ApprovalsCenter.tsx +327 -0
  167. package/src/session-kit/approvals/CapabilityApprovalStickyBar.tsx +118 -0
  168. package/src/session-kit/approvals/Obligation.tsx +111 -0
  169. package/src/session-kit/approvals/ScopedBody.tsx +392 -0
  170. package/src/session-kit/approvals/approval-icons.ts +31 -0
  171. package/src/session-kit/approvals/approval-model.ts +205 -0
  172. package/src/session-kit/approvals/index.ts +22 -0
  173. package/src/session-kit/approvals/obligation-display.ts +100 -0
  174. package/src/session-kit/approvals/risk-accent.ts +47 -0
  175. package/src/session-kit/approvals/scope-icons.ts +19 -0
  176. package/src/session-kit/combobox/Combobox.tsx +327 -0
  177. package/src/session-kit/combobox/use-click-outside.ts +21 -0
  178. package/src/session-kit/display/ContextHeader.tsx +148 -0
  179. package/src/session-kit/display/FileDiffCard.tsx +140 -0
  180. package/src/session-kit/display/MD.tsx +153 -0
  181. package/src/session-kit/display/MessageTurn.tsx +157 -0
  182. package/src/session-kit/display/ThinkingPanel.tsx +78 -0
  183. package/src/session-kit/display/TodoChecklist.tsx +120 -0
  184. package/src/session-kit/display/TokenMeter.tsx +89 -0
  185. package/src/session-kit/display/ToolStrip.tsx +278 -0
  186. package/src/session-kit/display/TypingDots.tsx +24 -0
  187. package/src/session-kit/index.ts +44 -0
  188. package/src/session-kit/lib/enums.ts +66 -0
  189. package/src/session-kit/lib/portal-accent.ts +30 -0
  190. package/src/session-kit/lib/status-dot.ts +68 -0
  191. package/src/session-kit/primitives/Avatar.tsx +44 -0
  192. package/src/session-kit/primitives/PortalGlyph.tsx +51 -0
  193. package/src/session-kit/primitives/RiskPill.tsx +95 -0
  194. package/src/session-kit/primitives/ScopeDot.tsx +47 -0
  195. package/src/session-kit/primitives/Segmented.tsx +71 -0
  196. /package/src/lib/biome-host/{composition-validation.ts → agent-validation.ts} +0 -0
  197. /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
+ }