hermium 0.2.0 → 0.3.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/bin/hermium.mjs +103 -58
- package/dist/api.mjs +35 -0
- package/dist/public/assets/css/styles-o9LLzp-x.css +1 -0
- package/dist/public/assets/js/{ChatInputBlock-Bw7AL70H.js → ChatInputBlock-BgJMx5nW.js} +1 -1
- package/dist/public/assets/js/{MarkdownMessage-8d7Y6VL-.js → MarkdownMessage-Dd6VsKhk.js} +1 -1
- package/dist/public/assets/js/{base-ui-BvQbAt_1.js → base-ui-Ls-HE_Zl.js} +1 -1
- package/dist/public/assets/js/{chat._sessionId-BG6lVraH.js → chat._sessionId-Yif-J0_7.js} +1 -1
- package/dist/public/assets/js/{chat.index-D2zdMPTT.js → chat.index-QgwoSqOs.js} +1 -1
- package/dist/public/assets/js/{index-C0AK45FU.js → index-BDLsrzTe.js} +26 -26
- package/dist/public/assets/js/{index-Cx5En4FK.js → index-DRH0t4ti.js} +1 -1
- package/dist/public/assets/js/{memory-CeSRdTkW.js → memory-CHpNNtU2.js} +1 -1
- package/dist/public/assets/js/{router-8uDKazL-.js → router-B8BATVn7.js} +1 -1
- package/dist/public/assets/js/{settings-Bc3Y5zXO.js → settings-CRV3Pp8x.js} +1 -1
- package/dist/public/assets/js/{skills-DZv7sA_5.js → skills-C8G72gXr.js} +1 -1
- package/dist/public/assets/js/{usage-DXQsT9_b.js → usage-Bev8hqwh.js} +1 -1
- package/dist/server/__23tanstack-start-plugin-adapters-Cwee5PKy.mjs +6 -0
- package/dist/server/_chunks/ssr-renderer.mjs +22 -0
- package/dist/server/_libs/bail.mjs +8 -0
- package/dist/server/_libs/base-ui__react.mjs +9858 -0
- package/dist/server/_libs/base-ui__utils.mjs +1106 -0
- package/dist/server/_libs/ccount.mjs +16 -0
- package/dist/server/_libs/character-entities.mjs +2130 -0
- package/dist/server/_libs/class-variance-authority.mjs +44 -0
- package/dist/server/_libs/clsx.mjs +16 -0
- package/dist/server/_libs/comma-separated-tokens.mjs +10 -0
- package/dist/server/_libs/cookie-es.mjs +1 -0
- package/dist/server/_libs/croner.mjs +1 -0
- package/dist/server/_libs/crossws.mjs +1 -0
- package/dist/server/_libs/decode-named-character-reference+[...].mjs +8 -0
- package/dist/server/_libs/devlop.mjs +8 -0
- package/dist/server/_libs/escape-string-regexp.mjs +9 -0
- package/dist/server/_libs/estree-util-is-identifier-name.mjs +11 -0
- package/dist/server/_libs/extend.mjs +97 -0
- package/dist/server/_libs/floating-ui__core.mjs +663 -0
- package/dist/server/_libs/floating-ui__dom.mjs +624 -0
- package/dist/server/_libs/floating-ui__react-dom.mjs +279 -0
- package/dist/server/_libs/floating-ui__utils.mjs +322 -0
- package/dist/server/_libs/h3.mjs +408 -0
- package/dist/server/_libs/hast-util-is-element.mjs +75 -0
- package/dist/server/_libs/hast-util-to-jsx-runtime.mjs +388 -0
- package/dist/server/_libs/hast-util-to-text.mjs +305 -0
- package/dist/server/_libs/hast-util-whitespace.mjs +10 -0
- package/dist/server/_libs/highlight.js.mjs +14756 -0
- package/dist/server/_libs/hookable.mjs +1 -0
- package/dist/server/_libs/html-url-attributes.mjs +26 -0
- package/dist/server/_libs/inline-style-parser.mjs +142 -0
- package/dist/server/_libs/is-plain-obj.mjs +10 -0
- package/dist/server/_libs/isbot.mjs +21 -0
- package/dist/server/_libs/longest-streak.mjs +25 -0
- package/dist/server/_libs/lowlight.mjs +262 -0
- package/dist/server/_libs/markdown-table.mjs +142 -0
- package/dist/server/_libs/mdast-util-find-and-replace.mjs +109 -0
- package/dist/server/_libs/mdast-util-from-markdown.mjs +717 -0
- package/dist/server/_libs/mdast-util-gfm-autolink-literal+[...].mjs +156 -0
- package/dist/server/_libs/mdast-util-gfm-footnote.mjs +117 -0
- package/dist/server/_libs/mdast-util-gfm-strikethrough.mjs +54 -0
- package/dist/server/_libs/mdast-util-gfm-table.mjs +157 -0
- package/dist/server/_libs/mdast-util-gfm-task-list-item.mjs +77 -0
- package/dist/server/_libs/mdast-util-gfm.mjs +29 -0
- package/dist/server/_libs/mdast-util-phrasing.mjs +30 -0
- package/dist/server/_libs/mdast-util-to-hast.mjs +710 -0
- package/dist/server/_libs/mdast-util-to-markdown.mjs +798 -0
- package/dist/server/_libs/mdast-util-to-string.mjs +38 -0
- package/dist/server/_libs/micromark-core-commonmark.mjs +2259 -0
- package/dist/server/_libs/micromark-extension-gfm-autolink-literal+[...].mjs +344 -0
- package/dist/server/_libs/micromark-extension-gfm-footnote+[...].mjs +279 -0
- package/dist/server/_libs/micromark-extension-gfm-strikethrough+[...].mjs +98 -0
- package/dist/server/_libs/micromark-extension-gfm-table.mjs +491 -0
- package/dist/server/_libs/micromark-extension-gfm-tagfilter+[...].mjs +1 -0
- package/dist/server/_libs/micromark-extension-gfm-task-list-item+[...].mjs +77 -0
- package/dist/server/_libs/micromark-extension-gfm.mjs +18 -0
- package/dist/server/_libs/micromark-factory-destination.mjs +94 -0
- package/dist/server/_libs/micromark-factory-label.mjs +63 -0
- package/dist/server/_libs/micromark-factory-space.mjs +24 -0
- package/dist/server/_libs/micromark-factory-title.mjs +65 -0
- package/dist/server/_libs/micromark-factory-whitespace.mjs +22 -0
- package/dist/server/_libs/micromark-util-character.mjs +44 -0
- package/dist/server/_libs/micromark-util-chunked.mjs +36 -0
- package/dist/server/_libs/micromark-util-classify-character+[...].mjs +12 -0
- package/dist/server/_libs/micromark-util-combine-extensions+[...].mjs +41 -0
- package/dist/server/_libs/micromark-util-decode-numeric-character-reference+[...].mjs +19 -0
- package/dist/server/_libs/micromark-util-decode-string.mjs +21 -0
- package/dist/server/_libs/micromark-util-encode.mjs +1 -0
- package/dist/server/_libs/micromark-util-html-tag-name.mjs +69 -0
- package/dist/server/_libs/micromark-util-normalize-identifier+[...].mjs +6 -0
- package/dist/server/_libs/micromark-util-resolve-all.mjs +15 -0
- package/dist/server/_libs/micromark-util-sanitize-uri.mjs +41 -0
- package/dist/server/_libs/micromark-util-subtokenize.mjs +346 -0
- package/dist/server/_libs/micromark.mjs +906 -0
- package/dist/server/_libs/ocache.mjs +1 -0
- package/dist/server/_libs/ohash.mjs +1 -0
- package/dist/server/_libs/property-information.mjs +1209 -0
- package/dist/server/_libs/react-dom.mjs +10779 -0
- package/dist/server/_libs/react-markdown.mjs +147 -0
- package/dist/server/_libs/react.mjs +513 -0
- package/dist/server/_libs/rehype-highlight.mjs +94 -0
- package/dist/server/_libs/remark-gfm.mjs +20 -0
- package/dist/server/_libs/remark-parse.mjs +19 -0
- package/dist/server/_libs/remark-rehype.mjs +21 -0
- package/dist/server/_libs/reselect.mjs +1 -0
- package/dist/server/_libs/rou3.mjs +8 -0
- package/dist/server/_libs/seroval-plugins.mjs +1 -0
- package/dist/server/_libs/seroval.mjs +1 -0
- package/dist/server/_libs/space-separated-tokens.mjs +6 -0
- package/dist/server/_libs/srvx.mjs +781 -0
- package/dist/server/_libs/style-to-js.mjs +72 -0
- package/dist/server/_libs/style-to-object.mjs +38 -0
- package/dist/server/_libs/tabler__icons-react.mjs +140 -0
- package/dist/server/_libs/tailwind-merge.mjs +3255 -0
- package/dist/server/_libs/tanstack__history.mjs +29 -0
- package/dist/server/_libs/tanstack__react-router.mjs +1120 -0
- package/dist/server/_libs/tanstack__react-store.mjs +2 -0
- package/dist/server/_libs/tanstack__router-core.mjs +3594 -0
- package/dist/server/_libs/tanstack__store.mjs +1 -0
- package/dist/server/_libs/trim-lines.mjs +41 -0
- package/dist/server/_libs/trough.mjs +85 -0
- package/dist/server/_libs/ufo.mjs +54 -0
- package/dist/server/_libs/unctx.mjs +1 -0
- package/dist/server/_libs/ungap__structured-clone.mjs +224 -0
- package/dist/server/_libs/unified.mjs +661 -0
- package/dist/server/_libs/unist-util-find-after.mjs +41 -0
- package/dist/server/_libs/unist-util-is.mjs +100 -0
- package/dist/server/_libs/unist-util-position.mjs +27 -0
- package/dist/server/_libs/unist-util-stringify-position.mjs +27 -0
- package/dist/server/_libs/unist-util-visit-parents.mjs +83 -0
- package/dist/server/_libs/unist-util-visit.mjs +24 -0
- package/dist/server/_libs/unstorage.mjs +1 -0
- package/dist/server/_libs/use-sync-external-store.mjs +139 -0
- package/dist/server/_libs/vfile-message.mjs +138 -0
- package/dist/server/_libs/vfile.mjs +467 -0
- package/dist/server/_libs/zustand.mjs +43 -0
- package/dist/server/_libs/zwitch.mjs +1 -0
- package/dist/server/_ssr/ChatInputBlock-Bu2-iop_.mjs +220 -0
- package/dist/server/_ssr/MarkdownMessage-CNS7OSKN.mjs +68 -0
- package/dist/server/_ssr/chat._sessionId-P02iSfut.mjs +477 -0
- package/dist/server/_ssr/chat.index-BYB_48NC.mjs +64 -0
- package/dist/server/_ssr/index-C1mT_2d8.mjs +4890 -0
- package/dist/server/_ssr/index-DFV9_oCk.mjs +43 -0
- package/dist/server/_ssr/memory-CW_fSOG9.mjs +257 -0
- package/dist/server/_ssr/router-CUAfx91O.mjs +2035 -0
- package/dist/server/_ssr/settings-DoXurzvn.mjs +10 -0
- package/dist/server/_ssr/skills-Cs7A5ZwO.mjs +422 -0
- package/dist/server/_ssr/theme-BK4-7E2h.mjs +42 -0
- package/dist/server/_ssr/usage-Bs2-LXGz.mjs +298 -0
- package/dist/server/_tanstack-start-manifest_v-C7Upe2TI.mjs +4 -0
- package/dist/server/index.mjs +506 -0
- package/dist/server/public/assets/css/index-Dfs9RUU9.css +1 -0
- package/dist/server/public/assets/css/styles-o9LLzp-x.css +1 -0
- package/dist/server/public/assets/js/ChatInputBlock-BgJMx5nW.js +1 -0
- package/dist/server/public/assets/js/MarkdownMessage-Dd6VsKhk.js +1 -0
- package/dist/server/public/assets/js/base-ui-Ls-HE_Zl.js +1 -0
- package/dist/server/public/assets/js/chat._sessionId-Yif-J0_7.js +1 -0
- package/dist/server/public/assets/js/chat.index-QgwoSqOs.js +1 -0
- package/dist/server/public/assets/js/index-BDLsrzTe.js +60 -0
- package/dist/server/public/assets/js/index-DRH0t4ti.js +1 -0
- package/dist/server/public/assets/js/memory-CHpNNtU2.js +3 -0
- package/dist/server/public/assets/js/router-B8BATVn7.js +1 -0
- package/dist/server/public/assets/js/settings-CRV3Pp8x.js +1 -0
- package/dist/server/public/assets/js/skills-C8G72gXr.js +1 -0
- package/dist/server/public/assets/js/theme-CPkdkpaj.js +1 -0
- package/dist/server/public/assets/js/usage-Bev8hqwh.js +1 -0
- package/dist/server/public/assets/woff2/geist-cyrillic-ext-wght-normal-DjL33-gN.woff2 +0 -0
- package/dist/server/public/assets/woff2/geist-cyrillic-wght-normal-BEAKL7Jp.woff2 +0 -0
- package/dist/server/public/assets/woff2/geist-latin-ext-wght-normal-DC-KSUi6.woff2 +0 -0
- package/dist/server/public/assets/woff2/geist-latin-wght-normal-BgDaEnEv.woff2 +0 -0
- package/dist/server/public/assets/woff2/geist-vietnamese-wght-normal-6IgcOCM7.woff2 +0 -0
- package/dist/server/public/favicon.ico +0 -0
- package/dist/server/public/logo.png +0 -0
- package/dist/server/public/manifest.json +25 -0
- package/dist/server/public/robots.txt +3 -0
- package/package.json +4 -3
- package/dist/public/assets/css/styles-B8p6jk5Z.css +0 -1
|
@@ -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
|
+
};
|