@useavalon/avalon 0.1.11 → 0.1.13

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 +54 -54
  2. package/mod.ts +302 -302
  3. package/package.json +49 -26
  4. package/src/build/integration-bundler-plugin.ts +116 -116
  5. package/src/build/integration-config.ts +168 -168
  6. package/src/build/integration-detection-plugin.ts +117 -117
  7. package/src/build/integration-resolver-plugin.ts +90 -90
  8. package/src/build/island-manifest.ts +269 -269
  9. package/src/build/island-types-generator.ts +476 -476
  10. package/src/build/mdx-island-transform.ts +464 -464
  11. package/src/build/mdx-plugin.ts +98 -98
  12. package/src/build/page-island-transform.ts +598 -598
  13. package/src/build/prop-extractors/index.ts +21 -21
  14. package/src/build/prop-extractors/lit.ts +140 -140
  15. package/src/build/prop-extractors/qwik.ts +16 -16
  16. package/src/build/prop-extractors/solid.ts +125 -125
  17. package/src/build/prop-extractors/svelte.ts +194 -194
  18. package/src/build/prop-extractors/vue.ts +111 -111
  19. package/src/build/sidecar-file-manager.ts +104 -104
  20. package/src/build/sidecar-renderer.ts +30 -30
  21. package/src/client/adapters/index.ts +21 -13
  22. package/src/client/components.ts +35 -35
  23. package/src/client/css-hmr-handler.ts +344 -344
  24. package/src/client/framework-adapter.ts +462 -462
  25. package/src/client/hmr-coordinator.ts +396 -396
  26. package/src/client/hmr-error-overlay.js +533 -533
  27. package/src/client/main.js +824 -816
  28. package/src/client/types/framework-runtime.d.ts +68 -68
  29. package/src/client/types/vite-hmr.d.ts +46 -46
  30. package/src/client/types/vite-virtual-modules.d.ts +70 -60
  31. package/src/components/Image.tsx +123 -123
  32. package/src/components/IslandErrorBoundary.tsx +145 -145
  33. package/src/components/LayoutDataErrorBoundary.tsx +141 -141
  34. package/src/components/LayoutErrorBoundary.tsx +127 -127
  35. package/src/components/PersistentIsland.tsx +52 -52
  36. package/src/components/StreamingErrorBoundary.tsx +233 -233
  37. package/src/components/StreamingLayout.tsx +538 -538
  38. package/src/core/components/component-analyzer.ts +192 -192
  39. package/src/core/components/component-detection.ts +508 -508
  40. package/src/core/components/enhanced-framework-detector.ts +500 -500
  41. package/src/core/components/framework-registry.ts +563 -563
  42. package/src/core/content/mdx-processor.ts +46 -46
  43. package/src/core/integrations/index.ts +19 -19
  44. package/src/core/integrations/loader.ts +125 -125
  45. package/src/core/integrations/registry.ts +175 -175
  46. package/src/core/islands/island-persistence.ts +325 -325
  47. package/src/core/islands/island-state-serializer.ts +258 -258
  48. package/src/core/islands/persistent-island-context.tsx +80 -80
  49. package/src/core/islands/use-persistent-state.ts +68 -68
  50. package/src/core/layout/enhanced-layout-resolver.ts +322 -322
  51. package/src/core/layout/layout-cache-manager.ts +485 -485
  52. package/src/core/layout/layout-composer.ts +357 -357
  53. package/src/core/layout/layout-data-loader.ts +516 -516
  54. package/src/core/layout/layout-discovery.ts +243 -243
  55. package/src/core/layout/layout-matcher.ts +299 -299
  56. package/src/core/layout/layout-types.ts +110 -110
  57. package/src/core/modules/framework-module-resolver.ts +273 -273
  58. package/src/islands/component-analysis.ts +213 -213
  59. package/src/islands/css-utils.ts +565 -565
  60. package/src/islands/discovery/index.ts +80 -80
  61. package/src/islands/discovery/registry.ts +340 -340
  62. package/src/islands/discovery/resolver.ts +477 -477
  63. package/src/islands/discovery/scanner.ts +386 -386
  64. package/src/islands/discovery/types.ts +117 -117
  65. package/src/islands/discovery/validator.ts +544 -544
  66. package/src/islands/discovery/watcher.ts +368 -368
  67. package/src/islands/framework-detection.ts +428 -428
  68. package/src/islands/integration-loader.ts +490 -490
  69. package/src/islands/island.tsx +565 -565
  70. package/src/islands/render-cache.ts +550 -550
  71. package/src/islands/types.ts +80 -80
  72. package/src/islands/universal-css-collector.ts +157 -157
  73. package/src/islands/universal-head-collector.ts +137 -137
  74. package/src/layout-system.d.ts +592 -592
  75. package/src/layout-system.ts +218 -218
  76. package/src/middleware/discovery.ts +268 -268
  77. package/src/middleware/executor.ts +315 -315
  78. package/src/middleware/index.ts +76 -76
  79. package/src/middleware/types.ts +99 -99
  80. package/src/nitro/build-config.ts +575 -575
  81. package/src/nitro/config.ts +483 -483
  82. package/src/nitro/error-handler.ts +636 -636
  83. package/src/nitro/index.ts +173 -173
  84. package/src/nitro/island-manifest.ts +584 -584
  85. package/src/nitro/middleware-adapter.ts +260 -260
  86. package/src/nitro/renderer.ts +1471 -1471
  87. package/src/nitro/route-discovery.ts +439 -439
  88. package/src/nitro/types.ts +321 -321
  89. package/src/render/collect-css.ts +198 -198
  90. package/src/render/error-pages.ts +79 -79
  91. package/src/render/isolated-ssr-renderer.ts +654 -654
  92. package/src/render/ssr.ts +1030 -1030
  93. package/src/schemas/api.ts +30 -30
  94. package/src/schemas/core.ts +64 -64
  95. package/src/schemas/index.ts +212 -212
  96. package/src/schemas/layout.ts +279 -279
  97. package/src/schemas/routing/index.ts +38 -38
  98. package/src/schemas/routing.ts +376 -376
  99. package/src/types/as-island.ts +20 -20
  100. package/src/types/image.d.ts +106 -106
  101. package/src/types/index.d.ts +22 -22
  102. package/src/types/island-jsx.d.ts +33 -33
  103. package/src/types/island-prop.d.ts +20 -20
  104. package/src/types/layout.ts +285 -285
  105. package/src/types/mdx.d.ts +6 -6
  106. package/src/types/routing.ts +555 -555
  107. package/src/types/types.ts +5 -5
  108. package/src/types/urlpattern.d.ts +49 -49
  109. package/src/types/vite-env.d.ts +11 -11
  110. package/src/utils/dev-logger.ts +299 -299
  111. package/src/utils/fs.ts +151 -151
  112. package/src/vite-plugin/auto-discover.ts +551 -551
  113. package/src/vite-plugin/config.ts +266 -266
  114. package/src/vite-plugin/errors.ts +127 -127
  115. package/src/vite-plugin/image-optimization.ts +156 -156
  116. package/src/vite-plugin/integration-activator.ts +126 -126
  117. package/src/vite-plugin/island-sidecar-plugin.ts +176 -176
  118. package/src/vite-plugin/module-discovery.ts +189 -189
  119. package/src/vite-plugin/nitro-integration.ts +1354 -1354
  120. package/src/vite-plugin/plugin.ts +403 -409
  121. package/src/vite-plugin/types.ts +327 -327
  122. package/src/vite-plugin/validation.ts +228 -228
  123. package/src/client/adapters/index.js +0 -12
  124. package/src/client/adapters/lit-adapter.js +0 -467
  125. package/src/client/adapters/lit-adapter.ts +0 -654
  126. package/src/client/adapters/preact-adapter.js +0 -223
  127. package/src/client/adapters/preact-adapter.ts +0 -331
  128. package/src/client/adapters/qwik-adapter.js +0 -259
  129. package/src/client/adapters/qwik-adapter.ts +0 -345
  130. package/src/client/adapters/react-adapter.js +0 -220
  131. package/src/client/adapters/react-adapter.ts +0 -353
  132. package/src/client/adapters/solid-adapter.js +0 -295
  133. package/src/client/adapters/solid-adapter.ts +0 -451
  134. package/src/client/adapters/svelte-adapter.js +0 -368
  135. package/src/client/adapters/svelte-adapter.ts +0 -524
  136. package/src/client/adapters/vue-adapter.js +0 -278
  137. package/src/client/adapters/vue-adapter.ts +0 -467
  138. package/src/client/components.js +0 -23
  139. package/src/client/css-hmr-handler.js +0 -263
  140. package/src/client/framework-adapter.js +0 -283
  141. package/src/client/hmr-coordinator.js +0 -274
