@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.
- package/dist/cjs/cli/index.js +22 -0
- package/dist/cjs/runtime/I18nLink.js +4 -12
- package/dist/cjs/runtime/context.js +32 -5
- package/dist/cjs/runtime/hooks.js +8 -5
- package/dist/cjs/runtime/i18n/backend/defaults.js +1 -1
- package/dist/cjs/runtime/i18n/backend/middleware.node.js +4 -4
- package/dist/cjs/runtime/i18n/instance.js +4 -2
- package/dist/cjs/runtime/index.js +7 -6
- package/dist/cjs/runtime/routerAdapter.js +163 -0
- package/dist/cjs/runtime/utils.js +63 -94
- package/dist/cjs/server/index.js +64 -8
- package/dist/cjs/shared/localisedUrls.js +237 -0
- package/dist/esm/cli/index.mjs +22 -0
- package/dist/esm/runtime/I18nLink.mjs +4 -12
- package/dist/esm/runtime/context.mjs +34 -7
- package/dist/esm/runtime/hooks.mjs +9 -6
- package/dist/esm/runtime/i18n/backend/defaults.mjs +1 -1
- package/dist/esm/runtime/i18n/backend/middleware.node.mjs +3 -3
- package/dist/esm/runtime/i18n/instance.mjs +4 -2
- package/dist/esm/runtime/index.mjs +7 -6
- package/dist/esm/runtime/routerAdapter.mjs +129 -0
- package/dist/esm/runtime/utils.mjs +11 -30
- package/dist/esm/server/index.mjs +57 -7
- package/dist/esm/shared/localisedUrls.mjs +191 -0
- package/dist/esm-node/cli/index.mjs +22 -0
- package/dist/esm-node/runtime/I18nLink.mjs +4 -12
- package/dist/esm-node/runtime/context.mjs +34 -7
- package/dist/esm-node/runtime/hooks.mjs +9 -6
- package/dist/esm-node/runtime/i18n/backend/defaults.mjs +1 -1
- package/dist/esm-node/runtime/i18n/backend/middleware.node.mjs +3 -3
- package/dist/esm-node/runtime/i18n/instance.mjs +4 -2
- package/dist/esm-node/runtime/index.mjs +7 -6
- package/dist/esm-node/runtime/routerAdapter.mjs +130 -0
- package/dist/esm-node/runtime/utils.mjs +11 -30
- package/dist/esm-node/server/index.mjs +57 -7
- package/dist/esm-node/shared/localisedUrls.mjs +192 -0
- package/dist/types/cli/index.d.ts +21 -0
- package/dist/types/runtime/I18nLink.d.ts +23 -0
- package/dist/types/runtime/context.d.ts +41 -0
- package/dist/types/runtime/hooks.d.ts +30 -0
- package/dist/types/runtime/i18n/backend/config.d.ts +2 -0
- package/dist/types/runtime/i18n/backend/defaults.d.ts +13 -0
- package/dist/types/runtime/i18n/backend/defaults.node.d.ts +8 -0
- package/dist/types/runtime/i18n/backend/index.d.ts +3 -0
- package/dist/types/runtime/i18n/backend/middleware.common.d.ts +14 -0
- package/dist/types/runtime/i18n/backend/middleware.d.ts +12 -0
- package/dist/types/runtime/i18n/backend/middleware.node.d.ts +13 -0
- package/dist/types/runtime/i18n/backend/sdk-backend.d.ts +53 -0
- package/dist/types/runtime/i18n/backend/sdk-event.d.ts +9 -0
- package/dist/types/runtime/i18n/detection/config.d.ts +11 -0
- package/dist/types/runtime/i18n/detection/index.d.ts +50 -0
- package/dist/types/runtime/i18n/detection/middleware.d.ts +24 -0
- package/dist/types/runtime/i18n/detection/middleware.node.d.ts +17 -0
- package/dist/types/runtime/i18n/index.d.ts +3 -0
- package/dist/types/runtime/i18n/instance.d.ts +96 -0
- package/dist/types/runtime/i18n/utils.d.ts +29 -0
- package/dist/types/runtime/index.d.ts +21 -0
- package/dist/types/runtime/routerAdapter.d.ts +26 -0
- package/dist/types/runtime/types.d.ts +15 -0
- package/dist/types/runtime/utils.d.ts +28 -0
- package/dist/types/server/index.d.ts +14 -0
- package/dist/types/shared/deepMerge.d.ts +1 -0
- package/dist/types/shared/detection.d.ts +11 -0
- package/dist/types/shared/localisedUrls.d.ts +13 -0
- package/dist/types/shared/type.d.ts +168 -0
- package/dist/types/shared/utils.d.ts +5 -0
- package/package.json +15 -15
- package/rstest.config.mts +39 -0
- package/src/cli/index.ts +43 -1
- package/src/runtime/I18nLink.tsx +10 -16
- package/src/runtime/context.tsx +45 -7
- package/src/runtime/hooks.ts +13 -4
- package/src/runtime/i18n/backend/defaults.ts +3 -1
- package/src/runtime/i18n/backend/middleware.node.ts +1 -1
- package/src/runtime/i18n/instance.ts +14 -5
- package/src/runtime/index.tsx +10 -2
- package/src/runtime/routerAdapter.tsx +333 -0
- package/src/runtime/utils.ts +22 -34
- package/src/server/index.ts +135 -10
- package/src/shared/localisedUrls.ts +393 -0
- package/src/shared/type.ts +12 -0
- package/tests/localisedUrls.test.ts +278 -0
- package/tests/routerAdapter.test.tsx +278 -0
- package/dist/esm/rslib-runtime.mjs +0 -18
- 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 {
|
|
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 } =
|
|
17
|
-
const
|
|
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 {
|
|
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 } =
|
|
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
|
|
2
|
+
import cjs from "i18next-fs-backend/cjs";
|
|
3
3
|
import { useI18nextBackendCommon } from "./middleware.common.mjs";
|
|
4
|
-
class FsBackendWithSave extends
|
|
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,
|
|
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
|
-
|
|
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 {
|
|
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
|
-
|
|
29
|
-
|
|
30
|
-
|
|
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
|
-
|
|
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
|
-
|
|
102
|
-
|
|
103
|
-
const
|
|
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 };
|