@pugi/cli 0.1.0-beta.5 → 0.1.0-beta.50

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 (263) hide show
  1. package/THIRD_PARTY_NOTICES.md +40 -0
  2. package/assets/pugi-mascot.ansi +15 -25
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/jobs-watch.js +201 -0
  6. package/dist/commands/jobs.js +15 -0
  7. package/dist/commands/smoke.js +133 -0
  8. package/dist/core/agent-progress/cleanup.js +134 -0
  9. package/dist/core/agent-progress/schema.js +144 -0
  10. package/dist/core/agent-progress/writer.js +101 -0
  11. package/dist/core/artifact-chain/dispatcher.js +148 -0
  12. package/dist/core/artifact-chain/exporter.js +164 -0
  13. package/dist/core/artifact-chain/state.js +243 -0
  14. package/dist/core/artifact-chain/steps.js +169 -0
  15. package/dist/core/auth/ensure-authenticated.js +129 -0
  16. package/dist/core/auth/env-provider.js +238 -0
  17. package/dist/core/auto-update/channels.js +122 -0
  18. package/dist/core/auto-update/checker.js +241 -0
  19. package/dist/core/auto-update/state.js +235 -0
  20. package/dist/core/bare-mode/index.js +107 -0
  21. package/dist/core/bash-classifier.js +400 -4
  22. package/dist/core/checkpoint/resumer.js +149 -0
  23. package/dist/core/checkpoint/rewinder.js +291 -0
  24. package/dist/core/codegraph/decision-store.js +248 -0
  25. package/dist/core/codegraph/detect-repo.js +459 -0
  26. package/dist/core/codegraph/install.js +134 -0
  27. package/dist/core/codegraph/offer-hook.js +220 -0
  28. package/dist/core/compact/auto-trigger.js +96 -0
  29. package/dist/core/compact/buffer-rewriter.js +115 -0
  30. package/dist/core/compact/summarizer.js +208 -0
  31. package/dist/core/compact/token-counter.js +108 -0
  32. package/dist/core/consensus/diff-capture.js +112 -3
  33. package/dist/core/context/index.js +7 -0
  34. package/dist/core/context/markdown-traverse.js +255 -0
  35. package/dist/core/cost/rate-card.js +129 -0
  36. package/dist/core/cost/tracker.js +221 -0
  37. package/dist/core/denial-tracking/index.js +8 -0
  38. package/dist/core/denial-tracking/state.js +264 -0
  39. package/dist/core/diagnostics/probe-runner.js +93 -0
  40. package/dist/core/diagnostics/probes/api.js +46 -0
  41. package/dist/core/diagnostics/probes/auth.js +86 -0
  42. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  43. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  44. package/dist/core/diagnostics/probes/config.js +72 -0
  45. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  46. package/dist/core/diagnostics/probes/disk.js +81 -0
  47. package/dist/core/diagnostics/probes/git.js +65 -0
  48. package/dist/core/diagnostics/probes/hooks.js +118 -0
  49. package/dist/core/diagnostics/probes/mcp.js +75 -0
  50. package/dist/core/diagnostics/probes/node.js +59 -0
  51. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  52. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  53. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  54. package/dist/core/diagnostics/probes/session.js +74 -0
  55. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  56. package/dist/core/diagnostics/probes/workspace.js +63 -0
  57. package/dist/core/diagnostics/types.js +70 -0
  58. package/dist/core/dispatch/cache-cleanup.js +197 -0
  59. package/dist/core/dispatch/cache-handoff.js +295 -0
  60. package/dist/core/edits/dispatch.js +218 -2
  61. package/dist/core/edits/journal.js +199 -0
  62. package/dist/core/edits/layer-d-ast.js +557 -14
  63. package/dist/core/edits/verify-hook.js +273 -0
  64. package/dist/core/edits/worktree.js +322 -0
  65. package/dist/core/engine/anvil-client.js +115 -5
  66. package/dist/core/engine/budgets.js +98 -0
  67. package/dist/core/engine/context-prefix.js +155 -0
  68. package/dist/core/engine/intent.js +260 -0
  69. package/dist/core/engine/native-pugi.js +860 -211
  70. package/dist/core/engine/prompts.js +88 -2
  71. package/dist/core/engine/strip-internal-fields.js +124 -0
  72. package/dist/core/engine/tool-bridge.js +1045 -36
  73. package/dist/core/feedback/queue.js +177 -0
  74. package/dist/core/feedback/submitter.js +145 -0
  75. package/dist/core/file-cache.js +113 -1
  76. package/dist/core/hooks/events.js +44 -0
  77. package/dist/core/hooks/index.js +15 -0
  78. package/dist/core/hooks/registry.js +213 -0
  79. package/dist/core/hooks/runner.js +236 -0
  80. package/dist/core/hooks/v2/event-emitter.js +115 -0
  81. package/dist/core/hooks/v2/executor.js +282 -0
  82. package/dist/core/hooks/v2/index.js +25 -0
  83. package/dist/core/hooks/v2/lifecycle.js +104 -0
  84. package/dist/core/hooks/v2/loader.js +216 -0
  85. package/dist/core/hooks/v2/matcher.js +125 -0
  86. package/dist/core/hooks/v2/trust.js +143 -0
  87. package/dist/core/hooks/v2/types.js +86 -0
  88. package/dist/core/lsp/cache.js +105 -0
  89. package/dist/core/lsp/client.js +776 -0
  90. package/dist/core/lsp/language-detect.js +66 -0
  91. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  92. package/dist/core/mcp/client.js +75 -6
  93. package/dist/core/mcp/http-server.js +553 -0
  94. package/dist/core/mcp/orchestrator-tools.js +662 -0
  95. package/dist/core/mcp/permission.js +190 -0
  96. package/dist/core/mcp/registry.js +24 -2
  97. package/dist/core/mcp/server-tools.js +219 -0
  98. package/dist/core/mcp/server.js +397 -0
  99. package/dist/core/memory/dual-write.js +416 -0
  100. package/dist/core/memory/phase1-kinds.js +20 -0
  101. package/dist/core/memory-sync/queue.js +158 -0
  102. package/dist/core/onboarding/ensure-initialized.js +133 -0
  103. package/dist/core/onboarding/marker.js +111 -0
  104. package/dist/core/onboarding/telemetry-state.js +108 -0
  105. package/dist/core/output-style/presets.js +176 -0
  106. package/dist/core/output-style/state.js +185 -0
  107. package/dist/core/path-security.js +284 -2
  108. package/dist/core/permissions/auto-classifier.js +124 -0
  109. package/dist/core/permissions/circuit-breaker.js +83 -0
  110. package/dist/core/permissions/gate.js +278 -0
  111. package/dist/core/permissions/index.js +20 -0
  112. package/dist/core/permissions/mode.js +174 -0
  113. package/dist/core/permissions/state.js +241 -0
  114. package/dist/core/permissions/tool-class.js +93 -0
  115. package/dist/core/prd-check/parser.js +215 -0
  116. package/dist/core/prd-check/reporter.js +127 -0
  117. package/dist/core/prd-check/session-review.js +557 -0
  118. package/dist/core/prd-check/verifiers.js +223 -0
  119. package/dist/core/pugi-md/context-injector.js +76 -0
  120. package/dist/core/pugi-md/walk-up.js +207 -0
  121. package/dist/core/release-notes/parser.js +241 -0
  122. package/dist/core/release-notes/state.js +116 -0
  123. package/dist/core/repl/history.js +11 -1
  124. package/dist/core/repl/model-pricing.js +135 -0
  125. package/dist/core/repl/session.js +1897 -37
  126. package/dist/core/repl/slash-commands.js +430 -15
  127. package/dist/core/repl/store/session-store.js +31 -2
  128. package/dist/core/repl/workspace-context.js +22 -0
  129. package/dist/core/repo-map/build.js +125 -0
  130. package/dist/core/repo-map/cache.js +185 -0
  131. package/dist/core/repo-map/extractor.js +254 -0
  132. package/dist/core/repo-map/formatter.js +145 -0
  133. package/dist/core/repo-map/scanner.js +211 -0
  134. package/dist/core/retry-budget/budget.js +284 -0
  135. package/dist/core/retry-budget/index.js +5 -0
  136. package/dist/core/session.js +92 -0
  137. package/dist/core/settings.js +80 -0
  138. package/dist/core/share/formatter.js +271 -0
  139. package/dist/core/share/redactor.js +221 -0
  140. package/dist/core/share/uploader.js +267 -0
  141. package/dist/core/skills/defaults.js +457 -0
  142. package/dist/core/smoke/headless-driver.js +174 -0
  143. package/dist/core/smoke/orchestrator.js +194 -0
  144. package/dist/core/smoke/runner.js +238 -0
  145. package/dist/core/smoke/scenario-parser.js +316 -0
  146. package/dist/core/subagents/dispatcher-real.js +600 -0
  147. package/dist/core/subagents/dispatcher.js +113 -24
  148. package/dist/core/subagents/index.js +18 -5
  149. package/dist/core/subagents/isolation-matrix.js +213 -0
  150. package/dist/core/subagents/spawn.js +19 -4
  151. package/dist/core/telemetry/emitter.js +229 -0
  152. package/dist/core/telemetry/queue.js +251 -0
  153. package/dist/core/theme/context.js +91 -0
  154. package/dist/core/theme/presets.js +228 -0
  155. package/dist/core/theme/state.js +181 -0
  156. package/dist/core/todos/invariant.js +10 -0
  157. package/dist/core/todos/state.js +177 -0
  158. package/dist/core/transport/version-interceptor.js +166 -0
  159. package/dist/core/vim/keymap.js +288 -0
  160. package/dist/core/vim/state.js +92 -0
  161. package/dist/core/worktree-manager/cleanup.js +123 -0
  162. package/dist/core/worktree-manager/manager.js +303 -0
  163. package/dist/index.js +28 -0
  164. package/dist/runtime/bootstrap.js +190 -0
  165. package/dist/runtime/cli.js +3241 -343
  166. package/dist/runtime/commands/cancel.js +231 -0
  167. package/dist/runtime/commands/chain.js +489 -0
  168. package/dist/runtime/commands/codegraph-status.js +227 -0
  169. package/dist/runtime/commands/compact.js +297 -0
  170. package/dist/runtime/commands/cost.js +199 -0
  171. package/dist/runtime/commands/delegate.js +242 -11
  172. package/dist/runtime/commands/dispatch.js +126 -0
  173. package/dist/runtime/commands/doctor.js +412 -0
  174. package/dist/runtime/commands/feedback.js +184 -0
  175. package/dist/runtime/commands/hooks.js +184 -0
  176. package/dist/runtime/commands/lsp.js +368 -0
  177. package/dist/runtime/commands/mcp.js +879 -0
  178. package/dist/runtime/commands/memory.js +508 -0
  179. package/dist/runtime/commands/model.js +237 -0
  180. package/dist/runtime/commands/onboarding.js +275 -0
  181. package/dist/runtime/commands/patch.js +128 -0
  182. package/dist/runtime/commands/permissions.js +112 -0
  183. package/dist/runtime/commands/plan.js +143 -0
  184. package/dist/runtime/commands/prd-check.js +285 -0
  185. package/dist/runtime/commands/redo-blob-store.js +92 -0
  186. package/dist/runtime/commands/redo.js +361 -0
  187. package/dist/runtime/commands/release-notes.js +229 -0
  188. package/dist/runtime/commands/repo-map.js +95 -0
  189. package/dist/runtime/commands/report.js +299 -0
  190. package/dist/runtime/commands/resume.js +118 -0
  191. package/dist/runtime/commands/review-consensus.js +17 -2
  192. package/dist/runtime/commands/rewind.js +333 -0
  193. package/dist/runtime/commands/sessions.js +163 -0
  194. package/dist/runtime/commands/share.js +316 -0
  195. package/dist/runtime/commands/status.js +186 -0
  196. package/dist/runtime/commands/stickers.js +82 -0
  197. package/dist/runtime/commands/style.js +194 -0
  198. package/dist/runtime/commands/theme.js +196 -0
  199. package/dist/runtime/commands/undo.js +32 -0
  200. package/dist/runtime/commands/update.js +289 -0
  201. package/dist/runtime/commands/vim.js +140 -0
  202. package/dist/runtime/commands/worktree.js +177 -0
  203. package/dist/runtime/commands/worktrees.js +155 -0
  204. package/dist/runtime/headless-repl.js +195 -0
  205. package/dist/runtime/headless.js +543 -0
  206. package/dist/runtime/load-hooks-or-exit.js +71 -0
  207. package/dist/runtime/plan-decompose.js +531 -0
  208. package/dist/runtime/version.js +65 -0
  209. package/dist/tools/agent-tool.js +229 -0
  210. package/dist/tools/apply-patch.js +556 -0
  211. package/dist/tools/ask-user-question.js +213 -0
  212. package/dist/tools/ask-user.js +115 -0
  213. package/dist/tools/bash.js +203 -4
  214. package/dist/tools/file-tools.js +85 -14
  215. package/dist/tools/lsp-tools.js +189 -0
  216. package/dist/tools/mcp-tool.js +260 -0
  217. package/dist/tools/multi-edit.js +361 -0
  218. package/dist/tools/powershell.js +268 -0
  219. package/dist/tools/registry.js +51 -0
  220. package/dist/tools/skill-tool.js +96 -0
  221. package/dist/tools/tasks.js +208 -0
  222. package/dist/tools/todo-write.js +184 -0
  223. package/dist/tools/web-fetch.js +147 -2
  224. package/dist/tools/web-search.js +458 -0
  225. package/dist/tui/agent-progress-card.js +111 -0
  226. package/dist/tui/agent-tree.js +10 -0
  227. package/dist/tui/ask-modal.js +2 -2
  228. package/dist/tui/ask-user-question-prompt.js +192 -0
  229. package/dist/tui/compact-banner.js +81 -0
  230. package/dist/tui/conversation-pane.js +82 -8
  231. package/dist/tui/cost-table.js +111 -0
  232. package/dist/tui/doctor-table.js +46 -0
  233. package/dist/tui/feedback-prompt.js +156 -0
  234. package/dist/tui/input-box.js +218 -3
  235. package/dist/tui/markdown-render.js +4 -4
  236. package/dist/tui/onboarding-wizard.js +240 -0
  237. package/dist/tui/permissions-picker.js +86 -0
  238. package/dist/tui/render.js +35 -0
  239. package/dist/tui/repl-render.js +313 -35
  240. package/dist/tui/repl-splash-art.js +1 -1
  241. package/dist/tui/repl-splash-mascot.js +32 -8
  242. package/dist/tui/repl-splash.js +2 -2
  243. package/dist/tui/repl.js +85 -5
  244. package/dist/tui/splash.js +1 -1
  245. package/dist/tui/status-bar.js +94 -16
  246. package/dist/tui/status-table.js +7 -0
  247. package/dist/tui/stickers-art.js +136 -0
  248. package/dist/tui/style-table.js +28 -0
  249. package/dist/tui/theme-table.js +29 -0
  250. package/dist/tui/thinking-spinner.js +123 -0
  251. package/dist/tui/tool-stream-pane.js +52 -3
  252. package/dist/tui/update-banner.js +27 -2
  253. package/dist/tui/vim-input.js +267 -0
  254. package/dist/tui/welcome-banner.js +107 -0
  255. package/dist/tui/welcome-data.js +293 -0
  256. package/docs/examples/codegraph.mcp.json +10 -0
  257. package/package.json +12 -6
  258. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  259. package/test/scenarios/compact-force.scenario.txt +11 -0
  260. package/test/scenarios/identity.scenario.txt +11 -0
  261. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  262. package/test/scenarios/walkback.scenario.txt +12 -0
  263. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,123 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Thinking spinner — animated dispatch indicator (CEO P0 #2, 2026-05-29).
4
+ *
5
+ * Replaces the static `dispatching` / `tool: <name>` strings the bottom
6
+ * status row used к paint during active brief dispatch with а Claude-
7
+ * Code-style rotating verb + glyph pair:
8
+ *
9
+ * ✽ Briefing… (awaiting_response, before the first tool call)
10
+ * ◆ Dispatching… (awaiting_response, после the first tool call)
11
+ * ◇ Reviewing… (tool_running)
12
+ * ✦ Synthesizing… (awaiting_response, late in turn)
13
+ * ❋ Shipping… (tool_running, edit/write/build family tool)
14
+ *
15
+ * The rotation cadence is ~800 ms per step so the verb feels alive but
16
+ * never strobes. The component owns its own timer (cleared on unmount)
17
+ * и takes the active dispatch state + the optional tool label so the
18
+ * "Shipping…" branch can light up specifically для write-class tools.
19
+ *
20
+ * Mount lifecycle: the REPL renders `<ThinkingSpinner />` only while
21
+ * `dispatchState` ∈ {`awaiting_response`, `tool_running`}. The
22
+ * status-bar's legacy `composeStatusLabel` row stays mounted too — they
23
+ * are not mutually exclusive — but the spinner row sits above the
24
+ * status row так the operator's eye lands on the live signal first.
25
+ *
26
+ * Brand discipline:
27
+ * - Verbs from the approved power-words list: `Briefing`,
28
+ * `Dispatching`, `Reviewing`, `Synthesizing`, `Shipping`. No
29
+ * `Thinking…` (forbidden cool word per DESIGN.md §3.2 voice gate).
30
+ * - Glyphs from the Pugi mascot brand glyph kit (`✽ ◆ ◇ ✦ ❋`). No
31
+ * emoji — the spinner must paint in а narrow Bash / WSL terminal
32
+ * где emoji fall back к `?`.
33
+ * - Cyan accent (`#3da9fc`) on the glyph; verb stays default-tone so
34
+ * the rotation does not compete with the cost-meter row above.
35
+ * - Trailing ellipsis is а single Unicode `…` (DESIGN.md §4 — three-
36
+ * dot ellipsis is the ONLY allowed truncation glyph).
37
+ *
38
+ * Test surface: `tickSpinnerFrame` is exported so the spec can drive
39
+ * the frame index without а real-clock timer.
40
+ */
41
+ import { useEffect, useState } from 'react';
42
+ import { Box, Text } from 'ink';
43
+ /* ------------------------------------------------------------------ */
44
+ /* Constants */
45
+ /* ------------------------------------------------------------------ */
46
+ const ACCENT = '#3da9fc';
47
+ const FRAME_INTERVAL_MS = 800;
48
+ const FRAMES = [
49
+ { glyph: '✽', verb: 'Briefing' },
50
+ { glyph: '◆', verb: 'Dispatching' },
51
+ { glyph: '◇', verb: 'Reviewing' },
52
+ { glyph: '✦', verb: 'Synthesizing' },
53
+ { glyph: '❋', verb: 'Shipping' },
54
+ ];
55
+ /**
56
+ * Tool labels that anchor к the `Shipping…` frame regardless of
57
+ * rotation index. These are the write-class tools — the operator sees
58
+ * "shipping" the moment а real file mutation fires так the indicator
59
+ * matches the visible filesystem effect.
60
+ */
61
+ const SHIPPING_TOOLS = new Set([
62
+ 'edit',
63
+ 'write',
64
+ 'build',
65
+ 'multi_edit',
66
+ 'apply_patch',
67
+ ]);
68
+ /* ------------------------------------------------------------------ */
69
+ /* Helpers */
70
+ /* ------------------------------------------------------------------ */
71
+ /**
72
+ * Compute the visible frame для the current tick. The base index is
73
+ * derived from а monotonically-incrementing counter (one increment per
74
+ * `FRAME_INTERVAL_MS` window), и а ship-class tool label overrides the
75
+ * base so write-tools light up `Shipping…` immediately.
76
+ *
77
+ * Exported for the spec so the frame selection is testable without
78
+ * mounting Ink.
79
+ */
80
+ export function tickSpinnerFrame(baseIndex, toolLabel) {
81
+ if (toolLabel && SHIPPING_TOOLS.has(toolLabel.toLowerCase())) {
82
+ return FRAMES[FRAMES.length - 1] ?? FRAMES[0];
83
+ }
84
+ const wrapped = ((baseIndex % FRAMES.length) + FRAMES.length) % FRAMES.length;
85
+ return FRAMES[wrapped] ?? FRAMES[0];
86
+ }
87
+ /**
88
+ * Decide whether the spinner should be visible for а dispatch state.
89
+ * `awaiting_response` и `tool_running` are the only active states; all
90
+ * other states (`idle`, `aborting`, `aborted`, `completed`, `failed`)
91
+ * suppress the spinner так the operator does not see а phantom verb
92
+ * after the dispatch settles.
93
+ */
94
+ export function isSpinnerActive(dispatchState) {
95
+ return dispatchState === 'awaiting_response' || dispatchState === 'tool_running';
96
+ }
97
+ /* ------------------------------------------------------------------ */
98
+ /* Component */
99
+ /* ------------------------------------------------------------------ */
100
+ export function ThinkingSpinner(props) {
101
+ const active = isSpinnerActive(props.dispatchState);
102
+ const [frameIndex, setFrameIndex] = useState(0);
103
+ const intervalMs = props.frameIntervalMs ?? FRAME_INTERVAL_MS;
104
+ useEffect(() => {
105
+ if (!active) {
106
+ // Reset так the next dispatch starts on `Briefing` instead of
107
+ // mid-rotation. Without this the spinner would resume one verb
108
+ // later on every turn и quickly drift to `Synthesizing` even на
109
+ // а fresh brief.
110
+ setFrameIndex(0);
111
+ return;
112
+ }
113
+ const timer = setInterval(() => {
114
+ setFrameIndex((previous) => (previous + 1) % FRAMES.length);
115
+ }, intervalMs);
116
+ return () => clearInterval(timer);
117
+ }, [active, intervalMs]);
118
+ if (!active)
119
+ return null;
120
+ const frame = tickSpinnerFrame(frameIndex, props.dispatchToolLabel ?? null);
121
+ return (_jsxs(Box, { children: [_jsx(Text, { color: ACCENT, children: `${frame.glyph} ` }), _jsx(Text, { children: `${frame.verb}…` })] }));
122
+ }
123
+ //# sourceMappingURL=thinking-spinner.js.map
@@ -1,5 +1,6 @@
1
1
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
2
  import { Box, Text } from 'ink';
3
+ import { RESULT_PREVIEW_MAX_CHARS, STREAMING_DELTA_MAX_CHARS, } from '../core/repl/session.js';
3
4
  const DEFAULT_COLLAPSE_THRESHOLD = 5;
4
5
  const DEFAULT_MAX_ROWS = 8;
5
6
  export function ToolStreamPane(props) {
@@ -28,7 +29,38 @@ function ToolCallRow({ call, collapseThreshold, }) {
28
29
  const label = formatToolLabel(call.tool, call.args);
29
30
  const summary = formatSummary(call);
30
31
  const showHint = (call.resultLines ?? 0) > collapseThreshold;
31
- return (_jsxs(Box, { children: [_jsx(Text, { color: color, children: glyph }), _jsx(Text, { children: ' ' }), _jsx(Text, { bold: true, children: label }), _jsx(Text, { dimColor: true, children: ` ${summary}` }), showHint ? (_jsx(Text, { dimColor: true, children: ` · ${call.resultLines} lines, Ctrl+O to expand` })) : null] }));
32
+ // Wave 6 small-CC-parity batch (2026-05-27): on a `running` row,
33
+ // surface the rolling streaming delta as a dim inline preview after
34
+ // the label. On a completed row, the same slot carries the
35
+ // `resultPreview` quoted head. Either way the row stays single-line —
36
+ // both fields are clamped к their respective char ceilings upstream.
37
+ const inlineTail = call.status === 'running'
38
+ ? call.streamingDelta
39
+ : call.resultPreview;
40
+ // Error rows: render the label too in red so the eye lands on the
41
+ // failure even при peripheral attention. The other states keep the
42
+ // glyph as the only color signal.
43
+ const labelColor = call.status === 'error' ? 'red' : undefined;
44
+ return (_jsxs(Box, { children: [_jsx(Text, { color: color, children: glyph }), _jsx(Text, { children: ' ' }), _jsx(Text, { bold: true, color: labelColor, children: label }), _jsx(Text, { dimColor: true, children: ` ${summary}` }), inlineTail ? (_jsx(Text, { dimColor: true, children: ` ${formatInlineTail(call.status, inlineTail)}` })) : null, showHint ? (_jsx(Text, { dimColor: true, children: ` · ${call.resultLines} lines, Ctrl+O to expand` })) : null] }));
45
+ }
46
+ /**
47
+ * Wave 6 small-CC-parity batch (2026-05-27): render the inline tail
48
+ * slot for either the live `streamingDelta` (during `running`) or the
49
+ * collapsed `resultPreview` (after completion). Pure helper so the
50
+ * spec can assert the exact shape.
51
+ *
52
+ * running → `… npm WARN deprecated…`
53
+ * ok / error → `"<preview head>"`
54
+ */
55
+ export function formatInlineTail(status, tail) {
56
+ if (status === 'running') {
57
+ return tail;
58
+ }
59
+ // Wrap the completed preview in quotes so the operator's eye groups
60
+ // the preview block visually distinct from the canonical detail
61
+ // (`OK`, `+12 -0`). Mirrors the Claude Code TUI's quoted-preview
62
+ // pattern.
63
+ return `"${tail}"`;
32
64
  }
33
65
  function statusGlyph(status) {
34
66
  switch (status) {
@@ -52,18 +84,35 @@ function statusColor(status) {
52
84
  }
53
85
  /**
54
86
  * Render the canonical `Tool(args)` form. Tool names are capitalised
55
- * the way Claude Code shows them; args are truncated to 60 chars so
56
- * the row stays single-line even on 80-col terminals.
87
+ * the way Claude Code shows them; args are truncated to keep the row
88
+ * single-line even on 80-col terminals. Cap = 60 chars (chosen empirically
89
+ * to leave room for the glyph + 1-space gap + summary + optional
90
+ * inline tail without overflow on narrow shells).
57
91
  */
58
92
  function formatToolLabel(tool, args) {
59
93
  const name = toolDisplayName(tool);
60
94
  const trimmedArgs = args.length > 60 ? `${args.slice(0, 57)}…` : args;
61
95
  return `${name}(${trimmedArgs})`;
62
96
  }
97
+ /**
98
+ * Re-export the upstream char caps so the operator-facing constants
99
+ * have one source of truth. The session module owns the canonical
100
+ * values (used both during ingest и during the pane's render lookup);
101
+ * tests assert against these names rather than literal numbers so a
102
+ * future tuning сtays diff-friendly.
103
+ */
104
+ export { RESULT_PREVIEW_MAX_CHARS, STREAMING_DELTA_MAX_CHARS };
63
105
  function toolDisplayName(tool) {
64
106
  switch (tool) {
65
107
  case 'read':
66
108
  return 'Read';
109
+ case 'write':
110
+ // 2026-05-27 — Write is the most operator-visible tool for the
111
+ // codegen-dispatch surface (Hiroshi writing index.html / style.css
112
+ // / script.js for a tic-tac-toe brief). Add the display name so
113
+ // the tool stream pane renders ✓ Write(index.html) instead of an
114
+ // unlabeled placeholder. Mirrors the Claude Code Write rendering.
115
+ return 'Write';
67
116
  case 'edit':
68
117
  return 'Edit';
69
118
  case 'bash':
@@ -1,8 +1,33 @@
1
1
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
2
  import { Box, Text } from 'ink';
3
- import { upgradeCommand } from '../runtime/update-check.js';
3
+ import { compareVersions, upgradeCommand, } from '../runtime/update-check.js';
4
+ import { getCachedServerRecommendation } from '../core/transport/version-interceptor.js';
5
+ /**
6
+ * Resolve the `latest` value the banner should show. Exported so the
7
+ * spec can lock the merge logic without rendering Ink.
8
+ */
9
+ export function resolveDisplayedLatest(npmLatest, serverRecommended) {
10
+ if (!serverRecommended)
11
+ return npmLatest;
12
+ return compareVersions(serverRecommended, npmLatest) > 0
13
+ ? serverRecommended
14
+ : npmLatest;
15
+ }
16
+ /**
17
+ * Claude-Code-style corner banner: single line, dim orange, right
18
+ * aligned. Renders to the bottom of the REPL frame so it never
19
+ * displaces conversation content. Only mounts when an update is
20
+ * actually available; identical to no-op when the registry poll
21
+ * reports current. Suppress with `PUGI_SKIP_UPDATE_BANNER=1`.
22
+ */
4
23
  export function UpdateBanner({ result }) {
5
24
  const command = upgradeCommand(result.method);
6
- return (_jsxs(Box, { flexDirection: "column", marginBottom: 1, children: [_jsxs(Box, { children: [_jsx(Text, { dimColor: true, children: '─ ' }), _jsx(Text, { bold: true, color: "cyan", children: 'Pugi ' }), _jsx(Text, { children: result.installed }), _jsx(Text, { dimColor: true, children: ' (installed) → ' }), _jsx(Text, { bold: true, children: result.latest }), _jsx(Text, { dimColor: true, children: ' (latest)' })] }), _jsxs(Box, { children: [_jsx(Text, { dimColor: true, children: ' Update: ' }), _jsx(Text, { color: "cyan", children: command })] }), _jsx(Box, { children: _jsx(Text, { dimColor: true, children: ' Skip with PUGI_SKIP_UPDATE_BANNER=1' }) })] }));
25
+ // Read the cache lazily inside the render so a server response that
26
+ // landed AFTER the banner was constructed still shows up on the next
27
+ // re-render. The cache lookup is a single map read — cheap enough to
28
+ // do per render.
29
+ const serverRecommended = getCachedServerRecommendation();
30
+ const displayedLatest = resolveDisplayedLatest(result.latest, serverRecommended);
31
+ return (_jsx(Box, { justifyContent: "flex-end", children: _jsxs(Text, { color: "#d97706", children: [_jsx(Text, { dimColor: true, children: 'Update available! ' }), _jsx(Text, { dimColor: true, children: `v${displayedLatest} — Run: ` }), _jsx(Text, { bold: true, children: command })] }) }));
7
32
  }
8
33
  //# sourceMappingURL=update-banner.js.map
@@ -0,0 +1,267 @@
1
+ import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Leak L26 (2026-05-27) — Vim-mode-aware REPL input.
4
+ *
5
+ * Thin wrapper that sits BETWEEN the REPL session and the legacy
6
+ * `InputBox`. When vim mode is off it forwards every prop unchanged
7
+ * so existing operators see zero behavioural delta. When vim mode is
8
+ * on it intercepts keystrokes through `useInput` BEFORE Ink's normal
9
+ * dispatch and either:
10
+ *
11
+ * - in `insert` mode, lets the keystroke fall through to `InputBox`
12
+ * so the legacy buffer / cursor / history / palette code stays
13
+ * authoritative (we do NOT re-implement insert-mode editing);
14
+ * - in `normal` mode, routes the key through `core/vim/keymap.ts`
15
+ * and applies the result to a shadow buffer + cursor mirror that
16
+ * it then pushes back into `InputBox` via the existing `initial`
17
+ * prop on remount.
18
+ *
19
+ * Why a wrapper instead of inlining into `InputBox`?
20
+ *
21
+ * - Keeps the legacy input surface untouched for non-vim operators
22
+ * (the L26 ship risks zero regression to ~year-old code).
23
+ * - The wrapper renders a thin status row ABOVE the input frame so
24
+ * the active mode + pending sequence + cheat sheet are visible.
25
+ * - Tests can exercise the wrapper without dragging in the full
26
+ * ink + clipboard stack of `InputBox` (the spec drives the
27
+ * keymap directly; the wrapper is exercised via the runtime
28
+ * command tests + manual smoke from the REPL).
29
+ *
30
+ * The component intentionally only models the SINGLE-LINE REPL prompt
31
+ * buffer — that is the surface Claude Code's `/vim` covers, and that
32
+ * is what the leak research validated. Multi-line + visual-mode +
33
+ * counts are out of scope for this sprint.
34
+ *
35
+ * ─── Closure-staleness contract (post-L26 fix, 2026-05-27) ───
36
+ *
37
+ * Three-of-three reviewers flagged the original `setShadowLine`
38
+ * functional updater for reading `shadowCursor` and `pending` from
39
+ * the closure, which goes stale across consecutive keystrokes
40
+ * (React batches dispatch, so the second `d` of `dd` saw the
41
+ * `pending` value from the render that scheduled the first `d`,
42
+ * not the post-first-d value). Same problem for cursor advancement
43
+ * across chained `l`/`h`/`w`/`b` presses.
44
+ *
45
+ * Additionally, calling `props.onSubmit` inline inside a setState
46
+ * updater is double-fired by React strict mode (updaters MUST be
47
+ * pure — strict mode runs them twice to surface impurity).
48
+ *
49
+ * The fix moves shadow cursor + pending + mode into `useRef`
50
+ * (refs survive across renders, are read synchronously, and are
51
+ * not subject to closure capture), drives the keymap dispatch
52
+ * OUTSIDE any setState callback, and defers `props.onSubmit` /
53
+ * `props.onExit` invocations until AFTER the render via the
54
+ * `useEffect` queued by toggling a transition ref.
55
+ *
56
+ * The `line` state is the only React state we mutate per keystroke
57
+ * because the inner `InputBox` re-reads it via the `initial` prop
58
+ * on remount — refs alone cannot trigger that remount. Cursor + mode
59
+ * + pending are surfaced to the render path through the same
60
+ * `line`/`tick` rerender, so the status bar stays in sync without
61
+ * itself being captured by a stale closure.
62
+ */
63
+ import { useEffect, useRef, useState } from 'react';
64
+ import { Box, Text, useInput } from 'ink';
65
+ import { handleNormalKey, PENDING_NONE, describePending, } from '../core/vim/keymap.js';
66
+ import { InputBox } from './input-box.js';
67
+ /**
68
+ * Render the mode badge + pending sequence + cheat sheet above the
69
+ * input frame. Two lines:
70
+ *
71
+ * ─ NORMAL ─ d: Esc=normal · i=insert · :w=submit · :q=cancel
72
+ * ─ INSERT ─ (mode-specific tail dropped when there's nothing to show)
73
+ *
74
+ * Plain ASCII + dim accents to match the rest of the REPL's chrome.
75
+ */
76
+ function VimStatusBar(props) {
77
+ const modeLabel = props.mode === 'normal' ? '-- NORMAL --' : '-- INSERT --';
78
+ const pendingLabel = describePending(props.pending);
79
+ const hint = props.mode === 'normal'
80
+ ? 'h/j/k/l move · i/a insert · x del · dd line · :w submit · :q cancel'
81
+ : 'Esc → normal mode';
82
+ return (_jsxs(Box, { children: [_jsx(Text, { color: "#3da9fc", bold: true, children: modeLabel }), pendingLabel.length > 0 ? (_jsxs(_Fragment, { children: [_jsx(Text, { children: " " }), _jsx(Text, { color: "yellow", children: `${pendingLabel}` })] })) : null, _jsx(Text, { children: " " }), _jsx(Text, { dimColor: true, children: hint })] }));
83
+ }
84
+ export function VimInput(props) {
85
+ const { vimEnabled, initialMode, ...inputBoxProps } = props;
86
+ // When vim mode is off we DO NOT mount the `useInput` overlay — the
87
+ // legacy `InputBox` keeps full ownership of the keystrokes and
88
+ // behaves byte-for-byte the way it always did. This is the same
89
+ // pattern `output-style` followed (off → pass through; on → activate
90
+ // the modal surface).
91
+ if (!vimEnabled) {
92
+ return _jsx(InputBox, { ...inputBoxProps });
93
+ }
94
+ // ─── Render-driving state ─────────────────────────────────────
95
+ //
96
+ // `line` + `tick` are the only pieces of state the render path
97
+ // reads — the inner `InputBox` is remounted on every `tick` bump
98
+ // so the `initial` prop is honoured. Cursor / pending / mode are
99
+ // held in refs (see below) and mirrored into render state via the
100
+ // same setLine/setTick burst at the end of each keystroke so the
101
+ // status bar stays in lockstep without ever being captured by a
102
+ // stale closure inside a setState updater.
103
+ const [line, setLine] = useState(props.initial ?? '');
104
+ const [mode, setMode] = useState(initialMode ?? 'normal');
105
+ const [pending, setPending] = useState(PENDING_NONE);
106
+ const [remountTick, setRemountTick] = useState(0);
107
+ // ─── Closure-safe scratch refs ────────────────────────────────
108
+ //
109
+ // These refs are written synchronously inside the `useInput`
110
+ // handler and read on the NEXT keystroke. They survive across
111
+ // renders, are not captured by stale closures, and are NEVER
112
+ // mutated inside a setState updater — so React strict-mode's
113
+ // double-invocation cannot corrupt them.
114
+ //
115
+ // Why not `useReducer`? A reducer would also see fresh state on
116
+ // every dispatch, but the keymap result is a tagged union whose
117
+ // side effects (mode flip, submit, cancel, remount) are easier
118
+ // to read as imperative steps after the pure `handleNormalKey`
119
+ // call. Refs keep the dispatcher linear and the diff vs the
120
+ // pre-fix file minimal.
121
+ const lineRef = useRef(props.initial ?? '');
122
+ const cursorRef = useRef(props.initial?.length ?? 0);
123
+ const pendingRef = useRef(PENDING_NONE);
124
+ const modeRef = useRef(initialMode ?? 'normal');
125
+ // Deferred side-effect queue — submit / exit must NOT run inline
126
+ // inside a setState updater (strict mode double-invokes them).
127
+ // We stash the payload here and flush it from a `useEffect` after
128
+ // the render commits.
129
+ const pendingSubmitRef = useRef(null);
130
+ const pendingExitRef = useRef(false);
131
+ const [sideEffectTick, setSideEffectTick] = useState(0);
132
+ useEffect(() => {
133
+ // Flush any deferred host callback exactly once per scheduled
134
+ // burst. The refs are nulled BEFORE the call so a re-entrant
135
+ // host onSubmit handler that re-renders us cannot re-fire the
136
+ // same payload.
137
+ const submit = pendingSubmitRef.current;
138
+ if (submit !== null) {
139
+ pendingSubmitRef.current = null;
140
+ props.onSubmit(submit);
141
+ }
142
+ if (pendingExitRef.current) {
143
+ pendingExitRef.current = false;
144
+ props.onExit();
145
+ }
146
+ // Intentionally only depends on `sideEffectTick`: we want this
147
+ // effect to fire ONLY when the keystroke handler asked for it
148
+ // (via setSideEffectTick), not on every prop change.
149
+ // eslint-disable-next-line react-hooks/exhaustive-deps
150
+ }, [sideEffectTick]);
151
+ useInput((input, key) => {
152
+ // Esc always returns to normal mode (and clears any pending
153
+ // sequence). This binding wins over `InputBox`'s own Esc handling
154
+ // because we own the `useInput` hook earlier in the render tree.
155
+ if (key.escape) {
156
+ modeRef.current = 'normal';
157
+ pendingRef.current = PENDING_NONE;
158
+ setMode('normal');
159
+ setPending(PENDING_NONE);
160
+ return;
161
+ }
162
+ // In insert mode we relinquish dispatch entirely so the legacy
163
+ // input box owns typing / palette / history / clipboard / kill
164
+ // ring without interference. Returning `undefined` from a
165
+ // `useInput` callback is a no-op — InputBox's own `useInput` will
166
+ // receive the same event on the next frame.
167
+ if (modeRef.current === 'insert')
168
+ return;
169
+ // In normal mode we drive the buffer through the keymap.
170
+ //
171
+ // CRITICAL: all inputs to `handleNormalKey` are read from refs,
172
+ // not from the React state closure captured at render time.
173
+ // This is what makes consecutive keystrokes ("ll", "dd", ":w")
174
+ // observe the post-previous-keystroke state instead of the
175
+ // pre-batch render snapshot.
176
+ const ch = input ?? '';
177
+ const curLine = lineRef.current;
178
+ const curCursor = cursorRef.current;
179
+ const curPending = pendingRef.current;
180
+ const out = handleNormalKey({
181
+ line: curLine,
182
+ cursor: curCursor,
183
+ pending: curPending,
184
+ ch,
185
+ enter: key.return,
186
+ escape: false,
187
+ backspace: key.backspace || key.delete,
188
+ });
189
+ // Step 1: write the new pending state to BOTH the ref (read by
190
+ // the next keystroke synchronously) AND the React state (drives
191
+ // the status bar render).
192
+ pendingRef.current = out.pending;
193
+ setPending(out.pending);
194
+ // Step 2: apply the discriminated result. Each branch updates
195
+ // the refs first (synchronous, closure-safe) and then schedules
196
+ // any required React state update or deferred side effect.
197
+ switch (out.result.kind) {
198
+ case 'move': {
199
+ cursorRef.current = out.result.cursor;
200
+ // Cursor lives in the ref; the inner InputBox does not need
201
+ // a remount for a pure motion, so we skip the tick bump.
202
+ break;
203
+ }
204
+ case 'edit': {
205
+ cursorRef.current = out.result.cursor;
206
+ lineRef.current = out.result.line;
207
+ setLine(out.result.line);
208
+ setRemountTick((t) => t + 1);
209
+ break;
210
+ }
211
+ case 'mode': {
212
+ cursorRef.current = out.result.cursor;
213
+ modeRef.current = out.result.mode;
214
+ setMode(out.result.mode);
215
+ break;
216
+ }
217
+ case 'submit': {
218
+ // Forward to the host's onSubmit and clear the shadow buffer
219
+ // so the next prompt starts empty. We DO NOT re-route
220
+ // through InputBox's Enter handler — that path also writes
221
+ // history; submitting from `:w` should mirror that, so we
222
+ // call props.onSubmit indirectly via the deferred-effect
223
+ // queue. (The host's history append happens inside InputBox;
224
+ // for the modal path the host can observe via props.onSubmit
225
+ // and append manually if needed. The leak-parity surface for
226
+ // L26 documents `:w` as equivalent to Enter so this is fine.)
227
+ const payload = out.result.payload;
228
+ if (payload.trim().length > 0) {
229
+ // Queue the host callback; useEffect flushes it after the
230
+ // render commits. This avoids React strict-mode's
231
+ // double-invocation of setState updaters double-firing
232
+ // onSubmit (which would, e.g., submit the same prompt
233
+ // twice to the agent on every `:w`).
234
+ pendingSubmitRef.current = payload;
235
+ }
236
+ cursorRef.current = 0;
237
+ lineRef.current = '';
238
+ setLine('');
239
+ setRemountTick((t) => t + 1);
240
+ setSideEffectTick((t) => t + 1);
241
+ break;
242
+ }
243
+ case 'cancel': {
244
+ cursorRef.current = 0;
245
+ lineRef.current = '';
246
+ setLine('');
247
+ setRemountTick((t) => t + 1);
248
+ break;
249
+ }
250
+ case 'noop': {
251
+ // No-op result still may have advanced `pending` (e.g. the
252
+ // first `d` of `dd` arms the pending state). The pending ref
253
+ // and state were already updated above, so nothing else to
254
+ // do here.
255
+ break;
256
+ }
257
+ }
258
+ });
259
+ return (_jsxs(Box, { flexDirection: "column", children: [_jsx(VimStatusBar, { mode: mode, pending: pending }), _jsx(InputBox, { ...inputBoxProps,
260
+ // In normal mode we seed the (possibly mutated) shadow line +
261
+ // suppress the blink so the operator's caret is unambiguously
262
+ // controlled by h/l/0/$/w/b. In insert mode we hand the
263
+ // buffer back to InputBox as-is so the legacy typing path
264
+ // takes over.
265
+ initial: mode === 'normal' ? line : (props.initial ?? line), blinkCursor: mode === 'normal' ? false : (props.blinkCursor ?? true) }, `vim-${remountTick}`)] }));
266
+ }
267
+ //# sourceMappingURL=vim-input.js.map
@@ -0,0 +1,107 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { Box, Text } from 'ink';
3
+ import { PUG_MASCOT, PUG_MASCOT_CYAN_MASK, PUG_MASCOT_MAX_WIDTH, } from './repl-splash-art.js';
4
+ /* ------------------------------------------------------------------ */
5
+ /* Layout constants */
6
+ /* ------------------------------------------------------------------ */
7
+ /** Default left-column width when the parent box doesn't pin one. */
8
+ const LEFT_COLUMN_WIDTH = 44;
9
+ /** Default right-column width. */
10
+ const RIGHT_COLUMN_WIDTH = 44;
11
+ const ACCENT = '#3da9fc';
12
+ const BULLET = '·';
13
+ /* ------------------------------------------------------------------ */
14
+ /* Component */
15
+ /* ------------------------------------------------------------------ */
16
+ export function WelcomeBanner(props) {
17
+ const { data } = props;
18
+ const showHandCraftedMascot = props.mascotPrePrinted !== true;
19
+ const initToast = props.autoInitStatus === 'initialized'
20
+ ? 'Pugi workspace initialised at .pugi/.'
21
+ : null;
22
+ const accountLine = formatAccountLine(data);
23
+ const modelLine = formatModelLine(data);
24
+ return (_jsx(Box, { flexDirection: "column", children: _jsxs(Box, { borderStyle: "round", borderColor: ACCENT, paddingX: 1, flexDirection: "row", children: [_jsx(LeftColumn, { greetingName: data.greetingName, cwd: data.cwd, modelLine: modelLine, accountLine: accountLine, cliVersion: data.cliVersion, showHandCraftedMascot: showHandCraftedMascot, initToast: initToast }), _jsx(Box, { width: 2 }), _jsx(RightColumn, { whatsNew: data.whatsNew })] }) }));
25
+ }
26
+ /* ------------------------------------------------------------------ */
27
+ /* Left column — greeting + mascot + status */
28
+ /* ------------------------------------------------------------------ */
29
+ function LeftColumn({ greetingName, cwd, modelLine, accountLine, cliVersion, showHandCraftedMascot, initToast, }) {
30
+ return (_jsxs(Box, { flexDirection: "column", width: LEFT_COLUMN_WIDTH, children: [_jsxs(Box, { children: [_jsx(Text, { bold: true, children: "Pugi" }), _jsx(Text, { bold: true, color: ACCENT, children: ".io" }), _jsx(Text, { dimColor: true, children: ` v${cliVersion}` })] }), _jsx(Box, { marginTop: 1, children: _jsx(Text, { children: `Welcome back ${greetingName}!` }) }), showHandCraftedMascot ? (_jsx(Box, { marginTop: 1, children: _jsx(MascotColumn, {}) })) : null, _jsxs(Box, { marginTop: 1, flexDirection: "column", children: [_jsx(Text, { children: modelLine }), _jsx(Text, { dimColor: true, children: accountLine }), _jsx(Text, { dimColor: true, children: truncatePath(cwd, LEFT_COLUMN_WIDTH - 2) })] }), initToast ? (_jsx(Box, { marginTop: 1, children: _jsx(Text, { color: ACCENT, children: initToast }) })) : null] }));
31
+ }
32
+ /* ------------------------------------------------------------------ */
33
+ /* Right column — tips + what's new */
34
+ /* ------------------------------------------------------------------ */
35
+ function RightColumn({ whatsNew, }) {
36
+ return (_jsxs(Box, { flexDirection: "column", width: RIGHT_COLUMN_WIDTH, children: [_jsx(Text, { dimColor: true, children: "Tips for getting started" }), _jsxs(Box, { flexDirection: "column", children: [_jsx(TipRow, { text: "Run /init to scaffold PUGI.md instructions" }), _jsx(TipRow, { text: "Brief Pugi \u2014 the workforce dispatches" }), _jsx(TipRow, { text: "Triple-review gate before push: /review --triple" }), _jsx(TipRow, { text: "/help for every slash command" })] }), _jsx(Box, { marginTop: 1, children: _jsx(Text, { dimColor: true, children: '─'.repeat(Math.min(RIGHT_COLUMN_WIDTH - 2, 18)) }) }), _jsx(Box, { marginTop: 1, children: _jsx(Text, { dimColor: true, children: "What's new" }) }), _jsx(Box, { flexDirection: "column", children: whatsNew.length > 0 ? (whatsNew.map((line, index) => (_jsx(TipRow, { text: line }, `whatsnew-${index}`)))) : (_jsx(Text, { dimColor: true, children: ` ${BULLET} /release-notes for the full changelog` })) })] }));
37
+ }
38
+ function TipRow({ text }) {
39
+ return (_jsxs(Text, { children: [_jsx(Text, { color: ACCENT, children: ` ${BULLET} ` }), _jsx(Text, { children: text })] }));
40
+ }
41
+ /* ------------------------------------------------------------------ */
42
+ /* Mascot column (hand-crafted ASCII fallback path) */
43
+ /* ------------------------------------------------------------------ */
44
+ function MascotColumn() {
45
+ return (_jsx(Box, { flexDirection: "column", minWidth: PUG_MASCOT_MAX_WIDTH, children: PUG_MASCOT.map((row, rowIndex) => (_jsx(MascotRow, { row: row, mask: PUG_MASCOT_CYAN_MASK[rowIndex] ?? [] }, `mascot-row-${rowIndex}`))) }));
46
+ }
47
+ function MascotRow({ row, mask, }) {
48
+ // Split into contiguous same-color runs so we emit one <Text> per
49
+ // run instead of one per character (keeps the Ink tree shallow и
50
+ // the snapshot diff readable).
51
+ const runs = [];
52
+ let buffer = '';
53
+ let bufferCyan = false;
54
+ for (let column = 0; column < row.length; column += 1) {
55
+ const ch = row.charAt(column);
56
+ const cyan = mask[column] === true;
57
+ if (buffer.length === 0) {
58
+ buffer = ch;
59
+ bufferCyan = cyan;
60
+ continue;
61
+ }
62
+ if (cyan === bufferCyan) {
63
+ buffer += ch;
64
+ }
65
+ else {
66
+ runs.push({ text: buffer, cyan: bufferCyan });
67
+ buffer = ch;
68
+ bufferCyan = cyan;
69
+ }
70
+ }
71
+ if (buffer.length > 0)
72
+ runs.push({ text: buffer, cyan: bufferCyan });
73
+ return (_jsx(Text, { children: runs.map((run, runIndex) => run.cyan ? (_jsx(Text, { color: ACCENT, children: run.text }, runIndex)) : (_jsx(Text, { color: "gray", children: run.text }, runIndex))) }));
74
+ }
75
+ /* ------------------------------------------------------------------ */
76
+ /* Formatters */
77
+ /* ------------------------------------------------------------------ */
78
+ function formatModelLine(data) {
79
+ const tierLabel = data.plan ? ` ${BULLET} ${capitalize(data.plan)} Tier` : '';
80
+ return `${data.model}${tierLabel}`;
81
+ }
82
+ function formatAccountLine(data) {
83
+ if (!data.email) {
84
+ return 'Anonymous (run /login to authenticate)';
85
+ }
86
+ const tenant = data.tenant ? ` ${BULLET} ${data.tenant} Org` : '';
87
+ return `${data.email}${tenant}`;
88
+ }
89
+ function capitalize(value) {
90
+ if (value.length === 0)
91
+ return value;
92
+ return value.charAt(0).toUpperCase() + value.slice(1).toLowerCase();
93
+ }
94
+ /**
95
+ * Trim the cwd from the LEFT (preserving the basename) so the operator
96
+ * always sees which project they're in even when the column is narrow.
97
+ * Mirrors the Claude Code boot card convention: long paths get an
98
+ * ellipsis on the head, never on the tail.
99
+ */
100
+ export function truncatePath(path, max) {
101
+ if (path.length <= max)
102
+ return path;
103
+ if (max <= 3)
104
+ return path.slice(-max);
105
+ return `…${path.slice(-(max - 1))}`;
106
+ }
107
+ //# sourceMappingURL=welcome-banner.js.map