vinext 0.0.15 → 0.0.16

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 (44) hide show
  1. package/README.md +6 -0
  2. package/dist/cli.js +21 -11
  3. package/dist/cli.js.map +1 -1
  4. package/dist/config/config-matchers.d.ts +4 -1
  5. package/dist/config/config-matchers.d.ts.map +1 -1
  6. package/dist/config/config-matchers.js +11 -1
  7. package/dist/config/config-matchers.js.map +1 -1
  8. package/dist/config/next-config.d.ts +2 -0
  9. package/dist/config/next-config.d.ts.map +1 -1
  10. package/dist/config/next-config.js.map +1 -1
  11. package/dist/deploy.d.ts.map +1 -1
  12. package/dist/deploy.js +2 -0
  13. package/dist/deploy.js.map +1 -1
  14. package/dist/index.d.ts.map +1 -1
  15. package/dist/index.js +68 -57
  16. package/dist/index.js.map +1 -1
  17. package/dist/server/app-dev-server.d.ts.map +1 -1
  18. package/dist/server/app-dev-server.js +25 -9
  19. package/dist/server/app-dev-server.js.map +1 -1
  20. package/dist/server/middleware.d.ts +23 -1
  21. package/dist/server/middleware.d.ts.map +1 -1
  22. package/dist/server/middleware.js +42 -8
  23. package/dist/server/middleware.js.map +1 -1
  24. package/dist/server/prod-server.js +1 -1
  25. package/dist/server/prod-server.js.map +1 -1
  26. package/dist/shims/fetch-cache.d.ts.map +1 -1
  27. package/dist/shims/fetch-cache.js +73 -41
  28. package/dist/shims/fetch-cache.js.map +1 -1
  29. package/dist/shims/font-google-base.d.ts +68 -0
  30. package/dist/shims/font-google-base.d.ts.map +1 -0
  31. package/dist/shims/font-google-base.js +365 -0
  32. package/dist/shims/font-google-base.js.map +1 -0
  33. package/dist/shims/font-google.d.ts +2 -121
  34. package/dist/shims/font-google.d.ts.map +1 -1
  35. package/dist/shims/font-google.generated.d.ts +1925 -0
  36. package/dist/shims/font-google.generated.d.ts.map +1 -0
  37. package/dist/shims/font-google.generated.js +1928 -0
  38. package/dist/shims/font-google.generated.js.map +1 -0
  39. package/dist/shims/font-google.js +2 -386
  40. package/dist/shims/font-google.js.map +1 -1
  41. package/dist/utils/project.d.ts.map +1 -1
  42. package/dist/utils/project.js +4 -0
  43. package/dist/utils/project.js.map +1 -1
  44. package/package.json +2 -1