@@ -1,368 +1,368 @@
1
- /**
2
- * Island File Watcher
3
- *
4
- * Watches all discovered island directories for file changes and emits
5
- * change events with affected island information. Supports HMR for
6
- * nested island directories.
7
- */
8
-
9
- import { resolve, relative, extname, basename } from "node:path";
10
- import { watch, type FSWatcher } from "node:fs";
11
- import type {
12
- IslandDirectory,
13
- DiscoveredIsland,
14
- IslandChangeEvent,
15
- IslandDiscoveryConfig,
16
- } from "./types.ts";
17
- import { isSupportedIslandExtension } from "./types.ts";
18
- import { discoverIslandsInDirectory } from "./scanner.ts";
19
- import { IslandRegistry } from "./registry.ts";
20
-
21
- /**
22
- * Callback type for island change events
23
- */
24
- export type IslandChangeCallback = (event: IslandChangeEvent) => void;
25
-
26
- /**
27
- * Options for the island watcher
28
- */
29
- export interface IslandWatcherOptions {
30
- /** Debounce delay in milliseconds (default: 100) */
31
- debounceMs?: number;
32
- /** Whether to emit events for initial discovery (default: false) */
33
- emitInitial?: boolean;
34
- }
35
-
36
- /**
37
- * Default watcher options
38
- */
39
- const DEFAULT_WATCHER_OPTIONS: Required<IslandWatcherOptions> = {
40
- debounceMs: 100,
41
- emitInitial: false,
42
- };
43
-
44
- /**
45
- * Island File Watcher
46
- *
47
- * Watches all discovered island directories for file changes.
48
- * Emits change events with affected island information for HMR support.
49
- */
50
- export class IslandWatcher {
51
- private _projectRoot: string;
52
- private _config: IslandDiscoveryConfig;
53
- private _options: Required<IslandWatcherOptions>;
54
- private _registry: IslandRegistry;
55
- private _watchers: FSWatcher[] = [];
56
- private _callbacks: Set<IslandChangeCallback> = new Set();
57
- private _isWatching = false;
58
- private _debounceTimers: Map<string, ReturnType<typeof setTimeout>> = new Map();
59
-
60
- constructor(
61
- projectRoot: string,
62
- registry: IslandRegistry,
63
- config: IslandDiscoveryConfig = {},
64
- options: IslandWatcherOptions = {}
65
- ) {
66
- this._projectRoot = projectRoot;
67
- this._registry = registry;
68
- this._config = config;
69
- this._options = { ...DEFAULT_WATCHER_OPTIONS, ...options };
70
- }
71
-
72
- /**
73
- * Check if the watcher is currently active
74
- */
75
- get isWatching(): boolean {
76
- return this._isWatching;
77
- }
78
-
79
- /**
80
- * Get the number of registered callbacks
81
- */
82
- get callbackCount(): number {
83
- return this._callbacks.size;
84
- }
85
-
86
- /**
87
- * Start watching all discovered island directories.
88
- *
89
- * @param callback - Callback to invoke on file changes
90
- * @returns Cleanup function to stop watching
91
- */
92
- async watch(callback: IslandChangeCallback): Promise<() => void> {
93
- this._callbacks.add(callback);
94
-
95
- // Start watching if not already
96
- if (!this._isWatching) {
97
- await this._startWatching();
98
- }
99
-
100
- // Return cleanup function
101
- return () => {
102
- this._callbacks.delete(callback);
103
-
104
- // Stop watching if no more callbacks
105
- if (this._callbacks.size === 0) {
106
- this.stop();
107
- }
108
- };
109
- }
110
-
111
- /**
112
- * Stop all file watchers and clear callbacks.
113
- */
114
- stop(): void {
115
- this._isWatching = false;
116
-
117
- // Close all watchers
118
- for (const watcher of this._watchers) {
119
- try {
120
- watcher.close();
121
- } catch {
122
- // Ignore errors when closing
123
- }
124
- }
125
- this._watchers = [];
126
-
127
- // Clear debounce timers
128
- for (const timerId of this._debounceTimers.values()) {
129
- clearTimeout(timerId);
130
- }
131
- this._debounceTimers.clear();
132
- }
133
-
134
- /**
135
- * Start watching all island directories.
136
- */
137
- private async _startWatching(): Promise<void> {
138
- if (this._isWatching) return;
139
-
140
- this._isWatching = true;
141
- const directories = this._registry.directories;
142
-
143
- for (const directory of directories) {
144
- try {
145
- this._watchDirectory(directory);
146
- } catch (error) {
147
- console.warn(
148
- `⚠️ Failed to watch island directory ${directory.relativePath}:`,
149
- error
150
- );
151
- }
152
- }
153
- }
154
-
155
- /**
156
- * Watch a single island directory for changes.
157
- */
158
- private _watchDirectory(directory: IslandDirectory): void {
159
- try {
160
- const fsWatcher = watch(directory.path, { recursive: false }, (eventType, filename) => {
161
- if (!this._isWatching || !filename) return;
162
-
163
- const filePath = resolve(directory.path, filename);
164
- const ext = extname(filename);
165
- if (!isSupportedIslandExtension(ext)) return;
166
-
167
- // Map Node.js event types to our event types
168
- const kind: "change" | "rename" = eventType as "change" | "rename";
169
- this._debounceEvent(filePath, kind, directory);
170
- });
171
-
172
- this._watchers.push(fsWatcher);
173
- } catch (error) {
174
- if (error instanceof Error && (error as NodeJS.ErrnoException).code === 'ENOENT') {
175
- console.warn(`⚠️ Island directory not found: ${directory.relativePath}`);
176
- } else {
177
- throw error;
178
- }
179
- }
180
- }
181
-
182
- /**
183
- * Debounce file change events to avoid duplicate processing.
184
- */
185
- private _debounceEvent(
186
- filePath: string,
187
- kind: "change" | "rename",
188
- directory: IslandDirectory
189
- ): void {
190
- // Clear existing timer for this file
191
- const existingTimer = this._debounceTimers.get(filePath);
192
- if (existingTimer) {
193
- clearTimeout(existingTimer);
194
- }
195
-
196
- // Set new timer
197
- const timerId = setTimeout(() => {
198
- this._debounceTimers.delete(filePath);
199
- this._handleFileChange(filePath, kind, directory);
200
- }, this._options.debounceMs);
201
-
202
- this._debounceTimers.set(filePath, timerId);
203
- }
204
-
205
- /**
206
- * Handle a file change event.
207
- */
208
- private async _handleFileChange(
209
- filePath: string,
210
- kind: "change" | "rename",
211
- directory: IslandDirectory
212
- ): Promise<void> {
213
- const relativePath = relative(this._projectRoot, filePath).replace(/\\/g, "/");
214
-
215
- // Node.js fs.watch emits "rename" for both add and remove, "change" for modifications
216
- // We need to check if the file exists to determine if it was added or removed
217
- let eventType: IslandChangeEvent["type"];
218
- if (kind === "change") {
219
- eventType = "change";
220
- } else {
221
- // "rename" - check if file exists to determine add vs remove
222
- try {
223
- const { stat } = await import("node:fs/promises");
224
- await stat(filePath);
225
- eventType = "add";
226
- } catch {
227
- eventType = "remove";
228
- }
229
- }
230
-
231
- // Try to find or create the island info
232
- let island: DiscoveredIsland | null = null;
233
-
234
- if (eventType === "remove") {
235
- // For remove events, try to find the island in the registry
236
- const qualifiedName = this._getQualifiedNameFromPath(filePath, directory);
237
- island = this._registry.resolve(qualifiedName) || null;
238
-
239
- // Remove from registry
240
- if (island) {
241
- this._registry.unregister(qualifiedName);
242
- }
243
- } else {
244
- // For add/change events, discover the island
245
- try {
246
- const islands = await discoverIslandsInDirectory(directory, this._projectRoot);
247
- island = islands.find(i => i.filePath === filePath) || null;
248
-
249
- // Update registry for new islands
250
- if (eventType === "add" && island) {
251
- this._registry.register(island);
252
- }
253
- } catch {
254
- // File might have been deleted between event and processing
255
- island = null;
256
- }
257
- }
258
-
259
- // Create and emit the change event
260
- const changeEvent: IslandChangeEvent = {
261
- type: eventType,
262
- island,
263
- filePath: relativePath,
264
- timestamp: Date.now(),
265
- };
266
-
267
- this._emitEvent(changeEvent);
268
- }
269
-
270
- /**
271
- * Get the qualified name for an island from its file path.
272
- */
273
- private _getQualifiedNameFromPath(
274
- filePath: string,
275
- directory: IslandDirectory
276
- ): string {
277
- const fileName = basename(filePath);
278
- const name = this._extractComponentName(fileName);
279
-
280
- if (directory.namespace === "") {
281
- return name;
282
- }
283
- return `${directory.namespace}/${name}`;
284
- }
285
-
286
- /**
287
- * Extract component name from file name.
288
- */
289
- private _extractComponentName(fileName: string): string {
290
- const frameworkPatterns = [
291
- ".solid.tsx", ".solid.jsx",
292
- ".react.tsx", ".react.jsx",
293
- ".lit.ts", ".lit.js",
294
- ".preact.tsx", ".preact.jsx",
295
- ];
296
-
297
- for (const pattern of frameworkPatterns) {
298
- if (fileName.endsWith(pattern)) {
299
- return fileName.slice(0, -pattern.length);
300
- }
301
- }
302
-
303
- const singleExtensions = [".tsx", ".ts", ".jsx", ".js", ".vue", ".svelte"];
304
- for (const ext of singleExtensions) {
305
- if (fileName.endsWith(ext)) {
306
- return fileName.slice(0, -ext.length);
307
- }
308
- }
309
-
310
- return fileName;
311
- }
312
-
313
- /**
314
- * Emit an event to all registered callbacks.
315
- */
316
- private _emitEvent(event: IslandChangeEvent): void {
317
- for (const callback of this._callbacks) {
318
- try {
319
- callback(event);
320
- } catch (error) {
321
- console.error("Error in island change callback:", error);
322
- }
323
- }
324
- }
325
-
326
- /**
327
- * Refresh the watcher to pick up new directories.
328
- * Call this after the registry is rebuilt.
329
- */
330
- async refresh(): Promise<void> {
331
- if (!this._isWatching) return;
332
-
333
- // Stop existing watchers
334
- for (const watcher of this._watchers) {
335
- try {
336
- watcher.close();
337
- } catch {
338
- // Ignore errors
339
- }
340
- }
341
- this._watchers = [];
342
-
343
- // Start watching new directories
344
- const directories = this._registry.directories;
345
- for (const directory of directories) {
346
- try {
347
- this._watchDirectory(directory);
348
- } catch (error) {
349
- console.warn(
350
- `⚠️ Failed to watch island directory ${directory.relativePath}:`,
351
- error
352
- );
353
- }
354
- }
355
- }
356
- }
357
-
358
- /**
359
- * Create an island watcher for the given registry.
360
- */
361
- export function createIslandWatcher(
362
- projectRoot: string,
363
- registry: IslandRegistry,
364
- config: IslandDiscoveryConfig = {},
365
- options: IslandWatcherOptions = {}
366
- ): IslandWatcher {
367
- return new IslandWatcher(projectRoot, registry, config, options);
368
- }
1
+ /**
2
+ * Island File Watcher
3
+ *
4
+ * Watches all discovered island directories for file changes and emits
5
+ * change events with affected island information. Supports HMR for
6
+ * nested island directories.
7
+ */
8
+
9
+ import { resolve, relative, extname, basename } from "node:path";
10
+ import { watch, type FSWatcher } from "node:fs";
11
+ import type {
12
+ IslandDirectory,
13
+ DiscoveredIsland,
14
+ IslandChangeEvent,
15
+ IslandDiscoveryConfig,
16
+ } from "./types.ts";
17
+ import { isSupportedIslandExtension } from "./types.ts";
18
+ import { discoverIslandsInDirectory } from "./scanner.ts";
19
+ import { IslandRegistry } from "./registry.ts";
20
+
21
+ /**
22
+ * Callback type for island change events
23
+ */
24
+ export type IslandChangeCallback = (event: IslandChangeEvent) => void;
25
+
26
+ /**
27
+ * Options for the island watcher
28
+ */
29
+ export interface IslandWatcherOptions {
30
+ /** Debounce delay in milliseconds (default: 100) */
31
+ debounceMs?: number;
32
+ /** Whether to emit events for initial discovery (default: false) */
33
+ emitInitial?: boolean;
34
+ }
35
+
36
+ /**
37
+ * Default watcher options
38
+ */
39
+ const DEFAULT_WATCHER_OPTIONS: Required<IslandWatcherOptions> = {
40
+ debounceMs: 100,
41
+ emitInitial: false,
42
+ };
43
+
44
+ /**
45
+ * Island File Watcher
46
+ *
47
+ * Watches all discovered island directories for file changes.
48
+ * Emits change events with affected island information for HMR support.
49
+ */
50
+ export class IslandWatcher {
51
+ private _projectRoot: string;
52
+ private _config: IslandDiscoveryConfig;
53
+ private _options: Required<IslandWatcherOptions>;
54
+ private _registry: IslandRegistry;
55
+ private _watchers: FSWatcher[] = [];
56
+ private _callbacks: Set<IslandChangeCallback> = new Set();
57
+ private _isWatching = false;
58
+ private _debounceTimers: Map<string, ReturnType<typeof setTimeout>> = new Map();
59
+
60
+ constructor(
61
+ projectRoot: string,
62
+ registry: IslandRegistry,
63
+ config: IslandDiscoveryConfig = {},
64
+ options: IslandWatcherOptions = {}
65
+ ) {
66
+ this._projectRoot = projectRoot;
67
+ this._registry = registry;
68
+ this._config = config;
69
+ this._options = { ...DEFAULT_WATCHER_OPTIONS, ...options };
70
+ }
71
+
72
+ /**
73
+ * Check if the watcher is currently active
74
+ */
75
+ get isWatching(): boolean {
76
+ return this._isWatching;
77
+ }
78
+
79
+ /**
80
+ * Get the number of registered callbacks
81
+ */
82
+ get callbackCount(): number {
83
+ return this._callbacks.size;
84
+ }
85
+
86
+ /**
87
+ * Start watching all discovered island directories.
88
+ *
89
+ * @param callback - Callback to invoke on file changes
90
+ * @returns Cleanup function to stop watching
91
+ */
92
+ async watch(callback: IslandChangeCallback): Promise<() => void> {
93
+ this._callbacks.add(callback);
94
+
95
+ // Start watching if not already
96
+ if (!this._isWatching) {
97
+ await this._startWatching();
98
+ }
99
+
100
+ // Return cleanup function
101
+ return () => {
102
+ this._callbacks.delete(callback);
103
+
104
+ // Stop watching if no more callbacks
105
+ if (this._callbacks.size === 0) {
106
+ this.stop();
107
+ }
108
+ };
109
+ }
110
+
111
+ /**
112
+ * Stop all file watchers and clear callbacks.
113
+ */
114
+ stop(): void {
115
+ this._isWatching = false;
116
+
117
+ // Close all watchers
118
+ for (const watcher of this._watchers) {
119
+ try {
120
+ watcher.close();
121
+ } catch {
122
+ // Ignore errors when closing
123
+ }
124
+ }
125
+ this._watchers = [];
126
+
127
+ // Clear debounce timers
128
+ for (const timerId of this._debounceTimers.values()) {
129
+ clearTimeout(timerId);
130
+ }
131
+ this._debounceTimers.clear();
132
+ }
133
+
134
+ /**
135
+ * Start watching all island directories.
136
+ */
137
+ private async _startWatching(): Promise<void> {
138
+ if (this._isWatching) return;
139
+
140
+ this._isWatching = true;
141
+ const directories = this._registry.directories;
142
+
143
+ for (const directory of directories) {
144
+ try {
145
+ this._watchDirectory(directory);
146
+ } catch (error) {
147
+ console.warn(
148
+ `⚠️ Failed to watch island directory ${directory.relativePath}:`,
149
+ error
150
+ );
151
+ }
152
+ }
153
+ }
154
+
155
+ /**
156
+ * Watch a single island directory for changes.
157
+ */
158
+ private _watchDirectory(directory: IslandDirectory): void {
159
+ try {
160
+ const fsWatcher = watch(directory.path, { recursive: false }, (eventType, filename) => {
161
+ if (!this._isWatching || !filename) return;
162
+
163
+ const filePath = resolve(directory.path, filename);
164
+ const ext = extname(filename);
165
+ if (!isSupportedIslandExtension(ext)) return;
166
+
167
+ // Map Node.js event types to our event types
168
+ const kind: "change" | "rename" = eventType as "change" | "rename";
169
+ this._debounceEvent(filePath, kind, directory);
170
+ });
171
+
172
+ this._watchers.push(fsWatcher);
173
+ } catch (error) {
174
+ if (error instanceof Error && (error as NodeJS.ErrnoException).code === 'ENOENT') {
175
+ console.warn(`⚠️ Island directory not found: ${directory.relativePath}`);
176
+ } else {
177
+ throw error;
178
+ }
179
+ }
180
+ }
181
+
182
+ /**
183
+ * Debounce file change events to avoid duplicate processing.
184
+ */
185
+ private _debounceEvent(
186
+ filePath: string,
187
+ kind: "change" | "rename",
188
+ directory: IslandDirectory
189
+ ): void {
190
+ // Clear existing timer for this file
191
+ const existingTimer = this._debounceTimers.get(filePath);
192
+ if (existingTimer) {
193
+ clearTimeout(existingTimer);
194
+ }
195
+
196
+ // Set new timer
197
+ const timerId = setTimeout(() => {
198
+ this._debounceTimers.delete(filePath);
199
+ this._handleFileChange(filePath, kind, directory);
200
+ }, this._options.debounceMs);
201
+
202
+ this._debounceTimers.set(filePath, timerId);
203
+ }
204
+
205
+ /**
206
+ * Handle a file change event.
207
+ */
208
+ private async _handleFileChange(
209
+ filePath: string,
210
+ kind: "change" | "rename",
211
+ directory: IslandDirectory
212
+ ): Promise<void> {
213
+ const relativePath = relative(this._projectRoot, filePath).replace(/\\/g, "/");
214
+
215
+ // Node.js fs.watch emits "rename" for both add and remove, "change" for modifications
216
+ // We need to check if the file exists to determine if it was added or removed
217
+ let eventType: IslandChangeEvent["type"];
218
+ if (kind === "change") {
219
+ eventType = "change";
220
+ } else {
221
+ // "rename" - check if file exists to determine add vs remove
222
+ try {
223
+ const { stat } = await import("node:fs/promises");
224
+ await stat(filePath);
225
+ eventType = "add";
226
+ } catch {
227
+ eventType = "remove";
228
+ }
229
+ }
230
+
231
+ // Try to find or create the island info
232
+ let island: DiscoveredIsland | null = null;
233
+
234
+ if (eventType === "remove") {
235
+ // For remove events, try to find the island in the registry
236
+ const qualifiedName = this._getQualifiedNameFromPath(filePath, directory);
237
+ island = this._registry.resolve(qualifiedName) || null;
238
+
239
+ // Remove from registry
240
+ if (island) {
241
+ this._registry.unregister(qualifiedName);
242
+ }
243
+ } else {
244
+ // For add/change events, discover the island
245
+ try {
246
+ const islands = await discoverIslandsInDirectory(directory, this._projectRoot);
247
+ island = islands.find(i => i.filePath === filePath) || null;
248
+
249
+ // Update registry for new islands
250
+ if (eventType === "add" && island) {
251
+ this._registry.register(island);
252
+ }
253
+ } catch {
254
+ // File might have been deleted between event and processing
255
+ island = null;
256
+ }
257
+ }
258
+
259
+ // Create and emit the change event
260
+ const changeEvent: IslandChangeEvent = {
261
+ type: eventType,
262
+ island,
263
+ filePath: relativePath,
264
+ timestamp: Date.now(),
265
+ };
266
+
267
+ this._emitEvent(changeEvent);
268
+ }
269
+
270
+ /**
271
+ * Get the qualified name for an island from its file path.
272
+ */
273
+ private _getQualifiedNameFromPath(
274
+ filePath: string,
275
+ directory: IslandDirectory
276
+ ): string {
277
+ const fileName = basename(filePath);
278
+ const name = this._extractComponentName(fileName);
279
+
280
+ if (directory.namespace === "") {
281
+ return name;
282
+ }
283
+ return `${directory.namespace}/${name}`;
284
+ }
285
+
286
+ /**
287
+ * Extract component name from file name.
288
+ */
289
+ private _extractComponentName(fileName: string): string {
290
+ const frameworkPatterns = [
291
+ ".solid.tsx", ".solid.jsx",
292
+ ".react.tsx", ".react.jsx",
293
+ ".lit.ts", ".lit.js",
294
+ ".preact.tsx", ".preact.jsx",
295
+ ];
296
+
297
+ for (const pattern of frameworkPatterns) {
298
+ if (fileName.endsWith(pattern)) {
299
+ return fileName.slice(0, -pattern.length);
300
+ }
301
+ }
302
+
303
+ const singleExtensions = [".tsx", ".ts", ".jsx", ".js", ".vue", ".svelte"];
304
+ for (const ext of singleExtensions) {
305
+ if (fileName.endsWith(ext)) {
306
+ return fileName.slice(0, -ext.length);
307
+ }
308
+ }
309
+
310
+ return fileName;
311
+ }
312
+
313
+ /**
314
+ * Emit an event to all registered callbacks.
315
+ */
316
+ private _emitEvent(event: IslandChangeEvent): void {
317
+ for (const callback of this._callbacks) {
318
+ try {
319
+ callback(event);
320
+ } catch (error) {
321
+ console.error("Error in island change callback:", error);
322
+ }
323
+ }
324
+ }
325
+
326
+ /**
327
+ * Refresh the watcher to pick up new directories.
328
+ * Call this after the registry is rebuilt.
329
+ */
330
+ async refresh(): Promise<void> {
331
+ if (!this._isWatching) return;
332
+
333
+ // Stop existing watchers
334
+ for (const watcher of this._watchers) {
335
+ try {
336
+ watcher.close();
337
+ } catch {
338
+ // Ignore errors
339
+ }
340
+ }
341
+ this._watchers = [];
342
+
343
+ // Start watching new directories
344
+ const directories = this._registry.directories;
345
+ for (const directory of directories) {
346
+ try {
347
+ this._watchDirectory(directory);
348
+ } catch (error) {
349
+ console.warn(
350
+ `⚠️ Failed to watch island directory ${directory.relativePath}:`,
351
+ error
352
+ );
353
+ }
354
+ }
355
+ }
356
+ }
357
+
358
+ /**
359
+ * Create an island watcher for the given registry.
360
+ */
361
+ export function createIslandWatcher(
362
+ projectRoot: string,
363
+ registry: IslandRegistry,
364
+ config: IslandDiscoveryConfig = {},
365
+ options: IslandWatcherOptions = {}
366
+ ): IslandWatcher {
367
+ return new IslandWatcher(projectRoot, registry, config, options);
368
+ }