@pi-agents/orchid 0.1.0-beta.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (163) hide show
  1. package/CHANGELOG.md +41 -0
  2. package/LICENSE +21 -0
  3. package/README.md +246 -0
  4. package/agents/AGENTS-MANIFEST.md +42 -0
  5. package/agents/brain.md +42 -0
  6. package/agents/context-builder.md +46 -0
  7. package/agents/delegate.md +12 -0
  8. package/agents/dev-1.md +42 -0
  9. package/agents/oracle.md +73 -0
  10. package/agents/planner.md +55 -0
  11. package/agents/researcher.md +52 -0
  12. package/agents/reviewer.md +79 -0
  13. package/agents/scout.md +50 -0
  14. package/agents/tester.md +45 -0
  15. package/agents/worker.md +55 -0
  16. package/extensions/ralph.ts +1 -0
  17. package/extensions/reviewer-extension.ts +125 -0
  18. package/extensions/task-orchestrator.ts +28 -0
  19. package/package.json +63 -0
  20. package/prompts/gather-context-and-clarify.md +13 -0
  21. package/prompts/parallel-cleanup.md +59 -0
  22. package/prompts/parallel-context-build.md +53 -0
  23. package/prompts/parallel-handoff-plan.md +59 -0
  24. package/prompts/parallel-research.md +50 -0
  25. package/prompts/parallel-review.md +54 -0
  26. package/prompts/review-loop.md +41 -0
  27. package/skills/orchid/SKILL.md +214 -0
  28. package/skills/orchid/orchid-cleanup/SKILL.md +122 -0
  29. package/skills/orchid/orchid-converge/SKILL.md +124 -0
  30. package/skills/orchid/orchid-decompose/SKILL.md +201 -0
  31. package/skills/orchid/orchid-doctor/SKILL.md +162 -0
  32. package/skills/orchid/orchid-investigate/SKILL.md +102 -0
  33. package/skills/orchid/orchid-launch/SKILL.md +147 -0
  34. package/skills/ralph/SKILL.md +73 -0
  35. package/skills/subagents/pi-subagents/SKILL.md +813 -0
  36. package/src/index.ts +7 -0
  37. package/src/orchestrator/abort.ts +534 -0
  38. package/src/orchestrator/agent-bridge-extension.ts +1020 -0
  39. package/src/orchestrator/agent-host.ts +954 -0
  40. package/src/orchestrator/cleanup.ts +776 -0
  41. package/src/orchestrator/config-loader.ts +1412 -0
  42. package/src/orchestrator/config-schema.ts +690 -0
  43. package/src/orchestrator/config.ts +81 -0
  44. package/src/orchestrator/context-window.ts +66 -0
  45. package/src/orchestrator/diagnostic-reports.ts +475 -0
  46. package/src/orchestrator/diagnostics.ts +394 -0
  47. package/src/orchestrator/discovery.ts +1833 -0
  48. package/src/orchestrator/engine-worker.ts +415 -0
  49. package/src/orchestrator/engine.ts +5940 -0
  50. package/src/orchestrator/execution.ts +3104 -0
  51. package/src/orchestrator/extension.ts +5934 -0
  52. package/src/orchestrator/formatting.ts +785 -0
  53. package/src/orchestrator/git.ts +88 -0
  54. package/src/orchestrator/index.ts +28 -0
  55. package/src/orchestrator/lane-runner.ts +1787 -0
  56. package/src/orchestrator/mailbox.ts +780 -0
  57. package/src/orchestrator/merge.ts +3414 -0
  58. package/src/orchestrator/messages.ts +1062 -0
  59. package/src/orchestrator/migrations.ts +278 -0
  60. package/src/orchestrator/naming.ts +117 -0
  61. package/src/orchestrator/path-resolver.ts +275 -0
  62. package/src/orchestrator/persistence.ts +2625 -0
  63. package/src/orchestrator/process-registry.ts +452 -0
  64. package/src/orchestrator/quality-gate.ts +1085 -0
  65. package/src/orchestrator/resume.ts +3488 -0
  66. package/src/orchestrator/sessions.ts +57 -0
  67. package/src/orchestrator/settings-loader.ts +136 -0
  68. package/src/orchestrator/settings-tui.ts +2208 -0
  69. package/src/orchestrator/sidecar-telemetry.ts +267 -0
  70. package/src/orchestrator/supervisor.ts +4548 -0
  71. package/src/orchestrator/task-executor-core.ts +675 -0
  72. package/src/orchestrator/tmux-compat.ts +37 -0
  73. package/src/orchestrator/tool-allowlist-constants.ts +37 -0
  74. package/src/orchestrator/types.ts +4465 -0
  75. package/src/orchestrator/verification.ts +547 -0
  76. package/src/orchestrator/waves.ts +1564 -0
  77. package/src/orchestrator/workspace.ts +707 -0
  78. package/src/orchestrator/worktree.ts +2725 -0
  79. package/src/ralph/index.ts +825 -0
  80. package/src/subagents/agents/agent-management.ts +648 -0
  81. package/src/subagents/agents/agent-scope.ts +6 -0
  82. package/src/subagents/agents/agent-selection.ts +23 -0
  83. package/src/subagents/agents/agent-serializer.ts +86 -0
  84. package/src/subagents/agents/agents.ts +832 -0
  85. package/src/subagents/agents/chain-serializer.ts +137 -0
  86. package/src/subagents/agents/frontmatter.ts +29 -0
  87. package/src/subagents/agents/identity.ts +30 -0
  88. package/src/subagents/agents/skills.ts +632 -0
  89. package/src/subagents/extension/config.ts +16 -0
  90. package/src/subagents/extension/control-notices.ts +92 -0
  91. package/src/subagents/extension/doctor.ts +199 -0
  92. package/src/subagents/extension/fanout-child.ts +170 -0
  93. package/src/subagents/extension/index.ts +573 -0
  94. package/src/subagents/extension/schemas.ts +168 -0
  95. package/src/subagents/intercom/intercom-bridge.ts +379 -0
  96. package/src/subagents/intercom/result-intercom.ts +377 -0
  97. package/src/subagents/runs/background/async-execution.ts +712 -0
  98. package/src/subagents/runs/background/async-job-tracker.ts +310 -0
  99. package/src/subagents/runs/background/async-resume.ts +345 -0
  100. package/src/subagents/runs/background/async-status.ts +325 -0
  101. package/src/subagents/runs/background/completion-dedupe.ts +63 -0
  102. package/src/subagents/runs/background/notify.ts +108 -0
  103. package/src/subagents/runs/background/parallel-groups.ts +45 -0
  104. package/src/subagents/runs/background/result-watcher.ts +307 -0
  105. package/src/subagents/runs/background/run-id-resolver.ts +83 -0
  106. package/src/subagents/runs/background/run-status.ts +269 -0
  107. package/src/subagents/runs/background/stale-run-reconciler.ts +336 -0
  108. package/src/subagents/runs/background/subagent-runner.ts +1808 -0
  109. package/src/subagents/runs/background/top-level-async.ts +13 -0
  110. package/src/subagents/runs/foreground/chain-clarify.ts +1333 -0
  111. package/src/subagents/runs/foreground/chain-execution.ts +938 -0
  112. package/src/subagents/runs/foreground/execution.ts +918 -0
  113. package/src/subagents/runs/foreground/subagent-executor.ts +2527 -0
  114. package/src/subagents/runs/shared/completion-guard.ts +147 -0
  115. package/src/subagents/runs/shared/long-running-guard.ts +175 -0
  116. package/src/subagents/runs/shared/mcp-direct-tool-allowlist.ts +365 -0
  117. package/src/subagents/runs/shared/model-fallback.ts +103 -0
  118. package/src/subagents/runs/shared/nested-events.ts +819 -0
  119. package/src/subagents/runs/shared/nested-path.ts +52 -0
  120. package/src/subagents/runs/shared/nested-render.ts +115 -0
  121. package/src/subagents/runs/shared/parallel-utils.ts +109 -0
  122. package/src/subagents/runs/shared/pi-args.ts +220 -0
  123. package/src/subagents/runs/shared/pi-spawn.ts +115 -0
  124. package/src/subagents/runs/shared/run-history.ts +60 -0
  125. package/src/subagents/runs/shared/single-output.ts +164 -0
  126. package/src/subagents/runs/shared/subagent-control.ts +226 -0
  127. package/src/subagents/runs/shared/subagent-prompt-runtime.ts +170 -0
  128. package/src/subagents/runs/shared/worktree.ts +577 -0
  129. package/src/subagents/shared/artifacts.ts +98 -0
  130. package/src/subagents/shared/atomic-json.ts +16 -0
  131. package/src/subagents/shared/file-coalescer.ts +40 -0
  132. package/src/subagents/shared/fork-context.ts +76 -0
  133. package/src/subagents/shared/formatters.ts +133 -0
  134. package/src/subagents/shared/jsonl-writer.ts +81 -0
  135. package/src/subagents/shared/model-info.ts +78 -0
  136. package/src/subagents/shared/post-exit-stdio-guard.ts +85 -0
  137. package/src/subagents/shared/session-identity.ts +10 -0
  138. package/src/subagents/shared/session-tokens.ts +44 -0
  139. package/src/subagents/shared/settings.ts +397 -0
  140. package/src/subagents/shared/status-format.ts +49 -0
  141. package/src/subagents/shared/types.ts +822 -0
  142. package/src/subagents/shared/utils.ts +450 -0
  143. package/src/subagents/slash/prompt-template-bridge.ts +397 -0
  144. package/src/subagents/slash/slash-bridge.ts +174 -0
  145. package/src/subagents/slash/slash-commands.ts +528 -0
  146. package/src/subagents/slash/slash-live-state.ts +292 -0
  147. package/src/subagents/tui/render-helpers.ts +80 -0
  148. package/src/subagents/tui/render.ts +1358 -0
  149. package/templates/agents/local/supervisor.md +33 -0
  150. package/templates/agents/local/task-merger.md +27 -0
  151. package/templates/agents/local/task-reviewer.md +30 -0
  152. package/templates/agents/local/task-worker.md +34 -0
  153. package/templates/agents/supervisor-routing.md +92 -0
  154. package/templates/agents/supervisor.md +229 -0
  155. package/templates/agents/task-merger.md +214 -0
  156. package/templates/agents/task-reviewer.md +260 -0
  157. package/templates/agents/task-worker-segment.md +44 -0
  158. package/templates/agents/task-worker.md +557 -0
  159. package/templates/tasks/CONTEXT.md +30 -0
  160. package/templates/tasks/EXAMPLE-001-hello-world/PROMPT.md +98 -0
  161. package/templates/tasks/EXAMPLE-001-hello-world/STATUS.md +73 -0
  162. package/templates/tasks/EXAMPLE-002-parallel-smoke/PROMPT.md +97 -0
  163. package/templates/tasks/EXAMPLE-002-parallel-smoke/STATUS.md +73 -0
