@useavalon/avalon 0.1.10 → 0.1.12

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 (250) hide show
  1. package/README.md +54 -54
  2. package/dist/mod.js +1 -0
  3. package/dist/src/build/integration-bundler-plugin.js +1 -0
  4. package/dist/src/build/integration-config.js +1 -0
  5. package/dist/src/build/integration-detection-plugin.js +1 -0
  6. package/dist/src/build/integration-resolver-plugin.js +1 -0
  7. package/dist/src/build/island-manifest.js +1 -0
  8. package/dist/src/build/island-types-generator.js +5 -0
  9. package/dist/src/build/mdx-island-transform.js +2 -0
  10. package/dist/src/build/mdx-plugin.js +1 -0
  11. package/dist/src/build/page-island-transform.js +3 -0
  12. package/dist/src/build/prop-extractors/index.js +1 -0
  13. package/dist/src/build/prop-extractors/lit.js +1 -0
  14. package/dist/src/build/prop-extractors/qwik.js +1 -0
  15. package/dist/src/build/prop-extractors/solid.js +1 -0
  16. package/dist/src/build/prop-extractors/svelte.js +1 -0
  17. package/dist/src/build/prop-extractors/vue.js +1 -0
  18. package/dist/src/build/sidecar-file-manager.js +1 -0
  19. package/dist/src/build/sidecar-renderer.js +6 -0
  20. package/dist/src/client/adapters/index.js +1 -0
  21. package/dist/src/client/components.js +1 -0
  22. package/dist/src/client/css-hmr-handler.js +1 -0
  23. package/dist/src/client/framework-adapter.js +13 -0
  24. package/dist/src/client/hmr-coordinator.js +1 -0
  25. package/dist/src/client/hmr-error-overlay.js +214 -0
  26. package/dist/src/client/main.js +39 -0
  27. package/{src → dist/src}/client/types/framework-runtime.d.ts +68 -68
  28. package/{src → dist/src}/client/types/vite-hmr.d.ts +46 -46
  29. package/dist/src/client/types/vite-virtual-modules.d.ts +70 -0
  30. package/dist/src/components/Image.js +1 -0
  31. package/dist/src/components/IslandErrorBoundary.js +1 -0
  32. package/dist/src/components/LayoutDataErrorBoundary.js +1 -0
  33. package/dist/src/components/LayoutErrorBoundary.js +1 -0
  34. package/dist/src/components/PersistentIsland.js +1 -0
  35. package/dist/src/components/StreamingErrorBoundary.js +1 -0
  36. package/dist/src/components/StreamingLayout.js +29 -0
  37. package/dist/src/core/components/component-analyzer.js +1 -0
  38. package/dist/src/core/components/component-detection.js +5 -0
  39. package/dist/src/core/components/enhanced-framework-detector.js +1 -0
  40. package/dist/src/core/components/framework-registry.js +1 -0
  41. package/dist/src/core/content/mdx-processor.js +1 -0
  42. package/dist/src/core/integrations/index.js +1 -0
  43. package/dist/src/core/integrations/loader.js +1 -0
  44. package/dist/src/core/integrations/registry.js +1 -0
  45. package/dist/src/core/islands/island-persistence.js +1 -0
  46. package/dist/src/core/islands/island-state-serializer.js +1 -0
  47. package/dist/src/core/islands/persistent-island-context.js +1 -0
  48. package/dist/src/core/islands/use-persistent-state.js +1 -0
  49. package/dist/src/core/layout/enhanced-layout-resolver.js +1 -0
  50. package/dist/src/core/layout/layout-cache-manager.js +1 -0
  51. package/dist/src/core/layout/layout-composer.js +1 -0
  52. package/dist/src/core/layout/layout-data-loader.js +1 -0
  53. package/dist/src/core/layout/layout-discovery.js +1 -0
  54. package/dist/src/core/layout/layout-matcher.js +1 -0
  55. package/dist/src/core/layout/layout-types.js +1 -0
  56. package/dist/src/core/modules/framework-module-resolver.js +1 -0
  57. package/dist/src/islands/component-analysis.js +1 -0
  58. package/dist/src/islands/css-utils.js +17 -0
  59. package/dist/src/islands/discovery/index.js +1 -0
  60. package/dist/src/islands/discovery/registry.js +1 -0
  61. package/dist/src/islands/discovery/resolver.js +2 -0
  62. package/dist/src/islands/discovery/scanner.js +1 -0
  63. package/dist/src/islands/discovery/types.js +1 -0
  64. package/dist/src/islands/discovery/validator.js +18 -0
  65. package/dist/src/islands/discovery/watcher.js +1 -0
  66. package/dist/src/islands/framework-detection.js +1 -0
  67. package/dist/src/islands/integration-loader.js +1 -0
  68. package/dist/src/islands/island.js +1 -0
  69. package/dist/src/islands/render-cache.js +1 -0
  70. package/dist/src/islands/types.js +1 -0
  71. package/dist/src/islands/universal-css-collector.js +5 -0
  72. package/dist/src/islands/universal-head-collector.js +2 -0
  73. package/{src → dist/src}/layout-system.d.ts +592 -592
  74. package/dist/src/layout-system.js +1 -0
  75. package/dist/src/middleware/discovery.js +1 -0
  76. package/dist/src/middleware/executor.js +1 -0
  77. package/dist/src/middleware/index.js +1 -0
  78. package/dist/src/middleware/types.js +1 -0
  79. package/dist/src/nitro/build-config.js +1 -0
  80. package/dist/src/nitro/config.js +1 -0
  81. package/dist/src/nitro/error-handler.js +198 -0
  82. package/dist/src/nitro/index.js +1 -0
  83. package/dist/src/nitro/island-manifest.js +2 -0
  84. package/dist/src/nitro/middleware-adapter.js +1 -0
  85. package/dist/src/nitro/renderer.js +183 -0
  86. package/dist/src/nitro/route-discovery.js +1 -0
  87. package/dist/src/nitro/types.js +1 -0
  88. package/dist/src/render/collect-css.js +3 -0
  89. package/{src/render/error-pages.ts → dist/src/render/error-pages.js} +7 -38
  90. package/dist/src/render/isolated-ssr-renderer.js +1 -0
  91. package/dist/src/render/ssr.js +90 -0
  92. package/dist/src/schemas/api.js +1 -0
  93. package/dist/src/schemas/core.js +1 -0
  94. package/dist/src/schemas/index.js +1 -0
  95. package/dist/src/schemas/layout.js +1 -0
  96. package/dist/src/schemas/routing/index.js +1 -0
  97. package/dist/src/schemas/routing.js +1 -0
  98. package/dist/src/types/as-island.js +1 -0
  99. package/{src → dist/src}/types/image.d.ts +106 -106
  100. package/{src → dist/src}/types/index.d.ts +22 -22
  101. package/{src → dist/src}/types/island-jsx.d.ts +33 -33
  102. package/{src → dist/src}/types/island-prop.d.ts +20 -20
  103. package/dist/src/types/layout.js +1 -0
  104. package/{src → dist/src}/types/mdx.d.ts +6 -6
  105. package/dist/src/types/routing.js +1 -0
  106. package/dist/src/types/types.js +1 -0
  107. package/{src → dist/src}/types/urlpattern.d.ts +49 -49
  108. package/{src → dist/src}/types/vite-env.d.ts +11 -11
  109. package/dist/src/utils/dev-logger.js +12 -0
  110. package/dist/src/utils/fs.js +1 -0
  111. package/dist/src/vite-plugin/auto-discover.js +1 -0
  112. package/dist/src/vite-plugin/config.js +1 -0
  113. package/dist/src/vite-plugin/errors.js +1 -0
  114. package/dist/src/vite-plugin/image-optimization.js +45 -0
  115. package/dist/src/vite-plugin/integration-activator.js +1 -0
  116. package/dist/src/vite-plugin/island-sidecar-plugin.js +1 -0
  117. package/dist/src/vite-plugin/module-discovery.js +1 -0
  118. package/dist/src/vite-plugin/nitro-integration.js +42 -0
  119. package/dist/src/vite-plugin/plugin.js +1 -0
  120. package/dist/src/vite-plugin/types.js +1 -0
  121. package/dist/src/vite-plugin/validation.js +2 -0
  122. package/package.json +57 -26
  123. package/mod.ts +0 -302
  124. package/src/build/integration-bundler-plugin.ts +0 -116
  125. package/src/build/integration-config.ts +0 -168
  126. package/src/build/integration-detection-plugin.ts +0 -117
  127. package/src/build/integration-resolver-plugin.ts +0 -90
  128. package/src/build/island-manifest.ts +0 -269
  129. package/src/build/island-types-generator.ts +0 -476
  130. package/src/build/mdx-island-transform.ts +0 -464
  131. package/src/build/mdx-plugin.ts +0 -98
  132. package/src/build/page-island-transform.ts +0 -598
  133. package/src/build/prop-extractors/index.ts +0 -21
  134. package/src/build/prop-extractors/lit.ts +0 -140
  135. package/src/build/prop-extractors/qwik.ts +0 -16
  136. package/src/build/prop-extractors/solid.ts +0 -125
  137. package/src/build/prop-extractors/svelte.ts +0 -194
  138. package/src/build/prop-extractors/vue.ts +0 -111
  139. package/src/build/sidecar-file-manager.ts +0 -104
  140. package/src/build/sidecar-renderer.ts +0 -30
  141. package/src/client/adapters/index.js +0 -12
  142. package/src/client/adapters/index.ts +0 -13
  143. package/src/client/adapters/lit-adapter.js +0 -467
  144. package/src/client/adapters/lit-adapter.ts +0 -654
  145. package/src/client/adapters/preact-adapter.js +0 -223
  146. package/src/client/adapters/preact-adapter.ts +0 -331
  147. package/src/client/adapters/qwik-adapter.js +0 -259
  148. package/src/client/adapters/qwik-adapter.ts +0 -345
  149. package/src/client/adapters/react-adapter.js +0 -220
  150. package/src/client/adapters/react-adapter.ts +0 -353
  151. package/src/client/adapters/solid-adapter.js +0 -295
  152. package/src/client/adapters/solid-adapter.ts +0 -451
  153. package/src/client/adapters/svelte-adapter.js +0 -368
  154. package/src/client/adapters/svelte-adapter.ts +0 -524
  155. package/src/client/adapters/vue-adapter.js +0 -278
  156. package/src/client/adapters/vue-adapter.ts +0 -467
  157. package/src/client/components.js +0 -23
  158. package/src/client/components.ts +0 -35
  159. package/src/client/css-hmr-handler.js +0 -263
  160. package/src/client/css-hmr-handler.ts +0 -344
  161. package/src/client/framework-adapter.js +0 -283
  162. package/src/client/framework-adapter.ts +0 -462
  163. package/src/client/hmr-coordinator.js +0 -274
  164. package/src/client/hmr-coordinator.ts +0 -396
  165. package/src/client/hmr-error-overlay.js +0 -533
  166. package/src/client/main.js +0 -816
  167. package/src/client/types/vite-virtual-modules.d.ts +0 -60
  168. package/src/components/Image.tsx +0 -123
  169. package/src/components/IslandErrorBoundary.tsx +0 -145
  170. package/src/components/LayoutDataErrorBoundary.tsx +0 -141
  171. package/src/components/LayoutErrorBoundary.tsx +0 -127
  172. package/src/components/PersistentIsland.tsx +0 -52
  173. package/src/components/StreamingErrorBoundary.tsx +0 -233
  174. package/src/components/StreamingLayout.tsx +0 -538
  175. package/src/core/components/component-analyzer.ts +0 -192
  176. package/src/core/components/component-detection.ts +0 -508
  177. package/src/core/components/enhanced-framework-detector.ts +0 -500
  178. package/src/core/components/framework-registry.ts +0 -563
  179. package/src/core/content/mdx-processor.ts +0 -46
  180. package/src/core/integrations/index.ts +0 -19
  181. package/src/core/integrations/loader.ts +0 -125
  182. package/src/core/integrations/registry.ts +0 -175
  183. package/src/core/islands/island-persistence.ts +0 -325
  184. package/src/core/islands/island-state-serializer.ts +0 -258
  185. package/src/core/islands/persistent-island-context.tsx +0 -80
  186. package/src/core/islands/use-persistent-state.ts +0 -68
  187. package/src/core/layout/enhanced-layout-resolver.ts +0 -322
  188. package/src/core/layout/layout-cache-manager.ts +0 -485
  189. package/src/core/layout/layout-composer.ts +0 -357
  190. package/src/core/layout/layout-data-loader.ts +0 -516
  191. package/src/core/layout/layout-discovery.ts +0 -243
  192. package/src/core/layout/layout-matcher.ts +0 -299
  193. package/src/core/layout/layout-types.ts +0 -110
  194. package/src/core/modules/framework-module-resolver.ts +0 -273
  195. package/src/islands/component-analysis.ts +0 -213
  196. package/src/islands/css-utils.ts +0 -565
  197. package/src/islands/discovery/index.ts +0 -80
  198. package/src/islands/discovery/registry.ts +0 -340
  199. package/src/islands/discovery/resolver.ts +0 -477
  200. package/src/islands/discovery/scanner.ts +0 -386
  201. package/src/islands/discovery/types.ts +0 -117
  202. package/src/islands/discovery/validator.ts +0 -544
  203. package/src/islands/discovery/watcher.ts +0 -368
  204. package/src/islands/framework-detection.ts +0 -428
  205. package/src/islands/integration-loader.ts +0 -490
  206. package/src/islands/island.tsx +0 -565
  207. package/src/islands/render-cache.ts +0 -550
  208. package/src/islands/types.ts +0 -80
  209. package/src/islands/universal-css-collector.ts +0 -157
  210. package/src/islands/universal-head-collector.ts +0 -137
  211. package/src/layout-system.ts +0 -218
  212. package/src/middleware/discovery.ts +0 -268
  213. package/src/middleware/executor.ts +0 -315
  214. package/src/middleware/index.ts +0 -76
  215. package/src/middleware/types.ts +0 -99
  216. package/src/nitro/build-config.ts +0 -576
  217. package/src/nitro/config.ts +0 -483
  218. package/src/nitro/error-handler.ts +0 -636
  219. package/src/nitro/index.ts +0 -173
  220. package/src/nitro/island-manifest.ts +0 -584
  221. package/src/nitro/middleware-adapter.ts +0 -260
  222. package/src/nitro/renderer.ts +0 -1471
  223. package/src/nitro/route-discovery.ts +0 -439
  224. package/src/nitro/types.ts +0 -321
  225. package/src/render/collect-css.ts +0 -198
  226. package/src/render/isolated-ssr-renderer.ts +0 -654
  227. package/src/render/ssr.ts +0 -1030
  228. package/src/schemas/api.ts +0 -30
  229. package/src/schemas/core.ts +0 -64
  230. package/src/schemas/index.ts +0 -212
  231. package/src/schemas/layout.ts +0 -279
  232. package/src/schemas/routing/index.ts +0 -38
  233. package/src/schemas/routing.ts +0 -376
  234. package/src/types/as-island.ts +0 -20
  235. package/src/types/layout.ts +0 -285
  236. package/src/types/routing.ts +0 -555
  237. package/src/types/types.ts +0 -5
  238. package/src/utils/dev-logger.ts +0 -299
  239. package/src/utils/fs.ts +0 -151
  240. package/src/vite-plugin/auto-discover.ts +0 -551
  241. package/src/vite-plugin/config.ts +0 -266
  242. package/src/vite-plugin/errors.ts +0 -127
  243. package/src/vite-plugin/image-optimization.ts +0 -156
  244. package/src/vite-plugin/integration-activator.ts +0 -126
  245. package/src/vite-plugin/island-sidecar-plugin.ts +0 -176
  246. package/src/vite-plugin/module-discovery.ts +0 -189
  247. package/src/vite-plugin/nitro-integration.ts +0 -1354
  248. package/src/vite-plugin/plugin.ts +0 -401
  249. package/src/vite-plugin/types.ts +0 -327
  250. package/src/vite-plugin/validation.ts +0 -228
