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.
Files changed (175) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +54 -0
  3. package/package.json +82 -0
  4. package/scripts/postinstall.ts +29 -0
  5. package/src/app/gateway.tsx +83 -0
  6. package/src/app/gatewayEvents.ts +203 -0
  7. package/src/app/launch.ts +41 -0
  8. package/src/app/skin.tsx +31 -0
  9. package/src/app/spawnHistory.ts +75 -0
  10. package/src/app/tabs.ts +23 -0
  11. package/src/app/turnReducer.ts +390 -0
  12. package/src/app/useAppKeys.ts +268 -0
  13. package/src/app/useAtRefPopover.ts +99 -0
  14. package/src/app/useInputHistory.ts +66 -0
  15. package/src/app/useSession.ts +102 -0
  16. package/src/app/useSlashCommands.ts +70 -0
  17. package/src/app/useSlashPopover.ts +48 -0
  18. package/src/app.tsx +917 -0
  19. package/src/commands/slash.ts +151 -0
  20. package/src/components/avatar/AnimatedAvatar.tsx +66 -0
  21. package/src/components/avatar/eikon.ts +144 -0
  22. package/src/components/avatar/states/error.ts +1155 -0
  23. package/src/components/avatar/states/idle.ts +1155 -0
  24. package/src/components/avatar/states/index.ts +30 -0
  25. package/src/components/avatar/states/listening.ts +1155 -0
  26. package/src/components/avatar/states/speaking.ts +1155 -0
  27. package/src/components/avatar/states/thinking.ts +1155 -0
  28. package/src/components/avatar/states/working.ts +1155 -0
  29. package/src/components/chat/AtRefPopover.tsx +54 -0
  30. package/src/components/chat/CodeBlock.tsx +67 -0
  31. package/src/components/chat/Composer.tsx +347 -0
  32. package/src/components/chat/DiffBlock.tsx +116 -0
  33. package/src/components/chat/ErrorBlock.tsx +70 -0
  34. package/src/components/chat/MediaChip.tsx +114 -0
  35. package/src/components/chat/MessageItem.tsx +282 -0
  36. package/src/components/chat/MessageList.tsx +114 -0
  37. package/src/components/chat/PromptCard.tsx +359 -0
  38. package/src/components/chat/SlashPopover.tsx +158 -0
  39. package/src/components/chat/ThoughtCloud.tsx +185 -0
  40. package/src/components/chat/TypingIndicator.tsx +25 -0
  41. package/src/components/chat/tool/Subagent.tsx +75 -0
  42. package/src/components/chat/tool/frame.tsx +69 -0
  43. package/src/components/chat/tool/index.tsx +65 -0
  44. package/src/components/chat/tool/preview.ts +57 -0
  45. package/src/components/sidebar/ContextGauge.tsx +102 -0
  46. package/src/components/sidebar/Sidebar.tsx +143 -0
  47. package/src/components/tabs/TabBar.tsx +50 -0
  48. package/src/components/ui/FileLink.tsx +52 -0
  49. package/src/config/index.ts +156 -0
  50. package/src/config/lane.ts +161 -0
  51. package/src/config/models.ts +95 -0
  52. package/src/config/rules.ts +80 -0
  53. package/src/config/schema.ts +308 -0
  54. package/src/dialogs/alert.tsx +52 -0
  55. package/src/dialogs/chafa.tsx +72 -0
  56. package/src/dialogs/confirm.tsx +58 -0
  57. package/src/dialogs/curator.tsx +153 -0
  58. package/src/dialogs/eikon-picker.tsx +95 -0
  59. package/src/dialogs/help.tsx +80 -0
  60. package/src/dialogs/history.tsx +92 -0
  61. package/src/dialogs/info.tsx +115 -0
  62. package/src/dialogs/keys.tsx +170 -0
  63. package/src/dialogs/logs.tsx +42 -0
  64. package/src/dialogs/message.tsx +38 -0
  65. package/src/dialogs/model-picker.tsx +123 -0
  66. package/src/dialogs/new-profile.tsx +69 -0
  67. package/src/dialogs/new-task.tsx +103 -0
  68. package/src/dialogs/profile.tsx +55 -0
  69. package/src/dialogs/rollback.tsx +190 -0
  70. package/src/dialogs/spawn-history.tsx +80 -0
  71. package/src/dialogs/text-prompt.tsx +68 -0
  72. package/src/dialogs/theme-picker.tsx +50 -0
  73. package/src/home/index.ts +23 -0
  74. package/src/home/store.ts +267 -0
  75. package/src/index.tsx +113 -0
  76. package/src/keys/catalog.ts +115 -0
  77. package/src/keys/chord.ts +125 -0
  78. package/src/keys/conflicts.ts +48 -0
  79. package/src/keys/context.tsx +112 -0
  80. package/src/keys/index.ts +5 -0
  81. package/src/keys/list.ts +94 -0
  82. package/src/keys/oc-compat.ts +87 -0
  83. package/src/tabs/Agents.tsx +607 -0
  84. package/src/tabs/Analytics.tsx +154 -0
  85. package/src/tabs/Chat.tsx +50 -0
  86. package/src/tabs/Config.tsx +605 -0
  87. package/src/tabs/Context.tsx +599 -0
  88. package/src/tabs/Cron.tsx +294 -0
  89. package/src/tabs/Env.tsx +227 -0
  90. package/src/tabs/Kanban.tsx +367 -0
  91. package/src/tabs/Memory.tsx +294 -0
  92. package/src/tabs/Sessions.tsx +786 -0
  93. package/src/tabs/Skills.tsx +507 -0
  94. package/src/tabs/Toolsets.tsx +266 -0
  95. package/src/theme/builtin.ts +78 -0
  96. package/src/theme/context.tsx +106 -0
  97. package/src/theme/index.ts +4 -0
  98. package/src/theme/resolve.ts +134 -0
  99. package/src/theme/syntax.ts +31 -0
  100. package/src/theme/themes/aura.json +69 -0
  101. package/src/theme/themes/ayu.json +80 -0
  102. package/src/theme/themes/carbonfox.json +248 -0
  103. package/src/theme/themes/catppuccin-frappe.json +233 -0
  104. package/src/theme/themes/catppuccin-macchiato.json +233 -0
  105. package/src/theme/themes/catppuccin.json +112 -0
  106. package/src/theme/themes/cobalt2.json +228 -0
  107. package/src/theme/themes/cursor.json +249 -0
  108. package/src/theme/themes/dracula.json +219 -0
  109. package/src/theme/themes/everforest.json +241 -0
  110. package/src/theme/themes/flexoki.json +237 -0
  111. package/src/theme/themes/github.json +233 -0
  112. package/src/theme/themes/gruvbox.json +242 -0
  113. package/src/theme/themes/kanagawa.json +77 -0
  114. package/src/theme/themes/lucent-orng.json +237 -0
  115. package/src/theme/themes/material.json +235 -0
  116. package/src/theme/themes/matrix.json +77 -0
  117. package/src/theme/themes/mercury.json +252 -0
  118. package/src/theme/themes/monokai.json +221 -0
  119. package/src/theme/themes/nightowl.json +221 -0
  120. package/src/theme/themes/nord.json +223 -0
  121. package/src/theme/themes/one-dark.json +84 -0
  122. package/src/theme/themes/opencode.json +245 -0
  123. package/src/theme/themes/orng.json +249 -0
  124. package/src/theme/themes/osaka-jade.json +93 -0
  125. package/src/theme/themes/palenight.json +222 -0
  126. package/src/theme/themes/rosepine.json +234 -0
  127. package/src/theme/themes/solarized.json +223 -0
  128. package/src/theme/themes/synthwave84.json +226 -0
  129. package/src/theme/themes/tokyonight.json +243 -0
  130. package/src/theme/themes/vercel.json +245 -0
  131. package/src/theme/themes/vesper.json +218 -0
  132. package/src/theme/themes/zenburn.json +223 -0
  133. package/src/theme/types.ts +119 -0
  134. package/src/types/message.ts +97 -0
  135. package/src/ui/ChafaImage.tsx +64 -0
  136. package/src/ui/Splash.tsx +118 -0
  137. package/src/ui/borders.ts +28 -0
  138. package/src/ui/command.tsx +104 -0
  139. package/src/ui/dialog-select.tsx +164 -0
  140. package/src/ui/dialog.tsx +102 -0
  141. package/src/ui/fmt.ts +82 -0
  142. package/src/ui/kv.tsx +28 -0
  143. package/src/ui/shell.tsx +45 -0
  144. package/src/ui/spinner.tsx +59 -0
  145. package/src/ui/splash-art.ts +123 -0
  146. package/src/ui/table.tsx +117 -0
  147. package/src/ui/ticker.tsx +90 -0
  148. package/src/ui/toast.tsx +130 -0
  149. package/src/utils/categorical.ts +77 -0
  150. package/src/utils/chafa.ts +173 -0
  151. package/src/utils/clipboard.ts +67 -0
  152. package/src/utils/context-segments.ts +317 -0
  153. package/src/utils/control.ts +495 -0
  154. package/src/utils/drop.ts +25 -0
  155. package/src/utils/editor.ts +33 -0
  156. package/src/utils/fuzzy.ts +45 -0
  157. package/src/utils/gateway-client.ts +253 -0
  158. package/src/utils/gateway-types.ts +282 -0
  159. package/src/utils/git.ts +57 -0
  160. package/src/utils/hermes-analytics.ts +134 -0
  161. package/src/utils/hermes-home.ts +821 -0
  162. package/src/utils/hermes-kanban.ts +154 -0
  163. package/src/utils/hermes-profiles.ts +217 -0
  164. package/src/utils/interpolate.ts +31 -0
  165. package/src/utils/math-unicode.ts +818 -0
  166. package/src/utils/memory-activity.ts +140 -0
  167. package/src/utils/open-file.ts +13 -0
  168. package/src/utils/paths.ts +52 -0
  169. package/src/utils/perf.ts +235 -0
  170. package/src/utils/preferences.ts +150 -0
  171. package/src/utils/sessions-db.ts +396 -0
  172. package/src/utils/subagent-tree.ts +146 -0
  173. package/src/utils/terminal-reset.ts +129 -0
  174. package/src/utils/tips.ts +67 -0
  175. 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
+ }