@imjp/writenex-astro 0.1.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 (141) hide show
  1. package/README.md +539 -0
  2. package/dist/chunk-5PM6EQE5.js +151 -0
  3. package/dist/chunk-5PM6EQE5.js.map +1 -0
  4. package/dist/chunk-7XU5X6CW.js +1331 -0
  5. package/dist/chunk-7XU5X6CW.js.map +1 -0
  6. package/dist/chunk-AAOQHQPU.js +574 -0
  7. package/dist/chunk-AAOQHQPU.js.map +1 -0
  8. package/dist/chunk-CF2XXJFF.js +1410 -0
  9. package/dist/chunk-CF2XXJFF.js.map +1 -0
  10. package/dist/chunk-CRPZUUDU.js +52 -0
  11. package/dist/chunk-CRPZUUDU.js.map +1 -0
  12. package/dist/chunk-CYLDJ3HZ.js +310 -0
  13. package/dist/chunk-CYLDJ3HZ.js.map +1 -0
  14. package/dist/chunk-KIKIPIFA.js +1 -0
  15. package/dist/chunk-KIKIPIFA.js.map +1 -0
  16. package/dist/chunk-XNTQTTJU.js +145 -0
  17. package/dist/chunk-XNTQTTJU.js.map +1 -0
  18. package/dist/client/index.css +2 -0
  19. package/dist/client/index.css.map +1 -0
  20. package/dist/client/index.js +375 -0
  21. package/dist/client/index.js.map +1 -0
  22. package/dist/client/styles.css +584 -0
  23. package/dist/client/variables.css +304 -0
  24. package/dist/config/index.d.ts +54 -0
  25. package/dist/config/index.js +38 -0
  26. package/dist/config/index.js.map +1 -0
  27. package/dist/config-BmEdBDo_.d.ts +220 -0
  28. package/dist/content-BWR52vD-.d.ts +64 -0
  29. package/dist/discovery/index.d.ts +310 -0
  30. package/dist/discovery/index.js +38 -0
  31. package/dist/discovery/index.js.map +1 -0
  32. package/dist/errors-C0iYiDTv.d.ts +107 -0
  33. package/dist/filesystem/index.d.ts +1292 -0
  34. package/dist/filesystem/index.js +203 -0
  35. package/dist/filesystem/index.js.map +1 -0
  36. package/dist/image-FP7w5ZIs.d.ts +47 -0
  37. package/dist/index.d.ts +64 -0
  38. package/dist/index.js +151 -0
  39. package/dist/index.js.map +1 -0
  40. package/dist/loader-55LWCXHA.js +12 -0
  41. package/dist/loader-55LWCXHA.js.map +1 -0
  42. package/dist/loader-CrdnaAWR.d.ts +327 -0
  43. package/dist/server/index.d.ts +357 -0
  44. package/dist/server/index.js +37 -0
  45. package/dist/server/index.js.map +1 -0
  46. package/package.json +94 -0
  47. package/src/client/App.tsx +900 -0
  48. package/src/client/components/ConfigPanel/ConfigPanel.css +553 -0
  49. package/src/client/components/ConfigPanel/ConfigPanel.tsx +396 -0
  50. package/src/client/components/ConfigPanel/index.ts +6 -0
  51. package/src/client/components/CreateContentModal/CreateContentModal.css +327 -0
  52. package/src/client/components/CreateContentModal/CreateContentModal.tsx +216 -0
  53. package/src/client/components/CreateContentModal/index.ts +7 -0
  54. package/src/client/components/Editor/Editor.css +885 -0
  55. package/src/client/components/Editor/Editor.tsx +484 -0
  56. package/src/client/components/Editor/ImageDialog.css +344 -0
  57. package/src/client/components/Editor/ImageDialog.tsx +367 -0
  58. package/src/client/components/Editor/LinkDialog.css +326 -0
  59. package/src/client/components/Editor/LinkDialog.tsx +332 -0
  60. package/src/client/components/Editor/index.ts +6 -0
  61. package/src/client/components/FrontmatterForm/FrontmatterForm.css +468 -0
  62. package/src/client/components/FrontmatterForm/FrontmatterForm.tsx +914 -0
  63. package/src/client/components/FrontmatterForm/index.ts +7 -0
  64. package/src/client/components/Header/Header.css +300 -0
  65. package/src/client/components/Header/Header.tsx +300 -0
  66. package/src/client/components/Header/index.ts +7 -0
  67. package/src/client/components/KeyboardShortcuts/KeyboardShortcuts.css +239 -0
  68. package/src/client/components/KeyboardShortcuts/KeyboardShortcuts.tsx +151 -0
  69. package/src/client/components/KeyboardShortcuts/index.ts +6 -0
  70. package/src/client/components/LazyEditor.tsx +75 -0
  71. package/src/client/components/LiveRegion/LiveRegion.css +19 -0
  72. package/src/client/components/LiveRegion/LiveRegion.tsx +60 -0
  73. package/src/client/components/LiveRegion/index.ts +7 -0
  74. package/src/client/components/SearchReplace/SearchReplacePanel.css +300 -0
  75. package/src/client/components/SearchReplace/SearchReplacePanel.tsx +332 -0
  76. package/src/client/components/SearchReplace/index.ts +7 -0
  77. package/src/client/components/SelectCollectionModal/SelectCollectionModal.css +308 -0
  78. package/src/client/components/SelectCollectionModal/SelectCollectionModal.tsx +223 -0
  79. package/src/client/components/SelectCollectionModal/index.ts +7 -0
  80. package/src/client/components/Sidebar/Sidebar.css +570 -0
  81. package/src/client/components/Sidebar/Sidebar.tsx +617 -0
  82. package/src/client/components/Sidebar/index.ts +7 -0
  83. package/src/client/components/SkipLink/SkipLink.css +51 -0
  84. package/src/client/components/SkipLink/SkipLink.tsx +67 -0
  85. package/src/client/components/SkipLink/index.ts +7 -0
  86. package/src/client/components/UnsavedChangesModal/UnsavedChangesModal.css +233 -0
  87. package/src/client/components/UnsavedChangesModal/UnsavedChangesModal.tsx +160 -0
  88. package/src/client/components/UnsavedChangesModal/index.ts +1 -0
  89. package/src/client/components/VersionHistory/DiffViewer.css +430 -0
  90. package/src/client/components/VersionHistory/DiffViewer.tsx +383 -0
  91. package/src/client/components/VersionHistory/VersionActions.css +318 -0
  92. package/src/client/components/VersionHistory/VersionActions.tsx +277 -0
  93. package/src/client/components/VersionHistory/VersionHistoryPanel.css +369 -0
  94. package/src/client/components/VersionHistory/VersionHistoryPanel.tsx +469 -0
  95. package/src/client/components/VersionHistory/index.ts +9 -0
  96. package/src/client/context/ApiContext.tsx +154 -0
  97. package/src/client/context/ThemeContext.tsx +172 -0
  98. package/src/client/hooks/useAnnounce.ts +201 -0
  99. package/src/client/hooks/useApi.ts +374 -0
  100. package/src/client/hooks/useArrowNavigation.ts +286 -0
  101. package/src/client/hooks/useAutosave.ts +241 -0
  102. package/src/client/hooks/useFocusTrap.ts +178 -0
  103. package/src/client/hooks/useKeyboardShortcuts.ts +203 -0
  104. package/src/client/hooks/useSearch.ts +206 -0
  105. package/src/client/hooks/useVersionHistory.ts +451 -0
  106. package/src/client/index.tsx +70 -0
  107. package/src/client/styles.css +584 -0
  108. package/src/client/utils/focus.ts +57 -0
  109. package/src/client/utils/openInEditor.ts +130 -0
  110. package/src/client/variables.css +304 -0
  111. package/src/config/defaults.ts +109 -0
  112. package/src/config/index.ts +32 -0
  113. package/src/config/loader.ts +174 -0
  114. package/src/config/schema.ts +161 -0
  115. package/src/core/constants.ts +39 -0
  116. package/src/core/errors.ts +739 -0
  117. package/src/core/index.ts +11 -0
  118. package/src/discovery/collections.ts +216 -0
  119. package/src/discovery/index.ts +33 -0
  120. package/src/discovery/patterns.ts +702 -0
  121. package/src/discovery/schema.ts +453 -0
  122. package/src/filesystem/images.ts +798 -0
  123. package/src/filesystem/index.ts +107 -0
  124. package/src/filesystem/reader.ts +452 -0
  125. package/src/filesystem/version-config.ts +390 -0
  126. package/src/filesystem/versions.ts +1339 -0
  127. package/src/filesystem/watcher.ts +226 -0
  128. package/src/filesystem/writer.ts +540 -0
  129. package/src/index.ts +61 -0
  130. package/src/integration.ts +228 -0
  131. package/src/server/assets.ts +254 -0
  132. package/src/server/cache.ts +355 -0
  133. package/src/server/index.ts +33 -0
  134. package/src/server/middleware.ts +209 -0
  135. package/src/server/routes.ts +1428 -0
  136. package/src/types/api.ts +61 -0
  137. package/src/types/config.ts +134 -0
  138. package/src/types/content.ts +64 -0
  139. package/src/types/image.ts +48 -0
  140. package/src/types/index.ts +58 -0
  141. package/src/types/version.ts +117 -0
