@intlayer/docs 7.0.7 → 7.0.8-canary.0
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/blog/ar/i18n_using_next-i18next.md +1068 -0
- package/blog/ar/i18n_using_next-intl.md +768 -0
- package/blog/ar/intlayer_with_react-intl.md +0 -4
- package/blog/ar/next-i18next_vs_next-intl_vs_intlayer.md +5 -4
- package/blog/de/i18n_using_next-i18next.md +1107 -0
- package/blog/de/i18n_using_next-intl.md +760 -0
- package/blog/de/intlayer_with_react-intl.md +0 -4
- package/blog/de/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/en/i18n_using_next-i18next.md +1073 -0
- package/blog/en/i18n_using_next-intl.md +757 -0
- package/blog/en/intlayer_with_i18next.md +71 -8
- package/blog/en/intlayer_with_next-i18next.md +71 -8
- package/blog/en/intlayer_with_next-intl.md +71 -8
- package/blog/en/intlayer_with_react-i18next.md +69 -8
- package/blog/en/intlayer_with_react-intl.md +68 -9
- package/blog/en/intlayer_with_vue-i18n.md +68 -7
- package/blog/en/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
- package/blog/en/vue-i18n_vs_intlayer.md +2 -0
- package/blog/en-GB/i18n_using_next-i18next.md +1074 -0
- package/blog/en-GB/i18n_using_next-intl.md +757 -0
- package/blog/en-GB/intlayer_with_i18next.md +15 -6
- package/blog/en-GB/intlayer_with_next-i18next.md +16 -6
- package/blog/en-GB/intlayer_with_next-intl.md +16 -6
- package/blog/en-GB/intlayer_with_react-i18next.md +16 -7
- package/blog/en-GB/intlayer_with_react-intl.md +14 -9
- package/blog/en-GB/intlayer_with_vue-i18n.md +16 -7
- package/blog/en-GB/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/en-GB/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
- package/blog/en-GB/vue-i18n_vs_intlayer.md +2 -0
- package/blog/es/i18n_using_next-i18next.md +1066 -0
- package/blog/es/i18n_using_next-intl.md +757 -0
- package/blog/es/intlayer_with_react-intl.md +0 -4
- package/blog/es/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/fr/i18n_using_next-i18next.md +1078 -0
- package/blog/fr/i18n_using_next-intl.md +759 -0
- package/blog/fr/intlayer_with_react-intl.md +0 -4
- package/blog/fr/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/hi/i18n_using_next-i18next.md +1068 -0
- package/blog/hi/i18n_using_next-intl.md +758 -0
- package/blog/hi/intlayer_with_react-intl.md +0 -4
- package/blog/hi/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/id/i18n_using_next-i18next.md +1078 -0
- package/blog/id/i18n_using_next-intl.md +757 -0
- package/blog/id/index.md +69 -0
- package/blog/id/internationalization_and_SEO.md +364 -0
- package/blog/id/intlayer_with_react-intl.md +0 -4
- package/blog/id/list_i18n_technologies/CMS/drupal.md +143 -0
- package/blog/id/list_i18n_technologies/CMS/wix.md +167 -0
- package/blog/id/list_i18n_technologies/CMS/wordpress.md +188 -0
- package/blog/id/list_i18n_technologies/frameworks/angular.md +125 -0
- package/blog/id/list_i18n_technologies/frameworks/flutter.md +150 -0
- package/blog/id/list_i18n_technologies/frameworks/react-native.md +217 -0
- package/blog/id/list_i18n_technologies/frameworks/react.md +155 -0
- package/blog/id/list_i18n_technologies/frameworks/svelte.md +131 -0
- package/blog/id/list_i18n_technologies/frameworks/vue.md +130 -0
- package/blog/id/next-i18next_vs_next-intl_vs_intlayer.md +1500 -0
- package/blog/id/nextjs-multilingual-seo-comparison.md +361 -0
- package/blog/id/rag_powered_documentation_assistant.md +288 -0
- package/blog/id/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
- package/blog/id/vue-i18n_vs_intlayer.md +278 -0
- package/blog/id/what_is_internationalization.md +166 -0
- package/blog/it/i18n_using_next-i18next.md +1078 -0
- package/blog/it/i18n_using_next-intl.md +758 -0
- package/blog/it/intlayer_with_react-intl.md +0 -4
- package/blog/it/react-i18next_vs_react-intl_vs_intlayer.md +4 -0
- package/blog/it/vue-i18n_vs_intlayer.md +2 -0
- package/blog/ja/i18n_using_next-i18next.md +1078 -0
- package/blog/ja/i18n_using_next-intl.md +758 -0
- package/blog/ja/intlayer_with_react-intl.md +0 -4
- package/blog/ja/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/ko/i18n_using_next-i18next.md +1075 -0
- package/blog/ko/i18n_using_next-intl.md +759 -0
- package/blog/ko/intlayer_with_react-intl.md +0 -4
- package/blog/ko/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/pl/i18n_using_next-i18next.md +1078 -0
- package/blog/pl/i18n_using_next-intl.md +758 -0
- package/blog/pl/index.md +69 -0
- package/blog/pl/internationalization_and_SEO.md +363 -0
- package/blog/pl/intlayer_with_react-intl.md +0 -4
- package/blog/pl/list_i18n_technologies/CMS/drupal.md +143 -0
- package/blog/pl/list_i18n_technologies/CMS/wix.md +167 -0
- package/blog/pl/list_i18n_technologies/CMS/wordpress.md +196 -0
- package/blog/pl/list_i18n_technologies/frameworks/angular.md +125 -0
- package/blog/pl/list_i18n_technologies/frameworks/flutter.md +151 -0
- package/blog/pl/list_i18n_technologies/frameworks/react-native.md +217 -0
- package/blog/pl/list_i18n_technologies/frameworks/react.md +155 -0
- package/blog/pl/list_i18n_technologies/frameworks/svelte.md +131 -0
- package/blog/pl/list_i18n_technologies/frameworks/vue.md +130 -0
- package/blog/pl/next-i18next_vs_next-intl_vs_intlayer.md +1501 -0
- package/blog/pl/nextjs-multilingual-seo-comparison.md +362 -0
- package/blog/pl/rag_powered_documentation_assistant.md +288 -0
- package/blog/pl/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
- package/blog/pl/vue-i18n_vs_intlayer.md +278 -0
- package/blog/pl/what_is_internationalization.md +167 -0
- package/blog/pt/i18n_using_next-i18next.md +1067 -0
- package/blog/pt/i18n_using_next-intl.md +760 -0
- package/blog/pt/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/ru/i18n_using_next-i18next.md +1106 -0
- package/blog/ru/i18n_using_next-intl.md +759 -0
- package/blog/ru/intlayer_with_react-intl.md +0 -4
- package/blog/ru/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/tr/i18n_using_next-i18next.md +1078 -0
- package/blog/tr/i18n_using_next-intl.md +760 -0
- package/blog/tr/intlayer_with_react-intl.md +0 -4
- package/blog/tr/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/tr/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
- package/blog/tr/vue-i18n_vs_intlayer.md +2 -0
- package/blog/vi/i18n_using_next-i18next.md +1080 -0
- package/blog/vi/i18n_using_next-intl.md +758 -0
- package/blog/vi/index.md +69 -0
- package/blog/vi/internationalization_and_SEO.md +363 -0
- package/blog/vi/intlayer_with_react-intl.md +0 -4
- package/blog/vi/list_i18n_technologies/CMS/drupal.md +143 -0
- package/blog/vi/list_i18n_technologies/CMS/wix.md +167 -0
- package/blog/vi/list_i18n_technologies/CMS/wordpress.md +188 -0
- package/blog/vi/list_i18n_technologies/frameworks/angular.md +125 -0
- package/blog/vi/list_i18n_technologies/frameworks/flutter.md +150 -0
- package/blog/vi/list_i18n_technologies/frameworks/react-native.md +217 -0
- package/blog/vi/list_i18n_technologies/frameworks/react.md +155 -0
- package/blog/vi/list_i18n_technologies/frameworks/svelte.md +131 -0
- package/blog/vi/list_i18n_technologies/frameworks/vue.md +130 -0
- package/blog/vi/next-i18next_vs_next-intl_vs_intlayer.md +1520 -0
- package/blog/vi/nextjs-multilingual-seo-comparison.md +362 -0
- package/blog/vi/rag_powered_documentation_assistant.md +288 -0
- package/blog/vi/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
- package/blog/vi/vue-i18n_vs_intlayer.md +278 -0
- package/blog/vi/what_is_internationalization.md +168 -0
- package/blog/zh/i18n_using_next-i18next.md +1105 -0
- package/blog/zh/i18n_using_next-intl.md +758 -0
- package/blog/zh/intlayer_with_react-intl.md +0 -4
- package/blog/zh/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
- package/blog/zh/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
- package/dist/cjs/common.cjs +0 -4
- package/dist/cjs/common.cjs.map +1 -1
- package/dist/cjs/generated/blog.entry.cjs +38 -6
- package/dist/cjs/generated/blog.entry.cjs.map +1 -1
- package/dist/cjs/generated/docs.entry.cjs +0 -6
- package/dist/cjs/generated/docs.entry.cjs.map +1 -1
- package/dist/cjs/generated/frequentQuestions.entry.cjs +0 -6
- package/dist/cjs/generated/frequentQuestions.entry.cjs.map +1 -1
- package/dist/cjs/generated/legal.entry.cjs +0 -6
- package/dist/cjs/generated/legal.entry.cjs.map +1 -1
- package/dist/esm/generated/blog.entry.mjs +38 -0
- package/dist/esm/generated/blog.entry.mjs.map +1 -1
- package/dist/types/generated/blog.entry.d.ts +2 -0
- package/dist/types/generated/blog.entry.d.ts.map +1 -1
- package/docs/ar/component_i18n.md +1 -1
- package/docs/ar/configuration.md +6 -0
- package/docs/ar/intlayer_cli.md +8 -3
- package/docs/ar/intlayer_with_next-i18next.md +619 -0
- package/docs/ar/intlayer_with_next-intl.md +446 -0
- package/docs/ar/intlayer_with_nextjs_16.md +21 -0
- package/docs/ar/intlayer_with_tanstack.md +4 -0
- package/docs/ar/intlayer_with_vite+react.md +4 -0
- package/docs/de/component_i18n.md +1 -1
- package/docs/de/configuration.md +6 -0
- package/docs/de/intlayer_cli.md +8 -3
- package/docs/de/intlayer_with_next-i18next.md +627 -0
- package/docs/de/intlayer_with_next-intl.md +451 -0
- package/docs/de/intlayer_with_nextjs_16.md +21 -0
- package/docs/de/intlayer_with_tanstack.md +4 -0
- package/docs/de/intlayer_with_vite+react.md +4 -0
- package/docs/en/component_i18n.md +1 -1
- package/docs/en/intlayer_cli.md +8 -1
- package/docs/en/intlayer_with_astro.md +10 -2
- package/docs/en/intlayer_with_create_react_app.md +8 -0
- package/docs/en/intlayer_with_lynx+react.md +8 -0
- package/docs/en/intlayer_with_nestjs.md +10 -0
- package/docs/en/intlayer_with_nextjs_14.md +10 -2
- package/docs/en/intlayer_with_nextjs_15.md +21 -4
- package/docs/en/intlayer_with_nextjs_16.md +17 -0
- package/docs/en/intlayer_with_nuxt.md +8 -0
- package/docs/en/intlayer_with_react_native+expo.md +10 -2
- package/docs/en/intlayer_with_react_router_v7.md +8 -0
- package/docs/en/intlayer_with_tanstack.md +10 -0
- package/docs/en/intlayer_with_vite+preact.md +10 -2
- package/docs/en/intlayer_with_vite+react.md +21 -4
- package/docs/en/intlayer_with_vite+vue.md +10 -2
- package/docs/en-GB/component_i18n.md +1 -1
- package/docs/en-GB/configuration.md +6 -0
- package/docs/en-GB/intlayer_cli.md +8 -3
- package/docs/en-GB/intlayer_with_angular.md +4 -4
- package/docs/en-GB/intlayer_with_express.md +4 -4
- package/docs/en-GB/intlayer_with_lynx+react.md +12 -12
- package/{blog/en/_intlayer_with_next-i18next.md → docs/en-GB/intlayer_with_next-i18next.md} +241 -42
- package/{blog/en/_intlayer_with_next-intl.md → docs/en-GB/intlayer_with_next-intl.md} +144 -29
- package/docs/en-GB/intlayer_with_nextjs_16.md +21 -0
- package/docs/en-GB/intlayer_with_tanstack.md +5 -1
- package/docs/en-GB/intlayer_with_vite+react.md +4 -0
- package/docs/en-GB/packages/next-intlayer/t.md +2 -2
- package/docs/es/component_i18n.md +1 -1
- package/docs/es/configuration.md +6 -0
- package/docs/es/intlayer_cli.md +8 -3
- package/docs/es/intlayer_with_next-i18next.md +628 -0
- package/docs/es/intlayer_with_next-intl.md +446 -0
- package/docs/es/intlayer_with_nextjs_16.md +21 -0
- package/docs/es/intlayer_with_tanstack.md +4 -0
- package/docs/es/intlayer_with_vite+react.md +4 -0
- package/docs/fr/configuration.md +6 -0
- package/docs/fr/intlayer_cli.md +8 -3
- package/docs/fr/intlayer_with_next-i18next.md +628 -0
- package/docs/fr/intlayer_with_next-intl.md +446 -0
- package/docs/fr/intlayer_with_nextjs_16.md +23 -2
- package/docs/fr/intlayer_with_tanstack.md +4 -0
- package/docs/fr/intlayer_with_vite+react.md +4 -0
- package/docs/hi/component_i18n.md +1 -1
- package/docs/hi/configuration.md +6 -0
- package/docs/hi/intlayer_cli.md +8 -0
- package/docs/hi/intlayer_with_next-i18next.md +628 -0
- package/docs/hi/intlayer_with_next-intl.md +446 -0
- package/docs/hi/intlayer_with_nextjs_16.md +21 -0
- package/docs/hi/intlayer_with_tanstack.md +4 -0
- package/docs/hi/intlayer_with_vite+react.md +4 -0
- package/docs/id/CI_CD.md +198 -0
- package/docs/id/autoFill.md +284 -0
- package/docs/id/component_i18n.md +186 -0
- package/docs/id/configuration.md +710 -0
- package/docs/id/dictionary/condition.md +231 -0
- package/docs/id/dictionary/content_file.md +1092 -0
- package/docs/id/dictionary/enumeration.md +245 -0
- package/docs/id/dictionary/file.md +237 -0
- package/docs/id/dictionary/function_fetching.md +214 -0
- package/docs/id/dictionary/gender.md +273 -0
- package/docs/id/dictionary/insertion.md +192 -0
- package/docs/id/dictionary/markdown.md +381 -0
- package/docs/id/dictionary/nesting.md +273 -0
- package/docs/id/dictionary/translation.md +310 -0
- package/docs/id/formatters.md +596 -0
- package/docs/id/how_works_intlayer.md +256 -0
- package/docs/id/index.md +176 -0
- package/docs/id/interest_of_intlayer.md +293 -0
- package/docs/id/intlayer_CMS.md +549 -0
- package/docs/id/intlayer_cli.md +850 -0
- package/docs/id/intlayer_visual_editor.md +288 -0
- package/docs/id/intlayer_with_angular.md +694 -0
- package/docs/id/intlayer_with_astro.md +252 -0
- package/docs/id/intlayer_with_create_react_app.md +1233 -0
- package/docs/id/intlayer_with_express.md +411 -0
- package/docs/id/intlayer_with_lynx+react.md +518 -0
- package/docs/id/intlayer_with_nestjs.md +272 -0
- package/docs/id/intlayer_with_next-i18next.md +628 -0
- package/docs/id/intlayer_with_next-intl.md +446 -0
- package/docs/id/intlayer_with_nextjs_14.md +1617 -0
- package/docs/id/intlayer_with_nextjs_15.md +1698 -0
- package/docs/id/intlayer_with_nextjs_16.md +21 -0
- package/docs/id/intlayer_with_nextjs_page_router.md +1478 -0
- package/docs/id/intlayer_with_nuxt.md +808 -0
- package/docs/id/intlayer_with_react_native+expo.md +699 -0
- package/docs/id/intlayer_with_react_router_v7.md +496 -0
- package/docs/id/intlayer_with_tanstack.md +564 -0
- package/docs/id/intlayer_with_vite+preact.md +1737 -0
- package/docs/id/intlayer_with_vite+react.md +1413 -0
- package/docs/id/intlayer_with_vite+solid.md +289 -0
- package/docs/id/intlayer_with_vite+svelte.md +289 -0
- package/docs/id/intlayer_with_vite+vue.md +1088 -0
- package/docs/id/introduction.md +218 -0
- package/docs/id/locale_mapper.md +242 -0
- package/docs/id/mcp_server.md +211 -0
- package/docs/id/packages/express-intlayer/t.md +458 -0
- package/docs/id/packages/intlayer/getConfiguration.md +145 -0
- package/docs/id/packages/intlayer/getEnumeration.md +159 -0
- package/docs/id/packages/intlayer/getHTMLTextDir.md +122 -0
- package/docs/id/packages/intlayer/getLocaleLang.md +81 -0
- package/docs/id/packages/intlayer/getLocaleName.md +119 -0
- package/docs/id/packages/intlayer/getLocalizedUrl.md +309 -0
- package/docs/id/packages/intlayer/getMultilingualUrls.md +223 -0
- package/docs/id/packages/intlayer/getPathWithoutLocale.md +75 -0
- package/docs/id/packages/intlayer/getTranslation.md +190 -0
- package/docs/id/packages/intlayer/getTranslationContent.md +188 -0
- package/docs/id/packages/next-intlayer/t.md +352 -0
- package/docs/id/packages/next-intlayer/useDictionary.md +271 -0
- package/docs/id/packages/next-intlayer/useIntlayer.md +264 -0
- package/docs/id/packages/next-intlayer/useLocale.md +166 -0
- package/docs/id/packages/react-intlayer/t.md +303 -0
- package/docs/id/packages/react-intlayer/useDictionary.md +287 -0
- package/docs/id/packages/react-intlayer/useI18n.md +267 -0
- package/docs/id/packages/react-intlayer/useIntlayer.md +254 -0
- package/docs/id/packages/react-intlayer/useLocale.md +210 -0
- package/docs/id/per_locale_file.md +323 -0
- package/docs/id/readme.md +261 -0
- package/docs/id/releases/v6.md +305 -0
- package/docs/id/roadmap.md +362 -0
- package/docs/id/testing.md +202 -0
- package/docs/id/vs_code_extension.md +126 -0
- package/docs/it/component_i18n.md +1 -1
- package/docs/it/configuration.md +6 -0
- package/docs/it/intlayer_cli.md +8 -3
- package/docs/it/intlayer_with_next-i18next.md +628 -0
- package/docs/it/intlayer_with_next-intl.md +446 -0
- package/docs/it/intlayer_with_nextjs_16.md +21 -0
- package/docs/it/intlayer_with_tanstack.md +4 -0
- package/docs/it/intlayer_with_vite+react.md +4 -0
- package/docs/ja/component_i18n.md +1 -1
- package/docs/ja/configuration.md +6 -0
- package/docs/ja/intlayer_cli.md +8 -3
- package/docs/ja/intlayer_with_next-i18next.md +627 -0
- package/docs/ja/intlayer_with_next-intl.md +446 -0
- package/docs/ja/intlayer_with_nextjs_16.md +21 -0
- package/docs/ja/intlayer_with_tanstack.md +4 -0
- package/docs/ja/intlayer_with_vite+react.md +4 -0
- package/docs/ko/configuration.md +6 -0
- package/docs/ko/intlayer_cli.md +8 -3
- package/docs/ko/intlayer_with_next-i18next.md +627 -0
- package/docs/ko/intlayer_with_next-intl.md +446 -0
- package/docs/ko/intlayer_with_nextjs_16.md +21 -0
- package/docs/ko/intlayer_with_tanstack.md +4 -0
- package/docs/ko/intlayer_with_vite+react.md +4 -0
- package/docs/pl/CI_CD.md +198 -0
- package/docs/pl/autoFill.md +284 -0
- package/docs/pl/component_i18n.md +186 -0
- package/docs/pl/configuration.md +710 -0
- package/docs/pl/dictionary/condition.md +232 -0
- package/docs/pl/dictionary/content_file.md +1130 -0
- package/docs/pl/dictionary/enumeration.md +245 -0
- package/docs/pl/dictionary/file.md +234 -0
- package/docs/pl/dictionary/function_fetching.md +214 -0
- package/docs/pl/dictionary/gender.md +276 -0
- package/docs/pl/dictionary/insertion.md +188 -0
- package/docs/pl/dictionary/markdown.md +408 -0
- package/docs/pl/dictionary/nesting.md +273 -0
- package/docs/pl/dictionary/translation.md +310 -0
- package/docs/pl/formatters.md +596 -0
- package/docs/pl/how_works_intlayer.md +256 -0
- package/docs/pl/index.md +176 -0
- package/docs/pl/interest_of_intlayer.md +291 -0
- package/docs/pl/intlayer_CMS.md +549 -0
- package/docs/pl/intlayer_cli.md +857 -0
- package/docs/pl/intlayer_visual_editor.md +288 -0
- package/docs/pl/intlayer_with_angular.md +690 -0
- package/docs/pl/intlayer_with_astro.md +280 -0
- package/docs/pl/intlayer_with_create_react_app.md +1235 -0
- package/docs/pl/intlayer_with_express.md +411 -0
- package/docs/pl/intlayer_with_lynx+react.md +518 -0
- package/docs/pl/intlayer_with_nestjs.md +272 -0
- package/docs/pl/intlayer_with_next-i18next.md +628 -0
- package/docs/pl/intlayer_with_next-intl.md +446 -0
- package/docs/pl/intlayer_with_nextjs_14.md +1594 -0
- package/docs/pl/intlayer_with_nextjs_15.md +1701 -0
- package/docs/pl/intlayer_with_nextjs_16.md +21 -0
- package/docs/pl/intlayer_with_nextjs_page_router.md +1513 -0
- package/docs/pl/intlayer_with_nuxt.md +885 -0
- package/docs/pl/intlayer_with_react_native+expo.md +698 -0
- package/docs/pl/intlayer_with_react_router_v7.md +503 -0
- package/docs/pl/intlayer_with_tanstack.md +562 -0
- package/docs/pl/intlayer_with_vite+preact.md +1736 -0
- package/docs/pl/intlayer_with_vite+react.md +1438 -0
- package/docs/pl/intlayer_with_vite+solid.md +290 -0
- package/docs/pl/intlayer_with_vite+svelte.md +289 -0
- package/docs/pl/intlayer_with_vite+vue.md +1116 -0
- package/docs/pl/introduction.md +209 -0
- package/docs/pl/locale_mapper.md +242 -0
- package/docs/pl/mcp_server.md +211 -0
- package/docs/pl/packages/express-intlayer/t.md +458 -0
- package/docs/pl/packages/intlayer/getConfiguration.md +146 -0
- package/docs/pl/packages/intlayer/getEnumeration.md +160 -0
- package/docs/pl/packages/intlayer/getHTMLTextDir.md +121 -0
- package/docs/pl/packages/intlayer/getLocaleLang.md +81 -0
- package/docs/pl/packages/intlayer/getLocaleName.md +118 -0
- package/docs/pl/packages/intlayer/getLocalizedUrl.md +300 -0
- package/docs/pl/packages/intlayer/getMultilingualUrls.md +221 -0
- package/docs/pl/packages/intlayer/getPathWithoutLocale.md +75 -0
- package/docs/pl/packages/intlayer/getTranslation.md +190 -0
- package/docs/pl/packages/intlayer/getTranslationContent.md +189 -0
- package/docs/pl/packages/next-intlayer/t.md +353 -0
- package/docs/pl/packages/next-intlayer/useDictionary.md +270 -0
- package/docs/pl/packages/next-intlayer/useIntlayer.md +263 -0
- package/docs/pl/packages/next-intlayer/useLocale.md +166 -0
- package/docs/pl/packages/react-intlayer/t.md +303 -0
- package/docs/pl/packages/react-intlayer/useDictionary.md +289 -0
- package/docs/pl/packages/react-intlayer/useI18n.md +249 -0
- package/docs/pl/packages/react-intlayer/useIntlayer.md +256 -0
- package/docs/pl/packages/react-intlayer/useLocale.md +210 -0
- package/docs/pl/per_locale_file.md +321 -0
- package/docs/pl/readme.md +261 -0
- package/docs/pl/releases/v6.md +305 -0
- package/docs/pl/roadmap.md +362 -0
- package/docs/pl/testing.md +202 -0
- package/docs/pl/vs_code_extension.md +126 -0
- package/docs/pt/component_i18n.md +1 -1
- package/docs/pt/configuration.md +6 -0
- package/docs/pt/intlayer_cli.md +8 -3
- package/docs/pt/intlayer_with_next-i18next.md +627 -0
- package/docs/pt/intlayer_with_next-intl.md +446 -0
- package/docs/pt/intlayer_with_nextjs_16.md +21 -0
- package/docs/pt/intlayer_with_tanstack.md +4 -0
- package/docs/pt/intlayer_with_vite+react.md +4 -0
- package/docs/ru/component_i18n.md +1 -1
- package/docs/ru/configuration.md +6 -0
- package/docs/ru/intlayer_cli.md +301 -22
- package/docs/ru/intlayer_with_next-i18next.md +629 -0
- package/docs/ru/intlayer_with_next-intl.md +448 -0
- package/docs/ru/intlayer_with_nextjs_16.md +21 -0
- package/docs/ru/intlayer_with_tanstack.md +4 -0
- package/docs/ru/intlayer_with_vite+react.md +4 -0
- package/docs/tr/component_i18n.md +1 -1
- package/docs/tr/configuration.md +6 -0
- package/docs/tr/intlayer_cli.md +8 -0
- package/docs/tr/intlayer_with_next-i18next.md +627 -0
- package/docs/tr/intlayer_with_next-intl.md +446 -0
- package/docs/tr/intlayer_with_nextjs_16.md +21 -0
- package/docs/tr/intlayer_with_tanstack.md +4 -0
- package/docs/tr/intlayer_with_vite+react.md +4 -0
- package/docs/vi/CI_CD.md +198 -0
- package/docs/vi/autoFill.md +284 -0
- package/docs/vi/component_i18n.md +186 -0
- package/docs/vi/configuration.md +710 -0
- package/docs/vi/dictionary/condition.md +237 -0
- package/docs/vi/dictionary/content_file.md +1115 -0
- package/docs/vi/dictionary/enumeration.md +255 -0
- package/docs/vi/dictionary/file.md +234 -0
- package/docs/vi/dictionary/function_fetching.md +212 -0
- package/docs/vi/dictionary/gender.md +275 -0
- package/docs/vi/dictionary/insertion.md +191 -0
- package/docs/vi/dictionary/markdown.md +381 -0
- package/docs/vi/dictionary/nesting.md +273 -0
- package/docs/vi/dictionary/translation.md +309 -0
- package/docs/vi/formatters.md +595 -0
- package/docs/vi/how_works_intlayer.md +256 -0
- package/docs/vi/index.md +174 -0
- package/docs/vi/interest_of_intlayer.md +292 -0
- package/docs/vi/intlayer_CMS.md +549 -0
- package/docs/vi/intlayer_cli.md +850 -0
- package/docs/vi/intlayer_visual_editor.md +288 -0
- package/docs/vi/intlayer_with_angular.md +692 -0
- package/docs/vi/intlayer_with_astro.md +252 -0
- package/docs/vi/intlayer_with_create_react_app.md +1230 -0
- package/docs/vi/intlayer_with_express.md +409 -0
- package/docs/vi/intlayer_with_lynx+react.md +520 -0
- package/docs/vi/intlayer_with_nestjs.md +272 -0
- package/docs/vi/intlayer_with_next-i18next.md +628 -0
- package/docs/vi/intlayer_with_next-intl.md +446 -0
- package/docs/vi/intlayer_with_nextjs_14.md +1584 -0
- package/docs/vi/intlayer_with_nextjs_15.md +1738 -0
- package/docs/vi/intlayer_with_nextjs_16.md +21 -0
- package/docs/vi/intlayer_with_nextjs_page_router.md +1504 -0
- package/docs/vi/intlayer_with_nuxt.md +821 -0
- package/docs/vi/intlayer_with_react_native+expo.md +700 -0
- package/docs/vi/intlayer_with_react_router_v7.md +498 -0
- package/docs/vi/intlayer_with_tanstack.md +562 -0
- package/docs/vi/intlayer_with_vite+preact.md +1722 -0
- package/docs/vi/intlayer_with_vite+react.md +1407 -0
- package/docs/vi/intlayer_with_vite+solid.md +287 -0
- package/docs/vi/intlayer_with_vite+svelte.md +289 -0
- package/docs/vi/intlayer_with_vite+vue.md +1071 -0
- package/docs/vi/introduction.md +215 -0
- package/docs/vi/locale_mapper.md +242 -0
- package/docs/vi/mcp_server.md +211 -0
- package/docs/vi/packages/express-intlayer/t.md +457 -0
- package/docs/vi/packages/intlayer/getConfiguration.md +145 -0
- package/docs/vi/packages/intlayer/getEnumeration.md +162 -0
- package/docs/vi/packages/intlayer/getHTMLTextDir.md +121 -0
- package/docs/vi/packages/intlayer/getLocaleLang.md +81 -0
- package/docs/vi/packages/intlayer/getLocaleName.md +129 -0
- package/docs/vi/packages/intlayer/getLocalizedUrl.md +309 -0
- package/docs/vi/packages/intlayer/getMultilingualUrls.md +221 -0
- package/docs/vi/packages/intlayer/getPathWithoutLocale.md +75 -0
- package/docs/vi/packages/intlayer/getTranslation.md +201 -0
- package/docs/vi/packages/intlayer/getTranslationContent.md +188 -0
- package/docs/vi/packages/next-intlayer/t.md +352 -0
- package/docs/vi/packages/next-intlayer/useDictionary.md +273 -0
- package/docs/vi/packages/next-intlayer/useIntlayer.md +264 -0
- package/docs/vi/packages/next-intlayer/useLocale.md +166 -0
- package/docs/vi/packages/react-intlayer/t.md +304 -0
- package/docs/vi/packages/react-intlayer/useDictionary.md +288 -0
- package/docs/vi/packages/react-intlayer/useI18n.md +295 -0
- package/docs/vi/packages/react-intlayer/useIntlayer.md +256 -0
- package/docs/vi/packages/react-intlayer/useLocale.md +210 -0
- package/docs/vi/per_locale_file.md +326 -0
- package/docs/vi/readme.md +261 -0
- package/docs/vi/releases/v6.md +305 -0
- package/docs/vi/roadmap.md +346 -0
- package/docs/vi/testing.md +202 -0
- package/docs/vi/vs_code_extension.md +126 -0
- package/docs/zh/configuration.md +6 -0
- package/docs/zh/intlayer_cli.md +8 -3
- package/docs/zh/intlayer_with_next-i18next.md +628 -0
- package/docs/zh/intlayer_with_next-intl.md +448 -0
- package/docs/zh/intlayer_with_nextjs_16.md +21 -0
- package/docs/zh/intlayer_with_tanstack.md +4 -0
- package/docs/zh/intlayer_with_vite+react.md +4 -0
- package/frequent_questions/ar/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/ar/array_as_content_declaration.md +1 -2
- package/frequent_questions/ar/build_dictionaries.md +1 -2
- package/frequent_questions/ar/build_error_CI_CD.md +1 -2
- package/frequent_questions/ar/bun_set_up.md +1 -2
- package/frequent_questions/ar/customized_locale_list.md +1 -2
- package/frequent_questions/ar/domain_routing.md +1 -2
- package/frequent_questions/ar/esbuild_error.md +1 -2
- package/frequent_questions/ar/get_locale_cookie.md +1 -2
- package/frequent_questions/ar/intlayer_command_undefined.md +1 -2
- package/frequent_questions/ar/locale_incorect_in_url.md +1 -2
- package/frequent_questions/ar/static_rendering.md +1 -3
- package/frequent_questions/ar/translated_path_url.md +1 -2
- package/frequent_questions/ar/unknown_command.md +1 -2
- package/frequent_questions/de/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/de/array_as_content_declaration.md +1 -2
- package/frequent_questions/de/build_dictionaries.md +1 -2
- package/frequent_questions/de/build_error_CI_CD.md +1 -2
- package/frequent_questions/de/bun_set_up.md +1 -2
- package/frequent_questions/de/customized_locale_list.md +1 -2
- package/frequent_questions/de/domain_routing.md +1 -2
- package/frequent_questions/de/esbuild_error.md +1 -2
- package/frequent_questions/de/get_locale_cookie.md +1 -2
- package/frequent_questions/de/intlayer_command_undefined.md +1 -2
- package/frequent_questions/de/locale_incorect_in_url.md +1 -2
- package/frequent_questions/de/static_rendering.md +1 -3
- package/frequent_questions/de/translated_path_url.md +1 -2
- package/frequent_questions/de/unknown_command.md +1 -2
- package/frequent_questions/en/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/en/array_as_content_declaration.md +1 -2
- package/frequent_questions/en/build_dictionaries.md +1 -2
- package/frequent_questions/en/build_error_CI_CD.md +1 -2
- package/frequent_questions/en/bun_set_up.md +1 -2
- package/frequent_questions/en/customized_locale_list.md +1 -2
- package/frequent_questions/en/domain_routing.md +1 -2
- package/frequent_questions/en/esbuild_error.md +1 -2
- package/frequent_questions/en/get_locale_cookie.md +1 -2
- package/frequent_questions/en/intlayer_command_undefined.md +1 -2
- package/frequent_questions/en/locale_incorect_in_url.md +1 -2
- package/frequent_questions/en/static_rendering.md +1 -3
- package/frequent_questions/en/translated_path_url.md +1 -2
- package/frequent_questions/en/unknown_command.md +1 -2
- package/frequent_questions/en-GB/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/en-GB/array_as_content_declaration.md +1 -2
- package/frequent_questions/en-GB/build_dictionaries.md +1 -2
- package/frequent_questions/en-GB/build_error_CI_CD.md +1 -2
- package/frequent_questions/en-GB/bun_set_up.md +1 -2
- package/frequent_questions/en-GB/customized_locale_list.md +1 -2
- package/frequent_questions/en-GB/domain_routing.md +1 -2
- package/frequent_questions/en-GB/esbuild_error.md +1 -2
- package/frequent_questions/en-GB/get_locale_cookie.md +1 -2
- package/frequent_questions/en-GB/intlayer_command_undefined.md +1 -2
- package/frequent_questions/en-GB/locale_incorect_in_url.md +1 -2
- package/frequent_questions/en-GB/static_rendering.md +1 -3
- package/frequent_questions/en-GB/translated_path_url.md +1 -2
- package/frequent_questions/en-GB/unknown_command.md +1 -2
- package/frequent_questions/es/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/es/array_as_content_declaration.md +1 -2
- package/frequent_questions/es/build_dictionaries.md +1 -2
- package/frequent_questions/es/build_error_CI_CD.md +1 -2
- package/frequent_questions/es/bun_set_up.md +1 -2
- package/frequent_questions/es/customized_locale_list.md +1 -2
- package/frequent_questions/es/domain_routing.md +1 -2
- package/frequent_questions/es/esbuild_error.md +1 -2
- package/frequent_questions/es/get_locale_cookie.md +1 -2
- package/frequent_questions/es/intlayer_command_undefined.md +1 -2
- package/frequent_questions/es/locale_incorect_in_url.md +1 -2
- package/frequent_questions/es/static_rendering.md +1 -3
- package/frequent_questions/es/translated_path_url.md +1 -2
- package/frequent_questions/es/unknown_command.md +1 -2
- package/frequent_questions/fr/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/fr/array_as_content_declaration.md +1 -2
- package/frequent_questions/fr/build_dictionaries.md +1 -2
- package/frequent_questions/fr/build_error_CI_CD.md +1 -2
- package/frequent_questions/fr/bun_set_up.md +1 -2
- package/frequent_questions/fr/customized_locale_list.md +1 -2
- package/frequent_questions/fr/domain_routing.md +1 -2
- package/frequent_questions/fr/esbuild_error.md +1 -2
- package/frequent_questions/fr/get_locale_cookie.md +1 -2
- package/frequent_questions/fr/intlayer_command_undefined.md +1 -2
- package/frequent_questions/fr/locale_incorect_in_url.md +1 -2
- package/frequent_questions/fr/static_rendering.md +1 -3
- package/frequent_questions/fr/translated_path_url.md +1 -2
- package/frequent_questions/fr/unknown_command.md +1 -2
- package/frequent_questions/hi/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/hi/array_as_content_declaration.md +1 -2
- package/frequent_questions/hi/build_dictionaries.md +1 -2
- package/frequent_questions/hi/build_error_CI_CD.md +1 -2
- package/frequent_questions/hi/bun_set_up.md +1 -2
- package/frequent_questions/hi/customized_locale_list.md +1 -2
- package/frequent_questions/hi/domain_routing.md +1 -2
- package/frequent_questions/hi/esbuild_error.md +1 -2
- package/frequent_questions/hi/get_locale_cookie.md +1 -2
- package/frequent_questions/hi/intlayer_command_undefined.md +1 -2
- package/frequent_questions/hi/locale_incorect_in_url.md +1 -2
- package/frequent_questions/hi/static_rendering.md +1 -3
- package/frequent_questions/hi/translated_path_url.md +1 -2
- package/frequent_questions/hi/unknown_command.md +1 -2
- package/frequent_questions/id/SSR_Next_no_[locale].md +104 -0
- package/frequent_questions/id/array_as_content_declaration.md +71 -0
- package/frequent_questions/id/build_dictionaries.md +58 -0
- package/frequent_questions/id/build_error_CI_CD.md +74 -0
- package/frequent_questions/id/bun_set_up.md +53 -0
- package/frequent_questions/id/customized_locale_list.md +64 -0
- package/frequent_questions/id/domain_routing.md +113 -0
- package/frequent_questions/id/esbuild_error.md +29 -0
- package/frequent_questions/id/get_locale_cookie.md +142 -0
- package/frequent_questions/id/intlayer_command_undefined.md +155 -0
- package/frequent_questions/id/locale_incorect_in_url.md +73 -0
- package/frequent_questions/id/static_rendering.md +44 -0
- package/frequent_questions/id/translated_path_url.md +55 -0
- package/frequent_questions/id/unknown_command.md +97 -0
- package/frequent_questions/it/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/it/array_as_content_declaration.md +1 -2
- package/frequent_questions/it/build_dictionaries.md +1 -2
- package/frequent_questions/it/build_error_CI_CD.md +1 -2
- package/frequent_questions/it/bun_set_up.md +1 -2
- package/frequent_questions/it/customized_locale_list.md +1 -2
- package/frequent_questions/it/domain_routing.md +1 -2
- package/frequent_questions/it/esbuild_error.md +1 -2
- package/frequent_questions/it/get_locale_cookie.md +1 -2
- package/frequent_questions/it/intlayer_command_undefined.md +1 -2
- package/frequent_questions/it/locale_incorect_in_url.md +1 -2
- package/frequent_questions/it/static_rendering.md +1 -3
- package/frequent_questions/it/translated_path_url.md +1 -2
- package/frequent_questions/it/unknown_command.md +1 -2
- package/frequent_questions/ja/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/ja/array_as_content_declaration.md +1 -2
- package/frequent_questions/ja/build_dictionaries.md +1 -2
- package/frequent_questions/ja/build_error_CI_CD.md +1 -2
- package/frequent_questions/ja/bun_set_up.md +1 -2
- package/frequent_questions/ja/customized_locale_list.md +1 -2
- package/frequent_questions/ja/domain_routing.md +1 -2
- package/frequent_questions/ja/esbuild_error.md +1 -2
- package/frequent_questions/ja/get_locale_cookie.md +1 -2
- package/frequent_questions/ja/intlayer_command_undefined.md +1 -2
- package/frequent_questions/ja/locale_incorect_in_url.md +1 -2
- package/frequent_questions/ja/static_rendering.md +1 -3
- package/frequent_questions/ja/translated_path_url.md +1 -2
- package/frequent_questions/ja/unknown_command.md +1 -2
- package/frequent_questions/ko/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/ko/array_as_content_declaration.md +1 -2
- package/frequent_questions/ko/build_dictionaries.md +1 -2
- package/frequent_questions/ko/build_error_CI_CD.md +1 -2
- package/frequent_questions/ko/bun_set_up.md +1 -2
- package/frequent_questions/ko/customized_locale_list.md +1 -2
- package/frequent_questions/ko/domain_routing.md +1 -2
- package/frequent_questions/ko/esbuild_error.md +1 -2
- package/frequent_questions/ko/get_locale_cookie.md +1 -2
- package/frequent_questions/ko/intlayer_command_undefined.md +1 -2
- package/frequent_questions/ko/locale_incorect_in_url.md +1 -2
- package/frequent_questions/ko/static_rendering.md +1 -3
- package/frequent_questions/ko/translated_path_url.md +1 -2
- package/frequent_questions/ko/unknown_command.md +1 -2
- package/frequent_questions/pl/SSR_Next_no_[locale].md +104 -0
- package/frequent_questions/pl/array_as_content_declaration.md +71 -0
- package/frequent_questions/pl/build_dictionaries.md +58 -0
- package/frequent_questions/pl/build_error_CI_CD.md +74 -0
- package/frequent_questions/pl/bun_set_up.md +54 -0
- package/frequent_questions/pl/customized_locale_list.md +64 -0
- package/frequent_questions/pl/domain_routing.md +113 -0
- package/frequent_questions/pl/esbuild_error.md +29 -0
- package/frequent_questions/pl/get_locale_cookie.md +142 -0
- package/frequent_questions/pl/intlayer_command_undefined.md +155 -0
- package/frequent_questions/pl/locale_incorect_in_url.md +73 -0
- package/frequent_questions/pl/static_rendering.md +44 -0
- package/frequent_questions/pl/translated_path_url.md +55 -0
- package/frequent_questions/pl/unknown_command.md +97 -0
- package/frequent_questions/pt/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/pt/array_as_content_declaration.md +1 -2
- package/frequent_questions/pt/build_dictionaries.md +1 -2
- package/frequent_questions/pt/build_error_CI_CD.md +1 -2
- package/frequent_questions/pt/bun_set_up.md +1 -2
- package/frequent_questions/pt/customized_locale_list.md +1 -2
- package/frequent_questions/pt/domain_routing.md +1 -2
- package/frequent_questions/pt/esbuild_error.md +1 -2
- package/frequent_questions/pt/get_locale_cookie.md +1 -2
- package/frequent_questions/pt/intlayer_command_undefined.md +1 -2
- package/frequent_questions/pt/locale_incorect_in_url.md +1 -2
- package/frequent_questions/pt/static_rendering.md +1 -3
- package/frequent_questions/pt/translated_path_url.md +1 -2
- package/frequent_questions/pt/unknown_command.md +1 -2
- package/frequent_questions/ru/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/ru/array_as_content_declaration.md +1 -2
- package/frequent_questions/ru/build_dictionaries.md +1 -2
- package/frequent_questions/ru/build_error_CI_CD.md +1 -2
- package/frequent_questions/ru/bun_set_up.md +1 -2
- package/frequent_questions/ru/customized_locale_list.md +1 -2
- package/frequent_questions/ru/domain_routing.md +1 -2
- package/frequent_questions/ru/esbuild_error.md +1 -2
- package/frequent_questions/ru/get_locale_cookie.md +1 -2
- package/frequent_questions/ru/intlayer_command_undefined.md +1 -2
- package/frequent_questions/ru/locale_incorect_in_url.md +1 -2
- package/frequent_questions/ru/static_rendering.md +1 -2
- package/frequent_questions/ru/translated_path_url.md +1 -2
- package/frequent_questions/ru/unknown_command.md +1 -2
- package/frequent_questions/tr/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/tr/array_as_content_declaration.md +1 -2
- package/frequent_questions/tr/build_dictionaries.md +1 -2
- package/frequent_questions/tr/build_error_CI_CD.md +1 -2
- package/frequent_questions/tr/bun_set_up.md +1 -2
- package/frequent_questions/tr/customized_locale_list.md +1 -2
- package/frequent_questions/tr/domain_routing.md +1 -2
- package/frequent_questions/tr/esbuild_error.md +1 -2
- package/frequent_questions/tr/get_locale_cookie.md +1 -2
- package/frequent_questions/tr/intlayer_command_undefined.md +1 -2
- package/frequent_questions/tr/locale_incorect_in_url.md +1 -2
- package/frequent_questions/tr/static_rendering.md +1 -2
- package/frequent_questions/tr/translated_path_url.md +1 -2
- package/frequent_questions/tr/unknown_command.md +1 -2
- package/frequent_questions/vi/SSR_Next_no_[locale].md +106 -0
- package/frequent_questions/vi/array_as_content_declaration.md +71 -0
- package/frequent_questions/vi/build_dictionaries.md +58 -0
- package/frequent_questions/vi/build_error_CI_CD.md +74 -0
- package/frequent_questions/vi/bun_set_up.md +53 -0
- package/frequent_questions/vi/customized_locale_list.md +64 -0
- package/frequent_questions/vi/domain_routing.md +113 -0
- package/frequent_questions/vi/esbuild_error.md +29 -0
- package/frequent_questions/vi/get_locale_cookie.md +142 -0
- package/frequent_questions/vi/intlayer_command_undefined.md +155 -0
- package/frequent_questions/vi/locale_incorect_in_url.md +73 -0
- package/frequent_questions/vi/static_rendering.md +44 -0
- package/frequent_questions/vi/translated_path_url.md +55 -0
- package/frequent_questions/vi/unknown_command.md +97 -0
- package/frequent_questions/zh/SSR_Next_no_[locale].md +1 -2
- package/frequent_questions/zh/array_as_content_declaration.md +1 -2
- package/frequent_questions/zh/build_dictionaries.md +1 -2
- package/frequent_questions/zh/build_error_CI_CD.md +1 -2
- package/frequent_questions/zh/bun_set_up.md +1 -2
- package/frequent_questions/zh/customized_locale_list.md +1 -2
- package/frequent_questions/zh/domain_routing.md +1 -2
- package/frequent_questions/zh/esbuild_error.md +1 -2
- package/frequent_questions/zh/get_locale_cookie.md +1 -2
- package/frequent_questions/zh/intlayer_command_undefined.md +1 -2
- package/frequent_questions/zh/locale_incorect_in_url.md +1 -2
- package/frequent_questions/zh/static_rendering.md +1 -3
- package/frequent_questions/zh/translated_path_url.md +1 -2
- package/frequent_questions/zh/unknown_command.md +1 -2
- package/legal/id/privacy_notice.md +83 -0
- package/legal/id/terms_of_service.md +55 -0
- package/legal/pl/privacy_notice.md +83 -0
- package/legal/pl/terms_of_service.md +55 -0
- package/legal/vi/privacy_notice.md +83 -0
- package/legal/vi/terms_of_service.md +55 -0
- package/package.json +19 -18
- package/src/generated/blog.entry.ts +38 -0
|
@@ -0,0 +1,1073 @@
|
|
|
1
|
+
---
|
|
2
|
+
createdAt: 2025-11-01
|
|
3
|
+
updatedAt: 2025-11-01
|
|
4
|
+
title: How to internationalize your Next.js application using next-i18next
|
|
5
|
+
description: Set up i18n with next-i18next - best practices and SEO tips for multilingual Next.js apps, covering internationalization, content organization, and technical setup.
|
|
6
|
+
slugs:
|
|
7
|
+
- blog
|
|
8
|
+
- nextjs-internationalization-using-next-i18next
|
|
9
|
+
applicationTemplate: https://github.com/aymericzip/next-i18next-template
|
|
10
|
+
history:
|
|
11
|
+
- version: 7.0.6
|
|
12
|
+
date: 2025-11-01
|
|
13
|
+
changes: Initial version
|
|
14
|
+
---
|
|
15
|
+
|
|
16
|
+
# How to internationalize your Next.js application using next-i18next in 2025
|
|
17
|
+
|
|
18
|
+
## Table of Contents
|
|
19
|
+
|
|
20
|
+
<TOC/>
|
|
21
|
+
|
|
22
|
+
## What is next-i18next?
|
|
23
|
+
|
|
24
|
+
**next-i18next** is a popular internationalization (i18n) solution for Next.js applications. While the original `next-i18next` package was designed for the Pages Router, this guide shows you how to implement i18next with the modern **App Router** using `i18next` and `react-i18next` directly.
|
|
25
|
+
|
|
26
|
+
With this approach, you can:
|
|
27
|
+
|
|
28
|
+
- **Organize translations** using namespaces (e.g., `common.json`, `about.json`) for better content management.
|
|
29
|
+
- **Load translations efficiently** by loading only the namespaces needed for each page, reducing bundle size.
|
|
30
|
+
- **Support both server and client components** with proper SSR and hydration handling.
|
|
31
|
+
- **Ensure TypeScript support** with type-safe locale configuration and translation keys.
|
|
32
|
+
- **Optimize for SEO** with proper metadata, sitemap, and robots.txt internationalization.
|
|
33
|
+
|
|
34
|
+
> As an alternative, you can also refer to the [next-intl guide](https://github.com/aymericzip/intlayer/blob/main/docs/blog/en/i18n_using_next-intl.md), or directly using [Intlayer](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/intlayer_with_nextjs_16.md).
|
|
35
|
+
|
|
36
|
+
> See the comparison in [next-i18next vs next-intl vs Intlayer](https://github.com/aymericzip/intlayer/blob/main/docs/blog/en/next-i18next_vs_next-intl_vs_intlayer.md).
|
|
37
|
+
|
|
38
|
+
## Practices you should follow
|
|
39
|
+
|
|
40
|
+
Before we dive into the implementation, here are some practices you should follow:
|
|
41
|
+
|
|
42
|
+
- **Set HTML `lang` and `dir` attributes**
|
|
43
|
+
In your layout, compute `dir` using `getLocaleDirection(locale)` and set `<html lang={locale} dir={dir}>` for proper accessibility and SEO.
|
|
44
|
+
- **Split messages by namespace**
|
|
45
|
+
Organize JSON files per locale and namespace (e.g., `common.json`, `about.json`) to load only what you need.
|
|
46
|
+
- **Minimize client payload**
|
|
47
|
+
On pages, send only required namespaces to `NextIntlClientProvider` (e.g., `pick(messages, ['common', 'about'])`).
|
|
48
|
+
- **Prefer static pages**
|
|
49
|
+
Use static page as much as possible for better performance and SEO.
|
|
50
|
+
- **I18n in server components**
|
|
51
|
+
Server components, like pages or all components not marked as `client` are statics and can be pre-rendered at build time. So we will have to pass the translation functions to them as props.
|
|
52
|
+
- **Set up TypeScript types**
|
|
53
|
+
For your locales to ensure type safety throughout your application.
|
|
54
|
+
- **Proxy for redirection**
|
|
55
|
+
Use a proxy to handle the locale detection and routing and redirect the user to the appropriate locale-prefixed URL.
|
|
56
|
+
- **Internationalization of your metadata, sitemap, robots.txt**
|
|
57
|
+
Internationalize your metadata, sitemap, robots.txt using the `generateMetadata` function provided by Next.js to ensure a better discovery by search engines in all locales.
|
|
58
|
+
- **Localize Links**
|
|
59
|
+
Localize Links using the `Link` component to redirect the user to the appropriate locale-prefixed URL. It's important to ensure the discovery of your pages in all locales.
|
|
60
|
+
- **Automate tests and translations**
|
|
61
|
+
Automate tests and translations help loosing time to maintain your multilingual application.
|
|
62
|
+
|
|
63
|
+
> See our doc listing everything you need to know about internationalization and SEO: [Internationalization (i18n) with next-intl](https://github.com/aymericzip/intlayer/blob/main/docs/blog/en/internationalization_and_SEO.md).
|
|
64
|
+
|
|
65
|
+
---
|
|
66
|
+
|
|
67
|
+
## Step-by-Step Guide to Set Up i18next in a Next.js Application
|
|
68
|
+
|
|
69
|
+
<iframe
|
|
70
|
+
src="https://stackblitz.com/github/aymericzip/next-i18next-template?embed=1&ctl=1&file=src/app/i18n.ts"
|
|
71
|
+
className="m-auto overflow-hidden rounded-lg border-0 max-md:size-full max-md:h-[700px] md:aspect-16/9 md:w-full"
|
|
72
|
+
title="Demo CodeSandbox - How to Internationalize your application using Intlayer"
|
|
73
|
+
sandbox="allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts"
|
|
74
|
+
loading="lazy"
|
|
75
|
+
|
|
76
|
+
> See [Application Template](https://github.com/aymericzip/next-i18next-template) on GitHub.
|
|
77
|
+
|
|
78
|
+
Here's the project structure we'll be creating:
|
|
79
|
+
|
|
80
|
+
```bash
|
|
81
|
+
.
|
|
82
|
+
├── i18n.config.ts
|
|
83
|
+
└── src # Src is optional
|
|
84
|
+
├── locales
|
|
85
|
+
│ ├── en
|
|
86
|
+
│ │ ├── common.json
|
|
87
|
+
│ │ └── about.json
|
|
88
|
+
│ └── fr
|
|
89
|
+
│ ├── common.json
|
|
90
|
+
│ └── about.json
|
|
91
|
+
├── types
|
|
92
|
+
│ └── i18next.d.ts
|
|
93
|
+
├── app
|
|
94
|
+
│ ├── proxy.ts
|
|
95
|
+
│ ├── i18n
|
|
96
|
+
│ │ └── server.ts
|
|
97
|
+
│ └── [locale]
|
|
98
|
+
│ ├── layout.tsx
|
|
99
|
+
│ ├── (home) # / (Route Group to not pollute all pages with home messages)
|
|
100
|
+
│ │ ├── layout.tsx
|
|
101
|
+
│ │ └── page.tsx
|
|
102
|
+
│ └── about # /about
|
|
103
|
+
│ ├── layout.tsx
|
|
104
|
+
│ └── page.tsx
|
|
105
|
+
└── components
|
|
106
|
+
├── I18nProvider.tsx
|
|
107
|
+
├── ClientComponent.tsx
|
|
108
|
+
└── ServerComponent.tsx
|
|
109
|
+
```
|
|
110
|
+
|
|
111
|
+
### Step 1: Install Dependencies
|
|
112
|
+
|
|
113
|
+
Install the necessary packages using npm:
|
|
114
|
+
|
|
115
|
+
```bash packageManager="npm"
|
|
116
|
+
npm install i18next react-i18next i18next-resources-to-backend
|
|
117
|
+
```
|
|
118
|
+
|
|
119
|
+
```bash packageManager="pnpm"
|
|
120
|
+
pnpm add i18next react-i18next i18next-resources-to-backend
|
|
121
|
+
```
|
|
122
|
+
|
|
123
|
+
```bash packageManager="yarn"
|
|
124
|
+
yarn add i18next react-i18next i18next-resources-to-backend
|
|
125
|
+
```
|
|
126
|
+
|
|
127
|
+
- **i18next**: The core internationalization framework that handles translation loading and management.
|
|
128
|
+
- **react-i18next**: React bindings for i18next that provide hooks like `useTranslation` for client components.
|
|
129
|
+
- **i18next-resources-to-backend**: A plugin that enables dynamic loading of translation files, allowing you to load only the namespaces you need.
|
|
130
|
+
|
|
131
|
+
### Step 2: Configure Your Project
|
|
132
|
+
|
|
133
|
+
Create a configuration file to define your supported locales, default locale, and helper functions for URL localization. This file serves as the single source of truth for your i18n setup and ensures type safety throughout your application.
|
|
134
|
+
|
|
135
|
+
Centralizing your locale configuration prevents inconsistencies and makes it easier to add or remove locales in the future. The helper functions ensure consistent URL generation for SEO and routing.
|
|
136
|
+
|
|
137
|
+
```ts fileName="i18n.config.ts"
|
|
138
|
+
// Define supported locales as a const array for type safety
|
|
139
|
+
// The 'as const' assertion makes TypeScript infer literal types instead of string[]
|
|
140
|
+
export const locales = ["en", "fr"] as const;
|
|
141
|
+
|
|
142
|
+
// Extract the Locale type from the locales array
|
|
143
|
+
// This creates a union type: "en" | "fr"
|
|
144
|
+
export type Locale = (typeof locales)[number];
|
|
145
|
+
|
|
146
|
+
// Set the default locale used when no locale is specified
|
|
147
|
+
export const defaultLocale: Locale = "en";
|
|
148
|
+
|
|
149
|
+
// Right-to-left languages that need special text direction handling
|
|
150
|
+
export const rtlLocales = ["ar", "he", "fa", "ur"] as const;
|
|
151
|
+
|
|
152
|
+
// Check if a locale requires RTL (right-to-left) text direction
|
|
153
|
+
// Used for languages like Arabic, Hebrew, Persian, and Urdu
|
|
154
|
+
export const isRtl = (locale: string) =>
|
|
155
|
+
(rtlLocales as readonly string[]).includes(locale);
|
|
156
|
+
|
|
157
|
+
// Generate a localized path for a given locale and path
|
|
158
|
+
// Default locale paths don't have a prefix (e.g., "/about" instead of "/en/about")
|
|
159
|
+
// Other locales are prefixed (e.g., "/fr/about")
|
|
160
|
+
export function localizedPath(locale: string, path: string) {
|
|
161
|
+
return locale === defaultLocale ? path : `/${locale}${path}`;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// Base URL for absolute URLs (used in sitemaps, metadata, etc.)
|
|
165
|
+
const ORIGIN = "https://example.com";
|
|
166
|
+
|
|
167
|
+
// Generate an absolute URL with locale prefix
|
|
168
|
+
// Used for SEO metadata, sitemaps, and canonical URLs
|
|
169
|
+
export function absoluteUrl(locale: string, path: string) {
|
|
170
|
+
return `${ORIGIN}${localizedPath(locale, path)}`;
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
// Used to set the locale cookie in the browser
|
|
174
|
+
export function getCookie(locale: Locale) {
|
|
175
|
+
return [
|
|
176
|
+
`NEXT_LOCALE=${locale}`,
|
|
177
|
+
"Path=/",
|
|
178
|
+
`Max-Age=${60 * 60 * 24 * 365}`, // 1 year
|
|
179
|
+
"SameSite=Lax",
|
|
180
|
+
].join("; ");
|
|
181
|
+
}
|
|
182
|
+
```
|
|
183
|
+
|
|
184
|
+
### Step 3: Centralize Translation Namespaces
|
|
185
|
+
|
|
186
|
+
Create a single source of truth for every namespace your application exposes. Reusing this list keeps server, client, and tooling code in sync and unlocks strong typing for translation helpers.
|
|
187
|
+
|
|
188
|
+
```ts fileName="src/i18n.namespaces.ts"
|
|
189
|
+
export const namespaces = ["common", "about"] as const;
|
|
190
|
+
|
|
191
|
+
export type Namespace = (typeof namespaces)[number];
|
|
192
|
+
```
|
|
193
|
+
|
|
194
|
+
### Step 4: Strongly Type Translation Keys with TypeScript
|
|
195
|
+
|
|
196
|
+
Augment `i18next` to point at your canonical language files (usually English). TypeScript then infers valid keys per namespace, so calls to `t()` are checked end-to-end.
|
|
197
|
+
|
|
198
|
+
```ts fileName="src/types/i18next.d.ts"
|
|
199
|
+
import "i18next";
|
|
200
|
+
|
|
201
|
+
declare module "i18next" {
|
|
202
|
+
interface CustomTypeOptions {
|
|
203
|
+
defaultNS: "common";
|
|
204
|
+
resources: {
|
|
205
|
+
common: typeof import("@/locales/en/common.json");
|
|
206
|
+
about: typeof import("@/locales/en/about.json");
|
|
207
|
+
};
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
```
|
|
211
|
+
|
|
212
|
+
> Tip: Store this declaration under `src/types` (create the folder if it doesn't exist). Next.js already includes `src` in `tsconfig.json`, so the augmentation is picked up automatically. If not, add the following to your `tsconfig.json` file:
|
|
213
|
+
|
|
214
|
+
```json5 fileName="tsconfig.json"
|
|
215
|
+
{
|
|
216
|
+
"include": ["src/types/**/*.ts"],
|
|
217
|
+
}
|
|
218
|
+
```
|
|
219
|
+
|
|
220
|
+
With this in place you can rely on autocomplete and compile-time checks:
|
|
221
|
+
|
|
222
|
+
```tsx
|
|
223
|
+
import { useTranslation, type TFunction } from "react-i18next";
|
|
224
|
+
|
|
225
|
+
const { t } = useTranslation("about");
|
|
226
|
+
|
|
227
|
+
// OK, typed: t("counter.increment")
|
|
228
|
+
// ERROR, compile error: t("doesNotExist")
|
|
229
|
+
export type AboutTranslator = TFunction<"about">;
|
|
230
|
+
```
|
|
231
|
+
|
|
232
|
+
### Step 5: Set Up Server-Side i18n Initialization
|
|
233
|
+
|
|
234
|
+
Create a server-side initialization function that loads translations for server components. This function creates a separate i18next instance for server-side rendering, ensuring that translations are loaded before rendering.
|
|
235
|
+
|
|
236
|
+
Server components need their own i18next instance because they run in a different context than client components. Pre-loading translations on the server prevents flash of untranslated content and improves SEO by ensuring search engines see translated content.
|
|
237
|
+
|
|
238
|
+
```ts fileName="src/app/i18n/server.ts"
|
|
239
|
+
import { createInstance } from "i18next";
|
|
240
|
+
import { initReactI18next } from "react-i18next/initReactI18next";
|
|
241
|
+
import resourcesToBackend from "i18next-resources-to-backend";
|
|
242
|
+
import { defaultLocale } from "@/i18n.config";
|
|
243
|
+
import { namespaces, type Namespace } from "@/i18n.namespaces";
|
|
244
|
+
|
|
245
|
+
// Configure dynamic resource loading for i18next
|
|
246
|
+
// This function dynamically imports translation JSON files based on locale and namespace
|
|
247
|
+
// Example: locale="fr", namespace="about" -> imports "@/locales/fr/about.json"
|
|
248
|
+
const backend = resourcesToBackend(
|
|
249
|
+
(locale: string, namespace: string) =>
|
|
250
|
+
import(`@/locales/${locale}/${namespace}.json`)
|
|
251
|
+
);
|
|
252
|
+
|
|
253
|
+
const DEFAULT_NAMESPACES = [
|
|
254
|
+
namespaces[0],
|
|
255
|
+
] as const satisfies readonly Namespace[];
|
|
256
|
+
|
|
257
|
+
/**
|
|
258
|
+
* Initialize i18next instance for server-side rendering
|
|
259
|
+
*
|
|
260
|
+
* @returns Initialized i18next instance ready for server-side use
|
|
261
|
+
*/
|
|
262
|
+
export async function initI18next(
|
|
263
|
+
locale: string,
|
|
264
|
+
ns: readonly Namespace[] = DEFAULT_NAMESPACES
|
|
265
|
+
) {
|
|
266
|
+
// Create a new i18next instance (separate from client-side instance)
|
|
267
|
+
const i18n = createInstance();
|
|
268
|
+
|
|
269
|
+
// Initialize with React integration and backend loader
|
|
270
|
+
await i18n
|
|
271
|
+
.use(initReactI18next) // Enable React hooks support
|
|
272
|
+
.use(backend) // Enable dynamic resource loading
|
|
273
|
+
.init({
|
|
274
|
+
lng: locale,
|
|
275
|
+
fallbackLng: defaultLocale,
|
|
276
|
+
ns, // Load only specified namespaces for better performance
|
|
277
|
+
defaultNS: "common", // Default namespace when none is specified
|
|
278
|
+
interpolation: { escapeValue: false }, // Don't escape HTML (React handles XSS protection)
|
|
279
|
+
react: { useSuspense: false }, // Disable Suspense for SSR compatibility
|
|
280
|
+
returnNull: false, // Return empty string instead of null for missing keys
|
|
281
|
+
initImmediate: false, // Defer initialization until resources are loaded (faster SSR)
|
|
282
|
+
});
|
|
283
|
+
return i18n;
|
|
284
|
+
}
|
|
285
|
+
```
|
|
286
|
+
|
|
287
|
+
### Step 6: Create Client-Side i18n Provider
|
|
288
|
+
|
|
289
|
+
Create a client component provider that wraps your application with i18next context. This provider receives pre-loaded translations from the server to prevent flash of untranslated content (FOUC) and avoid duplicate fetching.
|
|
290
|
+
|
|
291
|
+
Client components need their own i18next instance that runs in the browser. By accepting pre-loaded resources from the server, we ensure seamless hydration and prevent content flashing. The provider also manages locale changes and namespace loading dynamically.
|
|
292
|
+
|
|
293
|
+
```tsx fileName="src/components/I18nProvider.tsx"
|
|
294
|
+
"use client";
|
|
295
|
+
|
|
296
|
+
import { useEffect, useState } from "react";
|
|
297
|
+
import { I18nextProvider } from "react-i18next";
|
|
298
|
+
import { createInstance, type ResourceLanguage } from "i18next";
|
|
299
|
+
import { initReactI18next } from "react-i18next/initReactI18next";
|
|
300
|
+
import resourcesToBackend from "i18next-resources-to-backend";
|
|
301
|
+
import { defaultLocale } from "@/i18n.config";
|
|
302
|
+
import { namespaces as allNamespaces, type Namespace } from "@/i18n.namespaces";
|
|
303
|
+
|
|
304
|
+
// Configure dynamic resource loading for client-side
|
|
305
|
+
// Same pattern as server-side, but this instance runs in the browser
|
|
306
|
+
const backend = resourcesToBackend(
|
|
307
|
+
(locale: string, namespace: string) =>
|
|
308
|
+
import(`@/locales/${locale}/${namespace}.json`)
|
|
309
|
+
);
|
|
310
|
+
|
|
311
|
+
type Props = {
|
|
312
|
+
locale: string;
|
|
313
|
+
namespaces?: readonly Namespace[];
|
|
314
|
+
// Pre-loaded resources from server (prevents FOUC - Flash of Untranslated Content)
|
|
315
|
+
// Format: { namespace: translationBundle }
|
|
316
|
+
resources?: Record<Namespace, ResourceLanguage>;
|
|
317
|
+
children: React.ReactNode;
|
|
318
|
+
};
|
|
319
|
+
|
|
320
|
+
/**
|
|
321
|
+
* Client-side i18n provider that wraps the app with i18next context
|
|
322
|
+
* Receives pre-loaded resources from server to avoid re-fetching translations
|
|
323
|
+
*/
|
|
324
|
+
export default function I18nProvider({
|
|
325
|
+
locale,
|
|
326
|
+
namespaces = [allNamespaces[0]] as const,
|
|
327
|
+
resources,
|
|
328
|
+
children,
|
|
329
|
+
}: Props) {
|
|
330
|
+
// Create i18n instance once using useState lazy initializer
|
|
331
|
+
// This ensures the instance is created only once, not on every render
|
|
332
|
+
const [i18n] = useState(() => {
|
|
333
|
+
const i18nInstance = createInstance();
|
|
334
|
+
|
|
335
|
+
i18nInstance
|
|
336
|
+
.use(initReactI18next)
|
|
337
|
+
.use(backend)
|
|
338
|
+
.init({
|
|
339
|
+
lng: locale,
|
|
340
|
+
fallbackLng: defaultLocale,
|
|
341
|
+
ns: namespaces,
|
|
342
|
+
// If resources are provided (from server), use them to avoid client-side fetching
|
|
343
|
+
// This prevents FOUC and improves initial load performance
|
|
344
|
+
resources: resources ? { [locale]: resources } : undefined,
|
|
345
|
+
defaultNS: "common",
|
|
346
|
+
interpolation: { escapeValue: false },
|
|
347
|
+
react: { useSuspense: false },
|
|
348
|
+
returnNull: false, // Prevent undefined values from being returned
|
|
349
|
+
});
|
|
350
|
+
|
|
351
|
+
return i18nInstance;
|
|
352
|
+
});
|
|
353
|
+
|
|
354
|
+
// Update language when locale prop changes
|
|
355
|
+
useEffect(() => {
|
|
356
|
+
i18n.changeLanguage(locale);
|
|
357
|
+
}, [locale, i18n]);
|
|
358
|
+
|
|
359
|
+
// Ensure all required namespaces are loaded client-side
|
|
360
|
+
// Using join("|") as dependency to compare arrays correctly
|
|
361
|
+
useEffect(() => {
|
|
362
|
+
i18n.loadNamespaces(namespaces);
|
|
363
|
+
}, [namespaces.join("|"), i18n]);
|
|
364
|
+
|
|
365
|
+
// Provide i18n instance to all child components via React context
|
|
366
|
+
return <I18nextProvider i18n={i18n}>{children}</I18nextProvider>;
|
|
367
|
+
}
|
|
368
|
+
```
|
|
369
|
+
|
|
370
|
+
### Step 7: Define Dynamic Locale Routes
|
|
371
|
+
|
|
372
|
+
Set up dynamic routing for locales by creating a `[locale]` directory in your app folder. This allows Next.js to handle locale-based routing where each locale becomes a URL segment (e.g., `/en/about`, `/fr/about`).
|
|
373
|
+
|
|
374
|
+
Using dynamic routes enables Next.js to generate static pages for all locales at build time, improving performance and SEO. The layout component sets the HTML `lang` and `dir` attributes based on the locale, which is crucial for accessibility and search engine understanding.
|
|
375
|
+
|
|
376
|
+
```tsx fileName="src/app/[locale]/layout.tsx"
|
|
377
|
+
import type { ReactNode } from "react";
|
|
378
|
+
import { locales, defaultLocale, isRtl, type Locale } from "@/i18n.config";
|
|
379
|
+
|
|
380
|
+
// Disable dynamic params - all locales must be known at build time
|
|
381
|
+
// This ensures static generation for all locale routes
|
|
382
|
+
export const dynamicParams = false;
|
|
383
|
+
|
|
384
|
+
/**
|
|
385
|
+
* Generate static params for all locales at build time
|
|
386
|
+
* Next.js will pre-render pages for each locale returned here
|
|
387
|
+
* Example: [{ locale: "en" }, { locale: "fr" }]
|
|
388
|
+
*/
|
|
389
|
+
export function generateStaticParams() {
|
|
390
|
+
return locales.map((locale) => ({ locale }));
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Root layout component that handles locale-specific HTML attributes
|
|
395
|
+
* Sets the lang attribute and text direction (ltr/rtl) based on locale
|
|
396
|
+
*/
|
|
397
|
+
export default function LocaleLayout({
|
|
398
|
+
children,
|
|
399
|
+
params,
|
|
400
|
+
}: {
|
|
401
|
+
children: ReactNode;
|
|
402
|
+
params: { locale: string };
|
|
403
|
+
}) {
|
|
404
|
+
// Validate locale from URL params
|
|
405
|
+
// If invalid locale is provided, fall back to default locale
|
|
406
|
+
const locale: Locale = (locales as readonly string[]).includes(params.locale)
|
|
407
|
+
? (params.locale as any)
|
|
408
|
+
: defaultLocale;
|
|
409
|
+
|
|
410
|
+
// Determine text direction based on locale
|
|
411
|
+
// RTL languages like Arabic need dir="rtl" for proper text rendering
|
|
412
|
+
const dir = isRtl(locale) ? "rtl" : "ltr";
|
|
413
|
+
|
|
414
|
+
return (
|
|
415
|
+
<html lang={locale} dir={dir}>
|
|
416
|
+
<body>{children}</body>
|
|
417
|
+
</html>
|
|
418
|
+
);
|
|
419
|
+
}
|
|
420
|
+
```
|
|
421
|
+
|
|
422
|
+
### Step 8: Create Your Translation Files
|
|
423
|
+
|
|
424
|
+
Create JSON files for each locale and namespace. This structure allows you to organize translations logically and load only what you need for each page.
|
|
425
|
+
|
|
426
|
+
Organizing translations by namespace (e.g., `common.json`, `about.json`) enables code splitting and reduces bundle size. You only load the translations needed for each page, improving performance.
|
|
427
|
+
|
|
428
|
+
```json fileName="src/locales/en/common.json"
|
|
429
|
+
{
|
|
430
|
+
"appTitle": "Next.js i18n App",
|
|
431
|
+
"appDescription": "Example Next.js application with internationalization using i18next"
|
|
432
|
+
}
|
|
433
|
+
```
|
|
434
|
+
|
|
435
|
+
```json fileName="src/locales/fr/common.json"
|
|
436
|
+
{
|
|
437
|
+
"appTitle": "Application Next.js i18n",
|
|
438
|
+
"appDescription": "Exemple d'application Next.js avec internationalisation utilisant i18next"
|
|
439
|
+
}
|
|
440
|
+
```
|
|
441
|
+
|
|
442
|
+
```json fileName="src/locales/en/home.json"
|
|
443
|
+
{
|
|
444
|
+
"title": "Home",
|
|
445
|
+
"description": "Home page description",
|
|
446
|
+
"welcome": "Welcome",
|
|
447
|
+
"greeting": "Hello, world!",
|
|
448
|
+
"aboutPage": "About Page",
|
|
449
|
+
"documentation": "Documentation"
|
|
450
|
+
}
|
|
451
|
+
```
|
|
452
|
+
|
|
453
|
+
```json fileName="src/locales/fr/home.json"
|
|
454
|
+
{
|
|
455
|
+
"title": "Accueil",
|
|
456
|
+
"description": "Description de la page d'accueil",
|
|
457
|
+
"welcome": "Bienvenue",
|
|
458
|
+
"greeting": "Bonjour le monde!",
|
|
459
|
+
"aboutPage": "Page À propos",
|
|
460
|
+
"documentation": "Documentation"
|
|
461
|
+
}
|
|
462
|
+
```
|
|
463
|
+
|
|
464
|
+
```json fileName="src/locales/en/about.json"
|
|
465
|
+
{
|
|
466
|
+
"title": "About",
|
|
467
|
+
"description": "About page description",
|
|
468
|
+
"counter": {
|
|
469
|
+
"label": "Counter",
|
|
470
|
+
"increment": "Increment",
|
|
471
|
+
"description": "Click the button to increase the counter"
|
|
472
|
+
}
|
|
473
|
+
}
|
|
474
|
+
```
|
|
475
|
+
|
|
476
|
+
```json fileName="src/locales/fr/about.json"
|
|
477
|
+
{
|
|
478
|
+
"title": "À propos",
|
|
479
|
+
"description": "Description de la page À propos",
|
|
480
|
+
"counter": {
|
|
481
|
+
"label": "Compteur",
|
|
482
|
+
"increment": "Incrémenter",
|
|
483
|
+
"description": "Cliquez sur le bouton pour augmenter le compteur"
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
```
|
|
487
|
+
|
|
488
|
+
### Step 9: Utilize Translations in Your Pages
|
|
489
|
+
|
|
490
|
+
Create a page component that initializes i18next on the server and passes translations to both server and client components. This ensures that translations are loaded before rendering and prevents content flashing.
|
|
491
|
+
|
|
492
|
+
Server-side initialization loads translations before the page renders, improving SEO and preventing FOUC. By passing pre-loaded resources to the client provider, we avoid duplicate fetching and ensure smooth hydration.
|
|
493
|
+
|
|
494
|
+
```tsx fileName="src/app/[locale]/about/index.tsx"
|
|
495
|
+
import I18nProvider from "@/components/I18nProvider";
|
|
496
|
+
import { initI18next } from "@/app/i18n/server";
|
|
497
|
+
import type { Locale } from "@/i18n.config";
|
|
498
|
+
import { namespaces as allNamespaces, type Namespace } from "@/i18n.namespaces";
|
|
499
|
+
import type { ResourceLanguage } from "i18next";
|
|
500
|
+
import ClientComponent from "@/components/ClientComponent";
|
|
501
|
+
import ServerComponent from "@/components/ServerComponent";
|
|
502
|
+
|
|
503
|
+
/**
|
|
504
|
+
* Server component page that handles i18n initialization
|
|
505
|
+
* Pre-loads translations on the server and passes them to client components
|
|
506
|
+
*/
|
|
507
|
+
export default async function AboutPage({
|
|
508
|
+
params: { locale },
|
|
509
|
+
}: {
|
|
510
|
+
params: { locale: Locale };
|
|
511
|
+
}) {
|
|
512
|
+
// Define which translation namespaces this page needs
|
|
513
|
+
// Reuse the centralized list for type safety and autocomplete
|
|
514
|
+
const pageNamespaces = allNamespaces;
|
|
515
|
+
|
|
516
|
+
// Initialize i18next on the server with required namespaces
|
|
517
|
+
// This loads translation JSON files server-side
|
|
518
|
+
const i18n = await initI18next(locale, pageNamespaces);
|
|
519
|
+
|
|
520
|
+
// Get a fixed translation function for the "about" namespace
|
|
521
|
+
// getFixedT locks the namespace, so t("title") instead of t("about:title")
|
|
522
|
+
const tAbout = i18n.getFixedT(locale, "about");
|
|
523
|
+
|
|
524
|
+
// Extract translation bundles from the i18n instance
|
|
525
|
+
// This data is passed to I18nProvider to hydrate client-side i18n
|
|
526
|
+
// Prevents FOUC (Flash of Untranslated Content) and avoids duplicate fetching
|
|
527
|
+
const resources = Object.fromEntries(
|
|
528
|
+
pageNamespaces.map((ns) => [ns, i18n.getResourceBundle(locale, ns)])
|
|
529
|
+
) satisfies Record<Namespace, ResourceLanguage>;
|
|
530
|
+
|
|
531
|
+
return (
|
|
532
|
+
<I18nProvider
|
|
533
|
+
locale={locale}
|
|
534
|
+
namespaces={pageNamespaces}
|
|
535
|
+
resources={resources}
|
|
536
|
+
>
|
|
537
|
+
<main>
|
|
538
|
+
<h1>{tAbout("title")}</h1>
|
|
539
|
+
|
|
540
|
+
<ClientComponent />
|
|
541
|
+
<ServerComponent t={tAbout} locale={locale} count={0} />
|
|
542
|
+
</main>
|
|
543
|
+
</I18nProvider>
|
|
544
|
+
);
|
|
545
|
+
}
|
|
546
|
+
```
|
|
547
|
+
|
|
548
|
+
### Step 10: Use Translations in Client Components
|
|
549
|
+
|
|
550
|
+
Client components can use the `useTranslation` hook to access translations. This hook provides access to the translation function and the i18n instance, allowing you to translate content and access locale information.
|
|
551
|
+
|
|
552
|
+
Client components need React hooks to access translations. The `useTranslation` hook integrates seamlessly with i18next and provides reactive updates when the locale changes.
|
|
553
|
+
|
|
554
|
+
> Ensure the page/provider includes only the namespaces you need (e.g., `about`).
|
|
555
|
+
> If you use React < 19, memoize heavy formatters like `Intl.NumberFormat`.
|
|
556
|
+
|
|
557
|
+
```tsx fileName="src/components/ClientComponent.tsx"
|
|
558
|
+
"use client";
|
|
559
|
+
|
|
560
|
+
import { useState } from "react";
|
|
561
|
+
import { useTranslation } from "react-i18next";
|
|
562
|
+
|
|
563
|
+
/**
|
|
564
|
+
* Client component example using React hooks for translations
|
|
565
|
+
* Can use hooks like useState, useEffect, and useTranslation
|
|
566
|
+
*/
|
|
567
|
+
const ClientComponent = () => {
|
|
568
|
+
// useTranslation hook provides access to translation function and i18n instance
|
|
569
|
+
// Specify namespace to only load translations for "about" namespace
|
|
570
|
+
const { t, i18n } = useTranslation("about");
|
|
571
|
+
const [count, setCount] = useState(0);
|
|
572
|
+
|
|
573
|
+
// Create locale-aware number formatter
|
|
574
|
+
// i18n.language provides current locale (e.g., "en", "fr")
|
|
575
|
+
// Intl.NumberFormat formats numbers according to locale conventions
|
|
576
|
+
const numberFormat = new Intl.NumberFormat(i18n.language);
|
|
577
|
+
|
|
578
|
+
return (
|
|
579
|
+
<div className="flex flex-col items-center gap-4">
|
|
580
|
+
{/* Format number using locale-specific formatting */}
|
|
581
|
+
<p className="text-5xl font-bold text-white m-0">
|
|
582
|
+
{numberFormat.format(count)}
|
|
583
|
+
</p>
|
|
584
|
+
<button
|
|
585
|
+
type="button"
|
|
586
|
+
className="flex h-12 w-full items-center justify-center gap-2 rounded-full bg-foreground px-5 text-background transition-colors hover:bg-[#383838] dark:hover:bg-[#ccc] md:w-[158px]"
|
|
587
|
+
aria-label={t("counter.label")}
|
|
588
|
+
onClick={() => setCount((c) => c + 1)}
|
|
589
|
+
>
|
|
590
|
+
{t("counter.increment")}
|
|
591
|
+
</button>
|
|
592
|
+
</div>
|
|
593
|
+
);
|
|
594
|
+
};
|
|
595
|
+
|
|
596
|
+
export default ClientComponent;
|
|
597
|
+
```
|
|
598
|
+
|
|
599
|
+
### Step 11: Use Translations in Server Components
|
|
600
|
+
|
|
601
|
+
Server components cannot use React hooks, so they receive translations via props from their parent components. This approach keeps server components synchronous and allows them to be nested inside client components.
|
|
602
|
+
|
|
603
|
+
Server components that might be nested under client boundaries need to be synchronous. By passing translated strings and locale information as props, we avoid async operations and ensure proper rendering.
|
|
604
|
+
|
|
605
|
+
```tsx fileName="src/components/ServerComponent.tsx"
|
|
606
|
+
import type { TFunction } from "i18next";
|
|
607
|
+
|
|
608
|
+
type ServerComponentProps = {
|
|
609
|
+
// Translation function passed from parent server component
|
|
610
|
+
// Server components can't use hooks, so translations come via props
|
|
611
|
+
t: TFunction<"about">;
|
|
612
|
+
locale: string;
|
|
613
|
+
count: number;
|
|
614
|
+
};
|
|
615
|
+
|
|
616
|
+
/**
|
|
617
|
+
* Server component example - receives translations as props
|
|
618
|
+
* Can be nested inside client components (async server components)
|
|
619
|
+
* Cannot use React hooks, so all data must come from props or async operations
|
|
620
|
+
*/
|
|
621
|
+
const ServerComponent = ({ t, locale, count }: ServerComponentProps) => {
|
|
622
|
+
// Format number server-side using locale
|
|
623
|
+
// This runs on the server during SSR, improving initial page load
|
|
624
|
+
const formatted = new Intl.NumberFormat(locale).format(count);
|
|
625
|
+
|
|
626
|
+
return (
|
|
627
|
+
<div className="flex flex-col items-center gap-4">
|
|
628
|
+
<p className="text-5xl font-bold text-white m-0">{formatted}</p>
|
|
629
|
+
{/* Use translation function passed as prop */}
|
|
630
|
+
<div className="flex flex-col items-center gap-2">
|
|
631
|
+
<span className="text-xl font-semibold text-white">
|
|
632
|
+
{t("counter.label")}
|
|
633
|
+
</span>
|
|
634
|
+
<span className="text-sm opacity-80 italic">
|
|
635
|
+
{t("counter.description")}
|
|
636
|
+
</span>
|
|
637
|
+
</div>
|
|
638
|
+
</div>
|
|
639
|
+
);
|
|
640
|
+
};
|
|
641
|
+
|
|
642
|
+
export default ServerComponent;
|
|
643
|
+
```
|
|
644
|
+
|
|
645
|
+
---
|
|
646
|
+
|
|
647
|
+
### (Optional) Step 12: Change the language of your content
|
|
648
|
+
|
|
649
|
+
To change the language of your content in Next.js, the recommended way is to use locale-prefixed URLs and Next.js links. The example below reads the current locale from the route, strips it from the pathname, and renders one link per available locale.
|
|
650
|
+
|
|
651
|
+
```tsx fileName="src/components/LocaleSwitcher.tsx"
|
|
652
|
+
"use client";
|
|
653
|
+
|
|
654
|
+
import Link from "next/link";
|
|
655
|
+
import { useParams, usePathname } from "next/navigation";
|
|
656
|
+
import { useMemo } from "react";
|
|
657
|
+
import { defaultLocale, getCookie, type Locale, locales } from "@/i18n.config";
|
|
658
|
+
|
|
659
|
+
export default function LocaleSwitcher() {
|
|
660
|
+
const params = useParams();
|
|
661
|
+
const pathname = usePathname();
|
|
662
|
+
|
|
663
|
+
const activeLocale = (params?.locale as Locale | undefined) ?? defaultLocale;
|
|
664
|
+
|
|
665
|
+
const getLocaleLabel = (locale: Locale): string => {
|
|
666
|
+
try {
|
|
667
|
+
const displayNames = new Intl.DisplayNames([locale], {
|
|
668
|
+
type: "language",
|
|
669
|
+
});
|
|
670
|
+
return displayNames.of(locale) ?? locale.toUpperCase();
|
|
671
|
+
} catch {
|
|
672
|
+
return locale.toUpperCase();
|
|
673
|
+
}
|
|
674
|
+
};
|
|
675
|
+
|
|
676
|
+
const basePath = useMemo(() => {
|
|
677
|
+
if (!pathname) return "/";
|
|
678
|
+
|
|
679
|
+
const segments = pathname.split("/").filter(Boolean);
|
|
680
|
+
|
|
681
|
+
if (segments.length === 0) return "/";
|
|
682
|
+
|
|
683
|
+
const maybeLocale = segments[0] as Locale;
|
|
684
|
+
|
|
685
|
+
if ((locales as readonly string[]).includes(maybeLocale)) {
|
|
686
|
+
const rest = segments.slice(1).join("/");
|
|
687
|
+
return rest ? `/${rest}` : "/";
|
|
688
|
+
}
|
|
689
|
+
|
|
690
|
+
return pathname;
|
|
691
|
+
}, [pathname]);
|
|
692
|
+
|
|
693
|
+
return (
|
|
694
|
+
<nav aria-label="Language selector">
|
|
695
|
+
{(locales as readonly Locale[]).map((locale) => {
|
|
696
|
+
const isActive = locale === activeLocale;
|
|
697
|
+
|
|
698
|
+
const href =
|
|
699
|
+
locale === defaultLocale ? basePath : `/${locale}${basePath}`;
|
|
700
|
+
|
|
701
|
+
return (
|
|
702
|
+
<Link
|
|
703
|
+
key={locale}
|
|
704
|
+
href={href}
|
|
705
|
+
aria-current={isActive ? "page" : undefined}
|
|
706
|
+
onClick={() => {
|
|
707
|
+
document.cookie = getCookie(locale);
|
|
708
|
+
}}
|
|
709
|
+
>
|
|
710
|
+
{getLocaleLabel(locale)}
|
|
711
|
+
</Link>
|
|
712
|
+
);
|
|
713
|
+
})}
|
|
714
|
+
</nav>
|
|
715
|
+
);
|
|
716
|
+
}
|
|
717
|
+
```
|
|
718
|
+
|
|
719
|
+
### (Optional) Step 13: Build a localized Link component
|
|
720
|
+
|
|
721
|
+
Reusing localized URLs across your app keeps navigation consistent and SEO-friendly. Wrap `next/link` in a small helper that prefixes internal routes with the active locale while leaving external URLs untouched.
|
|
722
|
+
|
|
723
|
+
```tsx fileName="src/components/LocalizedLink.tsx"
|
|
724
|
+
"use client";
|
|
725
|
+
|
|
726
|
+
import NextLink, { type LinkProps } from "next/link";
|
|
727
|
+
import { useParams } from "next/navigation";
|
|
728
|
+
import type { ComponentProps, PropsWithChildren } from "react";
|
|
729
|
+
import {
|
|
730
|
+
defaultLocale,
|
|
731
|
+
type Locale,
|
|
732
|
+
locales,
|
|
733
|
+
localizedPath,
|
|
734
|
+
} from "@/i18n.config";
|
|
735
|
+
|
|
736
|
+
const isExternal = (href: string) => /^https?:\/\//.test(href);
|
|
737
|
+
|
|
738
|
+
type LocalizedLinkProps = PropsWithChildren<
|
|
739
|
+
Omit<LinkProps, "href"> &
|
|
740
|
+
Omit<ComponentProps<"a">, "href"> & { href: string; locale?: Locale }
|
|
741
|
+
>;
|
|
742
|
+
|
|
743
|
+
export default function LocalizedLink({
|
|
744
|
+
href,
|
|
745
|
+
locale,
|
|
746
|
+
children,
|
|
747
|
+
...props
|
|
748
|
+
}: LocalizedLinkProps) {
|
|
749
|
+
const params = useParams();
|
|
750
|
+
const fallback = (params?.locale as Locale | undefined) ?? defaultLocale;
|
|
751
|
+
const normalizedLocale = (locales as readonly string[]).includes(fallback)
|
|
752
|
+
? ((locale ?? fallback) as Locale)
|
|
753
|
+
: defaultLocale;
|
|
754
|
+
|
|
755
|
+
const normalizedPath = href.startsWith("/") ? href : `/${href}`;
|
|
756
|
+
const localizedHref = isExternal(href)
|
|
757
|
+
? href
|
|
758
|
+
: localizedPath(normalizedLocale, normalizedPath);
|
|
759
|
+
|
|
760
|
+
return (
|
|
761
|
+
<NextLink href={localizedHref} {...props}>
|
|
762
|
+
{children}
|
|
763
|
+
</NextLink>
|
|
764
|
+
);
|
|
765
|
+
}
|
|
766
|
+
```
|
|
767
|
+
|
|
768
|
+
> Tip: Because `LocalizedLink` is a drop-in replacement, migrate gradually by swapping imports and letting the component handle locale-specific URLs.
|
|
769
|
+
|
|
770
|
+
### (Optional) Step 14: Access the active locale inside Server Actions
|
|
771
|
+
|
|
772
|
+
Server Actions often need the current locale for emails, logging, or third-party integrations. Combine the locale cookie set by your proxy with the `Accept-Language` header as a fallback.
|
|
773
|
+
|
|
774
|
+
```ts fileName="src/app/actions/get-current-locale.ts"
|
|
775
|
+
"use server";
|
|
776
|
+
|
|
777
|
+
import { cookies, headers } from "next/headers";
|
|
778
|
+
import { defaultLocale, locales, type Locale } from "@/i18n.config";
|
|
779
|
+
|
|
780
|
+
const KNOWN_LOCALES = new Set(locales as readonly string[]);
|
|
781
|
+
|
|
782
|
+
const normalize = (value: string | undefined): Locale | undefined => {
|
|
783
|
+
if (!value) return undefined;
|
|
784
|
+
const base = value.toLowerCase().split("-")[0];
|
|
785
|
+
return KNOWN_LOCALES.has(base) ? (base as Locale) : undefined;
|
|
786
|
+
};
|
|
787
|
+
|
|
788
|
+
export async function getCurrentLocale(): Promise<Locale> {
|
|
789
|
+
const cookieLocale = normalize(cookies().get("NEXT_LOCALE")?.value);
|
|
790
|
+
|
|
791
|
+
if (cookieLocale) return cookieLocale;
|
|
792
|
+
|
|
793
|
+
const headerLocale = normalize(headers().get("accept-language"));
|
|
794
|
+
return headerLocale ?? defaultLocale;
|
|
795
|
+
}
|
|
796
|
+
|
|
797
|
+
// Example of a server action that uses the current locale
|
|
798
|
+
export async function stuffFromServer(formData: FormData) {
|
|
799
|
+
const locale = await getCurrentLocale();
|
|
800
|
+
|
|
801
|
+
// Use the locale for localized side effects (emails, CRM, etc.)
|
|
802
|
+
console.log(`Stuff from server with locale ${locale}`);
|
|
803
|
+
}
|
|
804
|
+
```
|
|
805
|
+
|
|
806
|
+
> Because the helper relies on Next.js cookies and headers, it works in Route Handlers, Server Actions, and other server-only contexts.
|
|
807
|
+
|
|
808
|
+
### (Optional) Step 15: Internationalize Your Metadata
|
|
809
|
+
|
|
810
|
+
Translating content is important, but the main goal of internationalization is to make your website more visible to the world. I18n is an incredible lever to improve your website visibility through proper SEO.
|
|
811
|
+
|
|
812
|
+
Properly internationalized metadata helps search engines understand what languages are available on your pages. This includes setting hreflang meta tags, translating titles and descriptions, and ensuring canonical URLs are correctly set for each locale.
|
|
813
|
+
|
|
814
|
+
Here's a list of good practices regarding multilingual SEO:
|
|
815
|
+
|
|
816
|
+
- Set hreflang meta tags in the `<head>` tag to help search engines understand what languages are available on the page
|
|
817
|
+
- List all page translations in the sitemap.xml using the `http://www.w3.org/1999/xhtml` XML schema
|
|
818
|
+
- Do not forget to exclude prefixed pages from the robots.txt (e.g., `/dashboard`, `/fr/dashboard`, `/es/dashboard`)
|
|
819
|
+
- Use custom Link component to redirect to the most localized page (e.g., in French `<a href="/fr/about">À propos</a>`)
|
|
820
|
+
|
|
821
|
+
Developers often forget to properly reference their pages across locales. Let's fix that:
|
|
822
|
+
|
|
823
|
+
```tsx fileName="src/app/[locale]/about/layout.tsx"
|
|
824
|
+
import type { Metadata } from "next";
|
|
825
|
+
import {
|
|
826
|
+
locales,
|
|
827
|
+
defaultLocale,
|
|
828
|
+
localizedPath,
|
|
829
|
+
absoluteUrl,
|
|
830
|
+
} from "@/i18n.config";
|
|
831
|
+
|
|
832
|
+
/**
|
|
833
|
+
* Generate SEO metadata for each locale version of the page
|
|
834
|
+
* This function runs for each locale at build time
|
|
835
|
+
*/
|
|
836
|
+
export async function generateMetadata({
|
|
837
|
+
params,
|
|
838
|
+
}: {
|
|
839
|
+
params: { locale: string };
|
|
840
|
+
}): Promise<Metadata> {
|
|
841
|
+
const { locale } = params;
|
|
842
|
+
|
|
843
|
+
// Dynamically import translation file for this locale
|
|
844
|
+
// Used to get translated title and description for metadata
|
|
845
|
+
const messages = (await import(`@/locales/${locale}/about.json`)).default;
|
|
846
|
+
|
|
847
|
+
// Create hreflang mapping for all locales
|
|
848
|
+
// Helps search engines understand language alternatives
|
|
849
|
+
// Format: { "en": "/about", "fr": "/fr/about" }
|
|
850
|
+
const languages = Object.fromEntries(
|
|
851
|
+
locales.map((locale) => [locale, localizedPath(locale, "/about")])
|
|
852
|
+
);
|
|
853
|
+
|
|
854
|
+
return {
|
|
855
|
+
title: messages.title,
|
|
856
|
+
description: messages.description,
|
|
857
|
+
alternates: {
|
|
858
|
+
// Canonical URL for this locale version
|
|
859
|
+
canonical: absoluteUrl(locale, "/about"),
|
|
860
|
+
// Language alternatives for SEO (hreflang tags)
|
|
861
|
+
// "x-default" specifies the default locale version
|
|
862
|
+
languages: {
|
|
863
|
+
...languages,
|
|
864
|
+
"x-default": absoluteUrl(defaultLocale, "/about"),
|
|
865
|
+
},
|
|
866
|
+
},
|
|
867
|
+
};
|
|
868
|
+
}
|
|
869
|
+
|
|
870
|
+
export default async function AboutPage() {
|
|
871
|
+
return <h1>About</h1>;
|
|
872
|
+
}
|
|
873
|
+
```
|
|
874
|
+
|
|
875
|
+
### (Optional) Step 16: Internationalize Your Sitemap
|
|
876
|
+
|
|
877
|
+
Generate a sitemap that includes all locale versions of your pages. This helps search engines discover and index all language versions of your content.
|
|
878
|
+
|
|
879
|
+
A properly internationalized sitemap ensures search engines can find and index all language versions of your pages. This improves visibility in international search results.
|
|
880
|
+
|
|
881
|
+
```ts fileName="src/app/sitemap.ts"
|
|
882
|
+
import type { MetadataRoute } from "next";
|
|
883
|
+
import { defaultLocale, locales } from "@/i18n";
|
|
884
|
+
|
|
885
|
+
const origin = "https://example.com";
|
|
886
|
+
|
|
887
|
+
const formatterLocalizedPath = (locale: string, path: string) =>
|
|
888
|
+
locale === defaultLocale ? `${origin}${path}` : `${origin}/${locale}${path}`;
|
|
889
|
+
|
|
890
|
+
/**
|
|
891
|
+
* Get a map of all locales and their localized paths
|
|
892
|
+
*
|
|
893
|
+
* Example output:
|
|
894
|
+
* {
|
|
895
|
+
* "en": "https://example.com",
|
|
896
|
+
* "fr": "https://example.com/fr",
|
|
897
|
+
* "es": "https://example.com/es",
|
|
898
|
+
* "x-default": "https://example.com"
|
|
899
|
+
* }
|
|
900
|
+
*/
|
|
901
|
+
const getLocalizedMap = (path: string) =>
|
|
902
|
+
Object.fromEntries([
|
|
903
|
+
...locales.map((locale) => [locale, formatterLocalizedPath(locale, path)]),
|
|
904
|
+
["x-default", formatterLocalizedPath(defaultLocale, path)],
|
|
905
|
+
]);
|
|
906
|
+
|
|
907
|
+
// Generate sitemap with all locale variants for better SEO
|
|
908
|
+
// The alternates field tells search engines about language versions
|
|
909
|
+
export default function sitemap(): MetadataRoute.Sitemap {
|
|
910
|
+
return [
|
|
911
|
+
{
|
|
912
|
+
url: formatterLocalizedPath(defaultLocale, "/"),
|
|
913
|
+
lastModified: new Date(),
|
|
914
|
+
changeFrequency: "monthly",
|
|
915
|
+
priority: 1.0,
|
|
916
|
+
alternates: { languages: getLocalizedMap("/") },
|
|
917
|
+
},
|
|
918
|
+
{
|
|
919
|
+
url: formatterLocalizedPath(defaultLocale, "/about"),
|
|
920
|
+
lastModified: new Date(),
|
|
921
|
+
changeFrequency: "monthly",
|
|
922
|
+
priority: 0.7,
|
|
923
|
+
alternates: { languages: getLocalizedMap("/about") },
|
|
924
|
+
},
|
|
925
|
+
];
|
|
926
|
+
}
|
|
927
|
+
```
|
|
928
|
+
|
|
929
|
+
### (Optional) Step 17: Internationalize Your robots.txt
|
|
930
|
+
|
|
931
|
+
Create a robots.txt file that properly handles all locale versions of your protected routes. This ensures that search engines don't index admin or dashboard pages in any language.
|
|
932
|
+
|
|
933
|
+
Properly configuring robots.txt for all locales prevents search engines from indexing sensitive pages in any language. This is crucial for security and privacy.
|
|
934
|
+
|
|
935
|
+
```ts fileName="src/app/robots.ts"
|
|
936
|
+
import type { MetadataRoute } from "next";
|
|
937
|
+
import { defaultLocale, locales } from "@/i18n";
|
|
938
|
+
|
|
939
|
+
const origin = "https://example.com";
|
|
940
|
+
|
|
941
|
+
// Generate paths for all locales (e.g., /admin, /fr/admin, /es/admin)
|
|
942
|
+
const withAllLocales = (path: string) => [
|
|
943
|
+
path,
|
|
944
|
+
...locales
|
|
945
|
+
.filter((locale) => locale !== defaultLocale)
|
|
946
|
+
.map((locale) => `/${locale}${path}`),
|
|
947
|
+
];
|
|
948
|
+
|
|
949
|
+
const disallow = [...withAllLocales("/dashboard"), ...withAllLocales("/admin")];
|
|
950
|
+
|
|
951
|
+
export default function robots(): MetadataRoute.Robots {
|
|
952
|
+
return {
|
|
953
|
+
rules: { userAgent: "*", allow: ["/"], disallow },
|
|
954
|
+
host: origin,
|
|
955
|
+
sitemap: `${origin}/sitemap.xml`,
|
|
956
|
+
};
|
|
957
|
+
}
|
|
958
|
+
```
|
|
959
|
+
|
|
960
|
+
### (Optional) Step 18: Set Up Middleware for Locale Routing
|
|
961
|
+
|
|
962
|
+
Create a proxy to automatically detect the user's preferred locale and redirect them to the appropriate locale-prefixed URL. This improves user experience by showing content in their preferred language.
|
|
963
|
+
|
|
964
|
+
Middleware ensures that users are automatically redirected to their preferred language when they visit your site. It also saves the user's preference in a cookie for future visits.
|
|
965
|
+
|
|
966
|
+
```ts fileName="src/proxy.ts"
|
|
967
|
+
import { NextResponse, type NextRequest } from "next/server";
|
|
968
|
+
import { defaultLocale, locales } from "@/i18n.config";
|
|
969
|
+
|
|
970
|
+
// Regex to match files with extensions (e.g., .js, .css, .png)
|
|
971
|
+
// Used to exclude static assets from locale routing
|
|
972
|
+
const PUBLIC_FILE = /\.[^/]+$/;
|
|
973
|
+
|
|
974
|
+
/**
|
|
975
|
+
* Extract locale from Accept-Language header
|
|
976
|
+
* Handles formats like "fr-CA", "en-US", etc.
|
|
977
|
+
* Falls back to default locale if browser language is not supported
|
|
978
|
+
*/
|
|
979
|
+
const pickLocale = (accept: string | null) => {
|
|
980
|
+
// Get first language preference (e.g., "fr-CA" from "fr-CA,en-US;q=0.9")
|
|
981
|
+
const raw = accept?.split(",")[0] ?? defaultLocale;
|
|
982
|
+
// Extract base language code (e.g., "fr" from "fr-CA")
|
|
983
|
+
const base = raw.toLowerCase().split("-")[0];
|
|
984
|
+
// Check if we support this locale, otherwise use default
|
|
985
|
+
return (locales as readonly string[]).includes(base) ? base : defaultLocale;
|
|
986
|
+
};
|
|
987
|
+
|
|
988
|
+
/**
|
|
989
|
+
* Next.js proxy for locale detection and routing
|
|
990
|
+
* Runs on every request before the page renders
|
|
991
|
+
* Automatically redirects to locale-prefixed URLs when needed
|
|
992
|
+
*/
|
|
993
|
+
export function proxy(request: NextRequest) {
|
|
994
|
+
const { pathname } = request.nextUrl;
|
|
995
|
+
|
|
996
|
+
// Skip proxy for Next.js internals, API routes, and static files
|
|
997
|
+
// These should not be locale-prefixed
|
|
998
|
+
if (
|
|
999
|
+
pathname.startsWith("/_next") ||
|
|
1000
|
+
pathname.startsWith("/api") ||
|
|
1001
|
+
pathname.startsWith("/static") ||
|
|
1002
|
+
PUBLIC_FILE.test(pathname)
|
|
1003
|
+
) {
|
|
1004
|
+
return;
|
|
1005
|
+
}
|
|
1006
|
+
|
|
1007
|
+
// Check if URL already has a locale prefix
|
|
1008
|
+
// Example: "/fr/about" or "/en" would return true
|
|
1009
|
+
const hasLocale = (locales as readonly string[]).some(
|
|
1010
|
+
(locale) => pathname === `/${locale}` || pathname.startsWith(`/${locale}/`)
|
|
1011
|
+
);
|
|
1012
|
+
|
|
1013
|
+
// If no locale prefix, detect locale and redirect
|
|
1014
|
+
if (!hasLocale) {
|
|
1015
|
+
// Try to get locale from cookie first (user preference)
|
|
1016
|
+
const cookieLocale = request.cookies.get("NEXT_LOCALE")?.value;
|
|
1017
|
+
|
|
1018
|
+
// Use cookie locale if valid, otherwise detect from browser headers
|
|
1019
|
+
const locale =
|
|
1020
|
+
cookieLocale && (locales as readonly string[]).includes(cookieLocale)
|
|
1021
|
+
? cookieLocale
|
|
1022
|
+
: pickLocale(request.headers.get("accept-language"));
|
|
1023
|
+
|
|
1024
|
+
// Clone URL to modify pathname
|
|
1025
|
+
const url = request.nextUrl.clone();
|
|
1026
|
+
// Add locale prefix to pathname
|
|
1027
|
+
// Handle root path specially to avoid double slash
|
|
1028
|
+
url.pathname = `/${locale}${pathname === "/" ? "" : pathname}`;
|
|
1029
|
+
|
|
1030
|
+
// Create redirect response and set locale cookie
|
|
1031
|
+
const res = NextResponse.redirect(url);
|
|
1032
|
+
res.cookies.set("NEXT_LOCALE", locale, { path: "/" });
|
|
1033
|
+
return res;
|
|
1034
|
+
}
|
|
1035
|
+
}
|
|
1036
|
+
|
|
1037
|
+
export const config = {
|
|
1038
|
+
matcher: [
|
|
1039
|
+
// Match all paths except:
|
|
1040
|
+
// - API routes (/api/*)
|
|
1041
|
+
// - Next.js internals (/_next/*)
|
|
1042
|
+
// - Static files (/static/*)
|
|
1043
|
+
// - Files with extensions (.*\\..*)
|
|
1044
|
+
"/((?!api|_next|static|.*\\..*).*)",
|
|
1045
|
+
],
|
|
1046
|
+
};
|
|
1047
|
+
```
|
|
1048
|
+
|
|
1049
|
+
### (Optional) Step 19: Automate Your Translations Using Intlayer
|
|
1050
|
+
|
|
1051
|
+
Intlayer is a **free** and **open-source** library designed to assist the localization process in your application. While i18next handles the translation loading and management, Intlayer helps automate the translation workflow.
|
|
1052
|
+
|
|
1053
|
+
Managing translations manually can be time-consuming and error-prone. Intlayer automates translation testing, generation, and management, saving you time and ensuring consistency across your application.
|
|
1054
|
+
|
|
1055
|
+
Intlayer will allows your to:
|
|
1056
|
+
|
|
1057
|
+
- **Declare your content where you want in your codebase**
|
|
1058
|
+
Intlayer allows to declare your content where you want in your codebase using `.content.{ts|js|json}` files. It will allow a better organization of your content, ensuring better readability and maintainability of your codebase.
|
|
1059
|
+
|
|
1060
|
+
- **Test missing translations**
|
|
1061
|
+
Intlayer provide test functions to that can be integrated in your CI/CD pipeline, or in your unit tests. Learn more about [testing your translations](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/testing.md).
|
|
1062
|
+
|
|
1063
|
+
- **Automate your translations**,
|
|
1064
|
+
Intlayer provide a CLI and a VSCode extension to automate your translations. It can be integrated in your CI/CD pipeline. Learn more about [automating your translations](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/intlayer_cli.md).
|
|
1065
|
+
You can use your **own API key, and the AI provider of your choice**. It also provide context aware translations, see [fill content](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/autoFill.md).
|
|
1066
|
+
|
|
1067
|
+
- **Connect external content**
|
|
1068
|
+
Intlayer allows you to connect your content to an external content management system (CMS). To fetch it in a optimized way and insert it in your JSON resources. Learn more about [fetching external content](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/dictionary/function_fetching.md).
|
|
1069
|
+
|
|
1070
|
+
- **Visual editor**
|
|
1071
|
+
Intlayer offers an free visual editor to edit your content using a visual editor. Learn more about [visual editing your translations](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/intlayer_visual_editor.md).
|
|
1072
|
+
|
|
1073
|
+
And more. To discover all the features provided by Intlayer, please refer to the [Interest of Intlayer documentation](https://github.com/aymericzip/intlayer/blob/main/docs/docs/en/interest_of_intlayer.md).
|