@mrquake/quakecode-cli 0.64.0 → 0.64.2

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 (180) hide show
  1. package/dist/bun/cli.d.ts.map +1 -1
  2. package/dist/bun/cli.js +1 -1
  3. package/dist/bun/cli.js.map +1 -1
  4. package/dist/config.d.ts.map +1 -1
  5. package/dist/config.js +7 -10
  6. package/dist/config.js.map +1 -1
  7. package/dist/core/extensions/loader.d.ts.map +1 -1
  8. package/dist/core/extensions/loader.js +21 -7
  9. package/dist/core/extensions/loader.js.map +1 -1
  10. package/dist/core/settings-manager.d.ts.map +1 -1
  11. package/dist/core/settings-manager.js +2 -2
  12. package/dist/core/settings-manager.js.map +1 -1
  13. package/dist/core/slash-commands.d.ts.map +1 -1
  14. package/dist/core/slash-commands.js +1 -1
  15. package/dist/core/slash-commands.js.map +1 -1
  16. package/dist/core/system-prompt.d.ts.map +1 -1
  17. package/dist/core/system-prompt.js +5 -5
  18. package/dist/core/system-prompt.js.map +1 -1
  19. package/dist/main.d.ts.map +1 -1
  20. package/dist/main.js +7 -7
  21. package/dist/main.js.map +1 -1
  22. package/dist/migrations.d.ts.map +1 -1
  23. package/dist/migrations.js +2 -2
  24. package/dist/migrations.js.map +1 -1
  25. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -1
  26. package/dist/modes/interactive/components/config-selector.js +1 -1
  27. package/dist/modes/interactive/components/config-selector.js.map +1 -1
  28. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -1
  29. package/dist/modes/interactive/components/session-selector.js +82 -105
  30. package/dist/modes/interactive/components/session-selector.js.map +1 -1
  31. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  32. package/dist/modes/interactive/interactive-mode.js +10 -5
  33. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  34. package/dist/utils/clipboard-image.d.ts.map +1 -1
  35. package/dist/utils/clipboard-image.js +3 -3
  36. package/dist/utils/clipboard-image.js.map +1 -1
  37. package/package.json +14 -7
  38. package/docs/compaction.md +0 -394
  39. package/docs/custom-provider.md +0 -596
  40. package/docs/development.md +0 -71
  41. package/docs/extensions.md +0 -2286
  42. package/docs/images/doom-extension.png +0 -0
  43. package/docs/images/exy.png +0 -0
  44. package/docs/images/interactive-mode.png +0 -0
  45. package/docs/images/tree-view.png +0 -0
  46. package/docs/json.md +0 -82
  47. package/docs/keybindings.md +0 -175
  48. package/docs/models.md +0 -341
  49. package/docs/packages.md +0 -218
  50. package/docs/prompt-templates.md +0 -67
  51. package/docs/providers.md +0 -195
  52. package/docs/rpc.md +0 -1377
  53. package/docs/sdk.md +0 -1064
  54. package/docs/session.md +0 -412
  55. package/docs/settings.md +0 -246
  56. package/docs/shell-aliases.md +0 -13
  57. package/docs/skills.md +0 -232
  58. package/docs/terminal-setup.md +0 -106
  59. package/docs/termux.md +0 -127
  60. package/docs/themes.md +0 -295
  61. package/docs/tmux.md +0 -61
  62. package/docs/tree.md +0 -231
  63. package/docs/tui.md +0 -887
  64. package/docs/windows.md +0 -17
  65. package/examples/README.md +0 -25
  66. package/examples/extensions/README.md +0 -206
  67. package/examples/extensions/antigravity-image-gen.ts +0 -418
  68. package/examples/extensions/auto-commit-on-exit.ts +0 -49
  69. package/examples/extensions/bash-spawn-hook.ts +0 -30
  70. package/examples/extensions/bookmark.ts +0 -50
  71. package/examples/extensions/built-in-tool-renderer.ts +0 -246
  72. package/examples/extensions/claude-rules.ts +0 -86
  73. package/examples/extensions/commands.ts +0 -72
  74. package/examples/extensions/confirm-destructive.ts +0 -59
  75. package/examples/extensions/custom-compaction.ts +0 -127
  76. package/examples/extensions/custom-footer.ts +0 -64
  77. package/examples/extensions/custom-header.ts +0 -73
  78. package/examples/extensions/custom-provider-anthropic/index.ts +0 -604
  79. package/examples/extensions/custom-provider-anthropic/package-lock.json +0 -24
  80. package/examples/extensions/custom-provider-anthropic/package.json +0 -19
  81. package/examples/extensions/custom-provider-gitlab-duo/index.ts +0 -349
  82. package/examples/extensions/custom-provider-gitlab-duo/package.json +0 -16
  83. package/examples/extensions/custom-provider-gitlab-duo/test.ts +0 -82
  84. package/examples/extensions/custom-provider-qwen-cli/index.ts +0 -345
  85. package/examples/extensions/custom-provider-qwen-cli/package.json +0 -16
  86. package/examples/extensions/dirty-repo-guard.ts +0 -56
  87. package/examples/extensions/doom-overlay/README.md +0 -46
  88. package/examples/extensions/doom-overlay/doom/build/doom.js +0 -21
  89. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  90. package/examples/extensions/doom-overlay/doom/build.sh +0 -152
  91. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +0 -72
  92. package/examples/extensions/doom-overlay/doom-component.ts +0 -132
  93. package/examples/extensions/doom-overlay/doom-engine.ts +0 -173
  94. package/examples/extensions/doom-overlay/doom-keys.ts +0 -104
  95. package/examples/extensions/doom-overlay/index.ts +0 -74
  96. package/examples/extensions/doom-overlay/wad-finder.ts +0 -51
  97. package/examples/extensions/dynamic-resources/SKILL.md +0 -8
  98. package/examples/extensions/dynamic-resources/dynamic.json +0 -79
  99. package/examples/extensions/dynamic-resources/dynamic.md +0 -5
  100. package/examples/extensions/dynamic-resources/index.ts +0 -15
  101. package/examples/extensions/dynamic-tools.ts +0 -74
  102. package/examples/extensions/event-bus.ts +0 -43
  103. package/examples/extensions/file-trigger.ts +0 -41
  104. package/examples/extensions/git-checkpoint.ts +0 -53
  105. package/examples/extensions/handoff.ts +0 -153
  106. package/examples/extensions/hello.ts +0 -26
  107. package/examples/extensions/hidden-thinking-label.ts +0 -53
  108. package/examples/extensions/inline-bash.ts +0 -94
  109. package/examples/extensions/input-transform.ts +0 -43
  110. package/examples/extensions/interactive-shell.ts +0 -196
  111. package/examples/extensions/mac-system-theme.ts +0 -47
  112. package/examples/extensions/message-renderer.ts +0 -59
  113. package/examples/extensions/minimal-mode.ts +0 -426
  114. package/examples/extensions/modal-editor.ts +0 -85
  115. package/examples/extensions/model-status.ts +0 -31
  116. package/examples/extensions/notify.ts +0 -55
  117. package/examples/extensions/overlay-qa-tests.ts +0 -1348
  118. package/examples/extensions/overlay-test.ts +0 -150
  119. package/examples/extensions/permission-gate.ts +0 -34
  120. package/examples/extensions/pirate.ts +0 -47
  121. package/examples/extensions/plan-mode/README.md +0 -65
  122. package/examples/extensions/plan-mode/index.ts +0 -340
  123. package/examples/extensions/plan-mode/utils.ts +0 -168
  124. package/examples/extensions/preset.ts +0 -397
  125. package/examples/extensions/protected-paths.ts +0 -30
  126. package/examples/extensions/provider-payload.ts +0 -14
  127. package/examples/extensions/qna.ts +0 -122
  128. package/examples/extensions/question.ts +0 -264
  129. package/examples/extensions/questionnaire.ts +0 -427
  130. package/examples/extensions/rainbow-editor.ts +0 -88
  131. package/examples/extensions/reload-runtime.ts +0 -37
  132. package/examples/extensions/rpc-demo.ts +0 -118
  133. package/examples/extensions/sandbox/index.ts +0 -321
  134. package/examples/extensions/sandbox/package-lock.json +0 -92
  135. package/examples/extensions/sandbox/package.json +0 -19
  136. package/examples/extensions/send-user-message.ts +0 -97
  137. package/examples/extensions/session-name.ts +0 -27
  138. package/examples/extensions/shutdown-command.ts +0 -63
  139. package/examples/extensions/snake.ts +0 -343
  140. package/examples/extensions/space-invaders.ts +0 -560
  141. package/examples/extensions/ssh.ts +0 -220
  142. package/examples/extensions/status-line.ts +0 -32
  143. package/examples/extensions/subagent/README.md +0 -172
  144. package/examples/extensions/subagent/agents/planner.md +0 -37
  145. package/examples/extensions/subagent/agents/reviewer.md +0 -35
  146. package/examples/extensions/subagent/agents/scout.md +0 -50
  147. package/examples/extensions/subagent/agents/worker.md +0 -24
  148. package/examples/extensions/subagent/agents.ts +0 -126
  149. package/examples/extensions/subagent/index.ts +0 -986
  150. package/examples/extensions/subagent/prompts/implement-and-review.md +0 -10
  151. package/examples/extensions/subagent/prompts/implement.md +0 -10
  152. package/examples/extensions/subagent/prompts/scout-and-plan.md +0 -9
  153. package/examples/extensions/summarize.ts +0 -206
  154. package/examples/extensions/system-prompt-header.ts +0 -17
  155. package/examples/extensions/timed-confirm.ts +0 -70
  156. package/examples/extensions/titlebar-spinner.ts +0 -58
  157. package/examples/extensions/todo.ts +0 -297
  158. package/examples/extensions/tool-override.ts +0 -144
  159. package/examples/extensions/tools.ts +0 -141
  160. package/examples/extensions/trigger-compact.ts +0 -50
  161. package/examples/extensions/truncated-tool.ts +0 -195
  162. package/examples/extensions/widget-placement.ts +0 -9
  163. package/examples/extensions/with-deps/index.ts +0 -32
  164. package/examples/extensions/with-deps/package-lock.json +0 -31
  165. package/examples/extensions/with-deps/package.json +0 -22
  166. package/examples/rpc-extension-ui.ts +0 -632
  167. package/examples/sdk/01-minimal.ts +0 -22
  168. package/examples/sdk/02-custom-model.ts +0 -49
  169. package/examples/sdk/03-custom-prompt.ts +0 -55
  170. package/examples/sdk/04-skills.ts +0 -52
  171. package/examples/sdk/05-tools.ts +0 -56
  172. package/examples/sdk/06-extensions.ts +0 -88
  173. package/examples/sdk/07-context-files.ts +0 -40
  174. package/examples/sdk/08-prompt-templates.ts +0 -48
  175. package/examples/sdk/09-api-keys-and-oauth.ts +0 -48
  176. package/examples/sdk/10-settings.ts +0 -51
  177. package/examples/sdk/11-sessions.ts +0 -48
  178. package/examples/sdk/12-full-control.ts +0 -81
  179. package/examples/sdk/13-session-runtime.ts +0 -49
  180. package/examples/sdk/README.md +0 -145
