hermium 0.2.0 → 0.3.0

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 (157) hide show
  1. package/bin/hermium.mjs +102 -58
  2. package/dist/server/__23tanstack-start-plugin-adapters-Cwee5PKy.mjs +6 -0
  3. package/dist/server/_chunks/ssr-renderer.mjs +22 -0
  4. package/dist/server/_libs/bail.mjs +8 -0
  5. package/dist/server/_libs/base-ui__react.mjs +9858 -0
  6. package/dist/server/_libs/base-ui__utils.mjs +1106 -0
  7. package/dist/server/_libs/ccount.mjs +16 -0
  8. package/dist/server/_libs/character-entities.mjs +2130 -0
  9. package/dist/server/_libs/class-variance-authority.mjs +44 -0
  10. package/dist/server/_libs/clsx.mjs +16 -0
  11. package/dist/server/_libs/comma-separated-tokens.mjs +10 -0
  12. package/dist/server/_libs/cookie-es.mjs +1 -0
  13. package/dist/server/_libs/croner.mjs +1 -0
  14. package/dist/server/_libs/crossws.mjs +1 -0
  15. package/dist/server/_libs/decode-named-character-reference+[...].mjs +8 -0
  16. package/dist/server/_libs/devlop.mjs +8 -0
  17. package/dist/server/_libs/escape-string-regexp.mjs +9 -0
  18. package/dist/server/_libs/estree-util-is-identifier-name.mjs +11 -0
  19. package/dist/server/_libs/extend.mjs +97 -0
  20. package/dist/server/_libs/floating-ui__core.mjs +663 -0
  21. package/dist/server/_libs/floating-ui__dom.mjs +624 -0
  22. package/dist/server/_libs/floating-ui__react-dom.mjs +279 -0
  23. package/dist/server/_libs/floating-ui__utils.mjs +322 -0
  24. package/dist/server/_libs/h3.mjs +408 -0
  25. package/dist/server/_libs/hast-util-is-element.mjs +75 -0
  26. package/dist/server/_libs/hast-util-to-jsx-runtime.mjs +388 -0
  27. package/dist/server/_libs/hast-util-to-text.mjs +305 -0
  28. package/dist/server/_libs/hast-util-whitespace.mjs +10 -0
  29. package/dist/server/_libs/highlight.js.mjs +14756 -0
  30. package/dist/server/_libs/hookable.mjs +1 -0
  31. package/dist/server/_libs/html-url-attributes.mjs +26 -0
  32. package/dist/server/_libs/inline-style-parser.mjs +142 -0
  33. package/dist/server/_libs/is-plain-obj.mjs +10 -0
  34. package/dist/server/_libs/isbot.mjs +21 -0
  35. package/dist/server/_libs/longest-streak.mjs +25 -0
  36. package/dist/server/_libs/lowlight.mjs +262 -0
  37. package/dist/server/_libs/markdown-table.mjs +142 -0
  38. package/dist/server/_libs/mdast-util-find-and-replace.mjs +109 -0
  39. package/dist/server/_libs/mdast-util-from-markdown.mjs +717 -0
  40. package/dist/server/_libs/mdast-util-gfm-autolink-literal+[...].mjs +156 -0
  41. package/dist/server/_libs/mdast-util-gfm-footnote.mjs +117 -0
  42. package/dist/server/_libs/mdast-util-gfm-strikethrough.mjs +54 -0
  43. package/dist/server/_libs/mdast-util-gfm-table.mjs +157 -0
  44. package/dist/server/_libs/mdast-util-gfm-task-list-item.mjs +77 -0
  45. package/dist/server/_libs/mdast-util-gfm.mjs +29 -0
  46. package/dist/server/_libs/mdast-util-phrasing.mjs +30 -0
  47. package/dist/server/_libs/mdast-util-to-hast.mjs +710 -0
  48. package/dist/server/_libs/mdast-util-to-markdown.mjs +798 -0
  49. package/dist/server/_libs/mdast-util-to-string.mjs +38 -0
  50. package/dist/server/_libs/micromark-core-commonmark.mjs +2259 -0
  51. package/dist/server/_libs/micromark-extension-gfm-autolink-literal+[...].mjs +344 -0
  52. package/dist/server/_libs/micromark-extension-gfm-footnote+[...].mjs +279 -0
  53. package/dist/server/_libs/micromark-extension-gfm-strikethrough+[...].mjs +98 -0
  54. package/dist/server/_libs/micromark-extension-gfm-table.mjs +491 -0
  55. package/dist/server/_libs/micromark-extension-gfm-tagfilter+[...].mjs +1 -0
  56. package/dist/server/_libs/micromark-extension-gfm-task-list-item+[...].mjs +77 -0
  57. package/dist/server/_libs/micromark-extension-gfm.mjs +18 -0
  58. package/dist/server/_libs/micromark-factory-destination.mjs +94 -0
  59. package/dist/server/_libs/micromark-factory-label.mjs +63 -0
  60. package/dist/server/_libs/micromark-factory-space.mjs +24 -0
  61. package/dist/server/_libs/micromark-factory-title.mjs +65 -0
  62. package/dist/server/_libs/micromark-factory-whitespace.mjs +22 -0
  63. package/dist/server/_libs/micromark-util-character.mjs +44 -0
  64. package/dist/server/_libs/micromark-util-chunked.mjs +36 -0
  65. package/dist/server/_libs/micromark-util-classify-character+[...].mjs +12 -0
  66. package/dist/server/_libs/micromark-util-combine-extensions+[...].mjs +41 -0
  67. package/dist/server/_libs/micromark-util-decode-numeric-character-reference+[...].mjs +19 -0
  68. package/dist/server/_libs/micromark-util-decode-string.mjs +21 -0
  69. package/dist/server/_libs/micromark-util-encode.mjs +1 -0
  70. package/dist/server/_libs/micromark-util-html-tag-name.mjs +69 -0
  71. package/dist/server/_libs/micromark-util-normalize-identifier+[...].mjs +6 -0
  72. package/dist/server/_libs/micromark-util-resolve-all.mjs +15 -0
  73. package/dist/server/_libs/micromark-util-sanitize-uri.mjs +41 -0
  74. package/dist/server/_libs/micromark-util-subtokenize.mjs +346 -0
  75. package/dist/server/_libs/micromark.mjs +906 -0
  76. package/dist/server/_libs/ocache.mjs +1 -0
  77. package/dist/server/_libs/ohash.mjs +1 -0
  78. package/dist/server/_libs/property-information.mjs +1209 -0
  79. package/dist/server/_libs/react-dom.mjs +10779 -0
  80. package/dist/server/_libs/react-markdown.mjs +147 -0
  81. package/dist/server/_libs/react.mjs +513 -0
  82. package/dist/server/_libs/rehype-highlight.mjs +94 -0
  83. package/dist/server/_libs/remark-gfm.mjs +20 -0
  84. package/dist/server/_libs/remark-parse.mjs +19 -0
  85. package/dist/server/_libs/remark-rehype.mjs +21 -0
  86. package/dist/server/_libs/reselect.mjs +1 -0
  87. package/dist/server/_libs/rou3.mjs +8 -0
  88. package/dist/server/_libs/seroval-plugins.mjs +1 -0
  89. package/dist/server/_libs/seroval.mjs +1 -0
  90. package/dist/server/_libs/space-separated-tokens.mjs +6 -0
  91. package/dist/server/_libs/srvx.mjs +781 -0
  92. package/dist/server/_libs/style-to-js.mjs +72 -0
  93. package/dist/server/_libs/style-to-object.mjs +38 -0
  94. package/dist/server/_libs/tabler__icons-react.mjs +140 -0
  95. package/dist/server/_libs/tailwind-merge.mjs +3255 -0
  96. package/dist/server/_libs/tanstack__history.mjs +29 -0
  97. package/dist/server/_libs/tanstack__react-router.mjs +1120 -0
  98. package/dist/server/_libs/tanstack__react-store.mjs +2 -0
  99. package/dist/server/_libs/tanstack__router-core.mjs +3594 -0
  100. package/dist/server/_libs/tanstack__store.mjs +1 -0
  101. package/dist/server/_libs/trim-lines.mjs +41 -0
  102. package/dist/server/_libs/trough.mjs +85 -0
  103. package/dist/server/_libs/ufo.mjs +54 -0
  104. package/dist/server/_libs/unctx.mjs +1 -0
  105. package/dist/server/_libs/ungap__structured-clone.mjs +224 -0
  106. package/dist/server/_libs/unified.mjs +661 -0
  107. package/dist/server/_libs/unist-util-find-after.mjs +41 -0
  108. package/dist/server/_libs/unist-util-is.mjs +100 -0
  109. package/dist/server/_libs/unist-util-position.mjs +27 -0
  110. package/dist/server/_libs/unist-util-stringify-position.mjs +27 -0
  111. package/dist/server/_libs/unist-util-visit-parents.mjs +83 -0
  112. package/dist/server/_libs/unist-util-visit.mjs +24 -0
  113. package/dist/server/_libs/unstorage.mjs +1 -0
  114. package/dist/server/_libs/use-sync-external-store.mjs +139 -0
  115. package/dist/server/_libs/vfile-message.mjs +138 -0
  116. package/dist/server/_libs/vfile.mjs +467 -0
  117. package/dist/server/_libs/zustand.mjs +43 -0
  118. package/dist/server/_libs/zwitch.mjs +1 -0
  119. package/dist/server/_ssr/ChatInputBlock-Bu2-iop_.mjs +220 -0
  120. package/dist/server/_ssr/MarkdownMessage-CNS7OSKN.mjs +68 -0
  121. package/dist/server/_ssr/chat._sessionId-P02iSfut.mjs +477 -0
  122. package/dist/server/_ssr/chat.index-BYB_48NC.mjs +64 -0
  123. package/dist/server/_ssr/index-C1mT_2d8.mjs +4890 -0
  124. package/dist/server/_ssr/index-DFV9_oCk.mjs +43 -0
  125. package/dist/server/_ssr/memory-CW_fSOG9.mjs +257 -0
  126. package/dist/server/_ssr/router-CUAfx91O.mjs +2035 -0
  127. package/dist/server/_ssr/settings-DoXurzvn.mjs +10 -0
  128. package/dist/server/_ssr/skills-Cs7A5ZwO.mjs +422 -0
  129. package/dist/server/_ssr/theme-BK4-7E2h.mjs +42 -0
  130. package/dist/server/_ssr/usage-Bs2-LXGz.mjs +298 -0
  131. package/dist/server/_tanstack-start-manifest_v-C7Upe2TI.mjs +4 -0
  132. package/dist/server/index.mjs +506 -0
  133. package/package.json +4 -3
  134. package/dist/public/assets/css/index-Dfs9RUU9.css +0 -1
  135. package/dist/public/assets/css/styles-B8p6jk5Z.css +0 -1
  136. package/dist/public/assets/js/ChatInputBlock-Bw7AL70H.js +0 -1
  137. package/dist/public/assets/js/MarkdownMessage-8d7Y6VL-.js +0 -1
  138. package/dist/public/assets/js/base-ui-BvQbAt_1.js +0 -1
  139. package/dist/public/assets/js/chat._sessionId-BG6lVraH.js +0 -1
  140. package/dist/public/assets/js/chat.index-D2zdMPTT.js +0 -1
  141. package/dist/public/assets/js/index-C0AK45FU.js +0 -60
  142. package/dist/public/assets/js/index-Cx5En4FK.js +0 -1
  143. package/dist/public/assets/js/memory-CeSRdTkW.js +0 -3
  144. package/dist/public/assets/js/router-8uDKazL-.js +0 -1
  145. package/dist/public/assets/js/settings-Bc3Y5zXO.js +0 -1
  146. package/dist/public/assets/js/skills-DZv7sA_5.js +0 -1
  147. package/dist/public/assets/js/theme-CPkdkpaj.js +0 -1
  148. package/dist/public/assets/js/usage-DXQsT9_b.js +0 -1
  149. package/dist/public/assets/woff2/geist-cyrillic-ext-wght-normal-DjL33-gN.woff2 +0 -0
  150. package/dist/public/assets/woff2/geist-cyrillic-wght-normal-BEAKL7Jp.woff2 +0 -0
  151. package/dist/public/assets/woff2/geist-latin-ext-wght-normal-DC-KSUi6.woff2 +0 -0
  152. package/dist/public/assets/woff2/geist-latin-wght-normal-BgDaEnEv.woff2 +0 -0
  153. package/dist/public/assets/woff2/geist-vietnamese-wght-normal-6IgcOCM7.woff2 +0 -0
  154. package/dist/public/favicon.ico +0 -0
  155. package/dist/public/logo.png +0 -0
  156. package/dist/public/manifest.json +0 -25
  157. package/dist/public/robots.txt +0 -3
