@bleedingdev/modern-js-plugin-i18n 3.2.0-ultramodern.12 → 3.2.0-ultramodern.120

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 (117) hide show
  1. package/README.md +221 -11
  2. package/dist/cjs/cli/index.js +32 -5
  3. package/dist/cjs/runtime/I18nLink.js +17 -28
  4. package/dist/cjs/runtime/Link.js +252 -0
  5. package/dist/cjs/runtime/canonicalRoutes.js +18 -0
  6. package/dist/cjs/runtime/context.js +41 -10
  7. package/dist/cjs/runtime/hooks.js +17 -10
  8. package/dist/cjs/runtime/i18n/backend/config.js +9 -5
  9. package/dist/cjs/runtime/i18n/backend/defaults.js +15 -10
  10. package/dist/cjs/runtime/i18n/backend/defaults.node.js +16 -11
  11. package/dist/cjs/runtime/i18n/backend/index.js +9 -5
  12. package/dist/cjs/runtime/i18n/backend/middleware.common.js +9 -5
  13. package/dist/cjs/runtime/i18n/backend/middleware.js +9 -5
  14. package/dist/cjs/runtime/i18n/backend/middleware.node.js +13 -9
  15. package/dist/cjs/runtime/i18n/backend/sdk-backend.js +9 -5
  16. package/dist/cjs/runtime/i18n/backend/sdk-event.js +16 -11
  17. package/dist/cjs/runtime/i18n/detection/config.js +9 -5
  18. package/dist/cjs/runtime/i18n/detection/index.js +9 -5
  19. package/dist/cjs/runtime/i18n/detection/middleware.js +9 -5
  20. package/dist/cjs/runtime/i18n/detection/middleware.node.js +9 -5
  21. package/dist/cjs/runtime/i18n/index.js +9 -5
  22. package/dist/cjs/runtime/i18n/instance.js +17 -37
  23. package/dist/cjs/runtime/i18n/react-i18next.js +53 -0
  24. package/dist/cjs/runtime/i18n/utils.js +9 -17
  25. package/dist/cjs/runtime/index.js +50 -15
  26. package/dist/cjs/runtime/localizedPaths.js +105 -0
  27. package/dist/cjs/runtime/routerAdapter.js +167 -0
  28. package/dist/cjs/runtime/utils.js +87 -97
  29. package/dist/cjs/server/index.js +69 -13
  30. package/dist/cjs/shared/deepMerge.js +12 -8
  31. package/dist/cjs/shared/detection.js +9 -5
  32. package/dist/cjs/shared/localisedUrls.js +271 -0
  33. package/dist/cjs/shared/utils.js +15 -11
  34. package/dist/esm/cli/index.mjs +23 -0
  35. package/dist/esm/runtime/I18nLink.mjs +7 -22
  36. package/dist/esm/runtime/Link.mjs +209 -0
  37. package/dist/esm/runtime/canonicalRoutes.mjs +0 -0
  38. package/dist/esm/runtime/context.mjs +34 -7
  39. package/dist/esm/runtime/hooks.mjs +9 -6
  40. package/dist/esm/runtime/i18n/backend/defaults.mjs +1 -1
  41. package/dist/esm/runtime/i18n/backend/defaults.node.mjs +2 -2
  42. package/dist/esm/runtime/i18n/backend/middleware.node.mjs +3 -3
  43. package/dist/esm/runtime/i18n/instance.mjs +1 -19
  44. package/dist/esm/runtime/i18n/react-i18next.mjs +15 -0
  45. package/dist/esm/runtime/i18n/utils.mjs +0 -12
  46. package/dist/esm/runtime/index.mjs +23 -13
  47. package/dist/esm/runtime/localizedPaths.mjs +58 -0
  48. package/dist/esm/runtime/routerAdapter.mjs +129 -0
  49. package/dist/esm/runtime/utils.mjs +25 -30
  50. package/dist/esm/server/index.mjs +53 -7
  51. package/dist/esm/shared/localisedUrls.mjs +212 -0
  52. package/dist/esm-node/cli/index.mjs +23 -0
  53. package/dist/esm-node/runtime/I18nLink.mjs +7 -22
  54. package/dist/esm-node/runtime/Link.mjs +210 -0
  55. package/dist/esm-node/runtime/canonicalRoutes.mjs +1 -0
  56. package/dist/esm-node/runtime/context.mjs +34 -7
  57. package/dist/esm-node/runtime/hooks.mjs +9 -6
  58. package/dist/esm-node/runtime/i18n/backend/defaults.mjs +1 -1
  59. package/dist/esm-node/runtime/i18n/backend/defaults.node.mjs +2 -2
  60. package/dist/esm-node/runtime/i18n/backend/middleware.node.mjs +3 -3
  61. package/dist/esm-node/runtime/i18n/instance.mjs +1 -19
  62. package/dist/esm-node/runtime/i18n/react-i18next.mjs +16 -0
  63. package/dist/esm-node/runtime/i18n/utils.mjs +0 -12
  64. package/dist/esm-node/runtime/index.mjs +23 -13
  65. package/dist/esm-node/runtime/localizedPaths.mjs +59 -0
  66. package/dist/esm-node/runtime/routerAdapter.mjs +130 -0
  67. package/dist/esm-node/runtime/utils.mjs +25 -30
  68. package/dist/esm-node/server/index.mjs +53 -7
  69. package/dist/esm-node/shared/localisedUrls.mjs +213 -0
  70. package/dist/types/cli/index.d.ts +1 -0
  71. package/dist/types/runtime/I18nLink.d.ts +6 -0
  72. package/dist/types/runtime/Link.d.ts +56 -0
  73. package/dist/types/runtime/canonicalRoutes.d.ts +60 -0
  74. package/dist/types/runtime/context.d.ts +3 -0
  75. package/dist/types/runtime/hooks.d.ts +4 -2
  76. package/dist/types/runtime/i18n/backend/middleware.node.d.ts +1 -1
  77. package/dist/types/runtime/i18n/instance.d.ts +4 -6
  78. package/dist/types/runtime/i18n/react-i18next.d.ts +7 -0
  79. package/dist/types/runtime/index.d.ts +6 -1
  80. package/dist/types/runtime/localizedPaths.d.ts +39 -0
  81. package/dist/types/runtime/routerAdapter.d.ts +26 -0
  82. package/dist/types/runtime/types.d.ts +1 -1
  83. package/dist/types/runtime/utils.d.ts +13 -9
  84. package/dist/types/server/index.d.ts +6 -0
  85. package/dist/types/shared/localisedUrls.d.ts +21 -0
  86. package/dist/types/shared/type.d.ts +12 -0
  87. package/package.json +24 -28
  88. package/rstest.config.mts +39 -0
  89. package/src/cli/index.ts +44 -1
  90. package/src/runtime/I18nLink.tsx +14 -51
  91. package/src/runtime/Link.tsx +414 -0
  92. package/src/runtime/canonicalRoutes.ts +93 -0
  93. package/src/runtime/context.tsx +45 -7
  94. package/src/runtime/hooks.ts +13 -4
  95. package/src/runtime/i18n/backend/defaults.node.ts +2 -2
  96. package/src/runtime/i18n/backend/defaults.ts +3 -1
  97. package/src/runtime/i18n/backend/middleware.node.ts +1 -1
  98. package/src/runtime/i18n/instance.ts +3 -30
  99. package/src/runtime/i18n/react-i18next.ts +25 -0
  100. package/src/runtime/i18n/utils.ts +4 -26
  101. package/src/runtime/index.tsx +47 -12
  102. package/src/runtime/localizedPaths.ts +118 -0
  103. package/src/runtime/routerAdapter.tsx +333 -0
  104. package/src/runtime/types.ts +1 -1
  105. package/src/runtime/utils.ts +44 -37
  106. package/src/server/index.ts +117 -10
  107. package/src/shared/localisedUrls.ts +453 -0
  108. package/src/shared/type.ts +12 -0
  109. package/tests/i18nUtils.test.ts +52 -0
  110. package/tests/link.test.tsx +475 -0
  111. package/tests/linkTypes.test.ts +28 -0
  112. package/tests/localisedUrls.test.ts +312 -0
  113. package/tests/routerAdapter.test.tsx +452 -0
  114. package/tests/type-fixture/linkTypes.fixture.tsx +51 -0
  115. package/tests/type-fixture/tsconfig.json +15 -0
  116. package/dist/esm/rslib-runtime.mjs +0 -18
  117. package/dist/esm-node/rslib-runtime.mjs +0 -19
