@apholdings/jensen-code 0.0.3 → 0.0.5

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 (166) hide show
  1. package/dist/cli/args.d.ts.map +1 -1
  2. package/dist/cli/args.js +6 -6
  3. package/dist/cli/args.js.map +1 -1
  4. package/dist/config.d.ts +6 -5
  5. package/dist/config.d.ts.map +1 -1
  6. package/dist/config.js +32 -25
  7. package/dist/config.js.map +1 -1
  8. package/dist/core/agent-session.d.ts +1 -0
  9. package/dist/core/agent-session.d.ts.map +1 -1
  10. package/dist/core/agent-session.js +25 -0
  11. package/dist/core/agent-session.js.map +1 -1
  12. package/dist/core/extensions/loader.d.ts.map +1 -1
  13. package/dist/core/extensions/loader.js +1 -1
  14. package/dist/core/extensions/loader.js.map +1 -1
  15. package/dist/core/footer-data-provider.d.ts +4 -1
  16. package/dist/core/footer-data-provider.d.ts.map +1 -1
  17. package/dist/core/footer-data-provider.js +25 -11
  18. package/dist/core/footer-data-provider.js.map +1 -1
  19. package/dist/index.d.ts +1 -1
  20. package/dist/index.d.ts.map +1 -1
  21. package/dist/index.js +1 -1
  22. package/dist/index.js.map +1 -1
  23. package/dist/modes/interactive/components/custom-editor.d.ts +1 -0
  24. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -1
  25. package/dist/modes/interactive/components/custom-editor.js +5 -0
  26. package/dist/modes/interactive/components/custom-editor.js.map +1 -1
  27. package/dist/modes/interactive/components/footer.d.ts +0 -2
  28. package/dist/modes/interactive/components/footer.d.ts.map +1 -1
  29. package/dist/modes/interactive/components/footer.js +8 -146
  30. package/dist/modes/interactive/components/footer.js.map +1 -1
  31. package/dist/modes/interactive/components/header.d.ts +9 -3
  32. package/dist/modes/interactive/components/header.d.ts.map +1 -1
  33. package/dist/modes/interactive/components/header.js +125 -196
  34. package/dist/modes/interactive/components/header.js.map +1 -1
  35. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  36. package/dist/modes/interactive/components/tool-execution.js +1 -2
  37. package/dist/modes/interactive/components/tool-execution.js.map +1 -1
  38. package/dist/modes/interactive/interactive-mode.d.ts +23 -4
  39. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  40. package/dist/modes/interactive/interactive-mode.js +657 -243
  41. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  42. package/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  43. package/dist/modes/interactive/theme/theme.js +2 -0
  44. package/dist/modes/interactive/theme/theme.js.map +1 -1
  45. package/dist/utils/frontmatter.d.ts.map +1 -1
  46. package/dist/utils/frontmatter.js +8 -4
  47. package/dist/utils/frontmatter.js.map +1 -1
  48. package/dist/utils/tools-manager.d.ts.map +1 -1
  49. package/dist/utils/tools-manager.js +2 -2
  50. package/dist/utils/tools-manager.js.map +1 -1
  51. package/examples/extensions/osgrep.ts +643 -0
  52. package/examples/extensions/subagent/agents.ts +150 -38
  53. package/examples/extensions/subagent/index.ts +634 -514
  54. package/package.json +4 -3
  55. package/examples/README.md +0 -25
  56. package/examples/extensions/README.md +0 -206
  57. package/examples/extensions/antigravity-image-gen.ts +0 -416
  58. package/examples/extensions/auto-commit-on-exit.ts +0 -50
  59. package/examples/extensions/bash-spawn-hook.ts +0 -31
  60. package/examples/extensions/bookmark.ts +0 -51
  61. package/examples/extensions/built-in-tool-renderer.ts +0 -247
  62. package/examples/extensions/claude-rules.ts +0 -87
  63. package/examples/extensions/commands.ts +0 -73
  64. package/examples/extensions/confirm-destructive.ts +0 -60
  65. package/examples/extensions/custom-compaction.ts +0 -115
  66. package/examples/extensions/custom-footer.ts +0 -65
  67. package/examples/extensions/custom-header.ts +0 -74
  68. package/examples/extensions/custom-provider-anthropic/index.ts +0 -605
  69. package/examples/extensions/custom-provider-anthropic/package-lock.json +0 -24
  70. package/examples/extensions/custom-provider-anthropic/package.json +0 -19
  71. package/examples/extensions/custom-provider-gitlab-duo/index.ts +0 -350
  72. package/examples/extensions/custom-provider-gitlab-duo/package.json +0 -16
  73. package/examples/extensions/custom-provider-gitlab-duo/test.ts +0 -82
  74. package/examples/extensions/custom-provider-qwen-cli/index.ts +0 -346
  75. package/examples/extensions/custom-provider-qwen-cli/package.json +0 -16
  76. package/examples/extensions/dirty-repo-guard.ts +0 -57
  77. package/examples/extensions/doom-overlay/README.md +0 -46
  78. package/examples/extensions/doom-overlay/doom/build/doom.js +0 -21
  79. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  80. package/examples/extensions/doom-overlay/doom/build.sh +0 -152
  81. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +0 -72
  82. package/examples/extensions/doom-overlay/doom-component.ts +0 -132
  83. package/examples/extensions/doom-overlay/doom-engine.ts +0 -173
  84. package/examples/extensions/doom-overlay/doom-keys.ts +0 -104
  85. package/examples/extensions/doom-overlay/index.ts +0 -75
  86. package/examples/extensions/doom-overlay/wad-finder.ts +0 -51
  87. package/examples/extensions/dynamic-resources/SKILL.md +0 -8
  88. package/examples/extensions/dynamic-resources/dynamic.json +0 -79
  89. package/examples/extensions/dynamic-resources/dynamic.md +0 -5
  90. package/examples/extensions/dynamic-resources/index.ts +0 -16
  91. package/examples/extensions/dynamic-tools.ts +0 -75
  92. package/examples/extensions/event-bus.ts +0 -44
  93. package/examples/extensions/file-trigger.ts +0 -42
  94. package/examples/extensions/git-checkpoint.ts +0 -54
  95. package/examples/extensions/handoff.ts +0 -151
  96. package/examples/extensions/hello.ts +0 -26
  97. package/examples/extensions/inline-bash.ts +0 -95
  98. package/examples/extensions/input-transform.ts +0 -44
  99. package/examples/extensions/interactive-shell.ts +0 -197
  100. package/examples/extensions/mac-system-theme.ts +0 -48
  101. package/examples/extensions/message-renderer.ts +0 -60
  102. package/examples/extensions/minimal-mode.ts +0 -427
  103. package/examples/extensions/modal-editor.ts +0 -86
  104. package/examples/extensions/model-status.ts +0 -32
  105. package/examples/extensions/notify.ts +0 -56
  106. package/examples/extensions/overlay-qa-tests.ts +0 -1349
  107. package/examples/extensions/overlay-test.ts +0 -151
  108. package/examples/extensions/permission-gate.ts +0 -35
  109. package/examples/extensions/pirate.ts +0 -48
  110. package/examples/extensions/plan-mode/README.md +0 -65
  111. package/examples/extensions/plan-mode/index.ts +0 -341
  112. package/examples/extensions/plan-mode/utils.ts +0 -168
  113. package/examples/extensions/preset.ts +0 -399
  114. package/examples/extensions/protected-paths.ts +0 -31
  115. package/examples/extensions/provider-payload.ts +0 -15
  116. package/examples/extensions/qna.ts +0 -120
  117. package/examples/extensions/question.ts +0 -265
  118. package/examples/extensions/questionnaire.ts +0 -428
  119. package/examples/extensions/rainbow-editor.ts +0 -89
  120. package/examples/extensions/reload-runtime.ts +0 -38
  121. package/examples/extensions/rpc-demo.ts +0 -125
  122. package/examples/extensions/sandbox/index.ts +0 -319
  123. package/examples/extensions/sandbox/package-lock.json +0 -92
  124. package/examples/extensions/sandbox/package.json +0 -19
  125. package/examples/extensions/send-user-message.ts +0 -98
  126. package/examples/extensions/session-name.ts +0 -28
  127. package/examples/extensions/shutdown-command.ts +0 -64
  128. package/examples/extensions/snake.ts +0 -344
  129. package/examples/extensions/space-invaders.ts +0 -561
  130. package/examples/extensions/ssh.ts +0 -221
  131. package/examples/extensions/status-line.ts +0 -41
  132. package/examples/extensions/subagent/README.md +0 -172
  133. package/examples/extensions/subagent/agents/planner.md +0 -37
  134. package/examples/extensions/subagent/agents/reviewer.md +0 -35
  135. package/examples/extensions/subagent/agents/scout.md +0 -50
  136. package/examples/extensions/subagent/agents/worker.md +0 -24
  137. package/examples/extensions/subagent/prompts/implement-and-review.md +0 -10
  138. package/examples/extensions/subagent/prompts/implement.md +0 -10
  139. package/examples/extensions/subagent/prompts/scout-and-plan.md +0 -9
  140. package/examples/extensions/summarize.ts +0 -196
  141. package/examples/extensions/system-prompt-header.ts +0 -18
  142. package/examples/extensions/timed-confirm.ts +0 -71
  143. package/examples/extensions/titlebar-spinner.ts +0 -59
  144. package/examples/extensions/todo.ts +0 -300
  145. package/examples/extensions/tool-override.ts +0 -144
  146. package/examples/extensions/tools.ts +0 -147
  147. package/examples/extensions/trigger-compact.ts +0 -41
  148. package/examples/extensions/truncated-tool.ts +0 -193
  149. package/examples/extensions/widget-placement.ts +0 -18
  150. package/examples/extensions/with-deps/index.ts +0 -33
  151. package/examples/extensions/with-deps/package-lock.json +0 -31
  152. package/examples/extensions/with-deps/package.json +0 -22
  153. package/examples/rpc-extension-ui.ts +0 -632
  154. package/examples/sdk/01-minimal.ts +0 -23
  155. package/examples/sdk/02-custom-model.ts +0 -50
  156. package/examples/sdk/03-custom-prompt.ts +0 -56
  157. package/examples/sdk/04-skills.ts +0 -47
  158. package/examples/sdk/05-tools.ts +0 -57
  159. package/examples/sdk/06-extensions.ts +0 -89
  160. package/examples/sdk/07-context-files.ts +0 -41
  161. package/examples/sdk/08-prompt-templates.ts +0 -48
  162. package/examples/sdk/09-api-keys-and-oauth.ts +0 -49
  163. package/examples/sdk/10-settings.ts +0 -52
  164. package/examples/sdk/11-sessions.ts +0 -49
  165. package/examples/sdk/12-full-control.ts +0 -83
  166. package/examples/sdk/README.md +0 -145
