@bleedingdev/modern-js-plugin-i18n 3.2.0-ultramodern.9 → 3.2.0-ultramodern.90

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 (80) 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/defaults.node.js +2 -2
  7. package/dist/cjs/runtime/i18n/backend/middleware.node.js +4 -4
  8. package/dist/cjs/runtime/i18n/instance.js +0 -24
  9. package/dist/cjs/runtime/i18n/react-i18next.js +49 -0
  10. package/dist/cjs/runtime/i18n/utils.js +0 -12
  11. package/dist/cjs/runtime/index.js +18 -10
  12. package/dist/cjs/runtime/routerAdapter.js +163 -0
  13. package/dist/cjs/runtime/utils.js +63 -94
  14. package/dist/cjs/server/index.js +60 -8
  15. package/dist/cjs/shared/localisedUrls.js +237 -0
  16. package/dist/esm/cli/index.mjs +22 -0
  17. package/dist/esm/runtime/I18nLink.mjs +4 -12
  18. package/dist/esm/runtime/context.mjs +34 -7
  19. package/dist/esm/runtime/hooks.mjs +9 -6
  20. package/dist/esm/runtime/i18n/backend/defaults.mjs +1 -1
  21. package/dist/esm/runtime/i18n/backend/defaults.node.mjs +2 -2
  22. package/dist/esm/runtime/i18n/backend/middleware.node.mjs +3 -3
  23. package/dist/esm/runtime/i18n/instance.mjs +1 -19
  24. package/dist/esm/runtime/i18n/react-i18next.mjs +15 -0
  25. package/dist/esm/runtime/i18n/utils.mjs +0 -12
  26. package/dist/esm/runtime/index.mjs +19 -11
  27. package/dist/esm/runtime/routerAdapter.mjs +129 -0
  28. package/dist/esm/runtime/utils.mjs +11 -30
  29. package/dist/esm/server/index.mjs +53 -7
  30. package/dist/esm/shared/localisedUrls.mjs +191 -0
  31. package/dist/esm-node/cli/index.mjs +22 -0
  32. package/dist/esm-node/runtime/I18nLink.mjs +4 -12
  33. package/dist/esm-node/runtime/context.mjs +34 -7
  34. package/dist/esm-node/runtime/hooks.mjs +9 -6
  35. package/dist/esm-node/runtime/i18n/backend/defaults.mjs +1 -1
  36. package/dist/esm-node/runtime/i18n/backend/defaults.node.mjs +2 -2
  37. package/dist/esm-node/runtime/i18n/backend/middleware.node.mjs +3 -3
  38. package/dist/esm-node/runtime/i18n/instance.mjs +1 -19
  39. package/dist/esm-node/runtime/i18n/react-i18next.mjs +16 -0
  40. package/dist/esm-node/runtime/i18n/utils.mjs +0 -12
  41. package/dist/esm-node/runtime/index.mjs +19 -11
  42. package/dist/esm-node/runtime/routerAdapter.mjs +130 -0
  43. package/dist/esm-node/runtime/utils.mjs +11 -30
  44. package/dist/esm-node/server/index.mjs +53 -7
  45. package/dist/esm-node/shared/localisedUrls.mjs +192 -0
  46. package/dist/types/runtime/I18nLink.d.ts +15 -0
  47. package/dist/types/runtime/context.d.ts +3 -0
  48. package/dist/types/runtime/hooks.d.ts +4 -2
  49. package/dist/types/runtime/i18n/backend/middleware.node.d.ts +1 -1
  50. package/dist/types/runtime/i18n/instance.d.ts +0 -5
  51. package/dist/types/runtime/i18n/react-i18next.d.ts +7 -0
  52. package/dist/types/runtime/index.d.ts +1 -0
  53. package/dist/types/runtime/routerAdapter.d.ts +26 -0
  54. package/dist/types/runtime/utils.d.ts +2 -7
  55. package/dist/types/server/index.d.ts +6 -0
  56. package/dist/types/shared/localisedUrls.d.ts +13 -0
  57. package/dist/types/shared/type.d.ts +12 -0
  58. package/package.json +18 -22
  59. package/rstest.config.mts +39 -0
  60. package/src/cli/index.ts +43 -1
  61. package/src/runtime/I18nLink.tsx +10 -16
  62. package/src/runtime/context.tsx +45 -7
  63. package/src/runtime/hooks.ts +13 -4
  64. package/src/runtime/i18n/backend/defaults.node.ts +2 -2
  65. package/src/runtime/i18n/backend/defaults.ts +3 -1
  66. package/src/runtime/i18n/backend/middleware.node.ts +1 -1
  67. package/src/runtime/i18n/instance.ts +0 -29
  68. package/src/runtime/i18n/react-i18next.ts +25 -0
  69. package/src/runtime/i18n/utils.ts +4 -26
  70. package/src/runtime/index.tsx +23 -10
  71. package/src/runtime/routerAdapter.tsx +333 -0
  72. package/src/runtime/utils.ts +22 -34
  73. package/src/server/index.ts +117 -10
  74. package/src/shared/localisedUrls.ts +393 -0
  75. package/src/shared/type.ts +12 -0
  76. package/tests/i18nUtils.test.ts +52 -0
  77. package/tests/localisedUrls.test.ts +312 -0
  78. package/tests/routerAdapter.test.tsx +382 -0
  79. package/dist/esm/rslib-runtime.mjs +0 -18
  80. package/dist/esm-node/rslib-runtime.mjs +0 -19
