-
Notifications
You must be signed in to change notification settings - Fork 36
Expand file tree
/
Copy pathbackground-widget.ts
More file actions
89 lines (82 loc) · 3.11 KB
/
background-widget.ts
File metadata and controls
89 lines (82 loc) · 3.11 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import { truncateToWidth, visibleWidth } from "@mariozechner/pi-tui";
import { formatDuration } from "./types.js";
import type { ShellSessionManager } from "./session-manager.js";
import type { InteractiveShellCoordinator } from "./runtime-coordinator.js";
export function setupBackgroundWidget(
ctx: { ui: { setWidget: Function }; hasUI?: boolean },
sessionManager: ShellSessionManager,
coordinator?: InteractiveShellCoordinator,
): (() => void) | null {
if (!ctx.hasUI) return null;
let durationTimer: ReturnType<typeof setInterval> | null = null;
let tuiRef: { requestRender: () => void } | null = null;
const requestRender = () => tuiRef?.requestRender();
const unsubscribe = sessionManager.onChange(() => {
manageDurationTimer();
requestRender();
});
function manageDurationTimer() {
const sessions = sessionManager.list();
const hasRunning = sessions.some((s) => !s.session.exited);
if (hasRunning && !durationTimer) {
durationTimer = setInterval(requestRender, 10_000);
} else if (!hasRunning && durationTimer) {
clearInterval(durationTimer);
durationTimer = null;
}
}
ctx.ui.setWidget(
"bg-sessions",
(tui: any, theme: any) => {
tuiRef = tui;
return {
render: (width: number) => {
const sessions = sessionManager.list();
if (sessions.length === 0) return [];
const cols = width || tui.terminal?.columns || 120;
const lines: string[] = [];
for (const s of sessions) {
const monitorState = coordinator?.getMonitorSessionState(s.id);
const exited = s.session.exited;
const dot = exited
? theme.fg("dim", "○")
: monitorState
? theme.fg("accent", "◆")
: theme.fg("accent", "●");
const id = theme.fg("dim", s.id);
const cmd = s.command.replace(/\s+/g, " ").trim();
const truncCmd = cmd.length > 60 ? cmd.slice(0, 57) + "..." : cmd;
const reason = s.reason ? theme.fg("dim", ` · ${s.reason}`) : "";
const statusText = monitorState
? `${monitorState.status === "running" ? "monitoring" : "monitor-stopped"}${monitorState.eventCount > 0 ? ` e:${monitorState.eventCount}` : ""}`
: exited
? "exited"
: "running";
const status = exited ? theme.fg("dim", statusText) : monitorState ? theme.fg("accent", statusText) : theme.fg("success", statusText);
const duration = theme.fg("dim", formatDuration(Date.now() - s.startedAt.getTime()));
const strategy = monitorState ? theme.fg("dim", ` · ${monitorState.strategy}`) : "";
const oneLine = ` ${dot} ${id} ${truncCmd}${reason}${strategy} ${status} ${duration}`;
if (visibleWidth(oneLine) <= cols) {
lines.push(oneLine);
} else {
lines.push(truncateToWidth(` ${dot} ${id} ${cmd}`, cols, "…"));
lines.push(truncateToWidth(` ${status} ${duration}${reason}`, cols, "…"));
}
}
return lines;
},
invalidate: () => {},
};
},
{ placement: "belowEditor" },
);
manageDurationTimer();
return () => {
unsubscribe();
if (durationTimer) {
clearInterval(durationTimer);
durationTimer = null;
}
ctx.ui.setWidget("bg-sessions", undefined);
};
}