@@ -0,0 +1,292 @@
1
+ import type { AgentToolResult } from "@earendil-works/pi-agent-core";
2
+ import type { Message } from "@earendil-works/pi-ai";
3
+ import type { SubagentParamsLike } from "../runs/foreground/subagent-executor.ts";
4
+ import type { SlashSubagentResponse, SlashSubagentUpdate } from "./slash-bridge.ts";
5
+ import { type Details, type SingleResult, type Usage, SLASH_RESULT_TYPE } from "../shared/types.ts";
6
+
7
+ export interface SlashMessageDetails {
8
+ requestId: string;
9
+ result: AgentToolResult<Details>;
10
+ }
11
+
12
+ interface SlashSnapshot {
13
+ result: AgentToolResult<Details>;
14
+ version: number;
15
+ }
16
+
17
+ interface SequentialChainStepLike {
18
+ agent: string;
19
+ task?: string;
20
+ }
21
+
22
+ interface ParallelChainStepLike {
23
+ parallel: Array<{ agent: string; task?: string }>;
24
+ }
25
+
26
+ type ChainStepLike = SequentialChainStepLike | ParallelChainStepLike;
27
+
28
+ const liveSnapshots = new Map<string, SlashSnapshot>();
29
+ const finalSnapshots = new Map<string, SlashSnapshot>();
30
+ let versionCounter = 1;
31
+
32
+ const EMPTY_MESSAGES: Message[] = [];
33
+ const EMPTY_USAGE: Usage = {
34
+ input: 0,
35
+ output: 0,
36
+ cacheRead: 0,
37
+ cacheWrite: 0,
38
+ cost: 0,
39
+ turns: 0,
40
+ };
41
+
42
+ function nextVersion(): number {
43
+ return versionCounter++;
44
+ }
45
+
46
+ function cloneUsage(): Usage {
47
+ return { ...EMPTY_USAGE };
48
+ }
49
+
50
+ function createPlaceholderResult(
51
+ agent: string,
52
+ task: string,
53
+ status: "pending" | "running",
54
+ index?: number,
55
+ ): SingleResult {
56
+ return {
57
+ agent,
58
+ task,
59
+ exitCode: 0,
60
+ messages: EMPTY_MESSAGES,
61
+ usage: cloneUsage(),
62
+ progress: {
63
+ ...(index !== undefined ? { index } : {}),
64
+ agent,
65
+ status,
66
+ task,
67
+ recentTools: [],
68
+ recentOutput: [],
69
+ toolCount: 0,
70
+ tokens: 0,
71
+ durationMs: 0,
72
+ },
73
+ };
74
+ }
75
+
76
+ function buildParallelInitialResult(params: SubagentParamsLike): AgentToolResult<Details> {
77
+ const tasks = params.tasks ?? [];
78
+ return {
79
+ content: [{ type: "text", text: tasks.map((task) => `${task.agent}: ${task.task}`).join("\n\n") }],
80
+ details: {
81
+ mode: "parallel",
82
+ ...(params.context ? { context: params.context } : {}),
83
+ results: tasks.map((task, index) => createPlaceholderResult(task.agent, task.task, "running", index)),
84
+ progress: tasks.map((task, index) => ({
85
+ index,
86
+ agent: task.agent,
87
+ status: "running" as const,
88
+ task: task.task,
89
+ recentTools: [],
90
+ recentOutput: [],
91
+ toolCount: 0,
92
+ tokens: 0,
93
+ durationMs: 0,
94
+ })),
95
+ },
96
+ };
97
+ }
98
+
99
+ function isParallelChainStep(step: ChainStepLike): step is ParallelChainStepLike {
100
+ return "parallel" in step && Array.isArray(step.parallel);
101
+ }
102
+
103
+ function chainStepLabel(step: ChainStepLike): string {
104
+ if (isParallelChainStep(step)) {
105
+ return `[${step.parallel.map((entry) => entry.agent).join("+")}]`;
106
+ }
107
+ return step.agent;
108
+ }
109
+
110
+ function flattenChainResults(chain: ChainStepLike[], fallbackTask: string | undefined): SingleResult[] {
111
+ const results: SingleResult[] = [];
112
+ let flatIndex = 0;
113
+ for (const step of chain) {
114
+ if (isParallelChainStep(step)) {
115
+ for (const task of step.parallel) {
116
+ results.push(createPlaceholderResult(task.agent, task.task ?? fallbackTask ?? "", results.length === 0 ? "running" : "pending", flatIndex));
117
+ flatIndex++;
118
+ }
119
+ continue;
120
+ }
121
+ results.push(createPlaceholderResult(step.agent, step.task ?? fallbackTask ?? "", results.length === 0 ? "running" : "pending", flatIndex));
122
+ flatIndex++;
123
+ }
124
+ return results;
125
+ }
126
+
127
+ function buildChainInitialResult(params: SubagentParamsLike): AgentToolResult<Details> {
128
+ const chain = (params.chain ?? []) as ChainStepLike[];
129
+ const results = flattenChainResults(chain, params.task);
130
+ return {
131
+ content: [{
132
+ type: "text",
133
+ text: results.map((result, index) => `Step ${index + 1}: ${result.agent}\n${result.task}`).join("\n\n"),
134
+ }],
135
+ details: {
136
+ mode: "chain",
137
+ ...(params.context ? { context: params.context } : {}),
138
+ results,
139
+ progress: results.map((result, index) => ({
140
+ index,
141
+ agent: result.agent,
142
+ status: index === 0 ? "running" as const : "pending" as const,
143
+ task: result.task,
144
+ recentTools: [],
145
+ recentOutput: [],
146
+ toolCount: 0,
147
+ tokens: 0,
148
+ durationMs: 0,
149
+ })),
150
+ chainAgents: chain.map((step) => chainStepLabel(step)),
151
+ totalSteps: chain.length,
152
+ currentStepIndex: 0,
153
+ },
154
+ };
155
+ }
156
+
157
+ function buildSingleInitialResult(params: SubagentParamsLike): AgentToolResult<Details> {
158
+ const agent = params.agent ?? "subagent";
159
+ const task = params.task ?? "";
160
+ return {
161
+ content: [{ type: "text", text: task }],
162
+ details: {
163
+ mode: "single",
164
+ ...(params.context ? { context: params.context } : {}),
165
+ results: [createPlaceholderResult(agent, task, "running")],
166
+ progress: [{
167
+ agent,
168
+ status: "running",
169
+ task,
170
+ recentTools: [],
171
+ recentOutput: [],
172
+ toolCount: 0,
173
+ tokens: 0,
174
+ durationMs: 0,
175
+ }],
176
+ },
177
+ };
178
+ }
179
+
180
+ export function buildSlashInitialResult(requestId: string, params: SubagentParamsLike): SlashMessageDetails {
181
+ const result = (params.tasks?.length ?? 0) > 0
182
+ ? buildParallelInitialResult(params)
183
+ : (params.chain?.length ?? 0) > 0
184
+ ? buildChainInitialResult(params)
185
+ : buildSingleInitialResult(params);
186
+ liveSnapshots.set(requestId, { result, version: nextVersion() });
187
+ finalSnapshots.delete(requestId);
188
+ return { requestId, result };
189
+ }
190
+
191
+ function cloneResultsWithProgress(
192
+ results: SingleResult[],
193
+ progress: NonNullable<Details["progress"]> | undefined,
194
+ ): SingleResult[] {
195
+ return results.map((result, index) => {
196
+ const nextProgress = progress?.find((entry) => entry.index === index)
197
+ ?? progress?.[index]
198
+ ?? result.progress;
199
+ return nextProgress ? { ...result, progress: nextProgress } : result;
200
+ });
201
+ }
202
+
203
+ export function applySlashUpdate(requestId: string, update: SlashSubagentUpdate): void {
204
+ const snapshot = liveSnapshots.get(requestId);
205
+ if (!snapshot) return;
206
+ const progress = update.progress;
207
+ if (!progress || !snapshot.result.details) return;
208
+ const currentStepIndex = progress.findIndex((entry) => entry.status === "running");
209
+ const nextDetails: Details = {
210
+ ...snapshot.result.details,
211
+ progress,
212
+ results: cloneResultsWithProgress(snapshot.result.details.results, progress),
213
+ ...(snapshot.result.details.mode === "chain" && currentStepIndex >= 0 ? { currentStepIndex } : {}),
214
+ };
215
+ liveSnapshots.set(requestId, {
216
+ result: {
217
+ ...snapshot.result,
218
+ details: nextDetails,
219
+ },
220
+ version: nextVersion(),
221
+ });
222
+ }
223
+
224
+ export function finalizeSlashResult(response: SlashSubagentResponse): SlashMessageDetails {
225
+ const snapshot = {
226
+ result: response.result,
227
+ version: nextVersion(),
228
+ };
229
+ finalSnapshots.set(response.requestId, snapshot);
230
+ liveSnapshots.delete(response.requestId);
231
+ return {
232
+ requestId: response.requestId,
233
+ result: response.result,
234
+ };
235
+ }
236
+
237
+ export function failSlashResult(requestId: string, params: SubagentParamsLike, message: string): SlashMessageDetails {
238
+ const initial = buildSlashInitialResult(requestId, params).result;
239
+ const failedResults = initial.details.results.map((result) => ({
240
+ ...result,
241
+ exitCode: 1,
242
+ error: message,
243
+ progress: result.progress ? { ...result.progress, status: "failed" as const } : result.progress,
244
+ }));
245
+ const result: AgentToolResult<Details> = {
246
+ content: [{ type: "text", text: message }],
247
+ details: {
248
+ ...initial.details,
249
+ results: failedResults,
250
+ progress: failedResults.map((entry) => entry.progress!).filter(Boolean),
251
+ },
252
+ };
253
+ const snapshot = { result, version: nextVersion() };
254
+ finalSnapshots.set(requestId, snapshot);
255
+ liveSnapshots.delete(requestId);
256
+ return { requestId, result };
257
+ }
258
+
259
+ function isSlashMessageDetails(value: unknown): value is SlashMessageDetails {
260
+ if (!value || typeof value !== "object") return false;
261
+ const v = value as { requestId?: string; result?: { content?: unknown; details?: { results?: unknown } } };
262
+ if (typeof v.requestId !== "string" || !v.requestId) return false;
263
+ if (!v.result || !Array.isArray(v.result.content)) return false;
264
+ return !!v.result.details && Array.isArray(v.result.details.results);
265
+ }
266
+
267
+ export function resolveSlashMessageDetails(value: unknown): SlashMessageDetails | undefined {
268
+ return isSlashMessageDetails(value) ? value : undefined;
269
+ }
270
+
271
+ export function getSlashRenderableSnapshot(details: SlashMessageDetails): SlashSnapshot {
272
+ return finalSnapshots.get(details.requestId)
273
+ ?? liveSnapshots.get(details.requestId)
274
+ ?? { result: details.result, version: 0 };
275
+ }
276
+
277
+ export function restoreSlashFinalSnapshots(entries: unknown[]): void {
278
+ liveSnapshots.clear();
279
+ finalSnapshots.clear();
280
+ for (const entry of entries) {
281
+ const e = entry as { type?: string; customType?: string; details?: unknown };
282
+ if (e?.type !== "custom_message" || e.customType !== SLASH_RESULT_TYPE) continue;
283
+ const details = resolveSlashMessageDetails(e.details);
284
+ if (!details) continue;
285
+ finalSnapshots.set(details.requestId, { result: details.result, version: nextVersion() });
286
+ }
287
+ }
288
+
289
+ export function clearSlashSnapshots(): void {
290
+ liveSnapshots.clear();
291
+ finalSnapshots.clear();
292
+ }
@@ -0,0 +1,80 @@
1
+ import type { Theme } from "@earendil-works/pi-coding-agent";
2
+ import { truncateToWidth, visibleWidth } from "@earendil-works/pi-tui";
3
+
4
+ function fuzzyScore(query: string, text: string): number {
5
+ const lq = query.toLowerCase();
6
+ const lt = text.toLowerCase();
7
+ if (lt.includes(lq)) return 100 + (lq.length / lt.length) * 50;
8
+ let score = 0;
9
+ let qi = 0;
10
+ let consecutive = 0;
11
+ for (let i = 0; i < lt.length && qi < lq.length; i++) {
12
+ if (lt[i] === lq[qi]) {
13
+ score += 10 + consecutive;
14
+ consecutive += 5;
15
+ qi++;
16
+ } else {
17
+ consecutive = 0;
18
+ }
19
+ }
20
+ return qi === lq.length ? score : 0;
21
+ }
22
+
23
+ export function fuzzyFilter<T extends { name: string; description: string; model?: string }>(items: T[], query: string): T[] {
24
+ const q = query.trim();
25
+ if (!q) return items;
26
+ return items
27
+ .map((item) => ({ item, score: Math.max(fuzzyScore(q, item.name), fuzzyScore(q, item.description) * 0.8, fuzzyScore(q, item.model ?? "") * 0.6) }))
28
+ .filter((x) => x.score > 0)
29
+ .sort((a, b) => b.score - a.score)
30
+ .map((x) => x.item);
31
+ }
32
+
33
+ export function pad(s: string, len: number): string {
34
+ const vis = visibleWidth(s);
35
+ return s + " ".repeat(Math.max(0, len - vis));
36
+ }
37
+
38
+ export function row(content: string, width: number, theme: Theme): string {
39
+ const innerW = width - 2;
40
+ const singleLine = content.replace(/[\r\n]+/g, " ").replace(/\t/g, " ");
41
+ const clipped = truncateToWidth(singleLine, innerW);
42
+ return theme.fg("border", "│") + pad(clipped, innerW) + theme.fg("border", "│");
43
+ }
44
+
45
+ export function renderHeader(text: string, width: number, theme: Theme): string {
46
+ const innerW = width - 2;
47
+ const padLen = Math.max(0, innerW - visibleWidth(text));
48
+ const padLeft = Math.floor(padLen / 2);
49
+ const padRight = padLen - padLeft;
50
+ return (
51
+ theme.fg("border", "╭" + "─".repeat(padLeft)) +
52
+ theme.fg("accent", text) +
53
+ theme.fg("border", "─".repeat(padRight) + "╮")
54
+ );
55
+ }
56
+
57
+ export function formatPath(filePath: string): string {
58
+ const home = process.env.HOME;
59
+ if (home && filePath.startsWith(home)) return `~${filePath.slice(home.length)}`;
60
+ return filePath;
61
+ }
62
+
63
+ export function formatScrollInfo(above: number, below: number): string {
64
+ let info = "";
65
+ if (above > 0) info += `↑ ${above} more`;
66
+ if (below > 0) info += `${info ? " " : ""}↓ ${below} more`;
67
+ return info;
68
+ }
69
+
70
+ export function renderFooter(text: string, width: number, theme: Theme): string {
71
+ const innerW = width - 2;
72
+ const padLen = Math.max(0, innerW - visibleWidth(text));
73
+ const padLeft = Math.floor(padLen / 2);
74
+ const padRight = padLen - padLeft;
75
+ return (
76
+ theme.fg("border", "╰" + "─".repeat(padLeft)) +
77
+ theme.fg("dim", text) +
78
+ theme.fg("border", "─".repeat(padRight) + "╯")
79
+ );
80
+ }