@xemahq/ui-kernel 0.1.5 → 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.
- 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/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 +1 -0
- package/dist/lib/biome-host/index.d.ts.map +1 -1
- package/dist/lib/biome-host/index.js +1 -0
- package/dist/lib/biome-host/index.js.map +1 -1
- 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 +3 -3
- package/src/lib/biome-host/create-biome-orval-config.ts +76 -0
- 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 +1 -0
- 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
|
@@ -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
|
+
}
|