@bleedingdev/modern-js-plugin-i18n 3.2.0-ultramodern.2 → 3.2.0-ultramodern.23

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 (85) hide show
  1. package/dist/cjs/cli/index.js +22 -0
  2. package/dist/cjs/runtime/I18nLink.js +4 -12
  3. package/dist/cjs/runtime/context.js +32 -5
  4. package/dist/cjs/runtime/hooks.js +8 -5
  5. package/dist/cjs/runtime/i18n/backend/defaults.js +1 -1
  6. package/dist/cjs/runtime/i18n/backend/middleware.node.js +4 -4
  7. package/dist/cjs/runtime/i18n/instance.js +4 -2
  8. package/dist/cjs/runtime/index.js +7 -6
  9. package/dist/cjs/runtime/routerAdapter.js +163 -0
  10. package/dist/cjs/runtime/utils.js +63 -94
  11. package/dist/cjs/server/index.js +64 -8
  12. package/dist/cjs/shared/localisedUrls.js +237 -0
  13. package/dist/esm/cli/index.mjs +22 -0
  14. package/dist/esm/runtime/I18nLink.mjs +4 -12
  15. package/dist/esm/runtime/context.mjs +34 -7
  16. package/dist/esm/runtime/hooks.mjs +9 -6
  17. package/dist/esm/runtime/i18n/backend/defaults.mjs +1 -1
  18. package/dist/esm/runtime/i18n/backend/middleware.node.mjs +3 -3
  19. package/dist/esm/runtime/i18n/instance.mjs +4 -2
  20. package/dist/esm/runtime/index.mjs +7 -6
  21. package/dist/esm/runtime/routerAdapter.mjs +129 -0
  22. package/dist/esm/runtime/utils.mjs +11 -30
  23. package/dist/esm/server/index.mjs +57 -7
  24. package/dist/esm/shared/localisedUrls.mjs +191 -0
  25. package/dist/esm-node/cli/index.mjs +22 -0
  26. package/dist/esm-node/runtime/I18nLink.mjs +4 -12
  27. package/dist/esm-node/runtime/context.mjs +34 -7
  28. package/dist/esm-node/runtime/hooks.mjs +9 -6
  29. package/dist/esm-node/runtime/i18n/backend/defaults.mjs +1 -1
  30. package/dist/esm-node/runtime/i18n/backend/middleware.node.mjs +3 -3
  31. package/dist/esm-node/runtime/i18n/instance.mjs +4 -2
  32. package/dist/esm-node/runtime/index.mjs +7 -6
  33. package/dist/esm-node/runtime/routerAdapter.mjs +130 -0
  34. package/dist/esm-node/runtime/utils.mjs +11 -30
  35. package/dist/esm-node/server/index.mjs +57 -7
  36. package/dist/esm-node/shared/localisedUrls.mjs +192 -0
  37. package/dist/types/cli/index.d.ts +21 -0
  38. package/dist/types/runtime/I18nLink.d.ts +23 -0
  39. package/dist/types/runtime/context.d.ts +41 -0
  40. package/dist/types/runtime/hooks.d.ts +30 -0
  41. package/dist/types/runtime/i18n/backend/config.d.ts +2 -0
  42. package/dist/types/runtime/i18n/backend/defaults.d.ts +13 -0
  43. package/dist/types/runtime/i18n/backend/defaults.node.d.ts +8 -0
  44. package/dist/types/runtime/i18n/backend/index.d.ts +3 -0
  45. package/dist/types/runtime/i18n/backend/middleware.common.d.ts +14 -0
  46. package/dist/types/runtime/i18n/backend/middleware.d.ts +12 -0
  47. package/dist/types/runtime/i18n/backend/middleware.node.d.ts +13 -0
  48. package/dist/types/runtime/i18n/backend/sdk-backend.d.ts +53 -0
  49. package/dist/types/runtime/i18n/backend/sdk-event.d.ts +9 -0
  50. package/dist/types/runtime/i18n/detection/config.d.ts +11 -0
  51. package/dist/types/runtime/i18n/detection/index.d.ts +50 -0
  52. package/dist/types/runtime/i18n/detection/middleware.d.ts +24 -0
  53. package/dist/types/runtime/i18n/detection/middleware.node.d.ts +17 -0
  54. package/dist/types/runtime/i18n/index.d.ts +3 -0
  55. package/dist/types/runtime/i18n/instance.d.ts +96 -0
  56. package/dist/types/runtime/i18n/utils.d.ts +29 -0
  57. package/dist/types/runtime/index.d.ts +21 -0
  58. package/dist/types/runtime/routerAdapter.d.ts +26 -0
  59. package/dist/types/runtime/types.d.ts +15 -0
  60. package/dist/types/runtime/utils.d.ts +28 -0
  61. package/dist/types/server/index.d.ts +14 -0
  62. package/dist/types/shared/deepMerge.d.ts +1 -0
  63. package/dist/types/shared/detection.d.ts +11 -0
  64. package/dist/types/shared/localisedUrls.d.ts +13 -0
  65. package/dist/types/shared/type.d.ts +168 -0
  66. package/dist/types/shared/utils.d.ts +5 -0
  67. package/package.json +15 -15
  68. package/rstest.config.mts +39 -0
  69. package/src/cli/index.ts +43 -1
  70. package/src/runtime/I18nLink.tsx +10 -16
  71. package/src/runtime/context.tsx +45 -7
  72. package/src/runtime/hooks.ts +13 -4
  73. package/src/runtime/i18n/backend/defaults.ts +3 -1
  74. package/src/runtime/i18n/backend/middleware.node.ts +1 -1
  75. package/src/runtime/i18n/instance.ts +14 -5
  76. package/src/runtime/index.tsx +10 -2
  77. package/src/runtime/routerAdapter.tsx +333 -0
  78. package/src/runtime/utils.ts +22 -34
  79. package/src/server/index.ts +135 -10
  80. package/src/shared/localisedUrls.ts +393 -0
  81. package/src/shared/type.ts +12 -0
  82. package/tests/localisedUrls.test.ts +278 -0
  83. package/tests/routerAdapter.test.tsx +278 -0
  84. package/dist/esm/rslib-runtime.mjs +0 -18
  85. package/dist/esm-node/rslib-runtime.mjs +0 -19
