@useavalon/avalon 0.1.12 → 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 (230) hide show
  1. package/mod.ts +302 -0
  2. package/package.json +9 -17
  3. package/src/build/integration-bundler-plugin.ts +116 -0
  4. package/src/build/integration-config.ts +168 -0
  5. package/src/build/integration-detection-plugin.ts +117 -0
  6. package/src/build/integration-resolver-plugin.ts +90 -0
  7. package/src/build/island-manifest.ts +269 -0
  8. package/src/build/island-types-generator.ts +476 -0
  9. package/src/build/mdx-island-transform.ts +464 -0
  10. package/src/build/mdx-plugin.ts +98 -0
  11. package/src/build/page-island-transform.ts +598 -0
  12. package/src/build/prop-extractors/index.ts +21 -0
  13. package/src/build/prop-extractors/lit.ts +140 -0
  14. package/src/build/prop-extractors/qwik.ts +16 -0
  15. package/src/build/prop-extractors/solid.ts +125 -0
  16. package/src/build/prop-extractors/svelte.ts +194 -0
  17. package/src/build/prop-extractors/vue.ts +111 -0
  18. package/src/build/sidecar-file-manager.ts +104 -0
  19. package/src/build/sidecar-renderer.ts +30 -0
  20. package/src/client/adapters/index.ts +21 -0
  21. package/src/client/components.ts +35 -0
  22. package/src/client/css-hmr-handler.ts +344 -0
  23. package/src/client/framework-adapter.ts +462 -0
  24. package/src/client/hmr-coordinator.ts +396 -0
  25. package/src/client/hmr-error-overlay.js +533 -0
  26. package/src/client/main.js +824 -0
  27. package/src/components/Image.tsx +123 -0
  28. package/src/components/IslandErrorBoundary.tsx +145 -0
  29. package/src/components/LayoutDataErrorBoundary.tsx +141 -0
  30. package/src/components/LayoutErrorBoundary.tsx +127 -0
  31. package/src/components/PersistentIsland.tsx +52 -0
  32. package/src/components/StreamingErrorBoundary.tsx +233 -0
  33. package/src/components/StreamingLayout.tsx +538 -0
  34. package/src/core/components/component-analyzer.ts +192 -0
  35. package/src/core/components/component-detection.ts +508 -0
  36. package/src/core/components/enhanced-framework-detector.ts +500 -0
  37. package/src/core/components/framework-registry.ts +563 -0
  38. package/src/core/content/mdx-processor.ts +46 -0
  39. package/src/core/integrations/index.ts +19 -0
  40. package/src/core/integrations/loader.ts +125 -0
  41. package/src/core/integrations/registry.ts +175 -0
  42. package/src/core/islands/island-persistence.ts +325 -0
  43. package/src/core/islands/island-state-serializer.ts +258 -0
  44. package/src/core/islands/persistent-island-context.tsx +80 -0
  45. package/src/core/islands/use-persistent-state.ts +68 -0
  46. package/src/core/layout/enhanced-layout-resolver.ts +322 -0
  47. package/src/core/layout/layout-cache-manager.ts +485 -0
  48. package/src/core/layout/layout-composer.ts +357 -0
  49. package/src/core/layout/layout-data-loader.ts +516 -0
  50. package/src/core/layout/layout-discovery.ts +243 -0
  51. package/src/core/layout/layout-matcher.ts +299 -0
  52. package/src/core/layout/layout-types.ts +110 -0
  53. package/src/core/modules/framework-module-resolver.ts +273 -0
  54. package/src/islands/component-analysis.ts +213 -0
  55. package/src/islands/css-utils.ts +565 -0
  56. package/src/islands/discovery/index.ts +80 -0
  57. package/src/islands/discovery/registry.ts +340 -0
  58. package/src/islands/discovery/resolver.ts +477 -0
  59. package/src/islands/discovery/scanner.ts +386 -0
  60. package/src/islands/discovery/types.ts +117 -0
  61. package/src/islands/discovery/validator.ts +544 -0
  62. package/src/islands/discovery/watcher.ts +368 -0
  63. package/src/islands/framework-detection.ts +428 -0
  64. package/src/islands/integration-loader.ts +490 -0
  65. package/src/islands/island.tsx +565 -0
  66. package/src/islands/render-cache.ts +550 -0
  67. package/src/islands/types.ts +80 -0
  68. package/src/islands/universal-css-collector.ts +157 -0
  69. package/src/islands/universal-head-collector.ts +137 -0
  70. package/src/layout-system.ts +218 -0
  71. package/src/middleware/discovery.ts +268 -0
  72. package/src/middleware/executor.ts +315 -0
  73. package/src/middleware/index.ts +76 -0
  74. package/src/middleware/types.ts +99 -0
  75. package/src/nitro/build-config.ts +576 -0
  76. package/src/nitro/config.ts +483 -0
  77. package/src/nitro/error-handler.ts +636 -0
  78. package/src/nitro/index.ts +173 -0
  79. package/src/nitro/island-manifest.ts +584 -0
  80. package/src/nitro/middleware-adapter.ts +260 -0
  81. package/src/nitro/renderer.ts +1471 -0
  82. package/src/nitro/route-discovery.ts +439 -0
  83. package/src/nitro/types.ts +321 -0
  84. package/src/render/collect-css.ts +198 -0
  85. package/src/render/error-pages.ts +79 -0
  86. package/src/render/isolated-ssr-renderer.ts +654 -0
  87. package/src/render/ssr.ts +1030 -0
  88. package/src/schemas/api.ts +30 -0
  89. package/src/schemas/core.ts +64 -0
  90. package/src/schemas/index.ts +212 -0
  91. package/src/schemas/layout.ts +279 -0
  92. package/src/schemas/routing/index.ts +38 -0
  93. package/src/schemas/routing.ts +376 -0
  94. package/src/types/as-island.ts +20 -0
  95. package/src/types/layout.ts +285 -0
  96. package/src/types/routing.ts +555 -0
  97. package/src/types/types.ts +5 -0
  98. package/src/utils/dev-logger.ts +299 -0
  99. package/src/utils/fs.ts +151 -0
  100. package/src/vite-plugin/auto-discover.ts +551 -0
  101. package/src/vite-plugin/config.ts +266 -0
  102. package/src/vite-plugin/errors.ts +127 -0
  103. package/src/vite-plugin/image-optimization.ts +156 -0
  104. package/src/vite-plugin/integration-activator.ts +126 -0
  105. package/src/vite-plugin/island-sidecar-plugin.ts +176 -0
  106. package/src/vite-plugin/module-discovery.ts +189 -0
  107. package/src/vite-plugin/nitro-integration.ts +1354 -0
  108. package/src/vite-plugin/plugin.ts +403 -0
  109. package/src/vite-plugin/types.ts +327 -0
  110. package/src/vite-plugin/validation.ts +228 -0
  111. package/dist/mod.js +0 -1
  112. package/dist/src/build/integration-bundler-plugin.js +0 -1
  113. package/dist/src/build/integration-config.js +0 -1
  114. package/dist/src/build/integration-detection-plugin.js +0 -1
  115. package/dist/src/build/integration-resolver-plugin.js +0 -1
  116. package/dist/src/build/island-manifest.js +0 -1
  117. package/dist/src/build/island-types-generator.js +0 -5
  118. package/dist/src/build/mdx-island-transform.js +0 -2
  119. package/dist/src/build/mdx-plugin.js +0 -1
  120. package/dist/src/build/page-island-transform.js +0 -3
  121. package/dist/src/build/prop-extractors/index.js +0 -1
  122. package/dist/src/build/prop-extractors/lit.js +0 -1
  123. package/dist/src/build/prop-extractors/qwik.js +0 -1
  124. package/dist/src/build/prop-extractors/solid.js +0 -1
  125. package/dist/src/build/prop-extractors/svelte.js +0 -1
  126. package/dist/src/build/prop-extractors/vue.js +0 -1
  127. package/dist/src/build/sidecar-file-manager.js +0 -1
  128. package/dist/src/build/sidecar-renderer.js +0 -6
  129. package/dist/src/client/adapters/index.js +0 -1
  130. package/dist/src/client/components.js +0 -1
  131. package/dist/src/client/css-hmr-handler.js +0 -1
  132. package/dist/src/client/framework-adapter.js +0 -13
  133. package/dist/src/client/hmr-coordinator.js +0 -1
  134. package/dist/src/client/hmr-error-overlay.js +0 -214
  135. package/dist/src/client/main.js +0 -39
  136. package/dist/src/components/Image.js +0 -1
  137. package/dist/src/components/IslandErrorBoundary.js +0 -1
  138. package/dist/src/components/LayoutDataErrorBoundary.js +0 -1
  139. package/dist/src/components/LayoutErrorBoundary.js +0 -1
  140. package/dist/src/components/PersistentIsland.js +0 -1
  141. package/dist/src/components/StreamingErrorBoundary.js +0 -1
  142. package/dist/src/components/StreamingLayout.js +0 -29
  143. package/dist/src/core/components/component-analyzer.js +0 -1
  144. package/dist/src/core/components/component-detection.js +0 -5
  145. package/dist/src/core/components/enhanced-framework-detector.js +0 -1
  146. package/dist/src/core/components/framework-registry.js +0 -1
  147. package/dist/src/core/content/mdx-processor.js +0 -1
  148. package/dist/src/core/integrations/index.js +0 -1
  149. package/dist/src/core/integrations/loader.js +0 -1
  150. package/dist/src/core/integrations/registry.js +0 -1
  151. package/dist/src/core/islands/island-persistence.js +0 -1
  152. package/dist/src/core/islands/island-state-serializer.js +0 -1
  153. package/dist/src/core/islands/persistent-island-context.js +0 -1
  154. package/dist/src/core/islands/use-persistent-state.js +0 -1
  155. package/dist/src/core/layout/enhanced-layout-resolver.js +0 -1
  156. package/dist/src/core/layout/layout-cache-manager.js +0 -1
  157. package/dist/src/core/layout/layout-composer.js +0 -1
  158. package/dist/src/core/layout/layout-data-loader.js +0 -1
  159. package/dist/src/core/layout/layout-discovery.js +0 -1
  160. package/dist/src/core/layout/layout-matcher.js +0 -1
  161. package/dist/src/core/layout/layout-types.js +0 -1
  162. package/dist/src/core/modules/framework-module-resolver.js +0 -1
  163. package/dist/src/islands/component-analysis.js +0 -1
  164. package/dist/src/islands/css-utils.js +0 -17
  165. package/dist/src/islands/discovery/index.js +0 -1
  166. package/dist/src/islands/discovery/registry.js +0 -1
  167. package/dist/src/islands/discovery/resolver.js +0 -2
  168. package/dist/src/islands/discovery/scanner.js +0 -1
  169. package/dist/src/islands/discovery/types.js +0 -1
  170. package/dist/src/islands/discovery/validator.js +0 -18
  171. package/dist/src/islands/discovery/watcher.js +0 -1
  172. package/dist/src/islands/framework-detection.js +0 -1
  173. package/dist/src/islands/integration-loader.js +0 -1
  174. package/dist/src/islands/island.js +0 -1
  175. package/dist/src/islands/render-cache.js +0 -1
  176. package/dist/src/islands/types.js +0 -1
  177. package/dist/src/islands/universal-css-collector.js +0 -5
  178. package/dist/src/islands/universal-head-collector.js +0 -2
  179. package/dist/src/layout-system.js +0 -1
  180. package/dist/src/middleware/discovery.js +0 -1
  181. package/dist/src/middleware/executor.js +0 -1
  182. package/dist/src/middleware/index.js +0 -1
  183. package/dist/src/middleware/types.js +0 -1
  184. package/dist/src/nitro/build-config.js +0 -1
  185. package/dist/src/nitro/config.js +0 -1
  186. package/dist/src/nitro/error-handler.js +0 -198
  187. package/dist/src/nitro/index.js +0 -1
  188. package/dist/src/nitro/island-manifest.js +0 -2
  189. package/dist/src/nitro/middleware-adapter.js +0 -1
  190. package/dist/src/nitro/renderer.js +0 -183
  191. package/dist/src/nitro/route-discovery.js +0 -1
  192. package/dist/src/nitro/types.js +0 -1
  193. package/dist/src/render/collect-css.js +0 -3
  194. package/dist/src/render/error-pages.js +0 -48
  195. package/dist/src/render/isolated-ssr-renderer.js +0 -1
  196. package/dist/src/render/ssr.js +0 -90
  197. package/dist/src/schemas/api.js +0 -1
  198. package/dist/src/schemas/core.js +0 -1
  199. package/dist/src/schemas/index.js +0 -1
  200. package/dist/src/schemas/layout.js +0 -1
  201. package/dist/src/schemas/routing/index.js +0 -1
  202. package/dist/src/schemas/routing.js +0 -1
  203. package/dist/src/types/as-island.js +0 -1
  204. package/dist/src/types/layout.js +0 -1
  205. package/dist/src/types/routing.js +0 -1
  206. package/dist/src/types/types.js +0 -1
  207. package/dist/src/utils/dev-logger.js +0 -12
  208. package/dist/src/utils/fs.js +0 -1
  209. package/dist/src/vite-plugin/auto-discover.js +0 -1
  210. package/dist/src/vite-plugin/config.js +0 -1
  211. package/dist/src/vite-plugin/errors.js +0 -1
  212. package/dist/src/vite-plugin/image-optimization.js +0 -45
  213. package/dist/src/vite-plugin/integration-activator.js +0 -1
  214. package/dist/src/vite-plugin/island-sidecar-plugin.js +0 -1
  215. package/dist/src/vite-plugin/module-discovery.js +0 -1
  216. package/dist/src/vite-plugin/nitro-integration.js +0 -42
  217. package/dist/src/vite-plugin/plugin.js +0 -1
  218. package/dist/src/vite-plugin/types.js +0 -1
  219. package/dist/src/vite-plugin/validation.js +0 -2
  220. /package/{dist/src → src}/client/types/framework-runtime.d.ts +0 -0
  221. /package/{dist/src → src}/client/types/vite-hmr.d.ts +0 -0
  222. /package/{dist/src → src}/client/types/vite-virtual-modules.d.ts +0 -0
  223. /package/{dist/src → src}/layout-system.d.ts +0 -0
  224. /package/{dist/src → src}/types/image.d.ts +0 -0
  225. /package/{dist/src → src}/types/index.d.ts +0 -0
  226. /package/{dist/src → src}/types/island-jsx.d.ts +0 -0
  227. /package/{dist/src → src}/types/island-prop.d.ts +0 -0
  228. /package/{dist/src → src}/types/mdx.d.ts +0 -0
  229. /package/{dist/src → src}/types/urlpattern.d.ts +0 -0
  230. /package/{dist/src → src}/types/vite-env.d.ts +0 -0
@@ -0,0 +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
+ }