@@ -0,0 +1,10 @@
1
+ import { j as jsxRuntimeExports } from "../_libs/react.mjs";
2
+ function SettingsPage() {
3
+ return /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "p-6", children: [
4
+ /* @__PURE__ */ jsxRuntimeExports.jsx("h1", { className: "text-2xl font-semibold", children: "Settings" }),
5
+ /* @__PURE__ */ jsxRuntimeExports.jsx("p", { className: "mt-2 text-muted-foreground", children: "Configure profiles, models, and preferences." })
6
+ ] });
7
+ }
8
+ export {
9
+ SettingsPage as component
10
+ };
@@ -0,0 +1,422 @@
1
+ import { r as reactExports, j as jsxRuntimeExports } from "../_libs/react.mjs";
2
+ import { d as useSidebar, S as SidebarTrigger, c as cn, B as Button, r as request } from "./router-CUAfx91O.mjs";
3
+ import { M as MarkdownMessage } from "./MarkdownMessage-CNS7OSKN.mjs";
4
+ import "./index-C1mT_2d8.mjs";
5
+ import { I as IconArrowLeft, m as IconLoader2, u as IconRefresh, k as IconFile, B as IconWand, v as IconSearch, f as IconChevronRight, e as IconChevronDown, r as IconPin } from "../_libs/tabler__icons-react.mjs";
6
+ import { I as Input$1, S as SwitchRoot, n as SwitchThumb } from "../_libs/base-ui__react.mjs";
7
+ import "../_libs/tanstack__react-router.mjs";
8
+ import "../_libs/tanstack__router-core.mjs";
9
+ import "../_libs/tanstack__history.mjs";
10
+ import "node:stream/web";
11
+ import "node:stream";
12
+ import "../_libs/react-dom.mjs";
13
+ import "util";
14
+ import "crypto";
15
+ import "async_hooks";
16
+ import "stream";
17
+ import "../_libs/isbot.mjs";
18
+ import "../_libs/clsx.mjs";
19
+ import "../_libs/tailwind-merge.mjs";
20
+ import "../_libs/class-variance-authority.mjs";
21
+ import "../_libs/zustand.mjs";
22
+ import "../_libs/base-ui__utils.mjs";
23
+ import "../_libs/use-sync-external-store.mjs";
24
+ import "../_libs/floating-ui__utils.mjs";
25
+ import "../_libs/floating-ui__react-dom.mjs";
26
+ import "../_libs/floating-ui__dom.mjs";
27
+ import "../_libs/floating-ui__core.mjs";
28
+ import "../_libs/react-markdown.mjs";
29
+ import "../_libs/devlop.mjs";
30
+ import "../_libs/unified.mjs";
31
+ import "../_libs/bail.mjs";
32
+ import "../_libs/extend.mjs";
33
+ import "../_libs/is-plain-obj.mjs";
34
+ import "../_libs/trough.mjs";
35
+ import "../_libs/vfile.mjs";
36
+ import "../_libs/vfile-message.mjs";
37
+ import "../_libs/unist-util-stringify-position.mjs";
38
+ import "node:process";
39
+ import "node:path";
40
+ import "node:url";
41
+ import "../_libs/remark-parse.mjs";
42
+ import "../_libs/mdast-util-from-markdown.mjs";
43
+ import "../_libs/micromark-util-decode-numeric-character-reference+[...].mjs";
44
+ import "../_libs/micromark-util-decode-string.mjs";
45
+ import "../_libs/decode-named-character-reference+[...].mjs";
46
+ import "../_libs/character-entities.mjs";
47
+ import "../_libs/micromark-util-normalize-identifier+[...].mjs";
48
+ import "../_libs/micromark.mjs";
49
+ import "../_libs/micromark-util-combine-extensions+[...].mjs";
50
+ import "../_libs/micromark-util-chunked.mjs";
51
+ import "../_libs/micromark-factory-space.mjs";
52
+ import "../_libs/micromark-util-character.mjs";
53
+ import "../_libs/micromark-core-commonmark.mjs";
54
+ import "../_libs/micromark-util-classify-character+[...].mjs";
55
+ import "../_libs/micromark-util-resolve-all.mjs";
56
+ import "../_libs/micromark-util-subtokenize.mjs";
57
+ import "../_libs/micromark-factory-destination.mjs";
58
+ import "../_libs/micromark-factory-label.mjs";
59
+ import "../_libs/micromark-factory-title.mjs";
60
+ import "../_libs/micromark-factory-whitespace.mjs";
61
+ import "../_libs/micromark-util-html-tag-name.mjs";
62
+ import "../_libs/mdast-util-to-string.mjs";
63
+ import "../_libs/remark-rehype.mjs";
64
+ import "../_libs/mdast-util-to-hast.mjs";
65
+ import "../_libs/ungap__structured-clone.mjs";
66
+ import "../_libs/micromark-util-sanitize-uri.mjs";
67
+ import "../_libs/unist-util-position.mjs";
68
+ import "../_libs/trim-lines.mjs";
69
+ import "../_libs/unist-util-visit.mjs";
70
+ import "../_libs/unist-util-visit-parents.mjs";
71
+ import "../_libs/unist-util-is.mjs";
72
+ import "../_libs/hast-util-to-jsx-runtime.mjs";
73
+ import "../_libs/comma-separated-tokens.mjs";
74
+ import "../_libs/property-information.mjs";
75
+ import "../_libs/space-separated-tokens.mjs";
76
+ import "../_libs/style-to-js.mjs";
77
+ import "../_libs/style-to-object.mjs";
78
+ import "../_libs/inline-style-parser.mjs";
79
+ import "../_libs/hast-util-whitespace.mjs";
80
+ import "../_libs/estree-util-is-identifier-name.mjs";
81
+ import "../_libs/html-url-attributes.mjs";
82
+ import "../_libs/rehype-highlight.mjs";
83
+ import "../_libs/lowlight.mjs";
84
+ import "../_libs/highlight.js.mjs";
85
+ import "../_libs/hast-util-to-text.mjs";
86
+ import "../_libs/hast-util-is-element.mjs";
87
+ import "../_libs/unist-util-find-after.mjs";
88
+ import "../_libs/remark-gfm.mjs";
89
+ import "../_libs/micromark-extension-gfm.mjs";
90
+ import "../_libs/micromark-extension-gfm-autolink-literal+[...].mjs";
91
+ import "../_libs/micromark-extension-gfm-footnote+[...].mjs";
92
+ import "../_libs/micromark-extension-gfm-strikethrough+[...].mjs";
93
+ import "../_libs/micromark-extension-gfm-table.mjs";
94
+ import "../_libs/micromark-extension-gfm-task-list-item+[...].mjs";
95
+ import "../_libs/mdast-util-gfm.mjs";
96
+ import "../_libs/mdast-util-gfm-autolink-literal+[...].mjs";
97
+ import "../_libs/ccount.mjs";
98
+ import "../_libs/mdast-util-find-and-replace.mjs";
99
+ import "../_libs/escape-string-regexp.mjs";
100
+ import "../_libs/mdast-util-gfm-footnote.mjs";
101
+ import "../_libs/mdast-util-gfm-strikethrough.mjs";
102
+ import "../_libs/mdast-util-gfm-table.mjs";
103
+ import "../_libs/markdown-table.mjs";
104
+ import "../_libs/mdast-util-to-markdown.mjs";
105
+ import "../_libs/longest-streak.mjs";
106
+ import "../_libs/mdast-util-phrasing.mjs";
107
+ import "../_libs/mdast-util-gfm-task-list-item.mjs";
108
+ import "node:async_hooks";
109
+ function Input({ className, type, ...props }) {
110
+ return /* @__PURE__ */ jsxRuntimeExports.jsx(
111
+ Input$1,
112
+ {
113
+ type,
114
+ "data-slot": "input",
115
+ className: cn(
116
+ "h-8 w-full min-w-0 rounded-lg border border-input bg-transparent px-2.5 py-1 text-base transition-colors outline-none file:inline-flex file:h-6 file:border-0 file:bg-transparent file:text-sm file:font-medium file:text-foreground placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-3 focus-visible:ring-ring/50 disabled:pointer-events-none disabled:cursor-not-allowed disabled:bg-input/50 disabled:opacity-50 aria-invalid:border-destructive aria-invalid:ring-3 aria-invalid:ring-destructive/20 md:text-sm dark:bg-input/30 dark:disabled:bg-input/80 dark:aria-invalid:border-destructive/50 dark:aria-invalid:ring-destructive/40",
117
+ className
118
+ ),
119
+ ...props
120
+ }
121
+ );
122
+ }
123
+ async function fetchSkills() {
124
+ return request("/api/hermes/skills");
125
+ }
126
+ async function fetchSkillContent(skillPath) {
127
+ const res = await request(`/api/hermes/skills/${skillPath}`);
128
+ return res.content;
129
+ }
130
+ async function fetchSkillFiles(category, skill) {
131
+ const res = await request(`/api/hermes/skills/${category}/${skill}/files`);
132
+ return res.files;
133
+ }
134
+ async function toggleSkill(name, enabled) {
135
+ await request("/api/hermes/skills/toggle", {
136
+ method: "PUT",
137
+ body: JSON.stringify({ name, enabled })
138
+ });
139
+ }
140
+ function Switch({
141
+ className,
142
+ size = "default",
143
+ ...props
144
+ }) {
145
+ return /* @__PURE__ */ jsxRuntimeExports.jsx(
146
+ SwitchRoot,
147
+ {
148
+ "data-slot": "switch",
149
+ "data-size": size,
150
+ className: cn(
151
+ "peer group/switch relative inline-flex shrink-0 items-center rounded-full border border-transparent transition-all outline-none after:absolute after:-inset-x-3 after:-inset-y-2 focus-visible:border-ring focus-visible:ring-3 focus-visible:ring-ring/50 aria-invalid:border-destructive aria-invalid:ring-3 aria-invalid:ring-destructive/20 data-[size=default]:h-[18.4px] data-[size=default]:w-[32px] data-[size=sm]:h-[14px] data-[size=sm]:w-[24px] dark:aria-invalid:border-destructive/50 dark:aria-invalid:ring-destructive/40 data-checked:bg-primary data-unchecked:bg-input dark:data-unchecked:bg-input/80 data-disabled:cursor-not-allowed data-disabled:opacity-50",
152
+ className
153
+ ),
154
+ ...props,
155
+ children: /* @__PURE__ */ jsxRuntimeExports.jsx(
156
+ SwitchThumb,
157
+ {
158
+ "data-slot": "switch-thumb",
159
+ className: "pointer-events-none block rounded-full bg-background ring-0 transition-transform group-data-[size=default]/switch:size-4 group-data-[size=sm]/switch:size-3 group-data-[size=default]/switch:data-checked:translate-x-[calc(100%-2px)] group-data-[size=sm]/switch:data-checked:translate-x-[calc(100%-2px)] dark:data-checked:bg-primary-foreground group-data-[size=default]/switch:data-unchecked:translate-x-0 group-data-[size=sm]/switch:data-unchecked:translate-x-0 dark:data-unchecked:bg-foreground"
160
+ }
161
+ )
162
+ }
163
+ );
164
+ }
165
+ function SourceDot({
166
+ source
167
+ }) {
168
+ const color = source === "builtin" ? "bg-muted-foreground" : source === "hub" ? "bg-blue-400" : "bg-emerald-400";
169
+ return /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: cn("size-2 rounded-full shrink-0", color) });
170
+ }
171
+ function SkillsPage() {
172
+ const {
173
+ state
174
+ } = useSidebar();
175
+ const isCollapsed = state === "collapsed";
176
+ const [data, setData] = reactExports.useState(null);
177
+ const [loading, setLoading] = reactExports.useState(true);
178
+ const [search, setSearch] = reactExports.useState("");
179
+ const [sourceFilter, setSourceFilter] = reactExports.useState("all");
180
+ const [collapsed, setCollapsed] = reactExports.useState(/* @__PURE__ */ new Set());
181
+ const [archiveCollapsed, setArchiveCollapsed] = reactExports.useState(true);
182
+ const [selected, setSelected] = reactExports.useState(null);
183
+ const [skillContent, setSkillContent] = reactExports.useState("");
184
+ const [skillFiles, setSkillFiles] = reactExports.useState([]);
185
+ const [skillLoading, setSkillLoading] = reactExports.useState(false);
186
+ const [viewingFile, setViewingFile] = reactExports.useState(null);
187
+ const [fileContent, setFileContent] = reactExports.useState("");
188
+ const [, setFileLoading] = reactExports.useState(false);
189
+ const load = reactExports.useCallback(async () => {
190
+ try {
191
+ const d = await fetchSkills();
192
+ setData(d);
193
+ } catch (err) {
194
+ console.error("Failed to load skills:", err);
195
+ }
196
+ }, []);
197
+ reactExports.useEffect(() => {
198
+ setLoading(true);
199
+ load().finally(() => setLoading(false));
200
+ }, [load]);
201
+ reactExports.useEffect(() => {
202
+ if (!selected) {
203
+ setSkillContent("");
204
+ setSkillFiles([]);
205
+ setViewingFile(null);
206
+ setFileContent("");
207
+ return;
208
+ }
209
+ setSkillLoading(true);
210
+ const skillPath = `${selected.category}/${selected.skill}/SKILL.md`;
211
+ Promise.all([fetchSkillContent(skillPath), fetchSkillFiles(selected.category, selected.skill)]).then(([content, files]) => {
212
+ setSkillContent(content);
213
+ setSkillFiles(files.filter((f) => !f.isDir));
214
+ }).catch((err) => {
215
+ setSkillContent(`Failed to load skill: ${err.message}`);
216
+ }).finally(() => setSkillLoading(false));
217
+ }, [selected]);
218
+ const filteredCategories = reactExports.useMemo(() => {
219
+ if (!data) return [];
220
+ let result = data.categories;
221
+ if (sourceFilter !== "all") {
222
+ result = result.map((cat) => ({
223
+ ...cat,
224
+ skills: cat.skills.filter((s) => (s.source || "local") === sourceFilter)
225
+ })).filter((cat) => cat.skills.length > 0);
226
+ }
227
+ if (search.trim()) {
228
+ const q = search.toLowerCase();
229
+ result = result.map((cat) => ({
230
+ ...cat,
231
+ skills: cat.skills.filter((s) => s.name.toLowerCase().includes(q) || s.description.toLowerCase().includes(q))
232
+ })).filter((cat) => cat.skills.length > 0 || cat.name.toLowerCase().includes(q));
233
+ }
234
+ return result;
235
+ }, [data, sourceFilter, search]);
236
+ const filteredArchived = reactExports.useMemo(() => {
237
+ if (!data) return [];
238
+ let result = data.archived;
239
+ if (sourceFilter !== "all") {
240
+ result = result.filter((s) => (s.source || "local") === sourceFilter);
241
+ }
242
+ if (search.trim()) {
243
+ const q = search.toLowerCase();
244
+ result = result.filter((s) => s.name.toLowerCase().includes(q) || s.description.toLowerCase().includes(q));
245
+ }
246
+ return result;
247
+ }, [data, sourceFilter, search]);
248
+ const toggleCategory = (name) => {
249
+ setCollapsed((prev) => {
250
+ const next = new Set(prev);
251
+ if (next.has(name)) next.delete(name);
252
+ else next.add(name);
253
+ return next;
254
+ });
255
+ };
256
+ const handleToggle = async (cat, name, enabled) => {
257
+ try {
258
+ await toggleSkill(name, enabled);
259
+ setData((prev) => {
260
+ if (!prev) return prev;
261
+ return {
262
+ ...prev,
263
+ categories: prev.categories.map((c) => c.name === cat ? {
264
+ ...c,
265
+ skills: c.skills.map((s) => s.name === name ? {
266
+ ...s,
267
+ enabled
268
+ } : s)
269
+ } : c)
270
+ };
271
+ });
272
+ } catch (err) {
273
+ console.error("Toggle failed:", err);
274
+ }
275
+ };
276
+ const viewFile = async (filePath) => {
277
+ if (!selected) return;
278
+ setFileLoading(true);
279
+ setViewingFile(filePath);
280
+ try {
281
+ const base = `${selected.category}/${selected.skill}/`;
282
+ const content = await fetchSkillContent(`${base}${filePath}`);
283
+ setFileContent(content);
284
+ } catch (err) {
285
+ setFileContent(`Failed to load file: ${err.message}`);
286
+ } finally {
287
+ setFileLoading(false);
288
+ }
289
+ };
290
+ const backToSkill = () => {
291
+ setViewingFile(null);
292
+ setFileContent("");
293
+ };
294
+ const sourceOptions = [{
295
+ label: "All",
296
+ value: "all"
297
+ }, {
298
+ label: "Builtin",
299
+ value: "builtin"
300
+ }, {
301
+ label: "Hub",
302
+ value: "hub"
303
+ }, {
304
+ label: "Local",
305
+ value: "local"
306
+ }];
307
+ if (selected) {
308
+ return /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col h-full", children: [
309
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("header", { className: "flex items-center justify-between border-b px-4 py-3 shrink-0 relative", children: [
310
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex items-center gap-2 z-10", children: [
311
+ isCollapsed && /* @__PURE__ */ jsxRuntimeExports.jsx(SidebarTrigger, { className: cn("mr-1 -ml-1 text-muted-foreground hover:text-foreground hover:bg-muted transition-colors shrink-0") }),
312
+ /* @__PURE__ */ jsxRuntimeExports.jsxs(Button, { variant: "ghost", size: "sm", className: "h-7 px-2 text-xs text-muted-foreground hover:text-foreground", onClick: () => setSelected(null), children: [
313
+ /* @__PURE__ */ jsxRuntimeExports.jsx(IconArrowLeft, { className: "size-3.5 mr-1" }),
314
+ "Back"
315
+ ] })
316
+ ] }),
317
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "absolute left-1/2 -translate-x-1/2 flex items-center gap-2 max-w-[50%]", children: [
318
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-muted-foreground text-sm truncate", children: selected.category }),
319
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-muted-foreground shrink-0", children: "/" }),
320
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-sm font-medium truncate", children: selected.skill })
321
+ ] }),
322
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "z-10", children: /* @__PURE__ */ jsxRuntimeExports.jsxs(Button, { variant: "ghost", size: "sm", className: "h-7 px-2 text-xs text-muted-foreground hover:text-foreground", onClick: () => {
323
+ setSkillLoading(true);
324
+ const skillPath = `${selected.category}/${selected.skill}/SKILL.md`;
325
+ Promise.all([fetchSkillContent(skillPath), fetchSkillFiles(selected.category, selected.skill)]).then(([content, files]) => {
326
+ setSkillContent(content);
327
+ setSkillFiles(files.filter((f) => !f.isDir));
328
+ }).finally(() => setSkillLoading(false));
329
+ }, disabled: skillLoading, children: [
330
+ skillLoading ? /* @__PURE__ */ jsxRuntimeExports.jsx(IconLoader2, { className: "size-3 animate-spin" }) : /* @__PURE__ */ jsxRuntimeExports.jsx(IconRefresh, { className: "size-3" }),
331
+ "Refresh"
332
+ ] }) })
333
+ ] }),
334
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex-1 overflow-y-auto p-4", children: /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "max-w-3xl mx-auto flex flex-col gap-4", children: viewingFile ? /* @__PURE__ */ jsxRuntimeExports.jsxs(jsxRuntimeExports.Fragment, { children: [
335
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("button", { onClick: backToSkill, className: "inline-flex items-center gap-1.5 text-xs text-primary hover:underline w-fit", children: [
336
+ /* @__PURE__ */ jsxRuntimeExports.jsx(IconArrowLeft, { className: "size-3" }),
337
+ "Back to ",
338
+ selected.skill
339
+ ] }),
340
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "text-sm text-muted-foreground mb-1", children: viewingFile }),
341
+ /* @__PURE__ */ jsxRuntimeExports.jsx(MarkdownMessage, { content: fileContent })
342
+ ] }) : /* @__PURE__ */ jsxRuntimeExports.jsxs(jsxRuntimeExports.Fragment, { children: [
343
+ /* @__PURE__ */ jsxRuntimeExports.jsx(MarkdownMessage, { content: skillContent }),
344
+ skillFiles.length > 0 && /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "border-t pt-4 mt-2", children: [
345
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "text-xs font-medium text-muted-foreground uppercase tracking-wider mb-2", children: "Attached Files" }),
346
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-wrap gap-2", children: skillFiles.map((f) => /* @__PURE__ */ jsxRuntimeExports.jsxs("button", { onClick: () => viewFile(f.path), className: "inline-flex items-center gap-1.5 rounded-md border bg-muted/50 px-2.5 py-1.5 text-xs text-foreground hover:bg-muted transition-colors", children: [
347
+ /* @__PURE__ */ jsxRuntimeExports.jsx(IconFile, { className: "size-3.5 text-muted-foreground shrink-0" }),
348
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "truncate max-w-[180px]", children: f.path })
349
+ ] }, f.path)) })
350
+ ] })
351
+ ] }) }) })
352
+ ] });
353
+ }
354
+ return /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex h-full", children: [
355
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "w-80 border-r flex flex-col h-full shrink-0", children: [
356
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("header", { className: "flex items-center gap-2 border-b px-3 py-3 shrink-0", children: [
357
+ isCollapsed && /* @__PURE__ */ jsxRuntimeExports.jsx(SidebarTrigger, { className: cn("mr-1 -ml-1 text-muted-foreground hover:text-foreground hover:bg-muted transition-colors shrink-0") }),
358
+ /* @__PURE__ */ jsxRuntimeExports.jsx(IconWand, { className: "size-5 text-muted-foreground" }),
359
+ /* @__PURE__ */ jsxRuntimeExports.jsx("h2", { className: "text-sm font-medium", children: "Skills" }),
360
+ /* @__PURE__ */ jsxRuntimeExports.jsx(Button, { variant: "ghost", size: "sm", className: "ml-auto h-7 px-2 text-xs text-muted-foreground hover:text-foreground", onClick: () => {
361
+ setLoading(true);
362
+ load().finally(() => setLoading(false));
363
+ }, disabled: loading, children: loading ? /* @__PURE__ */ jsxRuntimeExports.jsx(IconLoader2, { className: "size-3 animate-spin" }) : /* @__PURE__ */ jsxRuntimeExports.jsx(IconRefresh, { className: "size-3" }) })
364
+ ] }),
365
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "px-3 py-2 border-b flex flex-col gap-2 shrink-0", children: [
366
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "relative", children: [
367
+ /* @__PURE__ */ jsxRuntimeExports.jsx(IconSearch, { className: "absolute left-2 top-1/2 -translate-y-1/2 size-3.5 text-muted-foreground" }),
368
+ /* @__PURE__ */ jsxRuntimeExports.jsx(Input, { value: search, onChange: (e) => setSearch(e.target.value), placeholder: "Search skills...", className: "h-8 pl-7 text-xs" })
369
+ ] }),
370
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex items-center gap-1", children: sourceOptions.map((opt) => /* @__PURE__ */ jsxRuntimeExports.jsx("button", { onClick: () => setSourceFilter(opt.value), className: cn("px-2 py-0.5 text-[11px] rounded-sm transition-colors", sourceFilter === opt.value ? "bg-primary text-primary-foreground font-medium" : "text-muted-foreground hover:text-foreground hover:bg-muted"), children: opt.label }, opt.value)) })
371
+ ] }),
372
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex-1 overflow-y-auto p-2", children: loading ? /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex items-center justify-center py-8", children: /* @__PURE__ */ jsxRuntimeExports.jsx(IconLoader2, { className: "size-4 animate-spin text-muted-foreground" }) }) : !data || filteredCategories.length === 0 ? /* @__PURE__ */ jsxRuntimeExports.jsx("p", { className: "text-xs text-muted-foreground text-center py-8", children: search ? "No skills match your search." : "No skills found." }) : /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col gap-1", children: [
373
+ filteredCategories.map((cat) => /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col", children: [
374
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("button", { onClick: () => toggleCategory(cat.name), className: "flex items-center gap-1.5 px-2 py-1.5 text-xs font-semibold text-muted-foreground uppercase tracking-wider rounded-md hover:bg-muted/50 transition-colors", children: [
375
+ collapsed.has(cat.name) ? /* @__PURE__ */ jsxRuntimeExports.jsx(IconChevronRight, { className: "size-3 shrink-0" }) : /* @__PURE__ */ jsxRuntimeExports.jsx(IconChevronDown, { className: "size-3 shrink-0" }),
376
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "truncate", children: cat.name }),
377
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "ml-auto text-[10px] bg-muted px-1.5 py-0.5 rounded-full", children: cat.skills.length })
378
+ ] }),
379
+ !collapsed.has(cat.name) && /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-col", children: cat.skills.map((skill) => /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "group flex items-center gap-2 px-2 py-1.5 pl-7 rounded-md hover:bg-muted/50 cursor-pointer transition-colors", onClick: () => setSelected({
380
+ category: cat.name,
381
+ skill: skill.name
382
+ }), children: [
383
+ /* @__PURE__ */ jsxRuntimeExports.jsx(SourceDot, { source: skill.source }),
384
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col min-w-0 flex-1", children: [
385
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex items-center gap-1", children: [
386
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-xs font-medium truncate", children: skill.name }),
387
+ skill.modified && /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-[10px] text-amber-500", title: "Modified", children: "✎" }),
388
+ skill.pinned && /* @__PURE__ */ jsxRuntimeExports.jsx(IconPin, { className: "size-3 text-amber-500 shrink-0" })
389
+ ] }),
390
+ skill.description && /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-[10px] text-muted-foreground truncate", children: skill.description })
391
+ ] }),
392
+ /* @__PURE__ */ jsxRuntimeExports.jsx(Switch, { checked: skill.enabled !== false, onCheckedChange: (v) => handleToggle(cat.name, skill.name, v), className: "scale-75 shrink-0", onClick: (e) => e.stopPropagation() })
393
+ ] }, skill.name)) })
394
+ ] }, cat.name)),
395
+ (filteredArchived.length > 0 || (data?.archived.length ?? 0) > 0) && /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col mt-2 pt-2 border-t", children: [
396
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("button", { onClick: () => setArchiveCollapsed(!archiveCollapsed), className: "flex items-center gap-1.5 px-2 py-1.5 text-xs font-semibold text-muted-foreground uppercase tracking-wider rounded-md hover:bg-muted/50 transition-colors", children: [
397
+ archiveCollapsed ? /* @__PURE__ */ jsxRuntimeExports.jsx(IconChevronRight, { className: "size-3 shrink-0" }) : /* @__PURE__ */ jsxRuntimeExports.jsx(IconChevronDown, { className: "size-3 shrink-0" }),
398
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { children: "Archived" }),
399
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "ml-auto text-[10px] bg-muted px-1.5 py-0.5 rounded-full", children: data?.archived.length ?? 0 })
400
+ ] }),
401
+ !archiveCollapsed && filteredArchived.map((skill) => /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex items-center gap-2 px-2 py-1.5 pl-7 rounded-md hover:bg-muted/50 cursor-pointer transition-colors opacity-60", onClick: () => setSelected({
402
+ category: ".archive",
403
+ skill: skill.name
404
+ }), children: [
405
+ /* @__PURE__ */ jsxRuntimeExports.jsx(SourceDot, { source: skill.source }),
406
+ /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col min-w-0 flex-1", children: [
407
+ /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-xs font-medium truncate", children: skill.name }),
408
+ skill.description && /* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-[10px] text-muted-foreground truncate", children: skill.description })
409
+ ] })
410
+ ] }, skill.name))
411
+ ] })
412
+ ] }) })
413
+ ] }),
414
+ /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex-1 flex flex-col items-center justify-center", children: /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col items-center gap-3 text-muted-foreground", children: [
415
+ /* @__PURE__ */ jsxRuntimeExports.jsx(IconWand, { className: "size-12 opacity-20" }),
416
+ /* @__PURE__ */ jsxRuntimeExports.jsx("p", { className: "text-sm", children: "Select a skill to view its details" })
417
+ ] }) })
418
+ ] });
419
+ }
420
+ export {
421
+ SkillsPage as component
422
+ };
@@ -0,0 +1,42 @@
1
+ const THEME_KEY = "hermium-theme";
2
+ function getStoredTheme() {
3
+ try {
4
+ const raw = localStorage.getItem(THEME_KEY);
5
+ if (raw === "dark" || raw === "light" || raw === "system") return raw;
6
+ } catch {
7
+ }
8
+ return "system";
9
+ }
10
+ function storeTheme(theme) {
11
+ localStorage.setItem(THEME_KEY, theme);
12
+ }
13
+ function resolveTheme(theme) {
14
+ if (theme === "system") {
15
+ return window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light";
16
+ }
17
+ return theme;
18
+ }
19
+ function applyTheme(theme) {
20
+ const resolved = resolveTheme(theme);
21
+ const root = document.documentElement;
22
+ if (resolved === "dark") {
23
+ root.classList.add("dark");
24
+ } else {
25
+ root.classList.remove("dark");
26
+ }
27
+ }
28
+ function toggleTheme() {
29
+ const current = getStoredTheme();
30
+ const resolved = resolveTheme(current);
31
+ const next = resolved === "dark" ? "light" : "dark";
32
+ storeTheme(next);
33
+ applyTheme(next);
34
+ return next;
35
+ }
36
+ export {
37
+ applyTheme,
38
+ getStoredTheme,
39
+ resolveTheme,
40
+ storeTheme,
41
+ toggleTheme
42
+ };