@archznn/crewloop-skills 0.6.0 → 0.7.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 (245) hide show
  1. package/README.md +4 -16
  2. package/package.json +1 -2
  3. package/packages/cli/dist/agents.js +1 -1
  4. package/packages/cli/dist/agents.js.map +1 -1
  5. package/packages/cli/dist/cli.d.ts.map +1 -1
  6. package/packages/cli/dist/cli.js +2 -30
  7. package/packages/cli/dist/cli.js.map +1 -1
  8. package/packages/cli/dist/hooks.d.ts +6 -4
  9. package/packages/cli/dist/hooks.d.ts.map +1 -1
  10. package/packages/cli/dist/hooks.js +250 -98
  11. package/packages/cli/dist/hooks.js.map +1 -1
  12. package/packages/cli/dist/tests/hooks.test.js +245 -33
  13. package/packages/cli/dist/tests/hooks.test.js.map +1 -1
  14. package/references/conventions.md +1 -10
  15. package/references/workflow.md +1 -1
  16. package/servers/dashboard/README.md +55 -1
  17. package/servers/dashboard/dist/adapters/agy.d.ts +19 -0
  18. package/servers/dashboard/dist/adapters/agy.d.ts.map +1 -0
  19. package/servers/dashboard/dist/adapters/agy.js +108 -0
  20. package/servers/dashboard/dist/adapters/agy.js.map +1 -0
  21. package/servers/dashboard/dist/adapters/codex.d.ts.map +1 -1
  22. package/servers/dashboard/dist/adapters/codex.js +2 -0
  23. package/servers/dashboard/dist/adapters/codex.js.map +1 -1
  24. package/servers/dashboard/dist/adapters/kimi.d.ts +1 -1
  25. package/servers/dashboard/dist/adapters/kimi.d.ts.map +1 -1
  26. package/servers/dashboard/dist/adapters/kimi.js +9 -0
  27. package/servers/dashboard/dist/adapters/kimi.js.map +1 -1
  28. package/servers/dashboard/dist/adapters/shim.d.ts +1 -1
  29. package/servers/dashboard/dist/adapters/shim.d.ts.map +1 -1
  30. package/servers/dashboard/dist/adapters/shim.js +32 -11
  31. package/servers/dashboard/dist/adapters/shim.js.map +1 -1
  32. package/servers/dashboard/dist/adapters/shim.test.js +46 -4
  33. package/servers/dashboard/dist/adapters/shim.test.js.map +1 -1
  34. package/servers/dashboard/dist/lib/constants.d.ts +5 -0
  35. package/servers/dashboard/dist/lib/constants.d.ts.map +1 -0
  36. package/servers/dashboard/dist/lib/constants.js +46 -0
  37. package/servers/dashboard/dist/lib/constants.js.map +1 -0
  38. package/servers/dashboard/dist/lib/format.d.ts +6 -0
  39. package/servers/dashboard/dist/lib/format.d.ts.map +1 -0
  40. package/servers/dashboard/dist/lib/format.js +52 -0
  41. package/servers/dashboard/dist/lib/format.js.map +1 -0
  42. package/servers/dashboard/dist/lib/graph.d.ts +22 -0
  43. package/servers/dashboard/dist/lib/graph.d.ts.map +1 -0
  44. package/servers/dashboard/dist/lib/graph.js +45 -0
  45. package/servers/dashboard/dist/lib/graph.js.map +1 -0
  46. package/servers/dashboard/dist/lib/invocations.d.ts +32 -0
  47. package/servers/dashboard/dist/lib/invocations.d.ts.map +1 -0
  48. package/servers/dashboard/dist/lib/invocations.js +135 -0
  49. package/servers/dashboard/dist/lib/invocations.js.map +1 -0
  50. package/servers/dashboard/dist/lib/invocations.test.d.ts +2 -0
  51. package/servers/dashboard/dist/lib/invocations.test.d.ts.map +1 -0
  52. package/servers/dashboard/dist/lib/invocations.test.js +68 -0
  53. package/servers/dashboard/dist/lib/invocations.test.js.map +1 -0
  54. package/servers/dashboard/dist/lib/paths.d.ts +2 -0
  55. package/servers/dashboard/dist/lib/paths.d.ts.map +1 -0
  56. package/servers/dashboard/dist/lib/paths.js +40 -0
  57. package/servers/dashboard/dist/lib/paths.js.map +1 -0
  58. package/servers/dashboard/dist/presenter.d.ts.map +1 -1
  59. package/servers/dashboard/dist/presenter.js +2 -0
  60. package/servers/dashboard/dist/presenter.js.map +1 -1
  61. package/servers/dashboard/dist/public/assets/index-DjmMKbPN.css +1 -0
  62. package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js +5323 -0
  63. package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js.map +1 -0
  64. package/servers/dashboard/dist/public/index.html +16 -0
  65. package/servers/dashboard/dist/server.d.ts.map +1 -1
  66. package/servers/dashboard/dist/server.js +5 -1
  67. package/servers/dashboard/dist/server.js.map +1 -1
  68. package/servers/dashboard/dist/skills/infer.d.ts.map +1 -1
  69. package/servers/dashboard/dist/skills/infer.js +0 -6
  70. package/servers/dashboard/dist/skills/infer.js.map +1 -1
  71. package/servers/dashboard/dist/skills/infer.test.js +10 -3
  72. package/servers/dashboard/dist/skills/infer.test.js.map +1 -1
  73. package/servers/dashboard/dist/skills/mapping.d.ts +0 -3
  74. package/servers/dashboard/dist/skills/mapping.d.ts.map +1 -1
  75. package/servers/dashboard/dist/skills/mapping.js +0 -18
  76. package/servers/dashboard/dist/skills/mapping.js.map +1 -1
  77. package/servers/dashboard/dist/skills/registry.d.ts.map +1 -1
  78. package/servers/dashboard/dist/skills/registry.js +0 -1
  79. package/servers/dashboard/dist/skills/registry.js.map +1 -1
  80. package/servers/dashboard/dist/tests/adapters.test.d.ts +2 -0
  81. package/servers/dashboard/dist/tests/adapters.test.d.ts.map +1 -0
  82. package/servers/dashboard/dist/tests/adapters.test.js +180 -0
  83. package/servers/dashboard/dist/tests/adapters.test.js.map +1 -0
  84. package/servers/dashboard/dist/tests/lib-helpers.test.d.ts +2 -0
  85. package/servers/dashboard/dist/tests/lib-helpers.test.d.ts.map +1 -0
  86. package/servers/dashboard/dist/tests/lib-helpers.test.js +123 -0
  87. package/servers/dashboard/dist/tests/lib-helpers.test.js.map +1 -0
  88. package/servers/dashboard/dist/tests/shim.test.js +88 -2
  89. package/servers/dashboard/dist/tests/shim.test.js.map +1 -1
  90. package/servers/dashboard/dist/types.d.ts +5 -2
  91. package/servers/dashboard/dist/types.d.ts.map +1 -1
  92. package/servers/dashboard/package.json +22 -5
  93. package/servers/dashboard/src/adapters/agy.ts +136 -0
  94. package/servers/dashboard/src/adapters/codex.ts +2 -0
  95. package/servers/dashboard/src/adapters/kimi.ts +11 -1
  96. package/servers/dashboard/src/adapters/shim.test.ts +57 -4
  97. package/servers/dashboard/src/adapters/shim.ts +31 -11
  98. package/servers/dashboard/src/lib/constants.ts +44 -0
  99. package/servers/dashboard/src/lib/format.ts +44 -0
  100. package/servers/dashboard/src/lib/graph.ts +69 -0
  101. package/servers/dashboard/src/lib/invocations.test.ts +70 -0
  102. package/servers/dashboard/src/lib/invocations.ts +172 -0
  103. package/servers/dashboard/src/lib/paths.ts +35 -0
  104. package/servers/dashboard/src/presenter.ts +2 -0
  105. package/servers/dashboard/src/server.ts +5 -1
  106. package/servers/dashboard/src/skills/infer.test.ts +11 -3
  107. package/servers/dashboard/src/skills/infer.ts +1 -8
  108. package/servers/dashboard/src/skills/mapping.ts +0 -20
  109. package/servers/dashboard/src/skills/registry.ts +0 -1
  110. package/servers/dashboard/src/tests/adapters.test.ts +198 -0
  111. package/servers/dashboard/src/tests/lib-helpers.test.ts +133 -0
  112. package/servers/dashboard/src/tests/shim.test.ts +110 -2
  113. package/servers/dashboard/src/types.ts +5 -3
  114. package/servers/dashboard/ui/index.html +15 -0
  115. package/servers/dashboard/ui/postcss.config.js +6 -0
  116. package/servers/dashboard/ui/src/App.tsx +360 -0
  117. package/servers/dashboard/ui/src/components/ActiveSkillPanel.tsx +69 -0
  118. package/servers/dashboard/ui/src/components/ActivityGraph.tsx +74 -0
  119. package/servers/dashboard/ui/src/components/CommandPalette.tsx +200 -0
  120. package/servers/dashboard/ui/src/components/FileActivity.tsx +20 -0
  121. package/servers/dashboard/ui/src/components/FileDiff.tsx +68 -0
  122. package/servers/dashboard/ui/src/components/FileList.tsx +64 -0
  123. package/servers/dashboard/ui/src/components/FilterBar.tsx +208 -0
  124. package/servers/dashboard/ui/src/components/Network3D.tsx +178 -0
  125. package/servers/dashboard/ui/src/components/SessionSelector.tsx +95 -0
  126. package/servers/dashboard/ui/src/components/Sidebar.tsx +110 -0
  127. package/servers/dashboard/ui/src/components/TelemetryPanel.tsx +57 -0
  128. package/servers/dashboard/ui/src/components/Timeline.tsx +57 -0
  129. package/servers/dashboard/ui/src/components/TimelineRow.tsx +112 -0
  130. package/servers/dashboard/ui/src/components/TopBar.tsx +116 -0
  131. package/servers/dashboard/ui/src/components/ViewHeader.tsx +19 -0
  132. package/servers/dashboard/ui/src/components/ui/Icon.tsx +105 -0
  133. package/servers/dashboard/ui/src/components/ui/StatusBadge.tsx +19 -0
  134. package/servers/dashboard/ui/src/components/views/FilesView.tsx +23 -0
  135. package/servers/dashboard/ui/src/components/views/NetworkView.tsx +20 -0
  136. package/servers/dashboard/ui/src/components/views/Overview.tsx +135 -0
  137. package/servers/dashboard/ui/src/components/views/SessionsView.tsx +84 -0
  138. package/servers/dashboard/ui/src/components/views/SettingsView.tsx +138 -0
  139. package/servers/dashboard/ui/src/components/views/SkillsView.tsx +92 -0
  140. package/servers/dashboard/ui/src/components/views/TimelineView.tsx +46 -0
  141. package/servers/dashboard/ui/src/contexts/FilterContext.tsx +41 -0
  142. package/servers/dashboard/ui/src/contexts/PinnedSessionsContext.tsx +80 -0
  143. package/servers/dashboard/ui/src/contexts/SettingsContext.tsx +60 -0
  144. package/servers/dashboard/ui/src/hooks/useCommandPalette.ts +36 -0
  145. package/servers/dashboard/ui/src/hooks/useKeyboardShortcut.ts +38 -0
  146. package/servers/dashboard/ui/src/hooks/useNow.ts +12 -0
  147. package/servers/dashboard/ui/src/hooks/useReducedMotion.ts +15 -0
  148. package/servers/dashboard/ui/src/hooks/useSessions.ts +64 -0
  149. package/servers/dashboard/ui/src/hooks/useTheme.ts +30 -0
  150. package/servers/dashboard/ui/src/hooks/useViewport.ts +19 -0
  151. package/servers/dashboard/ui/src/hooks/useWebSocket.ts +118 -0
  152. package/servers/dashboard/ui/src/lib/export.test.ts +33 -0
  153. package/servers/dashboard/ui/src/lib/export.ts +39 -0
  154. package/servers/dashboard/ui/src/lib/filter.test.ts +95 -0
  155. package/servers/dashboard/ui/src/lib/filter.ts +178 -0
  156. package/servers/dashboard/ui/src/lib/format.test.ts +25 -0
  157. package/servers/dashboard/ui/src/lib/search.test.ts +52 -0
  158. package/servers/dashboard/ui/src/lib/search.ts +60 -0
  159. package/servers/dashboard/ui/src/lib/settings.test.ts +50 -0
  160. package/servers/dashboard/ui/src/lib/settings.ts +56 -0
  161. package/servers/dashboard/ui/src/lib/types.ts +124 -0
  162. package/servers/dashboard/ui/src/main.tsx +19 -0
  163. package/servers/dashboard/ui/src/styles/index.css +155 -0
  164. package/servers/dashboard/ui/tailwind.config.js +45 -0
  165. package/servers/dashboard/ui/tsconfig.json +33 -0
  166. package/servers/dashboard/ui/tsconfig.node.json +10 -0
  167. package/servers/dashboard/ui/vite.config.ts +37 -0
  168. package/servers/dashboard/ui/vitest.config.ts +8 -0
  169. package/skills/accessibility-auditor/SKILL.md +0 -20
  170. package/skills/architect/SKILL.md +0 -45
  171. package/skills/designer/SKILL.md +0 -30
  172. package/skills/docs-writer/SKILL.md +0 -13
  173. package/skills/engineer/SKILL.md +0 -30
  174. package/skills/maintainer/SKILL.md +0 -20
  175. package/skills/orchestrator/SKILL.md +0 -13
  176. package/skills/product-manager/SKILL.md +0 -20
  177. package/skills/researcher/SKILL.md +0 -20
  178. package/skills/reviewer/SKILL.md +0 -30
  179. package/skills/security-guard/SKILL.md +0 -20
  180. package/skills/shipper/SKILL.md +0 -33
  181. package/skills/tester/SKILL.md +0 -20
  182. package/packages/cli/dist/mcp.d.ts +0 -28
  183. package/packages/cli/dist/mcp.d.ts.map +0 -1
  184. package/packages/cli/dist/mcp.js +0 -148
  185. package/packages/cli/dist/mcp.js.map +0 -1
  186. package/packages/cli/dist/tests/mcp.test.d.ts +0 -2
  187. package/packages/cli/dist/tests/mcp.test.d.ts.map +0 -1
  188. package/packages/cli/dist/tests/mcp.test.js +0 -232
  189. package/packages/cli/dist/tests/mcp.test.js.map +0 -1
  190. package/references/obsidian-mcp-usage.md +0 -190
  191. package/servers/dashboard/public/app.js +0 -516
  192. package/servers/dashboard/public/index.html +0 -96
  193. package/servers/dashboard/public/styles.css +0 -819
  194. package/servers/obsidian-mcp/README.md +0 -82
  195. package/servers/obsidian-mcp/pyproject.toml +0 -32
  196. package/servers/obsidian-mcp/src/obsidian_mcp/__init__.py +0 -0
  197. package/servers/obsidian-mcp/src/obsidian_mcp/config.py +0 -47
  198. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/__init__.py +0 -0
  199. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/embeddings.py +0 -105
  200. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/indexer.py +0 -79
  201. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/store.py +0 -141
  202. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/sync.py +0 -37
  203. package/servers/obsidian-mcp/src/obsidian_mcp/learning/__init__.py +0 -0
  204. package/servers/obsidian-mcp/src/obsidian_mcp/learning/detector.py +0 -66
  205. package/servers/obsidian-mcp/src/obsidian_mcp/learning/note_generator.py +0 -40
  206. package/servers/obsidian-mcp/src/obsidian_mcp/main.py +0 -4
  207. package/servers/obsidian-mcp/src/obsidian_mcp/models.py +0 -42
  208. package/servers/obsidian-mcp/src/obsidian_mcp/privacy/__init__.py +0 -0
  209. package/servers/obsidian-mcp/src/obsidian_mcp/privacy/filter.py +0 -68
  210. package/servers/obsidian-mcp/src/obsidian_mcp/rag/__init__.py +0 -0
  211. package/servers/obsidian-mcp/src/obsidian_mcp/rag/engine.py +0 -50
  212. package/servers/obsidian-mcp/src/obsidian_mcp/rag/graph_search.py +0 -55
  213. package/servers/obsidian-mcp/src/obsidian_mcp/rag/text_search.py +0 -37
  214. package/servers/obsidian-mcp/src/obsidian_mcp/rag/vector_search.py +0 -118
  215. package/servers/obsidian-mcp/src/obsidian_mcp/server.py +0 -61
  216. package/servers/obsidian-mcp/src/obsidian_mcp/tools/__init__.py +0 -0
  217. package/servers/obsidian-mcp/src/obsidian_mcp/tools/create.py +0 -43
  218. package/servers/obsidian-mcp/src/obsidian_mcp/tools/delete.py +0 -16
  219. package/servers/obsidian-mcp/src/obsidian_mcp/tools/learn.py +0 -42
  220. package/servers/obsidian-mcp/src/obsidian_mcp/tools/list.py +0 -16
  221. package/servers/obsidian-mcp/src/obsidian_mcp/tools/read.py +0 -15
  222. package/servers/obsidian-mcp/src/obsidian_mcp/tools/registry.py +0 -130
  223. package/servers/obsidian-mcp/src/obsidian_mcp/tools/related.py +0 -20
  224. package/servers/obsidian-mcp/src/obsidian_mcp/tools/search.py +0 -26
  225. package/servers/obsidian-mcp/src/obsidian_mcp/tools/sync.py +0 -22
  226. package/servers/obsidian-mcp/src/obsidian_mcp/tools/update.py +0 -34
  227. package/servers/obsidian-mcp/src/obsidian_mcp/vault/__init__.py +0 -0
  228. package/servers/obsidian-mcp/src/obsidian_mcp/vault/parser.py +0 -82
  229. package/servers/obsidian-mcp/src/obsidian_mcp/vault/repository.py +0 -68
  230. package/servers/obsidian-mcp/src/obsidian_mcp/vault/writer.py +0 -61
  231. package/servers/obsidian-mcp/tests/conftest.py +0 -39
  232. package/servers/obsidian-mcp/tests/test_async_tools.py +0 -87
  233. package/servers/obsidian-mcp/tests/test_edge_cases.py +0 -59
  234. package/servers/obsidian-mcp/tests/test_indexer.py +0 -27
  235. package/servers/obsidian-mcp/tests/test_integration.py +0 -90
  236. package/servers/obsidian-mcp/tests/test_learning.py +0 -34
  237. package/servers/obsidian-mcp/tests/test_privacy.py +0 -31
  238. package/servers/obsidian-mcp/tests/test_privacy_config.py +0 -44
  239. package/servers/obsidian-mcp/tests/test_rag.py +0 -64
  240. package/servers/obsidian-mcp/tests/test_read_raw.py +0 -37
  241. package/servers/obsidian-mcp/tests/test_tfidf_fallback.py +0 -54
  242. package/servers/obsidian-mcp/tests/test_tools.py +0 -108
  243. package/servers/obsidian-mcp/tests/test_vault.py +0 -103
  244. package/servers/obsidian-mcp/tests/test_writer.py +0 -139
  245. package/skills/obsidian-second-brain/SKILL.md +0 -298
