vinext 0.0.14 → 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 (55) 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 +70 -59
  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 +29 -13
  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 +44 -10
  23. package/dist/server/middleware.js.map +1 -1
  24. package/dist/server/prod-server.d.ts.map +1 -1
  25. package/dist/server/prod-server.js +30 -4
  26. package/dist/server/prod-server.js.map +1 -1
  27. package/dist/shims/fetch-cache.d.ts.map +1 -1
  28. package/dist/shims/fetch-cache.js +73 -41
  29. package/dist/shims/fetch-cache.js.map +1 -1
  30. package/dist/shims/font-google-base.d.ts +68 -0
  31. package/dist/shims/font-google-base.d.ts.map +1 -0
  32. package/dist/shims/font-google-base.js +365 -0
  33. package/dist/shims/font-google-base.js.map +1 -0
  34. package/dist/shims/font-google.d.ts +2 -121
  35. package/dist/shims/font-google.d.ts.map +1 -1
  36. package/dist/shims/font-google.generated.d.ts +1925 -0
  37. package/dist/shims/font-google.generated.d.ts.map +1 -0
  38. package/dist/shims/font-google.generated.js +1928 -0
  39. package/dist/shims/font-google.generated.js.map +1 -0
  40. package/dist/shims/font-google.js +2 -386
  41. package/dist/shims/font-google.js.map +1 -1
  42. package/dist/shims/form.d.ts.map +1 -1
  43. package/dist/shims/form.js +32 -0
  44. package/dist/shims/form.js.map +1 -1
  45. package/dist/shims/link.d.ts.map +1 -1
  46. package/dist/shims/link.js +11 -0
  47. package/dist/shims/link.js.map +1 -1
  48. package/dist/shims/url-safety.d.ts +8 -0
  49. package/dist/shims/url-safety.d.ts.map +1 -0
  50. package/dist/shims/url-safety.js +16 -0
  51. package/dist/shims/url-safety.js.map +1 -0
  52. package/dist/utils/project.d.ts.map +1 -1
  53. package/dist/utils/project.js +4 -0
  54. package/dist/utils/project.js.map +1 -1
  55. 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":"form.d.ts","sourceRoot":"","sources":["../../src/shims/form.tsx"],"names":[],"mappings":"AAEA;;;;;;;;;;;;;;;;GAgBG;AAEH,OAAO,EAEL,cAAc,EACd,KAAK,kBAAkB,EAExB,MAAM,OAAO,CAAC;AAGf,OAAO,EAAE,cAAc,EAAE,CAAC;AAE1B,UAAU,SAAU,SAAQ,kBAAkB,CAAC,eAAe,CAAC;IAC7D,gEAAgE;IAChE,MAAM,EAAE,MAAM,GAAG,CAAC,CAAC,QAAQ,EAAE,QAAQ,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;IAChE,0DAA0D;IAC1D,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,qDAAqD;IACrD,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAED,QAAA,MAAM,IAAI,uGAmER,CAAC;AAEH,eAAe,IAAI,CAAC"}
1
+ {"version":3,"file":"form.d.ts","sourceRoot":"","sources":["../../src/shims/form.tsx"],"names":[],"mappings":"AAEA;;;;;;;;;;;;;;;;GAgBG;AAEH,OAAO,EAEL,cAAc,EACd,KAAK,kBAAkB,EAExB,MAAM,OAAO,CAAC;AAIf,OAAO,EAAE,cAAc,EAAE,CAAC;AAuB1B,UAAU,SAAU,SAAQ,kBAAkB,CAAC,eAAe,CAAC;IAC7D,gEAAgE;IAChE,MAAM,EAAE,MAAM,GAAG,CAAC,CAAC,QAAQ,EAAE,QAAQ,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;IAChE,0DAA0D;IAC1D,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,qDAAqD;IACrD,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAED,QAAA,MAAM,IAAI,uGA4ER,CAAC;AAEH,eAAe,IAAI,CAAC"}
@@ -18,14 +18,46 @@ import { jsx as _jsx } from "react/jsx-runtime";
18
18
  * </Form>
19
19
  */
20
20
  import { forwardRef, useActionState, } from "react";
21
+ import { isDangerousScheme } from "./url-safety.js";
21
22
  // Re-export useActionState from React 19 to match Next.js's next/form module
22
23
  export { useActionState };
