herm-tui 1.0.0-dev.1
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.
- package/LICENSE +21 -0
- package/README.md +54 -0
- package/package.json +82 -0
- package/scripts/postinstall.ts +29 -0
- package/src/app/gateway.tsx +83 -0
- package/src/app/gatewayEvents.ts +203 -0
- package/src/app/launch.ts +41 -0
- package/src/app/skin.tsx +31 -0
- package/src/app/spawnHistory.ts +75 -0
- package/src/app/tabs.ts +23 -0
- package/src/app/turnReducer.ts +390 -0
- package/src/app/useAppKeys.ts +268 -0
- package/src/app/useAtRefPopover.ts +99 -0
- package/src/app/useInputHistory.ts +66 -0
- package/src/app/useSession.ts +102 -0
- package/src/app/useSlashCommands.ts +70 -0
- package/src/app/useSlashPopover.ts +48 -0
- package/src/app.tsx +917 -0
- package/src/commands/slash.ts +151 -0
- package/src/components/avatar/AnimatedAvatar.tsx +66 -0
- package/src/components/avatar/eikon.ts +144 -0
- package/src/components/avatar/states/error.ts +1155 -0
- package/src/components/avatar/states/idle.ts +1155 -0
- package/src/components/avatar/states/index.ts +30 -0
- package/src/components/avatar/states/listening.ts +1155 -0
- package/src/components/avatar/states/speaking.ts +1155 -0
- package/src/components/avatar/states/thinking.ts +1155 -0
- package/src/components/avatar/states/working.ts +1155 -0
- package/src/components/chat/AtRefPopover.tsx +54 -0
- package/src/components/chat/CodeBlock.tsx +67 -0
- package/src/components/chat/Composer.tsx +347 -0
- package/src/components/chat/DiffBlock.tsx +116 -0
- package/src/components/chat/ErrorBlock.tsx +70 -0
- package/src/components/chat/MediaChip.tsx +114 -0
- package/src/components/chat/MessageItem.tsx +282 -0
- package/src/components/chat/MessageList.tsx +114 -0
- package/src/components/chat/PromptCard.tsx +359 -0
- package/src/components/chat/SlashPopover.tsx +158 -0
- package/src/components/chat/ThoughtCloud.tsx +185 -0
- package/src/components/chat/TypingIndicator.tsx +25 -0
- package/src/components/chat/tool/Subagent.tsx +75 -0
- package/src/components/chat/tool/frame.tsx +69 -0
- package/src/components/chat/tool/index.tsx +65 -0
- package/src/components/chat/tool/preview.ts +57 -0
- package/src/components/sidebar/ContextGauge.tsx +102 -0
- package/src/components/sidebar/Sidebar.tsx +143 -0
- package/src/components/tabs/TabBar.tsx +50 -0
- package/src/components/ui/FileLink.tsx +52 -0
- package/src/config/index.ts +156 -0
- package/src/config/lane.ts +161 -0
- package/src/config/models.ts +95 -0
- package/src/config/rules.ts +80 -0
- package/src/config/schema.ts +308 -0
- package/src/dialogs/alert.tsx +52 -0
- package/src/dialogs/chafa.tsx +72 -0
- package/src/dialogs/confirm.tsx +58 -0
- package/src/dialogs/curator.tsx +153 -0
- package/src/dialogs/eikon-picker.tsx +95 -0
- package/src/dialogs/help.tsx +80 -0
- package/src/dialogs/history.tsx +92 -0
- package/src/dialogs/info.tsx +115 -0
- package/src/dialogs/keys.tsx +170 -0
- package/src/dialogs/logs.tsx +42 -0
- package/src/dialogs/message.tsx +38 -0
- package/src/dialogs/model-picker.tsx +123 -0
- package/src/dialogs/new-profile.tsx +69 -0
- package/src/dialogs/new-task.tsx +103 -0
- package/src/dialogs/profile.tsx +55 -0
- package/src/dialogs/rollback.tsx +190 -0
- package/src/dialogs/spawn-history.tsx +80 -0
- package/src/dialogs/text-prompt.tsx +68 -0
- package/src/dialogs/theme-picker.tsx +50 -0
- package/src/home/index.ts +23 -0
- package/src/home/store.ts +267 -0
- package/src/index.tsx +113 -0
- package/src/keys/catalog.ts +115 -0
- package/src/keys/chord.ts +125 -0
- package/src/keys/conflicts.ts +48 -0
- package/src/keys/context.tsx +112 -0
- package/src/keys/index.ts +5 -0
- package/src/keys/list.ts +94 -0
- package/src/keys/oc-compat.ts +87 -0
- package/src/tabs/Agents.tsx +607 -0
- package/src/tabs/Analytics.tsx +154 -0
- package/src/tabs/Chat.tsx +50 -0
- package/src/tabs/Config.tsx +605 -0
- package/src/tabs/Context.tsx +599 -0
- package/src/tabs/Cron.tsx +294 -0
- package/src/tabs/Env.tsx +227 -0
- package/src/tabs/Kanban.tsx +367 -0
- package/src/tabs/Memory.tsx +294 -0
- package/src/tabs/Sessions.tsx +786 -0
- package/src/tabs/Skills.tsx +507 -0
- package/src/tabs/Toolsets.tsx +266 -0
- package/src/theme/builtin.ts +78 -0
- package/src/theme/context.tsx +106 -0
- package/src/theme/index.ts +4 -0
- package/src/theme/resolve.ts +134 -0
- package/src/theme/syntax.ts +31 -0
- package/src/theme/themes/aura.json +69 -0
- package/src/theme/themes/ayu.json +80 -0
- package/src/theme/themes/carbonfox.json +248 -0
- package/src/theme/themes/catppuccin-frappe.json +233 -0
- package/src/theme/themes/catppuccin-macchiato.json +233 -0
- package/src/theme/themes/catppuccin.json +112 -0
- package/src/theme/themes/cobalt2.json +228 -0
- package/src/theme/themes/cursor.json +249 -0
- package/src/theme/themes/dracula.json +219 -0
- package/src/theme/themes/everforest.json +241 -0
- package/src/theme/themes/flexoki.json +237 -0
- package/src/theme/themes/github.json +233 -0
- package/src/theme/themes/gruvbox.json +242 -0
- package/src/theme/themes/kanagawa.json +77 -0
- package/src/theme/themes/lucent-orng.json +237 -0
- package/src/theme/themes/material.json +235 -0
- package/src/theme/themes/matrix.json +77 -0
- package/src/theme/themes/mercury.json +252 -0
- package/src/theme/themes/monokai.json +221 -0
- package/src/theme/themes/nightowl.json +221 -0
- package/src/theme/themes/nord.json +223 -0
- package/src/theme/themes/one-dark.json +84 -0
- package/src/theme/themes/opencode.json +245 -0
- package/src/theme/themes/orng.json +249 -0
- package/src/theme/themes/osaka-jade.json +93 -0
- package/src/theme/themes/palenight.json +222 -0
- package/src/theme/themes/rosepine.json +234 -0
- package/src/theme/themes/solarized.json +223 -0
- package/src/theme/themes/synthwave84.json +226 -0
- package/src/theme/themes/tokyonight.json +243 -0
- package/src/theme/themes/vercel.json +245 -0
- package/src/theme/themes/vesper.json +218 -0
- package/src/theme/themes/zenburn.json +223 -0
- package/src/theme/types.ts +119 -0
- package/src/types/message.ts +97 -0
- package/src/ui/ChafaImage.tsx +64 -0
- package/src/ui/Splash.tsx +118 -0
- package/src/ui/borders.ts +28 -0
- package/src/ui/command.tsx +104 -0
- package/src/ui/dialog-select.tsx +164 -0
- package/src/ui/dialog.tsx +102 -0
- package/src/ui/fmt.ts +82 -0
- package/src/ui/kv.tsx +28 -0
- package/src/ui/shell.tsx +45 -0
- package/src/ui/spinner.tsx +59 -0
- package/src/ui/splash-art.ts +123 -0
- package/src/ui/table.tsx +117 -0
- package/src/ui/ticker.tsx +90 -0
- package/src/ui/toast.tsx +130 -0
- package/src/utils/categorical.ts +77 -0
- package/src/utils/chafa.ts +173 -0
- package/src/utils/clipboard.ts +67 -0
- package/src/utils/context-segments.ts +317 -0
- package/src/utils/control.ts +495 -0
- package/src/utils/drop.ts +25 -0
- package/src/utils/editor.ts +33 -0
- package/src/utils/fuzzy.ts +45 -0
- package/src/utils/gateway-client.ts +253 -0
- package/src/utils/gateway-types.ts +282 -0
- package/src/utils/git.ts +57 -0
- package/src/utils/hermes-analytics.ts +134 -0
- package/src/utils/hermes-home.ts +821 -0
- package/src/utils/hermes-kanban.ts +154 -0
- package/src/utils/hermes-profiles.ts +217 -0
- package/src/utils/interpolate.ts +31 -0
- package/src/utils/math-unicode.ts +818 -0
- package/src/utils/memory-activity.ts +140 -0
- package/src/utils/open-file.ts +13 -0
- package/src/utils/paths.ts +52 -0
- package/src/utils/perf.ts +235 -0
- package/src/utils/preferences.ts +150 -0
- package/src/utils/sessions-db.ts +396 -0
- package/src/utils/subagent-tree.ts +146 -0
- package/src/utils/terminal-reset.ts +129 -0
- package/src/utils/tips.ts +67 -0
- package/src/utils/tokens.ts +87 -0
|
@@ -0,0 +1,294 @@
|
|
|
1
|
+
import { useState, useEffect, useCallback, useRef, memo } from "react";
|
|
2
|
+
import { useTerminalDimensions } from "@opentui/react";
|
|
3
|
+
import { useGateway } from "../app/gateway";
|
|
4
|
+
import { useListKeys, useFollow } from "../keys";
|
|
5
|
+
import { useTheme } from "../theme";
|
|
6
|
+
import { useDialog } from "../ui/dialog";
|
|
7
|
+
import { useToast } from "../ui/toast";
|
|
8
|
+
import { openConfirm } from "../dialogs/confirm";
|
|
9
|
+
import { TabShell } from "../ui/shell";
|
|
10
|
+
import { KVBlock } from "../ui/kv";
|
|
11
|
+
import { Col, Hdr, VBAR } from "../ui/table";
|
|
12
|
+
import { openTextPrompt } from "../dialogs/text-prompt";
|
|
13
|
+
import { ago, until } from "../ui/fmt";
|
|
14
|
+
import { readCronOutput, type CronOutput } from "../utils/hermes-home";
|
|
15
|
+
|
|
16
|
+
// ─── Types ───────────────────────────────────────────────────────────
|
|
17
|
+
|
|
18
|
+
type CronJob = {
|
|
19
|
+
id: string
|
|
20
|
+
name: string
|
|
21
|
+
prompt: string
|
|
22
|
+
schedule: string
|
|
23
|
+
enabled: boolean
|
|
24
|
+
state: string
|
|
25
|
+
deliver: string
|
|
26
|
+
repeat?: string
|
|
27
|
+
last_run?: string
|
|
28
|
+
next_run?: string
|
|
29
|
+
last_status?: "ok" | "error"
|
|
30
|
+
last_error?: string
|
|
31
|
+
paused_reason?: string
|
|
32
|
+
model?: string
|
|
33
|
+
skills?: string[]
|
|
34
|
+
workdir?: string
|
|
35
|
+
script?: string
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
type RawJob = {
|
|
39
|
+
job_id?: string
|
|
40
|
+
id?: string
|
|
41
|
+
name?: string
|
|
42
|
+
prompt_preview?: string
|
|
43
|
+
prompt?: string
|
|
44
|
+
schedule?: string
|
|
45
|
+
enabled?: boolean
|
|
46
|
+
state?: string
|
|
47
|
+
deliver?: string
|
|
48
|
+
repeat?: string
|
|
49
|
+
last_run_at?: string
|
|
50
|
+
next_run_at?: string
|
|
51
|
+
last_status?: string
|
|
52
|
+
last_delivery_error?: string
|
|
53
|
+
paused_reason?: string
|
|
54
|
+
model?: string
|
|
55
|
+
skills?: string[]
|
|
56
|
+
workdir?: string
|
|
57
|
+
script?: string
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const normalize = (j: RawJob): CronJob => ({
|
|
61
|
+
id: j.job_id ?? j.id ?? "",
|
|
62
|
+
name: j.name ?? "",
|
|
63
|
+
prompt: j.prompt ?? j.prompt_preview ?? "",
|
|
64
|
+
schedule: j.schedule ?? "",
|
|
65
|
+
enabled: j.enabled ?? true,
|
|
66
|
+
state: j.state ?? "scheduled",
|
|
67
|
+
deliver: j.deliver ?? "local",
|
|
68
|
+
repeat: j.repeat,
|
|
69
|
+
last_run: j.last_run_at,
|
|
70
|
+
next_run: j.next_run_at,
|
|
71
|
+
last_status: j.last_status === "ok" || j.last_status === "error" ? j.last_status : undefined,
|
|
72
|
+
last_error: j.last_delivery_error,
|
|
73
|
+
paused_reason: j.paused_reason,
|
|
74
|
+
model: j.model,
|
|
75
|
+
skills: j.skills,
|
|
76
|
+
workdir: j.workdir,
|
|
77
|
+
script: j.script,
|
|
78
|
+
})
|
|
79
|
+
|
|
80
|
+
// gateway returns ISO timestamps; shared `ago`/`until` want unix seconds
|
|
81
|
+
const sec = (iso?: string) => iso ? new Date(iso).getTime() / 1000 : null
|
|
82
|
+
const last = (iso?: string) => { const t = sec(iso); return t ? ago(t) : "—" }
|
|
83
|
+
const next = (iso?: string) => { const t = sec(iso); return t ? until(t) : "—" }
|
|
84
|
+
|
|
85
|
+
// ─── Job Row ─────────────────────────────────────────────────────────
|
|
86
|
+
|
|
87
|
+
const JobRow = memo((props: {
|
|
88
|
+
id: string;
|
|
89
|
+
job: CronJob;
|
|
90
|
+
selected: boolean;
|
|
91
|
+
onSelect: () => void;
|
|
92
|
+
onHover: () => void;
|
|
93
|
+
}) => {
|
|
94
|
+
const theme = useTheme().theme;
|
|
95
|
+
const j = props.job;
|
|
96
|
+
const bg = props.selected ? theme.backgroundElement : undefined;
|
|
97
|
+
// ●/○ encodes enabled; color encodes last-run outcome.
|
|
98
|
+
const glyph = j.enabled ? "●" : "○";
|
|
99
|
+
const glyphColor = !j.enabled ? theme.textMuted
|
|
100
|
+
: j.last_status === "error" ? theme.error
|
|
101
|
+
: j.last_status === "ok" ? theme.success
|
|
102
|
+
: theme.textMuted;
|
|
103
|
+
|
|
104
|
+
return (
|
|
105
|
+
<box id={props.id} flexDirection="row" height={1} backgroundColor={bg}
|
|
106
|
+
onMouseDown={props.onSelect} onMouseMove={props.onHover}>
|
|
107
|
+
<Col w={2} fg={props.selected ? theme.primary : theme.text}>{props.selected ? "▸ " : " "}</Col>
|
|
108
|
+
<Col w={2} fg={glyphColor}>{`${glyph} `}</Col>
|
|
109
|
+
<Col grow fg={props.selected ? theme.accent : theme.text}>{j.name || j.id}</Col>
|
|
110
|
+
<Col w={18} fg={theme.textMuted}>{j.schedule || "—"}</Col>
|
|
111
|
+
<Col w={16} fg={theme.textMuted}>{`last: ${last(j.last_run)}`}</Col>
|
|
112
|
+
<Col w={16} fg={j.enabled ? theme.text : theme.textMuted}>
|
|
113
|
+
{`next: ${j.enabled ? next(j.next_run) : "paused"}`}
|
|
114
|
+
</Col>
|
|
115
|
+
</box>
|
|
116
|
+
);
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
// ─── Detail Panel ────────────────────────────────────────────────────
|
|
120
|
+
|
|
121
|
+
const DetailPanel = memo((props: { job: CronJob; reloadKey: number }) => {
|
|
122
|
+
const theme = useTheme().theme;
|
|
123
|
+
const j = props.job;
|
|
124
|
+
const [output, setOutput] = useState<CronOutput | null>(null);
|
|
125
|
+
|
|
126
|
+
useEffect(() => {
|
|
127
|
+
let live = true;
|
|
128
|
+
readCronOutput(j.id, 30).then(o => { if (live) setOutput(o) });
|
|
129
|
+
return () => { live = false };
|
|
130
|
+
}, [j.id, props.reloadKey]);
|
|
131
|
+
|
|
132
|
+
return (
|
|
133
|
+
<TabShell title="Job Detail" hint="" grow={2}>
|
|
134
|
+
<scrollbox scrollY flexGrow={1}>
|
|
135
|
+
<box flexDirection="column" width="100%">
|
|
136
|
+
<box minHeight={1}>
|
|
137
|
+
<text wrapMode="word"><span fg={theme.accent}><strong>{j.name || j.id}</strong></span></text>
|
|
138
|
+
</box>
|
|
139
|
+
<box height={1} />
|
|
140
|
+
<KVBlock rows={[
|
|
141
|
+
["ID", j.id],
|
|
142
|
+
["State", j.enabled ? "active" : "paused", j.enabled ? theme.success : theme.warning],
|
|
143
|
+
["Schedule", j.schedule || "—"],
|
|
144
|
+
["Repeat", j.repeat],
|
|
145
|
+
["Deliver", j.deliver ?? "local"],
|
|
146
|
+
["Last Run", j.last_run ? `${last(j.last_run)} · ${j.last_status ?? "?"}` : "never",
|
|
147
|
+
j.last_status === "error" ? theme.error : undefined],
|
|
148
|
+
["Next Run", j.enabled ? next(j.next_run) : "paused"],
|
|
149
|
+
["Model", j.model],
|
|
150
|
+
["Skills", j.skills?.length ? j.skills.join(", ") : undefined],
|
|
151
|
+
["Workdir", j.workdir],
|
|
152
|
+
["Script", j.script],
|
|
153
|
+
["Paused", j.paused_reason],
|
|
154
|
+
["Error", j.last_error, theme.error],
|
|
155
|
+
]} />
|
|
156
|
+
<box height={1} />
|
|
157
|
+
<box height={1}><text fg={theme.textMuted}>Prompt</text></box>
|
|
158
|
+
<text wrapMode="word"><span fg={theme.text}>{j.prompt}</span></text>
|
|
159
|
+
<box height={1} />
|
|
160
|
+
<box height={1}>
|
|
161
|
+
<text fg={theme.textMuted}>Last Output{output ? ` · ${ago(output.at.getTime() / 1000)}` : ""}</text>
|
|
162
|
+
</box>
|
|
163
|
+
{output
|
|
164
|
+
? <text wrapMode="word"><span fg={theme.text}>{output.text}</span></text>
|
|
165
|
+
: <text fg={theme.textMuted}>(none yet)</text>}
|
|
166
|
+
</box>
|
|
167
|
+
</scrollbox>
|
|
168
|
+
</TabShell>
|
|
169
|
+
);
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
// ─── Main Component ──────────────────────────────────────────────────
|
|
173
|
+
|
|
174
|
+
export const Cron = memo((props: { focused?: boolean }) => {
|
|
175
|
+
const theme = useTheme().theme;
|
|
176
|
+
const gw = useGateway();
|
|
177
|
+
const dialog = useDialog();
|
|
178
|
+
const toast = useToast();
|
|
179
|
+
const dims = useTerminalDimensions();
|
|
180
|
+
const [jobs, setJobs] = useState<CronJob[]>([]);
|
|
181
|
+
const [sel, setSel] = useState(0);
|
|
182
|
+
const [err, setErr] = useState<string | null>(null);
|
|
183
|
+
const [reloadKey, setReloadKey] = useState(0);
|
|
184
|
+
|
|
185
|
+
const live = useRef({ jobs, sel });
|
|
186
|
+
live.current = { jobs, sel };
|
|
187
|
+
|
|
188
|
+
const load = useCallback(() => {
|
|
189
|
+
gw.request<{ jobs?: RawJob[] }>("cron.manage", { action: "list" })
|
|
190
|
+
.then(res => {
|
|
191
|
+
setJobs((res.jobs ?? []).map(normalize));
|
|
192
|
+
setErr(null);
|
|
193
|
+
setReloadKey(k => k + 1);
|
|
194
|
+
})
|
|
195
|
+
.catch(e => setErr(e instanceof Error ? e.message : String(e)));
|
|
196
|
+
}, [gw]);
|
|
197
|
+
|
|
198
|
+
useEffect(() => { load(); }, [load]);
|
|
199
|
+
|
|
200
|
+
// ── Actions (stable via live ref) ─────────────────────────────────
|
|
201
|
+
|
|
202
|
+
const create = useCallback(async () => {
|
|
203
|
+
const schedule = await openTextPrompt(dialog, {
|
|
204
|
+
title: "New Cron Job", label: "Schedule (cron expr or 'every 30m')",
|
|
205
|
+
});
|
|
206
|
+
if (schedule === null) return;
|
|
207
|
+
const prompt = await openTextPrompt(dialog, {
|
|
208
|
+
title: "New Cron Job", label: "Prompt",
|
|
209
|
+
});
|
|
210
|
+
if (prompt === null) return;
|
|
211
|
+
// name left blank — server derives one from the prompt text.
|
|
212
|
+
gw.request("cron.manage", { action: "add", name: "", schedule, prompt })
|
|
213
|
+
.then(() => { toast.show({ variant: "success", message: "Job created" }); load(); })
|
|
214
|
+
.catch((e: Error) => toast.show({ variant: "error", message: e.message }));
|
|
215
|
+
}, [gw, dialog, toast, load]);
|
|
216
|
+
|
|
217
|
+
const toggle = useCallback(() => {
|
|
218
|
+
const j = live.current.jobs[live.current.sel];
|
|
219
|
+
if (!j) return;
|
|
220
|
+
const action = j.enabled ? "pause" : "resume";
|
|
221
|
+
gw.request("cron.manage", { action, name: j.id })
|
|
222
|
+
.then(() => { toast.show({ variant: "success", message: j.enabled ? "Paused" : "Resumed" }); load(); })
|
|
223
|
+
.catch((e: Error) => toast.show({ variant: "error", message: e.message }));
|
|
224
|
+
}, [gw, toast, load]);
|
|
225
|
+
|
|
226
|
+
const remove = useCallback(async () => {
|
|
227
|
+
const j = live.current.jobs[live.current.sel];
|
|
228
|
+
if (!j) return;
|
|
229
|
+
const ok = await openConfirm(dialog, {
|
|
230
|
+
title: "Delete Job?",
|
|
231
|
+
body: `Delete "${j.name || j.id}"? This cannot be undone.`,
|
|
232
|
+
yes: "delete", danger: true,
|
|
233
|
+
});
|
|
234
|
+
if (!ok) return;
|
|
235
|
+
gw.request("cron.manage", { action: "remove", name: j.id })
|
|
236
|
+
.then(() => {
|
|
237
|
+
toast.show({ variant: "success", message: "Deleted" });
|
|
238
|
+
setSel(s => Math.max(0, Math.min(s, live.current.jobs.length - 2)));
|
|
239
|
+
load();
|
|
240
|
+
})
|
|
241
|
+
.catch((e: Error) => toast.show({ variant: "error", message: e.message }));
|
|
242
|
+
}, [gw, dialog, toast, load]);
|
|
243
|
+
|
|
244
|
+
const follow = useFollow("cron");
|
|
245
|
+
const keys = useListKeys({
|
|
246
|
+
active: !!props.focused && dialog.stack.length === 0,
|
|
247
|
+
count: jobs.length, setSel, ...follow.opts,
|
|
248
|
+
onToggle: toggle,
|
|
249
|
+
onDelete: remove,
|
|
250
|
+
onNew: create,
|
|
251
|
+
onRefresh: () => { load(); toast.show({ variant: "info", message: "Reloaded", duration: 1000 }) },
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
const job = jobs[sel] ?? null;
|
|
255
|
+
const showDetail = dims.width >= 120 && job !== null;
|
|
256
|
+
|
|
257
|
+
return (
|
|
258
|
+
<box flexDirection="row" flexGrow={1}>
|
|
259
|
+
<TabShell title={`Cron Jobs (${jobs.length})`} error={err} grow={3}
|
|
260
|
+
hint={`↑↓ nav ${keys.print("list.new")} new ${keys.print("list.toggle")} pause/resume ${keys.print("list.delete")} delete ${keys.print("list.refresh")} refresh`}>
|
|
261
|
+
{jobs.length === 0 ? (
|
|
262
|
+
<box key="empty" flexGrow={1}>
|
|
263
|
+
<text fg={theme.textMuted}>No cron jobs. Press n to create one.</text>
|
|
264
|
+
</box>
|
|
265
|
+
) : (
|
|
266
|
+
<box key="table" flexDirection="column" flexGrow={1} minWidth={0}>
|
|
267
|
+
<Hdr>
|
|
268
|
+
<Col w={4} fg={theme.textMuted}>{""}</Col>
|
|
269
|
+
<Col grow fg={theme.textMuted} bold>Name</Col>
|
|
270
|
+
<Col w={18} fg={theme.textMuted} bold>Schedule</Col>
|
|
271
|
+
<Col w={16} fg={theme.textMuted} bold>Last</Col>
|
|
272
|
+
<Col w={16} fg={theme.textMuted} bold>Next</Col>
|
|
273
|
+
</Hdr>
|
|
274
|
+
<box height={1} />
|
|
275
|
+
<scrollbox ref={follow.ref} scrollY flexGrow={1} verticalScrollbarOptions={VBAR}>
|
|
276
|
+
{jobs.map((j, i) => (
|
|
277
|
+
<JobRow
|
|
278
|
+
key={j.id}
|
|
279
|
+
id={follow.id(i)}
|
|
280
|
+
job={j}
|
|
281
|
+
selected={i === sel}
|
|
282
|
+
onSelect={() => setSel(i)}
|
|
283
|
+
onHover={() => setSel(i)}
|
|
284
|
+
/>
|
|
285
|
+
))}
|
|
286
|
+
</scrollbox>
|
|
287
|
+
</box>
|
|
288
|
+
)}
|
|
289
|
+
</TabShell>
|
|
290
|
+
|
|
291
|
+
{showDetail ? <DetailPanel job={job} reloadKey={reloadKey} /> : null}
|
|
292
|
+
</box>
|
|
293
|
+
);
|
|
294
|
+
});
|
package/src/tabs/Env.tsx
ADDED
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
import { useState, useCallback, memo } from "react"
|
|
2
|
+
import { useKeyboard } from "@opentui/react"
|
|
3
|
+
import { useKeys, handleListKey, useFollow } from "../keys"
|
|
4
|
+
import { writeEnvVar, removeEnvVar, ENV_CATALOG } from "../utils/hermes-home"
|
|
5
|
+
import { useHome, home } from "../home"
|
|
6
|
+
import { useTheme } from "../theme"
|
|
7
|
+
import { useDialog } from "../ui/dialog"
|
|
8
|
+
import { useToast } from "../ui/toast"
|
|
9
|
+
import { TabShell } from "../ui/shell"
|
|
10
|
+
import { Col, Hdr, VBAR } from "../ui/table"
|
|
11
|
+
import { openTextPrompt } from "../dialogs/text-prompt"
|
|
12
|
+
import { openConfirm } from "../dialogs/confirm"
|
|
13
|
+
|
|
14
|
+
// ─── Types ────────────────────────────────────────────────────────
|
|
15
|
+
|
|
16
|
+
type Row =
|
|
17
|
+
| { type: "header"; category: string; collapsed: boolean }
|
|
18
|
+
| { type: "var"; key: string; value: string | undefined }
|
|
19
|
+
|
|
20
|
+
const mask = (val: string) => "•".repeat(Math.min(val.length, 12))
|
|
21
|
+
|
|
22
|
+
// ─── Var Row ──────────────────────────────────────────────────────
|
|
23
|
+
|
|
24
|
+
const VarRow = memo((props: {
|
|
25
|
+
id: string
|
|
26
|
+
name: string
|
|
27
|
+
value: string | undefined
|
|
28
|
+
shown: boolean
|
|
29
|
+
selected: boolean
|
|
30
|
+
onHover: () => void
|
|
31
|
+
onClick: () => void
|
|
32
|
+
}) => {
|
|
33
|
+
const theme = useTheme().theme
|
|
34
|
+
const set = props.value !== undefined
|
|
35
|
+
const bg = props.selected ? theme.backgroundElement : undefined
|
|
36
|
+
return (
|
|
37
|
+
<box id={props.id} flexDirection="row" height={1} backgroundColor={bg}
|
|
38
|
+
onMouseDown={props.onClick} onMouseMove={props.onHover}>
|
|
39
|
+
<Col w={2} fg={props.selected ? theme.primary : theme.text}>{props.selected ? "▸ " : " "}</Col>
|
|
40
|
+
<Col w={28} fg={props.selected ? theme.accent : theme.text}>{props.name}</Col>
|
|
41
|
+
<Col w={8} fg={set ? theme.success : theme.textMuted}>{set ? " SET " : "UNSET"}</Col>
|
|
42
|
+
<Col grow min={4} fg={props.shown ? theme.text : theme.textMuted}>
|
|
43
|
+
{set ? (props.shown ? props.value! : mask(props.value!)) : "—"}
|
|
44
|
+
</Col>
|
|
45
|
+
</box>
|
|
46
|
+
)
|
|
47
|
+
})
|
|
48
|
+
|
|
49
|
+
// ─── Main Component ───────────────────────────────────────────────
|
|
50
|
+
|
|
51
|
+
export const Env = memo((props: { focused?: boolean }) => {
|
|
52
|
+
const theme = useTheme().theme
|
|
53
|
+
const dialog = useDialog()
|
|
54
|
+
const toast = useToast()
|
|
55
|
+
|
|
56
|
+
const vars = useHome("env") ?? {}
|
|
57
|
+
const [sel, setSel] = useState(0)
|
|
58
|
+
const [reveal, setReveal] = useState<Set<string>>(new Set())
|
|
59
|
+
const [collapsed, setCollapsed] = useState<Record<string, boolean>>({})
|
|
60
|
+
const [searching, setSearching] = useState(false)
|
|
61
|
+
const [query, setQuery] = useState("")
|
|
62
|
+
|
|
63
|
+
// Catalog keys plus any extras present in .env that aren't catalogued.
|
|
64
|
+
const known = new Set(ENV_CATALOG.flatMap(g => g.keys))
|
|
65
|
+
const extra = Object.keys(vars).filter(k => !known.has(k)).sort()
|
|
66
|
+
const groups = extra.length > 0
|
|
67
|
+
? [...ENV_CATALOG, { category: "Other", keys: extra }]
|
|
68
|
+
: ENV_CATALOG
|
|
69
|
+
|
|
70
|
+
const rows: Row[] = groups.flatMap((g) => {
|
|
71
|
+
const keys = searching && query.trim()
|
|
72
|
+
? g.keys.filter(k => k.toLowerCase().includes(query.toLowerCase()))
|
|
73
|
+
: g.keys
|
|
74
|
+
if (keys.length === 0) return []
|
|
75
|
+
const hide = collapsed[g.category] ?? false
|
|
76
|
+
const header: Row = { type: "header", category: g.category, collapsed: hide }
|
|
77
|
+
if (hide) return [header]
|
|
78
|
+
return [header, ...keys.map((key): Row => ({ type: "var", key, value: vars[key] }))]
|
|
79
|
+
})
|
|
80
|
+
|
|
81
|
+
const count = rows.length
|
|
82
|
+
const cur = rows[sel]
|
|
83
|
+
const setKeys = rows.flatMap(r => r.type === "var" && r.value !== undefined ? [r.key] : [])
|
|
84
|
+
const follow = useFollow("env")
|
|
85
|
+
|
|
86
|
+
const edit = useCallback(async (key: string, initial: string) => {
|
|
87
|
+
const val = await openTextPrompt(dialog, { title: `Edit ${key}`, label: "Value", initial })
|
|
88
|
+
if (val == null) return
|
|
89
|
+
await writeEnvVar(key, val)
|
|
90
|
+
home.invalidate("env")
|
|
91
|
+
toast.show({ variant: "success", message: `${key} saved` })
|
|
92
|
+
}, [dialog, toast])
|
|
93
|
+
|
|
94
|
+
const add = useCallback(async () => {
|
|
95
|
+
const key = await openTextPrompt(dialog, { title: "New Variable", label: "Name (e.g. FOO_API_KEY)" })
|
|
96
|
+
if (!key) return
|
|
97
|
+
const val = await openTextPrompt(dialog, { title: `Set ${key}`, label: "Value" })
|
|
98
|
+
if (val == null) return
|
|
99
|
+
await writeEnvVar(key, val)
|
|
100
|
+
home.invalidate("env")
|
|
101
|
+
toast.show({ variant: "success", message: `${key} added` })
|
|
102
|
+
}, [dialog, toast])
|
|
103
|
+
|
|
104
|
+
const del = useCallback(async (key: string) => {
|
|
105
|
+
const ok = await openConfirm(dialog, {
|
|
106
|
+
title: "Delete Variable",
|
|
107
|
+
body: `Remove ${key} from .env?`,
|
|
108
|
+
yes: "delete", danger: true,
|
|
109
|
+
})
|
|
110
|
+
if (!ok) return
|
|
111
|
+
await removeEnvVar(key)
|
|
112
|
+
home.invalidate("env")
|
|
113
|
+
toast.show({ variant: "success", message: `${key} removed` })
|
|
114
|
+
}, [dialog, toast])
|
|
115
|
+
|
|
116
|
+
const revealAll = useCallback(() =>
|
|
117
|
+
setReveal(s => s.size === setKeys.length && setKeys.length > 0
|
|
118
|
+
? new Set()
|
|
119
|
+
: new Set(setKeys)), [setKeys])
|
|
120
|
+
|
|
121
|
+
const activateAt = useCallback((i: number) => {
|
|
122
|
+
const r = rows[i]
|
|
123
|
+
if (r?.type === "header")
|
|
124
|
+
return setCollapsed(p => ({ ...p, [r.category]: !p[r.category] }))
|
|
125
|
+
if (r?.type === "var") {
|
|
126
|
+
if (r.value !== undefined && !reveal.has(r.key))
|
|
127
|
+
return setReveal(s => new Set(s).add(r.key))
|
|
128
|
+
return void edit(r.key, r.value ?? "")
|
|
129
|
+
}
|
|
130
|
+
}, [rows, reveal, edit])
|
|
131
|
+
const activate = useCallback(() => activateAt(sel), [activateAt, sel])
|
|
132
|
+
|
|
133
|
+
const rowClick = useCallback((i: number) => { setSel(i); activateAt(i) }, [activateAt])
|
|
134
|
+
|
|
135
|
+
const keys = useKeys()
|
|
136
|
+
useKeyboard((key) => {
|
|
137
|
+
if (!props.focused || dialog.stack.length > 0) return
|
|
138
|
+
|
|
139
|
+
if (searching) {
|
|
140
|
+
if (key.name === "escape") { setSearching(false); setQuery(""); setSel(0); return }
|
|
141
|
+
if (key.name === "backspace") { setQuery(q => q.slice(0, -1)); setSel(0); return }
|
|
142
|
+
if (key.name === "up") return setSel(p => Math.max(0, p - 1))
|
|
143
|
+
if (key.name === "down") return setSel(p => Math.min(count - 1, p + 1))
|
|
144
|
+
if (key.name === "return") { setSearching(false); return activate() }
|
|
145
|
+
if (key.raw && key.raw.length === 1 && key.raw >= " ") {
|
|
146
|
+
setQuery(q => q + key.raw); setSel(0); return
|
|
147
|
+
}
|
|
148
|
+
return
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
handleListKey(keys, key, {
|
|
152
|
+
count, setSel, ...follow.opts,
|
|
153
|
+
onActivate: activate,
|
|
154
|
+
onToggle: revealAll,
|
|
155
|
+
onNew: add,
|
|
156
|
+
onDelete: () => { if (cur?.type === "var" && cur.value !== undefined) del(cur.key) },
|
|
157
|
+
onSearch: () => { setSearching(true); setQuery(""); setSel(0) },
|
|
158
|
+
onRefresh: () => { home.invalidate("env"); toast.show({ variant: "info", message: "Reloaded", duration: 1000 }) },
|
|
159
|
+
})
|
|
160
|
+
})
|
|
161
|
+
|
|
162
|
+
return (
|
|
163
|
+
<TabShell
|
|
164
|
+
title={searching ? "Env (searching)" : "Env / API Keys"}
|
|
165
|
+
hint={searching
|
|
166
|
+
? "↑↓ move Enter reveal/edit Esc cancel"
|
|
167
|
+
: `↑↓ move ${keys.print("list.activate")} reveal/edit ${keys.print("list.toggle")} show-all ${keys.print("list.new")} new ${keys.print("list.delete")} delete ${keys.print("list.search")} search ${keys.print("list.refresh")} reload`}
|
|
168
|
+
>
|
|
169
|
+
{searching ? (
|
|
170
|
+
<box height={1}>
|
|
171
|
+
<text>
|
|
172
|
+
<span fg={theme.accent}>/ </span>
|
|
173
|
+
<span fg={theme.text}>{query}</span>
|
|
174
|
+
<span fg={theme.accent}>█</span>
|
|
175
|
+
</text>
|
|
176
|
+
</box>
|
|
177
|
+
) : null}
|
|
178
|
+
|
|
179
|
+
<Hdr>
|
|
180
|
+
<Col w={2} fg={theme.textMuted}>{""}</Col>
|
|
181
|
+
<Col w={28} fg={theme.textMuted} bold>Name</Col>
|
|
182
|
+
<Col w={8} fg={theme.textMuted} bold>Status</Col>
|
|
183
|
+
<Col grow min={4} fg={theme.textMuted} bold>Value</Col>
|
|
184
|
+
</Hdr>
|
|
185
|
+
<box height={1} />
|
|
186
|
+
|
|
187
|
+
{count === 0 ? (
|
|
188
|
+
<box key="empty" flexGrow={1} padding={2}>
|
|
189
|
+
<text fg={theme.textMuted}>
|
|
190
|
+
{searching ? "No matching variables" : "No variables configured"}
|
|
191
|
+
</text>
|
|
192
|
+
</box>
|
|
193
|
+
) : (
|
|
194
|
+
<scrollbox ref={follow.ref} key="list" scrollY flexGrow={1}
|
|
195
|
+
verticalScrollbarOptions={VBAR}>
|
|
196
|
+
<box flexDirection="column" width="100%">
|
|
197
|
+
{rows.map((row, i) => row.type === "header" ? (
|
|
198
|
+
<box
|
|
199
|
+
id={follow.id(i)}
|
|
200
|
+
key={`h-${row.category}`}
|
|
201
|
+
marginTop={i > 0 ? 1 : 0}
|
|
202
|
+
backgroundColor={i === sel ? theme.backgroundElement : undefined}
|
|
203
|
+
onMouseMove={() => setSel(i)}
|
|
204
|
+
onMouseDown={() => rowClick(i)}
|
|
205
|
+
>
|
|
206
|
+
<text fg={theme.info}>
|
|
207
|
+
<strong>{`${row.collapsed ? "▸" : "▾"} ${row.category}`}</strong>
|
|
208
|
+
</text>
|
|
209
|
+
</box>
|
|
210
|
+
) : (
|
|
211
|
+
<VarRow
|
|
212
|
+
id={follow.id(i)}
|
|
213
|
+
key={row.key}
|
|
214
|
+
name={row.key}
|
|
215
|
+
value={row.value}
|
|
216
|
+
shown={reveal.has(row.key)}
|
|
217
|
+
selected={i === sel}
|
|
218
|
+
onHover={() => setSel(i)}
|
|
219
|
+
onClick={() => rowClick(i)}
|
|
220
|
+
/>
|
|
221
|
+
))}
|
|
222
|
+
</box>
|
|
223
|
+
</scrollbox>
|
|
224
|
+
)}
|
|
225
|
+
</TabShell>
|
|
226
|
+
)
|
|
227
|
+
})
|