@@ -1,428 +0,0 @@
1
- /**
2
- * Questionnaire Tool - Unified tool for asking single or multiple questions
3
- *
4
- * Single question: simple options list
5
- * Multiple questions: tab bar navigation between questions
6
- */
7
-
8
- import type { ExtensionAPI } from "@apholdings/jensen-code";
9
- import { Editor, type EditorTheme, Key, matchesKey, Text, truncateToWidth } from "@apholdings/jensen-tui";
10
- import { Type } from "@sinclair/typebox";
11
-
12
- // Types
13
- interface QuestionOption {
14
- value: string;
15
- label: string;
16
- description?: string;
17
- }
18
-
19
- type RenderOption = QuestionOption & { isOther?: boolean };
20
-
21
- interface Question {
22
- id: string;
23
- label: string;
24
- prompt: string;
25
- options: QuestionOption[];
26
- allowOther: boolean;
27
- }
28
-
29
- interface Answer {
30
- id: string;
31
- value: string;
32
- label: string;
33
- wasCustom: boolean;
34
- index?: number;
35
- }
36
-
37
- interface QuestionnaireResult {
38
- questions: Question[];
39
- answers: Answer[];
40
- cancelled: boolean;
41
- }
42
-
43
- // Schema
44
- const QuestionOptionSchema = Type.Object({
45
- value: Type.String({ description: "The value returned when selected" }),
46
- label: Type.String({ description: "Display label for the option" }),
47
- description: Type.Optional(Type.String({ description: "Optional description shown below label" })),
48
- });
49
-
50
- const QuestionSchema = Type.Object({
51
- id: Type.String({ description: "Unique identifier for this question" }),
52
- label: Type.Optional(
53
- Type.String({
54
- description: "Short contextual label for tab bar, e.g. 'Scope', 'Priority' (defaults to Q1, Q2)",
55
- }),
56
- ),
57
- prompt: Type.String({ description: "The full question text to display" }),
58
- options: Type.Array(QuestionOptionSchema, { description: "Available options to choose from" }),
59
- allowOther: Type.Optional(Type.Boolean({ description: "Allow 'Type something' option (default: true)" })),
60
- });
61
-
62
- const QuestionnaireParams = Type.Object({
63
- questions: Type.Array(QuestionSchema, { description: "Questions to ask the user" }),
64
- });
65
-
66
- function errorResult(
67
- message: string,
68
- questions: Question[] = [],
69
- ): { content: { type: "text"; text: string }[]; details: QuestionnaireResult } {
70
- return {
71
- content: [{ type: "text", text: message }],
72
- details: { questions, answers: [], cancelled: true },
73
- };
74
- }
75
-
76
- export default function questionnaire(pi: ExtensionAPI) {
77
- pi.registerTool({
78
- name: "questionnaire",
79
- label: "Questionnaire",
80
- description:
81
- "Ask the user one or more questions. Use for clarifying requirements, getting preferences, or confirming decisions. For single questions, shows a simple option list. For multiple questions, shows a tab-based interface.",
82
- parameters: QuestionnaireParams,
83
-
84
- async execute(_toolCallId, params, _signal, _onUpdate, ctx) {
85
- if (!ctx.hasUI) {
86
- return errorResult("Error: UI not available (running in non-interactive mode)");
87
- }
88
- if (params.questions.length === 0) {
89
- return errorResult("Error: No questions provided");
90
- }
91
-
92
- // Normalize questions with defaults
93
- const questions: Question[] = params.questions.map((q, i) => ({
94
- ...q,
95
- label: q.label || `Q${i + 1}`,
96
- allowOther: q.allowOther !== false,
97
- }));
98
-
99
- const isMulti = questions.length > 1;
100
- const totalTabs = questions.length + 1; // questions + Submit
101
-
102
- const result = await ctx.ui.custom<QuestionnaireResult>((tui, theme, _kb, done) => {
103
- // State
104
- let currentTab = 0;
105
- let optionIndex = 0;
106
- let inputMode = false;
107
- let inputQuestionId: string | null = null;
108
- let cachedLines: string[] | undefined;
109
- const answers = new Map<string, Answer>();
110
-
111
- // Editor for "Type something" option
112
- const editorTheme: EditorTheme = {
113
- borderColor: (s) => theme.fg("accent", s),
114
- selectList: {
115
- selectedPrefix: (t) => theme.fg("accent", t),
116
- selectedText: (t) => theme.fg("accent", t),
117
- description: (t) => theme.fg("muted", t),
118
- scrollInfo: (t) => theme.fg("dim", t),
119
- noMatch: (t) => theme.fg("warning", t),
120
- },
121
- };
122
- const editor = new Editor(tui, editorTheme);
123
-
124
- // Helpers
125
- function refresh() {
126
- cachedLines = undefined;
127
- tui.requestRender();
128
- }
129
-
130
- function submit(cancelled: boolean) {
131
- done({ questions, answers: Array.from(answers.values()), cancelled });
132
- }
133
-
134
- function currentQuestion(): Question | undefined {
135
- return questions[currentTab];
136
- }
137
-
138
- function currentOptions(): RenderOption[] {
139
- const q = currentQuestion();
140
- if (!q) return [];
141
- const opts: RenderOption[] = [...q.options];
142
- if (q.allowOther) {
143
- opts.push({ value: "__other__", label: "Type something.", isOther: true });
144
- }
145
- return opts;
146
- }
147
-
148
- function allAnswered(): boolean {
149
- return questions.every((q) => answers.has(q.id));
150
- }
151
-
152
- function advanceAfterAnswer() {
153
- if (!isMulti) {
154
- submit(false);
155
- return;
156
- }
157
- if (currentTab < questions.length - 1) {
158
- currentTab++;
159
- } else {
160
- currentTab = questions.length; // Submit tab
161
- }
162
- optionIndex = 0;
163
- refresh();
164
- }
165
-
166
- function saveAnswer(questionId: string, value: string, label: string, wasCustom: boolean, index?: number) {
167
- answers.set(questionId, { id: questionId, value, label, wasCustom, index });
168
- }
169
-
170
- // Editor submit callback
171
- editor.onSubmit = (value) => {
172
- if (!inputQuestionId) return;
173
- const trimmed = value.trim() || "(no response)";
174
- saveAnswer(inputQuestionId, trimmed, trimmed, true);
175
- inputMode = false;
176
- inputQuestionId = null;
177
- editor.setText("");
178
- advanceAfterAnswer();
179
- };
180
-
181
- function handleInput(data: string) {
182
- // Input mode: route to editor
183
- if (inputMode) {
184
- if (matchesKey(data, Key.escape)) {
185
- inputMode = false;
186
- inputQuestionId = null;
187
- editor.setText("");
188
- refresh();
189
- return;
190
- }
191
- editor.handleInput(data);
192
- refresh();
193
- return;
194
- }
195
-
196
- const q = currentQuestion();
197
- const opts = currentOptions();
198
-
199
- // Tab navigation (multi-question only)
200
- if (isMulti) {
201
- if (matchesKey(data, Key.tab) || matchesKey(data, Key.right)) {
202
- currentTab = (currentTab + 1) % totalTabs;
203
- optionIndex = 0;
204
- refresh();
205
- return;
206
- }
207
- if (matchesKey(data, Key.shift("tab")) || matchesKey(data, Key.left)) {
208
- currentTab = (currentTab - 1 + totalTabs) % totalTabs;
209
- optionIndex = 0;
210
- refresh();
211
- return;
212
- }
213
- }
214
-
215
- // Submit tab
216
- if (currentTab === questions.length) {
217
- if (matchesKey(data, Key.enter) && allAnswered()) {
218
- submit(false);
219
- } else if (matchesKey(data, Key.escape)) {
220
- submit(true);
221
- }
222
- return;
223
- }
224
-
225
- // Option navigation
226
- if (matchesKey(data, Key.up)) {
227
- optionIndex = Math.max(0, optionIndex - 1);
228
- refresh();
229
- return;
230
- }
231
- if (matchesKey(data, Key.down)) {
232
- optionIndex = Math.min(opts.length - 1, optionIndex + 1);
233
- refresh();
234
- return;
235
- }
236
-
237
- // Select option
238
- if (matchesKey(data, Key.enter) && q) {
239
- const opt = opts[optionIndex];
240
- if (opt.isOther) {
241
- inputMode = true;
242
- inputQuestionId = q.id;
243
- editor.setText("");
244
- refresh();
245
- return;
246
- }
247
- saveAnswer(q.id, opt.value, opt.label, false, optionIndex + 1);
248
- advanceAfterAnswer();
249
- return;
250
- }
251
-
252
- // Cancel
253
- if (matchesKey(data, Key.escape)) {
254
- submit(true);
255
- }
256
- }
257
-
258
- function render(width: number): string[] {
259
- if (cachedLines) return cachedLines;
260
-
261
- const lines: string[] = [];
262
- const q = currentQuestion();
263
- const opts = currentOptions();
264
-
265
- // Helper to add truncated line
266
- const add = (s: string) => lines.push(truncateToWidth(s, width));
267
-
268
- add(theme.fg("accent", "─".repeat(width)));
269
-
270
- // Tab bar (multi-question only)
271
- if (isMulti) {
272
- const tabs: string[] = ["← "];
273
- for (let i = 0; i < questions.length; i++) {
274
- const isActive = i === currentTab;
275
- const isAnswered = answers.has(questions[i].id);
276
- const lbl = questions[i].label;
277
- const box = isAnswered ? "■" : "□";
278
- const color = isAnswered ? "success" : "muted";
279
- const text = ` ${box} ${lbl} `;
280
- const styled = isActive ? theme.bg("selectedBg", theme.fg("text", text)) : theme.fg(color, text);
281
- tabs.push(`${styled} `);
282
- }
283
- const canSubmit = allAnswered();
284
- const isSubmitTab = currentTab === questions.length;
285
- const submitText = " ✓ Submit ";
286
- const submitStyled = isSubmitTab
287
- ? theme.bg("selectedBg", theme.fg("text", submitText))
288
- : theme.fg(canSubmit ? "success" : "dim", submitText);
289
- tabs.push(`${submitStyled} →`);
290
- add(` ${tabs.join("")}`);
291
- lines.push("");
292
- }
293
-
294
- // Helper to render options list
295
- function renderOptions() {
296
- for (let i = 0; i < opts.length; i++) {
297
- const opt = opts[i];
298
- const selected = i === optionIndex;
299
- const isOther = opt.isOther === true;
300
- const prefix = selected ? theme.fg("accent", "> ") : " ";
301
- const color = selected ? "accent" : "text";
302
- // Mark "Type something" differently when in input mode
303
- if (isOther && inputMode) {
304
- add(prefix + theme.fg("accent", `${i + 1}. ${opt.label} ✎`));
305
- } else {
306
- add(prefix + theme.fg(color, `${i + 1}. ${opt.label}`));
307
- }
308
- if (opt.description) {
309
- add(` ${theme.fg("muted", opt.description)}`);
310
- }
311
- }
312
- }
313
-
314
- // Content
315
- if (inputMode && q) {
316
- add(theme.fg("text", ` ${q.prompt}`));
317
- lines.push("");
318
- // Show options for reference
319
- renderOptions();
320
- lines.push("");
321
- add(theme.fg("muted", " Your answer:"));
322
- for (const line of editor.render(width - 2)) {
323
- add(` ${line}`);
324
- }
325
- lines.push("");
326
- add(theme.fg("dim", " Enter to submit • Esc to cancel"));
327
- } else if (currentTab === questions.length) {
328
- add(theme.fg("accent", theme.bold(" Ready to submit")));
329
- lines.push("");
330
- for (const question of questions) {
331
- const answer = answers.get(question.id);
332
- if (answer) {
333
- const prefix = answer.wasCustom ? "(wrote) " : "";
334
- add(`${theme.fg("muted", ` ${question.label}: `)}${theme.fg("text", prefix + answer.label)}`);
335
- }
336
- }
337
- lines.push("");
338
- if (allAnswered()) {
339
- add(theme.fg("success", " Press Enter to submit"));
340
- } else {
341
- const missing = questions
342
- .filter((q) => !answers.has(q.id))
343
- .map((q) => q.label)
344
- .join(", ");
345
- add(theme.fg("warning", ` Unanswered: ${missing}`));
346
- }
347
- } else if (q) {
348
- add(theme.fg("text", ` ${q.prompt}`));
349
- lines.push("");
350
- renderOptions();
351
- }
352
-
353
- lines.push("");
354
- if (!inputMode) {
355
- const help = isMulti
356
- ? " Tab/←→ navigate • ↑↓ select • Enter confirm • Esc cancel"
357
- : " ↑↓ navigate • Enter select • Esc cancel";
358
- add(theme.fg("dim", help));
359
- }
360
- add(theme.fg("accent", "─".repeat(width)));
361
-
362
- cachedLines = lines;
363
- return lines;
364
- }
365
-
366
- return {
367
- render,
368
- invalidate: () => {
369
- cachedLines = undefined;
370
- },
371
- handleInput,
372
- };
373
- });
374
-
375
- if (result.cancelled) {
376
- return {
377
- content: [{ type: "text", text: "User cancelled the questionnaire" }],
378
- details: result,
379
- };
380
- }
381
-
382
- const answerLines = result.answers.map((a) => {
383
- const qLabel = questions.find((q) => q.id === a.id)?.label || a.id;
384
- if (a.wasCustom) {
385
- return `${qLabel}: user wrote: ${a.label}`;
386
- }
387
- return `${qLabel}: user selected: ${a.index}. ${a.label}`;
388
- });
389
-
390
- return {
391
- content: [{ type: "text", text: answerLines.join("\n") }],
392
- details: result,
393
- };
394
- },
395
-
396
- renderCall(args, theme) {
397
- const qs = (args.questions as Question[]) || [];
398
- const count = qs.length;
399
- const labels = qs.map((q) => q.label || q.id).join(", ");
400
- let text = theme.fg("toolTitle", theme.bold("questionnaire "));
401
- text += theme.fg("muted", `${count} question${count !== 1 ? "s" : ""}`);
402
- if (labels) {
403
- text += theme.fg("dim", ` (${truncateToWidth(labels, 40)})`);
404
- }
405
- return new Text(text, 0, 0);
406
- },
407
-
408
- renderResult(result, _options, theme) {
409
- const details = result.details as QuestionnaireResult | undefined;
410
- if (!details) {
411
- const text = result.content[0];
412
- return new Text(text?.type === "text" ? text.text : "", 0, 0);
413
- }
414
- if (details.cancelled) {
415
- return new Text(theme.fg("warning", "Cancelled"), 0, 0);
416
- }
417
- const lines = details.answers.map((a) => {
418
- if (a.wasCustom) {
419
- return `${theme.fg("success", "✓ ")}${theme.fg("accent", a.id)}: ${theme.fg("muted", "(wrote) ")}${a.label}`;
420
- }
421
- const display = a.index ? `${a.index}. ${a.label}` : a.label;
422
- return `${theme.fg("success", "✓ ")}${theme.fg("accent", a.id)}: ${display}`;
423
- });
424
- return new Text(lines.join("\n"), 0, 0);
425
- },
426
- });
427
- }
428
-
@@ -1,89 +0,0 @@
1
- /**
2
- * Rainbow Editor - highlights "ultrathink" with animated shine effect
3
- *
4
- * Usage: pi --extension ./examples/extensions/rainbow-editor.ts
5
- */
6
-
7
- import { CustomEditor, type ExtensionAPI } from "@apholdings/jensen-code";
8
-
9
- // Base colors (coral → yellow → green → teal → blue → purple → pink)
10
- const COLORS: [number, number, number][] = [
11
- [233, 137, 115], // coral
12
- [228, 186, 103], // yellow
13
- [141, 192, 122], // green
14
- [102, 194, 179], // teal
15
- [121, 157, 207], // blue
16
- [157, 134, 195], // purple
17
- [206, 130, 172], // pink
18
- ];
19
- const RESET = "\x1b[0m";
20
-
21
- function brighten(rgb: [number, number, number], factor: number): string {
22
- const [r, g, b] = rgb.map((c) => Math.round(c + (255 - c) * factor));
23
- return `\x1b[38;2;${r};${g};${b}m`;
24
- }
25
-
26
- function colorize(text: string, shinePos: number): string {
27
- return (
28
- [...text]
29
- .map((c, i) => {
30
- const baseColor = COLORS[i % COLORS.length]!;
31
- // 3-letter shine: center bright, adjacent dimmer
32
- let factor = 0;
33
- if (shinePos >= 0) {
34
- const dist = Math.abs(i - shinePos);
35
- if (dist === 0) factor = 0.7;
36
- else if (dist === 1) factor = 0.35;
37
- }
38
- return `${brighten(baseColor, factor)}${c}`;
39
- })
40
- .join("") + RESET
41
- );
42
- }
43
-
44
- class RainbowEditor extends CustomEditor {
45
- private animationTimer?: ReturnType<typeof setInterval>;
46
- private frame = 0;
47
-
48
- private hasUltrathink(): boolean {
49
- return /ultrathink/i.test(this.getText());
50
- }
51
-
52
- private startAnimation(): void {
53
- if (this.animationTimer) return;
54
- this.animationTimer = setInterval(() => {
55
- this.frame++;
56
- this.tui.requestRender();
57
- }, 60);
58
- }
59
-
60
- private stopAnimation(): void {
61
- if (this.animationTimer) {
62
- clearInterval(this.animationTimer);
63
- this.animationTimer = undefined;
64
- }
65
- }
66
-
67
- handleInput(data: string): void {
68
- super.handleInput(data);
69
- if (this.hasUltrathink()) {
70
- this.startAnimation();
71
- } else {
72
- this.stopAnimation();
73
- }
74
- }
75
-
76
- render(width: number): string[] {
77
- // Cycle: 10 shine positions + 10 pause frames
78
- const cycle = this.frame % 20;
79
- const shinePos = cycle < 10 ? cycle : -1; // -1 means no shine (pause)
80
- return super.render(width).map((line) => line.replace(/ultrathink/gi, (m) => colorize(m, shinePos)));
81
- }
82
- }
83
-
84
- export default function (pi: ExtensionAPI) {
85
- pi.on("session_start", (_event, ctx) => {
86
- ctx.ui.setEditorComponent((tui, theme, kb) => new RainbowEditor(tui, theme, kb));
87
- });
88
- }
89
-
@@ -1,38 +0,0 @@
1
- /**
2
- * Reload Runtime Extension
3
- *
4
- * Demonstrates ctx.reload() from ExtensionCommandContext and an LLM-callable
5
- * tool that queues a follow-up command to trigger reload.
6
- */
7
-
8
- import type { ExtensionAPI } from "@apholdings/jensen-code";
9
- import { Type } from "@sinclair/typebox";
10
-
11
- export default function (pi: ExtensionAPI) {
12
- // Command entrypoint for reload.
13
- // Treat reload as terminal for this handler.
14
- pi.registerCommand("reload-runtime", {
15
- description: "Reload extensions, skills, prompts, and themes",
16
- handler: async (_args, ctx) => {
17
- await ctx.reload();
18
- return;
19
- },
20
- });
21
-
22
- // LLM-callable tool. Tools get ExtensionContext, so they cannot call ctx.reload() directly.
23
- // Instead, queue a follow-up user command that executes the command above.
24
- pi.registerTool({
25
- name: "reload_runtime",
26
- label: "Reload Runtime",
27
- description: "Reload extensions, skills, prompts, and themes",
28
- parameters: Type.Object({}),
29
- async execute() {
30
- pi.sendUserMessage("/reload-runtime", { deliverAs: "followUp" });
31
- return {
32
- content: [{ type: "text", text: "Queued /reload-runtime as a follow-up command." }],
33
- details: {},
34
- };
35
- },
36
- });
37
- }
38
-
@@ -1,125 +0,0 @@
1
- /**
2
- * RPC Extension UI Demo
3
- *
4
- * Purpose-built extension that exercises all RPC-supported extension UI methods.
5
- * Designed to be loaded alongside the rpc-extension-ui-example.ts script to
6
- * demonstrate the full extension UI protocol.
7
- *
8
- * UI methods exercised:
9
- * - select() - on tool_call for dangerous bash commands
10
- * - confirm() - on session_before_switch
11
- * - input() - via /rpc-input command
12
- * - editor() - via /rpc-editor command
13
- * - notify() - after each dialog completes
14
- * - setStatus() - on turn_start/turn_end
15
- * - setWidget() - on session_start
16
- * - setTitle() - on session_start and session_switch
17
- * - setEditorText() - via /rpc-prefill command
18
- */
19
-
20
- import type { ExtensionAPI } from "@apholdings/jensen-code";
21
-
22
- export default function (pi: ExtensionAPI) {
23
- let turnCount = 0;
24
-
25
- // -- setTitle, setWidget, setStatus on session lifecycle --
26
-
27
- pi.on("session_start", async (_event, ctx) => {
28
- ctx.ui.setTitle("pi RPC Demo");
29
- ctx.ui.setWidget("rpc-demo", ["--- RPC Extension UI Demo ---", "Loaded and ready."]);
30
- ctx.ui.setStatus("rpc-demo", `Turns: ${turnCount}`);
31
- });
32
-
33
- pi.on("session_switch", async (_event, ctx) => {
34
- turnCount = 0;
35
- ctx.ui.setTitle("pi RPC Demo (new session)");
36
- ctx.ui.setStatus("rpc-demo", `Turns: ${turnCount}`);
37
- });
38
-
39
- // -- setStatus on turn lifecycle --
40
-
41
- pi.on("turn_start", async (_event, ctx) => {
42
- turnCount++;
43
- ctx.ui.setStatus("rpc-demo", `Turn ${turnCount} running...`);
44
- });
45
-
46
- pi.on("turn_end", async (_event, ctx) => {
47
- ctx.ui.setStatus("rpc-demo", `Turn ${turnCount} done`);
48
- });
49
-
50
- // -- select on dangerous tool calls --
51
-
52
- pi.on("tool_call", async (event, ctx) => {
53
- if (event.toolName !== "bash") return undefined;
54
-
55
- const command = event.input.command as string;
56
- const isDangerous = /\brm\s+(-rf?|--recursive)/i.test(command) || /\bsudo\b/i.test(command);
57
-
58
- if (isDangerous) {
59
- if (!ctx.hasUI) {
60
- return { block: true, reason: "Dangerous command blocked (no UI)" };
61
- }
62
-
63
- const choice = await ctx.ui.select(`Dangerous command: ${command}`, ["Allow", "Block"]);
64
- if (choice !== "Allow") {
65
- ctx.ui.notify("Command blocked by user", "warning");
66
- return { block: true, reason: "Blocked by user" };
67
- }
68
- ctx.ui.notify("Command allowed", "info");
69
- }
70
-
71
- return undefined;
72
- });
73
-
74
- // -- confirm on session clear --
75
-
76
- pi.on("session_before_switch", async (event, ctx) => {
77
- if (event.reason !== "new") return;
78
- if (!ctx.hasUI) return;
79
-
80
- const confirmed = await ctx.ui.confirm("Clear session?", "All messages will be lost.");
81
- if (!confirmed) {
82
- ctx.ui.notify("Clear cancelled", "info");
83
- return { cancel: true };
84
- }
85
- });
86
-
87
- // -- input via command --
88
-
89
- pi.registerCommand("rpc-input", {
90
- description: "Prompt for text input (demonstrates ctx.ui.input in RPC)",
91
- handler: async (_args, ctx) => {
92
- const value = await ctx.ui.input("Enter a value", "type something...");
93
- if (value) {
94
- ctx.ui.notify(`You entered: ${value}`, "info");
95
- } else {
96
- ctx.ui.notify("Input cancelled", "info");
97
- }
98
- },
99
- });
100
-
101
- // -- editor via command --
102
-
103
- pi.registerCommand("rpc-editor", {
104
- description: "Open multi-line editor (demonstrates ctx.ui.editor in RPC)",
105
- handler: async (_args, ctx) => {
106
- const text = await ctx.ui.editor("Edit some text", "Line 1\nLine 2\nLine 3");
107
- if (text) {
108
- ctx.ui.notify(`Editor submitted (${text.split("\n").length} lines)`, "info");
109
- } else {
110
- ctx.ui.notify("Editor cancelled", "info");
111
- }
112
- },
113
- });
114
-
115
- // -- setEditorText via command --
116
-
117
- pi.registerCommand("rpc-prefill", {
118
- description: "Prefill the input editor (demonstrates ctx.ui.setEditorText in RPC)",
119
- handler: async (_args, ctx) => {
120
- ctx.ui.setEditorText("This text was set by the rpc-demo extension.");
121
- ctx.ui.notify("Editor prefilled", "info");
122
- },
123
- });
124
- }
125
-