@fragments-sdk/cli 0.2.2

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 (259) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +106 -0
  3. package/dist/bin.d.ts +1 -0
  4. package/dist/bin.js +4783 -0
  5. package/dist/bin.js.map +1 -0
  6. package/dist/chunk-4FDQSGKX.js +786 -0
  7. package/dist/chunk-4FDQSGKX.js.map +1 -0
  8. package/dist/chunk-7H2MMGYG.js +369 -0
  9. package/dist/chunk-7H2MMGYG.js.map +1 -0
  10. package/dist/chunk-BSCG3IP7.js +619 -0
  11. package/dist/chunk-BSCG3IP7.js.map +1 -0
  12. package/dist/chunk-LY2CFFPY.js +898 -0
  13. package/dist/chunk-LY2CFFPY.js.map +1 -0
  14. package/dist/chunk-MUZ6CM66.js +6636 -0
  15. package/dist/chunk-MUZ6CM66.js.map +1 -0
  16. package/dist/chunk-OAENNG3G.js +1489 -0
  17. package/dist/chunk-OAENNG3G.js.map +1 -0
  18. package/dist/chunk-XHNKNI6J.js +235 -0
  19. package/dist/chunk-XHNKNI6J.js.map +1 -0
  20. package/dist/core-DWKLGY4N.js +68 -0
  21. package/dist/core-DWKLGY4N.js.map +1 -0
  22. package/dist/generate-4LQNJ7SX.js +249 -0
  23. package/dist/generate-4LQNJ7SX.js.map +1 -0
  24. package/dist/index.d.ts +775 -0
  25. package/dist/index.js +41 -0
  26. package/dist/index.js.map +1 -0
  27. package/dist/init-EMVI47QG.js +416 -0
  28. package/dist/init-EMVI47QG.js.map +1 -0
  29. package/dist/mcp-bin.d.ts +1 -0
  30. package/dist/mcp-bin.js +1117 -0
  31. package/dist/mcp-bin.js.map +1 -0
  32. package/dist/scan-4YPRF7FV.js +12 -0
  33. package/dist/scan-4YPRF7FV.js.map +1 -0
  34. package/dist/service-QSZMZJBJ.js +208 -0
  35. package/dist/service-QSZMZJBJ.js.map +1 -0
  36. package/dist/static-viewer-MIPGZ4Z7.js +12 -0
  37. package/dist/static-viewer-MIPGZ4Z7.js.map +1 -0
  38. package/dist/test-SQ5ZHXWU.js +1067 -0
  39. package/dist/test-SQ5ZHXWU.js.map +1 -0
  40. package/dist/tokens-HSGMYK64.js +173 -0
  41. package/dist/tokens-HSGMYK64.js.map +1 -0
  42. package/dist/viewer-YRF4SQE4.js +11101 -0
  43. package/dist/viewer-YRF4SQE4.js.map +1 -0
  44. package/package.json +107 -0
  45. package/src/ai.ts +266 -0
  46. package/src/analyze.ts +265 -0
  47. package/src/bin.ts +916 -0
  48. package/src/build.ts +248 -0
  49. package/src/commands/a11y.ts +302 -0
  50. package/src/commands/add.ts +313 -0
  51. package/src/commands/audit.ts +195 -0
  52. package/src/commands/baseline.ts +221 -0
  53. package/src/commands/build.ts +144 -0
  54. package/src/commands/compare.ts +337 -0
  55. package/src/commands/context.ts +107 -0
  56. package/src/commands/dev.ts +107 -0
  57. package/src/commands/enhance.ts +858 -0
  58. package/src/commands/generate.ts +391 -0
  59. package/src/commands/init.ts +531 -0
  60. package/src/commands/link/figma.ts +645 -0
  61. package/src/commands/link/index.ts +10 -0
  62. package/src/commands/link/storybook.ts +267 -0
  63. package/src/commands/list.ts +49 -0
  64. package/src/commands/metrics.ts +114 -0
  65. package/src/commands/reset.ts +242 -0
  66. package/src/commands/scan.ts +537 -0
  67. package/src/commands/storygen.ts +207 -0
  68. package/src/commands/tokens.ts +251 -0
  69. package/src/commands/validate.ts +93 -0
  70. package/src/commands/verify.ts +215 -0
  71. package/src/core/composition.test.ts +262 -0
  72. package/src/core/composition.ts +255 -0
  73. package/src/core/config.ts +84 -0
  74. package/src/core/constants.ts +111 -0
  75. package/src/core/context.ts +380 -0
  76. package/src/core/defineSegment.ts +137 -0
  77. package/src/core/discovery.ts +337 -0
  78. package/src/core/figma.ts +263 -0
  79. package/src/core/fragment-types.ts +214 -0
  80. package/src/core/generators/context.ts +389 -0
  81. package/src/core/generators/index.ts +23 -0
  82. package/src/core/generators/registry.ts +364 -0
  83. package/src/core/generators/typescript-extractor.ts +374 -0
  84. package/src/core/importAnalyzer.ts +217 -0
  85. package/src/core/index.ts +149 -0
  86. package/src/core/loader.ts +155 -0
  87. package/src/core/node.ts +63 -0
  88. package/src/core/parser.ts +551 -0
  89. package/src/core/previewLoader.ts +172 -0
  90. package/src/core/schema/fragment.schema.json +189 -0
  91. package/src/core/schema/registry.schema.json +137 -0
  92. package/src/core/schema.ts +182 -0
  93. package/src/core/storyAdapter.test.ts +571 -0
  94. package/src/core/storyAdapter.ts +761 -0
  95. package/src/core/token-types.ts +287 -0
  96. package/src/core/types.ts +754 -0
  97. package/src/diff.ts +323 -0
  98. package/src/index.ts +43 -0
  99. package/src/mcp/__tests__/projectFields.test.ts +130 -0
  100. package/src/mcp/bin.ts +36 -0
  101. package/src/mcp/index.ts +8 -0
  102. package/src/mcp/server.ts +1310 -0
  103. package/src/mcp/utils.ts +54 -0
  104. package/src/mcp-bin.ts +36 -0
  105. package/src/migrate/__tests__/argTypes/argTypes.test.ts +189 -0
  106. package/src/migrate/__tests__/args/args.test.ts +452 -0
  107. package/src/migrate/__tests__/meta/meta.test.ts +198 -0
  108. package/src/migrate/__tests__/stories/stories.test.ts +278 -0
  109. package/src/migrate/__tests__/utils/utils.test.ts +371 -0
  110. package/src/migrate/__tests__/values/values.test.ts +303 -0
  111. package/src/migrate/bin.ts +108 -0
  112. package/src/migrate/converter.ts +658 -0
  113. package/src/migrate/detect.ts +196 -0
  114. package/src/migrate/index.ts +45 -0
  115. package/src/migrate/migrate.ts +163 -0
  116. package/src/migrate/parser.ts +1136 -0
  117. package/src/migrate/report.ts +624 -0
  118. package/src/migrate/types.ts +169 -0
  119. package/src/screenshot.ts +249 -0
  120. package/src/service/__tests__/ast-utils.test.ts +426 -0
  121. package/src/service/__tests__/enhance-scanner.test.ts +200 -0
  122. package/src/service/__tests__/figma/figma.test.ts +652 -0
  123. package/src/service/__tests__/metrics-store.test.ts +409 -0
  124. package/src/service/__tests__/patch-generator.test.ts +186 -0
  125. package/src/service/__tests__/props-extractor.test.ts +365 -0
  126. package/src/service/__tests__/token-registry.test.ts +267 -0
  127. package/src/service/analytics.ts +659 -0
  128. package/src/service/ast-utils.ts +444 -0
  129. package/src/service/browser-pool.ts +339 -0
  130. package/src/service/capture.ts +267 -0
  131. package/src/service/diff.ts +279 -0
  132. package/src/service/enhance/aggregator.ts +489 -0
  133. package/src/service/enhance/cache.ts +275 -0
  134. package/src/service/enhance/codebase-scanner.ts +357 -0
  135. package/src/service/enhance/context-generator.ts +529 -0
  136. package/src/service/enhance/doc-extractor.ts +523 -0
  137. package/src/service/enhance/index.ts +131 -0
  138. package/src/service/enhance/props-extractor.ts +665 -0
  139. package/src/service/enhance/scanner.ts +445 -0
  140. package/src/service/enhance/storybook-parser.ts +552 -0
  141. package/src/service/enhance/types.ts +346 -0
  142. package/src/service/enhance/variant-renderer.ts +479 -0
  143. package/src/service/figma.ts +1008 -0
  144. package/src/service/index.ts +249 -0
  145. package/src/service/metrics-store.ts +333 -0
  146. package/src/service/patch-generator.ts +349 -0
  147. package/src/service/report.ts +854 -0
  148. package/src/service/storage.ts +401 -0
  149. package/src/service/token-fixes.ts +281 -0
  150. package/src/service/token-parser.ts +504 -0
  151. package/src/service/token-registry.ts +721 -0
  152. package/src/service/utils.ts +172 -0
  153. package/src/setup.ts +241 -0
  154. package/src/shared/command-wrapper.ts +81 -0
  155. package/src/shared/dev-server-client.ts +199 -0
  156. package/src/shared/index.ts +8 -0
  157. package/src/shared/segment-loader.ts +59 -0
  158. package/src/shared/types.ts +147 -0
  159. package/src/static-viewer.ts +715 -0
  160. package/src/test/discovery.ts +172 -0
  161. package/src/test/index.ts +281 -0
  162. package/src/test/reporters/console.ts +194 -0
  163. package/src/test/reporters/json.ts +190 -0
  164. package/src/test/reporters/junit.ts +186 -0
  165. package/src/test/runner.ts +598 -0
  166. package/src/test/types.ts +245 -0
  167. package/src/test/watch.ts +200 -0
  168. package/src/validators.ts +152 -0
  169. package/src/viewer/__tests__/jsx-parser.test.ts +502 -0
  170. package/src/viewer/__tests__/render-utils.test.ts +232 -0
  171. package/src/viewer/__tests__/style-utils.test.ts +404 -0
  172. package/src/viewer/bin.ts +86 -0
  173. package/src/viewer/cli/health.ts +256 -0
  174. package/src/viewer/cli/index.ts +33 -0
  175. package/src/viewer/cli/scan.ts +124 -0
  176. package/src/viewer/cli/utils.ts +174 -0
  177. package/src/viewer/components/AccessibilityPanel.tsx +1404 -0
  178. package/src/viewer/components/ActionCapture.tsx +172 -0
  179. package/src/viewer/components/ActionsPanel.tsx +371 -0
  180. package/src/viewer/components/App.tsx +638 -0
  181. package/src/viewer/components/BottomPanel.tsx +224 -0
  182. package/src/viewer/components/CodePanel.tsx +589 -0
  183. package/src/viewer/components/CommandPalette.tsx +336 -0
  184. package/src/viewer/components/ComponentGraph.tsx +394 -0
  185. package/src/viewer/components/ComponentHeader.tsx +85 -0
  186. package/src/viewer/components/ContractPanel.tsx +234 -0
  187. package/src/viewer/components/ErrorBoundary.tsx +85 -0
  188. package/src/viewer/components/FigmaEmbed.tsx +231 -0
  189. package/src/viewer/components/FragmentEditor.tsx +485 -0
  190. package/src/viewer/components/HealthDashboard.tsx +452 -0
  191. package/src/viewer/components/HmrStatusIndicator.tsx +71 -0
  192. package/src/viewer/components/Icons.tsx +417 -0
  193. package/src/viewer/components/InteractionsPanel.tsx +720 -0
  194. package/src/viewer/components/IsolatedPreviewFrame.tsx +321 -0
  195. package/src/viewer/components/IsolatedRender.tsx +111 -0
  196. package/src/viewer/components/KeyboardShortcutsHelp.tsx +89 -0
  197. package/src/viewer/components/LandingPage.tsx +441 -0
  198. package/src/viewer/components/Layout.tsx +22 -0
  199. package/src/viewer/components/LeftSidebar.tsx +391 -0
  200. package/src/viewer/components/MultiViewportPreview.tsx +429 -0
  201. package/src/viewer/components/PreviewArea.tsx +404 -0
  202. package/src/viewer/components/PreviewFrameHost.tsx +310 -0
  203. package/src/viewer/components/PreviewPane.tsx +150 -0
  204. package/src/viewer/components/PreviewToolbar.tsx +176 -0
  205. package/src/viewer/components/PropsEditor.tsx +512 -0
  206. package/src/viewer/components/PropsTable.tsx +98 -0
  207. package/src/viewer/components/RelationsSection.tsx +57 -0
  208. package/src/viewer/components/ResizablePanel.tsx +328 -0
  209. package/src/viewer/components/RightSidebar.tsx +118 -0
  210. package/src/viewer/components/ScreenshotButton.tsx +90 -0
  211. package/src/viewer/components/Sidebar.tsx +169 -0
  212. package/src/viewer/components/SkeletonLoader.tsx +156 -0
  213. package/src/viewer/components/StoryRenderer.tsx +128 -0
  214. package/src/viewer/components/ThemeProvider.tsx +96 -0
  215. package/src/viewer/components/Toast.tsx +67 -0
  216. package/src/viewer/components/TokenStylePanel.tsx +708 -0
  217. package/src/viewer/components/UsageSection.tsx +95 -0
  218. package/src/viewer/components/VariantMatrix.tsx +350 -0
  219. package/src/viewer/components/VariantRenderer.tsx +131 -0
  220. package/src/viewer/components/VariantTabs.tsx +84 -0
  221. package/src/viewer/components/ViewportSelector.tsx +165 -0
  222. package/src/viewer/components/_future/CreatePage.tsx +836 -0
  223. package/src/viewer/composition-renderer.ts +381 -0
  224. package/src/viewer/constants/index.ts +1 -0
  225. package/src/viewer/constants/ui.ts +185 -0
  226. package/src/viewer/entry.tsx +299 -0
  227. package/src/viewer/hooks/index.ts +2 -0
  228. package/src/viewer/hooks/useA11yCache.ts +383 -0
  229. package/src/viewer/hooks/useA11yService.ts +498 -0
  230. package/src/viewer/hooks/useActions.ts +138 -0
  231. package/src/viewer/hooks/useAppState.ts +124 -0
  232. package/src/viewer/hooks/useFigmaIntegration.ts +132 -0
  233. package/src/viewer/hooks/useHmrStatus.ts +109 -0
  234. package/src/viewer/hooks/useKeyboardShortcuts.ts +222 -0
  235. package/src/viewer/hooks/usePreviewBridge.ts +347 -0
  236. package/src/viewer/hooks/useScrollSpy.ts +78 -0
  237. package/src/viewer/hooks/useUrlState.ts +330 -0
  238. package/src/viewer/hooks/useViewSettings.ts +125 -0
  239. package/src/viewer/index.html +28 -0
  240. package/src/viewer/index.ts +14 -0
  241. package/src/viewer/intelligence/healthReport.ts +505 -0
  242. package/src/viewer/intelligence/styleDrift.ts +340 -0
  243. package/src/viewer/intelligence/usageScanner.ts +309 -0
  244. package/src/viewer/jsx-parser.ts +485 -0
  245. package/src/viewer/postcss.config.js +6 -0
  246. package/src/viewer/preview-frame-entry.tsx +25 -0
  247. package/src/viewer/preview-frame.html +109 -0
  248. package/src/viewer/render-template.html +68 -0
  249. package/src/viewer/render-utils.ts +170 -0
  250. package/src/viewer/server.ts +276 -0
  251. package/src/viewer/style-utils.ts +414 -0
  252. package/src/viewer/styles/globals.css +355 -0
  253. package/src/viewer/tailwind.config.js +37 -0
  254. package/src/viewer/types/a11y.ts +197 -0
  255. package/src/viewer/utils/a11y-fixes.ts +471 -0
  256. package/src/viewer/utils/actionExport.ts +372 -0
  257. package/src/viewer/utils/colorSchemes.ts +201 -0
  258. package/src/viewer/utils/detectRelationships.ts +256 -0
  259. package/src/viewer/vite-plugin.ts +2143 -0