@@ -1,7 +1,38 @@
1
1
  import { DEFAULT_I18NEXT_DETECTION_OPTIONS, mergeDetectionOptions } from "../runtime/i18n/detection/config.mjs";
2
+ import { resolveLocalisedPath, resolveLocalisedUrlsConfig } from "../shared/localisedUrls.mjs";
2
3
  import { getLocaleDetectionOptions } from "../shared/utils.mjs";
3
4
  import * as __rspack_external__modern_js_server_core_hono_a76ca254 from "@modern-js/server-core/hono";
4
5
  const { languageDetector: languageDetector } = __rspack_external__modern_js_server_core_hono_a76ca254;
6
+ const normalizeApiPrefix = (prefix)=>{
7
+ const trimmedPrefix = prefix.trim();
8
+ if (!trimmedPrefix) return null;
9
+ const prefixedPath = trimmedPrefix.startsWith('/') ? trimmedPrefix : `/${trimmedPrefix}`;
10
+ const withoutWildcard = prefixedPath.replace(/\/\*$/, '');
11
+ const normalizedPrefix = withoutWildcard.length > 1 ? withoutWildcard.replace(/\/+$/, '') : withoutWildcard;
12
+ return '/' === normalizedPrefix ? null : normalizedPrefix;
13
+ };
14
+ const collectApiPrefixes = (routes, bffPrefix)=>{
15
+ const prefixes = new Set();
16
+ for (const route of routes){
17
+ if (!route.isApi || !route.urlPath) continue;
18
+ const normalizedPrefix = normalizeApiPrefix(route.urlPath);
19
+ if (normalizedPrefix) prefixes.add(normalizedPrefix);
20
+ }
21
+ const bffPrefixes = Array.isArray(bffPrefix) ? bffPrefix : bffPrefix ? [
22
+ bffPrefix
23
+ ] : [];
24
+ for (const prefix of bffPrefixes){
25
+ const normalizedPrefix = normalizeApiPrefix(prefix);
26
+ if (normalizedPrefix) prefixes.add(normalizedPrefix);
27
+ }
28
+ return [
29
+ ...prefixes
30
+ ];
31
+ };
32
+ const matchesApiPrefix = (pathname, apiPrefixes)=>{
33
+ const normalizedPathname = pathname.startsWith('/') ? pathname : `/${pathname}`;
34
+ return apiPrefixes.some((prefix)=>normalizedPathname === prefix || normalizedPathname.startsWith(`${prefix}/`));
35
+ };
5
36
  const convertToHonoLanguageDetectorOptions = (languages, fallbackLanguage, detectionOptions)=>{
6
37
  const mergedDetection = detectionOptions ? mergeDetectionOptions(detectionOptions) : DEFAULT_I18NEXT_DETECTION_OPTIONS;
7
38
  const order = (mergedDetection.order || []).filter((item)=>![
@@ -91,15 +122,20 @@ const getLanguageFromPath = (req, urlPath, languages)=>{
91
122
  if (languages.includes(firstSegment)) return firstSegment;
92
123
  return null;
93
124
  };
94
- const buildLocalizedUrl = (req, urlPath, language, languages)=>{
125
+ const buildLocalizedUrl = (req, urlPath, language, languages, localisedUrls)=>{
95
126
  const url = new URL(req.url);
96
127
  const pathname = url.pathname;
97
128
  const basePath = urlPath.replace('/*', '');
98
129
  const remainingPath = pathname.startsWith(basePath) ? pathname.slice(basePath.length) : pathname;
99
130
  const segments = remainingPath.split('/').filter(Boolean);
100
- if (segments.length > 0 && languages.includes(segments[0])) segments[0] = language;
101
- else segments.unshift(language);
102
- const newPathname = `/${segments.join('/')}`;
131
+ const localisedUrlsConfig = resolveLocalisedUrlsConfig(localisedUrls);
132
+ const pathWithoutLanguage = segments.length > 0 && languages.includes(segments[0]) ? `/${segments.slice(1).join('/')}` : remainingPath;
133
+ const resolvedPath = localisedUrlsConfig.enabled ? resolveLocalisedPath(pathWithoutLanguage, language, languages, localisedUrlsConfig.map) : pathWithoutLanguage;
134
+ const resolvedSegments = resolvedPath.split('/').filter(Boolean);
135
+ const newPathname = `/${[
136
+ language,
137
+ ...resolvedSegments
138
+ ].join('/')}`;
103
139
  const suffix = `${url.search}${url.hash}`;
104
140
  const localizedUrl = '/' === basePath ? newPathname + suffix : basePath + newPathname + suffix;
105
141
  return localizedUrl;
@@ -109,6 +145,9 @@ const i18nServerPlugin = (options)=>({
109
145
  setup: (api)=>{
110
146
  api.onPrepare(()=>{
111
147
  const { middlewares, routes } = api.getServerContext();
148
+ const serverConfig = api.getServerConfig();
149
+ const bffPrefix = serverConfig?.bff ? serverConfig.bff.prefix ?? '/api' : void 0;
150
+ const apiPrefixes = collectApiPrefixes(routes, bffPrefix);
112
151
  const entryPaths = new Set();
113
152
  routes.forEach((route)=>{
114
153
  if (route.entryName && route.urlPath && '/' !== route.urlPath) {
@@ -120,7 +159,7 @@ const i18nServerPlugin = (options)=>({
120
159
  const { entryName } = route;
121
160
  if (!entryName) return;
122
161
  if (!options.localeDetection) return;
123
- const { localePathRedirect, i18nextDetector = true, languages = [], fallbackLanguage = 'en', detection, ignoreRedirectRoutes } = getLocaleDetectionOptions(entryName, options.localeDetection);
162
+ const { localePathRedirect, i18nextDetector = true, languages = [], fallbackLanguage = 'en', detection, ignoreRedirectRoutes, localisedUrls } = getLocaleDetectionOptions(entryName, options.localeDetection);
124
163
  const staticRoutePrefixes = options.staticRoutePrefixes;
125
164
  const originUrlPath = route.urlPath;
126
165
  const urlPath = originUrlPath.endsWith('/') ? `${originUrlPath}*` : `${originUrlPath}/*`;
@@ -134,6 +173,7 @@ const i18nServerPlugin = (options)=>({
134
173
  handler: async (c, next)=>{
135
174
  const url = new URL(c.req.url);
136
175
  const pathname = url.pathname;
176
+ if (matchesApiPrefix(pathname, apiPrefixes)) return await next();
137
177
  if (isStaticResourceRequest(pathname, staticRoutePrefixes, languages)) return await next();
138
178
  if ('/' === originUrlPath) {
139
179
  const pathSegments = pathname.split('/').filter(Boolean);
@@ -152,6 +192,7 @@ const i18nServerPlugin = (options)=>({
152
192
  handler: async (c, next)=>{
153
193
  const url = new URL(c.req.url);
154
194
  const pathname = url.pathname;
195
+ if (matchesApiPrefix(pathname, apiPrefixes)) return await next();
155
196
  if (isStaticResourceRequest(pathname, staticRoutePrefixes, languages)) return await next();
156
197
  if (shouldIgnoreRedirect(pathname, urlPath, ignoreRedirectRoutes)) return await next();
157
198
  if ('/' === originUrlPath) {
@@ -166,9 +207,14 @@ const i18nServerPlugin = (options)=>({
166
207
  let detectedLanguage = null;
167
208
  if (i18nextDetector) detectedLanguage = c.get('language') || null;
168
209
  const targetLanguage = detectedLanguage || fallbackLanguage;
169
- const localizedUrl = buildLocalizedUrl(c.req, originUrlPath, targetLanguage, languages);
210
+ const localizedUrl = buildLocalizedUrl(c.req, originUrlPath, targetLanguage, languages, localisedUrls);
170
211
  return c.redirect(localizedUrl);
171
212
  }
213
+ const localisedUrlsConfig = resolveLocalisedUrlsConfig(localisedUrls);
214
+ if (localisedUrlsConfig.enabled) {
215
+ const expectedUrl = buildLocalizedUrl(c.req, originUrlPath, language, languages, localisedUrls);
216
+ if (expectedUrl !== `${pathname}${url.search}${url.hash}`) return c.redirect(expectedUrl);
217
+ }
172
218
  await next();
173
219
  }
174
220
  });
@@ -179,4 +225,4 @@ const i18nServerPlugin = (options)=>({
179
225
  });
180
226
  const server = i18nServerPlugin;
181
227
  export default server;
182
- export { i18nServerPlugin };
228
+ export { collectApiPrefixes, i18nServerPlugin, matchesApiPrefix };
@@ -0,0 +1,212 @@
1
+ const LOCALE_PARAM_NAMES = new Set([
2
+ 'lang',
3
+ 'locale',
4
+ 'language'
5
+ ]);
6
+ const normalisePathPattern = (path)=>{
7
+ const withoutDuplicateSlashes = path.replace(/\/+/g, '/');
8
+ const withLeadingSlash = withoutDuplicateSlashes.startsWith('/') ? withoutDuplicateSlashes : `/${withoutDuplicateSlashes}`;
9
+ const withoutTrailingSlash = withLeadingSlash.length > 1 ? withLeadingSlash.replace(/\/+$/, '') : withLeadingSlash;
10
+ return withoutTrailingSlash.replace(/\[(.+?)\]/g, ':$1');
11
+ };
12
+ const normaliseRoutePath = (path)=>{
13
+ const normalized = normalisePathPattern(path);
14
+ return '/' === normalized ? '' : normalized.slice(1);
15
+ };
16
+ const getLocaleParamSegment = (segment)=>{
17
+ if (!segment.startsWith(':')) return null;
18
+ const paramName = segment.slice(1).replace(/\?$/, '');
19
+ return LOCALE_PARAM_NAMES.has(paramName) ? segment : null;
20
+ };
21
+ const splitPathSegments = (path)=>{
22
+ if (!path) return [];
23
+ return normalisePathPattern(path).split('/').filter(Boolean);
24
+ };
25
+ const stripLeadingLocaleParam = (path)=>{
26
+ const segments = splitPathSegments(path);
27
+ const leadingLocaleParam = getLocaleParamSegment(segments[0] || '');
28
+ if (!leadingLocaleParam) return path;
29
+ const remainingPath = segments.slice(1).join('/');
30
+ return remainingPath ? `/${remainingPath}` : void 0;
31
+ };
32
+ const getLeadingLocaleParam = (path)=>{
33
+ const segments = splitPathSegments(path);
34
+ return getLocaleParamSegment(segments[0] || '');
35
+ };
36
+ const resolveLocalisedUrlsConfig = (option)=>{
37
+ if (false === option) return {
38
+ enabled: false,
39
+ map: {}
40
+ };
41
+ if (option && 'object' == typeof option) return {
42
+ enabled: true,
43
+ map: option
44
+ };
45
+ return {
46
+ enabled: true,
47
+ map: {}
48
+ };
49
+ };
50
+ const isLocalisableRoutePath = (path)=>{
51
+ const pathWithoutLocale = stripLeadingLocaleParam(path);
52
+ if (!pathWithoutLocale || '/' === pathWithoutLocale || '*' === pathWithoutLocale) return false;
53
+ return true;
54
+ };
55
+ const joinPath = (parentPath, routePath)=>{
56
+ if (!isLocalisableRoutePath(routePath)) return parentPath;
57
+ const segment = normaliseRoutePath(stripLeadingLocaleParam(routePath) || '');
58
+ return normalisePathPattern(`${parentPath}/${segment}`);
59
+ };
60
+ const ensureLocalisedUrlsForPath = (canonicalPath, languages, localisedUrls)=>{
61
+ const entry = localisedUrls[canonicalPath];
62
+ if (!entry) throw new Error(`localisedUrls is enabled, but route "${canonicalPath}" does not define localised URLs for languages: ${languages.join(', ')}. Add localisedUrls["${canonicalPath}"] or set localeDetection.localisedUrls to false.`);
63
+ const missingLanguages = languages.filter((language)=>!entry[language]);
64
+ if (missingLanguages.length > 0) throw new Error(`localisedUrls["${canonicalPath}"] is missing languages: ${missingLanguages.join(', ')}. Every configured language must have a localised URL.`);
65
+ return entry;
66
+ };
67
+ const validateLocalisedUrls = (routes, languages, localisedUrls)=>{
68
+ const visit = (route, parentPath)=>{
69
+ const canonicalPath = joinPath(parentPath, route.path);
70
+ if (isLocalisableRoutePath(route.path)) ensureLocalisedUrlsForPath(canonicalPath, languages, localisedUrls);
71
+ if ('children' in route && route.children) route.children.forEach((child)=>visit(child, canonicalPath));
72
+ };
73
+ routes.forEach((route)=>visit(route, ''));
74
+ };
75
+ const getLocalisedRoutePaths = (canonicalPath, parentLocalisedPaths, languages, entry)=>{
76
+ const paths = languages.map((language)=>{
77
+ const fullPath = normalisePathPattern(entry[language]);
78
+ const parentPath = normalisePathPattern(parentLocalisedPaths[language] || '/');
79
+ if ('/' === parentPath) return normaliseRoutePath(fullPath) || void 0;
80
+ const parentPrefix = `${parentPath}/`;
81
+ if (!fullPath.startsWith(parentPrefix)) throw new Error(`localisedUrls["${canonicalPath}"].${language} must be nested under "${parentPath}" because its parent route is localised there.`);
82
+ return normaliseRoutePath(fullPath.slice(parentPath.length));
83
+ });
84
+ return Array.from(new Set(paths.filter(Boolean)));
85
+ };
86
+ const transformLocalisedRoute = (route, parentCanonicalPath, parentLocalisedPaths, languages, localisedUrls)=>{
87
+ const canonicalPath = joinPath(parentCanonicalPath, route.path);
88
+ const localisedUrlEntry = isLocalisableRoutePath(route.path) ? ensureLocalisedUrlsForPath(canonicalPath, languages, localisedUrls) : void 0;
89
+ const routeLocalisedPaths = localisedUrlEntry ? languages.reduce((acc, language)=>{
90
+ acc[language] = normalisePathPattern(localisedUrlEntry[language]);
91
+ return acc;
92
+ }, {}) : parentLocalisedPaths;
93
+ const children = 'children' in route && route.children ? route.children.flatMap((child)=>transformLocalisedRoute(child, canonicalPath, routeLocalisedPaths, languages, localisedUrls)) : void 0;
94
+ const baseRoute = {
95
+ ...route,
96
+ ...children ? {
97
+ children
98
+ } : {}
99
+ };
100
+ if (!localisedUrlEntry) return [
101
+ baseRoute
102
+ ];
103
+ return getLocalisedRoutePaths(canonicalPath, parentLocalisedPaths, languages, localisedUrlEntry).map((localisedPath, index)=>cloneRouteWithLocalisedPath(baseRoute, localisedPath, index, canonicalPath));
104
+ };
105
+ const legalRouteIdPart = (value)=>value.replace(/[^a-zA-Z0-9_$-]+/g, '_').replace(/^_+|_+$/g, '') || 'index';
106
+ const suffixRouteIds = (route, suffix)=>{
107
+ const children = 'children' in route && route.children ? route.children.map((child)=>suffixRouteIds(child, suffix)) : void 0;
108
+ return {
109
+ ...route,
110
+ ...route.id ? {
111
+ id: `${route.id}__localised_${suffix}`
112
+ } : {},
113
+ ...children ? {
114
+ children
115
+ } : {}
116
+ };
117
+ };
118
+ const cloneRouteWithLocalisedPath = (route, path, index, canonicalPath)=>{
119
+ const leadingLocaleParam = getLeadingLocaleParam(route.path);
120
+ const localisedPath = leadingLocaleParam ? normaliseRoutePath(`${leadingLocaleParam}/${path}`) : path;
121
+ const routeWithPath = {
122
+ ...route,
123
+ path: localisedPath
124
+ };
125
+ routeWithPath.modernCanonicalPath = canonicalPath;
126
+ return 0 === index ? routeWithPath : suffixRouteIds(routeWithPath, legalRouteIdPart(localisedPath));
127
+ };
128
+ const applyLocalisedUrlsToRoutes = (routes, languages, localisedUrls)=>{
129
+ const rootLocalisedPaths = languages.reduce((acc, language)=>{
130
+ acc[language] = '/';
131
+ return acc;
132
+ }, {});
133
+ validateLocalisedUrls(routes, languages, localisedUrls);
134
+ return routes.flatMap((route)=>transformLocalisedRoute(route, '', rootLocalisedPaths, languages, localisedUrls));
135
+ };
136
+ const escapeRegExp = (value)=>value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
137
+ const getParamName = (segment)=>segment.slice(1).replace(/\?$/, '');
138
+ const compilePathPattern = (pattern)=>{
139
+ const names = [];
140
+ const segments = normalisePathPattern(pattern).split('/').filter(Boolean);
141
+ const source = segments.map((segment)=>{
142
+ if (segment.startsWith(':')) {
143
+ names.push(getParamName(segment));
144
+ const paramPattern = '([^/]+)';
145
+ return segment.endsWith('?') ? `(?:/${paramPattern})?` : `/${paramPattern}`;
146
+ }
147
+ if ('*' === segment) {
148
+ names.push('*');
149
+ return '/(.*)';
150
+ }
151
+ return `/${escapeRegExp(segment)}`;
152
+ }).join('');
153
+ return {
154
+ names,
155
+ regexp: new RegExp(`^${source || '/'}$`)
156
+ };
157
+ };
158
+ const matchPathPattern = (pathname, pattern)=>{
159
+ const { names, regexp } = compilePathPattern(pattern);
160
+ const match = regexp.exec(normalisePathPattern(pathname));
161
+ if (!match) return null;
162
+ return names.reduce((params, name, index)=>{
163
+ params[name] = decodeURIComponent(match[index + 1] || '');
164
+ return params;
165
+ }, {});
166
+ };
167
+ const buildPathFromPattern = (pattern, params)=>{
168
+ const segments = normalisePathPattern(pattern).split('/').filter(Boolean);
169
+ const path = segments.map((segment)=>{
170
+ if (segment.startsWith(':')) {
171
+ const param = params[getParamName(segment)];
172
+ return param ? encodeURIComponent(param) : '';
173
+ }
174
+ if ('*' === segment) return params['*'] || '';
175
+ return segment;
176
+ }).filter(Boolean).join('/');
177
+ return `/${path}`;
178
+ };
179
+ const resolveLocalisedPath = (pathname, targetLanguage, languages, localisedUrls)=>{
180
+ const normalizedPathname = normalisePathPattern(pathname);
181
+ for (const [canonicalPattern, localisedUrlEntry] of Object.entries(localisedUrls)){
182
+ const targetPattern = localisedUrlEntry[targetLanguage];
183
+ if (!targetPattern) continue;
184
+ const params = matchPathPattern(normalizedPathname, canonicalPattern);
185
+ if (params) return buildPathFromPattern(targetPattern, params);
186
+ }
187
+ for (const localisedUrlEntry of Object.values(localisedUrls)){
188
+ const targetPattern = localisedUrlEntry[targetLanguage];
189
+ if (targetPattern) for (const language of languages){
190
+ const sourcePattern = localisedUrlEntry[language];
191
+ if (!sourcePattern) continue;
192
+ const params = matchPathPattern(normalizedPathname, sourcePattern);
193
+ if (params) return buildPathFromPattern(targetPattern, params);
194
+ }
195
+ }
196
+ return normalizedPathname;
197
+ };
198
+ const resolveCanonicalLocalisedPath = (pathname, languages, localisedUrls)=>{
199
+ const normalizedPathname = normalisePathPattern(pathname);
200
+ for (const [canonicalPattern, localisedUrlEntry] of Object.entries(localisedUrls)){
201
+ const canonicalParams = matchPathPattern(normalizedPathname, canonicalPattern);
202
+ if (canonicalParams) return buildPathFromPattern(canonicalPattern, canonicalParams);
203
+ for (const language of languages){
204
+ const sourcePattern = localisedUrlEntry[language];
205
+ if (!sourcePattern) continue;
206
+ const params = matchPathPattern(normalizedPathname, sourcePattern);
207
+ if (params) return buildPathFromPattern(canonicalPattern, params);
208
+ }
209
+ }
210
+ return normalizedPathname;
211
+ };
212
+ export { applyLocalisedUrlsToRoutes, buildPathFromPattern, matchPathPattern, normalisePathPattern, resolveCanonicalLocalisedPath, resolveLocalisedPath, resolveLocalisedUrlsConfig, validateLocalisedUrls };
@@ -2,7 +2,9 @@ import "node:module";
2
2
  import { getPublicDirRoutePrefixes } from "@modern-js/server-core";
3
3
  import fs from "fs";
4
4
  import path from "path";
5
+ import { applyLocalisedUrlsToRoutes, resolveLocalisedUrlsConfig } from "../shared/localisedUrls.mjs";
5
6
  import { getBackendOptions, getLocaleDetectionOptions } from "../shared/utils.mjs";
7
+ import "../runtime/types.mjs";
6
8
  function hasJsonFiles(dirPath) {
7
9
  try {
8
10
  if (!fs.existsSync(dirPath) || !fs.statSync(dirPath).isDirectory()) return false;
@@ -81,6 +83,27 @@ const i18nPlugin = (options = {})=>({
81
83
  plugins
82
84
  };
83
85
  });
86
+ api.modifyFileSystemRoutes(({ entrypoint, routes })=>{
87
+ if (!localeDetection) return {
88
+ entrypoint,
89
+ routes
90
+ };
91
+ const localeDetectionOptions = getLocaleDetectionOptions(entrypoint.entryName, localeDetection);
92
+ const { localePathRedirect, languages = [], localisedUrls } = localeDetectionOptions;
93
+ if (!localePathRedirect || 0 === languages.length) return {
94
+ entrypoint,
95
+ routes
96
+ };
97
+ const localisedUrlsConfig = resolveLocalisedUrlsConfig(localisedUrls);
98
+ if (!localisedUrlsConfig.enabled) return {
99
+ entrypoint,
100
+ routes
101
+ };
102
+ return {
103
+ entrypoint,
104
+ routes: applyLocalisedUrlsToRoutes(routes, languages, localisedUrlsConfig.map)
105
+ };
106
+ });
84
107
  api._internalServerPlugins(({ plugins })=>{
85
108
  const { serverRoutes, metaName } = api.getAppContext();
86
109
  const normalizedConfig = api.getNormalizedConfig();
@@ -1,29 +1,14 @@
1
1
  import "node:module";
2
2
  import { jsx } from "react/jsx-runtime";
3
- import { Link as router_Link, useInRouterContext, useParams } from "@modern-js/runtime/router";
4
- import { useModernI18n } from "./context.mjs";
5
- import { buildLocalizedUrl } from "./utils.mjs";
6
- const useRouterHooks = ()=>{
7
- const inRouter = useInRouterContext();
8
- return {
9
- Link: inRouter ? router_Link : null,
10
- params: inRouter ? useParams() : {},
11
- hasRouter: inRouter
12
- };
13
- };
3
+ import { Link } from "./Link.mjs";
4
+ let warnedDeprecation = false;
14
5
  const I18nLink = ({ to, children, ...props })=>{
15
- const { Link, params, hasRouter } = useRouterHooks();
16
- const { language, supportedLanguages } = useModernI18n();
17
- const currentLang = language;
18
- const localizedTo = buildLocalizedUrl(to, currentLang, supportedLanguages);
19
- if ('development' === process.env.NODE_ENV && hasRouter && !params.lang) console.warn("I18nLink is being used outside of a :lang dynamic route context. This may cause unexpected behavior. Please ensure I18nLink is used within a route that has a :lang parameter.");
20
- if (!hasRouter || !Link) return /*#__PURE__*/ jsx("a", {
21
- href: localizedTo,
22
- ...props,
23
- children: children
24
- });
6
+ if ('development' === process.env.NODE_ENV && !warnedDeprecation) {
7
+ warnedDeprecation = true;
8
+ console.warn("[plugin-i18n] I18nLink is deprecated. Import { Link } from '@modern-js/plugin-i18n/runtime' instead — it accepts the same language-agnostic `to` values.");
9
+ }
25
10
  return /*#__PURE__*/ jsx(Link, {
26
- to: localizedTo,
11
+ to: to,
27
12
  ...props,
28
13
  children: children
29
14
  });
@@ -0,0 +1,210 @@
1
+ import "node:module";
2
+ import { jsx } from "react/jsx-runtime";
3
+ import { useMemo } from "react";
4
+ import { useModernI18n } from "./context.mjs";
5
+ import { canonicalPath } from "./localizedPaths.mjs";
6
+ import { useI18nRouterAdapter } from "./routerAdapter.mjs";
7
+ import { buildLocalizedUrl, splitUrlTarget } from "./utils.mjs";
8
+ const EXTERNAL_TARGET_RE = /^(?:[a-z][a-z0-9+.-]*:|\/\/)/i;
9
+ const warnedTargets = new Set();
10
+ const warnOnce = (key, message)=>{
11
+ if ('development' !== process.env.NODE_ENV || warnedTargets.has(key)) return;
12
+ warnedTargets.add(key);
13
+ console.warn(message);
14
+ };
15
+ const interpolateRouteParams = (pathname, params)=>{
16
+ if (!/[$:*{]/.test(pathname)) return pathname;
17
+ const resolveParam = (name)=>{
18
+ const value = params?.[name];
19
+ return void 0 === value ? void 0 : String(value);
20
+ };
21
+ const segments = pathname.split('/').map((segment)=>{
22
+ if (!segment) return segment;
23
+ if (segment.startsWith('{-$') && segment.endsWith('}')) {
24
+ const value = resolveParam(segment.slice(3, -1));
25
+ return void 0 === value ? null : encodeURIComponent(value);
26
+ }
27
+ if ('$' === segment || '*' === segment) {
28
+ const value = resolveParam('_splat') ?? resolveParam('*');
29
+ return void 0 === value ? null : value.split('/').map(encodeURIComponent).join('/');
30
+ }
31
+ if (segment.startsWith('$')) {
32
+ const value = resolveParam(segment.slice(1));
33
+ if (void 0 === value) {
34
+ warnOnce(`missing-param:${pathname}:${segment}`, `[plugin-i18n] <Link to="${pathname}"> is missing required param "${segment.slice(1)}".`);
35
+ return segment;
36
+ }
37
+ return encodeURIComponent(value);
38
+ }
39
+ if (segment.startsWith(':')) {
40
+ const optional = segment.endsWith('?');
41
+ const name = segment.slice(1, optional ? -1 : void 0);
42
+ const value = resolveParam(name);
43
+ if (void 0 === value) {
44
+ if (optional) return null;
45
+ warnOnce(`missing-param:${pathname}:${segment}`, `[plugin-i18n] <Link to="${pathname}"> is missing required param "${name}".`);
46
+ return segment;
47
+ }
48
+ return encodeURIComponent(value);
49
+ }
50
+ return segment;
51
+ }).filter((segment)=>null !== segment);
52
+ return segments.join('/') || '/';
53
+ };
54
+ const normalizeSearch = (search, searchFromTo)=>{
55
+ if (search && 'object' == typeof search) {
56
+ const entries = Object.entries(search).filter(([, value])=>null != value);
57
+ const searchObject = Object.fromEntries(entries.map(([key, value])=>[
58
+ key,
59
+ String(value)
60
+ ]));
61
+ const params = new URLSearchParams(searchObject);
62
+ const serialized = params.toString();
63
+ return {
64
+ searchString: serialized ? `?${serialized}` : '',
65
+ searchObject
66
+ };
67
+ }
68
+ const raw = 'string' == typeof search && search ? search : searchFromTo;
69
+ if (!raw) return {
70
+ searchString: '',
71
+ searchObject: void 0
72
+ };
73
+ const searchString = raw.startsWith('?') ? raw : `?${raw}`;
74
+ const searchObject = {};
75
+ new URLSearchParams(searchString).forEach((value, key)=>{
76
+ searchObject[key] = value;
77
+ });
78
+ return {
79
+ searchString,
80
+ searchObject
81
+ };
82
+ };
83
+ const splitActiveProps = (active, activeProps)=>{
84
+ if (!active || !activeProps) return {};
85
+ return activeProps;
86
+ };
87
+ const mergeClassNames = (...values)=>{
88
+ const classNames = values.filter((value)=>'string' == typeof value && value.length > 0);
89
+ return classNames.length > 0 ? classNames.join(' ') : void 0;
90
+ };
91
+ const Link = (props)=>{
92
+ const { to, params, children, hash: hashProp, search: searchProp, hashScrollIntoView, activeOptions, activeProps, ...rest } = props;
93
+ const adapter = useI18nRouterAdapter();
94
+ const { language, supportedLanguages, localisedUrls } = useModernI18n();
95
+ const config = {
96
+ languages: supportedLanguages,
97
+ localisedUrls
98
+ };
99
+ const isExternal = EXTERNAL_TARGET_RE.test(to);
100
+ const isBareHash = to.startsWith('#');
101
+ const target = useMemo(()=>{
102
+ if (isExternal || isBareHash) return null;
103
+ const { pathname, search: searchFromTo, hash: hashFromTo } = splitUrlTarget(to);
104
+ const interpolated = interpolateRouteParams(pathname || '/', params);
105
+ const firstSegment = interpolated.split('/').filter(Boolean)[0];
106
+ if (firstSegment && supportedLanguages.includes(firstSegment)) warnOnce(`lang-prefix:${to}`, `[plugin-i18n] <Link to="${to}"> starts with a language prefix. Write language-agnostic canonical paths; the Link localizes them automatically.`);
107
+ const localizedPathname = buildLocalizedUrl(interpolated, language, supportedLanguages, localisedUrls);
108
+ const hash = hashProp ?? (hashFromTo ? hashFromTo.slice(1) : '');
109
+ const { searchString, searchObject } = normalizeSearch(searchProp, searchFromTo);
110
+ return {
111
+ canonicalPathname: interpolated,
112
+ localizedPathname,
113
+ hash,
114
+ searchString,
115
+ searchObject,
116
+ href: `${localizedPathname}${searchString}${hash ? `#${hash}` : ''}`
117
+ };
118
+ }, [
119
+ to,
120
+ params,
121
+ hashProp,
122
+ searchProp,
123
+ isExternal,
124
+ isBareHash,
125
+ language,
126
+ supportedLanguages,
127
+ localisedUrls
128
+ ]);
129
+ const isActive = useMemo(()=>{
130
+ if (!target || !adapter.location) return false;
131
+ const current = canonicalPath(adapter.location.pathname, config);
132
+ const targetCanonical = canonicalPath(target.canonicalPathname, config);
133
+ const exact = activeOptions?.exact ?? '/' === targetCanonical;
134
+ if (current === targetCanonical) return true;
135
+ if (exact) return false;
136
+ return current.startsWith('/' === targetCanonical ? '/' : `${targetCanonical}/`);
137
+ }, [
138
+ target,
139
+ adapter.location,
140
+ activeOptions?.exact,
141
+ supportedLanguages,
142
+ localisedUrls
143
+ ]);
144
+ const resolvedActiveProps = splitActiveProps(isActive, activeProps);
145
+ const activeAttributes = isActive ? {
146
+ 'data-status': 'active',
147
+ 'aria-current': rest['aria-current'] ?? resolvedActiveProps['aria-current'] ?? 'page'
148
+ } : {};
149
+ if (!target) {
150
+ const { prefetch: _prefetch, preload: _preload, replace: _replace, ...anchorProps } = rest;
151
+ return /*#__PURE__*/ jsx("a", {
152
+ href: to,
153
+ ...anchorProps,
154
+ children: children
155
+ });
156
+ }
157
+ const { Link: RouterLink, hasRouter, framework } = adapter;
158
+ if (!hasRouter || !RouterLink) {
159
+ const { prefetch: _prefetch, preload: _preload, replace: _replace, ...anchorProps } = rest;
160
+ const { className: activeClassName, style: activeStyle, ...activeRest } = resolvedActiveProps;
161
+ return /*#__PURE__*/ jsx("a", {
162
+ href: target.href,
163
+ ...anchorProps,
164
+ ...activeRest,
165
+ ...activeAttributes,
166
+ className: mergeClassNames(rest.className, activeClassName),
167
+ style: {
168
+ ...rest.style,
169
+ ...activeStyle
170
+ },
171
+ children: children
172
+ });
173
+ }
174
+ const { className: activeClassName, style: activeStyle, ...activeRest } = resolvedActiveProps;
175
+ const mergedClassName = mergeClassNames(rest.className, activeClassName);
176
+ const mergedStyle = {
177
+ ...rest.style,
178
+ ...activeStyle
179
+ };
180
+ if ('tanstack' === framework) return /*#__PURE__*/ jsx(RouterLink, {
181
+ to: target.localizedPathname,
182
+ ...target.searchObject ? {
183
+ search: target.searchObject
184
+ } : {},
185
+ ...target.hash ? {
186
+ hash: target.hash
187
+ } : {},
188
+ ...void 0 === hashScrollIntoView ? {} : {
189
+ hashScrollIntoView
190
+ },
191
+ ...rest,
192
+ ...activeRest,
193
+ ...activeAttributes,
194
+ className: mergedClassName,
195
+ style: mergedStyle,
196
+ children: children
197
+ });
198
+ return /*#__PURE__*/ jsx(RouterLink, {
199
+ to: target.href,
200
+ ...rest,
201
+ ...activeRest,
202
+ ...activeAttributes,
203
+ className: mergedClassName,
204
+ style: mergedStyle,
205
+ children: children
206
+ });
207
+ };
208
+ const runtime_Link = Link;
209
+ export default runtime_Link;
210
+ export { Link, interpolateRouteParams };
@@ -0,0 +1 @@
1
+ import "node:module";