@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.
- 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 +2 -2
- 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,118 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Sticky CAPABILITY-approval bar (HANDOFF §6) — pinned above the composer
|
|
3
|
+
* so a pending capability approval is always actionable without scrolling
|
|
4
|
+
* back to the inline card.
|
|
5
|
+
*
|
|
6
|
+
* NOTE on naming: this is DISTINCT from the clarification-gate
|
|
7
|
+
* `StickyApprovalBar` (`src/components/clarification/StickyApprovalBar.tsx`),
|
|
8
|
+
* which approves a clarification CONTRACT (attempt counters, reject
|
|
9
|
+
* reasons, override rationale). This bar approves a suspended CAPABILITY
|
|
10
|
+
* CALL (risk tier + biome + Review/Deny/Approve). The two concerns are
|
|
11
|
+
* deliberately separate components so neither regresses the other.
|
|
12
|
+
*
|
|
13
|
+
* Title + RiskPill + biome + Review / Deny / Approve. The bar surface is
|
|
14
|
+
* tinted with the risk accent; critical/high force the risk colour on the
|
|
15
|
+
* approve button.
|
|
16
|
+
*/
|
|
17
|
+
import {
|
|
18
|
+
ApprovalDecision,
|
|
19
|
+
isExplicitApprovalTier,
|
|
20
|
+
type ApprovalRequest,
|
|
21
|
+
} from './approval-model';
|
|
22
|
+
import { ApprovalButton } from './ApprovalButton';
|
|
23
|
+
import { RISK_ACCENT } from './risk-accent';
|
|
24
|
+
import { RiskPill } from '../primitives/RiskPill';
|
|
25
|
+
|
|
26
|
+
import type { ApprovalIcons } from './approval-icons';
|
|
27
|
+
import type { ReactElement } from 'react';
|
|
28
|
+
|
|
29
|
+
export interface CapabilityApprovalStickyBarProps {
|
|
30
|
+
readonly request: ApprovalRequest;
|
|
31
|
+
readonly icons: Pick<ApprovalIcons, 'shield' | 'check'>;
|
|
32
|
+
readonly onResolve: (
|
|
33
|
+
decision: ApprovalDecision,
|
|
34
|
+
request: ApprovalRequest,
|
|
35
|
+
) => void;
|
|
36
|
+
/** "Review" scrolls/opens the inline card; optional. */
|
|
37
|
+
readonly onReview?: () => void;
|
|
38
|
+
/** Portal/brand accent for the (non-critical) approve button. */
|
|
39
|
+
readonly accent?: string;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export function CapabilityApprovalStickyBar({
|
|
43
|
+
request,
|
|
44
|
+
icons,
|
|
45
|
+
onResolve,
|
|
46
|
+
onReview,
|
|
47
|
+
accent = 'hsl(var(--primary))',
|
|
48
|
+
}: CapabilityApprovalStickyBarProps): ReactElement {
|
|
49
|
+
const r = RISK_ACCENT[request.riskTier];
|
|
50
|
+
const explicit = isExplicitApprovalTier(request.riskTier);
|
|
51
|
+
return (
|
|
52
|
+
<div
|
|
53
|
+
style={{
|
|
54
|
+
display: 'flex',
|
|
55
|
+
alignItems: 'center',
|
|
56
|
+
gap: 11,
|
|
57
|
+
padding: '9px 13px',
|
|
58
|
+
borderRadius: 11,
|
|
59
|
+
border: `1px solid ${r.bd}`,
|
|
60
|
+
background: r.bg,
|
|
61
|
+
marginBottom: 9,
|
|
62
|
+
}}
|
|
63
|
+
>
|
|
64
|
+
<span style={{ color: r.fg, flexShrink: 0, display: 'inline-flex' }}>
|
|
65
|
+
{icons.shield(17)}
|
|
66
|
+
</span>
|
|
67
|
+
<div style={{ flex: 1, minWidth: 0 }}>
|
|
68
|
+
<div style={{ display: 'flex', alignItems: 'center', gap: 7 }}>
|
|
69
|
+
<span
|
|
70
|
+
style={{
|
|
71
|
+
fontSize: 13,
|
|
72
|
+
fontWeight: 600,
|
|
73
|
+
color: 'hsl(var(--ink))',
|
|
74
|
+
overflow: 'hidden',
|
|
75
|
+
textOverflow: 'ellipsis',
|
|
76
|
+
whiteSpace: 'nowrap',
|
|
77
|
+
}}
|
|
78
|
+
>
|
|
79
|
+
{request.title}
|
|
80
|
+
</span>
|
|
81
|
+
<RiskPill tier={request.riskTier} size="xs" />
|
|
82
|
+
</div>
|
|
83
|
+
<div
|
|
84
|
+
style={{
|
|
85
|
+
fontSize: 11.5,
|
|
86
|
+
color: 'hsl(var(--ink-3))',
|
|
87
|
+
overflow: 'hidden',
|
|
88
|
+
textOverflow: 'ellipsis',
|
|
89
|
+
whiteSpace: 'nowrap',
|
|
90
|
+
}}
|
|
91
|
+
>
|
|
92
|
+
{request.biomeName} · awaiting your approval
|
|
93
|
+
</div>
|
|
94
|
+
</div>
|
|
95
|
+
{onReview && (
|
|
96
|
+
<ApprovalButton variant="ghost" size="sm" onClick={onReview}>
|
|
97
|
+
Review
|
|
98
|
+
</ApprovalButton>
|
|
99
|
+
)}
|
|
100
|
+
<ApprovalButton
|
|
101
|
+
variant="ghost"
|
|
102
|
+
size="sm"
|
|
103
|
+
onClick={() => onResolve(ApprovalDecision.Denied, request)}
|
|
104
|
+
>
|
|
105
|
+
Deny
|
|
106
|
+
</ApprovalButton>
|
|
107
|
+
<ApprovalButton
|
|
108
|
+
variant="primary"
|
|
109
|
+
size="sm"
|
|
110
|
+
icon={icons.check(13)}
|
|
111
|
+
accent={explicit ? r.fg : accent}
|
|
112
|
+
onClick={() => onResolve(ApprovalDecision.Approved, request)}
|
|
113
|
+
>
|
|
114
|
+
Approve
|
|
115
|
+
</ApprovalButton>
|
|
116
|
+
</div>
|
|
117
|
+
);
|
|
118
|
+
}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Obligation chip + quorum dots (HANDOFF §6).
|
|
3
|
+
*
|
|
4
|
+
* `Obligation` renders one obligation as a chip: host-supplied icon (via
|
|
5
|
+
* `renderIcon` from the stable `ObligationIcon` key) + label + optional
|
|
6
|
+
* pre-formatted mono value. An obligation whose `kind` is outside the
|
|
7
|
+
* known `PolicyObligationKind` map fails SAFE to a visible "unknown"
|
|
8
|
+
* chip (shield glyph + the raw kind) — never a silent drop.
|
|
9
|
+
*
|
|
10
|
+
* `QuorumDots` renders `have/need` approver dots (filled = collected).
|
|
11
|
+
*/
|
|
12
|
+
import { OBLIGATION_DISPLAY, ObligationIcon } from './obligation-display';
|
|
13
|
+
|
|
14
|
+
import type { ApprovalObligation } from './approval-model';
|
|
15
|
+
import type { ReactElement, ReactNode } from 'react';
|
|
16
|
+
|
|
17
|
+
export interface ObligationProps {
|
|
18
|
+
readonly obligation: ApprovalObligation;
|
|
19
|
+
/** Host icon renderer for a stable `ObligationIcon` key (kit is icon-agnostic). */
|
|
20
|
+
readonly renderIcon: (key: ObligationIcon, size: number) => ReactNode;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function Obligation({
|
|
24
|
+
obligation,
|
|
25
|
+
renderIcon,
|
|
26
|
+
}: ObligationProps): ReactElement {
|
|
27
|
+
// Known kinds resolve to a label + icon key; an unknown discriminant
|
|
28
|
+
// (a not-yet-mapped kernel kind) renders a visible fallback rather than
|
|
29
|
+
// disappearing — fail-fast, but on screen.
|
|
30
|
+
const meta = OBLIGATION_DISPLAY[obligation.kind];
|
|
31
|
+
const iconKey = meta?.iconKey ?? ObligationIcon.ShieldCheck;
|
|
32
|
+
const label = meta?.label ?? `unknown: ${obligation.kind}`;
|
|
33
|
+
return (
|
|
34
|
+
<span
|
|
35
|
+
style={{
|
|
36
|
+
display: 'inline-flex',
|
|
37
|
+
alignItems: 'center',
|
|
38
|
+
gap: 5,
|
|
39
|
+
padding: '3px 8px',
|
|
40
|
+
borderRadius: 6,
|
|
41
|
+
whiteSpace: 'nowrap',
|
|
42
|
+
background: 'hsl(var(--muted))',
|
|
43
|
+
border: '1px solid hsl(var(--rule))',
|
|
44
|
+
fontSize: 11.5,
|
|
45
|
+
color: 'hsl(var(--ink-2))',
|
|
46
|
+
}}
|
|
47
|
+
>
|
|
48
|
+
<span style={{ color: 'hsl(var(--ink-3))', display: 'inline-flex' }}>
|
|
49
|
+
{renderIcon(iconKey, 12)}
|
|
50
|
+
</span>
|
|
51
|
+
<span style={{ color: 'hsl(var(--ink-3))' }}>{label}</span>
|
|
52
|
+
{obligation.value && (
|
|
53
|
+
<span
|
|
54
|
+
className="mono"
|
|
55
|
+
style={{ fontSize: 11, color: 'hsl(var(--ink))', fontWeight: 500 }}
|
|
56
|
+
>
|
|
57
|
+
{obligation.value}
|
|
58
|
+
</span>
|
|
59
|
+
)}
|
|
60
|
+
</span>
|
|
61
|
+
);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
export interface QuorumDotsProps {
|
|
65
|
+
readonly have: number;
|
|
66
|
+
readonly need: number;
|
|
67
|
+
/** Host icon for the leading "users" glyph. */
|
|
68
|
+
readonly usersIcon?: ReactNode;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
export function QuorumDots({
|
|
72
|
+
have,
|
|
73
|
+
need,
|
|
74
|
+
usersIcon,
|
|
75
|
+
}: QuorumDotsProps): ReactElement {
|
|
76
|
+
return (
|
|
77
|
+
<span
|
|
78
|
+
style={{
|
|
79
|
+
display: 'inline-flex',
|
|
80
|
+
alignItems: 'center',
|
|
81
|
+
gap: 6,
|
|
82
|
+
fontSize: 12,
|
|
83
|
+
color: 'hsl(var(--ink-3))',
|
|
84
|
+
}}
|
|
85
|
+
>
|
|
86
|
+
{usersIcon}
|
|
87
|
+
<span style={{ display: 'inline-flex', gap: 3 }}>
|
|
88
|
+
{Array.from({ length: need }).map((_, i) => {
|
|
89
|
+
const filled = i < have;
|
|
90
|
+
return (
|
|
91
|
+
<span
|
|
92
|
+
key={`quorum-${i}`}
|
|
93
|
+
style={{
|
|
94
|
+
width: 7,
|
|
95
|
+
height: 7,
|
|
96
|
+
borderRadius: '50%',
|
|
97
|
+
background: filled ? 'hsl(var(--success))' : 'transparent',
|
|
98
|
+
border: `1.5px solid ${
|
|
99
|
+
filled ? 'hsl(var(--success))' : 'hsl(var(--rule-2))'
|
|
100
|
+
}`,
|
|
101
|
+
}}
|
|
102
|
+
/>
|
|
103
|
+
);
|
|
104
|
+
})}
|
|
105
|
+
</span>
|
|
106
|
+
<span className="mono" style={{ fontSize: 11 }}>
|
|
107
|
+
{have}/{need} approvers
|
|
108
|
+
</span>
|
|
109
|
+
</span>
|
|
110
|
+
);
|
|
111
|
+
}
|
|
@@ -0,0 +1,392 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Scoped approval bodies (HANDOFF §6) — the biome-supplied typed detail
|
|
3
|
+
* that renders UNDER the generic frame.
|
|
4
|
+
*
|
|
5
|
+
* `ScopedBody` is an EXHAUSTIVE switch over `ApprovalScopeKind`. A scope
|
|
6
|
+
* whose `kind` is outside the closed set fails SAFE to a visible
|
|
7
|
+
* "unknown scope" placeholder (it surfaces the raw kind) rather than
|
|
8
|
+
* silently dropping the body — the same fail-fast-but-visible contract as
|
|
9
|
+
* the unknown-obligation and unknown-chat-widget paths.
|
|
10
|
+
*
|
|
11
|
+
* Icon-agnostic: bodies that need a glyph take a `renderIcon` mapping a
|
|
12
|
+
* stable `ScopeIcon` key to a host icon node.
|
|
13
|
+
*/
|
|
14
|
+
import {
|
|
15
|
+
ApprovalScopeKind,
|
|
16
|
+
type ApprovalScope,
|
|
17
|
+
type DeployScope,
|
|
18
|
+
type FilesScope,
|
|
19
|
+
type PaymentScope,
|
|
20
|
+
type SlackScope,
|
|
21
|
+
type SqlScope,
|
|
22
|
+
} from './approval-model';
|
|
23
|
+
import { ScopeIcon, type ScopeIconRenderer } from './scope-icons';
|
|
24
|
+
|
|
25
|
+
import type { ReactElement, ReactNode } from 'react';
|
|
26
|
+
|
|
27
|
+
export interface ScopedBodyProps {
|
|
28
|
+
readonly scope: ApprovalScope;
|
|
29
|
+
readonly renderIcon: ScopeIconRenderer;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export function ScopedBody({
|
|
33
|
+
scope,
|
|
34
|
+
renderIcon,
|
|
35
|
+
}: ScopedBodyProps): ReactElement {
|
|
36
|
+
switch (scope.kind) {
|
|
37
|
+
case ApprovalScopeKind.Payment:
|
|
38
|
+
return <PaymentBody s={scope} />;
|
|
39
|
+
case ApprovalScopeKind.Sql:
|
|
40
|
+
return <SqlBody s={scope} renderIcon={renderIcon} />;
|
|
41
|
+
case ApprovalScopeKind.Slack:
|
|
42
|
+
return <SlackBody s={scope} renderIcon={renderIcon} />;
|
|
43
|
+
case ApprovalScopeKind.Deploy:
|
|
44
|
+
return <DeployBody s={scope} renderIcon={renderIcon} />;
|
|
45
|
+
case ApprovalScopeKind.Files:
|
|
46
|
+
return <FilesBody s={scope} renderIcon={renderIcon} />;
|
|
47
|
+
default:
|
|
48
|
+
return <UnknownScopeBody scope={scope} />;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/** Visible fallback for a scope kind the host does not (yet) understand. */
|
|
53
|
+
function UnknownScopeBody({
|
|
54
|
+
scope,
|
|
55
|
+
}: Readonly<{ scope: { readonly kind: string } }>): ReactElement {
|
|
56
|
+
return (
|
|
57
|
+
<div
|
|
58
|
+
style={{
|
|
59
|
+
padding: '8px 11px',
|
|
60
|
+
borderRadius: 7,
|
|
61
|
+
border: '1px dashed hsl(var(--rule-2))',
|
|
62
|
+
background: 'hsl(var(--muted))',
|
|
63
|
+
fontSize: 12,
|
|
64
|
+
color: 'hsl(var(--ink-3))',
|
|
65
|
+
}}
|
|
66
|
+
>
|
|
67
|
+
Unknown approval detail{' '}
|
|
68
|
+
<span className="mono" style={{ color: 'hsl(var(--ink-2))' }}>
|
|
69
|
+
{scope.kind}
|
|
70
|
+
</span>{' '}
|
|
71
|
+
— this biome ships a scope kind this host build cannot render. Review
|
|
72
|
+
the request before deciding.
|
|
73
|
+
</div>
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
function KV({
|
|
78
|
+
k,
|
|
79
|
+
v,
|
|
80
|
+
mono,
|
|
81
|
+
strong,
|
|
82
|
+
}: Readonly<{
|
|
83
|
+
k: string;
|
|
84
|
+
v: ReactNode;
|
|
85
|
+
mono?: boolean;
|
|
86
|
+
strong?: boolean;
|
|
87
|
+
}>): ReactElement {
|
|
88
|
+
return (
|
|
89
|
+
<div
|
|
90
|
+
style={{
|
|
91
|
+
display: 'flex',
|
|
92
|
+
justifyContent: 'space-between',
|
|
93
|
+
gap: 12,
|
|
94
|
+
padding: '3px 0',
|
|
95
|
+
}}
|
|
96
|
+
>
|
|
97
|
+
<span
|
|
98
|
+
style={{ fontSize: 12, color: 'hsl(var(--ink-3))', whiteSpace: 'nowrap' }}
|
|
99
|
+
>
|
|
100
|
+
{k}
|
|
101
|
+
</span>
|
|
102
|
+
<span
|
|
103
|
+
className={mono ? 'mono' : undefined}
|
|
104
|
+
style={{
|
|
105
|
+
fontSize: mono ? 12 : 12.5,
|
|
106
|
+
color: 'hsl(var(--ink))',
|
|
107
|
+
fontWeight: strong ? 600 : 450,
|
|
108
|
+
textAlign: 'right',
|
|
109
|
+
whiteSpace: 'nowrap',
|
|
110
|
+
}}
|
|
111
|
+
>
|
|
112
|
+
{v}
|
|
113
|
+
</span>
|
|
114
|
+
</div>
|
|
115
|
+
);
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
function PaymentBody({ s }: Readonly<{ s: PaymentScope }>): ReactElement {
|
|
119
|
+
return (
|
|
120
|
+
<div>
|
|
121
|
+
<div style={{ marginBottom: 9 }}>
|
|
122
|
+
<div
|
|
123
|
+
className="serif"
|
|
124
|
+
style={{
|
|
125
|
+
fontSize: 27,
|
|
126
|
+
fontWeight: 600,
|
|
127
|
+
color: 'hsl(var(--ink))',
|
|
128
|
+
lineHeight: 1.05,
|
|
129
|
+
}}
|
|
130
|
+
>
|
|
131
|
+
{s.amount}
|
|
132
|
+
</div>
|
|
133
|
+
<div
|
|
134
|
+
style={{ fontSize: 12.5, color: 'hsl(var(--ink-3))', marginTop: 2 }}
|
|
135
|
+
>
|
|
136
|
+
to {s.payee}
|
|
137
|
+
</div>
|
|
138
|
+
</div>
|
|
139
|
+
<div style={{ borderTop: '1px solid hsl(var(--rule))', paddingTop: 4 }}>
|
|
140
|
+
<KV k="Method" v={s.method} />
|
|
141
|
+
<KV k="Account" v={s.account} mono />
|
|
142
|
+
{s.memo && <KV k="Memo" v={s.memo} />}
|
|
143
|
+
</div>
|
|
144
|
+
</div>
|
|
145
|
+
);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
function SqlBody({
|
|
149
|
+
s,
|
|
150
|
+
renderIcon,
|
|
151
|
+
}: Readonly<{ s: SqlScope; renderIcon: ScopeIconRenderer }>): ReactElement {
|
|
152
|
+
return (
|
|
153
|
+
<div>
|
|
154
|
+
<div
|
|
155
|
+
style={{ display: 'flex', gap: 8, marginBottom: 7, flexWrap: 'wrap' }}
|
|
156
|
+
>
|
|
157
|
+
<Tag icon={renderIcon(ScopeIcon.Database, 12)} t={s.dialect} />
|
|
158
|
+
<Tag icon={renderIcon(ScopeIcon.Layers, 12)} t={s.table} />
|
|
159
|
+
<Tag
|
|
160
|
+
icon={renderIcon(ScopeIcon.Alert, 12)}
|
|
161
|
+
t={`${s.rowsAffected} rows`}
|
|
162
|
+
warn
|
|
163
|
+
/>
|
|
164
|
+
</div>
|
|
165
|
+
<pre
|
|
166
|
+
className="mono"
|
|
167
|
+
style={{
|
|
168
|
+
margin: 0,
|
|
169
|
+
padding: '10px 12px',
|
|
170
|
+
background: 'hsl(var(--ink) / 0.04)',
|
|
171
|
+
border: '1px solid hsl(var(--rule))',
|
|
172
|
+
borderRadius: 8,
|
|
173
|
+
fontSize: 12,
|
|
174
|
+
lineHeight: 1.6,
|
|
175
|
+
color: 'hsl(var(--ink))',
|
|
176
|
+
overflowX: 'auto',
|
|
177
|
+
whiteSpace: 'pre',
|
|
178
|
+
}}
|
|
179
|
+
>
|
|
180
|
+
{s.query}
|
|
181
|
+
</pre>
|
|
182
|
+
</div>
|
|
183
|
+
);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
function SlackBody({
|
|
187
|
+
s,
|
|
188
|
+
renderIcon,
|
|
189
|
+
}: Readonly<{ s: SlackScope; renderIcon: ScopeIconRenderer }>): ReactElement {
|
|
190
|
+
return (
|
|
191
|
+
<div>
|
|
192
|
+
<div
|
|
193
|
+
style={{
|
|
194
|
+
display: 'flex',
|
|
195
|
+
alignItems: 'center',
|
|
196
|
+
gap: 8,
|
|
197
|
+
marginBottom: 7,
|
|
198
|
+
fontSize: 12.5,
|
|
199
|
+
}}
|
|
200
|
+
>
|
|
201
|
+
<span style={{ color: 'hsl(var(--ink-3))', display: 'inline-flex' }}>
|
|
202
|
+
{renderIcon(ScopeIcon.Send, 13)}
|
|
203
|
+
</span>
|
|
204
|
+
<span
|
|
205
|
+
className="mono"
|
|
206
|
+
style={{ color: 'hsl(var(--ink))', fontWeight: 600 }}
|
|
207
|
+
>
|
|
208
|
+
{s.channel}
|
|
209
|
+
</span>
|
|
210
|
+
<span style={{ color: 'hsl(var(--ink-4))' }}>
|
|
211
|
+
· {s.workspace} · as {s.as}
|
|
212
|
+
</span>
|
|
213
|
+
</div>
|
|
214
|
+
<div
|
|
215
|
+
style={{
|
|
216
|
+
padding: '10px 12px',
|
|
217
|
+
background: 'hsl(var(--paper-elev))',
|
|
218
|
+
border: '1px solid hsl(var(--rule))',
|
|
219
|
+
borderRadius: 8,
|
|
220
|
+
fontSize: 13,
|
|
221
|
+
lineHeight: 1.55,
|
|
222
|
+
color: 'hsl(var(--ink-2))',
|
|
223
|
+
whiteSpace: 'pre-wrap',
|
|
224
|
+
}}
|
|
225
|
+
>
|
|
226
|
+
{s.message}
|
|
227
|
+
</div>
|
|
228
|
+
</div>
|
|
229
|
+
);
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
function DeployBody({
|
|
233
|
+
s,
|
|
234
|
+
renderIcon,
|
|
235
|
+
}: Readonly<{ s: DeployScope; renderIcon: ScopeIconRenderer }>): ReactElement {
|
|
236
|
+
return (
|
|
237
|
+
<div>
|
|
238
|
+
<div
|
|
239
|
+
style={{
|
|
240
|
+
display: 'flex',
|
|
241
|
+
alignItems: 'center',
|
|
242
|
+
gap: 8,
|
|
243
|
+
marginBottom: 8,
|
|
244
|
+
}}
|
|
245
|
+
>
|
|
246
|
+
<span
|
|
247
|
+
className="mono serif"
|
|
248
|
+
style={{ fontSize: 18, fontWeight: 600, color: 'hsl(var(--ink))' }}
|
|
249
|
+
>
|
|
250
|
+
{s.version}
|
|
251
|
+
</span>
|
|
252
|
+
<span style={{ color: 'hsl(var(--ink-4))', display: 'inline-flex' }}>
|
|
253
|
+
{renderIcon(ScopeIcon.ArrowRight, 14)}
|
|
254
|
+
</span>
|
|
255
|
+
<Tag icon={renderIcon(ScopeIcon.Globe, 12)} t={s.cluster} warn />
|
|
256
|
+
</div>
|
|
257
|
+
<div style={{ display: 'flex', gap: 14, marginBottom: 8 }}>
|
|
258
|
+
<Stat2 n={s.services} l="services" />
|
|
259
|
+
<Stat2 n={s.migrations} l="migrations" warn />
|
|
260
|
+
</div>
|
|
261
|
+
<div style={{ display: 'flex', flexDirection: 'column', gap: 3 }}>
|
|
262
|
+
{s.changelog.map((c) => (
|
|
263
|
+
<div
|
|
264
|
+
key={c}
|
|
265
|
+
style={{
|
|
266
|
+
display: 'flex',
|
|
267
|
+
gap: 8,
|
|
268
|
+
fontSize: 12,
|
|
269
|
+
color: 'hsl(var(--ink-2))',
|
|
270
|
+
}}
|
|
271
|
+
>
|
|
272
|
+
<span
|
|
273
|
+
style={{
|
|
274
|
+
marginTop: 7,
|
|
275
|
+
width: 3,
|
|
276
|
+
height: 3,
|
|
277
|
+
borderRadius: '50%',
|
|
278
|
+
background: 'hsl(var(--ink-4))',
|
|
279
|
+
flexShrink: 0,
|
|
280
|
+
}}
|
|
281
|
+
/>
|
|
282
|
+
<span className="mono" style={{ fontSize: 11.5 }}>
|
|
283
|
+
{c}
|
|
284
|
+
</span>
|
|
285
|
+
</div>
|
|
286
|
+
))}
|
|
287
|
+
</div>
|
|
288
|
+
</div>
|
|
289
|
+
);
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
function FilesBody({
|
|
293
|
+
s,
|
|
294
|
+
renderIcon,
|
|
295
|
+
}: Readonly<{ s: FilesScope; renderIcon: ScopeIconRenderer }>): ReactElement {
|
|
296
|
+
return (
|
|
297
|
+
<div style={{ display: 'flex', flexDirection: 'column', gap: 5 }}>
|
|
298
|
+
{s.files.map((f) => (
|
|
299
|
+
<div
|
|
300
|
+
key={f.path}
|
|
301
|
+
style={{
|
|
302
|
+
display: 'flex',
|
|
303
|
+
alignItems: 'center',
|
|
304
|
+
gap: 9,
|
|
305
|
+
padding: '5px 9px',
|
|
306
|
+
background: 'hsl(var(--paper-elev))',
|
|
307
|
+
border: '1px solid hsl(var(--rule))',
|
|
308
|
+
borderRadius: 7,
|
|
309
|
+
}}
|
|
310
|
+
>
|
|
311
|
+
<span style={{ color: 'hsl(var(--warning))', display: 'inline-flex' }}>
|
|
312
|
+
{renderIcon(ScopeIcon.FileEdit, 13)}
|
|
313
|
+
</span>
|
|
314
|
+
<span
|
|
315
|
+
className="mono"
|
|
316
|
+
style={{
|
|
317
|
+
fontSize: 12,
|
|
318
|
+
color: 'hsl(var(--ink))',
|
|
319
|
+
flex: 1,
|
|
320
|
+
overflow: 'hidden',
|
|
321
|
+
textOverflow: 'ellipsis',
|
|
322
|
+
whiteSpace: 'nowrap',
|
|
323
|
+
}}
|
|
324
|
+
>
|
|
325
|
+
{f.path}
|
|
326
|
+
</span>
|
|
327
|
+
<span
|
|
328
|
+
className="mono"
|
|
329
|
+
style={{ fontSize: 11, color: 'hsl(var(--success))' }}
|
|
330
|
+
>
|
|
331
|
+
+{f.add}
|
|
332
|
+
</span>
|
|
333
|
+
<span
|
|
334
|
+
className="mono"
|
|
335
|
+
style={{ fontSize: 11, color: 'hsl(var(--danger))' }}
|
|
336
|
+
>
|
|
337
|
+
−{f.del}
|
|
338
|
+
</span>
|
|
339
|
+
</div>
|
|
340
|
+
))}
|
|
341
|
+
</div>
|
|
342
|
+
);
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
function Tag({
|
|
346
|
+
icon,
|
|
347
|
+
t,
|
|
348
|
+
warn,
|
|
349
|
+
}: Readonly<{ icon: ReactNode; t: string; warn?: boolean }>): ReactElement {
|
|
350
|
+
return (
|
|
351
|
+
<span
|
|
352
|
+
style={{
|
|
353
|
+
display: 'inline-flex',
|
|
354
|
+
alignItems: 'center',
|
|
355
|
+
gap: 5,
|
|
356
|
+
padding: '2px 8px',
|
|
357
|
+
borderRadius: 6,
|
|
358
|
+
fontSize: 11.5,
|
|
359
|
+
background: warn ? 'hsl(var(--warning) / 0.1)' : 'hsl(var(--muted))',
|
|
360
|
+
border: `1px solid ${
|
|
361
|
+
warn ? 'hsl(var(--warning) / 0.3)' : 'hsl(var(--rule))'
|
|
362
|
+
}`,
|
|
363
|
+
color: warn ? 'hsl(var(--warning))' : 'hsl(var(--ink-2))',
|
|
364
|
+
}}
|
|
365
|
+
>
|
|
366
|
+
{icon}
|
|
367
|
+
{t}
|
|
368
|
+
</span>
|
|
369
|
+
);
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
function Stat2({
|
|
373
|
+
n,
|
|
374
|
+
l,
|
|
375
|
+
warn,
|
|
376
|
+
}: Readonly<{ n: number; l: string; warn?: boolean }>): ReactElement {
|
|
377
|
+
return (
|
|
378
|
+
<span style={{ display: 'inline-flex', alignItems: 'baseline', gap: 5 }}>
|
|
379
|
+
<span
|
|
380
|
+
className="serif"
|
|
381
|
+
style={{
|
|
382
|
+
fontSize: 18,
|
|
383
|
+
fontWeight: 600,
|
|
384
|
+
color: warn ? 'hsl(var(--warning))' : 'hsl(var(--ink))',
|
|
385
|
+
}}
|
|
386
|
+
>
|
|
387
|
+
{n}
|
|
388
|
+
</span>
|
|
389
|
+
<span style={{ fontSize: 11.5, color: 'hsl(var(--ink-3))' }}>{l}</span>
|
|
390
|
+
</span>
|
|
391
|
+
);
|
|
392
|
+
}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* The icon contract for the approval surfaces. The kit is icon-agnostic
|
|
3
|
+
* (it never imports an icon set), so the host supplies one bundle that
|
|
4
|
+
* resolves every glyph the approval frame, scoped bodies, obligation
|
|
5
|
+
* chips and Center need. Keeping it as a single typed bundle means a
|
|
6
|
+
* consumer wires icons ONCE and passes it to every approval surface.
|
|
7
|
+
*/
|
|
8
|
+
import type { ObligationIcon } from './obligation-display';
|
|
9
|
+
import type { ScopeIcon } from './scope-icons';
|
|
10
|
+
import type { ReactNode } from 'react';
|
|
11
|
+
|
|
12
|
+
export interface ApprovalIcons {
|
|
13
|
+
/** Map a stable obligation-chip glyph key → host icon node. */
|
|
14
|
+
readonly obligation: (key: ObligationIcon, size: number) => ReactNode;
|
|
15
|
+
/** Map a stable scoped-body glyph key → host icon node. */
|
|
16
|
+
readonly scope: (key: ScopeIcon, size: number) => ReactNode;
|
|
17
|
+
/** Frame: the "approval required" / approve shield. */
|
|
18
|
+
readonly shield: (size: number) => ReactNode;
|
|
19
|
+
/** Frame: capability-ref leading bolt. */
|
|
20
|
+
readonly capability: (size: number) => ReactNode;
|
|
21
|
+
/** Frame: approver-role person glyph. */
|
|
22
|
+
readonly user: (size: number) => ReactNode;
|
|
23
|
+
/** Quorum: multi-approver glyph. */
|
|
24
|
+
readonly users: (size: number) => ReactNode;
|
|
25
|
+
/** Resolved state: approved tick. */
|
|
26
|
+
readonly approved: (size: number) => ReactNode;
|
|
27
|
+
/** Resolved state: denied cross. */
|
|
28
|
+
readonly denied: (size: number) => ReactNode;
|
|
29
|
+
/** Generic small check (action buttons / Center). */
|
|
30
|
+
readonly check: (size: number) => ReactNode;
|
|
31
|
+
}
|