@@ -1,508 +0,0 @@
1
- /**
2
- * Component Detection System
3
- *
4
- * This module provides utilities to analyze components and determine their
5
- * hydration requirements based on script content and framework patterns.
6
- */
7
-
8
- export interface ComponentAnalysis {
9
- hasScript: boolean;
10
- hasHydrateFunction: boolean;
11
- framework: 'vue' | 'svelte' | 'solid' | 'preact' | 'react' | 'lit' | 'qwik' | 'unknown';
12
- recommendedStrategy: 'hydrate' | 'ssr-only';
13
- }
14
-
15
- export interface DetectionResult {
16
- shouldHydrate: boolean;
17
- reason: string;
18
- warnings?: string[];
19
- }
20
-
21
- export interface ComponentMetadata {
22
- path: string;
23
- framework: 'vue' | 'svelte' | 'solid' | 'preact' | 'react' | 'lit' | 'qwik';
24
- hasScript: boolean;
25
- hasHydrateFunction: boolean;
26
- renderStrategy: 'hydrate' | 'ssr-only';
27
- detectionConfidence: 'high' | 'medium' | 'low';
28
- }
29
-
30
- // Framework detection patterns
31
- const FRAMEWORK_PATTERNS = {
32
- vue: {
33
- fileExtensions: ['.vue'],
34
- scriptTags: ['<script>', '<script setup>', '<script lang="ts">', '<script setup lang="ts">'],
35
- hydratePatterns: ['hydrate', 'mount', 'createApp', 'Vue.createApp'],
36
- imports: ['vue', '@vue/', 'vue/'],
37
- },
38
- svelte: {
39
- fileExtensions: ['.svelte'],
40
- scriptTags: ['<script>', '<script lang="ts">', '<script context="module">'],
41
- hydratePatterns: ['hydrate', 'mount', '$:', 'onMount'],
42
- imports: ['svelte', 'svelte/'],
43
- },
44
- solid: {
45
- fileExtensions: ['.tsx', '.jsx', '.solid.tsx', '.solid.jsx'],
46
- scriptTags: [], // Solid uses JSX, no separate script tags
47
- hydratePatterns: ['hydrate', 'render', 'createSignal', 'createEffect'],
48
- imports: ['solid-js', 'solid-js/web'],
49
- },
50
- preact: {
51
- fileExtensions: ['.tsx', '.jsx', '.preact.tsx', '.preact.jsx'],
52
- scriptTags: [], // Preact uses JSX, no separate script tags
53
- hydratePatterns: ['hydrate', 'render'],
54
- imports: ['preact', 'preact/hooks'],
55
- },
56
- react: {
57
- fileExtensions: ['.tsx', '.jsx', '.react.tsx', '.react.jsx'],
58
- scriptTags: [], // React uses JSX, no separate script tags
59
- hydratePatterns: ['hydrate', 'render'],
60
- imports: ['react', 'react-dom'],
61
- },
62
- lit: {
63
- fileExtensions: ['.ts', '.js'],
64
- scriptTags: [], // Lit uses TypeScript/JavaScript classes
65
- hydratePatterns: ['LitElement', 'customElement', '@customElement'],
66
- imports: ['lit', 'lit-element', 'lit/'],
67
- },
68
- qwik: {
69
- fileExtensions: ['.tsx', '.jsx', '.qwik.tsx', '.qwik.jsx'],
70
- scriptTags: [], // Qwik uses JSX with component$
71
- hydratePatterns: ['component$', 'useSignal', 'useStore', 'useTask$', 'useVisibleTask$'],
72
- imports: ['@builder.io/qwik', '@builder.io/qwik/'],
73
- },
74
- } as const;
75
-
76
- /** Detect framework from explicit naming conventions in the file path */
77
- function detectByNamingConvention(filePath: string): ComponentAnalysis['framework'] | null {
78
- if (filePath.includes('.solid.')) return 'solid';
79
- if (filePath.includes('.preact.')) return 'preact';
80
- if (filePath.includes('.react.')) return 'react';
81
- if (filePath.includes('.qwik.')) return 'qwik';
82
- return null;
83
- }
84
-
85
- /** Detect framework from file extension alone */
86
- function detectByExtension(filePath: string, content: string): ComponentAnalysis['framework'] | null {
87
- if (filePath.endsWith('.vue')) return 'vue';
88
- if (filePath.endsWith('.svelte')) return 'svelte';
89
-
90
- if (filePath.endsWith('.ts') || filePath.endsWith('.js')) {
91
- if (content.includes('lit') || content.includes('LitElement') || content.includes('@customElement')) {
92
- return 'lit';
93
- }
94
- }
95
-
96
- if (filePath.endsWith('.tsx') || filePath.endsWith('.jsx')) {
97
- return detectJSXFramework(content);
98
- }
99
-
100
- return null;
101
- }
102
-
103
- /** Distinguish JSX frameworks by content patterns */
104
- function detectJSXFramework(content: string): ComponentAnalysis['framework'] {
105
- if (content.includes('solid-js') || content.includes('from "solid-js"') || content.includes("from 'solid-js'")) {
106
- return 'solid';
107
- }
108
- if (content.includes('@builder.io/qwik') || content.includes('from "@builder.io/qwik"') || content.includes("from '@builder.io/qwik'")) {
109
- return 'qwik';
110
- }
111
- if (content.includes('preact') || content.includes('from "preact"') || content.includes("from 'preact'")) {
112
- return 'preact';
113
- }
114
- if (content.includes('react') && !content.includes('preact')) {
115
- return 'react';
116
- }
117
- return 'preact';
118
- }
119
-
120
- /**
121
- * Detects the framework type based on file extension and content
122
- */
123
- export function detectFramework(filePath: string, content: string): ComponentAnalysis['framework'] {
124
- return (
125
- detectByNamingConvention(filePath) ??
126
- detectByExtension(filePath, content) ??
127
- detectByImports(content)
128
- );
129
- }
130
-
131
- /** Fallback: detect framework by scanning imports against known patterns */
132
- function detectByImports(content: string): ComponentAnalysis['framework'] {
133
- for (const [framework, patterns] of Object.entries(FRAMEWORK_PATTERNS)) {
134
- if (patterns.imports.some(importPattern => content.includes(importPattern))) {
135
- return framework as ComponentAnalysis['framework'];
136
- }
137
- }
138
- return 'unknown';
139
- }
140
-
141
- /**
142
- * Detects if a component has script sections
143
- */
144
- export function hasScriptSection(content: string, framework: ComponentAnalysis['framework']): boolean {
145
- switch (framework) {
146
- case 'vue':
147
- // Check for any <script> tag (more flexible than exact string matching)
148
- return /<script[^>]*>/i.test(content);
149
-
150
- case 'svelte':
151
- // Svelte components have script sections if they contain <script> tags
152
- // Also check for module context scripts
153
- return (
154
- content.includes('<script>') ||
155
- content.includes('<script ') ||
156
- content.includes('<script\n') ||
157
- content.includes('<script\t')
158
- );
159
-
160
- case 'solid':
161
- case 'preact':
162
- case 'react':
163
- case 'qwik':
164
- // JSX frameworks inherently have script content
165
- return (
166
- content.includes('function') || content.includes('=>') || content.includes('const') || content.includes('let')
167
- );
168
-
169
- case 'lit':
170
- // Lit components are TypeScript/JavaScript classes
171
- // They always have script content (class definitions)
172
- return (
173
- content.includes('class') || content.includes('LitElement') || content.includes('@customElement')
174
- );
175
-
176
- default:
177
- // For unknown frameworks, look for any script-like patterns
178
- return content.includes('<script>') || content.includes('function') || content.includes('=>');
179
- }
180
- }
181
-
182
- /**
183
- * Detects if a component has hydration functions
184
- */
185
- export function hasHydrateFunction(content: string, framework: ComponentAnalysis['framework']): boolean {
186
- const patterns = FRAMEWORK_PATTERNS[framework as keyof typeof FRAMEWORK_PATTERNS];
187
- if (!patterns) return false;
188
-
189
- // Look for explicit hydrate function exports or declarations
190
- const explicitHydratePatterns = [
191
- 'export function hydrate',
192
- 'export const hydrate',
193
- 'function hydrate(',
194
- 'const hydrate =',
195
- 'let hydrate =',
196
- 'var hydrate =',
197
- ];
198
-
199
- // Check for explicit hydrate functions first
200
- const hasExplicitHydrate = explicitHydratePatterns.some(pattern =>
201
- content.toLowerCase().includes(pattern.toLowerCase())
202
- );
203
-
204
- if (hasExplicitHydrate) return true;
205
-
206
- // For framework-specific patterns, be more selective
207
- switch (framework) {
208
- case 'vue':
209
- // Look for Vue-specific hydration patterns
210
- return content.includes('createApp') && content.includes('mount');
211
-
212
- case 'svelte':
213
- // Look for Svelte-specific hydration patterns - be more restrictive for explicit hydrate detection
214
- // Only consider it hydration-ready if it has explicit hydrate function
215
- // or uses Svelte's hydrate import
216
- return (
217
- content.includes('import') &&
218
- content.includes('hydrate') &&
219
- (content.includes('svelte') || content.includes('hydrate('))
220
- );
221
-
222
- case 'solid':
223
- // Look for Solid-specific hydration patterns
224
- return (
225
- (content.includes('hydrate') && content.includes('solid-js')) ||
226
- (content.includes('render') && content.includes('solid-js/web'))
227
- );
228
-
229
- case 'preact':
230
- // Look for Preact-specific hydration patterns
231
- return (
232
- (content.includes('hydrate') && content.includes('preact')) ||
233
- (content.includes('render') && content.includes('preact'))
234
- );
235
-
236
- case 'react':
237
- // Look for React-specific hydration patterns
238
- return (
239
- (content.includes('hydrate') && content.includes('react')) ||
240
- (content.includes('render') && content.includes('react-dom'))
241
- );
242
-
243
- case 'qwik':
244
- // Qwik uses resumability instead of hydration — component$ is the marker
245
- return (
246
- content.includes('component$') ||
247
- content.includes('@builder.io/qwik')
248
- );
249
-
250
- default:
251
- return false;
252
- }
253
- }
254
-
255
- /**
256
- * Extracts script content from Vue components
257
- */
258
- export function extractVueScript(content: string): string {
259
- return extractScriptTags(content);
260
- }
261
-
262
- /**
263
- * Extracts script content from Svelte components
264
- */
265
- export function extractSvelteScript(content: string): string {
266
- return extractScriptTags(content);
267
- }
268
-
269
- /** Shared: extract `<script>` tag contents from SFC-style components */
270
- function extractScriptTags(content: string): string {
271
- const scriptRegex = /<script[^>]*>([\s\S]*?)<\/script>/gi;
272
- const results: string[] = [];
273
- let match;
274
- while ((match = scriptRegex.exec(content)) !== null) {
275
- results.push(match[0]);
276
- }
277
- return results.join('\n');
278
- }
279
-
280
- const JSX_RETURN_RE = /^\s*return\s*\(/;
281
-
282
- /** Shared: extract script-like lines from JSX component files */
283
- function extractJSXScript(content: string): string {
284
- return content
285
- .split('\n')
286
- .filter(line => {
287
- const trimmed = line.trim();
288
- return !trimmed.startsWith('<') && !trimmed.startsWith('</') && !JSX_RETURN_RE.exec(trimmed);
289
- })
290
- .join('\n');
291
- }
292
-
293
- /**
294
- * Extracts script content from Solid components (JSX)
295
- */
296
- export function extractSolidScript(content: string): string {
297
- return extractJSXScript(content);
298
- }
299
-
300
- /**
301
- * Extracts script content from Preact components (JSX)
302
- */
303
- export function extractPreactScript(content: string): string {
304
- return extractJSXScript(content);
305
- }
306
-
307
- function extractQwikScript(content: string): string {
308
- return extractJSXScript(content);
309
- }
310
-
311
- /**
312
- * Extracts script content from React components (JSX)
313
- */
314
- export function extractReactScript(content: string): string {
315
- return extractJSXScript(content);
316
- }
317
-
318
- /**
319
- * Analyzes a component and returns detailed analysis
320
- */
321
- export function analyzeComponent(filePath: string, content: string): ComponentAnalysis {
322
- const framework = detectFramework(filePath, content);
323
- const hasScript = hasScriptSection(content, framework);
324
- const hasHydrate = hasScript ? hasHydrateFunction(content, framework) : false;
325
-
326
- // Determine recommended strategy with framework-specific logic
327
- let recommendedStrategy: ComponentAnalysis['recommendedStrategy'];
328
-
329
- if (hasScript) {
330
- // Framework-specific hydration strategy decisions
331
- switch (framework) {
332
- case 'svelte':
333
- // Svelte components with script sections are typically interactive
334
- // Use more intelligent detection based on content patterns
335
- recommendedStrategy = analyzeSvelteHydrationStrategy(content, hasHydrate);
336
- break;
337
-
338
- case 'solid':
339
- case 'preact':
340
- case 'react':
341
- // JSX frameworks typically need hydration if they have script content
342
- recommendedStrategy = 'hydrate';
343
- break;
344
-
345
- case 'qwik':
346
- // Qwik uses resumability — components with component$ always resume on client
347
- recommendedStrategy = 'hydrate';
348
- break;
349
-
350
- case 'vue':
351
- // Vue components need explicit hydration indicators
352
- recommendedStrategy = hasHydrate ? 'hydrate' : 'ssr-only';
353
- break;
354
-
355
- default:
356
- // Conservative approach for unknown frameworks
357
- recommendedStrategy = hasHydrate ? 'hydrate' : 'ssr-only';
358
- break;
359
- }
360
- } else {
361
- // No script = pure template component = SSR-only
362
- recommendedStrategy = 'ssr-only';
363
- }
364
-
365
- return {
366
- hasScript,
367
- hasHydrateFunction: hasHydrate,
368
- framework,
369
- recommendedStrategy,
370
- };
371
- }
372
-
373
- /**
374
- * Analyzes Svelte component to determine if it needs hydration
375
- */
376
- function analyzeSvelteHydrationStrategy(content: string, hasExplicitHydrate: boolean): ComponentAnalysis['recommendedStrategy'] {
377
- // If there's an explicit hydrate function, definitely hydrate
378
- if (hasExplicitHydrate) {
379
- return 'hydrate';
380
- }
381
-
382
- // Check for interactive patterns that indicate need for hydration
383
- const interactivePatterns = [
384
- 'on:', // Event handlers (Svelte 4 style)
385
- 'onclick', // Event handlers (HTML style)
386
- 'onchange',
387
- 'oninput',
388
- 'onsubmit',
389
- 'onkeydown',
390
- 'onkeyup',
391
- 'onmousedown',
392
- 'onmouseup',
393
- 'bind:', // Two-way bindings
394
- '$:', // Reactive statements
395
- '$state', // Svelte 5 runes
396
- '$derived',
397
- '$effect',
398
- '$props',
399
- 'onMount', // Lifecycle functions
400
- 'onDestroy',
401
- 'beforeUpdate',
402
- 'afterUpdate',
403
- 'tick',
404
- 'writable', // Stores
405
- 'readable',
406
- 'derived',
407
- 'get(',
408
- 'set(',
409
- 'update(',
410
- 'subscribe(',
411
- ];
412
-
413
- const hasInteractiveFeatures = interactivePatterns.some(pattern => content.includes(pattern));
414
-
415
- // Check for static-only patterns
416
- const staticPatterns = [
417
- 'export let', // Only props, no interactivity
418
- ];
419
-
420
- const isLikelyStatic = staticPatterns.some(pattern => content.includes(pattern)) && !hasInteractiveFeatures;
421
-
422
- if (isLikelyStatic) {
423
- return 'ssr-only';
424
- }
425
-
426
- // Default to hydration for Svelte components with script sections
427
- // This is safer and aligns with Svelte's typical usage patterns
428
- return 'hydrate';
429
- }
430
-
431
- /** Framework-specific hydration reasons for known frameworks with script sections */
432
- const FRAMEWORK_HYDRATION_REASONS: Record<string, string> = {
433
- solid: 'SolidJS component detected - uses integration system',
434
- preact: 'preact component with script content - likely needs hydration',
435
- react: 'react component with script content - likely needs hydration',
436
- svelte: 'Svelte component with script section - uses Svelte hydration system',
437
- vue: 'Vue component with script section - uses Vue integration system',
438
- lit: 'Lit component detected - Web Components require client-side registration',
439
- qwik: 'Qwik component detected - uses resumability instead of hydration',
440
- };
441
-
442
- /**
443
- * Determines if a component should be hydrated based on analysis
444
- */
445
- export function shouldHydrateComponent(
446
- analysis: ComponentAnalysis,
447
- options: { forceSSROnly?: boolean; detectScripts?: boolean } = {}
448
- ): DetectionResult {
449
- if (options.forceSSROnly) {
450
- return { shouldHydrate: false, reason: 'Explicitly configured for SSR-only rendering' };
451
- }
452
-
453
- if (options.detectScripts === false) {
454
- return { shouldHydrate: true, reason: 'Script detection disabled, defaulting to hydration' };
455
- }
456
-
457
- if (!analysis.hasScript) {
458
- return { shouldHydrate: false, reason: 'No script section detected, using SSR-only rendering' };
459
- }
460
-
461
- // Known frameworks with script sections always hydrate
462
- const frameworkReason = FRAMEWORK_HYDRATION_REASONS[analysis.framework];
463
- if (frameworkReason) {
464
- return { shouldHydrate: true, reason: frameworkReason };
465
- }
466
-
467
- // Unknown frameworks need explicit hydrate functions
468
- if (analysis.hasHydrateFunction) {
469
- return { shouldHydrate: true, reason: 'Component has script section with explicit hydration functions' };
470
- }
471
-
472
- return {
473
- shouldHydrate: false,
474
- reason: 'Component has script section but no explicit hydrate function, using SSR-only',
475
- warnings: ['Component has script section but no clear hydrate function detected'],
476
- };
477
- }
478
-
479
- /**
480
- * Creates component metadata with confidence scoring
481
- */
482
- export function createComponentMetadata(
483
- filePath: string,
484
- content: string,
485
- analysis: ComponentAnalysis
486
- ): ComponentMetadata {
487
- let detectionConfidence: ComponentMetadata['detectionConfidence'] = 'medium';
488
-
489
- // High confidence cases
490
- if (analysis.framework !== 'unknown' && analysis.hasScript && analysis.hasHydrateFunction) {
491
- detectionConfidence = 'high';
492
- } else if (analysis.framework !== 'unknown' && !analysis.hasScript) {
493
- detectionConfidence = 'high';
494
- }
495
- // Low confidence cases
496
- else if (analysis.framework === 'unknown') {
497
- detectionConfidence = 'low';
498
- }
499
-
500
- return {
501
- path: filePath,
502
- framework: analysis.framework === 'unknown' ? 'vue' : analysis.framework, // Default fallback to vue for backward compatibility
503
- hasScript: analysis.hasScript,
504
- hasHydrateFunction: analysis.hasHydrateFunction,
505
- renderStrategy: analysis.recommendedStrategy,
506
- detectionConfidence,
507
- };
508
- }