@intlayer/docs 7.3.10 → 7.3.12
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/generated/docs.entry.cjs +19 -0
- package/dist/cjs/generated/docs.entry.cjs.map +1 -1
- package/dist/esm/generated/docs.entry.mjs +19 -0
- package/dist/esm/generated/docs.entry.mjs.map +1 -1
- package/dist/types/generated/docs.entry.d.ts +1 -0
- package/dist/types/generated/docs.entry.d.ts.map +1 -1
- package/docs/ar/intlayer_with_nuxt.md +305 -421
- package/docs/ar/intlayer_with_react_router_v7.md +33 -4
- package/docs/ar/intlayer_with_react_router_v7_fs_routes.md +516 -0
- package/docs/ar/intlayer_with_tanstack.md +2 -12
- package/docs/ar/intlayer_with_vite+vue.md +1 -0
- package/docs/de/intlayer_with_nuxt.md +296 -394
- package/docs/de/intlayer_with_react_router_v7.md +33 -4
- package/docs/de/intlayer_with_react_router_v7_fs_routes.md +573 -0
- package/docs/de/intlayer_with_tanstack.md +1 -0
- package/docs/de/intlayer_with_vite+vue.md +1 -0
- package/docs/en/intlayer_with_nuxt.md +242 -321
- package/docs/en/intlayer_with_react_router_v7.md +24 -0
- package/docs/en/intlayer_with_react_router_v7_fs_routes.md +570 -0
- package/docs/en/intlayer_with_tanstack.md +2 -12
- package/docs/en/intlayer_with_vite+vue.md +49 -48
- package/docs/en-GB/intlayer_with_nuxt.md +262 -358
- package/docs/en-GB/intlayer_with_react_router_v7.md +33 -4
- package/docs/en-GB/intlayer_with_react_router_v7_fs_routes.md +513 -0
- package/docs/en-GB/intlayer_with_tanstack.md +2 -12
- package/docs/en-GB/intlayer_with_vite+vue.md +1 -0
- package/docs/es/intlayer_with_nuxt.md +284 -375
- package/docs/es/intlayer_with_react_router_v7.md +33 -4
- package/docs/es/intlayer_with_react_router_v7_fs_routes.md +575 -0
- package/docs/es/intlayer_with_tanstack.md +1 -0
- package/docs/es/intlayer_with_vite+vue.md +1 -2
- package/docs/fr/intlayer_with_nuxt.md +288 -387
- package/docs/fr/intlayer_with_react_router_v7.md +34 -5
- package/docs/fr/intlayer_with_react_router_v7_fs_routes.md +574 -0
- package/docs/fr/intlayer_with_tanstack.md +1 -0
- package/docs/fr/intlayer_with_vite+vue.md +1 -0
- package/docs/hi/intlayer_with_nuxt.md +318 -434
- package/docs/hi/intlayer_with_react_router_v7.md +33 -4
- package/docs/hi/intlayer_with_react_router_v7_fs_routes.md +518 -0
- package/docs/hi/intlayer_with_tanstack.md +2 -12
- package/docs/hi/intlayer_with_vite+vue.md +1 -0
- package/docs/id/intlayer_with_nuxt.md +275 -376
- package/docs/id/intlayer_with_react_router_v7.md +29 -4
- package/docs/id/intlayer_with_react_router_v7_fs_routes.md +521 -0
- package/docs/id/intlayer_with_tanstack.md +2 -12
- package/docs/id/intlayer_with_vite+vue.md +1 -0
- package/docs/it/intlayer_with_nuxt.md +312 -408
- package/docs/it/intlayer_with_react_router_v7.md +33 -4
- package/docs/it/intlayer_with_react_router_v7_fs_routes.md +574 -0
- package/docs/it/intlayer_with_tanstack.md +1 -0
- package/docs/ja/intlayer_with_nuxt.md +319 -414
- package/docs/ja/intlayer_with_react_router_v7.md +33 -4
- package/docs/ja/intlayer_with_react_router_v7_fs_routes.md +574 -0
- package/docs/ja/intlayer_with_tanstack.md +2 -12
- package/docs/ja/intlayer_with_vite+vue.md +1 -0
- package/docs/ko/intlayer_with_nuxt.md +307 -406
- package/docs/ko/intlayer_with_react_router_v7.md +33 -4
- package/docs/ko/intlayer_with_react_router_v7_fs_routes.md +515 -0
- package/docs/ko/intlayer_with_tanstack.md +2 -12
- package/docs/ko/intlayer_with_vite+vue.md +1 -0
- package/docs/pl/intlayer_with_nuxt.md +282 -457
- package/docs/pl/intlayer_with_react_router_v7.md +32 -5
- package/docs/pl/intlayer_with_react_router_v7_fs_routes.md +615 -0
- package/docs/pl/intlayer_with_tanstack.md +2 -12
- package/docs/pl/intlayer_with_vite+vue.md +1 -0
- package/docs/pt/intlayer_with_nuxt.md +288 -403
- package/docs/pt/intlayer_with_react_router_v7.md +28 -0
- package/docs/pt/intlayer_with_tanstack.md +1 -0
- package/docs/ru/intlayer_with_nuxt.md +300 -410
- package/docs/ru/intlayer_with_react_router_v7.md +33 -4
- package/docs/ru/intlayer_with_react_router_v7_fs_routes.md +574 -0
- package/docs/ru/intlayer_with_tanstack.md +1 -0
- package/docs/ru/intlayer_with_vite+vue.md +1 -0
- package/docs/tr/intlayer_with_nuxt.md +327 -392
- package/docs/tr/intlayer_with_react_router_v7.md +33 -4
- package/docs/tr/intlayer_with_react_router_v7_fs_routes.md +572 -0
- package/docs/tr/intlayer_with_tanstack.md +2 -12
- package/docs/tr/intlayer_with_vite+vue.md +1 -0
- package/docs/vi/intlayer_with_nuxt.md +282 -399
- package/docs/vi/intlayer_with_react_router_v7.md +29 -4
- package/docs/vi/intlayer_with_react_router_v7_fs_routes.md +523 -0
- package/docs/vi/intlayer_with_tanstack.md +2 -12
- package/docs/vi/intlayer_with_vite+vue.md +1 -0
- package/docs/zh/intlayer_with_nuxt.md +311 -444
- package/docs/zh/intlayer_with_react_router_v7.md +33 -4
- package/docs/zh/intlayer_with_react_router_v7_fs_routes.md +516 -0
- package/docs/zh/intlayer_with_tanstack.md +2 -12
- package/docs/zh/intlayer_with_vite+vue.md +1 -0
- package/package.json +6 -6
- package/src/generated/docs.entry.ts +19 -0
|
@@ -18,6 +18,7 @@ slugs:
|
|
|
18
18
|
- vite-and-react
|
|
19
19
|
- react-router-v7
|
|
20
20
|
applicationTemplate: https://github.com/aymericzip/intlayer-react-router-v7-template
|
|
21
|
+
youtubeVideo: https://www.youtube.com/watch?v=dS9L7uJeak4
|
|
21
22
|
history:
|
|
22
23
|
- version: 6.1.5
|
|
23
24
|
date: 2025-10-03
|
|
@@ -31,6 +32,10 @@ history:
|
|
|
31
32
|
|
|
32
33
|
Это руководство демонстрирует, как интегрировать **Intlayer** для бесшовной интернационализации в проектах на React Router v7 с маршрутизацией, учитывающей локаль, поддержкой TypeScript и современными практиками разработки.
|
|
33
34
|
|
|
35
|
+
## Table of Contents
|
|
36
|
+
|
|
37
|
+
<TOC/>
|
|
38
|
+
|
|
34
39
|
## Что такое Intlayer?
|
|
35
40
|
|
|
36
41
|
**Intlayer** — это инновационная, открытая библиотека интернационализации (i18n), созданная для упрощения поддержки многоязычности в современных веб-приложениях.
|
|
@@ -75,6 +80,29 @@ pnpm add vite-intlayer --save-dev
|
|
|
75
80
|
|
|
76
81
|
### Шаг 2: Конфигурация вашего проекта
|
|
77
82
|
|
|
83
|
+
## Пошаговое руководство по настройке Intlayer в приложении React Router v7 с маршрутами на основе файловой системы
|
|
84
|
+
|
|
85
|
+
<Tab defaultTab="video">
|
|
86
|
+
<TabItem label="Video" value="video">
|
|
87
|
+
|
|
88
|
+
<iframe title="How to translate your React Router v7 app using Intlayer" class="m-auto aspect-[16/9] w-full overflow-hidden rounded-lg border-0" allow="autoplay; gyroscope;" loading="lazy" width="1080" height="auto" src="https://www.youtube.com/embed/dS9L7uJeak4?autoplay=0&origin=http://intlayer.org&controls=0&rel=1"/>
|
|
89
|
+
|
|
90
|
+
</TabItem>
|
|
91
|
+
<TabItem label="Code" value="code">
|
|
92
|
+
|
|
93
|
+
<iframe
|
|
94
|
+
src="https://stackblitz.com/github/aymericzip/intlayer-react-router-v7-template?embed=1&ctl=1&file=intlayer.config.ts"
|
|
95
|
+
className="m-auto overflow-hidden rounded-lg border-0 max-md:size-full max-md:h-[700px] md:aspect-16/9 md:w-full"
|
|
96
|
+
title="Demo CodeSandbox - How to Internationalize your application using Intlayer"
|
|
97
|
+
sandbox="allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts"
|
|
98
|
+
loading="lazy"
|
|
99
|
+
/>
|
|
100
|
+
|
|
101
|
+
</TabItem>
|
|
102
|
+
</Tab>
|
|
103
|
+
|
|
104
|
+
See [Application Template](https://github.com/aymericzip/intlayer-react-router-v7-template) on GitHub.
|
|
105
|
+
|
|
78
106
|
Создайте файл конфигурации для настройки языков вашего приложения:
|
|
79
107
|
|
|
80
108
|
```typescript fileName="intlayer.config.ts" codeFormat="typescript"
|
|
@@ -259,7 +287,7 @@ export const LocalizedLink: FC<LinkProps> = (props) => {
|
|
|
259
287
|
Если вы хотите навигировать по локализованным маршрутам, вы можете использовать хук `useLocalizedNavigate`:
|
|
260
288
|
|
|
261
289
|
```tsx fileName="app/hooks/useLocalizedNavigate.ts"
|
|
262
|
-
import { useLocale } from "intlayer";
|
|
290
|
+
import { useLocale } from "react-intlayer";
|
|
263
291
|
import { type NavigateOptions, type To, useNavigate } from "react-router";
|
|
264
292
|
|
|
265
293
|
import { locacalizeTo } from "~/components/localized-link";
|
|
@@ -317,6 +345,7 @@ import {
|
|
|
317
345
|
getLocaleName,
|
|
318
346
|
getLocalizedUrl,
|
|
319
347
|
getPathWithoutLocale,
|
|
348
|
+
Locales,
|
|
320
349
|
} from "intlayer";
|
|
321
350
|
import { setLocaleInStorage, useIntlayer, useLocale } from "react-intlayer";
|
|
322
351
|
import { Link, useLocation } from "react-router";
|
|
@@ -410,13 +439,13 @@ export default function RootLayout() {
|
|
|
410
439
|
> Обратите внимание, что для использования `intlayerProxy` в продакшене необходимо переместить пакет `vite-intlayer` из `devDependencies` в `dependencies`.
|
|
411
440
|
|
|
412
441
|
```typescript {3,7} fileName="vite.config.ts"
|
|
442
|
+
import { reactRouter } from "@react-router/dev/vite";
|
|
413
443
|
import { defineConfig } from "vite";
|
|
414
|
-
import react from "@vitejs/plugin-react-swc";
|
|
415
444
|
import { intlayer, intlayerProxy } from "vite-intlayer";
|
|
445
|
+
import tsconfigPaths from "vite-tsconfig-paths";
|
|
416
446
|
|
|
417
|
-
// https://vitejs.dev/config/
|
|
418
447
|
export default defineConfig({
|
|
419
|
-
plugins: [
|
|
448
|
+
plugins: [reactRouter(), tsconfigPaths(), intlayer(), intlayerProxy()],
|
|
420
449
|
});
|
|
421
450
|
```
|
|
422
451
|
|
|
@@ -0,0 +1,574 @@
|
|
|
1
|
+
---
|
|
2
|
+
createdAt: 2025-12-07
|
|
3
|
+
updatedAt: 2025-12-07
|
|
4
|
+
title: Как перевести ваше React Router v7 (File-System Routes) – руководство i18n 2025
|
|
5
|
+
description: Узнайте, как добавить интернационализацию (i18n) в ваше приложение на React Router v7 с помощью Intlayer с маршрутизацией на основе файловой системы. Следуйте этому подробному руководству, чтобы сделать ваше приложение многоязычным с маршрутизацией, учитывающей локаль.
|
|
6
|
+
keywords:
|
|
7
|
+
- Интернационализация
|
|
8
|
+
- Документация
|
|
9
|
+
- Intlayer
|
|
10
|
+
- React Router v7
|
|
11
|
+
- fs-routes
|
|
12
|
+
- Маршруты на основе файловой системы
|
|
13
|
+
- React
|
|
14
|
+
- i18n
|
|
15
|
+
- TypeScript
|
|
16
|
+
- Маршрутизация по локали
|
|
17
|
+
slugs:
|
|
18
|
+
- doc
|
|
19
|
+
- environment
|
|
20
|
+
- vite-and-react
|
|
21
|
+
- react-router-v7-fs-routes
|
|
22
|
+
applicationTemplate: https://github.com/aymericzip/intlayer-react-router-v7-template-fs-routes
|
|
23
|
+
youtubeVideo: https://www.youtube.com/watch?v=dS9L7uJeak4
|
|
24
|
+
history:
|
|
25
|
+
- version: 7.3.4
|
|
26
|
+
date: 2025-12-08
|
|
27
|
+
changes: Init history
|
|
28
|
+
---
|
|
29
|
+
|
|
30
|
+
# Переведите ваш React Router v7 (File-System Routes) с Intlayer | Интернационализация (i18n)
|
|
31
|
+
|
|
32
|
+
Это руководство демонстрирует, как интегрировать **Intlayer** для бесшовной интернационализации в проектах на React Router v7 с использованием **маршрутизации на основе файловой системы** (`@react-router/fs-routes`) с маршрутизацией, учитывающей локаль, поддержкой TypeScript и современными практиками разработки.
|
|
33
|
+
|
|
34
|
+
## Table of Contents
|
|
35
|
+
|
|
36
|
+
<TOC/>
|
|
37
|
+
|
|
38
|
+
## Что такое Intlayer?
|
|
39
|
+
|
|
40
|
+
**Intlayer** — это инновационная, открытая библиотека интернационализации (i18n), созданная для упрощения поддержки многоязычности в современных веб-приложениях.
|
|
41
|
+
|
|
42
|
+
С помощью Intlayer вы можете:
|
|
43
|
+
|
|
44
|
+
- **Легко управлять переводами** с использованием декларативных словарей на уровне компонентов.
|
|
45
|
+
- **Динамически локализовать метаданные**, маршруты и контент.
|
|
46
|
+
- **Обеспечить поддержку TypeScript** с автогенерируемыми типами, улучшая автодополнение и обнаружение ошибок.
|
|
47
|
+
- **Воспользоваться расширенными возможностями**, такими как динамическое определение и переключение локали.
|
|
48
|
+
- **Включить маршрутизацию с учетом локали** с помощью системы маршрутизации на основе конфигурации React Router v7.
|
|
49
|
+
|
|
50
|
+
---
|
|
51
|
+
|
|
52
|
+
## Пошаговое руководство по настройке Intlayer в приложении React Router v7 с маршрутами на основе файловой системы
|
|
53
|
+
|
|
54
|
+
<Tab defaultTab="video">
|
|
55
|
+
<TabItem label="Video" value="video">
|
|
56
|
+
|
|
57
|
+
<iframe title="How to translate your React Router v7 (File-System Routes) app using Intlayer" class="m-auto aspect-[16/9] w-full overflow-hidden rounded-lg border-0" allow="autoplay; gyroscope;" loading="lazy" width="1080" height="auto" src="https://www.youtube.com/embed/dS9L7uJeak4?autoplay=0&origin=http://intlayer.org&controls=0&rel=1"/>
|
|
58
|
+
|
|
59
|
+
</TabItem>
|
|
60
|
+
<TabItem label="Code" value="code">
|
|
61
|
+
|
|
62
|
+
<iframe
|
|
63
|
+
src="https://stackblitz.com/github/aymericzip/intlayer-react-router-v7-template-fs-routes?embed=1&ctl=1&file=intlayer.config.ts"
|
|
64
|
+
className="m-auto overflow-hidden rounded-lg border-0 max-md:size-full max-md:h-[700px] md:aspect-16/9 md:w-full"
|
|
65
|
+
title="Demo CodeSandbox - How to Internationalize your application using Intlayer"
|
|
66
|
+
sandbox="allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts"
|
|
67
|
+
loading="lazy"
|
|
68
|
+
/>
|
|
69
|
+
|
|
70
|
+
</TabItem>
|
|
71
|
+
</Tab>
|
|
72
|
+
|
|
73
|
+
See [Application Template](https://github.com/aymericzip/intlayer-react-router-v7-template-fs-routes) on GitHub.
|
|
74
|
+
|
|
75
|
+
### Шаг 1: Установка зависимостей
|
|
76
|
+
|
|
77
|
+
Установите необходимые пакеты с помощью предпочитаемого менеджера пакетов:
|
|
78
|
+
|
|
79
|
+
```bash packageManager="npm"
|
|
80
|
+
npm install intlayer react-intlayer
|
|
81
|
+
npm install vite-intlayer --save-dev
|
|
82
|
+
npm install @react-router/fs-routes --save-dev
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
```bash packageManager="pnpm"
|
|
86
|
+
pnpm add intlayer react-intlayer
|
|
87
|
+
pnpm add vite-intlayer --save-dev
|
|
88
|
+
pnpm add @react-router/fs-routes --save-dev
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
- **intlayer**
|
|
92
|
+
|
|
93
|
+
- **intlayer**
|
|
94
|
+
|
|
95
|
+
Основной пакет, предоставляющий инструменты интернационализации для управления конфигурацией, перевода, [объявления контента](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/dictionary/get_started.md), транспиляции и [CLI-команд](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/intlayer_cli.md).
|
|
96
|
+
|
|
97
|
+
- **react-intlayer**
|
|
98
|
+
Пакет, который интегрирует Intlayer с приложением React. Он предоставляет провайдеры контекста и хуки для интернационализации в React.
|
|
99
|
+
|
|
100
|
+
- **vite-intlayer**
|
|
101
|
+
Включает плагин Vite для интеграции Intlayer с [сборщиком Vite](https://vite.dev/guide/why.html#why-bundle-for-production), а также промежуточное ПО для определения предпочтительной локали пользователя, управления куки и обработки перенаправления URL.
|
|
102
|
+
|
|
103
|
+
- **@react-router/fs-routes**
|
|
104
|
+
Пакет, который обеспечивает маршрутизацию на основе файловой системы для React Router v7.
|
|
105
|
+
|
|
106
|
+
### Шаг 2: Конфигурация вашего проекта
|
|
107
|
+
|
|
108
|
+
Создайте файл конфигурации для настройки языков вашего приложения:
|
|
109
|
+
|
|
110
|
+
```typescript fileName="intlayer.config.ts" codeFormat="typescript"
|
|
111
|
+
import { type IntlayerConfig, Locales } from "intlayer";
|
|
112
|
+
|
|
113
|
+
const config: IntlayerConfig = {
|
|
114
|
+
internationalization: {
|
|
115
|
+
defaultLocale: Locales.ENGLISH, // Язык по умолчанию
|
|
116
|
+
locales: [Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH], // Поддерживаемые языки
|
|
117
|
+
},
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
export default config;
|
|
121
|
+
```
|
|
122
|
+
|
|
123
|
+
```javascript fileName="intlayer.config.mjs" codeFormat="esm"
|
|
124
|
+
import { Locales } from "intlayer";
|
|
125
|
+
|
|
126
|
+
/** @type {import('intlayer').IntlayerConfig} */
|
|
127
|
+
const config = {
|
|
128
|
+
internationalization: {
|
|
129
|
+
defaultLocale: Locales.ENGLISH, // Язык по умолчанию
|
|
130
|
+
locales: [Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH], // Поддерживаемые языки
|
|
131
|
+
},
|
|
132
|
+
};
|
|
133
|
+
|
|
134
|
+
export default config;
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
```javascript fileName="intlayer.config.cjs" codeFormat="commonjs"
|
|
138
|
+
const { Locales } = require("intlayer");
|
|
139
|
+
|
|
140
|
+
/** @type {import('intlayer').IntlayerConfig} */
|
|
141
|
+
const config = {
|
|
142
|
+
internationalization: {
|
|
143
|
+
defaultLocale: Locales.ENGLISH, // Язык по умолчанию
|
|
144
|
+
locales: [Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH], // Поддерживаемые языки
|
|
145
|
+
},
|
|
146
|
+
};
|
|
147
|
+
|
|
148
|
+
module.exports = config;
|
|
149
|
+
```
|
|
150
|
+
|
|
151
|
+
> С помощью этого файла конфигурации вы можете настроить локализованные URL-адреса, перенаправления в middleware, имена cookie, расположение и расширение ваших деклараций контента, отключить логи Intlayer в консоли и многое другое. Для полного списка доступных параметров обратитесь к [документации по конфигурации](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/configuration.md).
|
|
152
|
+
|
|
153
|
+
### Шаг 3: Интеграция Intlayer в вашу конфигурацию Vite
|
|
154
|
+
|
|
155
|
+
Добавьте плагин intlayer в вашу конфигурацию:
|
|
156
|
+
|
|
157
|
+
```typescript fileName="vite.config.ts"
|
|
158
|
+
import { reactRouter } from "@react-router/dev/vite";
|
|
159
|
+
import { defineConfig } from "vite";
|
|
160
|
+
import { intlayer } from "vite-intlayer";
|
|
161
|
+
import tsconfigPaths from "vite-tsconfig-paths";
|
|
162
|
+
|
|
163
|
+
export default defineConfig({
|
|
164
|
+
plugins: [reactRouter(), tsconfigPaths(), intlayer()],
|
|
165
|
+
});
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
> Плагин Vite `intlayer()` используется для интеграции Intlayer с Vite. Он обеспечивает сборку файлов деклараций контента и отслеживает их в режиме разработки. Определяет переменные окружения Intlayer внутри приложения Vite. Кроме того, предоставляет алиасы для оптимизации производительности.
|
|
169
|
+
|
|
170
|
+
### Шаг 4: Настройка маршрутов на основе файловой системы React Router v7
|
|
171
|
+
|
|
172
|
+
Настройте конфигурацию маршрутов для использования маршрутов на основе файловой системы с `flatRoutes`:
|
|
173
|
+
|
|
174
|
+
```typescript fileName="app/routes.ts"
|
|
175
|
+
import type { RouteConfig } from "@react-router/dev/routes";
|
|
176
|
+
import { flatRoutes } from "@react-router/fs-routes";
|
|
177
|
+
import { configuration } from "intlayer";
|
|
178
|
+
|
|
179
|
+
const routes: RouteConfig = flatRoutes({
|
|
180
|
+
// Игнорировать файлы декларации контента, чтобы они не обрабатывались как маршруты
|
|
181
|
+
ignoredRouteFiles: configuration.content.fileExtensions.map(
|
|
182
|
+
(fileExtension) => `**/*${fileExtension}`
|
|
183
|
+
),
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
export default routes;
|
|
187
|
+
```
|
|
188
|
+
|
|
189
|
+
> Функция `flatRoutes` из `@react-router/fs-routes` включает маршрутизацию на основе файловой системы, где структура файлов в директории `routes/` определяет маршруты вашего приложения. Опция `ignoredRouteFiles` гарантирует, что файлы декларации контента Intlayer (`.content.ts`, и т.д.) не обрабатываются как файлы маршрутов.
|
|
190
|
+
|
|
191
|
+
### Шаг 5: Создайте файлы маршрутов с соглашениями файловой системы
|
|
192
|
+
|
|
193
|
+
При маршрутизации на основе файловой системы вы используете плоское соглашение об именовании, где точки (`.`) представляют сегменты пути, а скобки `()` обозначают необязательные сегменты.
|
|
194
|
+
|
|
195
|
+
Создайте следующие файлы в директории `app/routes/`:
|
|
196
|
+
|
|
197
|
+
#### Структура файлов
|
|
198
|
+
|
|
199
|
+
```bash
|
|
200
|
+
app/routes/
|
|
201
|
+
├── ($locale)._layout.tsx # Обертка layout для маршрутов локали
|
|
202
|
+
├── ($locale)._index.tsx # Главная страница (/:locale?)
|
|
203
|
+
├── ($locale)._index.content.ts # Контент главной страницы
|
|
204
|
+
├── ($locale).about.tsx # Страница "О нас" (/:locale?/about)
|
|
205
|
+
└── ($locale).about.content.ts # Контент страницы "О нас"
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
Соглашения об именовании:
|
|
209
|
+
|
|
210
|
+
- `($locale)` - Необязательный динамический сегмент для параметра локали
|
|
211
|
+
- `_layout` - Маршрут layout, который оборачивает дочерние маршруты
|
|
212
|
+
- `_index` - Индексный маршрут (отображается на родительском пути)
|
|
213
|
+
- `.` (точка) - Разделяет сегменты пути (например, `($locale).about` → `/:locale?/about`)
|
|
214
|
+
|
|
215
|
+
#### Компонент Layout
|
|
216
|
+
|
|
217
|
+
```tsx fileName="app/routes/($locale)._layout.tsx"
|
|
218
|
+
import { IntlayerProvider } from "react-intlayer";
|
|
219
|
+
import { Outlet } from "react-router";
|
|
220
|
+
|
|
221
|
+
import { useI18nHTMLAttributes } from "~/hooks/useI18nHTMLAttributes";
|
|
222
|
+
|
|
223
|
+
import type { Route } from "./+types/($locale)._layout";
|
|
224
|
+
|
|
225
|
+
export default function RootLayout({ params }: Route.ComponentProps) {
|
|
226
|
+
useI18nHTMLAttributes();
|
|
227
|
+
|
|
228
|
+
const { locale } = params;
|
|
229
|
+
|
|
230
|
+
return (
|
|
231
|
+
<IntlayerProvider locale={locale}>
|
|
232
|
+
<Outlet />
|
|
233
|
+
</IntlayerProvider>
|
|
234
|
+
);
|
|
235
|
+
}
|
|
236
|
+
```
|
|
237
|
+
|
|
238
|
+
#### Индексная страница
|
|
239
|
+
|
|
240
|
+
```tsx fileName="app/routes/($locale)._index.tsx"
|
|
241
|
+
import { useIntlayer } from "react-intlayer";
|
|
242
|
+
import { LocalizedLink } from "~/components/localized-link";
|
|
243
|
+
|
|
244
|
+
import type { Route } from "./+types/($locale)._index";
|
|
245
|
+
|
|
246
|
+
export default function Page() {
|
|
247
|
+
const { title, description, aboutLink } = useIntlayer("page");
|
|
248
|
+
|
|
249
|
+
return (
|
|
250
|
+
<div>
|
|
251
|
+
<h1>{title}</h1>
|
|
252
|
+
<p>{description}</p>
|
|
253
|
+
<nav>
|
|
254
|
+
<LocalizedLink to="/about">{aboutLink}</LocalizedLink>
|
|
255
|
+
</nav>
|
|
256
|
+
</div>
|
|
257
|
+
);
|
|
258
|
+
}
|
|
259
|
+
```
|
|
260
|
+
|
|
261
|
+
#### Страница "О нас"
|
|
262
|
+
|
|
263
|
+
```tsx fileName="app/routes/($locale).about.tsx"
|
|
264
|
+
import { useIntlayer } from "react-intlayer";
|
|
265
|
+
import { LocalizedLink } from "~/components/localized-link";
|
|
266
|
+
|
|
267
|
+
import type { Route } from "./+types/($locale).about";
|
|
268
|
+
|
|
269
|
+
export default function AboutPage() {
|
|
270
|
+
const { title, content, homeLink } = useIntlayer("about");
|
|
271
|
+
|
|
272
|
+
return (
|
|
273
|
+
<div>
|
|
274
|
+
<h1>{title}</h1>
|
|
275
|
+
<p>{content}</p>
|
|
276
|
+
<nav>
|
|
277
|
+
<LocalizedLink to="/">{homeLink}</LocalizedLink>
|
|
278
|
+
</nav>
|
|
279
|
+
</div>
|
|
280
|
+
);
|
|
281
|
+
}
|
|
282
|
+
```
|
|
283
|
+
|
|
284
|
+
### Шаг 6: Объявите ваш контент
|
|
285
|
+
|
|
286
|
+
Создайте и управляйте объявлениями контента для хранения переводов. Разместите файлы контента рядом с файлами маршрутов:
|
|
287
|
+
|
|
288
|
+
```tsx fileName="app/routes/($locale)._index.content.ts"
|
|
289
|
+
import { t, type Dictionary } from "intlayer";
|
|
290
|
+
|
|
291
|
+
const pageContent = {
|
|
292
|
+
key: "page",
|
|
293
|
+
content: {
|
|
294
|
+
title: t({
|
|
295
|
+
en: "Welcome to React Router v7 + Intlayer",
|
|
296
|
+
es: "Bienvenido a React Router v7 + Intlayer",
|
|
297
|
+
fr: "Bienvenue sur React Router v7 + Intlayer",
|
|
298
|
+
}),
|
|
299
|
+
description: t({
|
|
300
|
+
en: "Build multilingual applications with ease using React Router v7 and Intlayer.",
|
|
301
|
+
es: "Cree aplicaciones multilingües fácilmente usando React Router v7 y Intlayer.",
|
|
302
|
+
fr: "Créez des applications multilingues facilement avec React Router v7 et Intlayer.",
|
|
303
|
+
}),
|
|
304
|
+
aboutLink: t({
|
|
305
|
+
en: "Learn About Us",
|
|
306
|
+
es: "Aprender Sobre Nosotros",
|
|
307
|
+
fr: "En savoir plus sur nous",
|
|
308
|
+
}),
|
|
309
|
+
},
|
|
310
|
+
} satisfies Dictionary;
|
|
311
|
+
|
|
312
|
+
export default pageContent;
|
|
313
|
+
```
|
|
314
|
+
|
|
315
|
+
```tsx fileName="app/routes/($locale).about.content.ts"
|
|
316
|
+
import { t, type Dictionary } from "intlayer";
|
|
317
|
+
|
|
318
|
+
const aboutContent = {
|
|
319
|
+
key: "about",
|
|
320
|
+
content: {
|
|
321
|
+
title: t({
|
|
322
|
+
en: "About Us",
|
|
323
|
+
es: "Sobre Nosotros",
|
|
324
|
+
fr: "À propos de nous",
|
|
325
|
+
}),
|
|
326
|
+
content: t({
|
|
327
|
+
en: "This is the about page content.",
|
|
328
|
+
es: "Este es el contenido de la página de información.",
|
|
329
|
+
fr: "Ceci est le contenu de la page à propos.",
|
|
330
|
+
}),
|
|
331
|
+
homeLink: t({
|
|
332
|
+
en: "Home",
|
|
333
|
+
es: "Inicio",
|
|
334
|
+
fr: "Accueil",
|
|
335
|
+
}),
|
|
336
|
+
},
|
|
337
|
+
} satisfies Dictionary;
|
|
338
|
+
|
|
339
|
+
export default aboutContent;
|
|
340
|
+
```
|
|
341
|
+
|
|
342
|
+
> Ваши объявления контента могут быть определены в любом месте вашего приложения, как только они включены в директорию `contentDir` (по умолчанию, `./app`). И соответствовать расширению файла объявления контента (по умолчанию, `.content.{json,ts,tsx,js,jsx,mjs,mjx,cjs,cjx}`).
|
|
343
|
+
|
|
344
|
+
> Для получения дополнительной информации обратитесь к [документации по объявлениям контента](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/dictionary/get_started.md).
|
|
345
|
+
|
|
346
|
+
### Шаг 7: Создайте компоненты с поддержкой локали
|
|
347
|
+
|
|
348
|
+
Создайте компонент `LocalizedLink` для навигации с учётом локали:
|
|
349
|
+
|
|
350
|
+
```tsx fileName="app/components/localized-link.tsx"
|
|
351
|
+
import type { FC } from "react";
|
|
352
|
+
|
|
353
|
+
import { getLocalizedUrl, type LocalesValues } from "intlayer";
|
|
354
|
+
import { useLocale } from "react-intlayer";
|
|
355
|
+
import { Link, type LinkProps, type To } from "react-router";
|
|
356
|
+
|
|
357
|
+
const isExternalLink = (to: string) => /^(https?:)?\/\//.test(to);
|
|
358
|
+
|
|
359
|
+
// Проверяет, является ли ссылка внешней
|
|
360
|
+
export const locacalizeTo = (to: To, locale: LocalesValues): To => {
|
|
361
|
+
if (typeof to === "string") {
|
|
362
|
+
if (isExternalLink(to)) {
|
|
363
|
+
return to;
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
return getLocalizedUrl(to, locale);
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
if (isExternalLink(to.pathname ?? "")) {
|
|
370
|
+
return to;
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
return {
|
|
374
|
+
...to,
|
|
375
|
+
pathname: getLocalizedUrl(to.pathname ?? "", locale),
|
|
376
|
+
};
|
|
377
|
+
};
|
|
378
|
+
|
|
379
|
+
// Компонент локализованной ссылки, учитывающий текущую локаль
|
|
380
|
+
export const LocalizedLink: FC<LinkProps> = (props) => {
|
|
381
|
+
const { locale } = useLocale();
|
|
382
|
+
|
|
383
|
+
return <Link {...props} to={locacalizeTo(props.to, locale)} />;
|
|
384
|
+
};
|
|
385
|
+
```
|
|
386
|
+
|
|
387
|
+
Если вы хотите навигировать по локализованным маршрутам, вы можете использовать хук `useLocalizedNavigate`:
|
|
388
|
+
|
|
389
|
+
```tsx fileName="app/hooks/useLocalizedNavigate.ts"
|
|
390
|
+
import { useLocale } from "react-intlayer";
|
|
391
|
+
import { type NavigateOptions, type To, useNavigate } from "react-router";
|
|
392
|
+
|
|
393
|
+
import { locacalizeTo } from "~/components/localized-link";
|
|
394
|
+
|
|
395
|
+
export const useLocalizedNavigate = () => {
|
|
396
|
+
const navigate = useNavigate();
|
|
397
|
+
const { locale } = useLocale();
|
|
398
|
+
|
|
399
|
+
const localizedNavigate = (to: To, options?: NavigateOptions) => {
|
|
400
|
+
const localedTo = locacalizeTo(to, locale);
|
|
401
|
+
|
|
402
|
+
navigate(localedTo, options);
|
|
403
|
+
};
|
|
404
|
+
|
|
405
|
+
return localizedNavigate;
|
|
406
|
+
};
|
|
407
|
+
```
|
|
408
|
+
|
|
409
|
+
### Шаг 8: Создайте компонент переключателя локали
|
|
410
|
+
|
|
411
|
+
Создайте компонент, позволяющий пользователям менять язык:
|
|
412
|
+
|
|
413
|
+
```tsx fileName="app/components/locale-switcher.tsx"
|
|
414
|
+
import type { FC } from "react";
|
|
415
|
+
|
|
416
|
+
import {
|
|
417
|
+
getHTMLTextDir,
|
|
418
|
+
getLocaleName,
|
|
419
|
+
getLocalizedUrl,
|
|
420
|
+
getPathWithoutLocale,
|
|
421
|
+
Locales,
|
|
422
|
+
} from "intlayer";
|
|
423
|
+
import { useIntlayer, useLocale } from "react-intlayer";
|
|
424
|
+
import { Link, useLocation } from "react-router";
|
|
425
|
+
|
|
426
|
+
export const LocaleSwitcher: FC = () => {
|
|
427
|
+
const { localeSwitcherLabel } = useIntlayer("locale-switcher");
|
|
428
|
+
const { pathname } = useLocation();
|
|
429
|
+
|
|
430
|
+
const { availableLocales, locale } = useLocale();
|
|
431
|
+
|
|
432
|
+
const pathWithoutLocale = getPathWithoutLocale(pathname);
|
|
433
|
+
|
|
434
|
+
return (
|
|
435
|
+
<ol>
|
|
436
|
+
{availableLocales.map((localeItem) => (
|
|
437
|
+
<li key={localeItem}>
|
|
438
|
+
<Link
|
|
439
|
+
aria-current={localeItem === locale ? "page" : undefined}
|
|
440
|
+
aria-label={`${localeSwitcherLabel.value} ${getLocaleName(localeItem)}`}
|
|
441
|
+
reloadDocument // Перезагрузить страницу для применения новой локали
|
|
442
|
+
to={getLocalizedUrl(pathWithoutLocale, localeItem)}
|
|
443
|
+
>
|
|
444
|
+
<span>
|
|
445
|
+
{/* Локаль - например, FR */}
|
|
446
|
+
{localeItem}
|
|
447
|
+
</span>
|
|
448
|
+
<span>
|
|
449
|
+
{/* Язык на его собственной локали - например, Français */}
|
|
450
|
+
{getLocaleName(localeItem, locale)}
|
|
451
|
+
</span>
|
|
452
|
+
<span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
|
|
453
|
+
{/* Язык на текущей локали - например, Francés при установленной локали Locales.SPANISH */}
|
|
454
|
+
{getLocaleName(localeItem)}
|
|
455
|
+
</span>
|
|
456
|
+
<span dir="ltr" lang={Locales.ENGLISH}>
|
|
457
|
+
{/* Язык на английском - например, French */}
|
|
458
|
+
{getLocaleName(localeItem, Locales.ENGLISH)}
|
|
459
|
+
</span>
|
|
460
|
+
</Link>
|
|
461
|
+
</li>
|
|
462
|
+
))}
|
|
463
|
+
</ol>
|
|
464
|
+
);
|
|
465
|
+
};
|
|
466
|
+
```
|
|
467
|
+
|
|
468
|
+
> Чтобы узнать больше о хуке `useLocale`, обратитесь к [документации](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/packages/react-intlayer/useLocale.md).
|
|
469
|
+
|
|
470
|
+
### Шаг 9: Добавление управления атрибутами HTML (необязательно)
|
|
471
|
+
|
|
472
|
+
Создайте хук для управления атрибутами lang и dir в HTML:
|
|
473
|
+
|
|
474
|
+
```tsx fileName="app/hooks/useI18nHTMLAttributes.tsx"
|
|
475
|
+
import { getHTMLTextDir } from "intlayer";
|
|
476
|
+
import { useEffect } from "react";
|
|
477
|
+
import { useLocale } from "react-intlayer";
|
|
478
|
+
|
|
479
|
+
export const useI18nHTMLAttributes = () => {
|
|
480
|
+
const { locale } = useLocale();
|
|
481
|
+
|
|
482
|
+
useEffect(() => {
|
|
483
|
+
document.documentElement.lang = locale;
|
|
484
|
+
document.documentElement.dir = getHTMLTextDir(locale);
|
|
485
|
+
}, [locale]);
|
|
486
|
+
};
|
|
487
|
+
```
|
|
488
|
+
|
|
489
|
+
Этот хук уже используется в компоненте layout (`($locale)._layout.tsx`), показанном в Шаге 5.
|
|
490
|
+
|
|
491
|
+
### Шаг 10: Добавьте middleware (необязательно)
|
|
492
|
+
|
|
493
|
+
Вы также можете использовать `intlayerProxy` для добавления маршрутизации на стороне сервера в ваше приложение. Этот плагин автоматически определит текущую локаль на основе URL и установит соответствующее cookie с локалью. Если локаль не указана, плагин определит наиболее подходящую локаль на основе языковых предпочтений браузера пользователя. Если локаль не будет обнаружена, произойдет перенаправление на локаль по умолчанию.
|
|
494
|
+
|
|
495
|
+
> Обратите внимание, что для использования `intlayerProxy` в продакшене необходимо переместить пакет `vite-intlayer` из `devDependencies` в `dependencies`.
|
|
496
|
+
|
|
497
|
+
```typescript {3,7} fileName="vite.config.ts"
|
|
498
|
+
import { reactRouter } from "@react-router/dev/vite";
|
|
499
|
+
import { defineConfig } from "vite";
|
|
500
|
+
import { intlayer, intlayerProxy } from "vite-intlayer";
|
|
501
|
+
import tsconfigPaths from "vite-tsconfig-paths";
|
|
502
|
+
|
|
503
|
+
export default defineConfig({
|
|
504
|
+
plugins: [reactRouter(), tsconfigPaths(), intlayer(), intlayerProxy()],
|
|
505
|
+
});
|
|
506
|
+
```
|
|
507
|
+
|
|
508
|
+
---
|
|
509
|
+
|
|
510
|
+
## Настройка TypeScript
|
|
511
|
+
|
|
512
|
+
Intlayer использует расширение модулей (module augmentation), чтобы использовать преимущества TypeScript и сделать ваш код более надежным.
|
|
513
|
+
|
|
514
|
+
Убедитесь, что ваша конфигурация TypeScript включает автоматически сгенерированные типы:
|
|
515
|
+
|
|
516
|
+
```json5 fileName="tsconfig.json"
|
|
517
|
+
{
|
|
518
|
+
// ... ваши существующие настройки
|
|
519
|
+
include: [
|
|
520
|
+
// ... ваши существующие включения
|
|
521
|
+
".intlayer/**/*.ts", // Включить автоматически сгенерированные типы
|
|
522
|
+
],
|
|
523
|
+
}
|
|
524
|
+
```
|
|
525
|
+
|
|
526
|
+
---
|
|
527
|
+
|
|
528
|
+
## Конфигурация Git
|
|
529
|
+
|
|
530
|
+
Рекомендуется игнорировать файлы, сгенерированные Intlayer. Это позволит избежать их коммита в ваш Git-репозиторий.
|
|
531
|
+
|
|
532
|
+
Для этого вы можете добавить следующие инструкции в ваш файл `.gitignore`:
|
|
533
|
+
|
|
534
|
+
```plaintext fileName=".gitignore"
|
|
535
|
+
# Игнорировать файлы, сгенерированные Intlayer
|
|
536
|
+
.intlayer
|
|
537
|
+
```
|
|
538
|
+
|
|
539
|
+
---
|
|
540
|
+
|
|
541
|
+
## Расширение VS Code
|
|
542
|
+
|
|
543
|
+
Для улучшения вашего опыта разработки с Intlayer вы можете установить официальное **расширение Intlayer для VS Code**.
|
|
544
|
+
|
|
545
|
+
[Установить из VS Code Marketplace](https://marketplace.visualstudio.com/items?itemName=intlayer.intlayer-vs-code-extension)
|
|
546
|
+
|
|
547
|
+
Это расширение предоставляет:
|
|
548
|
+
|
|
549
|
+
- **Автозаполнение** ключей переводов.
|
|
550
|
+
- **Обнаружение ошибок в реальном времени** для отсутствующих переводов.
|
|
551
|
+
- **Встроенный просмотр** переведённого контента.
|
|
552
|
+
- **Быстрые действия** для простого создания и обновления переводов.
|
|
553
|
+
|
|
554
|
+
Для получения дополнительной информации о том, как использовать расширение, обратитесь к [документации расширения Intlayer для VS Code](https://intlayer.org/doc/vs-code-extension).
|
|
555
|
+
|
|
556
|
+
---
|
|
557
|
+
|
|
558
|
+
## Продвинутые возможности
|
|
559
|
+
|
|
560
|
+
Чтобы пойти дальше, вы можете реализовать [визуальный редактор](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/intlayer_visual_editor.md) или вынести ваш контент, используя [CMS](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/intlayer_CMS.md).
|
|
561
|
+
|
|
562
|
+
---
|
|
563
|
+
|
|
564
|
+
## Ссылки на документацию
|
|
565
|
+
|
|
566
|
+
- [Документация Intlayer](https://intlayer.org)
|
|
567
|
+
- [Документация React Router v7](https://reactrouter.com/)
|
|
568
|
+
- [Документация React Router fs-routes](https://reactrouter.com/how-to/file-route-conventions)
|
|
569
|
+
- [Хук useIntlayer](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/packages/react-intlayer/useIntlayer.md)
|
|
570
|
+
- [Хук useLocale](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/packages/react-intlayer/useLocale.md)
|
|
571
|
+
- [Объявление контента](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/dictionary/get_started.md)
|
|
572
|
+
- [Конфигурация](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ru/configuration.md)
|
|
573
|
+
|
|
574
|
+
Это подробное руководство содержит все необходимое для интеграции Intlayer с React Router v7 с использованием маршрутизации на основе файловой системы для полностью интернационализированного приложения с маршрутизацией, учитывающей локаль, и поддержкой TypeScript.
|