24
+ function isSafeAction(action) {
25
+ // Block dangerous URI schemes
26
+ if (isDangerousScheme(action))
27
+ return false;
28
+ // Block protocol-relative URLs (//evil.com/...)
29
+ if (action.startsWith("//"))
30
+ return false;
31
+ // Block absolute URLs to external origins (client-side: compare origins)
32
+ if (/^https?:\/\//i.test(action)) {
33
+ if (typeof window !== "undefined") {
34
+ try {
35
+ const actionUrl = new URL(action);
36
+ return actionUrl.origin === window.location.origin;
37
+ }
38
+ catch {
39
+ return false;
40
+ }
41
+ }
42
+ // Server-side: block all absolute URLs (can't compare origins)
43
+ return false;
44
+ }
45
+ return true;
46
+ }
23
47
  const Form = forwardRef(function Form(props, ref) {
24
48
  const { action, replace = false, scroll = true, onSubmit, ...rest } = props;
25
49
  // If action is a function (server action), pass it directly to React
26
50
  if (typeof action === "function") {
27
51
  return _jsx("form", { ref: ref, action: action, onSubmit: onSubmit, ...rest });
28
52
  }
53
+ // Block dangerous action URLs. Render <form> without action attribute
54
+ // so it submits to the current page (safe default).
55
+ if (!isSafeAction(action)) {
56
+ if (process.env.NODE_ENV !== "production") {
57
+ console.warn(`<Form> blocked unsafe action: ${action}`);
58
+ }
59
+ return _jsx("form", { ref: ref, onSubmit: onSubmit, ...rest });
60
+ }
29
61
  async function handleSubmit(e) {
30
62
  // Call user's onSubmit first
31
63
  if (onSubmit) {
@@ -1 +1 @@
1
- {"version":3,"file":"form.js","sourceRoot":"","sources":["../../src/shims/form.tsx"],"names":[],"mappings":"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","sourcesContent":["\"use client\";\n\n/**\n * next/form shim\n *\n * Progressive enhancement form component. In Next.js, this replaces\n * the standard <form> element with one that intercepts submissions\n * and performs client-side navigation for GET forms (search forms).\n *\n * For POST forms with server actions, it delegates to React's built-in\n * form action handling.\n *\n * Usage:\n * import Form from 'next/form';\n * <Form action=\"/search\">\n * <input name=\"q\" />\n * <button type=\"submit\">Search</button>\n * </Form>\n */\n\nimport {\n forwardRef,\n useActionState,\n type FormHTMLAttributes,\n type ForwardedRef,\n} from \"react\";\n\n// Re-export useActionState from React 19 to match Next.js's next/form module\nexport { useActionState };\n\ninterface FormProps extends FormHTMLAttributes<HTMLFormElement> {\n /** Target URL for GET forms, or server action for POST forms */\n action: string | ((formData: FormData) => void | Promise<void>);\n /** Replace instead of push in history (default: false) */\n replace?: boolean;\n /** Scroll to top after navigation (default: true) */\n scroll?: boolean;\n}\n\nconst Form = forwardRef(function Form(\n props: FormProps,\n ref: ForwardedRef<HTMLFormElement>,\n) {\n const { action, replace = false, scroll = true, onSubmit, ...rest } = props;\n\n // If action is a function (server action), pass it directly to React\n if (typeof action === \"function\") {\n return <form ref={ref} action={action as any} onSubmit={onSubmit as any} {...rest} />;\n }\n\n async function handleSubmit(e: any) {\n // Call user's onSubmit first\n if (onSubmit) {\n (onSubmit as any)(e);\n if (e.defaultPrevented) return;\n }\n\n // Only intercept GET forms for client-side navigation\n const method = (rest.method ?? \"GET\").toUpperCase();\n if (method !== \"GET\") return;\n\n e.preventDefault();\n\n const formData = new FormData(e.currentTarget);\n const params = new URLSearchParams();\n for (const [key, value] of formData) {\n if (typeof value === \"string\") {\n params.append(key, value);\n }\n }\n\n const url = `${action as string}?${params.toString()}`;\n\n // Navigate client-side\n const win = window as any;\n if (typeof win.__VINEXT_RSC_NAVIGATE__ === \"function\") {\n // App Router: RSC navigation. Await so scroll happens after new content renders.\n if (replace) {\n window.history.replaceState(null, \"\", url);\n } else {\n window.history.pushState(null, \"\", url);\n }\n await win.__VINEXT_RSC_NAVIGATE__(url);\n } else {\n // Pages Router: use router or fallback\n if (replace) {\n window.history.replaceState({}, \"\", url);\n } else {\n window.history.pushState({}, \"\", url);\n }\n window.dispatchEvent(new PopStateEvent(\"popstate\"));\n }\n\n if (scroll) {\n window.scrollTo(0, 0);\n }\n }\n\n return (\n <form\n ref={ref}\n action={action}\n onSubmit={handleSubmit}\n {...rest}\n />\n );\n});\n\nexport default Form;\n"]}
1
+ {"version":3,"file":"form.js","sourceRoot":"","sources":["../../src/shims/form.tsx"],"names":[],"mappings":"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","sourcesContent":["\"use client\";\n\n/**\n * next/form shim\n *\n * Progressive enhancement form component. In Next.js, this replaces\n * the standard <form> element with one that intercepts submissions\n * and performs client-side navigation for GET forms (search forms).\n *\n * For POST forms with server actions, it delegates to React's built-in\n * form action handling.\n *\n * Usage:\n * import Form from 'next/form';\n * <Form action=\"/search\">\n * <input name=\"q\" />\n * <button type=\"submit\">Search</button>\n * </Form>\n */\n\nimport {\n forwardRef,\n useActionState,\n type FormHTMLAttributes,\n type ForwardedRef,\n} from \"react\";\nimport { isDangerousScheme } from \"./url-safety.js\";\n\n// Re-export useActionState from React 19 to match Next.js's next/form module\nexport { useActionState };\n\nfunction isSafeAction(action: string): boolean {\n // Block dangerous URI schemes\n if (isDangerousScheme(action)) return false;\n // Block protocol-relative URLs (//evil.com/...)\n if (action.startsWith(\"//\")) return false;\n // Block absolute URLs to external origins (client-side: compare origins)\n if (/^https?:\\/\\//i.test(action)) {\n if (typeof window !== \"undefined\") {\n try {\n const actionUrl = new URL(action);\n return actionUrl.origin === window.location.origin;\n } catch {\n return false;\n }\n }\n // Server-side: block all absolute URLs (can't compare origins)\n return false;\n }\n return true;\n}\n\ninterface FormProps extends FormHTMLAttributes<HTMLFormElement> {\n /** Target URL for GET forms, or server action for POST forms */\n action: string | ((formData: FormData) => void | Promise<void>);\n /** Replace instead of push in history (default: false) */\n replace?: boolean;\n /** Scroll to top after navigation (default: true) */\n scroll?: boolean;\n}\n\nconst Form = forwardRef(function Form(\n props: FormProps,\n ref: ForwardedRef<HTMLFormElement>,\n) {\n const { action, replace = false, scroll = true, onSubmit, ...rest } = props;\n\n // If action is a function (server action), pass it directly to React\n if (typeof action === \"function\") {\n return <form ref={ref} action={action as any} onSubmit={onSubmit as any} {...rest} />;\n }\n\n // Block dangerous action URLs. Render <form> without action attribute\n // so it submits to the current page (safe default).\n if (!isSafeAction(action)) {\n if (process.env.NODE_ENV !== \"production\") {\n console.warn(`<Form> blocked unsafe action: ${action}`);\n }\n return <form ref={ref} onSubmit={onSubmit as any} {...rest} />;\n }\n\n async function handleSubmit(e: any) {\n // Call user's onSubmit first\n if (onSubmit) {\n (onSubmit as any)(e);\n if (e.defaultPrevented) return;\n }\n\n // Only intercept GET forms for client-side navigation\n const method = (rest.method ?? \"GET\").toUpperCase();\n if (method !== \"GET\") return;\n\n e.preventDefault();\n\n const formData = new FormData(e.currentTarget);\n const params = new URLSearchParams();\n for (const [key, value] of formData) {\n if (typeof value === \"string\") {\n params.append(key, value);\n }\n }\n\n const url = `${action as string}?${params.toString()}`;\n\n // Navigate client-side\n const win = window as any;\n if (typeof win.__VINEXT_RSC_NAVIGATE__ === \"function\") {\n // App Router: RSC navigation. Await so scroll happens after new content renders.\n if (replace) {\n window.history.replaceState(null, \"\", url);\n } else {\n window.history.pushState(null, \"\", url);\n }\n await win.__VINEXT_RSC_NAVIGATE__(url);\n } else {\n // Pages Router: use router or fallback\n if (replace) {\n window.history.replaceState({}, \"\", url);\n } else {\n window.history.pushState({}, \"\", url);\n }\n window.dispatchEvent(new PopStateEvent(\"popstate\"));\n }\n\n if (scroll) {\n window.scrollTo(0, 0);\n }\n }\n\n return (\n <form\n ref={ref}\n action={action}\n onSubmit={handleSubmit}\n {...rest}\n />\n );\n});\n\nexport default Form;\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"link.d.ts","sourceRoot":"","sources":["../../src/shims/link.tsx"],"names":[],"mappings":"AAEA;;;;;;GAMG;AACH,OAAO,KAAK,EAAE,EAAmF,KAAK,oBAAoB,EAAmB,MAAM,OAAO,CAAC;AAK3J,UAAU,aAAa;IACrB,GAAG,EAAE,GAAG,CAAC;IACT,iFAAiF;IACjF,cAAc,IAAI,IAAI,CAAC;IACvB,gDAAgD;IAChD,gBAAgB,EAAE,OAAO,CAAC;CAC3B;AAED,UAAU,SAAU,SAAQ,IAAI,CAAC,oBAAoB,CAAC,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC/E,IAAI,EAAE,MAAM,GAAG;QAAE,QAAQ,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;KAAE,CAAC;IACrE,kFAAkF;IAClF,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,2DAA2D;IAC3D,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,qFAAqF;IACrF,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,oDAAoD;IACpD,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,kDAAkD;IAClD,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,sDAAsD;IACtD,MAAM,CAAC,EAAE,MAAM,GAAG,KAAK,CAAC;IACxB,8EAA8E;IAC9E,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,aAAa,KAAK,IAAI,CAAC;IAC5C,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;CAC5B;AAMD,UAAU,sBAAsB;IAC9B,OAAO,EAAE,OAAO,CAAC;CAClB;AAID;;;;GAIG;AACH,wBAAgB,aAAa,IAAI,sBAAsB,CAEtD;AAoND,QAAA,MAAM,IAAI,qFAwMR,CAAC;AAEH,eAAe,IAAI,CAAC"}
1
+ {"version":3,"file":"link.d.ts","sourceRoot":"","sources":["../../src/shims/link.tsx"],"names":[],"mappings":"AAEA;;;;;;GAMG;AACH,OAAO,KAAK,EAAE,EAAmF,KAAK,oBAAoB,EAAmB,MAAM,OAAO,CAAC;AAM3J,UAAU,aAAa;IACrB,GAAG,EAAE,GAAG,CAAC;IACT,iFAAiF;IACjF,cAAc,IAAI,IAAI,CAAC;IACvB,gDAAgD;IAChD,gBAAgB,EAAE,OAAO,CAAC;CAC3B;AAED,UAAU,SAAU,SAAQ,IAAI,CAAC,oBAAoB,CAAC,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC/E,IAAI,EAAE,MAAM,GAAG;QAAE,QAAQ,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;KAAE,CAAC;IACrE,kFAAkF;IAClF,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,2DAA2D;IAC3D,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,qFAAqF;IACrF,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,oDAAoD;IACpD,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,kDAAkD;IAClD,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,sDAAsD;IACtD,MAAM,CAAC,EAAE,MAAM,GAAG,KAAK,CAAC;IACxB,8EAA8E;IAC9E,UAAU,CAAC,EAAE,CAAC,KAAK,EAAE,aAAa,KAAK,IAAI,CAAC;IAC5C,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;CAC5B;AAMD,UAAU,sBAAsB;IAC9B,OAAO,EAAE,OAAO,CAAC;CAClB;AAID;;;;GAIG;AACH,wBAAgB,aAAa,IAAI,sBAAsB,CAEtD;AAoND,QAAA,MAAM,IAAI,qFAoNR,CAAC;AAEH,eAAe,IAAI,CAAC"}
@@ -11,6 +11,7 @@ import React, { forwardRef, useRef, useEffect, useCallback, useContext, createCo
11
11
  // Import shared RSC prefetch utilities from navigation shim (relative path
12
12
  // so this resolves both via the Vite plugin and in direct vitest imports)
13
13
  import { toRscUrl, getPrefetchedUrls, storePrefetchResponse } from "./navigation.js";
14
+ import { isDangerousScheme } from "./url-safety.js";
14
15
  const LinkStatusContext = createContext({ pending: false });
15
16
  /**
16
17
  * useLinkStatus returns the pending state of the enclosing <Link>.
@@ -222,6 +223,16 @@ const Link = forwardRef(function Link({ href, as, replace = false, prefetch: pre
222
223
  // If `as` is provided, use it as the actual URL (legacy Next.js pattern
223
224
  // where href is a route pattern like "/user/[id]" and as is "/user/1")
224
225
  const resolvedHref = as ?? resolveHref(href);
226
+ // Block dangerous URI schemes (javascript:, data:, vbscript:).
227
+ // Render an inert <a> without href to prevent XSS while preserving
228
+ // styling and attributes like className, id, aria-*.
229
+ if (typeof resolvedHref === "string" && isDangerousScheme(resolvedHref)) {
230
+ if (process.env.NODE_ENV !== "production") {
231
+ console.warn(`<Link> blocked dangerous href: ${resolvedHref}`);
232
+ }
233
+ const { passHref: _p, ...safeProps } = restWithoutLocale;
234
+ return _jsx("a", { ...safeProps, children: children });
235
+ }
225
236
  // Apply locale prefix if specified
226
237
  const localizedHref = applyLocaleToHref(resolvedHref, locale);
227
238
  // Full href with basePath for browser URLs and fetches