@@ -1,9 +1,10 @@
1
1
  import "node:module";
2
2
  import { jsx } from "react/jsx-runtime";
3
3
  import { isBrowser } from "@modern-js/runtime";
4
- import { createContext, useCallback, useContext, useMemo } from "react";
4
+ import { createContext, useCallback, useContext, useEffect, useMemo } from "react";
5
5
  import { cacheUserLanguage } from "./i18n/detection/index.mjs";
6
- import { buildLocalizedUrl, detectLanguageFromPath, getEntryPath, shouldIgnoreRedirect, useRouterHooks } from "./utils.mjs";
6
+ import { useI18nRouterAdapter } from "./routerAdapter.mjs";
7
+ import { buildLocalizedUrl, detectLanguageFromPath, getEntryPath, shouldIgnoreRedirect } from "./utils.mjs";
7
8
  const ModernI18nContext = /*#__PURE__*/ createContext(null);
8
9
  const ModernI18nProvider = ({ children, value })=>/*#__PURE__*/ jsx(ModernI18nContext.Provider, {
9
10
  value: value,
@@ -12,9 +13,33 @@ const ModernI18nProvider = ({ children, value })=>/*#__PURE__*/ jsx(ModernI18nCo
12
13
  const useModernI18n = ()=>{
13
14
  const context = useContext(ModernI18nContext);
14
15
  if (!context) throw new Error('useModernI18n must be used within a ModernI18nProvider');
15
- const { language: contextLanguage, i18nInstance, languages, localePathRedirect, ignoreRedirectRoutes, updateLanguage } = context;
16
- const { navigate, location, hasRouter } = useRouterHooks();
17
- const currentLanguage = contextLanguage;
16
+ const { language: contextLanguage, i18nInstance, languages, localePathRedirect, ignoreRedirectRoutes, localisedUrls, updateLanguage } = context;
17
+ const { navigate, location, hasRouter } = useI18nRouterAdapter();
18
+ const pathLanguage = useMemo(()=>{
19
+ if (!localePathRedirect || !location?.pathname) return;
20
+ const detected = detectLanguageFromPath(location.pathname, languages || [], localePathRedirect);
21
+ return detected.detected ? detected.language : void 0;
22
+ }, [
23
+ languages,
24
+ localePathRedirect,
25
+ location?.pathname
26
+ ]);
27
+ const currentLanguage = pathLanguage || contextLanguage;
28
+ useEffect(()=>{
29
+ if (!pathLanguage || pathLanguage === contextLanguage) return;
30
+ updateLanguage?.(pathLanguage);
31
+ i18nInstance?.setLang?.(pathLanguage);
32
+ i18nInstance?.changeLanguage?.(pathLanguage);
33
+ if (isBrowser()) {
34
+ const detectionOptions = i18nInstance.options?.detection;
35
+ cacheUserLanguage(i18nInstance, pathLanguage, detectionOptions);
36
+ }
37
+ }, [
38
+ contextLanguage,
39
+ i18nInstance,
40
+ pathLanguage,
41
+ updateLanguage
42
+ ]);
18
43
  const changeLanguage = useCallback(async (newLang)=>{
19
44
  try {
20
45
  if (!newLang || 'string' != typeof newLang) throw new Error('Language must be a non-empty string');
@@ -31,7 +56,7 @@ const useModernI18n = ()=>{
31
56
  const pathLanguage = detectLanguageFromPath(currentPath, languages || [], localePathRedirect);
32
57
  if (pathLanguage.detected && pathLanguage.language === newLang) return;
33
58
  if (!shouldIgnoreRedirect(relativePath, languages || [], ignoreRedirectRoutes)) {
34
- const newPath = buildLocalizedUrl(relativePath, newLang, languages || []);
59
+ const newPath = buildLocalizedUrl(relativePath, newLang, languages || [], localisedUrls);
35
60
  const newUrl = entryPath + newPath + location.search + location.hash;
36
61
  await navigate(newUrl, {
37
62
  replace: true
@@ -44,7 +69,7 @@ const useModernI18n = ()=>{
44
69
  const pathLanguage = detectLanguageFromPath(currentPath, languages || [], localePathRedirect);
45
70
  if (pathLanguage.detected && pathLanguage.language === newLang) return;
46
71
  if (!shouldIgnoreRedirect(relativePath, languages || [], ignoreRedirectRoutes)) {
47
- const newPath = buildLocalizedUrl(relativePath, newLang, languages || []);
72
+ const newPath = buildLocalizedUrl(relativePath, newLang, languages || [], localisedUrls);
48
73
  const newUrl = entryPath + newPath + window.location.search + window.location.hash;
49
74
  window.history.pushState(null, '', newUrl);
50
75
  }
@@ -59,6 +84,7 @@ const useModernI18n = ()=>{
59
84
  updateLanguage,
60
85
  localePathRedirect,
61
86
  ignoreRedirectRoutes,
87
+ localisedUrls,
62
88
  languages,
63
89
  hasRouter,
64
90
  navigate,
@@ -99,6 +125,7 @@ const useModernI18n = ()=>{
99
125
  changeLanguage,
100
126
  i18nInstance,
101
127
  supportedLanguages: languages || [],
128
+ localisedUrls,
102
129
  isLanguageSupported,
103
130
  isResourcesReady
104
131
  };
@@ -3,7 +3,8 @@ import { isBrowser } from "@modern-js/runtime";
3
3
  import { useEffect, useRef } from "react";
4
4
  import { I18N_SDK_RESOURCES_LOADED_EVENT, getI18nSdkBackendId } from "./i18n/backend/sdk-event.mjs";
5
5
  import { cacheUserLanguage } from "./i18n/detection/index.mjs";
6
- import { buildLocalizedUrl, detectLanguageFromPath, getEntryPath, getPathname, shouldIgnoreRedirect, useRouterHooks } from "./utils.mjs";
6
+ import { useI18nRouterAdapter } from "./routerAdapter.mjs";
7
+ import { buildLocalizedUrl, detectLanguageFromPath, getEntryPath, getPathname, shouldIgnoreRedirect } from "./utils.mjs";
7
8
  function createMinimalI18nInstance(language) {
8
9
  const minimalInstance = {
9
10
  language,
@@ -15,7 +16,7 @@ function createMinimalI18nInstance(language) {
15
16
  };
16
17
  return minimalInstance;
17
18
  }
18
- function createContextValue(lang, i18nInstance, entryName, languages, localePathRedirect, ignoreRedirectRoutes, setLang) {
19
+ function createContextValue(lang, i18nInstance, entryName, languages, localePathRedirect, ignoreRedirectRoutes, localisedUrls, setLang) {
19
20
  const instance = i18nInstance || createMinimalI18nInstance(lang);
20
21
  return {
21
22
  language: lang,
@@ -24,6 +25,7 @@ function createContextValue(lang, i18nInstance, entryName, languages, localePath
24
25
  languages,
25
26
  localePathRedirect,
26
27
  ignoreRedirectRoutes,
28
+ localisedUrls,
27
29
  updateLanguage: setLang
28
30
  };
29
31
  }
@@ -73,9 +75,9 @@ function useSdkResourcesLoader(i18nInstance, setForceUpdate) {
73
75
  setForceUpdate
74
76
  ]);
75
77
  }
76
- function useClientSideRedirect(i18nInstance, localePathRedirect, languages, fallbackLanguage, ignoreRedirectRoutes) {
78
+ function useClientSideRedirect(i18nInstance, localePathRedirect, languages, fallbackLanguage, ignoreRedirectRoutes, localisedUrls) {
77
79
  const hasRedirectedRef = useRef(false);
78
- const { navigate, location, hasRouter } = useRouterHooks();
80
+ const { navigate, location, hasRouter } = useI18nRouterAdapter();
79
81
  useEffect(()=>{
80
82
  if ('browser' !== process.env.MODERN_TARGET) return;
81
83
  if (!localePathRedirect || !i18nInstance) return;
@@ -94,7 +96,7 @@ function useClientSideRedirect(i18nInstance, localePathRedirect, languages, fall
94
96
  const pathDetection = detectLanguageFromPath(currentPathname, languages, localePathRedirect);
95
97
  if (pathDetection.detected) return;
96
98
  const targetLanguage = i18nInstance.language || fallbackLanguage || languages[0] || 'en';
97
- const newPath = buildLocalizedUrl(relativePath, targetLanguage, languages);
99
+ const newPath = buildLocalizedUrl(relativePath, targetLanguage, languages, localisedUrls);
98
100
  const newUrl = entryPath + newPath + currentSearch + currentHash;
99
101
  if (newUrl !== currentPathname + currentSearch + currentHash) {
100
102
  hasRedirectedRef.current = true;
@@ -111,7 +113,8 @@ function useClientSideRedirect(i18nInstance, localePathRedirect, languages, fall
111
113
  i18nInstance,
112
114
  languages,
113
115
  fallbackLanguage,
114
- ignoreRedirectRoutes
116
+ ignoreRedirectRoutes,
117
+ localisedUrls
115
118
  ]);
116
119
  }
117
120
  function useLanguageSync(i18nInstance, localePathRedirect, languages, runtimeContextRef, prevLangRef, setLang) {
@@ -5,7 +5,7 @@ const DEFAULT_I18NEXT_BACKEND_OPTIONS = {
5
5
  };
6
6
  function convertPath(path) {
7
7
  if (!path) return path;
8
- if (path.startsWith('/')) return `${window.__assetPrefix__ || ''}${path}`;
8
+ if (path.startsWith('/')) return "u" < typeof window ? path : `${window.__assetPrefix__ || ''}${path}`;
9
9
  return path;
10
10
  }
11
11
  function convertBackendOptions(options) {
@@ -1,9 +1,9 @@
1
1
  import "node:module";
2
- import i18next_fs_backend from "i18next-fs-backend";
2
+ import cjs from "i18next-fs-backend/cjs";
3
3
  import { useI18nextBackendCommon } from "./middleware.common.mjs";
4
- class FsBackendWithSave extends i18next_fs_backend {
4
+ class FsBackendWithSave extends cjs {
5
5
  save(_language, _namespace, _data) {}
6
6
  }
7
7
  const HttpBackendWithSave = FsBackendWithSave;
8
- const useI18nextBackend = (i18nInstance, backend)=>useI18nextBackendCommon(i18nInstance, FsBackendWithSave, i18next_fs_backend, backend);
8
+ const useI18nextBackend = (i18nInstance, backend)=>useI18nextBackendCommon(i18nInstance, FsBackendWithSave, cjs, backend);
9
9
  export { FsBackendWithSave, HttpBackendWithSave, useI18nextBackend };
@@ -41,10 +41,12 @@ async function createI18nextInstance() {
41
41
  return null;
42
42
  }
43
43
  }
44
+ function getOptionalReactI18nextPackageName() {
45
+ return "react-i18next";
46
+ }
44
47
  async function tryImportReactI18next() {
45
48
  try {
46
- const reactI18next = await import("react-i18next");
47
- return reactI18next;
49
+ return await import(getOptionalReactI18nextPackageName());
48
50
  } catch (error) {
49
51
  return null;
50
52
  }
@@ -18,8 +18,8 @@ import "./types.mjs";
18
18
  const i18nPlugin = (options)=>({
19
19
  name: '@modern-js/plugin-i18n',
20
20
  setup: (api)=>{
21
- const { entryName, i18nInstance: userI18nInstance, initOptions, localeDetection, backend, htmlLangAttr = false } = options;
22
- const { localePathRedirect = false, i18nextDetector = true, languages = [], fallbackLanguage = 'en', detection, ignoreRedirectRoutes } = localeDetection || {};
21
+ const { entryName, i18nInstance: userI18nInstance, initOptions, localeDetection, backend, htmlLangAttr = false, reactI18next = true } = options;
22
+ const { localePathRedirect = false, i18nextDetector = true, languages = [], fallbackLanguage = 'en', detection, ignoreRedirectRoutes, localisedUrls } = localeDetection || {};
23
23
  const { enabled: backendEnabled = false } = backend || {};
24
24
  let latestI18nInstance;
25
25
  let I18nextProvider;
@@ -28,8 +28,8 @@ const i18nPlugin = (options)=>({
28
28
  const { i18n: otherConfig } = api.getRuntimeConfig();
29
29
  const { initOptions: otherInitOptions } = otherConfig || {};
30
30
  const userInitOptions = merge(otherInitOptions || {}, initOptions || {});
31
- const initReactI18next = await getInitReactI18next();
32
- I18nextProvider = await getI18nextProvider();
31
+ const initReactI18next = reactI18next ? await getInitReactI18next() : null;
32
+ I18nextProvider = reactI18next ? await getI18nextProvider() : null;
33
33
  if (initReactI18next) i18nInstance.use(initReactI18next);
34
34
  const pathname = getPathname(context);
35
35
  if (i18nextDetector) useI18nextLanguageDetector(i18nInstance);
@@ -90,14 +90,15 @@ const i18nPlugin = (options)=>({
90
90
  ]);
91
91
  useSdkResourcesLoader(i18nInstance, setForceUpdate);
92
92
  useLanguageSync(i18nInstance, localePathRedirect, languages, runtimeContextRef, prevLangRef, setLang);
93
- useClientSideRedirect(i18nInstance, localePathRedirect, languages, fallbackLanguage, ignoreRedirectRoutes);
94
- const contextValue = useMemo(()=>createContextValue(lang, i18nInstance, entryName, languages, localePathRedirect, ignoreRedirectRoutes, setLang), [
93
+ useClientSideRedirect(i18nInstance, localePathRedirect, languages, fallbackLanguage, ignoreRedirectRoutes, localisedUrls);
94
+ const contextValue = useMemo(()=>createContextValue(lang, i18nInstance, entryName, languages, localePathRedirect, ignoreRedirectRoutes, localisedUrls, setLang), [
95
95
  lang,
96
96
  i18nInstance,
97
97
  entryName,
98
98
  languages,
99
99
  localePathRedirect,
100
100
  ignoreRedirectRoutes,
101
+ localisedUrls,
101
102
  forceUpdate
102
103
  ]);
103
104
  const appContent = /*#__PURE__*/ jsxs(Fragment, {
@@ -0,0 +1,130 @@
1
+ import "node:module";
2
+ import { RuntimeContext, isBrowser } from "@modern-js/runtime";
3
+ import { InternalRuntimeContext } from "@modern-js/runtime/context";
4
+ import { Link as router_Link, useInRouterContext, useLocation, useNavigate, useParams } from "@modern-js/runtime/router";
5
+ import { useCallback, useContext, useEffect, useState } from "react";
6
+ const normalizeUrlPart = (value, prefix)=>{
7
+ if ('string' != typeof value || !value) return '';
8
+ return value.startsWith(prefix) ? value : `${prefix}${value}`;
9
+ };
10
+ const normalizeLocation = (location)=>{
11
+ if (!location || 'object' != typeof location) return null;
12
+ const locationValue = location;
13
+ if ('string' != typeof locationValue.pathname) return null;
14
+ return {
15
+ pathname: locationValue.pathname,
16
+ search: normalizeUrlPart('string' == typeof locationValue.search ? locationValue.search : locationValue.searchStr, '?'),
17
+ hash: normalizeUrlPart(locationValue.hash, '#')
18
+ };
19
+ };
20
+ const getWindowLocation = ()=>{
21
+ if (!isBrowser()) return null;
22
+ return {
23
+ pathname: window.location.pathname,
24
+ search: window.location.search,
25
+ hash: window.location.hash
26
+ };
27
+ };
28
+ const getRouterFramework = (runtimeContext, internalContext, inReactRouter)=>{
29
+ const framework = internalContext.routerFramework || internalContext.routerRuntime?.framework || runtimeContext.routerFramework;
30
+ if (framework) return framework;
31
+ if (internalContext.router?.useRouter || runtimeContext.router?.useRouter) return 'tanstack';
32
+ if (internalContext.router?.useLocation || internalContext.router?.useHref || runtimeContext.router?.useLocation || runtimeContext.router?.useHref) return 'react-router';
33
+ if (inReactRouter) return 'react-router';
34
+ };
35
+ const getRouterInstance = (internalContext, contextRouter)=>{
36
+ if (contextRouter) return contextRouter;
37
+ const router = internalContext.routerInstance || internalContext.routerRuntime?.instance;
38
+ if (!router || 'object' != typeof router) return null;
39
+ return router;
40
+ };
41
+ const getRouterStateLocation = (internalContext, contextRouter)=>{
42
+ const router = getRouterInstance(internalContext, contextRouter);
43
+ return normalizeLocation(router?.stores?.location?.get?.()) || normalizeLocation(router?.state?.location);
44
+ };
45
+ const getRouterParams = (internalContext, contextRouter)=>{
46
+ const router = getRouterInstance(internalContext, contextRouter);
47
+ const matches = router?.stores?.matches?.get?.() || router?.state?.matches;
48
+ if (!Array.isArray(matches)) return {};
49
+ return matches.reduce((params, match)=>{
50
+ if (match?.params) Object.assign(params, match.params);
51
+ return params;
52
+ }, {});
53
+ };
54
+ const useI18nRouterAdapter = ()=>{
55
+ const runtimeContext = useContext(RuntimeContext);
56
+ const internalContext = useContext(InternalRuntimeContext);
57
+ const inReactRouter = useInRouterContext();
58
+ const reactRouterNavigate = inReactRouter ? useNavigate() : null;
59
+ const reactRouterLocation = inReactRouter ? useLocation() : null;
60
+ const reactRouterParams = inReactRouter ? useParams() : {};
61
+ const framework = getRouterFramework(runtimeContext, internalContext, inReactRouter);
62
+ const contextUseRouter = inReactRouter || 'tanstack' !== framework ? void 0 : internalContext.router?.useRouter || runtimeContext.router?.useRouter;
63
+ const contextRouter = contextUseRouter ? contextUseRouter({
64
+ warn: false
65
+ }) : null;
66
+ const [, setRouterVersion] = useState(0);
67
+ const hasRouter = 'tanstack' === framework || 'react-router' === framework || Boolean(reactRouterNavigate);
68
+ useEffect(()=>{
69
+ if ('tanstack' !== framework) return;
70
+ const router = getRouterInstance(internalContext, contextRouter);
71
+ if (!router) return;
72
+ const update = ()=>setRouterVersion((version)=>version + 1);
73
+ const unsubscribers = [];
74
+ if ('function' == typeof router.stores?.location?.subscribe) {
75
+ const unsubscribe = router.stores.location.subscribe(update);
76
+ if ('function' == typeof unsubscribe) unsubscribers.push(unsubscribe);
77
+ }
78
+ if ('function' == typeof router.subscribe) for (const eventType of [
79
+ 'onBeforeNavigate',
80
+ 'onBeforeLoad'
81
+ ]){
82
+ const unsubscribe = router.subscribe(eventType, update);
83
+ if ('function' == typeof unsubscribe) unsubscribers.push(unsubscribe);
84
+ }
85
+ return ()=>{
86
+ for (const unsubscribe of unsubscribers)unsubscribe();
87
+ };
88
+ }, [
89
+ contextRouter,
90
+ framework,
91
+ internalContext
92
+ ]);
93
+ const navigate = useCallback((href, options)=>{
94
+ const router = getRouterInstance(internalContext, contextRouter);
95
+ const activeFramework = getRouterFramework(runtimeContext, internalContext, inReactRouter);
96
+ if ('tanstack' === activeFramework) {
97
+ if ('function' == typeof router?.navigate) return router.navigate({
98
+ to: href,
99
+ replace: options?.replace,
100
+ ...options?.state === void 0 ? {} : {
101
+ state: options.state
102
+ }
103
+ });
104
+ throw new Error('TanStack router instance is not available.');
105
+ }
106
+ if (reactRouterNavigate) return reactRouterNavigate(href, options);
107
+ if ('react-router' === activeFramework) {
108
+ if ('function' == typeof router?.navigate) return router.navigate(href, options);
109
+ throw new Error('React Router instance is not available.');
110
+ }
111
+ }, [
112
+ contextRouter,
113
+ internalContext,
114
+ inReactRouter,
115
+ reactRouterNavigate,
116
+ runtimeContext
117
+ ]);
118
+ const location = (reactRouterLocation ? normalizeLocation(reactRouterLocation) : getRouterStateLocation(internalContext, contextRouter)) || getWindowLocation();
119
+ const params = inReactRouter ? reactRouterParams : getRouterParams(internalContext, contextRouter);
120
+ const Link = 'tanstack' === framework ? internalContext.router?.Link || runtimeContext.router?.Link || null : 'react-router' === framework || inReactRouter ? router_Link : null;
121
+ return {
122
+ framework,
123
+ hasRouter,
124
+ location,
125
+ navigate: hasRouter ? navigate : null,
126
+ Link,
127
+ params
128
+ };
129
+ };
130
+ export { useI18nRouterAdapter };
@@ -1,13 +1,7 @@
1
1
  import "node:module";
2
2
  import { isBrowser } from "@modern-js/runtime";
3
3
  import { getGlobalBasename } from "@modern-js/runtime/context";
4
- import { __webpack_require__ } from "../rslib-runtime.mjs";
5
- import * as __rspack_external__modern_js_runtime_router_2dfd0c78 from "@modern-js/runtime/router";
6
- __webpack_require__.add({
7
- "@modern-js/runtime/router?f1fa" (module) {
8
- module.exports = __rspack_external__modern_js_runtime_router_2dfd0c78;
9
- }
10
- });
4
+ import { resolveLocalisedPath, resolveLocalisedUrlsConfig } from "../shared/localisedUrls.mjs";
11
5
  const getPathname = (context)=>{
12
6
  if (isBrowser()) return window.location.pathname;
13
7
  return context.ssrContext?.request?.pathname || '/';
@@ -23,11 +17,16 @@ const getLanguageFromPath = (pathname, languages, fallbackLanguage)=>{
23
17
  if (languages.includes(firstSegment)) return firstSegment;
24
18
  return fallbackLanguage;
25
19
  };
26
- const buildLocalizedUrl = (pathname, language, languages)=>{
20
+ const buildLocalizedUrl = (pathname, language, languages, localisedUrls)=>{
27
21
  const segments = pathname.split('/').filter(Boolean);
28
- if (segments.length > 0 && languages.includes(segments[0])) segments[0] = language;
29
- else segments.unshift(language);
30
- return `/${segments.join('/')}`;
22
+ const localisedUrlsConfig = resolveLocalisedUrlsConfig(localisedUrls);
23
+ const pathWithoutLanguage = segments.length > 0 && languages.includes(segments[0]) ? `/${segments.slice(1).join('/')}` : pathname;
24
+ const resolvedPath = localisedUrlsConfig.enabled ? resolveLocalisedPath(pathWithoutLanguage, language, languages, localisedUrlsConfig.map) : pathWithoutLanguage;
25
+ const resolvedSegments = resolvedPath.split('/').filter(Boolean);
26
+ return `/${[
27
+ language,
28
+ ...resolvedSegments
29
+ ].join('/')}`;
31
30
  };
32
31
  const detectLanguageFromPath = (pathname, languages, localePathRedirect)=>{
33
32
  if (!localePathRedirect) return {
@@ -55,22 +54,4 @@ const shouldIgnoreRedirect = (pathname, languages, ignoreRedirectRoutes)=>{
55
54
  if ('function' == typeof ignoreRedirectRoutes) return ignoreRedirectRoutes(normalizedPath);
56
55
  return ignoreRedirectRoutes.some((pattern)=>normalizedPath === pattern || normalizedPath.startsWith(`${pattern}/`));
57
56
  };
58
- const useRouterHooks = ()=>{
59
- try {
60
- const { useLocation, useNavigate, useParams } = __webpack_require__("@modern-js/runtime/router?f1fa");
61
- return {
62
- navigate: useNavigate(),
63
- location: useLocation(),
64
- params: useParams(),
65
- hasRouter: true
66
- };
67
- } catch (error) {
68
- return {
69
- navigate: null,
70
- location: null,
71
- params: {},
72
- hasRouter: false
73
- };
74
- }
75
- };
76
- export { buildLocalizedUrl, detectLanguageFromPath, getEntryPath, getLanguageFromPath, getPathname, shouldIgnoreRedirect, useRouterHooks };
57
+ export { buildLocalizedUrl, detectLanguageFromPath, getEntryPath, getLanguageFromPath, getPathname, shouldIgnoreRedirect };
@@ -1,8 +1,39 @@
1
1
  import "node:module";
2
2
  import { DEFAULT_I18NEXT_DETECTION_OPTIONS, mergeDetectionOptions } from "../runtime/i18n/detection/config.mjs";
3
+ import { resolveLocalisedPath, resolveLocalisedUrlsConfig } from "../shared/localisedUrls.mjs";
3
4
  import { getLocaleDetectionOptions } from "../shared/utils.mjs";
4
5
  import * as __rspack_external__modern_js_server_core_hono_a76ca254 from "@modern-js/server-core/hono";
5
6
  const { languageDetector: languageDetector } = __rspack_external__modern_js_server_core_hono_a76ca254;
7
+ const normalizeApiPrefix = (prefix)=>{
8
+ const trimmedPrefix = prefix.trim();
9
+ if (!trimmedPrefix) return null;
10
+ const prefixedPath = trimmedPrefix.startsWith('/') ? trimmedPrefix : `/${trimmedPrefix}`;
11
+ const withoutWildcard = prefixedPath.replace(/\/\*$/, '');
12
+ const normalizedPrefix = withoutWildcard.length > 1 ? withoutWildcard.replace(/\/+$/, '') : withoutWildcard;
13
+ return '/' === normalizedPrefix ? null : normalizedPrefix;
14
+ };
15
+ const collectApiPrefixes = (routes, bffPrefix)=>{
16
+ const prefixes = new Set();
17
+ for (const route of routes){
18
+ if (!route.isApi || !route.urlPath) continue;
19
+ const normalizedPrefix = normalizeApiPrefix(route.urlPath);
20
+ if (normalizedPrefix) prefixes.add(normalizedPrefix);
21
+ }
22
+ const bffPrefixes = Array.isArray(bffPrefix) ? bffPrefix : bffPrefix ? [
23
+ bffPrefix
24
+ ] : [];
25
+ for (const prefix of bffPrefixes){
26
+ const normalizedPrefix = normalizeApiPrefix(prefix);
27
+ if (normalizedPrefix) prefixes.add(normalizedPrefix);
28
+ }
29
+ return [
30
+ ...prefixes
31
+ ];
32
+ };
33
+ const matchesApiPrefix = (pathname, apiPrefixes)=>{
34
+ const normalizedPathname = pathname.startsWith('/') ? pathname : `/${pathname}`;
35
+ return apiPrefixes.some((prefix)=>normalizedPathname === prefix || normalizedPathname.startsWith(`${prefix}/`));
36
+ };
6
37
  const convertToHonoLanguageDetectorOptions = (languages, fallbackLanguage, detectionOptions)=>{
7
38
  const mergedDetection = detectionOptions ? mergeDetectionOptions(detectionOptions) : DEFAULT_I18NEXT_DETECTION_OPTIONS;
8
39
  const order = (mergedDetection.order || []).filter((item)=>![
@@ -92,15 +123,20 @@ const getLanguageFromPath = (req, urlPath, languages)=>{
92
123
  if (languages.includes(firstSegment)) return firstSegment;
93
124
  return null;
94
125
  };
95
- const buildLocalizedUrl = (req, urlPath, language, languages)=>{
126
+ const buildLocalizedUrl = (req, urlPath, language, languages, localisedUrls)=>{
96
127
  const url = new URL(req.url);
97
128
  const pathname = url.pathname;
98
129
  const basePath = urlPath.replace('/*', '');
99
130
  const remainingPath = pathname.startsWith(basePath) ? pathname.slice(basePath.length) : pathname;
100
131
  const segments = remainingPath.split('/').filter(Boolean);
101
- if (segments.length > 0 && languages.includes(segments[0])) segments[0] = language;
102
- else segments.unshift(language);
103
- const newPathname = `/${segments.join('/')}`;
132
+ const localisedUrlsConfig = resolveLocalisedUrlsConfig(localisedUrls);
133
+ const pathWithoutLanguage = segments.length > 0 && languages.includes(segments[0]) ? `/${segments.slice(1).join('/')}` : remainingPath;
134
+ const resolvedPath = localisedUrlsConfig.enabled ? resolveLocalisedPath(pathWithoutLanguage, language, languages, localisedUrlsConfig.map) : pathWithoutLanguage;
135
+ const resolvedSegments = resolvedPath.split('/').filter(Boolean);
136
+ const newPathname = `/${[
137
+ language,
138
+ ...resolvedSegments
139
+ ].join('/')}`;
104
140
  const suffix = `${url.search}${url.hash}`;
105
141
  const localizedUrl = '/' === basePath ? newPathname + suffix : basePath + newPathname + suffix;
106
142
  return localizedUrl;
@@ -110,6 +146,9 @@ const i18nServerPlugin = (options)=>({
110
146
  setup: (api)=>{
111
147
  api.onPrepare(()=>{
112
148
  const { middlewares, routes } = api.getServerContext();
149
+ const serverConfig = api.getServerConfig();
150
+ const bffPrefix = serverConfig?.bff ? serverConfig.bff.prefix ?? '/api' : void 0;
151
+ const apiPrefixes = collectApiPrefixes(routes, bffPrefix);
113
152
  const entryPaths = new Set();
114
153
  routes.forEach((route)=>{
115
154
  if (route.entryName && route.urlPath && '/' !== route.urlPath) {
@@ -121,7 +160,7 @@ const i18nServerPlugin = (options)=>({
121
160
  const { entryName } = route;
122
161
  if (!entryName) return;
123
162
  if (!options.localeDetection) return;
124
- const { localePathRedirect, i18nextDetector = true, languages = [], fallbackLanguage = 'en', detection, ignoreRedirectRoutes } = getLocaleDetectionOptions(entryName, options.localeDetection);
163
+ const { localePathRedirect, i18nextDetector = true, languages = [], fallbackLanguage = 'en', detection, ignoreRedirectRoutes, localisedUrls } = getLocaleDetectionOptions(entryName, options.localeDetection);
125
164
  const staticRoutePrefixes = options.staticRoutePrefixes;
126
165
  const originUrlPath = route.urlPath;
127
166
  const urlPath = originUrlPath.endsWith('/') ? `${originUrlPath}*` : `${originUrlPath}/*`;
@@ -135,6 +174,7 @@ const i18nServerPlugin = (options)=>({
135
174
  handler: async (c, next)=>{
136
175
  const url = new URL(c.req.url);
137
176
  const pathname = url.pathname;
177
+ if (matchesApiPrefix(pathname, apiPrefixes)) return await next();
138
178
  if (isStaticResourceRequest(pathname, staticRoutePrefixes, languages)) return await next();
139
179
  if ('/' === originUrlPath) {
140
180
  const pathSegments = pathname.split('/').filter(Boolean);
@@ -153,6 +193,7 @@ const i18nServerPlugin = (options)=>({
153
193
  handler: async (c, next)=>{
154
194
  const url = new URL(c.req.url);
155
195
  const pathname = url.pathname;
196
+ if (matchesApiPrefix(pathname, apiPrefixes)) return await next();
156
197
  if (isStaticResourceRequest(pathname, staticRoutePrefixes, languages)) return await next();
157
198
  if (shouldIgnoreRedirect(pathname, urlPath, ignoreRedirectRoutes)) return await next();
158
199
  if ('/' === originUrlPath) {
@@ -167,9 +208,18 @@ const i18nServerPlugin = (options)=>({
167
208
  let detectedLanguage = null;
168
209
  if (i18nextDetector) detectedLanguage = c.get('language') || null;
169
210
  const targetLanguage = detectedLanguage || fallbackLanguage;
170
- const localizedUrl = buildLocalizedUrl(c.req, originUrlPath, targetLanguage, languages);
211
+ const localizedUrl = buildLocalizedUrl(c.req, originUrlPath, targetLanguage, languages, localisedUrls);
171
212
  return c.redirect(localizedUrl);
172
213
  }
214
+ const localisedUrlsConfig = resolveLocalisedUrlsConfig(localisedUrls);
215
+ if (localisedUrlsConfig.enabled) {
216
+ const basePath = originUrlPath.replace('/*', '');
217
+ const remainingPath = pathname.startsWith(basePath) ? pathname.slice(basePath.length) : pathname;
218
+ const pathWithoutLanguage = remainingPath.split('/').filter(Boolean).slice(1).join('/');
219
+ const canonicalLocalizedPath = resolveLocalisedPath(`/${pathWithoutLanguage}`, language, languages, localisedUrlsConfig.map);
220
+ const expectedPathname = '/' === basePath ? `/${language}${'/' === canonicalLocalizedPath ? '' : canonicalLocalizedPath}` : `${basePath}/${language}${'/' === canonicalLocalizedPath ? '' : canonicalLocalizedPath}`;
221
+ if (expectedPathname !== pathname) return c.redirect(`${expectedPathname}${url.search}${url.hash}`);
222
+ }
173
223
  await next();
174
224
  }
175
225
  });
@@ -180,4 +230,4 @@ const i18nServerPlugin = (options)=>({
180
230
  });
181
231
  const server = i18nServerPlugin;
182
232
  export default server;
183
- export { i18nServerPlugin };
233
+ export { collectApiPrefixes, i18nServerPlugin, matchesApiPrefix };