@@ -1,387 +1,3 @@
1
- /**
2
- * next/font/google shim
3
- *
4
- * Provides a compatible shim for Next.js Google Fonts.
5
- *
6
- * Two modes:
7
- * 1. **Dev / CDN mode** (default): Loads fonts from Google Fonts CDN via <link> tags.
8
- * 2. **Self-hosted mode** (production build): The vinext:google-fonts Vite plugin
9
- * fetches font CSS + .woff2 files at build time, caches them locally, and injects
10
- * @font-face CSS pointing at local assets. No requests to Google at runtime.
11
- *
12
- * Usage:
13
- * import { Inter } from 'next/font/google';
14
- * const inter = Inter({ subsets: ['latin'], weight: ['400', '700'] });
15
- * // inter.className -> unique CSS class
16
- * // inter.style -> { fontFamily: "'Inter', sans-serif" }
17
- * // inter.variable -> CSS variable name like '--font-inter'
18
- */
19
- /**
20
- * Escape a string for safe interpolation inside a CSS single-quoted string.
21
- *
22
- * Prevents CSS injection by escaping characters that could break out of
23
- * a `'...'` CSS string context: backslashes, single quotes, and newlines.
24
- */
25
- function escapeCSSString(value) {
26
- return value
27
- .replace(/\\/g, "\\\\")
28
- .replace(/'/g, "\\'")
29
- .replace(/\n/g, "\\a ")
30
- .replace(/\r/g, "\\d ");
31
- }
32
- /**
33
- * Validate a CSS custom property name (e.g. `--font-inter`).
34
- *
35
- * Custom properties must start with `--` and only contain alphanumeric
36
- * characters, hyphens, and underscores. Anything else could be used to
37
- * break out of the CSS declaration and inject arbitrary rules.
38
- *
39
- * Returns the name if valid, undefined otherwise.
40
- */
41
- function sanitizeCSSVarName(name) {
42
- if (/^--[a-zA-Z0-9_-]+$/.test(name))
43
- return name;
44
- return undefined;
45
- }
46
- /**
47
- * Sanitize a CSS font-family fallback name.
48
- *
49
- * Generic family names (sans-serif, serif, monospace, etc.) are used as-is.
50
- * Named families are wrapped in escaped quotes. This prevents injection via
51
- * crafted fallback values like `); } body { color: red; } .x {`.
52
- */
53
- function sanitizeFallback(name) {
54
- // CSS generic font families — safe to use unquoted
55
- const generics = new Set([
56
- "serif", "sans-serif", "monospace", "cursive", "fantasy",
57
- "system-ui", "ui-serif", "ui-sans-serif", "ui-monospace", "ui-rounded",
58
- "emoji", "math", "fangsong",
59
- ]);
60
- const trimmed = name.trim();
61
- if (generics.has(trimmed))
62
- return trimmed;
63
- // Wrap in single quotes with escaping to prevent CSS injection
64
- return `'${escapeCSSString(trimmed)}'`;
65
- }
66
- // Counter for generating unique class names
67
- let classCounter = 0;
68
- // Track which font stylesheets have been injected (SSR + client)
69
- const injectedFonts = new Set();
70
- /**
71
- * Convert a font family name to a CSS variable name.
72
- * e.g., "Inter" -> "--font-inter", "Roboto Mono" -> "--font-roboto-mono"
73
- */
74
- function toVarName(family) {
75
- return "--font-" + family.toLowerCase().replace(/\s+/g, "-");
76
- }
77
- /**
78
- * Build a Google Fonts CSS URL.
79
- */
80
- function buildGoogleFontsUrl(family, options) {
81
- const params = new URLSearchParams();
82
- // Don't pre-replace spaces with "+". URLSearchParams handles encoding:
83
- // spaces become "+" in application/x-www-form-urlencoded format.
84
- // Pre-replacing would cause double-encoding: "+" -> "%2B" (400 error).
85
- let spec = family;
86
- // Build weight/style specs
87
- const weights = options.weight
88
- ? Array.isArray(options.weight)
89
- ? options.weight
90
- : [options.weight]
91
- : [];
92
- const styles = options.style
93
- ? Array.isArray(options.style)
94
- ? options.style
95
- : [options.style]
96
- : [];
97
- if (weights.length > 0 || styles.length > 0) {
98
- const hasItalic = styles.includes("italic");
99
- if (weights.length > 0) {
100
- if (hasItalic) {
101
- // Use ital axis: ital,wght@0,400;0,700;1,400;1,700
102
- const pairs = [];
103
- for (const w of weights) {
104
- pairs.push(`0,${w}`);
105
- pairs.push(`1,${w}`);
106
- }
107
- spec += `:ital,wght@${pairs.join(";")}`;
108
- }
109
- else {
110
- spec += `:wght@${weights.join(";")}`;
111
- }
112
- }
113
- }
114
- else {
115
- // When no weight is specified, request the full variable weight range.
116
- // Without this, Google Fonts returns only weight 400 (the default).
117
- // Next.js loads the full variable font by default, so we match that
118
- // behavior to ensure all font weights render correctly.
119
- spec += `:wght@100..900`;
120
- }
121
- params.set("family", spec);
122
- params.set("display", options.display ?? "swap");
123
- return `https://fonts.googleapis.com/css2?${params.toString()}`;
124
- }
125
- /**
126
- * Inject a <link> tag for the font (client-side only).
127
- * On the server, we track font URLs for SSR head injection.
128
- */
129
- function injectFontStylesheet(url) {
130
- if (injectedFonts.has(url))
131
- return;
132
- injectedFonts.add(url);
133
- if (typeof document !== "undefined") {
134
- const link = document.createElement("link");
135
- link.rel = "stylesheet";
136
- link.href = url;
137
- document.head.appendChild(link);
138
- }
139
- }
140
- /** Track which className CSS rules have been injected. */
141
- const injectedClassRules = new Set();
142
- /**
143
- * Inject a CSS rule that maps a className to a font-family.
144
- *
145
- * This is what makes `<div className={inter.className}>` apply the font.
146
- * Next.js generates equivalent rules at build time.
147
- *
148
- * In Next.js, the .className class ONLY sets font-family — it does NOT
149
- * set CSS variables. CSS variables are handled separately by the .variable class.
150
- */
151
- function injectClassNameRule(className, fontFamily) {
152
- if (injectedClassRules.has(className))
153
- return;
154
- injectedClassRules.add(className);
155
- const css = `.${className} { font-family: ${fontFamily}; }\n`;
156
- // On server, store the CSS for SSR injection
157
- if (typeof document === "undefined") {
158
- ssrFontStyles.push(css);
159
- return;
160
- }
161
- // On client, inject a <style> tag
162
- const style = document.createElement("style");
163
- style.textContent = css;
164
- style.setAttribute("data-vinext-font-class", className);
165
- document.head.appendChild(style);
166
- }
167
- /** Track which variable class CSS rules have been injected. */
168
- const injectedVariableRules = new Set();
169
- /** Track which :root CSS variable rules have been injected. */
170
- const injectedRootVariables = new Set();
171
- /**
172
- * Inject a CSS rule that sets a CSS variable on an element.
173
- * This is what makes `<html className={inter.variable}>` set the CSS variable
174
- * that can be referenced by other styles (e.g., Tailwind's font-sans).
175
- *
176
- * In Next.js, the .variable class ONLY sets the CSS variable — it does NOT
177
- * set font-family. This is critical because apps commonly apply multiple
178
- * .variable classes to <body> (e.g., geistSans.variable + geistMono.variable).
179
- * If we also set font-family here, the last class wins due to CSS cascade,
180
- * causing all text to use that font (e.g., everything becomes monospace).
181
- */
182
- function injectVariableClassRule(variableClassName, cssVarName, fontFamily) {
183
- if (injectedVariableRules.has(variableClassName))
184
- return;
185
- injectedVariableRules.add(variableClassName);
186
- // Only set the CSS variable — do NOT set font-family.
187
- // This matches Next.js behavior where .variable classes only define CSS variables.
188
- let css = `.${variableClassName} { ${cssVarName}: ${fontFamily}; }\n`;
189
- // Also inject at :root so CSS variable inheritance works throughout the page.
190
- // This ensures Tailwind utilities like `font-sans` that reference these
191
- // variables via var(--font-geist-sans) work correctly.
192
- if (!injectedRootVariables.has(cssVarName)) {
193
- injectedRootVariables.add(cssVarName);
194
- css += `:root { ${cssVarName}: ${fontFamily}; }\n`;
195
- }
196
- // On server, store the CSS for SSR injection
197
- if (typeof document === "undefined") {
198
- ssrFontStyles.push(css);
199
- return;
200
- }
201
- // On client, inject a <style> tag
202
- const style = document.createElement("style");
203
- style.textContent = css;
204
- style.setAttribute("data-vinext-font-variable", variableClassName);
205
- document.head.appendChild(style);
206
- }
207
- // SSR: collect font class CSS for injection in <head>
208
- const ssrFontStyles = [];
209
- /**
210
- * Get collected SSR font class styles (used by the renderer).
211
- * Note: We don't clear the arrays because fonts are loaded at module import
212
- * time and need to persist across all requests in the Workers environment.
213
- */
214
- export function getSSRFontStyles() {
215
- return [...ssrFontStyles];
216
- }
217
- // SSR: collect font URLs to inject in <head>
218
- const ssrFontUrls = [];
219
- /**
220
- * Get collected SSR font URLs (used by the renderer).
221
- * Note: We don't clear the arrays because fonts are loaded at module import
222
- * time and need to persist across all requests in the Workers environment.
223
- */
224
- export function getSSRFontLinks() {
225
- return [...ssrFontUrls];
226
- }
227
- // SSR: collect font file URLs for <link rel="preload"> injection (self-hosted Google fonts)
228
- const ssrFontPreloads = [];
229
- const ssrFontPreloadHrefs = new Set();
230
- /**
231
- * Get collected SSR font preload data (used by the renderer).
232
- * Returns an array of { href, type } objects for emitting
233
- * <link rel="preload" as="font" ...> tags.
234
- */
235
- export function getSSRFontPreloads() {
236
- return [...ssrFontPreloads];
237
- }
238
- /**
239
- * Determine the MIME type for a font file based on its extension.
240
- */
241
- function getFontMimeType(pathOrUrl) {
242
- if (pathOrUrl.endsWith(".woff2"))
243
- return "font/woff2";
244
- if (pathOrUrl.endsWith(".woff"))
245
- return "font/woff";
246
- if (pathOrUrl.endsWith(".ttf"))
247
- return "font/ttf";
248
- if (pathOrUrl.endsWith(".otf"))
249
- return "font/opentype";
250
- return "font/woff2";
251
- }
252
- /**
253
- * Extract font file URLs from @font-face CSS rules.
254
- * Parses url('...') references from the CSS text.
255
- */
256
- function extractFontUrlsFromCSS(css) {
257
- const urls = [];
258
- const urlRegex = /url\(['"]?([^'")]+)['"]?\)/g;
259
- let match;
260
- while ((match = urlRegex.exec(css)) !== null) {
261
- const url = match[1];
262
- // Only collect absolute paths (starting with /) — these are self-hosted font files
263
- if (url && url.startsWith("/")) {
264
- urls.push(url);
265
- }
266
- }
267
- return urls;
268
- }
269
- /**
270
- * Collect font file URLs from self-hosted CSS for preload link generation.
271
- * Only collects on the server (SSR). Deduplicates by href using a Set for O(1) lookups.
272
- */
273
- function collectFontPreloadsFromCSS(css) {
274
- if (typeof document !== "undefined")
275
- return; // client-side, skip
276
- const urls = extractFontUrlsFromCSS(css);
277
- for (const href of urls) {
278
- if (!ssrFontPreloadHrefs.has(href)) {
279
- ssrFontPreloadHrefs.add(href);
280
- ssrFontPreloads.push({ href, type: getFontMimeType(href) });
281
- }
282
- }
283
- }
284
- /** Track injected self-hosted @font-face blocks (deduplicate) */
285
- const injectedSelfHosted = new Set();
286
- /**
287
- * Inject self-hosted @font-face CSS (from the build plugin).
288
- * This replaces the CDN <link> tag with inline CSS.
289
- */
290
- function injectSelfHostedCSS(css) {
291
- if (injectedSelfHosted.has(css))
292
- return;
293
- injectedSelfHosted.add(css);
294
- // Extract font file URLs for preload hints (SSR only)
295
- collectFontPreloadsFromCSS(css);
296
- if (typeof document === "undefined") {
297
- // SSR: add to collected styles
298
- ssrFontStyles.push(css);
299
- return;
300
- }
301
- // Client: inject <style> tag
302
- const style = document.createElement("style");
303
- style.textContent = css;
304
- style.setAttribute("data-vinext-font-selfhosted", "true");
305
- document.head.appendChild(style);
306
- }
307
- function createFontLoader(family) {
308
- return function fontLoader(options = {}) {
309
- const id = classCounter++;
310
- const className = `__font_${family.toLowerCase().replace(/\s+/g, "_")}_${id}`;
311
- const fallback = options.fallback ?? ["sans-serif"];
312
- // Sanitize each fallback name to prevent CSS injection via crafted values
313
- const fontFamily = `'${escapeCSSString(family)}', ${fallback.map(sanitizeFallback).join(", ")}`;
314
- // Validate CSS variable name — reject anything that could inject CSS.
315
- // Fall back to auto-generated name if invalid.
316
- const defaultVarName = toVarName(family);
317
- const cssVarName = options.variable ? (sanitizeCSSVarName(options.variable) ?? defaultVarName) : defaultVarName;
318
- // In Next.js, `variable` returns a CLASS NAME that sets the CSS variable.
319
- // Users apply this class to set the CSS variable on that element.
320
- const variableClassName = `__variable_${family.toLowerCase().replace(/\s+/g, "_")}_${id}`;
321
- if (options._selfHostedCSS) {
322
- // Self-hosted mode: inject local @font-face CSS instead of CDN link
323
- injectSelfHostedCSS(options._selfHostedCSS);
324
- }
325
- else {
326
- // CDN mode: inject <link> to Google Fonts
327
- const url = buildGoogleFontsUrl(family, options);
328
- injectFontStylesheet(url);
329
- // On SSR, collect the URL for head injection
330
- if (typeof document === "undefined") {
331
- if (!ssrFontUrls.includes(url)) {
332
- ssrFontUrls.push(url);
333
- }
334
- }
335
- }
336
- // Inject a CSS rule that maps className to font-family.
337
- // This is what makes `<div className={inter.className}>` work.
338
- injectClassNameRule(className, fontFamily);
339
- // Inject a CSS rule for the variable class name.
340
- // This is what makes `<html className={inter.variable}>` set the CSS variable.
341
- injectVariableClassRule(variableClassName, cssVarName, fontFamily);
342
- return {
343
- className,
344
- style: { fontFamily },
345
- variable: variableClassName,
346
- };
347
- };
348
- }
349
- // Re-export for plugin use
350
- export { buildGoogleFontsUrl };
351
- // Export a Proxy that creates font loaders for any Google Font family.
352
- // Usage: import { Inter } from 'next/font/google'
353
- // The proxy intercepts property access and returns a loader for that font.
354
- const googleFonts = new Proxy({}, {
355
- get(_target, prop) {
356
- if (prop === "__esModule")
357
- return true;
358
- if (prop === "default")
359
- return googleFonts;
360
- // Convert camelCase/PascalCase to proper font family name
361
- // e.g., "Inter" -> "Inter", "RobotoMono" -> "Roboto Mono"
362
- const family = prop.replace(/([a-z])([A-Z])/g, "$1 $2");
363
- return createFontLoader(family);
364
- },
365
- });
366
- export default googleFonts;
367
- // Named exports for common fonts (provides better IDE autocomplete)
368
- export const Inter = createFontLoader("Inter");
369
- export const Roboto = createFontLoader("Roboto");
370
- export const Roboto_Mono = createFontLoader("Roboto Mono");
371
- export const Open_Sans = createFontLoader("Open Sans");
372
- export const Lato = createFontLoader("Lato");
373
- export const Poppins = createFontLoader("Poppins");
374
- export const Montserrat = createFontLoader("Montserrat");
375
- export const Source_Code_Pro = createFontLoader("Source Code Pro");
376
- export const Noto_Sans = createFontLoader("Noto Sans");
377
- export const Raleway = createFontLoader("Raleway");
378
- export const Ubuntu = createFontLoader("Ubuntu");
379
- export const Nunito = createFontLoader("Nunito");
380
- export const Playfair_Display = createFontLoader("Playfair Display");
381
- export const Merriweather = createFontLoader("Merriweather");
382
- export const PT_Sans = createFontLoader("PT Sans");
383
- export const Fira_Code = createFontLoader("Fira Code");
384
- export const JetBrains_Mono = createFontLoader("JetBrains Mono");
385
- export const Geist = createFontLoader("Geist");
386
- export const Geist_Mono = createFontLoader("Geist Mono");
1
+ export { default, buildGoogleFontsUrl, getSSRFontLinks, getSSRFontStyles, getSSRFontPreloads } from "./font-google-base";
2
+ export * from "./font-google.generated";
387
3
  //# sourceMappingURL=font-google.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"font-google.js","sourceRoot":"","sources":["../../src/shims/font-google.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * next/font/google shim\n *\n * Provides a compatible shim for Next.js Google Fonts.\n *\n * Two modes:\n * 1. **Dev / CDN mode** (default): Loads fonts from Google Fonts CDN via <link> tags.\n * 2. **Self-hosted mode** (production build): The vinext:google-fonts Vite plugin\n * fetches font CSS + .woff2 files at build time, caches them locally, and injects\n * @font-face CSS pointing at local assets. No requests to Google at runtime.\n *\n * Usage:\n * import { Inter } from 'next/font/google';\n * const inter = Inter({ subsets: ['latin'], weight: ['400', '700'] });\n * // inter.className -> unique CSS class\n * // inter.style -> { fontFamily: \"'Inter', sans-serif\" }\n * // inter.variable -> CSS variable name like '--font-inter'\n */\n\n/**\n * Escape a string for safe interpolation inside a CSS single-quoted string.\n *\n * Prevents CSS injection by escaping characters that could break out of\n * a `'...'` CSS string context: backslashes, single quotes, and newlines.\n */\nfunction escapeCSSString(value: string): string {\n return value\n .replace(/\\\\/g, \"\\\\\\\\\")\n .replace(/'/g, \"\\\\'\")\n .replace(/\\n/g, \"\\\\a \")\n .replace(/\\r/g, \"\\\\d \");\n}\n\n/**\n * Validate a CSS custom property name (e.g. `--font-inter`).\n *\n * Custom properties must start with `--` and only contain alphanumeric\n * characters, hyphens, and underscores. Anything else could be used to\n * break out of the CSS declaration and inject arbitrary rules.\n *\n * Returns the name if valid, undefined otherwise.\n */\nfunction sanitizeCSSVarName(name: string): string | undefined {\n if (/^--[a-zA-Z0-9_-]+$/.test(name)) return name;\n return undefined;\n}\n\n/**\n * Sanitize a CSS font-family fallback name.\n *\n * Generic family names (sans-serif, serif, monospace, etc.) are used as-is.\n * Named families are wrapped in escaped quotes. This prevents injection via\n * crafted fallback values like `); } body { color: red; } .x {`.\n */\nfunction sanitizeFallback(name: string): string {\n // CSS generic font families — safe to use unquoted\n const generics = new Set([\n \"serif\", \"sans-serif\", \"monospace\", \"cursive\", \"fantasy\",\n \"system-ui\", \"ui-serif\", \"ui-sans-serif\", \"ui-monospace\", \"ui-rounded\",\n \"emoji\", \"math\", \"fangsong\",\n ]);\n const trimmed = name.trim();\n if (generics.has(trimmed)) return trimmed;\n // Wrap in single quotes with escaping to prevent CSS injection\n return `'${escapeCSSString(trimmed)}'`;\n}\n\n// Counter for generating unique class names\nlet classCounter = 0;\n\n// Track which font stylesheets have been injected (SSR + client)\nconst injectedFonts = new Set<string>();\n\ninterface FontOptions {\n weight?: string | string[];\n style?: string | string[];\n subsets?: string[];\n display?: string;\n preload?: boolean;\n fallback?: string[];\n adjustFontFallback?: boolean | string;\n variable?: string;\n axes?: string[];\n}\n\ninterface FontResult {\n className: string;\n style: { fontFamily: string };\n variable?: string;\n}\n\n/**\n * Convert a font family name to a CSS variable name.\n * e.g., \"Inter\" -> \"--font-inter\", \"Roboto Mono\" -> \"--font-roboto-mono\"\n */\nfunction toVarName(family: string): string {\n return \"--font-\" + family.toLowerCase().replace(/\\s+/g, \"-\");\n}\n\n/**\n * Build a Google Fonts CSS URL.\n */\nfunction buildGoogleFontsUrl(family: string, options: FontOptions): string {\n const params = new URLSearchParams();\n // Don't pre-replace spaces with \"+\". URLSearchParams handles encoding:\n // spaces become \"+\" in application/x-www-form-urlencoded format.\n // Pre-replacing would cause double-encoding: \"+\" -> \"%2B\" (400 error).\n let spec = family;\n\n // Build weight/style specs\n const weights = options.weight\n ? Array.isArray(options.weight)\n ? options.weight\n : [options.weight]\n : [];\n const styles = options.style\n ? Array.isArray(options.style)\n ? options.style\n : [options.style]\n : [];\n\n if (weights.length > 0 || styles.length > 0) {\n const hasItalic = styles.includes(\"italic\");\n if (weights.length > 0) {\n if (hasItalic) {\n // Use ital axis: ital,wght@0,400;0,700;1,400;1,700\n const pairs: string[] = [];\n for (const w of weights) {\n pairs.push(`0,${w}`);\n pairs.push(`1,${w}`);\n }\n spec += `:ital,wght@${pairs.join(\";\")}`;\n } else {\n spec += `:wght@${weights.join(\";\")}`;\n }\n }\n } else {\n // When no weight is specified, request the full variable weight range.\n // Without this, Google Fonts returns only weight 400 (the default).\n // Next.js loads the full variable font by default, so we match that\n // behavior to ensure all font weights render correctly.\n spec += `:wght@100..900`;\n }\n\n params.set(\"family\", spec);\n params.set(\"display\", options.display ?? \"swap\");\n\n return `https://fonts.googleapis.com/css2?${params.toString()}`;\n}\n\n/**\n * Inject a <link> tag for the font (client-side only).\n * On the server, we track font URLs for SSR head injection.\n */\nfunction injectFontStylesheet(url: string): void {\n if (injectedFonts.has(url)) return;\n injectedFonts.add(url);\n\n if (typeof document !== \"undefined\") {\n const link = document.createElement(\"link\");\n link.rel = \"stylesheet\";\n link.href = url;\n document.head.appendChild(link);\n }\n}\n\n/** Track which className CSS rules have been injected. */\nconst injectedClassRules = new Set<string>();\n\n/**\n * Inject a CSS rule that maps a className to a font-family.\n *\n * This is what makes `<div className={inter.className}>` apply the font.\n * Next.js generates equivalent rules at build time.\n *\n * In Next.js, the .className class ONLY sets font-family — it does NOT\n * set CSS variables. CSS variables are handled separately by the .variable class.\n */\nfunction injectClassNameRule(\n className: string,\n fontFamily: string,\n): void {\n if (injectedClassRules.has(className)) return;\n injectedClassRules.add(className);\n\n const css = `.${className} { font-family: ${fontFamily}; }\\n`;\n\n // On server, store the CSS for SSR injection\n if (typeof document === \"undefined\") {\n ssrFontStyles.push(css);\n return;\n }\n\n // On client, inject a <style> tag\n const style = document.createElement(\"style\");\n style.textContent = css;\n style.setAttribute(\"data-vinext-font-class\", className);\n document.head.appendChild(style);\n}\n\n/** Track which variable class CSS rules have been injected. */\nconst injectedVariableRules = new Set<string>();\n\n/** Track which :root CSS variable rules have been injected. */\nconst injectedRootVariables = new Set<string>();\n\n/**\n * Inject a CSS rule that sets a CSS variable on an element.\n * This is what makes `<html className={inter.variable}>` set the CSS variable\n * that can be referenced by other styles (e.g., Tailwind's font-sans).\n *\n * In Next.js, the .variable class ONLY sets the CSS variable — it does NOT\n * set font-family. This is critical because apps commonly apply multiple\n * .variable classes to <body> (e.g., geistSans.variable + geistMono.variable).\n * If we also set font-family here, the last class wins due to CSS cascade,\n * causing all text to use that font (e.g., everything becomes monospace).\n */\nfunction injectVariableClassRule(\n variableClassName: string,\n cssVarName: string,\n fontFamily: string,\n): void {\n if (injectedVariableRules.has(variableClassName)) return;\n injectedVariableRules.add(variableClassName);\n\n // Only set the CSS variable — do NOT set font-family.\n // This matches Next.js behavior where .variable classes only define CSS variables.\n let css = `.${variableClassName} { ${cssVarName}: ${fontFamily}; }\\n`;\n \n // Also inject at :root so CSS variable inheritance works throughout the page.\n // This ensures Tailwind utilities like `font-sans` that reference these\n // variables via var(--font-geist-sans) work correctly.\n if (!injectedRootVariables.has(cssVarName)) {\n injectedRootVariables.add(cssVarName);\n css += `:root { ${cssVarName}: ${fontFamily}; }\\n`;\n }\n\n // On server, store the CSS for SSR injection\n if (typeof document === \"undefined\") {\n ssrFontStyles.push(css);\n return;\n }\n\n // On client, inject a <style> tag\n const style = document.createElement(\"style\");\n style.textContent = css;\n style.setAttribute(\"data-vinext-font-variable\", variableClassName);\n document.head.appendChild(style);\n}\n\n// SSR: collect font class CSS for injection in <head>\nconst ssrFontStyles: string[] = [];\n\n/**\n * Get collected SSR font class styles (used by the renderer).\n * Note: We don't clear the arrays because fonts are loaded at module import\n * time and need to persist across all requests in the Workers environment.\n */\nexport function getSSRFontStyles(): string[] {\n return [...ssrFontStyles];\n}\n\n// SSR: collect font URLs to inject in <head>\nconst ssrFontUrls: string[] = [];\n\n/**\n * Get collected SSR font URLs (used by the renderer).\n * Note: We don't clear the arrays because fonts are loaded at module import\n * time and need to persist across all requests in the Workers environment.\n */\nexport function getSSRFontLinks(): string[] {\n return [...ssrFontUrls];\n}\n\n// SSR: collect font file URLs for <link rel=\"preload\"> injection (self-hosted Google fonts)\nconst ssrFontPreloads: Array<{ href: string; type: string }> = [];\nconst ssrFontPreloadHrefs = new Set<string>();\n\n/**\n * Get collected SSR font preload data (used by the renderer).\n * Returns an array of { href, type } objects for emitting\n * <link rel=\"preload\" as=\"font\" ...> tags.\n */\nexport function getSSRFontPreloads(): Array<{ href: string; type: string }> {\n return [...ssrFontPreloads];\n}\n\n/**\n * Determine the MIME type for a font file based on its extension.\n */\nfunction getFontMimeType(pathOrUrl: string): string {\n if (pathOrUrl.endsWith(\".woff2\")) return \"font/woff2\";\n if (pathOrUrl.endsWith(\".woff\")) return \"font/woff\";\n if (pathOrUrl.endsWith(\".ttf\")) return \"font/ttf\";\n if (pathOrUrl.endsWith(\".otf\")) return \"font/opentype\";\n return \"font/woff2\";\n}\n\n/**\n * Extract font file URLs from @font-face CSS rules.\n * Parses url('...') references from the CSS text.\n */\nfunction extractFontUrlsFromCSS(css: string): string[] {\n const urls: string[] = [];\n const urlRegex = /url\\(['\"]?([^'\")]+)['\"]?\\)/g;\n let match: RegExpExecArray | null;\n while ((match = urlRegex.exec(css)) !== null) {\n const url = match[1];\n // Only collect absolute paths (starting with /) — these are self-hosted font files\n if (url && url.startsWith(\"/\")) {\n urls.push(url);\n }\n }\n return urls;\n}\n\n/**\n * Collect font file URLs from self-hosted CSS for preload link generation.\n * Only collects on the server (SSR). Deduplicates by href using a Set for O(1) lookups.\n */\nfunction collectFontPreloadsFromCSS(css: string): void {\n if (typeof document !== \"undefined\") return; // client-side, skip\n\n const urls = extractFontUrlsFromCSS(css);\n for (const href of urls) {\n if (!ssrFontPreloadHrefs.has(href)) {\n ssrFontPreloadHrefs.add(href);\n ssrFontPreloads.push({ href, type: getFontMimeType(href) });\n }\n }\n}\n\n/** Track injected self-hosted @font-face blocks (deduplicate) */\nconst injectedSelfHosted = new Set<string>();\n\n/**\n * Inject self-hosted @font-face CSS (from the build plugin).\n * This replaces the CDN <link> tag with inline CSS.\n */\nfunction injectSelfHostedCSS(css: string): void {\n if (injectedSelfHosted.has(css)) return;\n injectedSelfHosted.add(css);\n\n // Extract font file URLs for preload hints (SSR only)\n collectFontPreloadsFromCSS(css);\n\n if (typeof document === \"undefined\") {\n // SSR: add to collected styles\n ssrFontStyles.push(css);\n return;\n }\n\n // Client: inject <style> tag\n const style = document.createElement(\"style\");\n style.textContent = css;\n style.setAttribute(\"data-vinext-font-selfhosted\", \"true\");\n document.head.appendChild(style);\n}\n\nfunction createFontLoader(family: string) {\n return function fontLoader(options: FontOptions & { _selfHostedCSS?: string } = {}): FontResult {\n const id = classCounter++;\n const className = `__font_${family.toLowerCase().replace(/\\s+/g, \"_\")}_${id}`;\n const fallback = options.fallback ?? [\"sans-serif\"];\n // Sanitize each fallback name to prevent CSS injection via crafted values\n const fontFamily = `'${escapeCSSString(family)}', ${fallback.map(sanitizeFallback).join(\", \")}`;\n // Validate CSS variable name — reject anything that could inject CSS.\n // Fall back to auto-generated name if invalid.\n const defaultVarName = toVarName(family);\n const cssVarName = options.variable ? (sanitizeCSSVarName(options.variable) ?? defaultVarName) : defaultVarName;\n // In Next.js, `variable` returns a CLASS NAME that sets the CSS variable.\n // Users apply this class to set the CSS variable on that element.\n const variableClassName = `__variable_${family.toLowerCase().replace(/\\s+/g, \"_\")}_${id}`;\n\n if (options._selfHostedCSS) {\n // Self-hosted mode: inject local @font-face CSS instead of CDN link\n injectSelfHostedCSS(options._selfHostedCSS);\n } else {\n // CDN mode: inject <link> to Google Fonts\n const url = buildGoogleFontsUrl(family, options);\n injectFontStylesheet(url);\n\n // On SSR, collect the URL for head injection\n if (typeof document === \"undefined\") {\n if (!ssrFontUrls.includes(url)) {\n ssrFontUrls.push(url);\n }\n }\n }\n\n // Inject a CSS rule that maps className to font-family.\n // This is what makes `<div className={inter.className}>` work.\n injectClassNameRule(className, fontFamily);\n \n // Inject a CSS rule for the variable class name.\n // This is what makes `<html className={inter.variable}>` set the CSS variable.\n injectVariableClassRule(variableClassName, cssVarName, fontFamily);\n\n return {\n className,\n style: { fontFamily },\n variable: variableClassName,\n };\n };\n}\n\n// Re-export for plugin use\nexport { buildGoogleFontsUrl };\n\n// Export a Proxy that creates font loaders for any Google Font family.\n// Usage: import { Inter } from 'next/font/google'\n// The proxy intercepts property access and returns a loader for that font.\nconst googleFonts = new Proxy(\n {} as Record<string, (options?: FontOptions) => FontResult>,\n {\n get(_target, prop: string) {\n if (prop === \"__esModule\") return true;\n if (prop === \"default\") return googleFonts;\n // Convert camelCase/PascalCase to proper font family name\n // e.g., \"Inter\" -> \"Inter\", \"RobotoMono\" -> \"Roboto Mono\"\n const family = prop.replace(/([a-z])([A-Z])/g, \"$1 $2\");\n return createFontLoader(family);\n },\n },\n);\n\nexport default googleFonts;\n\n// Named exports for common fonts (provides better IDE autocomplete)\nexport const Inter = createFontLoader(\"Inter\");\nexport const Roboto = createFontLoader(\"Roboto\");\nexport const Roboto_Mono = createFontLoader(\"Roboto Mono\");\nexport const Open_Sans = createFontLoader(\"Open Sans\");\nexport const Lato = createFontLoader(\"Lato\");\nexport const Poppins = createFontLoader(\"Poppins\");\nexport const Montserrat = createFontLoader(\"Montserrat\");\nexport const Source_Code_Pro = createFontLoader(\"Source Code Pro\");\nexport const Noto_Sans = createFontLoader(\"Noto Sans\");\nexport const Raleway = createFontLoader(\"Raleway\");\nexport const Ubuntu = createFontLoader(\"Ubuntu\");\nexport const Nunito = createFontLoader(\"Nunito\");\nexport const Playfair_Display = createFontLoader(\"Playfair Display\");\nexport const Merriweather = createFontLoader(\"Merriweather\");\nexport const PT_Sans = createFontLoader(\"PT Sans\");\nexport const Fira_Code = createFontLoader(\"Fira Code\");\nexport const JetBrains_Mono = createFontLoader(\"JetBrains Mono\");\nexport const Geist = createFontLoader(\"Geist\");\nexport const Geist_Mono = createFontLoader(\"Geist Mono\");\n"]}
1
+ {"version":3,"file":"font-google.js","sourceRoot":"","sources":["../../src/shims/font-google.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,mBAAmB,EAAE,eAAe,EAAE,gBAAgB,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACzH,cAAc,yBAAyB,CAAC","sourcesContent":["export { default, buildGoogleFontsUrl, getSSRFontLinks, getSSRFontStyles, getSSRFontPreloads } from \"./font-google-base\";\nexport * from \"./font-google.generated\";\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"project.d.ts","sourceRoot":"","sources":["../../src/utils/project.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAoBH;;;GAGG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAepD;AAED;;;;GAIG;AACH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,KAAK,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAsBtE;AAID;;;GAGG;AACH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAKzD;AAED;;;GAGG;AACH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAK7D;AAID;;GAEG;AACH,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAMnD;AAED;;GAEG;AACH,wBAAgB,SAAS,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAK/C"}
1
+ {"version":3,"file":"project.d.ts","sourceRoot":"","sources":["../../src/utils/project.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAoBH;;;GAGG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAepD;AAED;;;;GAIG;AACH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,KAAK,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAsBtE;AAID;;;GAGG;AACH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAMzD;AAED;;;GAGG;AACH,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAM7D;AAID;;GAEG;AACH,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAMnD;AAED;;GAEG;AACH,wBAAgB,SAAS,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAK/C"}
@@ -77,6 +77,8 @@ export function detectPackageManager(root) {
77
77
  return "pnpm add -D";
78
78
  if (fs.existsSync(path.join(root, "yarn.lock")))
79
79
  return "yarn add -D";
80
+ if (fs.existsSync(path.join(root, "bun.lock")))
81
+ return "bun add -D";
80
82
  if (fs.existsSync(path.join(root, "bun.lockb")))
81
83
  return "bun add -D";
82
84
  return "npm install -D";
@@ -90,6 +92,8 @@ export function detectPackageManagerName(root) {
90
92
  return "pnpm";
91
93
  if (fs.existsSync(path.join(root, "yarn.lock")))
92
94
  return "yarn";
95
+ if (fs.existsSync(path.join(root, "bun.lock")))
96
+ return "bun";
93
97
  if (fs.existsSync(path.join(root, "bun.lockb")))
94
98
  return "bun";
95
99
  return "npm";
@@ -1 +1 @@
1
- {"version":3,"file":"project.js","sourceRoot":"","sources":["../../src/utils/project.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Shared project utilities — used by both `vinext init` and `vinext deploy`.\n *\n * These functions detect and modify project configuration without touching\n * any Next.js source files, config files, or tsconfig.json.\n */\n\nimport fs from \"node:fs\";\nimport path from \"node:path\";\n\n// ─── CJS Config Handling ─────────────────────────────────────────────────────\n\n/** Common CJS config files that may need renaming when adding \"type\": \"module\" */\nconst CJS_CONFIG_FILES = [\n \"postcss.config.js\",\n \"tailwind.config.js\",\n \".eslintrc.js\",\n \"prettier.config.js\",\n \"stylelint.config.js\",\n \"commitlint.config.js\",\n \"jest.config.js\",\n \"babel.config.js\",\n \".babelrc.js\",\n];\n\n/**\n * Ensure package.json has \"type\": \"module\".\n * Returns true if it was added (i.e. it wasn't already there).\n */\nexport function ensureESModule(root: string): boolean {\n const pkgPath = path.join(root, \"package.json\");\n if (!fs.existsSync(pkgPath)) return false;\n\n try {\n const raw = fs.readFileSync(pkgPath, \"utf-8\");\n const pkg = JSON.parse(raw);\n if (pkg.type === \"module\") return false;\n\n pkg.type = \"module\";\n fs.writeFileSync(pkgPath, JSON.stringify(pkg, null, 2) + \"\\n\", \"utf-8\");\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Rename CJS config files (that use `module.exports`) to .cjs\n * to avoid breakage when \"type\": \"module\" is added.\n * Returns array of [oldName, newName] pairs that were renamed.\n */\nexport function renameCJSConfigs(root: string): Array<[string, string]> {\n const renamed: Array<[string, string]> = [];\n\n for (const fileName of CJS_CONFIG_FILES) {\n const filePath = path.join(root, fileName);\n if (!fs.existsSync(filePath)) continue;\n\n try {\n const content = fs.readFileSync(filePath, \"utf-8\");\n // Only rename if it actually uses CJS patterns\n if (/\\bmodule\\.exports\\b/.test(content) || /\\brequire\\s*\\(/.test(content)) {\n const newName = fileName.replace(/\\.js$/, \".cjs\");\n const newPath = path.join(root, newName);\n fs.renameSync(filePath, newPath);\n renamed.push([fileName, newName]);\n }\n } catch {\n // skip unreadable files\n }\n }\n\n return renamed;\n}\n\n// ─── Package Manager Detection ───────────────────────────────────────────────\n\n/**\n * Detect which package manager is used by looking at lock files.\n * Returns the install command string (e.g. \"pnpm add -D\").\n */\nexport function detectPackageManager(root: string): string {\n if (fs.existsSync(path.join(root, \"pnpm-lock.yaml\"))) return \"pnpm add -D\";\n if (fs.existsSync(path.join(root, \"yarn.lock\"))) return \"yarn add -D\";\n if (fs.existsSync(path.join(root, \"bun.lockb\"))) return \"bun add -D\";\n return \"npm install -D\";\n}\n\n/**\n * Detect which package manager name is used (without install args).\n * Returns \"pnpm\", \"yarn\", \"bun\", or \"npm\".\n */\nexport function detectPackageManagerName(root: string): string {\n if (fs.existsSync(path.join(root, \"pnpm-lock.yaml\"))) return \"pnpm\";\n if (fs.existsSync(path.join(root, \"yarn.lock\"))) return \"yarn\";\n if (fs.existsSync(path.join(root, \"bun.lockb\"))) return \"bun\";\n return \"npm\";\n}\n\n// ─── Vite Config Detection ───────────────────────────────────────────────────\n\n/**\n * Check if a vite.config file exists in the project root.\n */\nexport function hasViteConfig(root: string): boolean {\n return (\n fs.existsSync(path.join(root, \"vite.config.ts\")) ||\n fs.existsSync(path.join(root, \"vite.config.js\")) ||\n fs.existsSync(path.join(root, \"vite.config.mjs\"))\n );\n}\n\n/**\n * Check if the project uses App Router (has an app/ directory).\n */\nexport function hasAppDir(root: string): boolean {\n return (\n fs.existsSync(path.join(root, \"app\")) ||\n fs.existsSync(path.join(root, \"src\", \"app\"))\n );\n}\n"]}
1
+ {"version":3,"file":"project.js","sourceRoot":"","sources":["../../src/utils/project.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Shared project utilities — used by both `vinext init` and `vinext deploy`.\n *\n * These functions detect and modify project configuration without touching\n * any Next.js source files, config files, or tsconfig.json.\n */\n\nimport fs from \"node:fs\";\nimport path from \"node:path\";\n\n// ─── CJS Config Handling ─────────────────────────────────────────────────────\n\n/** Common CJS config files that may need renaming when adding \"type\": \"module\" */\nconst CJS_CONFIG_FILES = [\n \"postcss.config.js\",\n \"tailwind.config.js\",\n \".eslintrc.js\",\n \"prettier.config.js\",\n \"stylelint.config.js\",\n \"commitlint.config.js\",\n \"jest.config.js\",\n \"babel.config.js\",\n \".babelrc.js\",\n];\n\n/**\n * Ensure package.json has \"type\": \"module\".\n * Returns true if it was added (i.e. it wasn't already there).\n */\nexport function ensureESModule(root: string): boolean {\n const pkgPath = path.join(root, \"package.json\");\n if (!fs.existsSync(pkgPath)) return false;\n\n try {\n const raw = fs.readFileSync(pkgPath, \"utf-8\");\n const pkg = JSON.parse(raw);\n if (pkg.type === \"module\") return false;\n\n pkg.type = \"module\";\n fs.writeFileSync(pkgPath, JSON.stringify(pkg, null, 2) + \"\\n\", \"utf-8\");\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Rename CJS config files (that use `module.exports`) to .cjs\n * to avoid breakage when \"type\": \"module\" is added.\n * Returns array of [oldName, newName] pairs that were renamed.\n */\nexport function renameCJSConfigs(root: string): Array<[string, string]> {\n const renamed: Array<[string, string]> = [];\n\n for (const fileName of CJS_CONFIG_FILES) {\n const filePath = path.join(root, fileName);\n if (!fs.existsSync(filePath)) continue;\n\n try {\n const content = fs.readFileSync(filePath, \"utf-8\");\n // Only rename if it actually uses CJS patterns\n if (/\\bmodule\\.exports\\b/.test(content) || /\\brequire\\s*\\(/.test(content)) {\n const newName = fileName.replace(/\\.js$/, \".cjs\");\n const newPath = path.join(root, newName);\n fs.renameSync(filePath, newPath);\n renamed.push([fileName, newName]);\n }\n } catch {\n // skip unreadable files\n }\n }\n\n return renamed;\n}\n\n// ─── Package Manager Detection ───────────────────────────────────────────────\n\n/**\n * Detect which package manager is used by looking at lock files.\n * Returns the install command string (e.g. \"pnpm add -D\").\n */\nexport function detectPackageManager(root: string): string {\n if (fs.existsSync(path.join(root, \"pnpm-lock.yaml\"))) return \"pnpm add -D\";\n if (fs.existsSync(path.join(root, \"yarn.lock\"))) return \"yarn add -D\";\n if (fs.existsSync(path.join(root, \"bun.lock\"))) return \"bun add -D\";\n if (fs.existsSync(path.join(root, \"bun.lockb\"))) return \"bun add -D\";\n return \"npm install -D\";\n}\n\n/**\n * Detect which package manager name is used (without install args).\n * Returns \"pnpm\", \"yarn\", \"bun\", or \"npm\".\n */\nexport function detectPackageManagerName(root: string): string {\n if (fs.existsSync(path.join(root, \"pnpm-lock.yaml\"))) return \"pnpm\";\n if (fs.existsSync(path.join(root, \"yarn.lock\"))) return \"yarn\";\n if (fs.existsSync(path.join(root, \"bun.lock\"))) return \"bun\";\n if (fs.existsSync(path.join(root, \"bun.lockb\"))) return \"bun\";\n return \"npm\";\n}\n\n// ─── Vite Config Detection ───────────────────────────────────────────────────\n\n/**\n * Check if a vite.config file exists in the project root.\n */\nexport function hasViteConfig(root: string): boolean {\n return (\n fs.existsSync(path.join(root, \"vite.config.ts\")) ||\n fs.existsSync(path.join(root, \"vite.config.js\")) ||\n fs.existsSync(path.join(root, \"vite.config.mjs\"))\n );\n}\n\n/**\n * Check if the project uses App Router (has an app/ directory).\n */\nexport function hasAppDir(root: string): boolean {\n return (\n fs.existsSync(path.join(root, \"app\")) ||\n fs.existsSync(path.join(root, \"src\", \"app\"))\n );\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "vinext",
3
- "version": "0.0.15",
3
+ "version": "0.0.16",
4
4
  "description": "Run Next.js apps on Vite. Drop-in replacement for the next CLI.",
5
5
  "license": "MIT",
6
6
  "repository": {
@@ -66,6 +66,7 @@
66
66
  "magic-string": "^0.30.21",
67
67
  "react-server-dom-webpack": "^19.2.4",
68
68
  "rsc-html-stream": "^0.0.7",
69
+ "vite-plugin-commonjs": "^0.10.4",
69
70
  "vite-tsconfig-paths": "^6.1.1"
70
71
  },
71
72
  "devDependencies": {