herm-tui 1.0.0-dev.1 → 1.0.0-dev.3
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/db.worker.js +81 -0
- package/highlights-eq9cgrbb.scm +604 -0
- package/highlights-ghv9g403.scm +205 -0
- package/highlights-hk7bwhj4.scm +284 -0
- package/highlights-r812a2qc.scm +150 -0
- package/highlights-x6tmsnaa.scm +115 -0
- package/index.js +10374 -0
- package/injections-73j83es3.scm +27 -0
- package/package.json +14 -64
- package/parser.worker.js +8 -0
- package/tree-sitter-3jzf13jk.wasm +0 -0
- package/tree-sitter-javascript-nd0q4pe9.wasm +0 -0
- package/tree-sitter-markdown-411r6y9b.wasm +0 -0
- package/tree-sitter-markdown_inline-j5349f42.wasm +0 -0
- package/tree-sitter-typescript-zxjzwt75.wasm +0 -0
- package/tree-sitter-zig-e78zbjpm.wasm +0 -0
- package/scripts/postinstall.ts +0 -29
- package/src/app/gateway.tsx +0 -83
- package/src/app/gatewayEvents.ts +0 -203
- package/src/app/launch.ts +0 -41
- package/src/app/skin.tsx +0 -31
- package/src/app/spawnHistory.ts +0 -75
- package/src/app/tabs.ts +0 -23
- package/src/app/turnReducer.ts +0 -390
- package/src/app/useAppKeys.ts +0 -268
- package/src/app/useAtRefPopover.ts +0 -99
- package/src/app/useInputHistory.ts +0 -66
- package/src/app/useSession.ts +0 -102
- package/src/app/useSlashCommands.ts +0 -70
- package/src/app/useSlashPopover.ts +0 -48
- package/src/app.tsx +0 -917
- package/src/commands/slash.ts +0 -151
- package/src/components/avatar/AnimatedAvatar.tsx +0 -66
- package/src/components/avatar/eikon.ts +0 -144
- package/src/components/avatar/states/error.ts +0 -1155
- package/src/components/avatar/states/idle.ts +0 -1155
- package/src/components/avatar/states/index.ts +0 -30
- package/src/components/avatar/states/listening.ts +0 -1155
- package/src/components/avatar/states/speaking.ts +0 -1155
- package/src/components/avatar/states/thinking.ts +0 -1155
- package/src/components/avatar/states/working.ts +0 -1155
- package/src/components/chat/AtRefPopover.tsx +0 -54
- package/src/components/chat/CodeBlock.tsx +0 -67
- package/src/components/chat/Composer.tsx +0 -347
- package/src/components/chat/DiffBlock.tsx +0 -116
- package/src/components/chat/ErrorBlock.tsx +0 -70
- package/src/components/chat/MediaChip.tsx +0 -114
- package/src/components/chat/MessageItem.tsx +0 -282
- package/src/components/chat/MessageList.tsx +0 -114
- package/src/components/chat/PromptCard.tsx +0 -359
- package/src/components/chat/SlashPopover.tsx +0 -158
- package/src/components/chat/ThoughtCloud.tsx +0 -185
- package/src/components/chat/TypingIndicator.tsx +0 -25
- package/src/components/chat/tool/Subagent.tsx +0 -75
- package/src/components/chat/tool/frame.tsx +0 -69
- package/src/components/chat/tool/index.tsx +0 -65
- package/src/components/chat/tool/preview.ts +0 -57
- package/src/components/sidebar/ContextGauge.tsx +0 -102
- package/src/components/sidebar/Sidebar.tsx +0 -143
- package/src/components/tabs/TabBar.tsx +0 -50
- package/src/components/ui/FileLink.tsx +0 -52
- package/src/config/index.ts +0 -156
- package/src/config/lane.ts +0 -161
- package/src/config/models.ts +0 -95
- package/src/config/rules.ts +0 -80
- package/src/config/schema.ts +0 -308
- package/src/dialogs/alert.tsx +0 -52
- package/src/dialogs/chafa.tsx +0 -72
- package/src/dialogs/confirm.tsx +0 -58
- package/src/dialogs/curator.tsx +0 -153
- package/src/dialogs/eikon-picker.tsx +0 -95
- package/src/dialogs/help.tsx +0 -80
- package/src/dialogs/history.tsx +0 -92
- package/src/dialogs/info.tsx +0 -115
- package/src/dialogs/keys.tsx +0 -170
- package/src/dialogs/logs.tsx +0 -42
- package/src/dialogs/message.tsx +0 -38
- package/src/dialogs/model-picker.tsx +0 -123
- package/src/dialogs/new-profile.tsx +0 -69
- package/src/dialogs/new-task.tsx +0 -103
- package/src/dialogs/profile.tsx +0 -55
- package/src/dialogs/rollback.tsx +0 -190
- package/src/dialogs/spawn-history.tsx +0 -80
- package/src/dialogs/text-prompt.tsx +0 -68
- package/src/dialogs/theme-picker.tsx +0 -50
- package/src/home/index.ts +0 -23
- package/src/home/store.ts +0 -267
- package/src/index.tsx +0 -113
- package/src/keys/catalog.ts +0 -115
- package/src/keys/chord.ts +0 -125
- package/src/keys/conflicts.ts +0 -48
- package/src/keys/context.tsx +0 -112
- package/src/keys/index.ts +0 -5
- package/src/keys/list.ts +0 -94
- package/src/keys/oc-compat.ts +0 -87
- package/src/tabs/Agents.tsx +0 -607
- package/src/tabs/Analytics.tsx +0 -154
- package/src/tabs/Chat.tsx +0 -50
- package/src/tabs/Config.tsx +0 -605
- package/src/tabs/Context.tsx +0 -599
- package/src/tabs/Cron.tsx +0 -294
- package/src/tabs/Env.tsx +0 -227
- package/src/tabs/Kanban.tsx +0 -367
- package/src/tabs/Memory.tsx +0 -294
- package/src/tabs/Sessions.tsx +0 -786
- package/src/tabs/Skills.tsx +0 -507
- package/src/tabs/Toolsets.tsx +0 -266
- package/src/theme/builtin.ts +0 -78
- package/src/theme/context.tsx +0 -106
- package/src/theme/index.ts +0 -4
- package/src/theme/resolve.ts +0 -134
- package/src/theme/syntax.ts +0 -31
- package/src/theme/themes/aura.json +0 -69
- package/src/theme/themes/ayu.json +0 -80
- package/src/theme/themes/carbonfox.json +0 -248
- package/src/theme/themes/catppuccin-frappe.json +0 -233
- package/src/theme/themes/catppuccin-macchiato.json +0 -233
- package/src/theme/themes/catppuccin.json +0 -112
- package/src/theme/themes/cobalt2.json +0 -228
- package/src/theme/themes/cursor.json +0 -249
- package/src/theme/themes/dracula.json +0 -219
- package/src/theme/themes/everforest.json +0 -241
- package/src/theme/themes/flexoki.json +0 -237
- package/src/theme/themes/github.json +0 -233
- package/src/theme/themes/gruvbox.json +0 -242
- package/src/theme/themes/kanagawa.json +0 -77
- package/src/theme/themes/lucent-orng.json +0 -237
- package/src/theme/themes/material.json +0 -235
- package/src/theme/themes/matrix.json +0 -77
- package/src/theme/themes/mercury.json +0 -252
- package/src/theme/themes/monokai.json +0 -221
- package/src/theme/themes/nightowl.json +0 -221
- package/src/theme/themes/nord.json +0 -223
- package/src/theme/themes/one-dark.json +0 -84
- package/src/theme/themes/opencode.json +0 -245
- package/src/theme/themes/orng.json +0 -249
- package/src/theme/themes/osaka-jade.json +0 -93
- package/src/theme/themes/palenight.json +0 -222
- package/src/theme/themes/rosepine.json +0 -234
- package/src/theme/themes/solarized.json +0 -223
- package/src/theme/themes/synthwave84.json +0 -226
- package/src/theme/themes/tokyonight.json +0 -243
- package/src/theme/themes/vercel.json +0 -245
- package/src/theme/themes/vesper.json +0 -218
- package/src/theme/themes/zenburn.json +0 -223
- package/src/theme/types.ts +0 -119
- package/src/types/message.ts +0 -97
- package/src/ui/ChafaImage.tsx +0 -64
- package/src/ui/Splash.tsx +0 -118
- package/src/ui/borders.ts +0 -28
- package/src/ui/command.tsx +0 -104
- package/src/ui/dialog-select.tsx +0 -164
- package/src/ui/dialog.tsx +0 -102
- package/src/ui/fmt.ts +0 -82
- package/src/ui/kv.tsx +0 -28
- package/src/ui/shell.tsx +0 -45
- package/src/ui/spinner.tsx +0 -59
- package/src/ui/splash-art.ts +0 -123
- package/src/ui/table.tsx +0 -117
- package/src/ui/ticker.tsx +0 -90
- package/src/ui/toast.tsx +0 -130
- package/src/utils/categorical.ts +0 -77
- package/src/utils/chafa.ts +0 -173
- package/src/utils/clipboard.ts +0 -67
- package/src/utils/context-segments.ts +0 -317
- package/src/utils/control.ts +0 -495
- package/src/utils/drop.ts +0 -25
- package/src/utils/editor.ts +0 -33
- package/src/utils/fuzzy.ts +0 -45
- package/src/utils/gateway-client.ts +0 -253
- package/src/utils/gateway-types.ts +0 -282
- package/src/utils/git.ts +0 -57
- package/src/utils/hermes-analytics.ts +0 -134
- package/src/utils/hermes-home.ts +0 -821
- package/src/utils/hermes-kanban.ts +0 -154
- package/src/utils/hermes-profiles.ts +0 -217
- package/src/utils/interpolate.ts +0 -31
- package/src/utils/math-unicode.ts +0 -818
- package/src/utils/memory-activity.ts +0 -140
- package/src/utils/open-file.ts +0 -13
- package/src/utils/paths.ts +0 -52
- package/src/utils/perf.ts +0 -235
- package/src/utils/preferences.ts +0 -150
- package/src/utils/sessions-db.ts +0 -396
- package/src/utils/subagent-tree.ts +0 -146
- package/src/utils/terminal-reset.ts +0 -129
- package/src/utils/tips.ts +0 -67
- package/src/utils/tokens.ts +0 -87
package/src/tabs/Cron.tsx
DELETED
|
@@ -1,294 +0,0 @@
|
|
|
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
DELETED
|
@@ -1,227 +0,0 @@
|
|
|
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
|
-
})
|