@@ -0,0 +1,391 @@
1
+ import { useState, useMemo, useRef, useEffect, useCallback } from 'react';
2
+ import type { SegmentDefinition } from '../../core/index.js';
3
+ import { BRAND } from '../../core/index.js';
4
+ import { useTheme } from './ThemeProvider.js';
5
+ import clsx from 'clsx';
6
+ import { SearchIcon, SunIcon, MoonIcon, DashboardIcon } from './Icons.js';
7
+
8
+ // Fuzzy matching utility
9
+ interface FuzzyMatch {
10
+ score: number;
11
+ indices: number[];
12
+ }
13
+
14
+ function fuzzyMatch(text: string, pattern: string): FuzzyMatch | null {
15
+ if (!pattern) return { score: 0, indices: [] };
16
+
17
+ const textLower = text.toLowerCase();
18
+ const patternLower = pattern.toLowerCase();
19
+
20
+ const indices: number[] = [];
21
+ let patternIdx = 0;
22
+ let score = 0;
23
+ let consecutiveBonus = 0;
24
+
25
+ for (let i = 0; i < textLower.length && patternIdx < patternLower.length; i++) {
26
+ if (textLower[i] === patternLower[patternIdx]) {
27
+ indices.push(i);
28
+ if (indices.length > 1 && indices[indices.length - 2] === i - 1) {
29
+ consecutiveBonus += 5;
30
+ }
31
+ if (i === 0 || text[i - 1] === ' ' || text[i - 1] === '-' || text[i - 1] === '_') {
32
+ score += 10;
33
+ }
34
+ patternIdx++;
35
+ }
36
+ }
37
+
38
+ if (patternIdx !== patternLower.length) {
39
+ return null;
40
+ }
41
+
42
+ score += consecutiveBonus;
43
+ score += (patternLower.length / textLower.length) * 20;
44
+
45
+ return { score, indices };
46
+ }
47
+
48
+ interface SearchResult {
49
+ item: { path: string; segment: SegmentDefinition };
50
+ score: number;
51
+ nameIndices: number[];
52
+ }
53
+
54
+ function searchSegment(
55
+ item: { path: string; segment: SegmentDefinition },
56
+ query: string
57
+ ): SearchResult | null {
58
+ const { segment } = item;
59
+ // Skip invalid segments
60
+ if (!segment?.meta) return null;
61
+ const { name, category, tags } = segment.meta;
62
+
63
+ const nameMatch = fuzzyMatch(name, query);
64
+ if (nameMatch) {
65
+ return { item, score: nameMatch.score + 100, nameIndices: nameMatch.indices };
66
+ }
67
+
68
+ const categoryMatch = fuzzyMatch(category, query);
69
+ if (categoryMatch) {
70
+ return { item, score: categoryMatch.score + 50, nameIndices: [] };
71
+ }
72
+
73
+ if (tags) {
74
+ for (const tag of tags) {
75
+ const tagMatch = fuzzyMatch(tag, query);
76
+ if (tagMatch) {
77
+ return { item, score: tagMatch.score + 25, nameIndices: [] };
78
+ }
79
+ }
80
+ }
81
+
82
+ return null;
83
+ }
84
+
85
+ function useDebounce<T>(value: T, delay: number): T {
86
+ const [debouncedValue, setDebouncedValue] = useState(value);
87
+
88
+ useEffect(() => {
89
+ const timer = setTimeout(() => setDebouncedValue(value), delay);
90
+ return () => clearTimeout(timer);
91
+ }, [value, delay]);
92
+
93
+ return debouncedValue;
94
+ }
95
+
96
+ function HighlightedText({ text, indices }: { text: string; indices: number[] }) {
97
+ if (indices.length === 0) return <>{text}</>;
98
+
99
+ const result: React.ReactNode[] = [];
100
+ let lastIndex = 0;
101
+
102
+ for (let i = 0; i < indices.length; i++) {
103
+ const matchIndex = indices[i];
104
+ if (matchIndex > lastIndex) {
105
+ result.push(text.slice(lastIndex, matchIndex));
106
+ }
107
+ result.push(
108
+ <span key={matchIndex} className="text-primary font-medium">
109
+ {text[matchIndex]}
110
+ </span>
111
+ );
112
+ lastIndex = matchIndex + 1;
113
+ }
114
+
115
+ if (lastIndex < text.length) {
116
+ result.push(text.slice(lastIndex));
117
+ }
118
+
119
+ return <>{result}</>;
120
+ }
121
+
122
+ interface LeftSidebarProps {
123
+ segments: Array<{ path: string; segment: SegmentDefinition }>;
124
+ activeSegment: string | null;
125
+ onSelect: (path: string) => void;
126
+ showHealth?: boolean;
127
+ onHealthClick?: () => void;
128
+ }
129
+
130
+ export function LeftSidebar({ segments, activeSegment, onSelect, showHealth, onHealthClick }: LeftSidebarProps) {
131
+ const [search, setSearch] = useState('');
132
+ const [focusedIndex, setFocusedIndex] = useState(-1);
133
+ const { theme, setTheme, resolvedTheme } = useTheme();
134
+ const searchInputRef = useRef<HTMLInputElement>(null);
135
+ const itemRefs = useRef<Map<number, HTMLButtonElement>>(new Map());
136
+
137
+ const debouncedSearch = useDebounce(search, 150);
138
+
139
+ const searchResults = useMemo(() => {
140
+ if (!debouncedSearch) return null;
141
+
142
+ const results: SearchResult[] = [];
143
+ for (const item of segments) {
144
+ const result = searchSegment(item, debouncedSearch);
145
+ if (result) results.push(result);
146
+ }
147
+
148
+ results.sort((a, b) => b.score - a.score);
149
+ return results;
150
+ }, [segments, debouncedSearch]);
151
+
152
+ const highlightMap = useMemo(() => {
153
+ const map = new Map<string, number[]>();
154
+ if (searchResults) {
155
+ for (const result of searchResults) {
156
+ map.set(result.item.path, result.nameIndices);
157
+ }
158
+ }
159
+ return map;
160
+ }, [searchResults]);
161
+
162
+ const grouped = useMemo(() => {
163
+ const source = searchResults
164
+ ? searchResults.map(r => r.item)
165
+ : segments;
166
+
167
+ const groups: Record<string, typeof segments> = {};
168
+ for (const item of source) {
169
+ // Skip invalid segments
170
+ if (!item.segment?.meta) continue;
171
+ const category = item.segment.meta.category || 'uncategorized';
172
+ if (!groups[category]) groups[category] = [];
173
+ groups[category].push(item);
174
+ }
175
+ return groups;
176
+ }, [segments, searchResults]);
177
+
178
+ const toggleTheme = () => {
179
+ // Simple toggle between light and dark
180
+ setTheme(resolvedTheme === 'light' ? 'dark' : 'light');
181
+ };
182
+
183
+ const flatItems = useMemo(() => {
184
+ const items: Array<{ path: string; segment: SegmentDefinition }> = [];
185
+ const sortedEntries = Object.entries(grouped).sort(([a], [b]) =>
186
+ a.toLowerCase().localeCompare(b.toLowerCase())
187
+ );
188
+ for (const [, categoryItems] of sortedEntries) {
189
+ const sorted = [...categoryItems]
190
+ .filter(item => item.segment?.meta?.name)
191
+ .sort((a, b) =>
192
+ a.segment.meta.name.toLowerCase().localeCompare(b.segment.meta.name.toLowerCase())
193
+ );
194
+ items.push(...sorted);
195
+ }
196
+ return items;
197
+ }, [grouped]);
198
+
199
+ useEffect(() => {
200
+ if (focusedIndex >= 0 && focusedIndex < flatItems.length) {
201
+ itemRefs.current.get(focusedIndex)?.focus();
202
+ }
203
+ }, [focusedIndex, flatItems.length]);
204
+
205
+ useEffect(() => {
206
+ setFocusedIndex(-1);
207
+ }, [search]);
208
+
209
+ const handleKeyDown = useCallback((e: KeyboardEvent) => {
210
+ const target = e.target as HTMLElement;
211
+ if ((target.tagName === 'INPUT' && target !== searchInputRef.current) ||
212
+ target.tagName === 'TEXTAREA' || target.isContentEditable) {
213
+ return;
214
+ }
215
+
216
+ if ((e.key === '/' && !e.metaKey && !e.ctrlKey) ||
217
+ (e.key === 'k' && (e.metaKey || e.ctrlKey))) {
218
+ e.preventDefault();
219
+ searchInputRef.current?.focus();
220
+ searchInputRef.current?.select();
221
+ return;
222
+ }
223
+
224
+ if (e.key === 'Escape') {
225
+ if (document.activeElement === searchInputRef.current) {
226
+ setSearch('');
227
+ searchInputRef.current?.blur();
228
+ if (flatItems.length > 0) setFocusedIndex(0);
229
+ } else {
230
+ setSearch('');
231
+ setFocusedIndex(-1);
232
+ }
233
+ return;
234
+ }
235
+
236
+ if (document.activeElement === searchInputRef.current) return;
237
+
238
+ if (e.key === 'ArrowDown') {
239
+ e.preventDefault();
240
+ setFocusedIndex(prev => (prev + 1) >= flatItems.length ? 0 : prev + 1);
241
+ } else if (e.key === 'ArrowUp') {
242
+ e.preventDefault();
243
+ setFocusedIndex(prev => (prev - 1) < 0 ? flatItems.length - 1 : prev - 1);
244
+ } else if (e.key === 'Enter' && focusedIndex >= 0 && focusedIndex < flatItems.length) {
245
+ e.preventDefault();
246
+ onSelect(flatItems[focusedIndex].path);
247
+ }
248
+ }, [flatItems, focusedIndex, onSelect]);
249
+
250
+ useEffect(() => {
251
+ document.addEventListener('keydown', handleKeyDown);
252
+ return () => document.removeEventListener('keydown', handleKeyDown);
253
+ }, [handleKeyDown]);
254
+
255
+ const setItemRef = useCallback((index: number, el: HTMLButtonElement | null) => {
256
+ if (el) itemRefs.current.set(index, el);
257
+ else itemRefs.current.delete(index);
258
+ }, []);
259
+
260
+ let globalIndex = 0;
261
+ const sortedEntries = Object.entries(grouped).sort(([a], [b]) =>
262
+ a.toLowerCase().localeCompare(b.toLowerCase())
263
+ );
264
+
265
+ return (
266
+ <div className="flex flex-col h-full bg-[--bg-secondary]">
267
+ {/* Header - matches toolbar height (py-2 + border) */}
268
+ <div className="flex items-center justify-between px-4 py-2 border-b border-[--border]">
269
+ <span className="text-sm font-medium text-primary">{BRAND.name}</span>
270
+ <button
271
+ onClick={toggleTheme}
272
+ className={clsx(
273
+ 'p-1.5 rounded-md',
274
+ 'text-tertiary hover:text-secondary hover:bg-[--bg-hover]',
275
+ 'focus:outline-none'
276
+ )}
277
+ title={`Theme: ${resolvedTheme}`}
278
+ >
279
+ {resolvedTheme === 'dark' ? (
280
+ <MoonIcon className="w-4 h-4" />
281
+ ) : (
282
+ <SunIcon className="w-4 h-4" />
283
+ )}
284
+ </button>
285
+ </div>
286
+
287
+ {/* Search */}
288
+ <div className="px-3 py-3">
289
+ <div className="relative">
290
+ <SearchIcon className="absolute left-2.5 top-1/2 -translate-y-1/2 w-3.5 h-3.5 text-tertiary pointer-events-none" />
291
+ <input
292
+ ref={searchInputRef}
293
+ type="text"
294
+ placeholder="Search"
295
+ value={search}
296
+ onChange={(e) => setSearch(e.target.value)}
297
+ className={clsx(
298
+ 'w-full pl-8 pr-3 py-1.5 text-sm rounded-md',
299
+ 'bg-[--bg-primary] text-primary placeholder:text-tertiary',
300
+ 'border border-[--border-subtle]',
301
+ 'focus:outline-none focus:border-[--border-strong]'
302
+ )}
303
+ />
304
+ </div>
305
+ </div>
306
+
307
+ {/* Dashboard Link */}
308
+ {onHealthClick && (
309
+ <div className="px-2 pb-3">
310
+ <button
311
+ onClick={onHealthClick}
312
+ className={clsx(
313
+ 'w-full flex items-center gap-2 px-2 py-2 rounded-md text-sm',
314
+ 'focus:outline-none',
315
+ showHealth
316
+ ? 'bg-[--bg-hover] text-primary'
317
+ : 'text-secondary hover:bg-[--bg-hover] hover:text-primary'
318
+ )}
319
+ >
320
+ <DashboardIcon className="w-4 h-4" />
321
+ <span>Dashboard</span>
322
+ </button>
323
+ </div>
324
+ )}
325
+
326
+ {/* Component list */}
327
+ <nav className="flex-1 overflow-y-auto px-2 pb-4">
328
+ {sortedEntries.map(([category, items]) => {
329
+ const sortedItems = [...items].sort((a, b) =>
330
+ a.segment.meta.name.toLowerCase().localeCompare(b.segment.meta.name.toLowerCase())
331
+ );
332
+
333
+ return (
334
+ <div key={category} className="mb-1">
335
+ {/* Category divider */}
336
+ <div className="px-2 py-1.5 text-xs font-medium text-tertiary mt-2 first:mt-0">
337
+ {category}
338
+ </div>
339
+
340
+ {/* Items */}
341
+ <div className="space-y-0.5">
342
+ {sortedItems.map((item) => {
343
+ const isActive = activeSegment === item.path;
344
+ const currentIndex = globalIndex++;
345
+ const isFocused = focusedIndex === currentIndex;
346
+ const nameIndices = highlightMap.get(item.path) || [];
347
+
348
+ return (
349
+ <button
350
+ key={item.path}
351
+ ref={(el) => setItemRef(currentIndex, el)}
352
+ onClick={() => onSelect(item.path)}
353
+ onFocus={() => setFocusedIndex(currentIndex)}
354
+ className={clsx(
355
+ 'w-full text-left px-2 py-1.5 rounded-md text-sm',
356
+ 'focus:outline-none',
357
+ isActive
358
+ ? 'bg-[--bg-hover] text-primary'
359
+ : isFocused
360
+ ? 'bg-[--bg-hover] text-primary'
361
+ : 'text-secondary hover:bg-[--bg-hover] hover:text-primary'
362
+ )}
363
+ >
364
+ <HighlightedText
365
+ text={item.segment.meta.name}
366
+ indices={nameIndices}
367
+ />
368
+ </button>
369
+ );
370
+ })}
371
+ </div>
372
+ </div>
373
+ );
374
+ })}
375
+
376
+ {Object.keys(grouped).length === 0 && (
377
+ <div className="px-2 py-8 text-center text-tertiary text-sm">
378
+ No results
379
+ </div>
380
+ )}
381
+ </nav>
382
+
383
+ {/* Footer */}
384
+ <div className="px-4 py-3 border-t border-[--border-subtle]">
385
+ <div className="text-xs text-tertiary">
386
+ {segments.length} components
387
+ </div>
388
+ </div>
389
+ </div>
390
+ );
391
+ }