@usetheo/ui 0.12.0 → 0.13.0

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 (52) hide show
  1. package/CHANGELOG.md +94 -0
  2. package/README.md +20 -19
  3. package/dist/chunk-6ZQKEY54.js +149 -0
  4. package/dist/chunk-6ZQKEY54.js.map +1 -0
  5. package/dist/chunk-AVPHVQZS.js +73 -0
  6. package/dist/chunk-AVPHVQZS.js.map +1 -0
  7. package/dist/chunk-GXBFGWQN.js +81 -0
  8. package/dist/chunk-GXBFGWQN.js.map +1 -0
  9. package/dist/chunk-I32I36LW.js +113 -0
  10. package/dist/chunk-I32I36LW.js.map +1 -0
  11. package/dist/chunk-JPTPIZ5V.js +120 -0
  12. package/dist/chunk-JPTPIZ5V.js.map +1 -0
  13. package/dist/{chunk-TO3UAT6O.js → chunk-K7PYLTMP.js} +3 -3
  14. package/dist/{chunk-TO3UAT6O.js.map → chunk-K7PYLTMP.js.map} +1 -1
  15. package/dist/{chunk-R2PAGRDP.js → chunk-MYEHGDC2.js} +2 -2
  16. package/dist/{chunk-R2PAGRDP.js.map → chunk-MYEHGDC2.js.map} +1 -1
  17. package/dist/{chunk-IPEYGWA7.js → chunk-PTHRL242.js} +4 -4
  18. package/dist/{chunk-IPEYGWA7.js.map → chunk-PTHRL242.js.map} +1 -1
  19. package/dist/chunk-RC5XME4T.js +33 -0
  20. package/dist/chunk-RC5XME4T.js.map +1 -0
  21. package/dist/chunk-UK27KR35.js +73 -0
  22. package/dist/chunk-UK27KR35.js.map +1 -0
  23. package/dist/chunk-UOMQPIB4.js +48 -0
  24. package/dist/chunk-UOMQPIB4.js.map +1 -0
  25. package/dist/{chunk-TNBJ36XJ.js → chunk-XZKEGEPT.js} +4 -4
  26. package/dist/{chunk-TNBJ36XJ.js.map → chunk-XZKEGEPT.js.map} +1 -1
  27. package/dist/components.css +1 -1
  28. package/dist/composites/agent-editor/index.js +2 -2
  29. package/dist/composites/rule-editor/index.js +3 -3
  30. package/dist/composites/skill-editor/index.js +3 -3
  31. package/dist/composites/stability-bundle-viewer/index.js +4 -0
  32. package/dist/composites/stability-bundle-viewer/index.js.map +1 -0
  33. package/dist/index.d.ts +162 -1
  34. package/dist/index.js +44 -36
  35. package/dist/index.js.map +1 -1
  36. package/dist/primitives/branch-indicator/index.js +4 -0
  37. package/dist/primitives/branch-indicator/index.js.map +1 -0
  38. package/dist/primitives/channel-card/index.js +4 -0
  39. package/dist/primitives/channel-card/index.js.map +1 -0
  40. package/dist/primitives/export-chat-dialog/index.js +4 -0
  41. package/dist/primitives/export-chat-dialog/index.js.map +1 -0
  42. package/dist/primitives/gateway-status-indicator/index.js +4 -0
  43. package/dist/primitives/gateway-status-indicator/index.js.map +1 -0
  44. package/dist/primitives/pin-input/index.js +1 -1
  45. package/dist/primitives/run-status-pill/index.js +4 -0
  46. package/dist/primitives/run-status-pill/index.js.map +1 -0
  47. package/dist/primitives/thinking-level-selector/index.js +4 -0
  48. package/dist/primitives/thinking-level-selector/index.js.map +1 -0
  49. package/dist/primitives/update-banner/index.js +4 -0
  50. package/dist/primitives/update-banner/index.js.map +1 -0
  51. package/package.json +123 -99
  52. package/registry/r/pin-input.json +1 -1
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/primitives/branch-indicator/branch-indicator.tsx"],"names":[],"mappings":";;;;AAiBO,IAAM,eAAA,GAAkB,UAAA;AAAA,EAC7B,CAAC,EAAE,WAAA,EAAa,WAAA,EAAa,SAAA,EAAW,eAAe,UAAA,EAAY,GAAG,IAAA,EAAK,EAAG,GAAA,KAAQ;AACpF,IAAA,IAAI,CAAC,MAAA,CAAO,SAAA,CAAU,WAAW,CAAA,IAAK,WAAA,GAAc,GAAG,OAAO,IAAA;AAC9D,IAAA,uBACE,IAAA;AAAA,MAAC,MAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,SAAA,EAAW,EAAA;AAAA,UACT,qEAAA;AAAA,UACA,yDAAA;AAAA,UACA;AAAA,SACF;AAAA,QACA,KAAA,EAAO,eAAe,CAAA,aAAA,EAAgB,WAAA,GAAc,CAAC,CAAA,KAAA,EAAQ,WAAA,GAAc,CAAA,GAAI,GAAA,GAAM,EAAE,CAAA,CAAA;AAAA,QACvF,eAAa,UAAA,IAAc,kBAAA;AAAA,QAC3B,mBAAA,EAAmB,WAAA;AAAA,QAClB,GAAG,IAAA;AAAA,QACL,QAAA,EAAA;AAAA,UAAA,MAAA;AAAA,UACG;AAAA;AAAA;AAAA,KACJ;AAAA,EAEJ;AACF;AACA,eAAA,CAAgB,WAAA,GAAc,iBAAA","file":"chunk-RC5XME4T.js","sourcesContent":["import { type HTMLAttributes, forwardRef } from \"react\";\n\nimport { cn } from \"../../../lib/cn.js\";\n\n/**\n * BranchIndicator — small \"×N\" pill that shows when a run was retried/branched.\n *\n * Renders `null` when `branchCount < 2` OR when not a positive integer.\n * EC-10 guard: parametrized over [-1, 0, 0.5] in tests.\n */\n\nexport interface BranchIndicatorProps extends HTMLAttributes<HTMLSpanElement> {\n branchCount: number;\n tooltipText?: string;\n \"data-testid\"?: string;\n}\n\nexport const BranchIndicator = forwardRef<HTMLSpanElement, BranchIndicatorProps>(\n ({ branchCount, tooltipText, className, \"data-testid\": dataTestId, ...rest }, ref) => {\n if (!Number.isInteger(branchCount) || branchCount < 2) return null;\n return (\n <span\n ref={ref}\n className={cn(\n \"inline-flex items-center rounded-full border border-border bg-muted\",\n \"px-1.5 py-0.5 font-medium text-muted-foreground text-xs\",\n className,\n )}\n title={tooltipText ?? `Run branched ${branchCount - 1} time${branchCount > 2 ? \"s\" : \"\"}`}\n data-testid={dataTestId ?? \"branch-indicator\"}\n data-branch-count={branchCount}\n {...rest}\n >\n ×{branchCount}\n </span>\n );\n },\n);\nBranchIndicator.displayName = \"BranchIndicator\";\n"]}
@@ -0,0 +1,73 @@
1
+ import { cn } from './chunk-ZSRJCIWF.js';
2
+ import { PauseCircle, XCircle, AlertCircle, CheckCircle2, Loader2, Clock } from 'lucide-react';
3
+ import { forwardRef } from 'react';
4
+ import { jsxs, jsx } from 'react/jsx-runtime';
5
+
6
+ var STATUS_META = {
7
+ queued: {
8
+ icon: Clock,
9
+ label: "Queued",
10
+ className: "border-border bg-muted text-muted-foreground"
11
+ },
12
+ in_progress: {
13
+ icon: Loader2,
14
+ label: "In progress",
15
+ className: "border-primary/40 bg-primary/10 text-primary",
16
+ spin: true
17
+ },
18
+ finished: {
19
+ icon: CheckCircle2,
20
+ label: "Done",
21
+ className: "border-emerald-500/40 bg-emerald-500/10 text-emerald-600 dark:text-emerald-400"
22
+ },
23
+ error: {
24
+ icon: AlertCircle,
25
+ label: "Error",
26
+ className: "border-destructive/40 bg-destructive/10 text-destructive"
27
+ },
28
+ cancelled: {
29
+ icon: XCircle,
30
+ label: "Cancelled",
31
+ className: "border-border bg-muted text-muted-foreground"
32
+ },
33
+ interrupted: {
34
+ icon: PauseCircle,
35
+ label: "Interrupted",
36
+ className: "border-amber-500/40 bg-amber-500/10 text-amber-600 dark:text-amber-400"
37
+ }
38
+ };
39
+ var RunStatusPill = forwardRef(
40
+ ({ status, detail, className, "data-testid": dataTestId, ...rest }, ref) => {
41
+ const meta = STATUS_META[status];
42
+ const Icon = meta.icon;
43
+ return /* @__PURE__ */ jsxs(
44
+ "span",
45
+ {
46
+ ref,
47
+ role: "status",
48
+ "aria-live": "polite",
49
+ className: cn(
50
+ "inline-flex items-center gap-1.5 rounded-full border px-2 py-0.5 font-medium text-xs",
51
+ meta.className,
52
+ className
53
+ ),
54
+ "data-testid": dataTestId ?? "run-status-pill",
55
+ "data-status": status,
56
+ ...rest,
57
+ children: [
58
+ /* @__PURE__ */ jsx(Icon, { className: cn("size-3", meta.spin && "animate-spin"), "aria-hidden": true }),
59
+ /* @__PURE__ */ jsx("span", { children: meta.label }),
60
+ detail !== void 0 && detail.length > 0 && /* @__PURE__ */ jsxs("span", { className: "text-muted-foreground", "data-testid": "run-status-detail", children: [
61
+ " \xB7 ",
62
+ detail
63
+ ] })
64
+ ]
65
+ }
66
+ );
67
+ }
68
+ );
69
+ RunStatusPill.displayName = "RunStatusPill";
70
+
71
+ export { RunStatusPill };
72
+ //# sourceMappingURL=chunk-UK27KR35.js.map
73
+ //# sourceMappingURL=chunk-UK27KR35.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/primitives/run-status-pill/run-status-pill.tsx"],"names":[],"mappings":";;;;;AAwCA,IAAM,WAAA,GAA6C;AAAA,EACjD,MAAA,EAAQ;AAAA,IACN,IAAA,EAAM,KAAA;AAAA,IACN,KAAA,EAAO,QAAA;AAAA,IACP,SAAA,EAAW;AAAA,GACb;AAAA,EACA,WAAA,EAAa;AAAA,IACX,IAAA,EAAM,OAAA;AAAA,IACN,KAAA,EAAO,aAAA;AAAA,IACP,SAAA,EAAW,8CAAA;AAAA,IACX,IAAA,EAAM;AAAA,GACR;AAAA,EACA,QAAA,EAAU;AAAA,IACR,IAAA,EAAM,YAAA;AAAA,IACN,KAAA,EAAO,MAAA;AAAA,IACP,SAAA,EAAW;AAAA,GACb;AAAA,EACA,KAAA,EAAO;AAAA,IACL,IAAA,EAAM,WAAA;AAAA,IACN,KAAA,EAAO,OAAA;AAAA,IACP,SAAA,EAAW;AAAA,GACb;AAAA,EACA,SAAA,EAAW;AAAA,IACT,IAAA,EAAM,OAAA;AAAA,IACN,KAAA,EAAO,WAAA;AAAA,IACP,SAAA,EAAW;AAAA,GACb;AAAA,EACA,WAAA,EAAa;AAAA,IACX,IAAA,EAAM,WAAA;AAAA,IACN,KAAA,EAAO,aAAA;AAAA,IACP,SAAA,EAAW;AAAA;AAEf,CAAA;AAEO,IAAM,aAAA,GAAgB,UAAA;AAAA,EAC3B,CAAC,EAAE,MAAA,EAAQ,MAAA,EAAQ,SAAA,EAAW,eAAe,UAAA,EAAY,GAAG,IAAA,EAAK,EAAG,GAAA,KAAQ;AAC1E,IAAA,MAAM,IAAA,GAAO,YAAY,MAAM,CAAA;AAC/B,IAAA,MAAM,OAAO,IAAA,CAAK,IAAA;AAClB,IAAA,uBACE,IAAA;AAAA,MAAC,MAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QAEA,IAAA,EAAK,QAAA;AAAA,QACL,WAAA,EAAU,QAAA;AAAA,QACV,SAAA,EAAW,EAAA;AAAA,UACT,sFAAA;AAAA,UACA,IAAA,CAAK,SAAA;AAAA,UACL;AAAA,SACF;AAAA,QACA,eAAa,UAAA,IAAc,iBAAA;AAAA,QAC3B,aAAA,EAAa,MAAA;AAAA,QACZ,GAAG,IAAA;AAAA,QAEJ,QAAA,EAAA;AAAA,0BAAA,GAAA,CAAC,IAAA,EAAA,EAAK,WAAW,EAAA,CAAG,QAAA,EAAU,KAAK,IAAA,IAAQ,cAAc,CAAA,EAAG,aAAA,EAAW,IAAA,EAAC,CAAA;AAAA,0BACxE,GAAA,CAAC,MAAA,EAAA,EAAM,QAAA,EAAA,IAAA,CAAK,KAAA,EAAM,CAAA;AAAA,UACjB,MAAA,KAAW,MAAA,IAAa,MAAA,CAAO,MAAA,GAAS,CAAA,yBACtC,MAAA,EAAA,EAAK,SAAA,EAAU,uBAAA,EAAwB,aAAA,EAAY,mBAAA,EACjD,QAAA,EAAA;AAAA,YAAA,QAAA;AAAA,YACA;AAAA,WAAA,EACH;AAAA;AAAA;AAAA,KAEJ;AAAA,EAEJ;AACF;AACA,aAAA,CAAc,WAAA,GAAc,eAAA","file":"chunk-UK27KR35.js","sourcesContent":["import {\n AlertCircle,\n CheckCircle2,\n Clock,\n Loader2,\n type LucideIcon,\n PauseCircle,\n XCircle,\n} from \"lucide-react\";\nimport { type HTMLAttributes, forwardRef } from \"react\";\n\nimport { cn } from \"../../../lib/cn.js\";\n\n/**\n * RunStatusPill — compact status indicator for Run/Task lifecycle states.\n * Mirrors OpenClaw's `Run status: In progress | Done | Interrupted` pill.\n * Closed enum mirrors SDK `Task` state (D362).\n */\n\nexport type RunStatus =\n | \"queued\"\n | \"in_progress\"\n | \"finished\"\n | \"error\"\n | \"cancelled\"\n | \"interrupted\";\n\nexport interface RunStatusPillProps extends Omit<HTMLAttributes<HTMLSpanElement>, \"title\"> {\n status: RunStatus;\n detail?: string;\n \"data-testid\"?: string;\n}\n\ninterface StatusMeta {\n icon: LucideIcon;\n label: string;\n className: string;\n spin?: boolean;\n}\n\nconst STATUS_META: Record<RunStatus, StatusMeta> = {\n queued: {\n icon: Clock,\n label: \"Queued\",\n className: \"border-border bg-muted text-muted-foreground\",\n },\n in_progress: {\n icon: Loader2,\n label: \"In progress\",\n className: \"border-primary/40 bg-primary/10 text-primary\",\n spin: true,\n },\n finished: {\n icon: CheckCircle2,\n label: \"Done\",\n className: \"border-emerald-500/40 bg-emerald-500/10 text-emerald-600 dark:text-emerald-400\",\n },\n error: {\n icon: AlertCircle,\n label: \"Error\",\n className: \"border-destructive/40 bg-destructive/10 text-destructive\",\n },\n cancelled: {\n icon: XCircle,\n label: \"Cancelled\",\n className: \"border-border bg-muted text-muted-foreground\",\n },\n interrupted: {\n icon: PauseCircle,\n label: \"Interrupted\",\n className: \"border-amber-500/40 bg-amber-500/10 text-amber-600 dark:text-amber-400\",\n },\n};\n\nexport const RunStatusPill = forwardRef<HTMLSpanElement, RunStatusPillProps>(\n ({ status, detail, className, \"data-testid\": dataTestId, ...rest }, ref) => {\n const meta = STATUS_META[status];\n const Icon = meta.icon;\n return (\n <span\n ref={ref}\n // biome-ignore lint/a11y/useSemanticElements: inline status badge — <span role=\"status\"> keeps it phrasing-content compatible inside chat headers and table cells, where the implicit <output> placement would break layout\n role=\"status\"\n aria-live=\"polite\"\n className={cn(\n \"inline-flex items-center gap-1.5 rounded-full border px-2 py-0.5 font-medium text-xs\",\n meta.className,\n className,\n )}\n data-testid={dataTestId ?? \"run-status-pill\"}\n data-status={status}\n {...rest}\n >\n <Icon className={cn(\"size-3\", meta.spin && \"animate-spin\")} aria-hidden />\n <span>{meta.label}</span>\n {detail !== undefined && detail.length > 0 && (\n <span className=\"text-muted-foreground\" data-testid=\"run-status-detail\">\n {\" · \"}\n {detail}\n </span>\n )}\n </span>\n );\n },\n);\nRunStatusPill.displayName = \"RunStatusPill\";\n"]}
@@ -0,0 +1,48 @@
1
+ import { cn } from './chunk-ZSRJCIWF.js';
2
+ import { forwardRef } from 'react';
3
+ import { jsxs, jsx } from 'react/jsx-runtime';
4
+
5
+ var OVERRIDE_OPTIONS = ["off", "minimal", "low", "medium", "high", "xhigh"];
6
+ function inheritedLabel(inheritedValue) {
7
+ if (inheritedValue === void 0) return "Inherited";
8
+ return `Inherited: ${inheritedValue}`;
9
+ }
10
+ function overrideLabel(level) {
11
+ if (level === "off") return "Off";
12
+ return `Override: ${level}`;
13
+ }
14
+ var ThinkingLevelSelector = forwardRef(
15
+ ({ value, inheritedValue, onChange, disabled = false, className, "data-testid": dataTestId }, ref) => {
16
+ if (process.env.NODE_ENV !== "production" && value === "inherited" && inheritedValue === void 0) {
17
+ console.warn(
18
+ '[ThinkingLevelSelector] value="inherited" but no inheritedValue prop \u2014 rendering as "Inherited: ?".'
19
+ );
20
+ }
21
+ return /* @__PURE__ */ jsxs(
22
+ "select",
23
+ {
24
+ ref,
25
+ "aria-label": "Thinking level",
26
+ className: cn(
27
+ "rounded-md border border-border bg-background px-2 py-1 text-sm",
28
+ "focus:outline-none focus:ring-2 focus:ring-primary",
29
+ disabled && "cursor-not-allowed opacity-50",
30
+ className
31
+ ),
32
+ disabled,
33
+ value,
34
+ onChange: (e) => onChange(e.target.value),
35
+ "data-testid": dataTestId ?? "thinking-level-selector",
36
+ children: [
37
+ /* @__PURE__ */ jsx("option", { value: "inherited", children: inheritedLabel(inheritedValue) }),
38
+ OVERRIDE_OPTIONS.map((level) => /* @__PURE__ */ jsx("option", { value: level, children: overrideLabel(level) }, level))
39
+ ]
40
+ }
41
+ );
42
+ }
43
+ );
44
+ ThinkingLevelSelector.displayName = "ThinkingLevelSelector";
45
+
46
+ export { ThinkingLevelSelector };
47
+ //# sourceMappingURL=chunk-UOMQPIB4.js.map
48
+ //# sourceMappingURL=chunk-UOMQPIB4.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/primitives/thinking-level-selector/thinking-level-selector.tsx"],"names":[],"mappings":";;;;AA+BA,IAAM,mBAAoC,CAAC,KAAA,EAAO,WAAW,KAAA,EAAO,QAAA,EAAU,QAAQ,OAAO,CAAA;AAE7F,SAAS,eAAe,cAAA,EAAmD;AACzE,EAAA,IAAI,cAAA,KAAmB,QAAW,OAAO,WAAA;AACzC,EAAA,OAAO,cAAc,cAAc,CAAA,CAAA;AACrC;AAEA,SAAS,cAAc,KAAA,EAA8B;AACnD,EAAA,IAAI,KAAA,KAAU,OAAO,OAAO,KAAA;AAC5B,EAAA,OAAO,aAAa,KAAK,CAAA,CAAA;AAC3B;AAEO,IAAM,qBAAA,GAAwB,UAAA;AAAA,EACnC,CACE,EAAE,KAAA,EAAO,cAAA,EAAgB,QAAA,EAAU,QAAA,GAAW,KAAA,EAAO,SAAA,EAAW,aAAA,EAAe,UAAA,EAAW,EAC1F,GAAA,KACG;AACH,IAAA,IACE,QAAQ,GAAA,CAAI,QAAA,KAAa,gBACzB,KAAA,KAAU,WAAA,IACV,mBAAmB,MAAA,EACnB;AAEA,MAAA,OAAA,CAAQ,IAAA;AAAA,QACN;AAAA,OACF;AAAA,IACF;AACA,IAAA,uBACE,IAAA;AAAA,MAAC,QAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,YAAA,EAAW,gBAAA;AAAA,QACX,SAAA,EAAW,EAAA;AAAA,UACT,iEAAA;AAAA,UACA,oDAAA;AAAA,UACA,QAAA,IAAY,+BAAA;AAAA,UACZ;AAAA,SACF;AAAA,QACA,QAAA;AAAA,QACA,KAAA;AAAA,QACA,UAAU,CAAC,CAAA,KAAM,QAAA,CAAS,CAAA,CAAE,OAAO,KAAiC,CAAA;AAAA,QACpE,eAAa,UAAA,IAAc,yBAAA;AAAA,QAE3B,QAAA,EAAA;AAAA,0BAAA,GAAA,CAAC,QAAA,EAAA,EAAO,KAAA,EAAM,WAAA,EAAa,QAAA,EAAA,cAAA,CAAe,cAAc,CAAA,EAAE,CAAA;AAAA,UACzD,gBAAA,CAAiB,GAAA,CAAI,CAAC,KAAA,qBACrB,GAAA,CAAC,QAAA,EAAA,EAAmB,KAAA,EAAO,KAAA,EACxB,QAAA,EAAA,aAAA,CAAc,KAAK,CAAA,EAAA,EADT,KAEb,CACD;AAAA;AAAA;AAAA,KACH;AAAA,EAEJ;AACF;AACA,qBAAA,CAAsB,WAAA,GAAc,uBAAA","file":"chunk-UOMQPIB4.js","sourcesContent":["import { forwardRef } from \"react\";\n\nimport { cn } from \"../../../lib/cn.js\";\n\n/**\n * ThinkingLevelSelector — multi-state combobox for LLM reasoning budget.\n *\n * Mirrors the OpenClaw Control UI thinking selector. Values follow the SDK's\n * `Agent.thinking` parameter (\"minimal\" | \"low\" | \"medium\" | \"high\" | \"xhigh\")\n * plus a sentinel \"off\" and the special \"inherited\" mode that means \"use the\n * value resolved by the parent agent/session\".\n *\n * Pure UI primitive. No backend dependency. Consumer wires `value` from agent\n * state and persists `onChange` selections wherever it pleases.\n *\n * Phase 1 of `theokit-ui-parity-plan.md` (T1.1).\n */\n\nexport type ThinkingLevel = \"off\" | \"minimal\" | \"low\" | \"medium\" | \"high\" | \"xhigh\";\n\nexport type ThinkingLevelOrInherited = ThinkingLevel | \"inherited\";\n\nexport interface ThinkingLevelSelectorProps {\n value: ThinkingLevelOrInherited;\n inheritedValue?: ThinkingLevel;\n onChange: (value: ThinkingLevelOrInherited) => void;\n disabled?: boolean;\n className?: string;\n \"data-testid\"?: string;\n}\n\nconst OVERRIDE_OPTIONS: ThinkingLevel[] = [\"off\", \"minimal\", \"low\", \"medium\", \"high\", \"xhigh\"];\n\nfunction inheritedLabel(inheritedValue: ThinkingLevel | undefined): string {\n if (inheritedValue === undefined) return \"Inherited\";\n return `Inherited: ${inheritedValue}`;\n}\n\nfunction overrideLabel(level: ThinkingLevel): string {\n if (level === \"off\") return \"Off\";\n return `Override: ${level}`;\n}\n\nexport const ThinkingLevelSelector = forwardRef<HTMLSelectElement, ThinkingLevelSelectorProps>(\n (\n { value, inheritedValue, onChange, disabled = false, className, \"data-testid\": dataTestId },\n ref,\n ) => {\n if (\n process.env.NODE_ENV !== \"production\" &&\n value === \"inherited\" &&\n inheritedValue === undefined\n ) {\n // biome-ignore lint/suspicious/noConsole: dev-only warning when caller passes value=\"inherited\" without inheritedValue\n console.warn(\n '[ThinkingLevelSelector] value=\"inherited\" but no inheritedValue prop — rendering as \"Inherited: ?\".',\n );\n }\n return (\n <select\n ref={ref}\n aria-label=\"Thinking level\"\n className={cn(\n \"rounded-md border border-border bg-background px-2 py-1 text-sm\",\n \"focus:outline-none focus:ring-2 focus:ring-primary\",\n disabled && \"cursor-not-allowed opacity-50\",\n className,\n )}\n disabled={disabled}\n value={value}\n onChange={(e) => onChange(e.target.value as ThinkingLevelOrInherited)}\n data-testid={dataTestId ?? \"thinking-level-selector\"}\n >\n <option value=\"inherited\">{inheritedLabel(inheritedValue)}</option>\n {OVERRIDE_OPTIONS.map((level) => (\n <option key={level} value={level}>\n {overrideLabel(level)}\n </option>\n ))}\n </select>\n );\n },\n);\nThinkingLevelSelector.displayName = \"ThinkingLevelSelector\";\n"]}
@@ -1,9 +1,9 @@
1
1
  import { ALL_MODES, MODE_LABEL } from './chunk-VM4RMQQN.js';
