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,67 @@
|
|
|
1
|
+
// Hermes CLI ships ~200 one-line tips in hermes_cli/tips.py. There's
|
|
2
|
+
// no `tips.list` RPC (see UPSTREAM.md); rather than shell.exec a python
|
|
3
|
+
// one-liner on every boot, read the source file once and scrape the
|
|
4
|
+
// string literals out of the `TIPS = [ ... ]` block. Brittle-by-design
|
|
5
|
+
// but zero-cost; falls back to a small built-in set if the file moved.
|
|
6
|
+
|
|
7
|
+
import { readFileSync } from "node:fs"
|
|
8
|
+
import { join } from "node:path"
|
|
9
|
+
import { hermesAgentRoot } from "./gateway-client"
|
|
10
|
+
|
|
11
|
+
const FALLBACK = [
|
|
12
|
+
"`@file:path/to/file.py` injects file contents directly into your message.",
|
|
13
|
+
"`/title <name>` names the session — resume it later from the Sessions tab.",
|
|
14
|
+
"Ctrl+G opens $EDITOR seeded with the composer contents.",
|
|
15
|
+
"Ctrl+Z suspends to the shell; `fg` resumes.",
|
|
16
|
+
"Pasting 5+ lines collapses to a `[Pasted #N …]` placeholder.",
|
|
17
|
+
"Click a user message in the transcript to rewind to that point.",
|
|
18
|
+
]
|
|
19
|
+
|
|
20
|
+
// Tokens worth accenting in a tip line: /slash, @refs, keybinds,
|
|
21
|
+
// `code`, "quoted".
|
|
22
|
+
const HL = /(\/[a-z][\w-]*|@[\w:./-]+|(?:Ctrl|Alt|Shift)\+\S+|`[^`]+`|"[^"]+")/g
|
|
23
|
+
|
|
24
|
+
type TipPart = { t: string; hl: boolean }
|
|
25
|
+
|
|
26
|
+
export function splitTip(tip: string): TipPart[] {
|
|
27
|
+
const out: TipPart[] = []
|
|
28
|
+
let i = 0
|
|
29
|
+
for (const m of tip.matchAll(HL)) {
|
|
30
|
+
const j = m.index
|
|
31
|
+
if (j > i) out.push({ t: tip.slice(i, j), hl: false })
|
|
32
|
+
out.push({ t: m[0].replace(/^`|`$/g, ""), hl: true })
|
|
33
|
+
i = j + m[0].length
|
|
34
|
+
}
|
|
35
|
+
if (i < tip.length) out.push({ t: tip.slice(i), hl: false })
|
|
36
|
+
return out
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
let cache: string[] | null = null
|
|
40
|
+
|
|
41
|
+
export function loadTips(): string[] {
|
|
42
|
+
if (cache) return cache
|
|
43
|
+
try {
|
|
44
|
+
const src = readFileSync(join(hermesAgentRoot(), "hermes_cli", "tips.py"), "utf8")
|
|
45
|
+
const body = src.split(/^TIPS\s*=\s*\[/m)[1]?.split(/^\]/m)[0] ?? ""
|
|
46
|
+
// Each tip is a double-quoted single-line string literal. Pull the
|
|
47
|
+
// inner text, unescape \" and \\, drop comments/blank lines.
|
|
48
|
+
const tips: string[] = []
|
|
49
|
+
for (const line of body.split("\n")) {
|
|
50
|
+
const m = line.match(/^\s+"((?:[^"\\]|\\.)*)",?\s*$/)
|
|
51
|
+
if (m) tips.push(m[1].replace(/\\"/g, '"').replace(/\\\\/g, "\\"))
|
|
52
|
+
}
|
|
53
|
+
cache = tips.length > 10 ? tips : FALLBACK
|
|
54
|
+
} catch {
|
|
55
|
+
cache = FALLBACK
|
|
56
|
+
}
|
|
57
|
+
return cache
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/** Random tip; never the same twice in a row. */
|
|
61
|
+
export function randomTip(prev?: string): string {
|
|
62
|
+
const t = loadTips()
|
|
63
|
+
if (t.length < 2) return t[0] ?? ""
|
|
64
|
+
let pick = t[Math.floor(Math.random() * t.length)]
|
|
65
|
+
while (pick === prev) pick = t[Math.floor(Math.random() * t.length)]
|
|
66
|
+
return pick
|
|
67
|
+
}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Token count estimator.
|
|
3
|
+
*
|
|
4
|
+
* Uses gpt-tokenizer with o200k_base (GPT-4o / GPT-5 / Claude-close-enough)
|
|
5
|
+
* for accurate counts on static blocks. Falls back to chars/4 if the
|
|
6
|
+
* tokenizer import fails at runtime (shouldn't in production, but keeps
|
|
7
|
+
* tests resilient and prevents a bad dep from breaking renders).
|
|
8
|
+
*
|
|
9
|
+
* Load-time: gpt-tokenizer is 55MB and costs ~170ms to import — roughly
|
|
10
|
+
* half the cold-start import graph. Nothing on the first-frame path
|
|
11
|
+
* (splash, composer, sidebar) needs real token counts, so the module is
|
|
12
|
+
* require()'d lazily on first count() call. Bun's require() is sync and
|
|
13
|
+
* cached, so the first call takes the hit and subsequent calls are free.
|
|
14
|
+
* Background warmup() lets app.tsx kick the import after first render
|
|
15
|
+
* so the first Context-tab visit doesn't stall.
|
|
16
|
+
*
|
|
17
|
+
* Cached by content hash (DJB2) so repeated counts of the same text
|
|
18
|
+
* are free. The grid recomputes on every snapshot refresh (10s) but
|
|
19
|
+
* most content is stable across refreshes — skills, tool schemas, SOUL.
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
type Enc = { countTokens: (s: string) => number }
|
|
23
|
+
let enc: Enc | null | undefined
|
|
24
|
+
|
|
25
|
+
const load = (): Enc | null => {
|
|
26
|
+
if (enc !== undefined) return enc
|
|
27
|
+
try { enc = require("gpt-tokenizer") as Enc }
|
|
28
|
+
catch { enc = null }
|
|
29
|
+
return enc
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/** Kick the lazy import off the hot path. Fire-and-forget. */
|
|
33
|
+
export const warmup = () => { queueMicrotask(load) }
|
|
34
|
+
|
|
35
|
+
// Simple DJB2 hash — fast, collision-tolerant for cache keys.
|
|
36
|
+
const hash = (s: string): string => {
|
|
37
|
+
let h = 5381
|
|
38
|
+
for (let i = 0; i < s.length; i++) h = ((h << 5) + h + s.charCodeAt(i)) | 0
|
|
39
|
+
return String(h)
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
const cache = new Map<string, number>()
|
|
43
|
+
const CACHE_MAX = 1024
|
|
44
|
+
|
|
45
|
+
// Fallback when tokenizer unavailable — matches the old chars/4 behavior.
|
|
46
|
+
const roughCount = (s: string): number => Math.ceil(s.length / 4)
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Count tokens in a string. Cached by content hash.
|
|
50
|
+
* Returns 0 for empty string.
|
|
51
|
+
*/
|
|
52
|
+
export function count(s: string): number {
|
|
53
|
+
if (!s) return 0
|
|
54
|
+
const k = hash(s)
|
|
55
|
+
const hit = cache.get(k)
|
|
56
|
+
if (hit !== undefined) return hit
|
|
57
|
+
let n: number
|
|
58
|
+
try { n = load()?.countTokens(s) ?? roughCount(s) }
|
|
59
|
+
catch { n = roughCount(s) }
|
|
60
|
+
if (cache.size >= CACHE_MAX) {
|
|
61
|
+
const first = cache.keys().next().value
|
|
62
|
+
if (first !== undefined) cache.delete(first)
|
|
63
|
+
}
|
|
64
|
+
cache.set(k, n)
|
|
65
|
+
return n
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
/** Clear the count cache. For tests. */
|
|
69
|
+
export function clearCache(): void {
|
|
70
|
+
cache.clear()
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Human-format a token count for compact display:
|
|
75
|
+
* 999 → "999" 1_000 → "1.0K" 10_000 → "10K"
|
|
76
|
+
* 258_000 → "258K" 1_000_000 → "1M" 1_250_000 → "1.2M"
|
|
77
|
+
*/
|
|
78
|
+
export function formatTokens(n: number): string {
|
|
79
|
+
if (!Number.isFinite(n) || n < 0) return "0"
|
|
80
|
+
if (n >= 1_000_000) {
|
|
81
|
+
const m = n / 1_000_000
|
|
82
|
+
return m === Math.floor(m) ? `${m}M` : `${m.toFixed(1)}M`
|
|
83
|
+
}
|
|
84
|
+
if (n >= 10_000) return `${Math.round(n / 1000)}K`
|
|
85
|
+
if (n >= 1_000) return `${(n / 1000).toFixed(1)}K`
|
|
86
|
+
return String(Math.round(n))
|
|
87
|
+
}
|