@@ -0,0 +1,312 @@
1
+ import type { NestedRouteForCli } from '@modern-js/types';
2
+ import { describe, expect, test } from '@rstest/core';
3
+ import {
4
+ collectApiPrefixes,
5
+ i18nServerPlugin,
6
+ matchesApiPrefix,
7
+ } from '../src/server';
8
+ import {
9
+ applyLocalisedUrlsToRoutes,
10
+ resolveLocalisedPath,
11
+ validateLocalisedUrls,
12
+ } from '../src/shared/localisedUrls';
13
+
14
+ const createRoute = (
15
+ path: string,
16
+ children?: NestedRouteForCli[],
17
+ ): NestedRouteForCli => ({
18
+ id: path,
19
+ path,
20
+ type: 'nested',
21
+ origin: 'file-system',
22
+ routeType: children ? 'layout' : 'page',
23
+ _component: `${path}.tsx`,
24
+ children,
25
+ });
26
+
27
+ const createRequestContext = (pathname: string) =>
28
+ ({
29
+ req: {
30
+ url: `http://localhost${pathname}`,
31
+ },
32
+ }) as any;
33
+
34
+ describe('localisedUrls', () => {
35
+ test('requires every localisable route path to define every language', () => {
36
+ const routes = [createRoute(':lang', [createRoute('terms-of-service')])];
37
+
38
+ expect(() =>
39
+ validateLocalisedUrls(routes, ['en', 'cs'], {
40
+ '/terms-of-service': {
41
+ en: '/terms-of-service',
42
+ },
43
+ }),
44
+ ).toThrow('missing languages: cs');
45
+ });
46
+
47
+ test('expands route paths to localised aliases', () => {
48
+ const routes = [
49
+ createRoute(':lang', [
50
+ createRoute('terms-of-service'),
51
+ createRoute('products', [createRoute(':slug')]),
52
+ ]),
53
+ ];
54
+
55
+ const localisedRoutes = applyLocalisedUrlsToRoutes(routes, ['en', 'cs'], {
56
+ '/terms-of-service': {
57
+ en: '/terms-of-service',
58
+ cs: '/podminky-pouzivani',
59
+ },
60
+ '/products': {
61
+ en: '/products',
62
+ cs: '/produkty',
63
+ },
64
+ '/products/:slug': {
65
+ en: '/products/:slug',
66
+ cs: '/produkty/:slug',
67
+ },
68
+ });
69
+
70
+ const localeRoute = localisedRoutes[0] as NestedRouteForCli;
71
+ expect(localeRoute.children?.map(route => route.path)).toEqual([
72
+ 'terms-of-service',
73
+ 'podminky-pouzivani',
74
+ 'products',
75
+ 'produkty',
76
+ ]);
77
+
78
+ const productRoutes = localeRoute.children?.filter(
79
+ route => route.path === 'products' || route.path === 'produkty',
80
+ );
81
+ expect(productRoutes?.[0].children?.map(route => route.path)).toEqual([
82
+ ':slug',
83
+ ]);
84
+ expect(productRoutes?.[1].children?.map(route => route.path)).toEqual([
85
+ ':slug',
86
+ ]);
87
+ });
88
+
89
+ test('expands flat locale-prefixed route paths with canonical keys', () => {
90
+ const routes = [
91
+ createRoute('/:lang/about'),
92
+ createRoute('/:lang/products/:slug'),
93
+ ];
94
+
95
+ const localisedRoutes = applyLocalisedUrlsToRoutes(routes, ['en', 'cs'], {
96
+ '/about': {
97
+ en: '/about',
98
+ cs: '/o-nas',
99
+ },
100
+ '/products/:slug': {
101
+ en: '/products/:slug',
102
+ cs: '/produkty/:slug',
103
+ },
104
+ });
105
+
106
+ expect(localisedRoutes.map(route => route.path)).toEqual([
107
+ ':lang/about',
108
+ ':lang/o-nas',
109
+ ':lang/products/:slug',
110
+ ':lang/produkty/:slug',
111
+ ]);
112
+ });
113
+
114
+ test('resolves current localized path to the target language', () => {
115
+ const localisedUrls = {
116
+ '/terms-of-service': {
117
+ en: '/terms-of-service',
118
+ cs: '/podminky-pouzivani',
119
+ },
120
+ '/products/:slug': {
121
+ en: '/products/:slug',
122
+ cs: '/produkty/:slug',
123
+ },
124
+ };
125
+
126
+ expect(
127
+ resolveLocalisedPath(
128
+ '/terms-of-service',
129
+ 'cs',
130
+ ['en', 'cs'],
131
+ localisedUrls,
132
+ ),
133
+ ).toBe('/podminky-pouzivani');
134
+ expect(
135
+ resolveLocalisedPath('/produkty/cervena-bota', 'en', ['en', 'cs'], {
136
+ ...localisedUrls,
137
+ '/products/:slug': {
138
+ en: '/products/:slug',
139
+ cs: '/produkty/:slug',
140
+ },
141
+ }),
142
+ ).toBe('/products/cervena-bota');
143
+ });
144
+
145
+ test('resolves optional route params', () => {
146
+ const localisedUrls = {
147
+ '/products/:slug?': {
148
+ en: '/products/:slug?',
149
+ cs: '/produkty/:slug?',
150
+ },
151
+ };
152
+
153
+ expect(
154
+ resolveLocalisedPath('/products', 'cs', ['en', 'cs'], localisedUrls),
155
+ ).toBe('/produkty');
156
+ expect(
157
+ resolveLocalisedPath(
158
+ '/produkty/cervena-bota',
159
+ 'en',
160
+ ['en', 'cs'],
161
+ localisedUrls,
162
+ ),
163
+ ).toBe('/products/cervena-bota');
164
+ });
165
+
166
+ test('resolves nested optional route params with translated ancestors', () => {
167
+ const localisedUrls = {
168
+ '/checkout': {
169
+ en: '/checkout',
170
+ cs: '/pokladna',
171
+ },
172
+ '/checkout/thank-you': {
173
+ en: '/checkout/thank-you',
174
+ cs: '/pokladna/dekujeme',
175
+ },
176
+ '/checkout/thank-you/:orderId?': {
177
+ en: '/checkout/thank-you/:orderId?',
178
+ cs: '/pokladna/dekujeme/:orderId?',
179
+ },
180
+ };
181
+
182
+ expect(
183
+ resolveLocalisedPath(
184
+ '/checkout/thank-you',
185
+ 'cs',
186
+ ['en', 'cs'],
187
+ localisedUrls,
188
+ ),
189
+ ).toBe('/pokladna/dekujeme');
190
+ expect(
191
+ resolveLocalisedPath(
192
+ '/pokladna/dekujeme/ABC-123',
193
+ 'en',
194
+ ['en', 'cs'],
195
+ localisedUrls,
196
+ ),
197
+ ).toBe('/checkout/thank-you/ABC-123');
198
+ });
199
+ });
200
+
201
+ describe('i18n server API prefix skips', () => {
202
+ test('collects API route prefixes and normalized BFF config prefixes', () => {
203
+ expect(
204
+ collectApiPrefixes(
205
+ [
206
+ { entryName: 'main', isApi: false, urlPath: '/' },
207
+ { isApi: true, urlPath: '/bff-api' },
208
+ { isApi: true, urlPath: '/rpc/*' },
209
+ { isApi: true, urlPath: '/' },
210
+ { isApi: true },
211
+ ],
212
+ ['bff-api/', '/internal-api'],
213
+ ),
214
+ ).toEqual(['/bff-api', '/rpc', '/internal-api']);
215
+ });
216
+
217
+ test('matches API prefixes by exact path or slash-delimited segment', () => {
218
+ const prefixes = ['/bff-api'];
219
+
220
+ expect(matchesApiPrefix('/bff-api', prefixes)).toBe(true);
221
+ expect(matchesApiPrefix('/bff-api/ping', prefixes)).toBe(true);
222
+ expect(matchesApiPrefix('/bff-api-v2', prefixes)).toBe(false);
223
+ expect(matchesApiPrefix('/bff-api-v2/ping', prefixes)).toBe(false);
224
+ });
225
+
226
+ test('skips language detector and redirect middleware for API routes', async () => {
227
+ const middlewares: any[] = [];
228
+ const routes = [
229
+ { entryName: 'main', entryPath: '', urlPath: '/' },
230
+ { entryPath: '', isApi: true, urlPath: '/bff-api' },
231
+ ];
232
+ let prepare: (() => void) | undefined;
233
+
234
+ i18nServerPlugin({
235
+ localeDetection: {
236
+ fallbackLanguage: 'en',
237
+ languages: ['en', 'cs'],
238
+ localePathRedirect: true,
239
+ },
240
+ staticRoutePrefixes: [],
241
+ }).setup({
242
+ getServerConfig: () => ({}),
243
+ getServerContext: () => ({ middlewares, routes }),
244
+ onPrepare: fn => {
245
+ prepare = fn;
246
+ },
247
+ } as any);
248
+
249
+ prepare?.();
250
+
251
+ const detectorMiddleware = middlewares.find(
252
+ middleware => middleware.name === 'i18n-language-detector',
253
+ );
254
+ const redirectMiddleware = middlewares.find(
255
+ middleware => middleware.name === 'i18n-server-middleware',
256
+ );
257
+
258
+ expect(detectorMiddleware).toBeDefined();
259
+ expect(redirectMiddleware).toBeDefined();
260
+
261
+ for (const middleware of [detectorMiddleware, redirectMiddleware]) {
262
+ let nextCalls = 0;
263
+ const response = await middleware.handler(
264
+ createRequestContext('/bff-api/ping'),
265
+ async () => {
266
+ nextCalls++;
267
+ },
268
+ );
269
+
270
+ expect(response).toBeUndefined();
271
+ expect(nextCalls).toBe(1);
272
+ }
273
+ });
274
+
275
+ test('uses /api as the BFF prefix when BFF config is present without prefix', async () => {
276
+ const middlewares: any[] = [];
277
+ const routes = [{ entryName: 'main', entryPath: '', urlPath: '/' }];
278
+ let prepare: (() => void) | undefined;
279
+
280
+ i18nServerPlugin({
281
+ localeDetection: {
282
+ fallbackLanguage: 'en',
283
+ languages: ['en', 'cs'],
284
+ localePathRedirect: true,
285
+ },
286
+ staticRoutePrefixes: [],
287
+ }).setup({
288
+ getServerConfig: () => ({ bff: {} }),
289
+ getServerContext: () => ({ middlewares, routes }),
290
+ onPrepare: fn => {
291
+ prepare = fn;
292
+ },
293
+ } as any);
294
+
295
+ prepare?.();
296
+
297
+ const redirectMiddleware = middlewares.find(
298
+ middleware => middleware.name === 'i18n-server-middleware',
299
+ );
300
+
301
+ let nextCalls = 0;
302
+ const response = await redirectMiddleware.handler(
303
+ createRequestContext('/api/ping'),
304
+ async () => {
305
+ nextCalls++;
306
+ },
307
+ );
308
+
309
+ expect(response).toBeUndefined();
310
+ expect(nextCalls).toBe(1);
311
+ });
312
+ });
@@ -0,0 +1,382 @@
1
+ import {
2
+ InternalRuntimeContext,
3
+ RuntimeContext,
4
+ } from '@modern-js/runtime/context';
5
+ import type React from 'react';
6
+ import type { ComponentType, PropsWithChildren } from 'react';
7
+ import { act } from 'react';
8
+ import { createRoot, type Root } from 'react-dom/client';
9
+ import { i18nPlugin } from '../src/runtime';
10
+ import { ModernI18nProvider, useModernI18n } from '../src/runtime/context';
11
+ import { I18nLink } from '../src/runtime/I18nLink';
12
+ import type { I18nInstance } from '../src/runtime/i18n';
13
+ import { getReactI18nextIntegration } from '../src/runtime/i18n/react-i18next';
14
+
15
+ (
16
+ globalThis as { IS_REACT_ACT_ENVIRONMENT?: boolean }
17
+ ).IS_REACT_ACT_ENVIRONMENT = true;
18
+
19
+ const localisedUrls = {
20
+ '/terms-of-service': {
21
+ en: '/terms-of-service',
22
+ cs: '/podminky-pouzivani',
23
+ },
24
+ };
25
+
26
+ const requestContext = {
27
+ request: {},
28
+ response: {},
29
+ };
30
+
31
+ const TanstackLink = ({ to, children, ...props }: any) => (
32
+ <a href={to} data-router-link="tanstack" {...props}>
33
+ {children}
34
+ </a>
35
+ );
36
+
37
+ function createI18nInstance(language = 'en'): I18nInstance {
38
+ return {
39
+ language,
40
+ isInitialized: true,
41
+ init: () => Promise.resolve(undefined),
42
+ use: () => {},
43
+ createInstance: () => createI18nInstance(language),
44
+ setLang: rstest.fn(async () => undefined),
45
+ changeLanguage: rstest.fn(async () => undefined),
46
+ services: {},
47
+ options: {},
48
+ };
49
+ }
50
+
51
+ function createRuntimeContext(
52
+ router: unknown,
53
+ framework: 'tanstack' | 'react-router',
54
+ ) {
55
+ return {
56
+ isBrowser: true,
57
+ requestContext,
58
+ context: requestContext,
59
+ routerFramework: framework,
60
+ routerInstance: router,
61
+ routerRuntime: {
62
+ framework,
63
+ instance: router,
64
+ },
65
+ router: {
66
+ ...(framework === 'tanstack'
67
+ ? { Link: TanstackLink, useRouter: () => router }
68
+ : { useLocation: () => undefined, useHref: () => undefined }),
69
+ },
70
+ } as any;
71
+ }
72
+
73
+ function createTanstackRuntimeContext(router: unknown) {
74
+ return createRuntimeContext(router, 'tanstack');
75
+ }
76
+
77
+ function createReactRouterRuntimeContext(router: unknown) {
78
+ return createRuntimeContext(router, 'react-router');
79
+ }
80
+
81
+ function collectI18nWrapRoot() {
82
+ let wrapRoot: ((App: ComponentType<any>) => ComponentType<any>) | undefined;
83
+
84
+ i18nPlugin({
85
+ reactI18next: false,
86
+ localeDetection: {
87
+ fallbackLanguage: 'en',
88
+ },
89
+ }).setup?.({
90
+ getRuntimeConfig: () => ({}),
91
+ onBeforeRender: () => undefined,
92
+ wrapRoot: (callback: (App: ComponentType<any>) => ComponentType<any>) => {
93
+ wrapRoot = callback;
94
+ },
95
+ } as any);
96
+
97
+ if (!wrapRoot) {
98
+ throw new Error('Expected i18n runtime plugin to register wrapRoot');
99
+ }
100
+
101
+ return wrapRoot;
102
+ }
103
+
104
+ function createTanstackRouter(pathname = '/en/terms-of-service', lang = 'en') {
105
+ const url = new URL(pathname, 'https://modernjs.test');
106
+
107
+ return {
108
+ navigate: rstest.fn(async () => undefined),
109
+ state: {
110
+ location: {
111
+ pathname: url.pathname,
112
+ searchStr: url.search,
113
+ hash: url.hash,
114
+ },
115
+ matches: [
116
+ {
117
+ params: {
118
+ lang,
119
+ },
120
+ },
121
+ ],
122
+ },
123
+ };
124
+ }
125
+
126
+ async function renderI18nRoot(node: React.ReactNode) {
127
+ const container = document.createElement('div');
128
+ document.body.appendChild(container);
129
+ const root = createRoot(container);
130
+
131
+ await act(async () => {
132
+ root.render(
133
+ <RuntimeContext.Provider
134
+ value={{
135
+ isBrowser: true,
136
+ requestContext,
137
+ context: requestContext,
138
+ }}
139
+ >
140
+ {node}
141
+ </RuntimeContext.Provider>,
142
+ );
143
+ });
144
+
145
+ return {
146
+ container,
147
+ root,
148
+ };
149
+ }
150
+
151
+ async function renderWithRuntime(
152
+ node: React.ReactNode,
153
+ runtimeContext: ReturnType<typeof createTanstackRuntimeContext>,
154
+ ) {
155
+ const container = document.createElement('div');
156
+ document.body.appendChild(container);
157
+ const root = createRoot(container);
158
+
159
+ await act(async () => {
160
+ root.render(
161
+ <InternalRuntimeContext.Provider value={runtimeContext}>
162
+ {node}
163
+ </InternalRuntimeContext.Provider>,
164
+ );
165
+ });
166
+
167
+ return {
168
+ container,
169
+ root,
170
+ };
171
+ }
172
+
173
+ function cleanup(rendered?: { container: HTMLElement; root: Root }) {
174
+ if (!rendered) {
175
+ return;
176
+ }
177
+ act(() => {
178
+ rendered.root.unmount();
179
+ });
180
+ rendered.container.remove();
181
+ }
182
+
183
+ describe('i18n runtime wrapRoot', () => {
184
+ let rendered: { container: HTMLElement; root: Root } | undefined;
185
+
186
+ afterEach(() => {
187
+ cleanup(rendered);
188
+ rendered = undefined;
189
+ });
190
+
191
+ test('renders children when no root App exists yet', async () => {
192
+ const wrapRoot = collectI18nWrapRoot();
193
+ const I18nRoot = wrapRoot(undefined as unknown as ComponentType<any>);
194
+
195
+ rendered = await renderI18nRoot(
196
+ <I18nRoot>
197
+ <main>router content</main>
198
+ </I18nRoot>,
199
+ );
200
+
201
+ expect(rendered.container.textContent).toContain('router content');
202
+ });
203
+
204
+ test('preserves App props and children', async () => {
205
+ const wrapRoot = collectI18nWrapRoot();
206
+ const App = ({ children, label }: PropsWithChildren<{ label: string }>) => (
207
+ <main data-label={label}>{children}</main>
208
+ );
209
+ const I18nRoot = wrapRoot(App);
210
+
211
+ rendered = await renderI18nRoot(
212
+ <I18nRoot label="root">
213
+ <span>router content</span>
214
+ </I18nRoot>,
215
+ );
216
+
217
+ expect(
218
+ rendered.container.querySelector('main')?.getAttribute('data-label'),
219
+ ).toBe('root');
220
+ expect(rendered.container.textContent).toContain('router content');
221
+ });
222
+ });
223
+
224
+ describe('i18n react-i18next integration', () => {
225
+ test('loads the bundled react-i18next integration', async () => {
226
+ const integration = await getReactI18nextIntegration();
227
+
228
+ expect(integration.I18nextProvider).toEqual(expect.any(Function));
229
+ expect(integration.initReactI18next).toBeDefined();
230
+ });
231
+ });
232
+
233
+ describe('i18n router adapter', () => {
234
+ let rendered: { container: HTMLElement; root: Root } | undefined;
235
+
236
+ afterEach(() => {
237
+ cleanup(rendered);
238
+ rendered = undefined;
239
+ window.history.replaceState(null, '', '/');
240
+ });
241
+
242
+ test('uses the TanStack router Link for I18nLink rendering', async () => {
243
+ const router = createTanstackRouter('/cs/podminky-pouzivani', 'cs');
244
+ rendered = await renderWithRuntime(
245
+ <ModernI18nProvider
246
+ value={{
247
+ language: 'cs',
248
+ i18nInstance: createI18nInstance('cs'),
249
+ languages: ['en', 'cs'],
250
+ localePathRedirect: true,
251
+ localisedUrls,
252
+ }}
253
+ >
254
+ <I18nLink to="/terms-of-service" data-testid="terms-link">
255
+ Terms
256
+ </I18nLink>
257
+ </ModernI18nProvider>,
258
+ createTanstackRuntimeContext(router),
259
+ );
260
+
261
+ const link = rendered.container.querySelector<HTMLAnchorElement>(
262
+ '[data-testid="terms-link"]',
263
+ );
264
+ expect(link?.getAttribute('href')).toBe('/cs/podminky-pouzivani');
265
+ expect(link?.getAttribute('data-router-link')).toBe('tanstack');
266
+ });
267
+
268
+ test('uses TanStack-shaped replacement when changeLanguage updates the URL', async () => {
269
+ window.history.replaceState(
270
+ null,
271
+ '',
272
+ '/en/terms-of-service?from=test#section',
273
+ );
274
+
275
+ const router = createTanstackRouter(
276
+ '/en/terms-of-service?from=test#section',
277
+ );
278
+ let changeLanguagePromise: Promise<void> | undefined;
279
+
280
+ const Harness = () => {
281
+ const { changeLanguage } = useModernI18n();
282
+ return (
283
+ <button
284
+ type="button"
285
+ onClick={() => {
286
+ changeLanguagePromise = changeLanguage('cs');
287
+ }}
288
+ >
289
+ Change language
290
+ </button>
291
+ );
292
+ };
293
+
294
+ rendered = await renderWithRuntime(
295
+ <ModernI18nProvider
296
+ value={{
297
+ language: 'en',
298
+ i18nInstance: createI18nInstance('en'),
299
+ languages: ['en', 'cs'],
300
+ localePathRedirect: true,
301
+ localisedUrls,
302
+ }}
303
+ >
304
+ <Harness />
305
+ </ModernI18nProvider>,
306
+ createTanstackRuntimeContext(router),
307
+ );
308
+
309
+ const button = rendered.container.querySelector('button');
310
+
311
+ await act(async () => {
312
+ button?.dispatchEvent(
313
+ new MouseEvent('click', {
314
+ bubbles: true,
315
+ cancelable: true,
316
+ button: 0,
317
+ }),
318
+ );
319
+ await changeLanguagePromise;
320
+ });
321
+
322
+ expect(router.navigate).toHaveBeenCalledWith({
323
+ to: '/cs/podminky-pouzivani?from=test#section',
324
+ replace: true,
325
+ });
326
+ });
327
+
328
+ test('keeps React Router positional replacement when changeLanguage updates the URL', async () => {
329
+ window.history.replaceState(null, '', '/en/terms-of-service');
330
+
331
+ const router = {
332
+ navigate: rstest.fn(async () => undefined),
333
+ };
334
+ let changeLanguagePromise: Promise<void> | undefined;
335
+
336
+ const Harness = () => {
337
+ const { changeLanguage } = useModernI18n();
338
+ return (
339
+ <button
340
+ type="button"
341
+ onClick={() => {
342
+ changeLanguagePromise = changeLanguage('cs');
343
+ }}
344
+ >
345
+ Change language
346
+ </button>
347
+ );
348
+ };
349
+
350
+ rendered = await renderWithRuntime(
351
+ <ModernI18nProvider
352
+ value={{
353
+ language: 'en',
354
+ i18nInstance: createI18nInstance('en'),
355
+ languages: ['en', 'cs'],
356
+ localePathRedirect: true,
357
+ localisedUrls,
358
+ }}
359
+ >
360
+ <Harness />
361
+ </ModernI18nProvider>,
362
+ createReactRouterRuntimeContext(router),
363
+ );
364
+
365
+ const button = rendered.container.querySelector('button');
366
+
367
+ await act(async () => {
368
+ button?.dispatchEvent(
369
+ new MouseEvent('click', {
370
+ bubbles: true,
371
+ cancelable: true,
372
+ button: 0,
373
+ }),
374
+ );
375
+ await changeLanguagePromise;
376
+ });
377
+
378
+ expect(router.navigate).toHaveBeenCalledWith('/cs/podminky-pouzivani', {
379
+ replace: true,
380
+ });
381
+ });
382
+ });
@@ -1,18 +0,0 @@
1
- var __webpack_modules__ = {};
2
- var __webpack_module_cache__ = {};
3
- function __webpack_require__(moduleId) {
4
- var cachedModule = __webpack_module_cache__[moduleId];
5
- if (void 0 !== cachedModule) return cachedModule.exports;
6
- var module = __webpack_module_cache__[moduleId] = {
7
- exports: {}
8
- };
9
- __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
10
- return module.exports;
11
- }
12
- __webpack_require__.m = __webpack_modules__;
13
- (()=>{
14
- __webpack_require__.add = function(modules) {
15
- Object.assign(__webpack_require__.m, modules);
16
- };
17
- })();
18
- export { __webpack_require__ };