@@ -0,0 +1,308 @@
1
+ /**
2
+ * @fileoverview SelectCollectionModal styles
3
+ *
4
+ * Modal styling for collection selection dialog.
5
+ */
6
+
7
+ /* ============================================================================
8
+ MODAL OVERLAY
9
+ ============================================================================ */
10
+
11
+ .wn-select-collection-overlay {
12
+ position: fixed;
13
+ inset: 0;
14
+ z-index: var(--wn-z-modal);
15
+ display: flex;
16
+ align-items: center;
17
+ justify-content: center;
18
+ background-color: var(--wn-backdrop);
19
+ animation: wn-select-collection-fadeIn var(--wn-transition-fast) ease-out;
20
+ }
21
+
22
+ @keyframes wn-select-collection-fadeIn {
23
+ from {
24
+ opacity: 0;
25
+ }
26
+ to {
27
+ opacity: 1;
28
+ }
29
+ }
30
+
31
+ /* ============================================================================
32
+ MODAL CONTAINER
33
+ ============================================================================ */
34
+
35
+ .wn-select-collection-modal {
36
+ width: 100%;
37
+ max-width: var(--wn-modal-sm);
38
+ max-height: 80vh;
39
+ overflow: hidden;
40
+ border-radius: var(--wn-radius-lg);
41
+ border: 1px solid var(--wn-zinc-700);
42
+ background-color: var(--wn-zinc-900);
43
+ display: flex;
44
+ flex-direction: column;
45
+ animation: wn-select-collection-zoomIn var(--wn-transition-fast) ease-out;
46
+ }
47
+
48
+ @keyframes wn-select-collection-zoomIn {
49
+ from {
50
+ opacity: 0;
51
+ transform: scale(0.95) translateY(-10px);
52
+ }
53
+ to {
54
+ opacity: 1;
55
+ transform: scale(1) translateY(0);
56
+ }
57
+ }
58
+
59
+ /* ============================================================================
60
+ MODAL HEADER
61
+ ============================================================================ */
62
+
63
+ .wn-select-collection-header {
64
+ display: flex;
65
+ align-items: center;
66
+ justify-content: space-between;
67
+ padding: var(--wn-space-5) var(--wn-space-6);
68
+ border-bottom: 1px solid var(--wn-zinc-700);
69
+ flex-shrink: 0;
70
+ }
71
+
72
+ .wn-select-collection-header-content {
73
+ display: flex;
74
+ align-items: center;
75
+ gap: var(--wn-space-4);
76
+ }
77
+
78
+ .wn-select-collection-icon {
79
+ color: var(--wn-brand-500);
80
+ }
81
+
82
+ .wn-select-collection-title {
83
+ font-size: var(--wn-font-md);
84
+ font-weight: 600;
85
+ color: var(--wn-zinc-50);
86
+ margin: 0;
87
+ }
88
+
89
+ .wn-select-collection-close {
90
+ display: flex;
91
+ align-items: center;
92
+ justify-content: center;
93
+ width: var(--wn-icon-btn-md);
94
+ height: var(--wn-icon-btn-md);
95
+ padding: 0;
96
+ border: none;
97
+ border-radius: var(--wn-radius-md);
98
+ background-color: transparent;
99
+ color: var(--wn-zinc-400);
100
+ cursor: pointer;
101
+ transition:
102
+ background-color var(--wn-transition-fast),
103
+ color var(--wn-transition-fast);
104
+ }
105
+
106
+ .wn-select-collection-close:hover {
107
+ background-color: var(--wn-overlay-10);
108
+ color: var(--wn-zinc-50);
109
+ }
110
+
111
+ /* ============================================================================
112
+ MODAL BODY
113
+ ============================================================================ */
114
+
115
+ .wn-select-collection-body {
116
+ padding: var(--wn-space-5) var(--wn-space-6);
117
+ flex: 1;
118
+ overflow-y: auto;
119
+ }
120
+
121
+ .wn-select-collection-hint {
122
+ font-size: var(--wn-font-base);
123
+ color: var(--wn-zinc-500);
124
+ margin: 0 0 var(--wn-space-5) 0;
125
+ }
126
+
127
+ /* Collection list */
128
+ .wn-select-collection-list {
129
+ list-style: none;
130
+ margin: 0;
131
+ padding: 0;
132
+ display: flex;
133
+ flex-direction: column;
134
+ gap: var(--wn-space-1);
135
+ }
136
+
137
+ .wn-select-collection-item {
138
+ display: flex;
139
+ align-items: center;
140
+ gap: var(--wn-space-4);
141
+ width: 100%;
142
+ padding: var(--wn-space-4) var(--wn-space-5);
143
+ border-radius: var(--wn-radius-md);
144
+ border: 1px solid transparent;
145
+ background-color: transparent;
146
+ color: var(--wn-zinc-400);
147
+ font-size: var(--wn-font-base);
148
+ text-align: left;
149
+ cursor: pointer;
150
+ transition: all var(--wn-transition-fast) ease;
151
+ }
152
+
153
+ .wn-select-collection-item:hover {
154
+ background-color: var(--wn-overlay-5);
155
+ color: var(--wn-zinc-50);
156
+ }
157
+
158
+ .wn-select-collection-item--focused {
159
+ background-color: var(--wn-brand-alpha-10);
160
+ border-color: var(--wn-brand-alpha-30);
161
+ color: var(--wn-brand-400);
162
+ }
163
+
164
+ .wn-select-collection-item-name {
165
+ flex: 1;
166
+ font-weight: 500;
167
+ }
168
+
169
+ .wn-select-collection-item-count {
170
+ font-size: var(--wn-font-xs);
171
+ color: var(--wn-zinc-500);
172
+ }
173
+
174
+ .wn-select-collection-item--focused .wn-select-collection-item-count {
175
+ color: var(--wn-brand-400);
176
+ opacity: 0.7;
177
+ }
178
+
179
+ .wn-select-collection-item-arrow {
180
+ opacity: 0;
181
+ transition: opacity var(--wn-transition-fast);
182
+ }
183
+
184
+ .wn-select-collection-item:hover .wn-select-collection-item-arrow,
185
+ .wn-select-collection-item--focused .wn-select-collection-item-arrow {
186
+ opacity: 1;
187
+ }
188
+
189
+ /* Loading & Empty states */
190
+ .wn-select-collection-loading,
191
+ .wn-select-collection-empty {
192
+ padding: var(--wn-space-8);
193
+ text-align: center;
194
+ font-size: var(--wn-font-base);
195
+ color: var(--wn-zinc-500);
196
+ }
197
+
198
+ /* ============================================================================
199
+ MODAL FOOTER
200
+ ============================================================================ */
201
+
202
+ .wn-select-collection-footer {
203
+ display: flex;
204
+ align-items: center;
205
+ justify-content: center;
206
+ gap: var(--wn-space-7);
207
+ padding: var(--wn-space-4) var(--wn-space-6);
208
+ border-top: 1px solid var(--wn-zinc-700);
209
+ background-color: var(--wn-overlay-light-10);
210
+ flex-shrink: 0;
211
+ }
212
+
213
+ .wn-select-collection-shortcut {
214
+ display: flex;
215
+ align-items: center;
216
+ gap: var(--wn-space-1);
217
+ font-size: var(--wn-font-xs);
218
+ color: var(--wn-zinc-500);
219
+ }
220
+
221
+ .wn-select-collection-shortcut kbd {
222
+ display: inline-flex;
223
+ align-items: center;
224
+ justify-content: center;
225
+ min-width: 1.25rem;
226
+ height: 1.125rem;
227
+ padding: 0 var(--wn-space-1);
228
+ background-color: var(--wn-zinc-800);
229
+ border: 1px solid var(--wn-zinc-700);
230
+ border-radius: var(--wn-radius-sm);
231
+ font-family: inherit;
232
+ font-size: var(--wn-font-xs);
233
+ font-weight: 500;
234
+ color: var(--wn-zinc-400);
235
+ }
236
+
237
+ /* ============================================================================
238
+ LIGHT MODE OVERRIDES
239
+ ============================================================================ */
240
+
241
+ .wn-light .wn-select-collection-overlay {
242
+ background-color: var(--wn-backdrop-light);
243
+ }
244
+
245
+ .wn-light .wn-select-collection-modal {
246
+ border-color: var(--wn-zinc-200);
247
+ background-color: #fff;
248
+ }
249
+
250
+ .wn-light .wn-select-collection-header {
251
+ border-bottom-color: var(--wn-zinc-200);
252
+ }
253
+
254
+ .wn-light .wn-select-collection-title {
255
+ color: var(--wn-zinc-900);
256
+ }
257
+
258
+ .wn-light .wn-select-collection-close {
259
+ color: var(--wn-zinc-500);
260
+ }
261
+
262
+ .wn-light .wn-select-collection-close:hover {
263
+ background-color: var(--wn-overlay-light-5);
264
+ color: var(--wn-zinc-900);
265
+ }
266
+
267
+ .wn-light .wn-select-collection-hint {
268
+ color: var(--wn-zinc-500);
269
+ }
270
+
271
+ .wn-light .wn-select-collection-item {
272
+ color: var(--wn-zinc-500);
273
+ }
274
+
275
+ .wn-light .wn-select-collection-item:hover {
276
+ background-color: var(--wn-overlay-light-5);
277
+ color: var(--wn-zinc-900);
278
+ }
279
+
280
+ .wn-light .wn-select-collection-item--focused {
281
+ background-color: var(--wn-brand-alpha-10);
282
+ border-color: var(--wn-brand-alpha-30);
283
+ color: var(--wn-brand-600);
284
+ }
285
+
286
+ .wn-light .wn-select-collection-item-count {
287
+ color: var(--wn-zinc-400);
288
+ }
289
+
290
+ .wn-light .wn-select-collection-item--focused .wn-select-collection-item-count {
291
+ color: var(--wn-brand-600);
292
+ }
293
+
294
+ .wn-light .wn-select-collection-loading,
295
+ .wn-light .wn-select-collection-empty {
296
+ color: var(--wn-zinc-500);
297
+ }
298
+
299
+ .wn-light .wn-select-collection-footer {
300
+ border-top-color: var(--wn-zinc-200);
301
+ background-color: var(--wn-zinc-50);
302
+ }
303
+
304
+ .wn-light .wn-select-collection-shortcut kbd {
305
+ background-color: #fff;
306
+ border-color: var(--wn-zinc-200);
307
+ color: var(--wn-zinc-600);
308
+ }
@@ -0,0 +1,223 @@
1
+ /**
2
+ * @fileoverview Select collection modal component
3
+ *
4
+ * Modal dialog for selecting a collection when creating new content
5
+ * without a pre-selected collection. Triggered by Alt+N shortcut.
6
+ *
7
+ * @module @writenex/astro/client/components/SelectCollectionModal
8
+ */
9
+
10
+ import { useCallback, useEffect, useRef, useState } from "react";
11
+ import { X, Folder, ChevronRight } from "lucide-react";
12
+ import { useFocusTrap } from "../../hooks/useFocusTrap";
13
+ import type { Collection } from "../../hooks/useApi";
14
+ import "./SelectCollectionModal.css";
15
+
16
+ /**
17
+ * Props for SelectCollectionModal component
18
+ */
19
+ interface SelectCollectionModalProps {
20
+ /** Whether the modal is open */
21
+ isOpen: boolean;
22
+ /** Callback to close the modal */
23
+ onClose: () => void;
24
+ /** Callback when a collection is selected */
25
+ onSelect: (collectionName: string) => void;
26
+ /** Available collections */
27
+ collections: Collection[];
28
+ /** Whether collections are loading */
29
+ isLoading?: boolean;
30
+ }
31
+
32
+ /**
33
+ * Modal dialog for selecting a collection
34
+ *
35
+ * @component
36
+ */
37
+ export function SelectCollectionModal({
38
+ isOpen,
39
+ onClose,
40
+ onSelect,
41
+ collections,
42
+ isLoading = false,
43
+ }: SelectCollectionModalProps): React.ReactElement | null {
44
+ const [focusedIndex, setFocusedIndex] = useState(0);
45
+ const listRef = useRef<HTMLUListElement>(null);
46
+ const triggerRef = useRef<HTMLElement | null>(null);
47
+
48
+ // Store the trigger element when modal opens
49
+ useEffect(() => {
50
+ if (isOpen) {
51
+ triggerRef.current = document.activeElement as HTMLElement;
52
+ setFocusedIndex(0);
53
+ }
54
+ }, [isOpen]);
55
+
56
+ // Focus trap for accessibility
57
+ const { containerRef } = useFocusTrap({
58
+ enabled: isOpen,
59
+ onEscape: onClose,
60
+ returnFocusTo: triggerRef.current,
61
+ });
62
+
63
+ // Keyboard navigation
64
+ useEffect(() => {
65
+ if (!isOpen || collections.length === 0) return;
66
+
67
+ const handleKeyDown = (e: KeyboardEvent) => {
68
+ switch (e.key) {
69
+ case "ArrowDown":
70
+ e.preventDefault();
71
+ setFocusedIndex((prev) =>
72
+ prev < collections.length - 1 ? prev + 1 : 0
73
+ );
74
+ break;
75
+ case "ArrowUp":
76
+ e.preventDefault();
77
+ setFocusedIndex((prev) =>
78
+ prev > 0 ? prev - 1 : collections.length - 1
79
+ );
80
+ break;
81
+ case "Enter":
82
+ e.preventDefault();
83
+ if (collections[focusedIndex]) {
84
+ onSelect(collections[focusedIndex].name);
85
+ }
86
+ break;
87
+ }
88
+ };
89
+
90
+ document.addEventListener("keydown", handleKeyDown);
91
+ return () => document.removeEventListener("keydown", handleKeyDown);
92
+ }, [isOpen, collections, focusedIndex, onSelect]);
93
+
94
+ // Scroll focused item into view
95
+ useEffect(() => {
96
+ if (listRef.current && collections.length > 0) {
97
+ const focusedItem = listRef.current.children[focusedIndex] as HTMLElement;
98
+ focusedItem?.scrollIntoView({ block: "nearest" });
99
+ }
100
+ }, [focusedIndex, collections.length]);
101
+
102
+ const handleOverlayClick = useCallback(
103
+ (e: React.MouseEvent) => {
104
+ if (e.target === e.currentTarget) {
105
+ onClose();
106
+ }
107
+ },
108
+ [onClose]
109
+ );
110
+
111
+ const handleItemClick = useCallback(
112
+ (collectionName: string) => {
113
+ onSelect(collectionName);
114
+ },
115
+ [onSelect]
116
+ );
117
+
118
+ if (!isOpen) return null;
119
+
120
+ return (
121
+ <div
122
+ className="wn-select-collection-overlay"
123
+ onClick={handleOverlayClick}
124
+ role="presentation"
125
+ >
126
+ <div
127
+ ref={containerRef}
128
+ className="wn-select-collection-modal"
129
+ role="dialog"
130
+ aria-modal="true"
131
+ aria-labelledby="select-collection-title"
132
+ >
133
+ {/* Header */}
134
+ <div className="wn-select-collection-header">
135
+ <div className="wn-select-collection-header-content">
136
+ <Folder size={18} className="wn-select-collection-icon" />
137
+ <h2
138
+ id="select-collection-title"
139
+ className="wn-select-collection-title"
140
+ >
141
+ Select Collection
142
+ </h2>
143
+ </div>
144
+ <button
145
+ className="wn-select-collection-close"
146
+ onClick={onClose}
147
+ title="Close (Esc)"
148
+ aria-label="Close modal"
149
+ >
150
+ <X size={16} />
151
+ </button>
152
+ </div>
153
+
154
+ {/* Body */}
155
+ <div className="wn-select-collection-body">
156
+ <p className="wn-select-collection-hint">
157
+ Choose a collection to create new content in:
158
+ </p>
159
+
160
+ {isLoading ? (
161
+ <div className="wn-select-collection-loading">
162
+ Loading collections...
163
+ </div>
164
+ ) : collections.length === 0 ? (
165
+ <div className="wn-select-collection-empty">
166
+ No collections found
167
+ </div>
168
+ ) : (
169
+ <ul
170
+ ref={listRef}
171
+ className="wn-select-collection-list"
172
+ role="listbox"
173
+ aria-label="Collections"
174
+ >
175
+ {collections.map((collection, index) => (
176
+ <li
177
+ key={collection.name}
178
+ role="option"
179
+ aria-selected={index === focusedIndex}
180
+ >
181
+ <button
182
+ className={`wn-select-collection-item ${
183
+ index === focusedIndex
184
+ ? "wn-select-collection-item--focused"
185
+ : ""
186
+ }`}
187
+ onClick={() => handleItemClick(collection.name)}
188
+ tabIndex={index === focusedIndex ? 0 : -1}
189
+ >
190
+ <Folder size={16} />
191
+ <span className="wn-select-collection-item-name">
192
+ {collection.name}
193
+ </span>
194
+ <span className="wn-select-collection-item-count">
195
+ {collection.count} items
196
+ </span>
197
+ <ChevronRight
198
+ size={14}
199
+ className="wn-select-collection-item-arrow"
200
+ />
201
+ </button>
202
+ </li>
203
+ ))}
204
+ </ul>
205
+ )}
206
+ </div>
207
+
208
+ {/* Footer hint */}
209
+ <div className="wn-select-collection-footer">
210
+ <span className="wn-select-collection-shortcut">
211
+ <kbd>↑</kbd> <kbd>↓</kbd> Navigate
212
+ </span>
213
+ <span className="wn-select-collection-shortcut">
214
+ <kbd>Enter</kbd> Select
215
+ </span>
216
+ <span className="wn-select-collection-shortcut">
217
+ <kbd>Esc</kbd> Cancel
218
+ </span>
219
+ </div>
220
+ </div>
221
+ </div>
222
+ );
223
+ }
@@ -0,0 +1,7 @@
1
+ /**
2
+ * @fileoverview SelectCollectionModal component exports
3
+ *
4
+ * @module @writenex/astro/client/components/SelectCollectionModal
5
+ */
6
+
7
+ export { SelectCollectionModal } from "./SelectCollectionModal";