@@ -0,0 +1,60 @@
1
+ import { createContext, useCallback, useContext, useEffect, useMemo, useState, type ReactNode } from 'react';
2
+ import type { DashboardSettings, Theme } from '../lib/types';
3
+ import { loadSettings, saveSettings } from '../lib/settings';
4
+ import { useReducedMotion } from '../hooks/useReducedMotion';
5
+
6
+ interface SettingsContextValue {
7
+ settings: DashboardSettings;
8
+ resolvedTheme: 'dark' | 'light';
9
+ reducedMotion: boolean;
10
+ setSettings: (updater: DashboardSettings | ((prev: DashboardSettings) => DashboardSettings)) => void;
11
+ }
12
+
13
+ const SettingsContext = createContext<SettingsContextValue | null>(null);
14
+
15
+ function resolveTheme(theme: Theme): 'dark' | 'light' {
16
+ if (theme !== 'system') return theme;
17
+ return window.matchMedia('(prefers-color-scheme: light)').matches ? 'light' : 'dark';
18
+ }
19
+
20
+ export function SettingsProvider({ children }: { children: ReactNode }) {
21
+ const [settings, setSettingsState] = useState<DashboardSettings>(() => loadSettings());
22
+ const systemReducedMotion = useReducedMotion();
23
+ const resolvedTheme = useMemo(() => resolveTheme(settings.theme), [settings.theme]);
24
+ const reducedMotion = settings.reducedMotion || systemReducedMotion;
25
+
26
+ useEffect(() => {
27
+ const root = document.documentElement;
28
+ root.classList.remove('dark', 'light');
29
+ root.classList.add(resolvedTheme);
30
+ }, [resolvedTheme]);
31
+
32
+ useEffect(() => {
33
+ const root = document.documentElement;
34
+ root.classList.remove('density-compact', 'density-comfortable');
35
+ root.classList.add(`density-${settings.density}`);
36
+ }, [settings.density]);
37
+
38
+ const setSettings = useCallback(
39
+ (updater: DashboardSettings | ((prev: DashboardSettings) => DashboardSettings)) => {
40
+ setSettingsState((prev) => {
41
+ const next = typeof updater === 'function' ? (updater as (p: DashboardSettings) => DashboardSettings)(prev) : updater;
42
+ saveSettings(next);
43
+ return next;
44
+ });
45
+ },
46
+ []
47
+ );
48
+
49
+ return (
50
+ <SettingsContext.Provider value={{ settings, resolvedTheme, reducedMotion, setSettings }}>
51
+ {children}
52
+ </SettingsContext.Provider>
53
+ );
54
+ }
55
+
56
+ export function useSettings(): SettingsContextValue {
57
+ const ctx = useContext(SettingsContext);
58
+ if (!ctx) throw new Error('useSettings must be used within SettingsProvider');
59
+ return ctx;
60
+ }
@@ -0,0 +1,36 @@
1
+ import { useCallback, useEffect, useState } from 'react';
2
+
3
+ export interface CommandPaletteState {
4
+ open: boolean;
5
+ query: string;
6
+ selectedIndex: number;
7
+ }
8
+
9
+ export interface CommandPaletteActions {
10
+ setOpen: (open: boolean) => void;
11
+ setQuery: (query: string) => void;
12
+ setSelectedIndex: (index: number | ((prev: number) => number)) => void;
13
+ }
14
+
15
+ export function useCommandPalette(): CommandPaletteState & CommandPaletteActions {
16
+ const [open, setOpen] = useState(false);
17
+ const [query, setQuery] = useState('');
18
+ const [selectedIndex, setSelectedIndexState] = useState(0);
19
+
20
+ const setSelectedIndex = useCallback(
21
+ (index: number | ((prev: number) => number)) => {
22
+ setSelectedIndexState((prev) => (typeof index === 'function' ? index(prev) : index));
23
+ },
24
+ []
25
+ );
26
+
27
+ useEffect(() => {
28
+ setSelectedIndexState(0);
29
+ }, [query]);
30
+
31
+ useEffect(() => {
32
+ if (open) setQuery('');
33
+ }, [open]);
34
+
35
+ return { open, setOpen, query, setQuery, selectedIndex, setSelectedIndex };
36
+ }
@@ -0,0 +1,38 @@
1
+ import { useEffect } from 'react';
2
+
3
+ interface ShortcutOptions {
4
+ meta?: boolean;
5
+ ctrl?: boolean;
6
+ preventDefault?: boolean;
7
+ disabled?: boolean;
8
+ }
9
+
10
+ export function useKeyboardShortcut(
11
+ key: string,
12
+ callback: () => void,
13
+ options: ShortcutOptions = {}
14
+ ): void {
15
+ const { meta = false, ctrl = false, preventDefault = true, disabled = false } = options;
16
+
17
+ useEffect(() => {
18
+ if (disabled) return;
19
+
20
+ function handler(e: KeyboardEvent) {
21
+ const target = e.target as HTMLElement | null;
22
+ const tag = target?.tagName ?? '';
23
+ if (tag === 'INPUT' || tag === 'TEXTAREA' || tag === 'SELECT' || target?.isContentEditable) {
24
+ return;
25
+ }
26
+ if (e.key.toLowerCase() !== key.toLowerCase()) return;
27
+ if (meta && !(e.metaKey || e.ctrlKey)) return;
28
+ if (ctrl && !e.ctrlKey) return;
29
+ if (!meta && !ctrl && (e.metaKey || e.ctrlKey)) return;
30
+
31
+ if (preventDefault) e.preventDefault();
32
+ callback();
33
+ }
34
+
35
+ document.addEventListener('keydown', handler);
36
+ return () => document.removeEventListener('keydown', handler);
37
+ }, [key, callback, meta, ctrl, preventDefault, disabled]);
38
+ }
@@ -0,0 +1,12 @@
1
+ import { useEffect, useState } from 'react';
2
+
3
+ export function useNow(interval = 10_000): number {
4
+ const [now, setNow] = useState(Date.now());
5
+
6
+ useEffect(() => {
7
+ const id = window.setInterval(() => setNow(Date.now()), interval);
8
+ return () => window.clearInterval(id);
9
+ }, [interval]);
10
+
11
+ return now;
12
+ }
@@ -0,0 +1,15 @@
1
+ import { useEffect, useState } from 'react';
2
+
3
+ export function useReducedMotion(): boolean {
4
+ const [reduced, setReduced] = useState(false);
5
+
6
+ useEffect(() => {
7
+ const mq = window.matchMedia('(prefers-reduced-motion: reduce)');
8
+ setReduced(mq.matches);
9
+ const handler = (e: MediaQueryListEvent) => setReduced(e.matches);
10
+ mq.addEventListener('change', handler);
11
+ return () => mq.removeEventListener('change', handler);
12
+ }, []);
13
+
14
+ return reduced;
15
+ }
@@ -0,0 +1,64 @@
1
+ import { useCallback, useState } from 'react';
2
+ import type { ClientSession, ClientWebSocketMessage } from '../../../src/types';
3
+
4
+ export interface SessionsState {
5
+ sessions: Map<string, ClientSession>;
6
+ selectedSessionId: string | null;
7
+ }
8
+
9
+ function defaultSelected(
10
+ sessions: Map<string, ClientSession>,
11
+ activeSessionId: string | undefined,
12
+ current: string | null
13
+ ): string | null {
14
+ if (current && sessions.has(current)) return current;
15
+ if (activeSessionId && sessions.has(activeSessionId)) return activeSessionId;
16
+ const first = sessions.keys().next().value;
17
+ return first || null;
18
+ }
19
+
20
+ export function useSessions() {
21
+ const [state, setState] = useState<SessionsState>({
22
+ sessions: new Map(),
23
+ selectedSessionId: null,
24
+ });
25
+
26
+ const selectSession = useCallback((id: string | null) => {
27
+ setState((prev) => ({ ...prev, selectedSessionId: id }));
28
+ }, []);
29
+
30
+ const handleMessage = useCallback((msg: ClientWebSocketMessage) => {
31
+ setState((prev) => {
32
+ const sessions = new Map(prev.sessions);
33
+ let activeSessionId: string | undefined;
34
+
35
+ if (msg.type === 'snapshot') {
36
+ sessions.clear();
37
+ for (const s of msg.sessions) {
38
+ sessions.set(s.id, s);
39
+ }
40
+ } else if (msg.type === 'update') {
41
+ const s = msg.session;
42
+ sessions.set(s.id, s);
43
+ if (msg.isActive) activeSessionId = s.id;
44
+ }
45
+
46
+ return {
47
+ sessions,
48
+ selectedSessionId: defaultSelected(sessions, activeSessionId, prev.selectedSessionId),
49
+ };
50
+ });
51
+ }, []);
52
+
53
+ const sortedSessions = Array.from(state.sessions.values()).sort(
54
+ (a, b) => (b.lastActivity || 0) - (a.lastActivity || 0)
55
+ );
56
+
57
+ return {
58
+ sessions: state.sessions,
59
+ selectedSessionId: state.selectedSessionId,
60
+ selectSession,
61
+ handleMessage,
62
+ sortedSessions,
63
+ };
64
+ }
@@ -0,0 +1,30 @@
1
+ import { useEffect, useState } from 'react';
2
+
3
+ type Theme = 'dark' | 'light' | 'system';
4
+
5
+ function resolveTheme(theme: Theme): 'dark' | 'light' {
6
+ if (theme !== 'system') return theme;
7
+ if (window.matchMedia('(prefers-color-scheme: light)').matches) return 'light';
8
+ return 'dark';
9
+ }
10
+
11
+ export function useTheme() {
12
+ const [theme, setThemeState] = useState<Theme>(() => {
13
+ const stored = localStorage.getItem('crewloop-theme') as Theme | null;
14
+ return stored || 'system';
15
+ });
16
+ const resolved = resolveTheme(theme);
17
+
18
+ useEffect(() => {
19
+ const root = document.documentElement;
20
+ root.classList.remove('dark', 'light');
21
+ root.classList.add(resolved);
22
+ }, [resolved]);
23
+
24
+ const setTheme = (value: Theme) => {
25
+ localStorage.setItem('crewloop-theme', value);
26
+ setThemeState(value);
27
+ };
28
+
29
+ return { theme, resolved, setTheme };
30
+ }
@@ -0,0 +1,19 @@
1
+ import { useEffect, useState } from 'react';
2
+
3
+ export type Breakpoint = 'mobile' | 'tablet' | 'desktop';
4
+
5
+ export function useViewport(): { width: number; breakpoint: Breakpoint } {
6
+ const [width, setWidth] = useState(window.innerWidth);
7
+
8
+ useEffect(() => {
9
+ const handler = () => setWidth(window.innerWidth);
10
+ window.addEventListener('resize', handler);
11
+ return () => window.removeEventListener('resize', handler);
12
+ }, []);
13
+
14
+ let breakpoint: Breakpoint = 'desktop';
15
+ if (width < 768) breakpoint = 'mobile';
16
+ else if (width < 1024) breakpoint = 'tablet';
17
+
18
+ return { width, breakpoint };
19
+ }
@@ -0,0 +1,118 @@
1
+ import { useEffect, useRef, useState } from 'react';
2
+ import type { ClientWebSocketMessage } from '../../../src/types';
3
+
4
+ type ConnectionStatus = 'connecting' | 'connected' | 'disconnected';
5
+
6
+ export function useWebSocket(
7
+ url: string,
8
+ onMessage: (msg: ClientWebSocketMessage) => void
9
+ ): { status: ConnectionStatus; send: (data: unknown) => void } {
10
+ const [status, setStatus] = useState<ConnectionStatus>('connecting');
11
+ const wsRef = useRef<WebSocket | null>(null);
12
+ const reconnectTimerRef = useRef<number | null>(null);
13
+ const pingTimerRef = useRef<number | null>(null);
14
+ const lastPongRef = useRef<number>(0);
15
+ const onMessageRef = useRef(onMessage);
16
+
17
+ useEffect(() => {
18
+ onMessageRef.current = onMessage;
19
+ }, [onMessage]);
20
+
21
+ useEffect(() => {
22
+ let active = true;
23
+
24
+ function scheduleReconnect() {
25
+ if (reconnectTimerRef.current) return;
26
+ reconnectTimerRef.current = window.setTimeout(() => {
27
+ reconnectTimerRef.current = null;
28
+ if (active) connect();
29
+ }, 3000);
30
+ }
31
+
32
+ function startPing(ws: WebSocket) {
33
+ stopPing();
34
+ lastPongRef.current = Date.now();
35
+ pingTimerRef.current = window.setInterval(() => {
36
+ if (ws.readyState === WebSocket.OPEN) {
37
+ ws.send(JSON.stringify({ type: 'ping' }));
38
+ }
39
+ if (Date.now() - lastPongRef.current > 35000) {
40
+ ws.close();
41
+ }
42
+ }, 15000);
43
+ }
44
+
45
+ function stopPing() {
46
+ if (pingTimerRef.current) {
47
+ clearInterval(pingTimerRef.current);
48
+ pingTimerRef.current = null;
49
+ }
50
+ }
51
+
52
+ function connect() {
53
+ setStatus('connecting');
54
+ try {
55
+ const ws = new WebSocket(url);
56
+ wsRef.current = ws;
57
+
58
+ ws.addEventListener('open', () => {
59
+ if (!active) {
60
+ ws.close();
61
+ return;
62
+ }
63
+ setStatus('connected');
64
+ startPing(ws);
65
+ });
66
+
67
+ ws.addEventListener('message', (event) => {
68
+ let msg: ClientWebSocketMessage;
69
+ try {
70
+ msg = JSON.parse(event.data);
71
+ } catch {
72
+ return;
73
+ }
74
+ if ((msg as { type: string }).type === 'pong') {
75
+ lastPongRef.current = Date.now();
76
+ return;
77
+ }
78
+ onMessageRef.current(msg);
79
+ });
80
+
81
+ ws.addEventListener('close', () => {
82
+ setStatus('disconnected');
83
+ stopPing();
84
+ if (active) scheduleReconnect();
85
+ });
86
+
87
+ ws.addEventListener('error', () => {
88
+ ws.close();
89
+ });
90
+ } catch {
91
+ setStatus('disconnected');
92
+ scheduleReconnect();
93
+ }
94
+ }
95
+
96
+ connect();
97
+
98
+ return () => {
99
+ active = false;
100
+ if (reconnectTimerRef.current) {
101
+ clearTimeout(reconnectTimerRef.current);
102
+ reconnectTimerRef.current = null;
103
+ }
104
+ stopPing();
105
+ wsRef.current?.close();
106
+ wsRef.current = null;
107
+ };
108
+ }, [url]);
109
+
110
+ const send = (data: unknown) => {
111
+ const ws = wsRef.current;
112
+ if (ws && ws.readyState === WebSocket.OPEN) {
113
+ ws.send(JSON.stringify(data));
114
+ }
115
+ };
116
+
117
+ return { status, send };
118
+ }
@@ -0,0 +1,33 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { toExportableEvent, toJson, filename } from './export';
3
+ import type { ToolInvocation } from '../../../src/lib/invocations';
4
+
5
+ describe('export', () => {
6
+ it('maps an invocation to an exportable event', () => {
7
+ const inv: ToolInvocation = {
8
+ id: 'e1',
9
+ tool: 'Read',
10
+ eventType: 'tool_end',
11
+ status: 'success',
12
+ startTime: 1000,
13
+ skill: 'engineer',
14
+ detail: 'opened file',
15
+ input: { path: 'src/index.ts' },
16
+ durationMs: 50,
17
+ };
18
+ const ev = toExportableEvent(inv);
19
+ expect(ev.id).toBe('e1');
20
+ expect(ev.tool).toBe('Read');
21
+ expect(ev.path).toBe('src/index.ts');
22
+ expect(ev.durationMs).toBe(50);
23
+ });
24
+
25
+ it('produces valid JSON', () => {
26
+ const blob = toJson([{ id: 'e1', timestamp: 1, eventType: 'tool_end', status: 'success' }]);
27
+ expect(blob.type).toBe('application/json');
28
+ });
29
+
30
+ it('generates a filename with timestamp', () => {
31
+ expect(filename('json')).toMatch(/^crewloop-events-\d{4}-\d{2}-\d{2}-\d{6}\.json$/);
32
+ });
33
+ });
@@ -0,0 +1,39 @@
1
+ import type { ToolInvocation } from '../../../src/lib/invocations';
2
+ import { resolvePath } from '../../../src/lib/paths';
3
+ import type { ExportableEvent } from './types';
4
+
5
+ export function toExportableEvent(inv: ToolInvocation): ExportableEvent {
6
+ return {
7
+ id: inv.id,
8
+ timestamp: inv.startTime,
9
+ tool: inv.tool,
10
+ eventType: inv.eventType,
11
+ status: inv.status,
12
+ skill: inv.skill,
13
+ detail: inv.detail,
14
+ path: resolvePath(inv.input, inv.output),
15
+ durationMs: inv.durationMs,
16
+ };
17
+ }
18
+
19
+ export function toJson(events: ExportableEvent[]): Blob {
20
+ return new Blob([JSON.stringify(events, null, 2)], { type: 'application/json' });
21
+ }
22
+
23
+ export function download(blob: Blob, filename: string): void {
24
+ const url = URL.createObjectURL(blob);
25
+ const a = document.createElement('a');
26
+ a.href = url;
27
+ a.download = filename;
28
+ document.body.appendChild(a);
29
+ a.click();
30
+ document.body.removeChild(a);
31
+ URL.revokeObjectURL(url);
32
+ }
33
+
34
+ export function filename(extension: 'json'): string {
35
+ const now = new Date();
36
+ const pad = (n: number) => n.toString().padStart(2, '0');
37
+ const stamp = `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}-${pad(now.getHours())}${pad(now.getMinutes())}${pad(now.getSeconds())}`;
38
+ return `crewloop-events-${stamp}.${extension}`;
39
+ }
@@ -0,0 +1,95 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { buildOptions, filterInvocations, filterSessions, filterGraph } from './filter';
3
+ import type { AgentSource, ClientSession, EventStatus } from '../../../src/types';
4
+ import type { ToolInvocation } from '../../../src/lib/invocations';
5
+ import type { Graph3D } from '../../../src/lib/graph';
6
+ import { DEFAULT_FILTER_STATE } from './types';
7
+
8
+ function makeSession(id: string, source: ClientSession['source'], overrides?: Partial<ClientSession>): ClientSession {
9
+ return {
10
+ id,
11
+ source,
12
+ lifecycle: 'running',
13
+ events: [],
14
+ startTime: 0,
15
+ lastActivity: 1000,
16
+ toolCounts: {},
17
+ ...overrides,
18
+ } as ClientSession;
19
+ }
20
+
21
+ function makeInv(tool: string, overrides?: Partial<ToolInvocation>): ToolInvocation {
22
+ return {
23
+ id: Math.random().toString(),
24
+ tool,
25
+ eventType: 'tool_end',
26
+ status: 'success',
27
+ startTime: 500,
28
+ ...overrides,
29
+ } as ToolInvocation;
30
+ }
31
+
32
+ describe('filter', () => {
33
+ it('builds options from selected session', () => {
34
+ const session = makeSession('s1', 'kimi', {
35
+ activeSkill: { name: 'engineer', confidence: 'explicit' },
36
+ events: [{ id: 'e1', timestamp: 0, event_type: 'tool_end', tool: 'Read' }],
37
+ });
38
+ const sessions = new Map<string, ClientSession>([['s1', session]]);
39
+ expect(buildOptions(sessions, 's1')).toEqual({
40
+ sources: ['kimi'],
41
+ skills: ['engineer'],
42
+ statuses: [],
43
+ tools: ['Read'],
44
+ opTypes: ['read'],
45
+ });
46
+ });
47
+
48
+ it('filters invocations by tool and status', () => {
49
+ const invs = [
50
+ makeInv('Read', { status: 'success' }),
51
+ makeInv('Edit', { status: 'error' }),
52
+ makeInv('Bash', { status: 'running' }),
53
+ ];
54
+ const filters = { ...DEFAULT_FILTER_STATE, tools: ['Read', 'Edit'], statuses: ['success' as EventStatus] };
55
+ expect(filterInvocations(invs, undefined, filters, 1000)).toHaveLength(1);
56
+ });
57
+
58
+ it('filters invocations by time range', () => {
59
+ const invs = [
60
+ makeInv('Read', { startTime: 64000 }),
61
+ makeInv('Edit', { startTime: 100 }),
62
+ ];
63
+ const filters = { ...DEFAULT_FILTER_STATE, timeRange: '1m' as const };
64
+ expect(filterInvocations(invs, undefined, filters, 65000)).toHaveLength(1);
65
+ });
66
+
67
+ it('filters sessions by source and pin order', () => {
68
+ const a = makeSession('a', 'kimi', { lastActivity: 2000 });
69
+ const b = makeSession('b', 'codex', { lastActivity: 3000 });
70
+ const filters = { ...DEFAULT_FILTER_STATE, sources: ['kimi', 'codex'] as AgentSource[] };
71
+ const result = filterSessions([a, b], filters, [{ id: 'a', pinnedAt: 0 }], 4000);
72
+ expect(result.map((s) => s.id)).toEqual(['a', 'b']);
73
+ });
74
+
75
+ it('filters graph while keeping skill root', () => {
76
+ const graph: Graph3D = {
77
+ nodes: [
78
+ { id: 'skill:eng', type: 'skill', label: 'engineer', weight: 1 },
79
+ { id: 'tool:Read', type: 'tool', label: 'Read', weight: 1 },
80
+ { id: 'tool:Edit', type: 'tool', label: 'Edit', weight: 1 },
81
+ { id: 'file:a', type: 'file', label: 'a', weight: 1 },
82
+ ],
83
+ links: [
84
+ { source: 'skill:eng', target: 'tool:Read', weight: 1 },
85
+ { source: 'skill:eng', target: 'tool:Edit', weight: 1 },
86
+ { source: 'tool:Read', target: 'file:a', weight: 1 },
87
+ ],
88
+ };
89
+ const invs = [makeInv('Read'), makeInv('Edit')];
90
+ const filters = { ...DEFAULT_FILTER_STATE, tools: ['Read'] };
91
+ const result = filterGraph(graph, invs, filters);
92
+ expect(result.nodes.map((n) => n.id).sort()).toEqual(['file:a', 'skill:eng', 'tool:Read']);
93
+ expect(result.links).toHaveLength(2);
94
+ });
95
+ });