hermium 0.1.2 → 0.1.4
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/bin/hermium.mjs +184 -145
- package/dist/server/index.mjs +65 -65
- package/dist/web-server/__23tanstack-start-plugin-adapters-Cwee5PKy.mjs +6 -0
- package/dist/web-server/_chunks/ssr-renderer.mjs +22 -0
- package/dist/web-server/_libs/babel__runtime.mjs +237 -0
- package/dist/web-server/_libs/bail.mjs +8 -0
- package/dist/web-server/_libs/base-ui__react.mjs +9554 -0
- package/dist/web-server/_libs/base-ui__utils.mjs +1101 -0
- package/dist/web-server/_libs/ccount.mjs +16 -0
- package/dist/web-server/_libs/character-entities-legacy.mjs +111 -0
- package/dist/web-server/_libs/character-entities.mjs +2130 -0
- package/dist/web-server/_libs/character-reference-invalid.mjs +33 -0
- package/dist/web-server/_libs/class-variance-authority.mjs +44 -0
- package/dist/web-server/_libs/clsx.mjs +16 -0
- package/dist/web-server/_libs/comma-separated-tokens.mjs +31 -0
- package/dist/web-server/_libs/cookie-es.mjs +44 -0
- package/dist/web-server/_libs/croner.mjs +1 -0
- package/dist/web-server/_libs/crossws.mjs +1 -0
- package/dist/web-server/_libs/decode-named-character-reference+[...].mjs +8 -0
- package/dist/web-server/_libs/devlop.mjs +8 -0
- package/dist/web-server/_libs/escape-string-regexp.mjs +9 -0
- package/dist/web-server/_libs/estree-util-is-identifier-name.mjs +11 -0
- package/dist/web-server/_libs/extend.mjs +97 -0
- package/dist/web-server/_libs/fault.mjs +1 -0
- package/dist/web-server/_libs/floating-ui__core.mjs +663 -0
- package/dist/web-server/_libs/floating-ui__dom.mjs +624 -0
- package/dist/web-server/_libs/floating-ui__react-dom.mjs +279 -0
- package/dist/web-server/_libs/floating-ui__utils.mjs +322 -0
- package/dist/web-server/_libs/format.mjs +1 -0
- package/dist/web-server/_libs/h3.mjs +408 -0
- package/dist/web-server/_libs/hast-util-parse-selector.mjs +39 -0
- package/dist/web-server/_libs/hast-util-to-jsx-runtime.mjs +388 -0
- package/dist/web-server/_libs/hast-util-whitespace.mjs +10 -0
- package/dist/web-server/_libs/hastscript.mjs +200 -0
- package/dist/web-server/_libs/highlight.js.mjs +1 -0
- package/dist/web-server/_libs/hookable.mjs +1 -0
- package/dist/web-server/_libs/html-url-attributes.mjs +26 -0
- package/dist/web-server/_libs/inline-style-parser.mjs +142 -0
- package/dist/web-server/_libs/is-alphabetical.mjs +7 -0
- package/dist/web-server/_libs/is-alphanumerical.mjs +8 -0
- package/dist/web-server/_libs/is-decimal.mjs +7 -0
- package/dist/web-server/_libs/is-hexadecimal.mjs +7 -0
- package/dist/web-server/_libs/is-plain-obj.mjs +10 -0
- package/dist/web-server/_libs/isbot.mjs +21 -0
- package/dist/web-server/_libs/longest-streak.mjs +25 -0
- package/dist/web-server/_libs/lowlight.mjs +1 -0
- package/dist/web-server/_libs/markdown-table.mjs +142 -0
- package/dist/web-server/_libs/mdast-util-find-and-replace.mjs +109 -0
- package/dist/web-server/_libs/mdast-util-from-markdown.mjs +717 -0
- package/dist/web-server/_libs/mdast-util-gfm-autolink-literal+[...].mjs +156 -0
- package/dist/web-server/_libs/mdast-util-gfm-footnote.mjs +117 -0
- package/dist/web-server/_libs/mdast-util-gfm-strikethrough.mjs +54 -0
- package/dist/web-server/_libs/mdast-util-gfm-table.mjs +157 -0
- package/dist/web-server/_libs/mdast-util-gfm-task-list-item.mjs +77 -0
- package/dist/web-server/_libs/mdast-util-gfm.mjs +29 -0
- package/dist/web-server/_libs/mdast-util-phrasing.mjs +30 -0
- package/dist/web-server/_libs/mdast-util-to-hast.mjs +710 -0
- package/dist/web-server/_libs/mdast-util-to-markdown.mjs +798 -0
- package/dist/web-server/_libs/mdast-util-to-string.mjs +38 -0
- package/dist/web-server/_libs/micromark-core-commonmark.mjs +2259 -0
- package/dist/web-server/_libs/micromark-extension-gfm-autolink-literal+[...].mjs +344 -0
- package/dist/web-server/_libs/micromark-extension-gfm-footnote+[...].mjs +279 -0
- package/dist/web-server/_libs/micromark-extension-gfm-strikethrough+[...].mjs +98 -0
- package/dist/web-server/_libs/micromark-extension-gfm-table.mjs +491 -0
- package/dist/web-server/_libs/micromark-extension-gfm-tagfilter+[...].mjs +1 -0
- package/dist/web-server/_libs/micromark-extension-gfm-task-list-item+[...].mjs +77 -0
- package/dist/web-server/_libs/micromark-extension-gfm.mjs +18 -0
- package/dist/web-server/_libs/micromark-factory-destination.mjs +94 -0
- package/dist/web-server/_libs/micromark-factory-label.mjs +63 -0
- package/dist/web-server/_libs/micromark-factory-space.mjs +24 -0
- package/dist/web-server/_libs/micromark-factory-title.mjs +65 -0
- package/dist/web-server/_libs/micromark-factory-whitespace.mjs +22 -0
- package/dist/web-server/_libs/micromark-util-character.mjs +44 -0
- package/dist/web-server/_libs/micromark-util-chunked.mjs +36 -0
- package/dist/web-server/_libs/micromark-util-classify-character+[...].mjs +12 -0
- package/dist/web-server/_libs/micromark-util-combine-extensions+[...].mjs +41 -0
- package/dist/web-server/_libs/micromark-util-decode-numeric-character-reference+[...].mjs +19 -0
- package/dist/web-server/_libs/micromark-util-decode-string.mjs +21 -0
- package/dist/web-server/_libs/micromark-util-encode.mjs +1 -0
- package/dist/web-server/_libs/micromark-util-html-tag-name.mjs +69 -0
- package/dist/web-server/_libs/micromark-util-normalize-identifier+[...].mjs +6 -0
- package/dist/web-server/_libs/micromark-util-resolve-all.mjs +15 -0
- package/dist/web-server/_libs/micromark-util-sanitize-uri.mjs +41 -0
- package/dist/web-server/_libs/micromark-util-subtokenize.mjs +346 -0
- package/dist/web-server/_libs/micromark.mjs +906 -0
- package/dist/web-server/_libs/ocache.mjs +1 -0
- package/dist/web-server/_libs/ohash.mjs +1 -0
- package/dist/web-server/_libs/parse-entities.mjs +245 -0
- package/dist/web-server/_libs/property-information.mjs +1210 -0
- package/dist/web-server/_libs/react-dom.mjs +10779 -0
- package/dist/web-server/_libs/react-markdown.mjs +147 -0
- package/dist/web-server/_libs/react-syntax-highlighter.mjs +941 -0
- package/dist/web-server/_libs/react.mjs +513 -0
- package/dist/web-server/_libs/refractor.mjs +2425 -0
- package/dist/web-server/_libs/remark-gfm.mjs +20 -0
- package/dist/web-server/_libs/remark-parse.mjs +19 -0
- package/dist/web-server/_libs/remark-rehype.mjs +21 -0
- package/dist/web-server/_libs/reselect.mjs +1 -0
- package/dist/web-server/_libs/rou3.mjs +8 -0
- package/dist/web-server/_libs/seroval-plugins.mjs +58 -0
- package/dist/web-server/_libs/seroval.mjs +1775 -0
- package/dist/web-server/_libs/space-separated-tokens.mjs +11 -0
- package/dist/web-server/_libs/srvx.mjs +781 -0
- package/dist/web-server/_libs/style-to-js.mjs +72 -0
- package/dist/web-server/_libs/style-to-object.mjs +38 -0
- package/dist/web-server/_libs/tabler__icons-react.mjs +224 -0
- package/dist/web-server/_libs/tanstack__history.mjs +204 -0
- package/dist/web-server/_libs/tanstack__query-core.mjs +2552 -0
- package/dist/web-server/_libs/tanstack__react-query.mjs +190 -0
- package/dist/web-server/_libs/tanstack__react-router.mjs +1120 -0
- package/dist/web-server/_libs/tanstack__react-store.mjs +2 -0
- package/dist/web-server/_libs/tanstack__router-core.mjs +4288 -0
- package/dist/web-server/_libs/tanstack__store.mjs +1 -0
- package/dist/web-server/_libs/trim-lines.mjs +41 -0
- package/dist/web-server/_libs/trough.mjs +85 -0
- package/dist/web-server/_libs/ufo.mjs +54 -0
- package/dist/web-server/_libs/unctx.mjs +1 -0
- package/dist/web-server/_libs/ungap__structured-clone.mjs +224 -0
- package/dist/web-server/_libs/unified.mjs +661 -0
- package/dist/web-server/_libs/unist-util-is.mjs +100 -0
- package/dist/web-server/_libs/unist-util-position.mjs +27 -0
- package/dist/web-server/_libs/unist-util-stringify-position.mjs +27 -0
- package/dist/web-server/_libs/unist-util-visit-parents.mjs +83 -0
- package/dist/web-server/_libs/unist-util-visit.mjs +24 -0
- package/dist/web-server/_libs/unstorage.mjs +1 -0
- package/dist/web-server/_libs/use-sync-external-store.mjs +139 -0
- package/dist/web-server/_libs/vfile-message.mjs +138 -0
- package/dist/web-server/_libs/vfile.mjs +467 -0
- package/dist/web-server/_libs/zod.mjs +3915 -0
- package/dist/web-server/_libs/zustand.mjs +343 -0
- package/dist/web-server/_libs/zwitch.mjs +1 -0
- package/dist/web-server/_ssr/index-BLK6uN4p.mjs +612 -0
- package/dist/web-server/_ssr/index-BkkxTg0a.mjs +1855 -0
- package/dist/web-server/_ssr/index-Bp9a_nTf.mjs +66 -0
- package/dist/web-server/_ssr/index-C8t8AZQG.mjs +513 -0
- package/dist/web-server/_ssr/index-DSIu0x-q.mjs +449 -0
- package/dist/web-server/_ssr/index-DqFrn6kj.mjs +278 -0
- package/dist/web-server/_ssr/index-EKE8NFy_.mjs +189 -0
- package/dist/web-server/_ssr/index-JzLhPyir.mjs +213 -0
- package/dist/web-server/_ssr/index-wTy_4MhH.mjs +369 -0
- package/dist/web-server/_ssr/index.mjs +1558 -0
- package/dist/web-server/_ssr/input-BQFduUUo.mjs +20 -0
- package/dist/web-server/_ssr/router-59cN5lqo.mjs +1998 -0
- package/dist/web-server/_ssr/start-HYkvq4Ni.mjs +4 -0
- package/dist/web-server/_ssr/switch-Bim4kX8N.mjs +33 -0
- package/dist/web-server/_ssr/syntax-highlighter-5vezNTce.mjs +62 -0
- package/dist/web-server/_ssr/textarea-CK0ROhfF.mjs +18 -0
- package/dist/web-server/_tanstack-start-manifest_v-DLw6M7p4.mjs +4 -0
- package/dist/web-server/index.mjs +611 -0
- package/package.json +1 -1
|
@@ -0,0 +1,449 @@
|
|
|
1
|
+
import { r as reactExports, j as jsxRuntimeExports } from "../_libs/react.mjs";
|
|
2
|
+
import { a as cn, g as get, p as post } from "./router-59cN5lqo.mjs";
|
|
3
|
+
import { S as Switch } from "./switch-Bim4kX8N.mjs";
|
|
4
|
+
import { G as IconLoader2, H as IconAlertCircle, F as IconRefresh, p as IconPuzzle, I as IconSearch, a as IconX, m as IconChevronDown, J as IconVersions, K as IconTag, L as IconLink } from "../_libs/tabler__icons-react.mjs";
|
|
5
|
+
import { M as Markdown } from "../_libs/react-markdown.mjs";
|
|
6
|
+
import { r as remarkGfm } from "../_libs/remark-gfm.mjs";
|
|
7
|
+
import "../_libs/tanstack__react-router.mjs";
|
|
8
|
+
import "../_libs/tanstack__router-core.mjs";
|
|
9
|
+
import "../_libs/tanstack__history.mjs";
|
|
10
|
+
import "../_libs/cookie-es.mjs";
|
|
11
|
+
import "../_libs/seroval.mjs";
|
|
12
|
+
import "../_libs/seroval-plugins.mjs";
|
|
13
|
+
import "node:stream/web";
|
|
14
|
+
import "node:stream";
|
|
15
|
+
import "../_libs/react-dom.mjs";
|
|
16
|
+
import "util";
|
|
17
|
+
import "crypto";
|
|
18
|
+
import "async_hooks";
|
|
19
|
+
import "stream";
|
|
20
|
+
import "../_libs/isbot.mjs";
|
|
21
|
+
import "../_libs/tanstack__query-core.mjs";
|
|
22
|
+
import "../_libs/tanstack__react-query.mjs";
|
|
23
|
+
import "../_libs/clsx.mjs";
|
|
24
|
+
import "../_libs/class-variance-authority.mjs";
|
|
25
|
+
import "../_libs/zustand.mjs";
|
|
26
|
+
import "../_libs/base-ui__react.mjs";
|
|
27
|
+
import "../_libs/base-ui__utils.mjs";
|
|
28
|
+
import "../_libs/use-sync-external-store.mjs";
|
|
29
|
+
import "../_libs/floating-ui__utils.mjs";
|
|
30
|
+
import "../_libs/floating-ui__react-dom.mjs";
|
|
31
|
+
import "../_libs/floating-ui__dom.mjs";
|
|
32
|
+
import "../_libs/floating-ui__core.mjs";
|
|
33
|
+
import "../_libs/zod.mjs";
|
|
34
|
+
import "../_libs/devlop.mjs";
|
|
35
|
+
import "../_libs/unified.mjs";
|
|
36
|
+
import "../_libs/bail.mjs";
|
|
37
|
+
import "../_libs/extend.mjs";
|
|
38
|
+
import "../_libs/is-plain-obj.mjs";
|
|
39
|
+
import "../_libs/trough.mjs";
|
|
40
|
+
import "../_libs/vfile.mjs";
|
|
41
|
+
import "../_libs/vfile-message.mjs";
|
|
42
|
+
import "../_libs/unist-util-stringify-position.mjs";
|
|
43
|
+
import "node:process";
|
|
44
|
+
import "node:path";
|
|
45
|
+
import "node:url";
|
|
46
|
+
import "../_libs/remark-parse.mjs";
|
|
47
|
+
import "../_libs/mdast-util-from-markdown.mjs";
|
|
48
|
+
import "../_libs/micromark-util-decode-numeric-character-reference+[...].mjs";
|
|
49
|
+
import "../_libs/micromark-util-decode-string.mjs";
|
|
50
|
+
import "../_libs/decode-named-character-reference+[...].mjs";
|
|
51
|
+
import "../_libs/character-entities.mjs";
|
|
52
|
+
import "../_libs/micromark-util-normalize-identifier+[...].mjs";
|
|
53
|
+
import "../_libs/micromark.mjs";
|
|
54
|
+
import "../_libs/micromark-util-combine-extensions+[...].mjs";
|
|
55
|
+
import "../_libs/micromark-util-chunked.mjs";
|
|
56
|
+
import "../_libs/micromark-factory-space.mjs";
|
|
57
|
+
import "../_libs/micromark-util-character.mjs";
|
|
58
|
+
import "../_libs/micromark-core-commonmark.mjs";
|
|
59
|
+
import "../_libs/micromark-util-classify-character+[...].mjs";
|
|
60
|
+
import "../_libs/micromark-util-resolve-all.mjs";
|
|
61
|
+
import "../_libs/micromark-util-subtokenize.mjs";
|
|
62
|
+
import "../_libs/micromark-factory-destination.mjs";
|
|
63
|
+
import "../_libs/micromark-factory-label.mjs";
|
|
64
|
+
import "../_libs/micromark-factory-title.mjs";
|
|
65
|
+
import "../_libs/micromark-factory-whitespace.mjs";
|
|
66
|
+
import "../_libs/micromark-util-html-tag-name.mjs";
|
|
67
|
+
import "../_libs/mdast-util-to-string.mjs";
|
|
68
|
+
import "../_libs/remark-rehype.mjs";
|
|
69
|
+
import "../_libs/mdast-util-to-hast.mjs";
|
|
70
|
+
import "../_libs/ungap__structured-clone.mjs";
|
|
71
|
+
import "../_libs/micromark-util-sanitize-uri.mjs";
|
|
72
|
+
import "../_libs/unist-util-position.mjs";
|
|
73
|
+
import "../_libs/trim-lines.mjs";
|
|
74
|
+
import "../_libs/unist-util-visit.mjs";
|
|
75
|
+
import "../_libs/unist-util-visit-parents.mjs";
|
|
76
|
+
import "../_libs/unist-util-is.mjs";
|
|
77
|
+
import "../_libs/hast-util-to-jsx-runtime.mjs";
|
|
78
|
+
import "../_libs/comma-separated-tokens.mjs";
|
|
79
|
+
import "../_libs/property-information.mjs";
|
|
80
|
+
import "../_libs/space-separated-tokens.mjs";
|
|
81
|
+
import "../_libs/style-to-js.mjs";
|
|
82
|
+
import "../_libs/style-to-object.mjs";
|
|
83
|
+
import "../_libs/inline-style-parser.mjs";
|
|
84
|
+
import "../_libs/hast-util-whitespace.mjs";
|
|
85
|
+
import "../_libs/estree-util-is-identifier-name.mjs";
|
|
86
|
+
import "../_libs/html-url-attributes.mjs";
|
|
87
|
+
import "../_libs/micromark-extension-gfm.mjs";
|
|
88
|
+
import "../_libs/micromark-extension-gfm-autolink-literal+[...].mjs";
|
|
89
|
+
import "../_libs/micromark-extension-gfm-footnote+[...].mjs";
|
|
90
|
+
import "../_libs/micromark-extension-gfm-strikethrough+[...].mjs";
|
|
91
|
+
import "../_libs/micromark-extension-gfm-table.mjs";
|
|
92
|
+
import "../_libs/micromark-extension-gfm-task-list-item+[...].mjs";
|
|
93
|
+
import "../_libs/mdast-util-gfm.mjs";
|
|
94
|
+
import "../_libs/mdast-util-gfm-autolink-literal+[...].mjs";
|
|
95
|
+
import "../_libs/ccount.mjs";
|
|
96
|
+
import "../_libs/mdast-util-find-and-replace.mjs";
|
|
97
|
+
import "../_libs/escape-string-regexp.mjs";
|
|
98
|
+
import "../_libs/mdast-util-gfm-footnote.mjs";
|
|
99
|
+
import "../_libs/mdast-util-gfm-strikethrough.mjs";
|
|
100
|
+
import "../_libs/mdast-util-gfm-table.mjs";
|
|
101
|
+
import "../_libs/markdown-table.mjs";
|
|
102
|
+
import "../_libs/mdast-util-to-markdown.mjs";
|
|
103
|
+
import "../_libs/longest-streak.mjs";
|
|
104
|
+
import "../_libs/mdast-util-phrasing.mjs";
|
|
105
|
+
import "../_libs/mdast-util-gfm-task-list-item.mjs";
|
|
106
|
+
async function fetchSkills() {
|
|
107
|
+
return get("/api/hermes/skills");
|
|
108
|
+
}
|
|
109
|
+
async function fetchSkillContent(category, skill) {
|
|
110
|
+
return get(`/api/hermes/skills/${category}/${skill}/SKILL.md`);
|
|
111
|
+
}
|
|
112
|
+
async function toggleSkill(category, skill) {
|
|
113
|
+
return post(`/api/hermes/skills/${category}/${skill}/toggle`);
|
|
114
|
+
}
|
|
115
|
+
const markdownComponents = {
|
|
116
|
+
a({ href, children }) {
|
|
117
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
118
|
+
"a",
|
|
119
|
+
{
|
|
120
|
+
href,
|
|
121
|
+
target: "_blank",
|
|
122
|
+
rel: "noopener noreferrer",
|
|
123
|
+
className: "text-blue-600 dark:text-blue-400 underline underline-offset-2 decoration-blue-400/30 hover:decoration-blue-600/60 transition-colors",
|
|
124
|
+
children
|
|
125
|
+
}
|
|
126
|
+
);
|
|
127
|
+
},
|
|
128
|
+
img({ src, alt }) {
|
|
129
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx("img", { src, alt, className: "rounded-xl max-w-full my-4", loading: "lazy" });
|
|
130
|
+
},
|
|
131
|
+
pre({ children }) {
|
|
132
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx("pre", { className: "my-5 rounded-xl bg-muted/50 border border-border/60 px-5 py-4 text-[13px] leading-relaxed overflow-x-auto font-mono whitespace-pre-wrap", children });
|
|
133
|
+
},
|
|
134
|
+
code({ children, className }) {
|
|
135
|
+
const isInline = !className;
|
|
136
|
+
if (isInline) {
|
|
137
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx("code", { className: "rounded-md bg-muted/70 px-1.5 py-0.5 text-[13px] font-mono text-foreground/90 border border-border/50", children });
|
|
138
|
+
}
|
|
139
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx("code", { className, children });
|
|
140
|
+
}
|
|
141
|
+
};
|
|
142
|
+
function SkillsPage() {
|
|
143
|
+
const [categories, setCategories] = reactExports.useState([]);
|
|
144
|
+
const [total, setTotal] = reactExports.useState(0);
|
|
145
|
+
const [loading, setLoading] = reactExports.useState(true);
|
|
146
|
+
const [error, setError] = reactExports.useState(null);
|
|
147
|
+
const [searchQuery, setSearchQuery] = reactExports.useState("");
|
|
148
|
+
const [selectedCategory, setSelectedCategory] = reactExports.useState(null);
|
|
149
|
+
const [selectedSkill, setSelectedSkill] = reactExports.useState(null);
|
|
150
|
+
const [selectedSkillName, setSelectedSkillName] = reactExports.useState(null);
|
|
151
|
+
const [skillContent, setSkillContent] = reactExports.useState("");
|
|
152
|
+
const [skillLoading, setSkillLoading] = reactExports.useState(false);
|
|
153
|
+
const [collapsedCategories, setCollapsedCategories] = reactExports.useState(/* @__PURE__ */ new Set());
|
|
154
|
+
const load = reactExports.useCallback(async () => {
|
|
155
|
+
setLoading(true);
|
|
156
|
+
setError(null);
|
|
157
|
+
try {
|
|
158
|
+
const result = await fetchSkills();
|
|
159
|
+
setCategories(result.categories);
|
|
160
|
+
setTotal(result.total);
|
|
161
|
+
} catch (e) {
|
|
162
|
+
setError(e instanceof Error ? e.message : "Failed to load skills");
|
|
163
|
+
} finally {
|
|
164
|
+
setLoading(false);
|
|
165
|
+
}
|
|
166
|
+
}, []);
|
|
167
|
+
reactExports.useEffect(() => {
|
|
168
|
+
load();
|
|
169
|
+
}, [load]);
|
|
170
|
+
reactExports.useEffect(() => {
|
|
171
|
+
if (!selectedCategory || !selectedSkill) {
|
|
172
|
+
setSkillContent("");
|
|
173
|
+
return;
|
|
174
|
+
}
|
|
175
|
+
let cancelled = false;
|
|
176
|
+
setSkillLoading(true);
|
|
177
|
+
fetchSkillContent(selectedCategory, selectedSkill).then((res) => {
|
|
178
|
+
if (!cancelled) {
|
|
179
|
+
const content = res.content;
|
|
180
|
+
if (content.startsWith("---")) {
|
|
181
|
+
const endIdx = content.indexOf("\n---", 3);
|
|
182
|
+
if (endIdx !== -1) {
|
|
183
|
+
setSkillContent(content.slice(endIdx + 4).trim());
|
|
184
|
+
} else {
|
|
185
|
+
setSkillContent(content);
|
|
186
|
+
}
|
|
187
|
+
} else {
|
|
188
|
+
setSkillContent(content);
|
|
189
|
+
}
|
|
190
|
+
setSkillLoading(false);
|
|
191
|
+
}
|
|
192
|
+
}).catch((e) => {
|
|
193
|
+
if (!cancelled) {
|
|
194
|
+
setSkillContent(e instanceof Error ? e.message : "Failed to load skill");
|
|
195
|
+
setSkillLoading(false);
|
|
196
|
+
}
|
|
197
|
+
});
|
|
198
|
+
return () => {
|
|
199
|
+
cancelled = true;
|
|
200
|
+
};
|
|
201
|
+
}, [selectedCategory, selectedSkill]);
|
|
202
|
+
const q = searchQuery.toLowerCase().trim();
|
|
203
|
+
const filteredCategories = q ? categories.map((cat) => ({
|
|
204
|
+
...cat,
|
|
205
|
+
skills: (cat.skills ?? []).filter(
|
|
206
|
+
(s) => s.name.toLowerCase().includes(q) || s.description.toLowerCase().includes(q) || (s.tags ?? []).some((t) => t.toLowerCase().includes(q))
|
|
207
|
+
)
|
|
208
|
+
})).filter((cat) => cat.skills.length > 0 || cat.name.toLowerCase().includes(q)) : categories;
|
|
209
|
+
const toggleCategory = (name) => {
|
|
210
|
+
setCollapsedCategories((prev) => {
|
|
211
|
+
const next = new Set(prev);
|
|
212
|
+
if (next.has(name)) next.delete(name);
|
|
213
|
+
else next.add(name);
|
|
214
|
+
return next;
|
|
215
|
+
});
|
|
216
|
+
};
|
|
217
|
+
const selectSkill = (catName, skillName, displayName) => {
|
|
218
|
+
setSelectedCategory(catName);
|
|
219
|
+
setSelectedSkill(skillName);
|
|
220
|
+
setSelectedSkillName(displayName);
|
|
221
|
+
};
|
|
222
|
+
const handleToggle = async (catName, skillName) => {
|
|
223
|
+
setCategories(
|
|
224
|
+
(prev) => prev.map(
|
|
225
|
+
(cat) => cat.name === catName ? {
|
|
226
|
+
...cat,
|
|
227
|
+
skills: cat.skills.map(
|
|
228
|
+
(s) => s.name === skillName ? { ...s, disabled: !s.disabled } : s
|
|
229
|
+
)
|
|
230
|
+
} : cat
|
|
231
|
+
)
|
|
232
|
+
);
|
|
233
|
+
try {
|
|
234
|
+
await toggleSkill(catName, skillName);
|
|
235
|
+
} catch {
|
|
236
|
+
load();
|
|
237
|
+
}
|
|
238
|
+
};
|
|
239
|
+
if (loading) {
|
|
240
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-1 items-center justify-center", children: /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col items-center gap-3 text-muted-foreground", children: [
|
|
241
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconLoader2, { className: "size-6 animate-spin" }),
|
|
242
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-sm", children: "Loading skills..." })
|
|
243
|
+
] }) });
|
|
244
|
+
}
|
|
245
|
+
if (error) {
|
|
246
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-1 items-center justify-center", children: /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col items-center gap-3 text-muted-foreground", children: [
|
|
247
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconAlertCircle, { className: "size-8 text-red-500" }),
|
|
248
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-sm", children: error }),
|
|
249
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs(
|
|
250
|
+
"button",
|
|
251
|
+
{
|
|
252
|
+
onClick: load,
|
|
253
|
+
className: "inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-xs font-medium bg-muted hover:bg-muted/80 transition-colors",
|
|
254
|
+
children: [
|
|
255
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconRefresh, { className: "size-3.5" }),
|
|
256
|
+
"Retry"
|
|
257
|
+
]
|
|
258
|
+
}
|
|
259
|
+
)
|
|
260
|
+
] }) });
|
|
261
|
+
}
|
|
262
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-1 overflow-hidden", children: [
|
|
263
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("aside", { className: "flex w-64 shrink-0 flex-col border-r border-border", children: [
|
|
264
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex items-center gap-2.5 px-3 py-3.5 shrink-0", children: [
|
|
265
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconPuzzle, { className: "size-5 text-muted-foreground" }),
|
|
266
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("span", { className: "text-sm font-semibold text-foreground", children: [
|
|
267
|
+
"Skills",
|
|
268
|
+
total > 0 && /* @__PURE__ */ jsxRuntimeExports.jsxs("span", { className: "ml-1.5 text-[11px] font-normal text-muted-foreground", children: [
|
|
269
|
+
"(",
|
|
270
|
+
total,
|
|
271
|
+
")"
|
|
272
|
+
] })
|
|
273
|
+
] })
|
|
274
|
+
] }),
|
|
275
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "px-3 pb-2 shrink-0", children: /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "relative", children: [
|
|
276
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconSearch, { className: "absolute left-2.5 top-1/2 -translate-y-1/2 size-3.5 text-muted-foreground/50" }),
|
|
277
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
278
|
+
"input",
|
|
279
|
+
{
|
|
280
|
+
value: searchQuery,
|
|
281
|
+
onChange: (e) => setSearchQuery(e.target.value),
|
|
282
|
+
placeholder: "Search skills...",
|
|
283
|
+
className: "w-full rounded-md border border-border/60 bg-muted/30 pl-8 pr-8 py-1.5 text-xs text-foreground placeholder:text-muted-foreground/40 outline-none focus:ring-1 focus:ring-primary/20 focus:border-primary/30"
|
|
284
|
+
}
|
|
285
|
+
),
|
|
286
|
+
searchQuery && /* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
287
|
+
"button",
|
|
288
|
+
{
|
|
289
|
+
onClick: () => setSearchQuery(""),
|
|
290
|
+
className: "absolute right-2 top-1/2 -translate-y-1/2 text-muted-foreground/40 hover:text-muted-foreground",
|
|
291
|
+
children: /* @__PURE__ */ jsxRuntimeExports.jsx(IconX, { className: "size-3" })
|
|
292
|
+
}
|
|
293
|
+
)
|
|
294
|
+
] }) }),
|
|
295
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex-1 overflow-y-auto min-h-0", children: filteredCategories.length === 0 ? /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "px-4 py-8 text-center", children: /* @__PURE__ */ jsxRuntimeExports.jsx("p", { className: "text-xs text-muted-foreground", children: searchQuery ? "No skills match your search" : "No skills found" }) }) : filteredCategories.map((cat) => /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "mb-1", children: [
|
|
296
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs(
|
|
297
|
+
"button",
|
|
298
|
+
{
|
|
299
|
+
onClick: () => toggleCategory(cat.name),
|
|
300
|
+
className: "flex w-full items-center gap-1.5 px-3 py-1.5 text-left hover:bg-muted/40 transition-colors",
|
|
301
|
+
children: [
|
|
302
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
303
|
+
IconChevronDown,
|
|
304
|
+
{
|
|
305
|
+
className: cn(
|
|
306
|
+
"size-3 shrink-0 text-muted-foreground/50 transition-transform",
|
|
307
|
+
collapsedCategories.has(cat.name) && "-rotate-90"
|
|
308
|
+
)
|
|
309
|
+
}
|
|
310
|
+
),
|
|
311
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-[11px] font-semibold uppercase tracking-wide text-muted-foreground truncate", children: cat.name.replace(/-/g, " ") }),
|
|
312
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-[10px] text-muted-foreground/50 ml-auto shrink-0", children: cat.skills.length })
|
|
313
|
+
]
|
|
314
|
+
}
|
|
315
|
+
),
|
|
316
|
+
!collapsedCategories.has(cat.name) && /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "pb-1", children: (cat.skills ?? []).map((skill) => {
|
|
317
|
+
const isActive = selectedCategory === cat.name && selectedSkill === skill.name;
|
|
318
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsxs(
|
|
319
|
+
"button",
|
|
320
|
+
{
|
|
321
|
+
onClick: () => selectSkill(cat.name, skill.name, skill.name),
|
|
322
|
+
className: cn(
|
|
323
|
+
"flex w-full items-start gap-2 px-3 py-1.5 pl-8 text-left transition-colors",
|
|
324
|
+
isActive ? "bg-muted text-foreground" : skill.disabled ? "text-muted-foreground/40 hover:bg-muted/20" : "text-muted-foreground hover:bg-muted/30 hover:text-foreground"
|
|
325
|
+
),
|
|
326
|
+
children: [
|
|
327
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: cn("min-w-0 flex-1", skill.disabled && "opacity-50"), children: [
|
|
328
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: cn("truncate text-xs font-medium", skill.disabled && "line-through"), children: skill.name }),
|
|
329
|
+
skill.description && /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "truncate text-[10px] text-muted-foreground/60 mt-0.5", children: skill.description })
|
|
330
|
+
] }),
|
|
331
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "shrink-0", onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
332
|
+
Switch,
|
|
333
|
+
{
|
|
334
|
+
checked: !skill.disabled,
|
|
335
|
+
onCheckedChange: () => handleToggle(cat.name, skill.name)
|
|
336
|
+
}
|
|
337
|
+
) })
|
|
338
|
+
]
|
|
339
|
+
},
|
|
340
|
+
skill.name
|
|
341
|
+
);
|
|
342
|
+
}) })
|
|
343
|
+
] }, cat.name)) })
|
|
344
|
+
] }),
|
|
345
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-1 flex-col min-w-0 overflow-y-auto", children: !selectedSkill ? (
|
|
346
|
+
/* Empty state */
|
|
347
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-1 items-center justify-center", children: /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-col items-center gap-4 text-muted-foreground max-w-sm text-center", children: [
|
|
348
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex size-16 items-center justify-center rounded-2xl bg-muted/50 border border-border/50", children: /* @__PURE__ */ jsxRuntimeExports.jsx(IconPuzzle, { className: "size-7 text-muted-foreground/25" }) }),
|
|
349
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("div", { children: [
|
|
350
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("p", { className: "text-sm font-medium text-foreground/60", children: "Select a skill to view details" }),
|
|
351
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("p", { className: "text-xs text-muted-foreground/50 mt-1", children: "Browse skills from your Hermes agent library. Each skill provides instructions and context for specific tasks." })
|
|
352
|
+
] })
|
|
353
|
+
] }) })
|
|
354
|
+
) : /* @__PURE__ */ jsxRuntimeExports.jsxs(jsxRuntimeExports.Fragment, { children: [
|
|
355
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "shrink-0 border-b border-border px-6 py-4", children: [
|
|
356
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex items-center gap-3", children: [
|
|
357
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex size-8 items-center justify-center rounded-lg bg-muted/50 border border-border/50", children: /* @__PURE__ */ jsxRuntimeExports.jsx(IconPuzzle, { className: "size-4 text-muted-foreground" }) }),
|
|
358
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "min-w-0", children: [
|
|
359
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("h2", { className: "text-base font-semibold text-foreground truncate", children: selectedSkillName }),
|
|
360
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("p", { className: "text-xs text-muted-foreground/60 mt-0.5 flex items-center gap-1", children: [
|
|
361
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-[10px] uppercase tracking-wider text-muted-foreground/40", children: selectedCategory?.replace(/-/g, " ") }),
|
|
362
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { className: "text-muted-foreground/30", children: "/" }),
|
|
363
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("span", { children: selectedSkillName })
|
|
364
|
+
] })
|
|
365
|
+
] })
|
|
366
|
+
] }),
|
|
367
|
+
(() => {
|
|
368
|
+
const cat = categories.find((c) => c.name === selectedCategory);
|
|
369
|
+
const skill = cat?.skills.find((s) => s.name === selectedSkill);
|
|
370
|
+
if (!skill) return null;
|
|
371
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "flex flex-wrap items-center gap-2 mt-3", children: [
|
|
372
|
+
skill.version && /* @__PURE__ */ jsxRuntimeExports.jsxs("span", { className: "inline-flex items-center gap-1 text-[10px] text-muted-foreground/60 bg-muted/40 rounded-full px-2 py-0.5", children: [
|
|
373
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconVersions, { className: "size-3" }),
|
|
374
|
+
"v",
|
|
375
|
+
skill.version
|
|
376
|
+
] }),
|
|
377
|
+
(skill.tags ?? []).map((tag) => /* @__PURE__ */ jsxRuntimeExports.jsxs(
|
|
378
|
+
"span",
|
|
379
|
+
{
|
|
380
|
+
className: "inline-flex items-center gap-1 text-[10px] text-muted-foreground/60 bg-muted/40 rounded-full px-2 py-0.5",
|
|
381
|
+
children: [
|
|
382
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconTag, { className: "size-3" }),
|
|
383
|
+
tag
|
|
384
|
+
]
|
|
385
|
+
},
|
|
386
|
+
tag
|
|
387
|
+
))
|
|
388
|
+
] });
|
|
389
|
+
})()
|
|
390
|
+
] }),
|
|
391
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex-1 overflow-y-auto min-h-0 px-6 py-6", children: skillLoading ? /* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex items-center justify-center py-16", children: /* @__PURE__ */ jsxRuntimeExports.jsx(IconLoader2, { className: "size-5 animate-spin text-muted-foreground" }) }) : /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "max-w-3xl mx-auto", children: [
|
|
392
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
393
|
+
"div",
|
|
394
|
+
{
|
|
395
|
+
className: cn(
|
|
396
|
+
"prose-sm max-w-none",
|
|
397
|
+
// Headings
|
|
398
|
+
"[&_h1]:mt-8 [&_h1]:mb-4 [&_h1]:text-2xl [&_h1]:font-bold [&_h1]:tracking-tight [&_h1]:text-foreground",
|
|
399
|
+
"[&_h2]:mt-7 [&_h2]:mb-3 [&_h2]:text-xl [&_h2]:font-semibold [&_h2]:tracking-tight [&_h2]:text-foreground",
|
|
400
|
+
"[&_h3]:mt-6 [&_h3]:mb-2.5 [&_h3]:text-lg [&_h3]:font-semibold [&_h3]:text-foreground",
|
|
401
|
+
"[&_h4]:mt-5 [&_h4]:mb-2 [&_h4]:text-base [&_h4]:font-medium [&_h4]:text-foreground",
|
|
402
|
+
// Paragraphs
|
|
403
|
+
"[&_p]:my-4 [&_p]:leading-[1.75] [&_p]:text-foreground/85",
|
|
404
|
+
// Lists
|
|
405
|
+
"[&_ul]:my-4 [&_ul]:pl-6 [&_ul]:list-disc [&_ul]:space-y-1.5 [&_ul]:marker:text-muted-foreground/40",
|
|
406
|
+
"[&_ol]:my-4 [&_ol]:pl-6 [&_ol]:list-decimal [&_ol]:space-y-1.5 [&_ol]:marker:text-muted-foreground/40",
|
|
407
|
+
"[&_li]:leading-relaxed [&_li]:text-foreground/85",
|
|
408
|
+
// Blockquotes
|
|
409
|
+
"[&_blockquote]:my-5 [&_blockquote]:border-l-[3px] [&_blockquote]:border-amber-400 dark:[&_blockquote]:border-amber-500 [&_blockquote]:pl-4 [&_blockquote]:text-muted-foreground [&_blockquote]:italic [&_blockquote]:leading-relaxed",
|
|
410
|
+
// Horizontal rules
|
|
411
|
+
"[&_hr]:my-8 [&_hr]:border-border/40",
|
|
412
|
+
// Tables
|
|
413
|
+
"[&_table]:w-full [&_table]:my-4 [&_table]:text-sm",
|
|
414
|
+
"[&_th]:border [&_th]:border-border [&_th]:bg-muted/60 [&_th]:px-4 [&_th]:py-2.5 [&_th]:text-left [&_th]:font-medium [&_th]:text-foreground/80 [&_th]:text-xs [&_th]:uppercase [&_th]:tracking-wider",
|
|
415
|
+
"[&_td]:border [&_td]:border-border [&_td]:px-4 [&_td]:py-2.5 [&_td]:text-foreground/80",
|
|
416
|
+
// First/last margin reset
|
|
417
|
+
"[&>*:first-child]:mt-0 [&>*:last-child]:mb-0"
|
|
418
|
+
),
|
|
419
|
+
children: /* @__PURE__ */ jsxRuntimeExports.jsx(Markdown, { remarkPlugins: [remarkGfm], components: markdownComponents, children: skillContent })
|
|
420
|
+
}
|
|
421
|
+
),
|
|
422
|
+
(() => {
|
|
423
|
+
const cat = categories.find((c) => c.name === selectedCategory);
|
|
424
|
+
const skill = cat?.skills.find((s) => s.name === selectedSkill);
|
|
425
|
+
const relatedSkills = skill?.related_skills ?? [];
|
|
426
|
+
if (!relatedSkills.length) return null;
|
|
427
|
+
return /* @__PURE__ */ jsxRuntimeExports.jsxs("div", { className: "mt-8 pt-6 border-t border-border", children: [
|
|
428
|
+
/* @__PURE__ */ jsxRuntimeExports.jsxs("h4", { className: "flex items-center gap-1.5 text-xs font-semibold text-muted-foreground uppercase tracking-wide mb-3", children: [
|
|
429
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx(IconLink, { className: "size-3" }),
|
|
430
|
+
"Related Skills"
|
|
431
|
+
] }),
|
|
432
|
+
/* @__PURE__ */ jsxRuntimeExports.jsx("div", { className: "flex flex-wrap gap-1.5", children: relatedSkills.map((related) => /* @__PURE__ */ jsxRuntimeExports.jsx(
|
|
433
|
+
"span",
|
|
434
|
+
{
|
|
435
|
+
className: "rounded-md bg-muted/40 px-2.5 py-1 text-xs text-muted-foreground border border-border/40",
|
|
436
|
+
children: related
|
|
437
|
+
},
|
|
438
|
+
related
|
|
439
|
+
)) })
|
|
440
|
+
] });
|
|
441
|
+
})()
|
|
442
|
+
] }) })
|
|
443
|
+
] }) })
|
|
444
|
+
] });
|
|
445
|
+
}
|
|
446
|
+
const SplitComponent = SkillsPage;
|
|
447
|
+
export {
|
|
448
|
+
SplitComponent as component
|
|
449
|
+
};
|