mirror of
https://github.com/badlogic/pi-mono.git
synced 2026-05-23 21:25:27 +00:00
150 lines
4.3 KiB
TypeScript
150 lines
4.3 KiB
TypeScript
import {
|
|
CustomEditor,
|
|
type ExtensionAPI,
|
|
type ExtensionContext,
|
|
type KeybindingsManager,
|
|
} from "@earendil-works/pi-coding-agent";
|
|
import type { Component, EditorTheme, TUI } from "@earendil-works/pi-tui";
|
|
import { truncateToWidth, visibleWidth } from "@earendil-works/pi-tui";
|
|
|
|
function fitBorder(
|
|
left: string,
|
|
right: string,
|
|
width: number,
|
|
border: (text: string) => string,
|
|
fill: (text: string) => string = border,
|
|
): string {
|
|
if (width <= 0) return "";
|
|
if (width === 1) return border("─");
|
|
|
|
let leftText = left;
|
|
let rightText = right;
|
|
const fixedWidth = 2;
|
|
const minimumGap = 3;
|
|
|
|
while (
|
|
fixedWidth + visibleWidth(leftText) + visibleWidth(rightText) + minimumGap > width &&
|
|
visibleWidth(rightText) > 0
|
|
) {
|
|
rightText = truncateToWidth(rightText, Math.max(0, visibleWidth(rightText) - 1), "");
|
|
}
|
|
while (
|
|
fixedWidth + visibleWidth(leftText) + visibleWidth(rightText) + minimumGap > width &&
|
|
visibleWidth(leftText) > 0
|
|
) {
|
|
leftText = truncateToWidth(leftText, Math.max(0, visibleWidth(leftText) - 1), "");
|
|
}
|
|
|
|
const gapWidth = Math.max(0, width - fixedWidth - visibleWidth(leftText) - visibleWidth(rightText));
|
|
return `${border("─")}${leftText}${fill("─".repeat(gapWidth))}${rightText}${border("─")}`;
|
|
}
|
|
|
|
function formatCwd(cwd: string): string {
|
|
const home = process.env.HOME;
|
|
if (home && cwd.startsWith(home)) {
|
|
return `~${cwd.slice(home.length)}`;
|
|
}
|
|
return cwd;
|
|
}
|
|
|
|
function formatContext(ctx: ExtensionContext): string {
|
|
const usage = ctx.getContextUsage();
|
|
const contextWindow = usage?.contextWindow ?? ctx.model?.contextWindow;
|
|
if (!contextWindow || !usage || usage.percent === null) {
|
|
return "ctx ?";
|
|
}
|
|
return `ctx ${Math.round(usage.percent)}%/${(contextWindow / 1000).toFixed(0)}k`;
|
|
}
|
|
|
|
function formatThinking(level: string): string {
|
|
return level === "off" ? "off" : level;
|
|
}
|
|
|
|
class EmptyFooter implements Component {
|
|
render(): string[] {
|
|
return [];
|
|
}
|
|
|
|
invalidate(): void {}
|
|
}
|
|
|
|
export default function (pi: ExtensionAPI) {
|
|
let isWorking = false;
|
|
let spinnerIndex = 0;
|
|
let spinnerTimer: ReturnType<typeof setInterval> | undefined;
|
|
let activeTui: TUI | undefined;
|
|
const spinnerFrames = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
|
|
|
|
const stopSpinner = () => {
|
|
if (spinnerTimer) {
|
|
clearInterval(spinnerTimer);
|
|
spinnerTimer = undefined;
|
|
}
|
|
};
|
|
|
|
pi.on("agent_start", () => {
|
|
isWorking = true;
|
|
stopSpinner();
|
|
spinnerTimer = setInterval(() => {
|
|
spinnerIndex = (spinnerIndex + 1) % spinnerFrames.length;
|
|
activeTui?.requestRender();
|
|
}, 80);
|
|
activeTui?.requestRender();
|
|
});
|
|
|
|
pi.on("agent_end", () => {
|
|
isWorking = false;
|
|
stopSpinner();
|
|
activeTui?.requestRender();
|
|
});
|
|
|
|
pi.on("session_shutdown", () => {
|
|
stopSpinner();
|
|
activeTui = undefined;
|
|
});
|
|
|
|
pi.on("session_start", (_event, ctx) => {
|
|
ctx.ui.setWorkingVisible(false);
|
|
ctx.ui.setFooter(() => new EmptyFooter());
|
|
|
|
let branch: string | undefined;
|
|
|
|
const refreshBranch = async () => {
|
|
const result = await pi.exec("git", ["branch", "--show-current"], { cwd: ctx.cwd }).catch(() => undefined);
|
|
const stdout = result?.stdout.trim();
|
|
branch = stdout && stdout.length > 0 ? stdout : undefined;
|
|
activeTui?.requestRender();
|
|
};
|
|
void refreshBranch();
|
|
|
|
class BorderStatusEditor extends CustomEditor {
|
|
constructor(tui: TUI, theme: EditorTheme, keybindings: KeybindingsManager) {
|
|
super(tui, theme, keybindings, { paddingX: 0 });
|
|
activeTui = tui;
|
|
}
|
|
|
|
render(width: number): string[] {
|
|
const lines = super.render(width);
|
|
if (lines.length < 2) return lines;
|
|
|
|
const thm = ctx.ui.theme;
|
|
const model = ctx.model ? `${ctx.model.provider}/${ctx.model.id}` : "no model";
|
|
const thinking = pi.getThinkingLevel();
|
|
const topLeft = isWorking ? thm.fg("accent", ` ${spinnerFrames[spinnerIndex]} `) : "";
|
|
const topRight = "";
|
|
const bottomLeft = thm.fg("muted", ` ${model} · ${formatThinking(thinking)} `);
|
|
const bottomRight = thm.fg(
|
|
"muted",
|
|
` ${formatContext(ctx)} · ${formatCwd(ctx.cwd)}${branch ? ` (${branch})` : ""} `,
|
|
);
|
|
const borderColor = (text: string) => this.borderColor(text);
|
|
|
|
lines[0] = fitBorder(topLeft, topRight, width, borderColor);
|
|
lines[lines.length - 1] = fitBorder(bottomLeft, bottomRight, width, borderColor);
|
|
return lines;
|
|
}
|
|
}
|
|
|
|
ctx.ui.setEditorComponent((tui, theme, keybindings) => new BorderStatusEditor(tui, theme, keybindings));
|
|
});
|
|
}
|