@@ -1,10 +0,0 @@
1
- ---
2
- description: Worker implements, reviewer reviews, worker applies feedback
3
- ---
4
- Use the subagent tool with the chain parameter to execute this workflow:
5
-
6
- 1. First, use the "worker" agent to implement: $@
7
- 2. Then, use the "reviewer" agent to review the implementation from the previous step (use {previous} placeholder)
8
- 3. Finally, use the "worker" agent to apply the feedback from the review (use {previous} placeholder)
9
-
10
- Execute this as a chain, passing output between steps via {previous}.
@@ -1,10 +0,0 @@
1
- ---
2
- description: Full implementation workflow - scout gathers context, planner creates plan, worker implements
3
- ---
4
- Use the subagent tool with the chain parameter to execute this workflow:
5
-
6
- 1. First, use the "scout" agent to find all code relevant to: $@
7
- 2. Then, use the "planner" agent to create an implementation plan for "$@" using the context from the previous step (use {previous} placeholder)
8
- 3. Finally, use the "worker" agent to implement the plan from the previous step (use {previous} placeholder)
9
-
10
- Execute this as a chain, passing output between steps via {previous}.
@@ -1,9 +0,0 @@
1
- ---
2
- description: Scout gathers context, planner creates implementation plan (no implementation)
3
- ---
4
- Use the subagent tool with the chain parameter to execute this workflow:
5
-
6
- 1. First, use the "scout" agent to find all code relevant to: $@
7
- 2. Then, use the "planner" agent to create an implementation plan for "$@" using the context from the previous step (use {previous} placeholder)
8
-
9
- Execute this as a chain, passing output between steps via {previous}. Do NOT implement - just return the plan.
@@ -1,206 +0,0 @@
1
- import { complete, getModel } from "@mariozechner/pi-ai";
2
- import type { ExtensionAPI, ExtensionCommandContext } from "@mariozechner/pi-coding-agent";
3
- import { DynamicBorder, getMarkdownTheme } from "@mariozechner/pi-coding-agent";
4
- import { Container, Markdown, matchesKey, Text } from "@mariozechner/pi-tui";
5
-
6
- type ContentBlock = {
7
- type?: string;
8
- text?: string;
9
- name?: string;
10
- arguments?: Record<string, unknown>;
11
- };
12
-
13
- type SessionEntry = {
14
- type: string;
15
- message?: {
16
- role?: string;
17
- content?: unknown;
18
- };
19
- };
20
-
21
- const extractTextParts = (content: unknown): string[] => {
22
- if (typeof content === "string") {
23
- return [content];
24
- }
25
-
26
- if (!Array.isArray(content)) {
27
- return [];
28
- }
29
-
30
- const textParts: string[] = [];
31
- for (const part of content) {
32
- if (!part || typeof part !== "object") {
33
- continue;
34
- }
35
-
36
- const block = part as ContentBlock;
37
- if (block.type === "text" && typeof block.text === "string") {
38
- textParts.push(block.text);
39
- }
40
- }
41
-
42
- return textParts;
43
- };
44
-
45
- const extractToolCallLines = (content: unknown): string[] => {
46
- if (!Array.isArray(content)) {
47
- return [];
48
- }
49
-
50
- const toolCalls: string[] = [];
51
- for (const part of content) {
52
- if (!part || typeof part !== "object") {
53
- continue;
54
- }
55
-
56
- const block = part as ContentBlock;
57
- if (block.type !== "toolCall" || typeof block.name !== "string") {
58
- continue;
59
- }
60
-
61
- const args = block.arguments ?? {};
62
- toolCalls.push(`Tool ${block.name} was called with args ${JSON.stringify(args)}`);
63
- }
64
-
65
- return toolCalls;
66
- };
67
-
68
- const buildConversationText = (entries: SessionEntry[]): string => {
69
- const sections: string[] = [];
70
-
71
- for (const entry of entries) {
72
- if (entry.type !== "message" || !entry.message?.role) {
73
- continue;
74
- }
75
-
76
- const role = entry.message.role;
77
- const isUser = role === "user";
78
- const isAssistant = role === "assistant";
79
-
80
- if (!isUser && !isAssistant) {
81
- continue;
82
- }
83
-
84
- const entryLines: string[] = [];
85
- const textParts = extractTextParts(entry.message.content);
86
- if (textParts.length > 0) {
87
- const roleLabel = isUser ? "User" : "Assistant";
88
- const messageText = textParts.join("\n").trim();
89
- if (messageText.length > 0) {
90
- entryLines.push(`${roleLabel}: ${messageText}`);
91
- }
92
- }
93
-
94
- if (isAssistant) {
95
- entryLines.push(...extractToolCallLines(entry.message.content));
96
- }
97
-
98
- if (entryLines.length > 0) {
99
- sections.push(entryLines.join("\n"));
100
- }
101
- }
102
-
103
- return sections.join("\n\n");
104
- };
105
-
106
- const buildSummaryPrompt = (conversationText: string): string =>
107
- [
108
- "Summarize this conversation so I can resume it later.",
109
- "Include goals, key decisions, progress, open questions, and next steps.",
110
- "Keep it concise and structured with headings.",
111
- "",
112
- "<conversation>",
113
- conversationText,
114
- "</conversation>",
115
- ].join("\n");
116
-
117
- const showSummaryUi = async (summary: string, ctx: ExtensionCommandContext) => {
118
- if (!ctx.hasUI) {
119
- return;
120
- }
121
-
122
- await ctx.ui.custom((_tui, theme, _kb, done) => {
123
- const container = new Container();
124
- const border = new DynamicBorder((s: string) => theme.fg("accent", s));
125
- const mdTheme = getMarkdownTheme();
126
-
127
- container.addChild(border);
128
- container.addChild(new Text(theme.fg("accent", theme.bold("Conversation Summary")), 1, 0));
129
- container.addChild(new Markdown(summary, 1, 1, mdTheme));
130
- container.addChild(new Text(theme.fg("dim", "Press Enter or Esc to close"), 1, 0));
131
- container.addChild(border);
132
-
133
- return {
134
- render: (width: number) => container.render(width),
135
- invalidate: () => container.invalidate(),
136
- handleInput: (data: string) => {
137
- if (matchesKey(data, "enter") || matchesKey(data, "escape")) {
138
- done(undefined);
139
- }
140
- },
141
- };
142
- });
143
- };
144
-
145
- export default function (pi: ExtensionAPI) {
146
- pi.registerCommand("summarize", {
147
- description: "Summarize the current conversation in a custom UI",
148
- handler: async (_args, ctx) => {
149
- const branch = ctx.sessionManager.getBranch();
150
- const conversationText = buildConversationText(branch);
151
-
152
- if (!conversationText.trim()) {
153
- if (ctx.hasUI) {
154
- ctx.ui.notify("No conversation text found", "warning");
155
- }
156
- return;
157
- }
158
-
159
- if (ctx.hasUI) {
160
- ctx.ui.notify("Preparing summary...", "info");
161
- }
162
-
163
- const model = getModel("openai", "gpt-5.2");
164
- if (!model && ctx.hasUI) {
165
- ctx.ui.notify("Model openai/gpt-5.2 not found", "warning");
166
- }
167
-
168
- const auth = model ? await ctx.modelRegistry.getApiKeyAndHeaders(model) : undefined;
169
- if (auth && !auth.ok && ctx.hasUI) {
170
- ctx.ui.notify(auth.error, "warning");
171
- }
172
- if (auth?.ok && !auth.apiKey && ctx.hasUI) {
173
- ctx.ui.notify("No API key for openai/gpt-5.2", "warning");
174
- }
175
-
176
- if (!model || !auth?.ok || !auth.apiKey) {
177
- return;
178
- }
179
-
180
- const summaryMessages = [
181
- {
182
- role: "user" as const,
183
- content: [{ type: "text" as const, text: buildSummaryPrompt(conversationText) }],
184
- timestamp: Date.now(),
185
- },
186
- ];
187
-
188
- const response = await complete(
189
- model,
190
- { messages: summaryMessages },
191
- {
192
- apiKey: auth.apiKey,
193
- headers: auth.headers,
194
- reasoningEffort: "high",
195
- },
196
- );
197
-
198
- const summary = response.content
199
- .filter((c): c is { type: "text"; text: string } => c.type === "text")
200
- .map((c) => c.text)
201
- .join("\n");
202
-
203
- await showSummaryUi(summary, ctx);
204
- },
205
- });
206
- }
@@ -1,17 +0,0 @@
1
- /**
2
- * Displays a status widget showing the system prompt length.
3
- *
4
- * Demonstrates ctx.getSystemPrompt() for accessing the effective system prompt.
5
- */
6
- import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
7
-
8
- export default function (pi: ExtensionAPI) {
9
- pi.on("agent_start", (_event, ctx) => {
10
- const prompt = ctx.getSystemPrompt();
11
- ctx.ui.setStatus("system-prompt", `System: ${prompt.length} chars`);
12
- });
13
-
14
- pi.on("session_shutdown", (_event, ctx) => {
15
- ctx.ui.setStatus("system-prompt", undefined);
16
- });
17
- }
@@ -1,70 +0,0 @@
1
- /**
2
- * Example extension demonstrating timed dialogs with live countdown.
3
- *
4
- * Commands:
5
- * - /timed - Shows confirm dialog that auto-cancels after 5 seconds with countdown
6
- * - /timed-select - Shows select dialog that auto-cancels after 10 seconds with countdown
7
- * - /timed-signal - Shows confirm using AbortSignal (manual approach)
8
- */
9
-
10
- import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
11
-
12
- export default function (pi: ExtensionAPI) {
13
- // Simple approach: use timeout option (recommended)
14
- pi.registerCommand("timed", {
15
- description: "Show a timed confirmation dialog (auto-cancels in 5s with countdown)",
16
- handler: async (_args, ctx) => {
17
- const confirmed = await ctx.ui.confirm(
18
- "Timed Confirmation",
19
- "This dialog will auto-cancel in 5 seconds. Confirm?",
20
- { timeout: 5000 },
21
- );
22
-
23
- if (confirmed) {
24
- ctx.ui.notify("Confirmed by user!", "info");
25
- } else {
26
- ctx.ui.notify("Cancelled or timed out", "info");
27
- }
28
- },
29
- });
30
-
31
- pi.registerCommand("timed-select", {
32
- description: "Show a timed select dialog (auto-cancels in 10s with countdown)",
33
- handler: async (_args, ctx) => {
34
- const choice = await ctx.ui.select("Pick an option", ["Option A", "Option B", "Option C"], { timeout: 10000 });
35
-
36
- if (choice) {
37
- ctx.ui.notify(`Selected: ${choice}`, "info");
38
- } else {
39
- ctx.ui.notify("Selection cancelled or timed out", "info");
40
- }
41
- },
42
- });
43
-
44
- // Manual approach: use AbortSignal for more control
45
- pi.registerCommand("timed-signal", {
46
- description: "Show a timed confirm using AbortSignal (manual approach)",
47
- handler: async (_args, ctx) => {
48
- const controller = new AbortController();
49
- const timeoutId = setTimeout(() => controller.abort(), 5000);
50
-
51
- ctx.ui.notify("Dialog will auto-cancel in 5 seconds...", "info");
52
-
53
- const confirmed = await ctx.ui.confirm(
54
- "Timed Confirmation",
55
- "This dialog will auto-cancel in 5 seconds. Confirm?",
56
- { signal: controller.signal },
57
- );
58
-
59
- clearTimeout(timeoutId);
60
-
61
- if (confirmed) {
62
- ctx.ui.notify("Confirmed by user!", "info");
63
- } else if (controller.signal.aborted) {
64
- ctx.ui.notify("Dialog timed out (auto-cancelled)", "warning");
65
- } else {
66
- ctx.ui.notify("Cancelled by user", "info");
67
- }
68
- },
69
- });
70
- }
@@ -1,58 +0,0 @@
1
- /**
2
- * Titlebar Spinner Extension
3
- *
4
- * Shows a braille spinner animation in the terminal title while the agent is working.
5
- * Uses `ctx.ui.setTitle()` to update the terminal title via the extension API.
6
- *
7
- * Usage:
8
- * pi --extension examples/extensions/titlebar-spinner.ts
9
- */
10
-
11
- import path from "node:path";
12
- import type { ExtensionAPI, ExtensionContext } from "@mariozechner/pi-coding-agent";
13
-
14
- const BRAILLE_FRAMES = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
15
-
16
- function getBaseTitle(pi: ExtensionAPI): string {
17
- const cwd = path.basename(process.cwd());
18
- const session = pi.getSessionName();
19
- return session ? `π - ${session} - ${cwd}` : `π - ${cwd}`;
20
- }
21
-
22
- export default function (pi: ExtensionAPI) {
23
- let timer: ReturnType<typeof setInterval> | null = null;
24
- let frameIndex = 0;
25
-
26
- function stopAnimation(ctx: ExtensionContext) {
27
- if (timer) {
28
- clearInterval(timer);
29
- timer = null;
30
- }
31
- frameIndex = 0;
32
- ctx.ui.setTitle(getBaseTitle(pi));
33
- }
34
-
35
- function startAnimation(ctx: ExtensionContext) {
36
- stopAnimation(ctx);
37
- timer = setInterval(() => {
38
- const frame = BRAILLE_FRAMES[frameIndex % BRAILLE_FRAMES.length];
39
- const cwd = path.basename(process.cwd());
40
- const session = pi.getSessionName();
41
- const title = session ? `${frame} π - ${session} - ${cwd}` : `${frame} π - ${cwd}`;
42
- ctx.ui.setTitle(title);
43
- frameIndex++;
44
- }, 80);
45
- }
46
-
47
- pi.on("agent_start", async (_event, ctx) => {
48
- startAnimation(ctx);
49
- });
50
-
51
- pi.on("agent_end", async (_event, ctx) => {
52
- stopAnimation(ctx);
53
- });
54
-
55
- pi.on("session_shutdown", async (_event, ctx) => {
56
- stopAnimation(ctx);
57
- });
58
- }
@@ -1,297 +0,0 @@
1
- /**
2
- * Todo Extension - Demonstrates state management via session entries
3
- *
4
- * This extension:
5
- * - Registers a `todo` tool for the LLM to manage todos
6
- * - Registers a `/todos` command for users to view the list
7
- *
8
- * State is stored in tool result details (not external files), which allows
9
- * proper branching - when you branch, the todo state is automatically
10
- * correct for that point in history.
11
- */
12
-
13
- import { StringEnum } from "@mariozechner/pi-ai";
14
- import type { ExtensionAPI, ExtensionContext, Theme } from "@mariozechner/pi-coding-agent";
15
- import { matchesKey, Text, truncateToWidth } from "@mariozechner/pi-tui";
16
- import { Type } from "@sinclair/typebox";
17
-
18
- interface Todo {
19
- id: number;
20
- text: string;
21
- done: boolean;
22
- }
23
-
24
- interface TodoDetails {
25
- action: "list" | "add" | "toggle" | "clear";
26
- todos: Todo[];
27
- nextId: number;
28
- error?: string;
29
- }
30
-
31
- const TodoParams = Type.Object({
32
- action: StringEnum(["list", "add", "toggle", "clear"] as const),
33
- text: Type.Optional(Type.String({ description: "Todo text (for add)" })),
34
- id: Type.Optional(Type.Number({ description: "Todo ID (for toggle)" })),
35
- });
36
-
37
- /**
38
- * UI component for the /todos command
39
- */
40
- class TodoListComponent {
41
- private todos: Todo[];
42
- private theme: Theme;
43
- private onClose: () => void;
44
- private cachedWidth?: number;
45
- private cachedLines?: string[];
46
-
47
- constructor(todos: Todo[], theme: Theme, onClose: () => void) {
48
- this.todos = todos;
49
- this.theme = theme;
50
- this.onClose = onClose;
51
- }
52
-
53
- handleInput(data: string): void {
54
- if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
55
- this.onClose();
56
- }
57
- }
58
-
59
- render(width: number): string[] {
60
- if (this.cachedLines && this.cachedWidth === width) {
61
- return this.cachedLines;
62
- }
63
-
64
- const lines: string[] = [];
65
- const th = this.theme;
66
-
67
- lines.push("");
68
- const title = th.fg("accent", " Todos ");
69
- const headerLine =
70
- th.fg("borderMuted", "─".repeat(3)) + title + th.fg("borderMuted", "─".repeat(Math.max(0, width - 10)));
71
- lines.push(truncateToWidth(headerLine, width));
72
- lines.push("");
73
-
74
- if (this.todos.length === 0) {
75
- lines.push(truncateToWidth(` ${th.fg("dim", "No todos yet. Ask the agent to add some!")}`, width));
76
- } else {
77
- const done = this.todos.filter((t) => t.done).length;
78
- const total = this.todos.length;
79
- lines.push(truncateToWidth(` ${th.fg("muted", `${done}/${total} completed`)}`, width));
80
- lines.push("");
81
-
82
- for (const todo of this.todos) {
83
- const check = todo.done ? th.fg("success", "✓") : th.fg("dim", "○");
84
- const id = th.fg("accent", `#${todo.id}`);
85
- const text = todo.done ? th.fg("dim", todo.text) : th.fg("text", todo.text);
86
- lines.push(truncateToWidth(` ${check} ${id} ${text}`, width));
87
- }
88
- }
89
-
90
- lines.push("");
91
- lines.push(truncateToWidth(` ${th.fg("dim", "Press Escape to close")}`, width));
92
- lines.push("");
93
-
94
- this.cachedWidth = width;
95
- this.cachedLines = lines;
96
- return lines;
97
- }
98
-
99
- invalidate(): void {
100
- this.cachedWidth = undefined;
101
- this.cachedLines = undefined;
102
- }
103
- }
104
-
105
- export default function (pi: ExtensionAPI) {
106
- // In-memory state (reconstructed from session on load)
107
- let todos: Todo[] = [];
108
- let nextId = 1;
109
-
110
- /**
111
- * Reconstruct state from session entries.
112
- * Scans tool results for this tool and applies them in order.
113
- */
114
- const reconstructState = (ctx: ExtensionContext) => {
115
- todos = [];
116
- nextId = 1;
117
-
118
- for (const entry of ctx.sessionManager.getBranch()) {
119
- if (entry.type !== "message") continue;
120
- const msg = entry.message;
121
- if (msg.role !== "toolResult" || msg.toolName !== "todo") continue;
122
-
123
- const details = msg.details as TodoDetails | undefined;
124
- if (details) {
125
- todos = details.todos;
126
- nextId = details.nextId;
127
- }
128
- }
129
- };
130
-
131
- // Reconstruct state on session events
132
- pi.on("session_start", async (_event, ctx) => reconstructState(ctx));
133
- pi.on("session_tree", async (_event, ctx) => reconstructState(ctx));
134
-
135
- // Register the todo tool for the LLM
136
- pi.registerTool({
137
- name: "todo",
138
- label: "Todo",
139
- description: "Manage a todo list. Actions: list, add (text), toggle (id), clear",
140
- parameters: TodoParams,
141
-
142
- async execute(_toolCallId, params, _signal, _onUpdate, _ctx) {
143
- switch (params.action) {
144
- case "list":
145
- return {
146
- content: [
147
- {
148
- type: "text",
149
- text: todos.length
150
- ? todos.map((t) => `[${t.done ? "x" : " "}] #${t.id}: ${t.text}`).join("\n")
151
- : "No todos",
152
- },
153
- ],
154
- details: { action: "list", todos: [...todos], nextId } as TodoDetails,
155
- };
156
-
157
- case "add": {
158
- if (!params.text) {
159
- return {
160
- content: [{ type: "text", text: "Error: text required for add" }],
161
- details: { action: "add", todos: [...todos], nextId, error: "text required" } as TodoDetails,
162
- };
163
- }
164
- const newTodo: Todo = { id: nextId++, text: params.text, done: false };
165
- todos.push(newTodo);
166
- return {
167
- content: [{ type: "text", text: `Added todo #${newTodo.id}: ${newTodo.text}` }],
168
- details: { action: "add", todos: [...todos], nextId } as TodoDetails,
169
- };
170
- }
171
-
172
- case "toggle": {
173
- if (params.id === undefined) {
174
- return {
175
- content: [{ type: "text", text: "Error: id required for toggle" }],
176
- details: { action: "toggle", todos: [...todos], nextId, error: "id required" } as TodoDetails,
177
- };
178
- }
179
- const todo = todos.find((t) => t.id === params.id);
180
- if (!todo) {
181
- return {
182
- content: [{ type: "text", text: `Todo #${params.id} not found` }],
183
- details: {
184
- action: "toggle",
185
- todos: [...todos],
186
- nextId,
187
- error: `#${params.id} not found`,
188
- } as TodoDetails,
189
- };
190
- }
191
- todo.done = !todo.done;
192
- return {
193
- content: [{ type: "text", text: `Todo #${todo.id} ${todo.done ? "completed" : "uncompleted"}` }],
194
- details: { action: "toggle", todos: [...todos], nextId } as TodoDetails,
195
- };
196
- }
197
-
198
- case "clear": {
199
- const count = todos.length;
200
- todos = [];
201
- nextId = 1;
202
- return {
203
- content: [{ type: "text", text: `Cleared ${count} todos` }],
204
- details: { action: "clear", todos: [], nextId: 1 } as TodoDetails,
205
- };
206
- }
207
-
208
- default:
209
- return {
210
- content: [{ type: "text", text: `Unknown action: ${params.action}` }],
211
- details: {
212
- action: "list",
213
- todos: [...todos],
214
- nextId,
215
- error: `unknown action: ${params.action}`,
216
- } as TodoDetails,
217
- };
218
- }
219
- },
220
-
221
- renderCall(args, theme, _context) {
222
- let text = theme.fg("toolTitle", theme.bold("todo ")) + theme.fg("muted", args.action);
223
- if (args.text) text += ` ${theme.fg("dim", `"${args.text}"`)}`;
224
- if (args.id !== undefined) text += ` ${theme.fg("accent", `#${args.id}`)}`;
225
- return new Text(text, 0, 0);
226
- },
227
-
228
- renderResult(result, { expanded }, theme, _context) {
229
- const details = result.details as TodoDetails | undefined;
230
- if (!details) {
231
- const text = result.content[0];
232
- return new Text(text?.type === "text" ? text.text : "", 0, 0);
233
- }
234
-
235
- if (details.error) {
236
- return new Text(theme.fg("error", `Error: ${details.error}`), 0, 0);
237
- }
238
-
239
- const todoList = details.todos;
240
-
241
- switch (details.action) {
242
- case "list": {
243
- if (todoList.length === 0) {
244
- return new Text(theme.fg("dim", "No todos"), 0, 0);
245
- }
246
- let listText = theme.fg("muted", `${todoList.length} todo(s):`);
247
- const display = expanded ? todoList : todoList.slice(0, 5);
248
- for (const t of display) {
249
- const check = t.done ? theme.fg("success", "✓") : theme.fg("dim", "○");
250
- const itemText = t.done ? theme.fg("dim", t.text) : theme.fg("muted", t.text);
251
- listText += `\n${check} ${theme.fg("accent", `#${t.id}`)} ${itemText}`;
252
- }
253
- if (!expanded && todoList.length > 5) {
254
- listText += `\n${theme.fg("dim", `... ${todoList.length - 5} more`)}`;
255
- }
256
- return new Text(listText, 0, 0);
257
- }
258
-
259
- case "add": {
260
- const added = todoList[todoList.length - 1];
261
- return new Text(
262
- theme.fg("success", "✓ Added ") +
263
- theme.fg("accent", `#${added.id}`) +
264
- " " +
265
- theme.fg("muted", added.text),
266
- 0,
267
- 0,
268
- );
269
- }
270
-
271
- case "toggle": {
272
- const text = result.content[0];
273
- const msg = text?.type === "text" ? text.text : "";
274
- return new Text(theme.fg("success", "✓ ") + theme.fg("muted", msg), 0, 0);
275
- }
276
-
277
- case "clear":
278
- return new Text(theme.fg("success", "✓ ") + theme.fg("muted", "Cleared all todos"), 0, 0);
279
- }
280
- },
281
- });
282
-
283
- // Register the /todos command for users
284
- pi.registerCommand("todos", {
285
- description: "Show all todos on the current branch",
286
- handler: async (_args, ctx) => {
287
- if (!ctx.hasUI) {
288
- ctx.ui.notify("/todos requires interactive mode", "error");
289
- return;
290
- }
291
-
292
- await ctx.ui.custom<void>((_tui, theme, _kb, done) => {
293
- return new TodoListComponent(todos, theme, () => done());
294
- });
295
- },
296
- });
297
- }