@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,122 +0,0 @@
1
- /**
2
- * Q&A extraction extension - extracts questions from assistant responses
3
- *
4
- * Demonstrates the "prompt generator" pattern:
5
- * 1. /qna command gets the last assistant message
6
- * 2. Shows a spinner while extracting (hides editor)
7
- * 3. Loads the result into the editor for user to fill in answers
8
- */
9
-
10
- import { complete, type UserMessage } from "@mariozechner/pi-ai";
11
- import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
12
- import { BorderedLoader } from "@mariozechner/pi-coding-agent";
13
-
14
- const SYSTEM_PROMPT = `You are a question extractor. Given text from a conversation, extract any questions that need answering and format them for the user to fill in.
15
-
16
- Output format:
17
- - List each question on its own line, prefixed with "Q: "
18
- - After each question, add a blank line for the answer prefixed with "A: "
19
- - If no questions are found, output "No questions found in the last message."
20
-
21
- Example output:
22
- Q: What is your preferred database?
23
- A:
24
-
25
- Q: Should we use TypeScript or JavaScript?
26
- A:
27
-
28
- Keep questions in the order they appeared. Be concise.`;
29
-
30
- export default function (pi: ExtensionAPI) {
31
- pi.registerCommand("qna", {
32
- description: "Extract questions from last assistant message into editor",
33
- handler: async (_args, ctx) => {
34
- if (!ctx.hasUI) {
35
- ctx.ui.notify("qna requires interactive mode", "error");
36
- return;
37
- }
38
-
39
- if (!ctx.model) {
40
- ctx.ui.notify("No model selected", "error");
41
- return;
42
- }
43
-
44
- // Find the last assistant message on the current branch
45
- const branch = ctx.sessionManager.getBranch();
46
- let lastAssistantText: string | undefined;
47
-
48
- for (let i = branch.length - 1; i >= 0; i--) {
49
- const entry = branch[i];
50
- if (entry.type === "message") {
51
- const msg = entry.message;
52
- if ("role" in msg && msg.role === "assistant") {
53
- if (msg.stopReason !== "stop") {
54
- ctx.ui.notify(`Last assistant message incomplete (${msg.stopReason})`, "error");
55
- return;
56
- }
57
- const textParts = msg.content
58
- .filter((c): c is { type: "text"; text: string } => c.type === "text")
59
- .map((c) => c.text);
60
- if (textParts.length > 0) {
61
- lastAssistantText = textParts.join("\n");
62
- break;
63
- }
64
- }
65
- }
66
- }
67
-
68
- if (!lastAssistantText) {
69
- ctx.ui.notify("No assistant messages found", "error");
70
- return;
71
- }
72
-
73
- // Run extraction with loader UI
74
- const result = await ctx.ui.custom<string | null>((tui, theme, _kb, done) => {
75
- const loader = new BorderedLoader(tui, theme, `Extracting questions using ${ctx.model!.id}...`);
76
- loader.onAbort = () => done(null);
77
-
78
- // Do the work
79
- const doExtract = async () => {
80
- const auth = await ctx.modelRegistry.getApiKeyAndHeaders(ctx.model!);
81
- if (!auth.ok || !auth.apiKey) {
82
- throw new Error(auth.ok ? `No API key for ${ctx.model!.provider}` : auth.error);
83
- }
84
- const userMessage: UserMessage = {
85
- role: "user",
86
- content: [{ type: "text", text: lastAssistantText! }],
87
- timestamp: Date.now(),
88
- };
89
-
90
- const response = await complete(
91
- ctx.model!,
92
- { systemPrompt: SYSTEM_PROMPT, messages: [userMessage] },
93
- { apiKey: auth.apiKey, headers: auth.headers, signal: loader.signal },
94
- );
95
-
96
- if (response.stopReason === "aborted") {
97
- return null;
98
- }
99
-
100
- return response.content
101
- .filter((c): c is { type: "text"; text: string } => c.type === "text")
102
- .map((c) => c.text)
103
- .join("\n");
104
- };
105
-
106
- doExtract()
107
- .then(done)
108
- .catch(() => done(null));
109
-
110
- return loader;
111
- });
112
-
113
- if (result === null) {
114
- ctx.ui.notify("Cancelled", "info");
115
- return;
116
- }
117
-
118
- ctx.ui.setEditorText(result);
119
- ctx.ui.notify("Questions loaded. Edit and submit when ready.", "info");
120
- },
121
- });
122
- }
@@ -1,264 +0,0 @@
1
- /**
2
- * Question Tool - Single question with options
3
- * Full custom UI: options list + inline editor for "Type something..."
4
- * Escape in editor returns to options, Escape in options cancels
5
- */
6
-
7
- import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
8
- import { Editor, type EditorTheme, Key, matchesKey, Text, truncateToWidth } from "@mariozechner/pi-tui";
9
- import { Type } from "@sinclair/typebox";
10
-
11
- interface OptionWithDesc {
12
- label: string;
13
- description?: string;
14
- }
15
-
16
- type DisplayOption = OptionWithDesc & { isOther?: boolean };
17
-
18
- interface QuestionDetails {
19
- question: string;
20
- options: string[];
21
- answer: string | null;
22
- wasCustom?: boolean;
23
- }
24
-
25
- // Options with labels and optional descriptions
26
- const OptionSchema = Type.Object({
27
- label: Type.String({ description: "Display label for the option" }),
28
- description: Type.Optional(Type.String({ description: "Optional description shown below label" })),
29
- });
30
-
31
- const QuestionParams = Type.Object({
32
- question: Type.String({ description: "The question to ask the user" }),
33
- options: Type.Array(OptionSchema, { description: "Options for the user to choose from" }),
34
- });
35
-
36
- export default function question(pi: ExtensionAPI) {
37
- pi.registerTool({
38
- name: "question",
39
- label: "Question",
40
- description: "Ask the user a question and let them pick from options. Use when you need user input to proceed.",
41
- parameters: QuestionParams,
42
-
43
- async execute(_toolCallId, params, _signal, _onUpdate, ctx) {
44
- if (!ctx.hasUI) {
45
- return {
46
- content: [{ type: "text", text: "Error: UI not available (running in non-interactive mode)" }],
47
- details: {
48
- question: params.question,
49
- options: params.options.map((o) => o.label),
50
- answer: null,
51
- } as QuestionDetails,
52
- };
53
- }
54
-
55
- if (params.options.length === 0) {
56
- return {
57
- content: [{ type: "text", text: "Error: No options provided" }],
58
- details: { question: params.question, options: [], answer: null } as QuestionDetails,
59
- };
60
- }
61
-
62
- const allOptions: DisplayOption[] = [...params.options, { label: "Type something.", isOther: true }];
63
-
64
- const result = await ctx.ui.custom<{ answer: string; wasCustom: boolean; index?: number } | null>(
65
- (tui, theme, _kb, done) => {
66
- let optionIndex = 0;
67
- let editMode = false;
68
- let cachedLines: string[] | undefined;
69
-
70
- const editorTheme: EditorTheme = {
71
- borderColor: (s) => theme.fg("accent", s),
72
- selectList: {
73
- selectedPrefix: (t) => theme.fg("accent", t),
74
- selectedText: (t) => theme.fg("accent", t),
75
- description: (t) => theme.fg("muted", t),
76
- scrollInfo: (t) => theme.fg("dim", t),
77
- noMatch: (t) => theme.fg("warning", t),
78
- },
79
- };
80
- const editor = new Editor(tui, editorTheme);
81
-
82
- editor.onSubmit = (value) => {
83
- const trimmed = value.trim();
84
- if (trimmed) {
85
- done({ answer: trimmed, wasCustom: true });
86
- } else {
87
- editMode = false;
88
- editor.setText("");
89
- refresh();
90
- }
91
- };
92
-
93
- function refresh() {
94
- cachedLines = undefined;
95
- tui.requestRender();
96
- }
97
-
98
- function handleInput(data: string) {
99
- if (editMode) {
100
- if (matchesKey(data, Key.escape)) {
101
- editMode = false;
102
- editor.setText("");
103
- refresh();
104
- return;
105
- }
106
- editor.handleInput(data);
107
- refresh();
108
- return;
109
- }
110
-
111
- if (matchesKey(data, Key.up)) {
112
- optionIndex = Math.max(0, optionIndex - 1);
113
- refresh();
114
- return;
115
- }
116
- if (matchesKey(data, Key.down)) {
117
- optionIndex = Math.min(allOptions.length - 1, optionIndex + 1);
118
- refresh();
119
- return;
120
- }
121
-
122
- if (matchesKey(data, Key.enter)) {
123
- const selected = allOptions[optionIndex];
124
- if (selected.isOther) {
125
- editMode = true;
126
- refresh();
127
- } else {
128
- done({ answer: selected.label, wasCustom: false, index: optionIndex + 1 });
129
- }
130
- return;
131
- }
132
-
133
- if (matchesKey(data, Key.escape)) {
134
- done(null);
135
- }
136
- }
137
-
138
- function render(width: number): string[] {
139
- if (cachedLines) return cachedLines;
140
-
141
- const lines: string[] = [];
142
- const add = (s: string) => lines.push(truncateToWidth(s, width));
143
-
144
- add(theme.fg("accent", "─".repeat(width)));
145
- add(theme.fg("text", ` ${params.question}`));
146
- lines.push("");
147
-
148
- for (let i = 0; i < allOptions.length; i++) {
149
- const opt = allOptions[i];
150
- const selected = i === optionIndex;
151
- const isOther = opt.isOther === true;
152
- const prefix = selected ? theme.fg("accent", "> ") : " ";
153
-
154
- if (isOther && editMode) {
155
- add(prefix + theme.fg("accent", `${i + 1}. ${opt.label} ✎`));
156
- } else if (selected) {
157
- add(prefix + theme.fg("accent", `${i + 1}. ${opt.label}`));
158
- } else {
159
- add(` ${theme.fg("text", `${i + 1}. ${opt.label}`)}`);
160
- }
161
-
162
- // Show description if present
163
- if (opt.description) {
164
- add(` ${theme.fg("muted", opt.description)}`);
165
- }
166
- }
167
-
168
- if (editMode) {
169
- lines.push("");
170
- add(theme.fg("muted", " Your answer:"));
171
- for (const line of editor.render(width - 2)) {
172
- add(` ${line}`);
173
- }
174
- }
175
-
176
- lines.push("");
177
- if (editMode) {
178
- add(theme.fg("dim", " Enter to submit • Esc to go back"));
179
- } else {
180
- add(theme.fg("dim", " ↑↓ navigate • Enter to select • Esc to cancel"));
181
- }
182
- add(theme.fg("accent", "─".repeat(width)));
183
-
184
- cachedLines = lines;
185
- return lines;
186
- }
187
-
188
- return {
189
- render,
190
- invalidate: () => {
191
- cachedLines = undefined;
192
- },
193
- handleInput,
194
- };
195
- },
196
- );
197
-
198
- // Build simple options list for details
199
- const simpleOptions = params.options.map((o) => o.label);
200
-
201
- if (!result) {
202
- return {
203
- content: [{ type: "text", text: "User cancelled the selection" }],
204
- details: { question: params.question, options: simpleOptions, answer: null } as QuestionDetails,
205
- };
206
- }
207
-
208
- if (result.wasCustom) {
209
- return {
210
- content: [{ type: "text", text: `User wrote: ${result.answer}` }],
211
- details: {
212
- question: params.question,
213
- options: simpleOptions,
214
- answer: result.answer,
215
- wasCustom: true,
216
- } as QuestionDetails,
217
- };
218
- }
219
- return {
220
- content: [{ type: "text", text: `User selected: ${result.index}. ${result.answer}` }],
221
- details: {
222
- question: params.question,
223
- options: simpleOptions,
224
- answer: result.answer,
225
- wasCustom: false,
226
- } as QuestionDetails,
227
- };
228
- },
229
-
230
- renderCall(args, theme, _context) {
231
- let text = theme.fg("toolTitle", theme.bold("question ")) + theme.fg("muted", args.question);
232
- const opts = Array.isArray(args.options) ? args.options : [];
233
- if (opts.length) {
234
- const labels = opts.map((o: OptionWithDesc) => o.label);
235
- const numbered = [...labels, "Type something."].map((o, i) => `${i + 1}. ${o}`);
236
- text += `\n${theme.fg("dim", ` Options: ${numbered.join(", ")}`)}`;
237
- }
238
- return new Text(text, 0, 0);
239
- },
240
-
241
- renderResult(result, _options, theme, _context) {
242
- const details = result.details as QuestionDetails | undefined;
243
- if (!details) {
244
- const text = result.content[0];
245
- return new Text(text?.type === "text" ? text.text : "", 0, 0);
246
- }
247
-
248
- if (details.answer === null) {
249
- return new Text(theme.fg("warning", "Cancelled"), 0, 0);
250
- }
251
-
252
- if (details.wasCustom) {
253
- return new Text(
254
- theme.fg("success", "✓ ") + theme.fg("muted", "(wrote) ") + theme.fg("accent", details.answer),
255
- 0,
256
- 0,
257
- );
258
- }
259
- const idx = details.options.indexOf(details.answer) + 1;
260
- const display = idx > 0 ? `${idx}. ${details.answer}` : details.answer;
261
- return new Text(theme.fg("success", "✓ ") + theme.fg("accent", display), 0, 0);
262
- },
263
- });
264
- }