2
- import { Textarea } from './chunk-WWNH5ENT.js';
3
2
  import { Switch } from './chunk-3HOXC25T.js';
3
+ import { Textarea } from './chunk-WWNH5ENT.js';
4
4
  import { Select } from './chunk-EP25QJ4N.js';
5
- import { FormField } from './chunk-TK24HQJJ.js';
6
5
  import { Input } from './chunk-H3VJMFJQ.js';
6
+ import { FormField } from './chunk-TK24HQJJ.js';
7
7
  import { Button } from './chunk-57NXT3OX.js';
8
8
  import { cn } from './chunk-ZSRJCIWF.js';
9
9
  import { useState } from 'react';
@@ -152,5 +152,5 @@ function RuleEditor({
152
152
  }
153
153
 
154
154
  export { RuleEditor };
155
- //# sourceMappingURL=chunk-TNBJ36XJ.js.map
156
- //# sourceMappingURL=chunk-TNBJ36XJ.js.map
155
+ //# sourceMappingURL=chunk-XZKEGEPT.js.map
156
+ //# sourceMappingURL=chunk-XZKEGEPT.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/components/composites/rule-editor/rule-editor.tsx"],"names":[],"mappings":"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","file":"chunk-TNBJ36XJ.js","sourcesContent":["import { useState } from \"react\";\nimport type { FormEvent, HTMLAttributes } from \"react\";\nimport { cn } from \"../../../lib/cn.js\";\nimport { ALL_MODES, MODE_LABEL, type Mode } from \"../../../types/mode.js\";\nimport type { Rule, RuleScope, RuleState } from \"../../../types/rule.js\";\nimport { Button } from \"../../primitives/button/index.js\";\nimport { FormField } from \"../../primitives/form-field/index.js\";\nimport { Input } from \"../../primitives/input/index.js\";\nimport { Select } from \"../../primitives/select/index.js\";\nimport { Switch } from \"../../primitives/switch/index.js\";\nimport { Textarea } from \"../../primitives/textarea/index.js\";\n\n/**\n * RuleEditor — form for creating or editing a Rule (behavior instruction\n * injected into the system prompt).\n */\n\ntype RuleDraft = Omit<Rule, \"id\" | \"updatedAt\"> & {\n id?: string;\n tags?: string[];\n};\n\ninterface RuleEditorProps extends Omit<HTMLAttributes<HTMLFormElement>, \"onSubmit\" | \"onChange\"> {\n initial?: Partial<Rule>;\n onSave: (draft: RuleDraft) => void;\n onCancel?: () => void;\n onDelete?: () => void;\n}\n\nconst SCOPES: Array<{ id: RuleScope; label: string }> = [\n { id: \"global\", label: \"Global — applies to every session\" },\n { id: \"project\", label: \"Project — only this workspace\" },\n];\n\nexport function RuleEditor({\n className,\n initial,\n onSave,\n onCancel,\n onDelete,\n ...formProps\n}: RuleEditorProps) {\n const [title, setTitle] = useState(initial?.title ?? \"\");\n const [body, setBody] = useState(initial?.body ?? \"\");\n const [scope, setScope] = useState<RuleScope>(initial?.scope ?? \"global\");\n const [tagsRaw, setTagsRaw] = useState(initial?.tags?.join(\", \") ?? \"\");\n const [enabled, setEnabled] = useState<RuleState>(initial?.state ?? \"enabled\");\n const [modes, setModes] = useState<Mode[]>(initial?.modes ?? []);\n\n // Note: state is only seeded once on mount. To reset the form when editing\n // a different rule, use the React `key` pattern at the call site:\n // <RuleEditor key={rule.id} initial={rule} ... />\n const toggleMode = (m: Mode) =>\n setModes((prev) => (prev.includes(m) ? prev.filter((x) => x !== m) : [...prev, m]));\n\n const canSave = title.trim().length > 0 && body.trim().length > 0;\n const handleSubmit = (e: FormEvent) => {\n e.preventDefault();\n if (!canSave) return;\n onSave({\n id: initial?.id,\n title: title.trim(),\n body: body.trim(),\n scope,\n state: enabled,\n tags: tagsRaw\n .split(\",\")\n .map((t) => t.trim())\n .filter(Boolean),\n modes: modes.length > 0 ? modes : undefined,\n });\n };\n\n return (\n <form\n onSubmit={handleSubmit}\n className={cn(\"flex h-full flex-col gap-4\", className)}\n {...formProps}\n >\n <FormField>\n <FormField.Label>Title</FormField.Label>\n <FormField.Control>\n <Input\n value={title}\n onChange={(e) => setTitle(e.target.value)}\n placeholder=\"Always write tests before fixes\"\n required\n />\n </FormField.Control>\n <FormField.Hint>Short, imperative summary the agent will keep in memory.</FormField.Hint>\n </FormField>\n\n <FormField className=\"flex-1\">\n <FormField.Label>Body (markdown)</FormField.Label>\n <FormField.Control>\n <Textarea\n value={body}\n onChange={(e) => setBody(e.target.value)}\n rows={8}\n placeholder=\"When fixing a bug, first write a failing regression test, then the fix.\"\n required\n className=\"min-h-[12rem] flex-1 font-mono text-code-sm\"\n />\n </FormField.Control>\n <FormField.Hint>Injected into the system prompt verbatim.</FormField.Hint>\n </FormField>\n\n <div className=\"grid grid-cols-2 gap-3\">\n <FormField>\n <FormField.Label>Scope</FormField.Label>\n <FormField.Control>\n <Select\n value={scope}\n onValueChange={(v) => {\n // Re-audit Issue 7: narrow Select string value against\n // SCOPES.id before casting. Silent no-op for unknown.\n const next = SCOPES.find((s) => s.id === v);\n if (next) setScope(next.id);\n }}\n >\n <Select.Trigger aria-label=\"Select rule scope\">\n <Select.Value />\n </Select.Trigger>\n <Select.Content>\n {SCOPES.map((s) => (\n <Select.Item key={s.id} value={s.id}>\n {s.label}\n </Select.Item>\n ))}\n </Select.Content>\n </Select>\n </FormField.Control>\n </FormField>\n <FormField>\n <FormField.Label>Tags (comma-separated)</FormField.Label>\n <FormField.Control>\n <Input\n value={tagsRaw}\n onChange={(e) => setTagsRaw(e.target.value)}\n placeholder=\"testing, process\"\n />\n </FormField.Control>\n </FormField>\n </div>\n\n <FormField>\n <FormField.Label>Active modes</FormField.Label>\n <div className=\"flex flex-wrap gap-1.5\">\n {ALL_MODES.map((m) => {\n const on = modes.includes(m);\n return (\n <button\n key={m}\n type=\"button\"\n onClick={() => toggleMode(m)}\n aria-pressed={on}\n className={cn(\n \"inline-flex h-7 items-center rounded-full border px-3 font-mono text-body-sm transition-colors\",\n on\n ? \"border-primary bg-primary/15 text-primary\"\n : \"border-border/60 bg-card text-muted-foreground hover:text-foreground\",\n )}\n >\n {MODE_LABEL[m]}\n </button>\n );\n })}\n </div>\n <FormField.Hint>\n {modes.length === 0\n ? \"Empty = global (applies to every mode).\"\n : `Only visible in: ${modes.map((m) => MODE_LABEL[m]).join(\", \")}.`}\n </FormField.Hint>\n </FormField>\n\n <div className=\"flex items-center gap-3\">\n <Switch\n checked={enabled === \"enabled\"}\n onCheckedChange={(v) => setEnabled(v ? \"enabled\" : \"disabled\")}\n aria-label=\"Enabled\"\n />\n <span className=\"text-body-sm text-muted-foreground\">\n {enabled === \"enabled\"\n ? \"Enabled — agent will follow this rule.\"\n : \"Disabled — kept but ignored.\"}\n </span>\n </div>\n\n <footer className=\"flex items-center justify-between gap-2 border-border/40 border-t pt-4\">\n <div>\n {onDelete ? (\n <Button type=\"button\" variant=\"ghost\" onClick={onDelete}>\n Delete\n </Button>\n ) : null}\n </div>\n <div className=\"flex items-center gap-2\">\n {onCancel ? (\n <Button type=\"button\" variant=\"secondary\" onClick={onCancel}>\n Cancel\n </Button>\n ) : null}\n <Button type=\"submit\" disabled={!canSave}>\n {initial?.id ? \"Save changes\" : \"Create rule\"}\n </Button>\n </div>\n </footer>\n </form>\n );\n}\n"]}
1
+ {"version":3,"sources":["../src/components/composites/rule-editor/rule-editor.tsx"],"names":[],"mappings":"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","file":"chunk-XZKEGEPT.js","sourcesContent":["import { useState } from \"react\";\nimport type { FormEvent, HTMLAttributes } from \"react\";\nimport { cn } from \"../../../lib/cn.js\";\nimport { ALL_MODES, MODE_LABEL, type Mode } from \"../../../types/mode.js\";\nimport type { Rule, RuleScope, RuleState } from \"../../../types/rule.js\";\nimport { Button } from \"../../primitives/button/index.js\";\nimport { FormField } from \"../../primitives/form-field/index.js\";\nimport { Input } from \"../../primitives/input/index.js\";\nimport { Select } from \"../../primitives/select/index.js\";\nimport { Switch } from \"../../primitives/switch/index.js\";\nimport { Textarea } from \"../../primitives/textarea/index.js\";\n\n/**\n * RuleEditor — form for creating or editing a Rule (behavior instruction\n * injected into the system prompt).\n */\n\ntype RuleDraft = Omit<Rule, \"id\" | \"updatedAt\"> & {\n id?: string;\n tags?: string[];\n};\n\ninterface RuleEditorProps extends Omit<HTMLAttributes<HTMLFormElement>, \"onSubmit\" | \"onChange\"> {\n initial?: Partial<Rule>;\n onSave: (draft: RuleDraft) => void;\n onCancel?: () => void;\n onDelete?: () => void;\n}\n\nconst SCOPES: Array<{ id: RuleScope; label: string }> = [\n { id: \"global\", label: \"Global — applies to every session\" },\n { id: \"project\", label: \"Project — only this workspace\" },\n];\n\nexport function RuleEditor({\n className,\n initial,\n onSave,\n onCancel,\n onDelete,\n ...formProps\n}: RuleEditorProps) {\n const [title, setTitle] = useState(initial?.title ?? \"\");\n const [body, setBody] = useState(initial?.body ?? \"\");\n const [scope, setScope] = useState<RuleScope>(initial?.scope ?? \"global\");\n const [tagsRaw, setTagsRaw] = useState(initial?.tags?.join(\", \") ?? \"\");\n const [enabled, setEnabled] = useState<RuleState>(initial?.state ?? \"enabled\");\n const [modes, setModes] = useState<Mode[]>(initial?.modes ?? []);\n\n // Note: state is only seeded once on mount. To reset the form when editing\n // a different rule, use the React `key` pattern at the call site:\n // <RuleEditor key={rule.id} initial={rule} ... />\n const toggleMode = (m: Mode) =>\n setModes((prev) => (prev.includes(m) ? prev.filter((x) => x !== m) : [...prev, m]));\n\n const canSave = title.trim().length > 0 && body.trim().length > 0;\n const handleSubmit = (e: FormEvent) => {\n e.preventDefault();\n if (!canSave) return;\n onSave({\n id: initial?.id,\n title: title.trim(),\n body: body.trim(),\n scope,\n state: enabled,\n tags: tagsRaw\n .split(\",\")\n .map((t) => t.trim())\n .filter(Boolean),\n modes: modes.length > 0 ? modes : undefined,\n });\n };\n\n return (\n <form\n onSubmit={handleSubmit}\n className={cn(\"flex h-full flex-col gap-4\", className)}\n {...formProps}\n >\n <FormField>\n <FormField.Label>Title</FormField.Label>\n <FormField.Control>\n <Input\n value={title}\n onChange={(e) => setTitle(e.target.value)}\n placeholder=\"Always write tests before fixes\"\n required\n />\n </FormField.Control>\n <FormField.Hint>Short, imperative summary the agent will keep in memory.</FormField.Hint>\n </FormField>\n\n <FormField className=\"flex-1\">\n <FormField.Label>Body (markdown)</FormField.Label>\n <FormField.Control>\n <Textarea\n value={body}\n onChange={(e) => setBody(e.target.value)}\n rows={8}\n placeholder=\"When fixing a bug, first write a failing regression test, then the fix.\"\n required\n className=\"min-h-[12rem] flex-1 font-mono text-code-sm\"\n />\n </FormField.Control>\n <FormField.Hint>Injected into the system prompt verbatim.</FormField.Hint>\n </FormField>\n\n <div className=\"grid grid-cols-2 gap-3\">\n <FormField>\n <FormField.Label>Scope</FormField.Label>\n <FormField.Control>\n <Select\n value={scope}\n onValueChange={(v) => {\n // Re-audit Issue 7: narrow Select string value against\n // SCOPES.id before casting. Silent no-op for unknown.\n const next = SCOPES.find((s) => s.id === v);\n if (next) setScope(next.id);\n }}\n >\n <Select.Trigger aria-label=\"Select rule scope\">\n <Select.Value />\n </Select.Trigger>\n <Select.Content>\n {SCOPES.map((s) => (\n <Select.Item key={s.id} value={s.id}>\n {s.label}\n </Select.Item>\n ))}\n </Select.Content>\n </Select>\n </FormField.Control>\n </FormField>\n <FormField>\n <FormField.Label>Tags (comma-separated)</FormField.Label>\n <FormField.Control>\n <Input\n value={tagsRaw}\n onChange={(e) => setTagsRaw(e.target.value)}\n placeholder=\"testing, process\"\n />\n </FormField.Control>\n </FormField>\n </div>\n\n <FormField>\n <FormField.Label>Active modes</FormField.Label>\n <div className=\"flex flex-wrap gap-1.5\">\n {ALL_MODES.map((m) => {\n const on = modes.includes(m);\n return (\n <button\n key={m}\n type=\"button\"\n onClick={() => toggleMode(m)}\n aria-pressed={on}\n className={cn(\n \"inline-flex h-7 items-center rounded-full border px-3 font-mono text-body-sm transition-colors\",\n on\n ? \"border-primary bg-primary/15 text-primary\"\n : \"border-border/60 bg-card text-muted-foreground hover:text-foreground\",\n )}\n >\n {MODE_LABEL[m]}\n </button>\n );\n })}\n </div>\n <FormField.Hint>\n {modes.length === 0\n ? \"Empty = global (applies to every mode).\"\n : `Only visible in: ${modes.map((m) => MODE_LABEL[m]).join(\", \")}.`}\n </FormField.Hint>\n </FormField>\n\n <div className=\"flex items-center gap-3\">\n <Switch\n checked={enabled === \"enabled\"}\n onCheckedChange={(v) => setEnabled(v ? \"enabled\" : \"disabled\")}\n aria-label=\"Enabled\"\n />\n <span className=\"text-body-sm text-muted-foreground\">\n {enabled === \"enabled\"\n ? \"Enabled — agent will follow this rule.\"\n : \"Disabled — kept but ignored.\"}\n </span>\n </div>\n\n <footer className=\"flex items-center justify-between gap-2 border-border/40 border-t pt-4\">\n <div>\n {onDelete ? (\n <Button type=\"button\" variant=\"ghost\" onClick={onDelete}>\n Delete\n </Button>\n ) : null}\n </div>\n <div className=\"flex items-center gap-2\">\n {onCancel ? (\n <Button type=\"button\" variant=\"secondary\" onClick={onCancel}>\n Cancel\n </Button>\n ) : null}\n <Button type=\"submit\" disabled={!canSave}>\n {initial?.id ? \"Save changes\" : \"Create rule\"}\n </Button>\n </div>\n </footer>\n </form>\n );\n}\n"]}