@xemahq/ui-kernel 0.1.6 → 0.1.7

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 (178) hide show
  1. package/dist/lib/biome-host/create-biome-orval-config.d.ts +14 -0
  2. package/dist/lib/biome-host/create-biome-orval-config.d.ts.map +1 -0
  3. package/dist/lib/biome-host/create-biome-orval-config.js +22 -0
  4. package/dist/lib/biome-host/create-biome-orval-config.js.map +1 -0
  5. package/dist/lib/biome-host/host-bridge.d.ts +2 -0
  6. package/dist/lib/biome-host/host-bridge.d.ts.map +1 -1
  7. package/dist/lib/biome-host/host-bridge.js.map +1 -1
  8. package/dist/lib/biome-host/host-sources.d.ts +2 -0
  9. package/dist/lib/biome-host/host-sources.d.ts.map +1 -1
  10. package/dist/lib/biome-host/index.d.ts +1 -0
  11. package/dist/lib/biome-host/index.d.ts.map +1 -1
  12. package/dist/lib/biome-host/index.js +1 -0
  13. package/dist/lib/biome-host/index.js.map +1 -1
  14. package/dist/session-kit/approvals/ApprovalButton.d.ts +14 -0
  15. package/dist/session-kit/approvals/ApprovalButton.d.ts.map +1 -0
  16. package/dist/session-kit/approvals/ApprovalButton.js +45 -0
  17. package/dist/session-kit/approvals/ApprovalButton.js.map +1 -0
  18. package/dist/session-kit/approvals/ApprovalCard.d.ts +12 -0
  19. package/dist/session-kit/approvals/ApprovalCard.d.ts.map +1 -0
  20. package/dist/session-kit/approvals/ApprovalCard.js +117 -0
  21. package/dist/session-kit/approvals/ApprovalCard.js.map +1 -0
  22. package/dist/session-kit/approvals/ApprovalsCenter.d.ts +11 -0
  23. package/dist/session-kit/approvals/ApprovalsCenter.d.ts.map +1 -0
  24. package/dist/session-kit/approvals/ApprovalsCenter.js +127 -0
  25. package/dist/session-kit/approvals/ApprovalsCenter.js.map +1 -0
  26. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.d.ts +12 -0
  27. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.d.ts.map +1 -0
  28. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.js +36 -0
  29. package/dist/session-kit/approvals/CapabilityApprovalStickyBar.js.map +1 -0
  30. package/dist/session-kit/approvals/Obligation.d.ts +15 -0
  31. package/dist/session-kit/approvals/Obligation.d.ts.map +1 -0
  32. package/dist/session-kit/approvals/Obligation.js +42 -0
  33. package/dist/session-kit/approvals/Obligation.js.map +1 -0
  34. package/dist/session-kit/approvals/ScopedBody.d.ts +9 -0
  35. package/dist/session-kit/approvals/ScopedBody.d.ts.map +1 -0
  36. package/dist/session-kit/approvals/ScopedBody.js +145 -0
  37. package/dist/session-kit/approvals/ScopedBody.js.map +1 -0
  38. package/dist/session-kit/approvals/approval-icons.d.ts +15 -0
  39. package/dist/session-kit/approvals/approval-icons.d.ts.map +1 -0
  40. package/dist/session-kit/approvals/approval-icons.js +3 -0
  41. package/dist/session-kit/approvals/approval-icons.js.map +1 -0
  42. package/dist/session-kit/approvals/approval-model.d.ts +83 -0
  43. package/dist/session-kit/approvals/approval-model.d.ts.map +1 -0
  44. package/dist/session-kit/approvals/approval-model.js +25 -0
  45. package/dist/session-kit/approvals/approval-model.js.map +1 -0
  46. package/dist/session-kit/approvals/index.d.ts +12 -0
  47. package/dist/session-kit/approvals/index.d.ts.map +1 -0
  48. package/dist/session-kit/approvals/index.js +28 -0
  49. package/dist/session-kit/approvals/index.js.map +1 -0
  50. package/dist/session-kit/approvals/obligation-display.d.ts +17 -0
  51. package/dist/session-kit/approvals/obligation-display.d.ts.map +1 -0
  52. package/dist/session-kit/approvals/obligation-display.js +58 -0
  53. package/dist/session-kit/approvals/obligation-display.js.map +1 -0
  54. package/dist/session-kit/approvals/risk-accent.d.ts +8 -0
  55. package/dist/session-kit/approvals/risk-accent.d.ts.map +1 -0
  56. package/dist/session-kit/approvals/risk-accent.js +28 -0
  57. package/dist/session-kit/approvals/risk-accent.js.map +1 -0
  58. package/dist/session-kit/approvals/scope-icons.d.ts +12 -0
  59. package/dist/session-kit/approvals/scope-icons.d.ts.map +1 -0
  60. package/dist/session-kit/approvals/scope-icons.js +14 -0
  61. package/dist/session-kit/approvals/scope-icons.js.map +1 -0
  62. package/dist/session-kit/combobox/Combobox.d.ts +46 -0
  63. package/dist/session-kit/combobox/Combobox.d.ts.map +1 -0
  64. package/dist/session-kit/combobox/Combobox.js +113 -0
  65. package/dist/session-kit/combobox/Combobox.js.map +1 -0
  66. package/dist/session-kit/combobox/use-click-outside.d.ts +3 -0
  67. package/dist/session-kit/combobox/use-click-outside.d.ts.map +1 -0
  68. package/dist/session-kit/combobox/use-click-outside.js +18 -0
  69. package/dist/session-kit/combobox/use-click-outside.js.map +1 -0
  70. package/dist/session-kit/display/ContextHeader.d.ts +27 -0
  71. package/dist/session-kit/display/ContextHeader.d.ts.map +1 -0
  72. package/dist/session-kit/display/ContextHeader.js +47 -0
  73. package/dist/session-kit/display/ContextHeader.js.map +1 -0
  74. package/dist/session-kit/display/FileDiffCard.d.ts +18 -0
  75. package/dist/session-kit/display/FileDiffCard.d.ts.map +1 -0
  76. package/dist/session-kit/display/FileDiffCard.js +58 -0
  77. package/dist/session-kit/display/FileDiffCard.js.map +1 -0
  78. package/dist/session-kit/display/MD.d.ts +7 -0
  79. package/dist/session-kit/display/MD.d.ts.map +1 -0
  80. package/dist/session-kit/display/MD.js +89 -0
  81. package/dist/session-kit/display/MD.js.map +1 -0
  82. package/dist/session-kit/display/MessageTurn.d.ts +21 -0
  83. package/dist/session-kit/display/MessageTurn.d.ts.map +1 -0
  84. package/dist/session-kit/display/MessageTurn.js +62 -0
  85. package/dist/session-kit/display/MessageTurn.js.map +1 -0
  86. package/dist/session-kit/display/ThinkingPanel.d.ts +12 -0
  87. package/dist/session-kit/display/ThinkingPanel.d.ts.map +1 -0
  88. package/dist/session-kit/display/ThinkingPanel.js +30 -0
  89. package/dist/session-kit/display/ThinkingPanel.js.map +1 -0
  90. package/dist/session-kit/display/TodoChecklist.d.ts +17 -0
  91. package/dist/session-kit/display/TodoChecklist.d.ts.map +1 -0
  92. package/dist/session-kit/display/TodoChecklist.js +50 -0
  93. package/dist/session-kit/display/TodoChecklist.js.map +1 -0
  94. package/dist/session-kit/display/TokenMeter.d.ts +15 -0
  95. package/dist/session-kit/display/TokenMeter.d.ts.map +1 -0
  96. package/dist/session-kit/display/TokenMeter.js +35 -0
  97. package/dist/session-kit/display/TokenMeter.js.map +1 -0
  98. package/dist/session-kit/display/ToolStrip.d.ts +31 -0
  99. package/dist/session-kit/display/ToolStrip.d.ts.map +1 -0
  100. package/dist/session-kit/display/ToolStrip.js +99 -0
  101. package/dist/session-kit/display/ToolStrip.js.map +1 -0
  102. package/dist/session-kit/display/TypingDots.d.ts +3 -0
  103. package/dist/session-kit/display/TypingDots.d.ts.map +1 -0
  104. package/dist/session-kit/display/TypingDots.js +14 -0
  105. package/dist/session-kit/display/TypingDots.js.map +1 -0
  106. package/dist/session-kit/index.d.ts +21 -0
  107. package/dist/session-kit/index.d.ts.map +1 -0
  108. package/dist/session-kit/index.js +37 -0
  109. package/dist/session-kit/index.js.map +1 -0
  110. package/dist/session-kit/lib/enums.d.ts +34 -0
  111. package/dist/session-kit/lib/enums.d.ts.map +1 -0
  112. package/dist/session-kit/lib/enums.js +44 -0
  113. package/dist/session-kit/lib/enums.js.map +1 -0
  114. package/dist/session-kit/lib/portal-accent.d.ts +3 -0
  115. package/dist/session-kit/lib/portal-accent.d.ts.map +1 -0
  116. package/dist/session-kit/lib/portal-accent.js +9 -0
  117. package/dist/session-kit/lib/portal-accent.js.map +1 -0
  118. package/dist/session-kit/lib/status-dot.d.ts +10 -0
  119. package/dist/session-kit/lib/status-dot.d.ts.map +1 -0
  120. package/dist/session-kit/lib/status-dot.js +43 -0
  121. package/dist/session-kit/lib/status-dot.js.map +1 -0
  122. package/dist/session-kit/primitives/Avatar.d.ts +10 -0
  123. package/dist/session-kit/primitives/Avatar.d.ts.map +1 -0
  124. package/dist/session-kit/primitives/Avatar.js +21 -0
  125. package/dist/session-kit/primitives/Avatar.js.map +1 -0
  126. package/dist/session-kit/primitives/PortalGlyph.d.ts +12 -0
  127. package/dist/session-kit/primitives/PortalGlyph.d.ts.map +1 -0
  128. package/dist/session-kit/primitives/PortalGlyph.js +21 -0
  129. package/dist/session-kit/primitives/PortalGlyph.js.map +1 -0
  130. package/dist/session-kit/primitives/RiskPill.d.ts +12 -0
  131. package/dist/session-kit/primitives/RiskPill.d.ts.map +1 -0
  132. package/dist/session-kit/primitives/RiskPill.js +53 -0
  133. package/dist/session-kit/primitives/RiskPill.js.map +1 -0
  134. package/dist/session-kit/primitives/ScopeDot.d.ts +9 -0
  135. package/dist/session-kit/primitives/ScopeDot.d.ts.map +1 -0
  136. package/dist/session-kit/primitives/ScopeDot.js +23 -0
  137. package/dist/session-kit/primitives/ScopeDot.js.map +1 -0
  138. package/dist/session-kit/primitives/Segmented.d.ts +16 -0
  139. package/dist/session-kit/primitives/Segmented.d.ts.map +1 -0
  140. package/dist/session-kit/primitives/Segmented.js +31 -0
  141. package/dist/session-kit/primitives/Segmented.js.map +1 -0
  142. package/package.json +2 -2
  143. package/src/lib/biome-host/create-biome-orval-config.ts +76 -0
  144. package/src/lib/biome-host/host-bridge.ts +22 -0
  145. package/src/lib/biome-host/host-sources.ts +13 -0
  146. package/src/lib/biome-host/index.ts +1 -0
  147. package/src/session-kit/approvals/ApprovalButton.tsx +89 -0
  148. package/src/session-kit/approvals/ApprovalCard.tsx +336 -0
  149. package/src/session-kit/approvals/ApprovalsCenter.tsx +327 -0
  150. package/src/session-kit/approvals/CapabilityApprovalStickyBar.tsx +118 -0
  151. package/src/session-kit/approvals/Obligation.tsx +111 -0
  152. package/src/session-kit/approvals/ScopedBody.tsx +392 -0
  153. package/src/session-kit/approvals/approval-icons.ts +31 -0
  154. package/src/session-kit/approvals/approval-model.ts +205 -0
  155. package/src/session-kit/approvals/index.ts +22 -0
  156. package/src/session-kit/approvals/obligation-display.ts +100 -0
  157. package/src/session-kit/approvals/risk-accent.ts +47 -0
  158. package/src/session-kit/approvals/scope-icons.ts +19 -0
  159. package/src/session-kit/combobox/Combobox.tsx +327 -0
  160. package/src/session-kit/combobox/use-click-outside.ts +21 -0
  161. package/src/session-kit/display/ContextHeader.tsx +148 -0
  162. package/src/session-kit/display/FileDiffCard.tsx +140 -0
  163. package/src/session-kit/display/MD.tsx +153 -0
  164. package/src/session-kit/display/MessageTurn.tsx +157 -0
  165. package/src/session-kit/display/ThinkingPanel.tsx +78 -0
  166. package/src/session-kit/display/TodoChecklist.tsx +120 -0
  167. package/src/session-kit/display/TokenMeter.tsx +89 -0
  168. package/src/session-kit/display/ToolStrip.tsx +278 -0
  169. package/src/session-kit/display/TypingDots.tsx +24 -0
  170. package/src/session-kit/index.ts +44 -0
  171. package/src/session-kit/lib/enums.ts +66 -0
  172. package/src/session-kit/lib/portal-accent.ts +30 -0
  173. package/src/session-kit/lib/status-dot.ts +68 -0
  174. package/src/session-kit/primitives/Avatar.tsx +44 -0
  175. package/src/session-kit/primitives/PortalGlyph.tsx +51 -0
  176. package/src/session-kit/primitives/RiskPill.tsx +95 -0
  177. package/src/session-kit/primitives/ScopeDot.tsx +47 -0
  178. package/src/session-kit/primitives/Segmented.tsx +71 -0
@@ -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
+ }
@@ -0,0 +1,140 @@
1
+ /**
2
+ * Collapsible file-diff card — path + add/del counts in the header, an
3
+ * optional expandable unified-diff body. Disclosure + file icons are
4
+ * host-supplied.
5
+ */
6
+ import { useState, type ReactElement, type ReactNode } from 'react';
7
+
8
+ import { SessionSkin } from '../lib/enums';
9
+
10
+ /**
11
+ * A single diff line: first character is the marker (`+` add, `-` del,
12
+ * ` ` context), the remainder is the line text.
13
+ */
14
+ export type DiffLine = string;
15
+
16
+ export interface DiffFile {
17
+ readonly path: string;
18
+ readonly add: number;
19
+ readonly del: number;
20
+ readonly hunks?: ReadonlyArray<DiffLine>;
21
+ }
22
+
23
+ export interface FileDiffCardProps {
24
+ readonly file: DiffFile;
25
+ readonly skin?: SessionSkin;
26
+ readonly collapsedIcon?: ReactNode;
27
+ readonly expandedIcon?: ReactNode;
28
+ readonly fileIcon?: ReactNode;
29
+ }
30
+
31
+ const DANGER_VAR = 'var(--destructive, var(--danger))';
32
+ const DANGER = `hsl(${DANGER_VAR})`;
33
+ const DANGER_SOFT = `hsl(${DANGER_VAR} / 0.06)`;
34
+
35
+ export function FileDiffCard({
36
+ file,
37
+ skin = SessionSkin.Minimal,
38
+ collapsedIcon,
39
+ expandedIcon,
40
+ fileIcon,
41
+ }: FileDiffCardProps): ReactElement {
42
+ const [open, setOpen] = useState(false);
43
+ const hasHunks = Boolean(file.hunks && file.hunks.length > 0);
44
+ return (
45
+ <div
46
+ style={{
47
+ marginTop: 9,
48
+ border: '1px solid hsl(var(--rule))',
49
+ borderRadius: 9,
50
+ overflow: 'hidden',
51
+ background: 'hsl(var(--paper-elev))',
52
+ }}
53
+ >
54
+ <button
55
+ type="button"
56
+ onClick={() => hasHunks && setOpen((v) => !v)}
57
+ style={{
58
+ display: 'flex',
59
+ alignItems: 'center',
60
+ gap: 9,
61
+ width: '100%',
62
+ padding: '8px 11px',
63
+ textAlign: 'left',
64
+ cursor: hasHunks ? 'pointer' : 'default',
65
+ background:
66
+ skin === SessionSkin.Functional ? 'hsl(var(--muted))' : 'transparent',
67
+ }}
68
+ >
69
+ {hasHunks && (open ? expandedIcon : collapsedIcon)}
70
+ {fileIcon && (
71
+ <span style={{ display: 'inline-flex', color: 'hsl(var(--warning))' }}>
72
+ {fileIcon}
73
+ </span>
74
+ )}
75
+ <span
76
+ className="mono"
77
+ style={{
78
+ fontSize: 12,
79
+ color: 'hsl(var(--ink))',
80
+ overflow: 'hidden',
81
+ textOverflow: 'ellipsis',
82
+ whiteSpace: 'nowrap',
83
+ }}
84
+ >
85
+ {file.path}
86
+ </span>
87
+ <span style={{ flex: 1 }} />
88
+ <span className="mono" style={{ fontSize: 11, color: 'hsl(var(--success))' }}>
89
+ +{file.add}
90
+ </span>
91
+ <span className="mono" style={{ fontSize: 11, color: DANGER }}>
92
+ −{file.del}
93
+ </span>
94
+ </button>
95
+ {open && file.hunks && (
96
+ <div
97
+ className="mono"
98
+ style={{
99
+ borderTop: '1px solid hsl(var(--rule))',
100
+ fontSize: 11.5,
101
+ lineHeight: 1.6,
102
+ overflowX: 'auto',
103
+ }}
104
+ >
105
+ {file.hunks.map((line, index) => {
106
+ const marker = line[0];
107
+ const text = line.slice(1);
108
+ const bg =
109
+ marker === '+'
110
+ ? 'hsl(var(--success) / 0.07)'
111
+ : marker === '-'
112
+ ? DANGER_SOFT
113
+ : 'transparent';
114
+ const color =
115
+ marker === '+'
116
+ ? 'hsl(var(--success))'
117
+ : marker === '-'
118
+ ? DANGER
119
+ : 'hsl(var(--ink-3))';
120
+ return (
121
+ <div key={index} style={{ display: 'flex', background: bg, padding: '0 11px' }}>
122
+ <span style={{ width: 14, color, flexShrink: 0 }}>
123
+ {marker === ' ' ? '' : marker}
124
+ </span>
125
+ <span
126
+ style={{
127
+ color: marker === ' ' ? 'hsl(var(--ink-3))' : 'hsl(var(--ink-2))',
128
+ whiteSpace: 'pre',
129
+ }}
130
+ >
131
+ {text}
132
+ </span>
133
+ </div>
134
+ );
135
+ })}
136
+ </div>
137
+ )}
138
+ </div>
139
+ );
140
+ }