@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.
Files changed (726) hide show
  1. package/blog/ar/i18n_using_next-i18next.md +1068 -0
  2. package/blog/ar/i18n_using_next-intl.md +768 -0
  3. package/blog/ar/intlayer_with_react-intl.md +0 -4
  4. package/blog/ar/next-i18next_vs_next-intl_vs_intlayer.md +5 -4
  5. package/blog/de/i18n_using_next-i18next.md +1107 -0
  6. package/blog/de/i18n_using_next-intl.md +760 -0
  7. package/blog/de/intlayer_with_react-intl.md +0 -4
  8. package/blog/de/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  9. package/blog/en/i18n_using_next-i18next.md +1073 -0
  10. package/blog/en/i18n_using_next-intl.md +757 -0
  11. package/blog/en/intlayer_with_i18next.md +71 -8
  12. package/blog/en/intlayer_with_next-i18next.md +71 -8
  13. package/blog/en/intlayer_with_next-intl.md +71 -8
  14. package/blog/en/intlayer_with_react-i18next.md +69 -8
  15. package/blog/en/intlayer_with_react-intl.md +68 -9
  16. package/blog/en/intlayer_with_vue-i18n.md +68 -7
  17. package/blog/en/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
  18. package/blog/en/vue-i18n_vs_intlayer.md +2 -0
  19. package/blog/en-GB/i18n_using_next-i18next.md +1074 -0
  20. package/blog/en-GB/i18n_using_next-intl.md +757 -0
  21. package/blog/en-GB/intlayer_with_i18next.md +15 -6
  22. package/blog/en-GB/intlayer_with_next-i18next.md +16 -6
  23. package/blog/en-GB/intlayer_with_next-intl.md +16 -6
  24. package/blog/en-GB/intlayer_with_react-i18next.md +16 -7
  25. package/blog/en-GB/intlayer_with_react-intl.md +14 -9
  26. package/blog/en-GB/intlayer_with_vue-i18n.md +16 -7
  27. package/blog/en-GB/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  28. package/blog/en-GB/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
  29. package/blog/en-GB/vue-i18n_vs_intlayer.md +2 -0
  30. package/blog/es/i18n_using_next-i18next.md +1066 -0
  31. package/blog/es/i18n_using_next-intl.md +757 -0
  32. package/blog/es/intlayer_with_react-intl.md +0 -4
  33. package/blog/es/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  34. package/blog/fr/i18n_using_next-i18next.md +1078 -0
  35. package/blog/fr/i18n_using_next-intl.md +759 -0
  36. package/blog/fr/intlayer_with_react-intl.md +0 -4
  37. package/blog/fr/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  38. package/blog/hi/i18n_using_next-i18next.md +1068 -0
  39. package/blog/hi/i18n_using_next-intl.md +758 -0
  40. package/blog/hi/intlayer_with_react-intl.md +0 -4
  41. package/blog/hi/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  42. package/blog/id/i18n_using_next-i18next.md +1078 -0
  43. package/blog/id/i18n_using_next-intl.md +757 -0
  44. package/blog/id/index.md +69 -0
  45. package/blog/id/internationalization_and_SEO.md +364 -0
  46. package/blog/id/intlayer_with_react-intl.md +0 -4
  47. package/blog/id/list_i18n_technologies/CMS/drupal.md +143 -0
  48. package/blog/id/list_i18n_technologies/CMS/wix.md +167 -0
  49. package/blog/id/list_i18n_technologies/CMS/wordpress.md +188 -0
  50. package/blog/id/list_i18n_technologies/frameworks/angular.md +125 -0
  51. package/blog/id/list_i18n_technologies/frameworks/flutter.md +150 -0
  52. package/blog/id/list_i18n_technologies/frameworks/react-native.md +217 -0
  53. package/blog/id/list_i18n_technologies/frameworks/react.md +155 -0
  54. package/blog/id/list_i18n_technologies/frameworks/svelte.md +131 -0
  55. package/blog/id/list_i18n_technologies/frameworks/vue.md +130 -0
  56. package/blog/id/next-i18next_vs_next-intl_vs_intlayer.md +1500 -0
  57. package/blog/id/nextjs-multilingual-seo-comparison.md +361 -0
  58. package/blog/id/rag_powered_documentation_assistant.md +288 -0
  59. package/blog/id/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
  60. package/blog/id/vue-i18n_vs_intlayer.md +278 -0
  61. package/blog/id/what_is_internationalization.md +166 -0
  62. package/blog/it/i18n_using_next-i18next.md +1078 -0
  63. package/blog/it/i18n_using_next-intl.md +758 -0
  64. package/blog/it/intlayer_with_react-intl.md +0 -4
  65. package/blog/it/react-i18next_vs_react-intl_vs_intlayer.md +4 -0
  66. package/blog/it/vue-i18n_vs_intlayer.md +2 -0
  67. package/blog/ja/i18n_using_next-i18next.md +1078 -0
  68. package/blog/ja/i18n_using_next-intl.md +758 -0
  69. package/blog/ja/intlayer_with_react-intl.md +0 -4
  70. package/blog/ja/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  71. package/blog/ko/i18n_using_next-i18next.md +1075 -0
  72. package/blog/ko/i18n_using_next-intl.md +759 -0
  73. package/blog/ko/intlayer_with_react-intl.md +0 -4
  74. package/blog/ko/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  75. package/blog/pl/i18n_using_next-i18next.md +1078 -0
  76. package/blog/pl/i18n_using_next-intl.md +758 -0
  77. package/blog/pl/index.md +69 -0
  78. package/blog/pl/internationalization_and_SEO.md +363 -0
  79. package/blog/pl/intlayer_with_react-intl.md +0 -4
  80. package/blog/pl/list_i18n_technologies/CMS/drupal.md +143 -0
  81. package/blog/pl/list_i18n_technologies/CMS/wix.md +167 -0
  82. package/blog/pl/list_i18n_technologies/CMS/wordpress.md +196 -0
  83. package/blog/pl/list_i18n_technologies/frameworks/angular.md +125 -0
  84. package/blog/pl/list_i18n_technologies/frameworks/flutter.md +151 -0
  85. package/blog/pl/list_i18n_technologies/frameworks/react-native.md +217 -0
  86. package/blog/pl/list_i18n_technologies/frameworks/react.md +155 -0
  87. package/blog/pl/list_i18n_technologies/frameworks/svelte.md +131 -0
  88. package/blog/pl/list_i18n_technologies/frameworks/vue.md +130 -0
  89. package/blog/pl/next-i18next_vs_next-intl_vs_intlayer.md +1501 -0
  90. package/blog/pl/nextjs-multilingual-seo-comparison.md +362 -0
  91. package/blog/pl/rag_powered_documentation_assistant.md +288 -0
  92. package/blog/pl/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
  93. package/blog/pl/vue-i18n_vs_intlayer.md +278 -0
  94. package/blog/pl/what_is_internationalization.md +167 -0
  95. package/blog/pt/i18n_using_next-i18next.md +1067 -0
  96. package/blog/pt/i18n_using_next-intl.md +760 -0
  97. package/blog/pt/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  98. package/blog/ru/i18n_using_next-i18next.md +1106 -0
  99. package/blog/ru/i18n_using_next-intl.md +759 -0
  100. package/blog/ru/intlayer_with_react-intl.md +0 -4
  101. package/blog/ru/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  102. package/blog/tr/i18n_using_next-i18next.md +1078 -0
  103. package/blog/tr/i18n_using_next-intl.md +760 -0
  104. package/blog/tr/intlayer_with_react-intl.md +0 -4
  105. package/blog/tr/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  106. package/blog/tr/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
  107. package/blog/tr/vue-i18n_vs_intlayer.md +2 -0
  108. package/blog/vi/i18n_using_next-i18next.md +1080 -0
  109. package/blog/vi/i18n_using_next-intl.md +758 -0
  110. package/blog/vi/index.md +69 -0
  111. package/blog/vi/internationalization_and_SEO.md +363 -0
  112. package/blog/vi/intlayer_with_react-intl.md +0 -4
  113. package/blog/vi/list_i18n_technologies/CMS/drupal.md +143 -0
  114. package/blog/vi/list_i18n_technologies/CMS/wix.md +167 -0
  115. package/blog/vi/list_i18n_technologies/CMS/wordpress.md +188 -0
  116. package/blog/vi/list_i18n_technologies/frameworks/angular.md +125 -0
  117. package/blog/vi/list_i18n_technologies/frameworks/flutter.md +150 -0
  118. package/blog/vi/list_i18n_technologies/frameworks/react-native.md +217 -0
  119. package/blog/vi/list_i18n_technologies/frameworks/react.md +155 -0
  120. package/blog/vi/list_i18n_technologies/frameworks/svelte.md +131 -0
  121. package/blog/vi/list_i18n_technologies/frameworks/vue.md +130 -0
  122. package/blog/vi/next-i18next_vs_next-intl_vs_intlayer.md +1520 -0
  123. package/blog/vi/nextjs-multilingual-seo-comparison.md +362 -0
  124. package/blog/vi/rag_powered_documentation_assistant.md +288 -0
  125. package/blog/vi/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
  126. package/blog/vi/vue-i18n_vs_intlayer.md +278 -0
  127. package/blog/vi/what_is_internationalization.md +168 -0
  128. package/blog/zh/i18n_using_next-i18next.md +1105 -0
  129. package/blog/zh/i18n_using_next-intl.md +758 -0
  130. package/blog/zh/intlayer_with_react-intl.md +0 -4
  131. package/blog/zh/next-i18next_vs_next-intl_vs_intlayer.md +2 -0
  132. package/blog/zh/react-i18next_vs_react-intl_vs_intlayer.md +2 -0
  133. package/dist/cjs/common.cjs +0 -4
  134. package/dist/cjs/common.cjs.map +1 -1
  135. package/dist/cjs/generated/blog.entry.cjs +38 -6
  136. package/dist/cjs/generated/blog.entry.cjs.map +1 -1
  137. package/dist/cjs/generated/docs.entry.cjs +0 -6
  138. package/dist/cjs/generated/docs.entry.cjs.map +1 -1
  139. package/dist/cjs/generated/frequentQuestions.entry.cjs +0 -6
  140. package/dist/cjs/generated/frequentQuestions.entry.cjs.map +1 -1
  141. package/dist/cjs/generated/legal.entry.cjs +0 -6
  142. package/dist/cjs/generated/legal.entry.cjs.map +1 -1
  143. package/dist/esm/generated/blog.entry.mjs +38 -0
  144. package/dist/esm/generated/blog.entry.mjs.map +1 -1
  145. package/dist/types/generated/blog.entry.d.ts +2 -0
  146. package/dist/types/generated/blog.entry.d.ts.map +1 -1
  147. package/docs/ar/component_i18n.md +1 -1
  148. package/docs/ar/configuration.md +6 -0
  149. package/docs/ar/intlayer_cli.md +8 -3
  150. package/docs/ar/intlayer_with_next-i18next.md +619 -0
  151. package/docs/ar/intlayer_with_next-intl.md +446 -0
  152. package/docs/ar/intlayer_with_nextjs_16.md +21 -0
  153. package/docs/ar/intlayer_with_tanstack.md +4 -0
  154. package/docs/ar/intlayer_with_vite+react.md +4 -0
  155. package/docs/de/component_i18n.md +1 -1
  156. package/docs/de/configuration.md +6 -0
  157. package/docs/de/intlayer_cli.md +8 -3
  158. package/docs/de/intlayer_with_next-i18next.md +627 -0
  159. package/docs/de/intlayer_with_next-intl.md +451 -0
  160. package/docs/de/intlayer_with_nextjs_16.md +21 -0
  161. package/docs/de/intlayer_with_tanstack.md +4 -0
  162. package/docs/de/intlayer_with_vite+react.md +4 -0
  163. package/docs/en/component_i18n.md +1 -1
  164. package/docs/en/intlayer_cli.md +8 -1
  165. package/docs/en/intlayer_with_astro.md +10 -2
  166. package/docs/en/intlayer_with_create_react_app.md +8 -0
  167. package/docs/en/intlayer_with_lynx+react.md +8 -0
  168. package/docs/en/intlayer_with_nestjs.md +10 -0
  169. package/docs/en/intlayer_with_nextjs_14.md +10 -2
  170. package/docs/en/intlayer_with_nextjs_15.md +21 -4
  171. package/docs/en/intlayer_with_nextjs_16.md +17 -0
  172. package/docs/en/intlayer_with_nuxt.md +8 -0
  173. package/docs/en/intlayer_with_react_native+expo.md +10 -2
  174. package/docs/en/intlayer_with_react_router_v7.md +8 -0
  175. package/docs/en/intlayer_with_tanstack.md +10 -0
  176. package/docs/en/intlayer_with_vite+preact.md +10 -2
  177. package/docs/en/intlayer_with_vite+react.md +21 -4
  178. package/docs/en/intlayer_with_vite+vue.md +10 -2
  179. package/docs/en-GB/component_i18n.md +1 -1
  180. package/docs/en-GB/configuration.md +6 -0
  181. package/docs/en-GB/intlayer_cli.md +8 -3
  182. package/docs/en-GB/intlayer_with_angular.md +4 -4
  183. package/docs/en-GB/intlayer_with_express.md +4 -4
  184. package/docs/en-GB/intlayer_with_lynx+react.md +12 -12
  185. package/{blog/en/_intlayer_with_next-i18next.md → docs/en-GB/intlayer_with_next-i18next.md} +241 -42
  186. package/{blog/en/_intlayer_with_next-intl.md → docs/en-GB/intlayer_with_next-intl.md} +144 -29
  187. package/docs/en-GB/intlayer_with_nextjs_16.md +21 -0
  188. package/docs/en-GB/intlayer_with_tanstack.md +5 -1
  189. package/docs/en-GB/intlayer_with_vite+react.md +4 -0
  190. package/docs/en-GB/packages/next-intlayer/t.md +2 -2
  191. package/docs/es/component_i18n.md +1 -1
  192. package/docs/es/configuration.md +6 -0
  193. package/docs/es/intlayer_cli.md +8 -3
  194. package/docs/es/intlayer_with_next-i18next.md +628 -0
  195. package/docs/es/intlayer_with_next-intl.md +446 -0
  196. package/docs/es/intlayer_with_nextjs_16.md +21 -0
  197. package/docs/es/intlayer_with_tanstack.md +4 -0
  198. package/docs/es/intlayer_with_vite+react.md +4 -0
  199. package/docs/fr/configuration.md +6 -0
  200. package/docs/fr/intlayer_cli.md +8 -3
  201. package/docs/fr/intlayer_with_next-i18next.md +628 -0
  202. package/docs/fr/intlayer_with_next-intl.md +446 -0
  203. package/docs/fr/intlayer_with_nextjs_16.md +23 -2
  204. package/docs/fr/intlayer_with_tanstack.md +4 -0
  205. package/docs/fr/intlayer_with_vite+react.md +4 -0
  206. package/docs/hi/component_i18n.md +1 -1
  207. package/docs/hi/configuration.md +6 -0
  208. package/docs/hi/intlayer_cli.md +8 -0
  209. package/docs/hi/intlayer_with_next-i18next.md +628 -0
  210. package/docs/hi/intlayer_with_next-intl.md +446 -0
  211. package/docs/hi/intlayer_with_nextjs_16.md +21 -0
  212. package/docs/hi/intlayer_with_tanstack.md +4 -0
  213. package/docs/hi/intlayer_with_vite+react.md +4 -0
  214. package/docs/id/CI_CD.md +198 -0
  215. package/docs/id/autoFill.md +284 -0
  216. package/docs/id/component_i18n.md +186 -0
  217. package/docs/id/configuration.md +710 -0
  218. package/docs/id/dictionary/condition.md +231 -0
  219. package/docs/id/dictionary/content_file.md +1092 -0
  220. package/docs/id/dictionary/enumeration.md +245 -0
  221. package/docs/id/dictionary/file.md +237 -0
  222. package/docs/id/dictionary/function_fetching.md +214 -0
  223. package/docs/id/dictionary/gender.md +273 -0
  224. package/docs/id/dictionary/insertion.md +192 -0
  225. package/docs/id/dictionary/markdown.md +381 -0
  226. package/docs/id/dictionary/nesting.md +273 -0
  227. package/docs/id/dictionary/translation.md +310 -0
  228. package/docs/id/formatters.md +596 -0
  229. package/docs/id/how_works_intlayer.md +256 -0
  230. package/docs/id/index.md +176 -0
  231. package/docs/id/interest_of_intlayer.md +293 -0
  232. package/docs/id/intlayer_CMS.md +549 -0
  233. package/docs/id/intlayer_cli.md +850 -0
  234. package/docs/id/intlayer_visual_editor.md +288 -0
  235. package/docs/id/intlayer_with_angular.md +694 -0
  236. package/docs/id/intlayer_with_astro.md +252 -0
  237. package/docs/id/intlayer_with_create_react_app.md +1233 -0
  238. package/docs/id/intlayer_with_express.md +411 -0
  239. package/docs/id/intlayer_with_lynx+react.md +518 -0
  240. package/docs/id/intlayer_with_nestjs.md +272 -0
  241. package/docs/id/intlayer_with_next-i18next.md +628 -0
  242. package/docs/id/intlayer_with_next-intl.md +446 -0
  243. package/docs/id/intlayer_with_nextjs_14.md +1617 -0
  244. package/docs/id/intlayer_with_nextjs_15.md +1698 -0
  245. package/docs/id/intlayer_with_nextjs_16.md +21 -0
  246. package/docs/id/intlayer_with_nextjs_page_router.md +1478 -0
  247. package/docs/id/intlayer_with_nuxt.md +808 -0
  248. package/docs/id/intlayer_with_react_native+expo.md +699 -0
  249. package/docs/id/intlayer_with_react_router_v7.md +496 -0
  250. package/docs/id/intlayer_with_tanstack.md +564 -0
  251. package/docs/id/intlayer_with_vite+preact.md +1737 -0
  252. package/docs/id/intlayer_with_vite+react.md +1413 -0
  253. package/docs/id/intlayer_with_vite+solid.md +289 -0
  254. package/docs/id/intlayer_with_vite+svelte.md +289 -0
  255. package/docs/id/intlayer_with_vite+vue.md +1088 -0
  256. package/docs/id/introduction.md +218 -0
  257. package/docs/id/locale_mapper.md +242 -0
  258. package/docs/id/mcp_server.md +211 -0
  259. package/docs/id/packages/express-intlayer/t.md +458 -0
  260. package/docs/id/packages/intlayer/getConfiguration.md +145 -0
  261. package/docs/id/packages/intlayer/getEnumeration.md +159 -0
  262. package/docs/id/packages/intlayer/getHTMLTextDir.md +122 -0
  263. package/docs/id/packages/intlayer/getLocaleLang.md +81 -0
  264. package/docs/id/packages/intlayer/getLocaleName.md +119 -0
  265. package/docs/id/packages/intlayer/getLocalizedUrl.md +309 -0
  266. package/docs/id/packages/intlayer/getMultilingualUrls.md +223 -0
  267. package/docs/id/packages/intlayer/getPathWithoutLocale.md +75 -0
  268. package/docs/id/packages/intlayer/getTranslation.md +190 -0
  269. package/docs/id/packages/intlayer/getTranslationContent.md +188 -0
  270. package/docs/id/packages/next-intlayer/t.md +352 -0
  271. package/docs/id/packages/next-intlayer/useDictionary.md +271 -0
  272. package/docs/id/packages/next-intlayer/useIntlayer.md +264 -0
  273. package/docs/id/packages/next-intlayer/useLocale.md +166 -0
  274. package/docs/id/packages/react-intlayer/t.md +303 -0
  275. package/docs/id/packages/react-intlayer/useDictionary.md +287 -0
  276. package/docs/id/packages/react-intlayer/useI18n.md +267 -0
  277. package/docs/id/packages/react-intlayer/useIntlayer.md +254 -0
  278. package/docs/id/packages/react-intlayer/useLocale.md +210 -0
  279. package/docs/id/per_locale_file.md +323 -0
  280. package/docs/id/readme.md +261 -0
  281. package/docs/id/releases/v6.md +305 -0
  282. package/docs/id/roadmap.md +362 -0
  283. package/docs/id/testing.md +202 -0
  284. package/docs/id/vs_code_extension.md +126 -0
  285. package/docs/it/component_i18n.md +1 -1
  286. package/docs/it/configuration.md +6 -0
  287. package/docs/it/intlayer_cli.md +8 -3
  288. package/docs/it/intlayer_with_next-i18next.md +628 -0
  289. package/docs/it/intlayer_with_next-intl.md +446 -0
  290. package/docs/it/intlayer_with_nextjs_16.md +21 -0
  291. package/docs/it/intlayer_with_tanstack.md +4 -0
  292. package/docs/it/intlayer_with_vite+react.md +4 -0
  293. package/docs/ja/component_i18n.md +1 -1
  294. package/docs/ja/configuration.md +6 -0
  295. package/docs/ja/intlayer_cli.md +8 -3
  296. package/docs/ja/intlayer_with_next-i18next.md +627 -0
  297. package/docs/ja/intlayer_with_next-intl.md +446 -0
  298. package/docs/ja/intlayer_with_nextjs_16.md +21 -0
  299. package/docs/ja/intlayer_with_tanstack.md +4 -0
  300. package/docs/ja/intlayer_with_vite+react.md +4 -0
  301. package/docs/ko/configuration.md +6 -0
  302. package/docs/ko/intlayer_cli.md +8 -3
  303. package/docs/ko/intlayer_with_next-i18next.md +627 -0
  304. package/docs/ko/intlayer_with_next-intl.md +446 -0
  305. package/docs/ko/intlayer_with_nextjs_16.md +21 -0
  306. package/docs/ko/intlayer_with_tanstack.md +4 -0
  307. package/docs/ko/intlayer_with_vite+react.md +4 -0
  308. package/docs/pl/CI_CD.md +198 -0
  309. package/docs/pl/autoFill.md +284 -0
  310. package/docs/pl/component_i18n.md +186 -0
  311. package/docs/pl/configuration.md +710 -0
  312. package/docs/pl/dictionary/condition.md +232 -0
  313. package/docs/pl/dictionary/content_file.md +1130 -0
  314. package/docs/pl/dictionary/enumeration.md +245 -0
  315. package/docs/pl/dictionary/file.md +234 -0
  316. package/docs/pl/dictionary/function_fetching.md +214 -0
  317. package/docs/pl/dictionary/gender.md +276 -0
  318. package/docs/pl/dictionary/insertion.md +188 -0
  319. package/docs/pl/dictionary/markdown.md +408 -0
  320. package/docs/pl/dictionary/nesting.md +273 -0
  321. package/docs/pl/dictionary/translation.md +310 -0
  322. package/docs/pl/formatters.md +596 -0
  323. package/docs/pl/how_works_intlayer.md +256 -0
  324. package/docs/pl/index.md +176 -0
  325. package/docs/pl/interest_of_intlayer.md +291 -0
  326. package/docs/pl/intlayer_CMS.md +549 -0
  327. package/docs/pl/intlayer_cli.md +857 -0
  328. package/docs/pl/intlayer_visual_editor.md +288 -0
  329. package/docs/pl/intlayer_with_angular.md +690 -0
  330. package/docs/pl/intlayer_with_astro.md +280 -0
  331. package/docs/pl/intlayer_with_create_react_app.md +1235 -0
  332. package/docs/pl/intlayer_with_express.md +411 -0
  333. package/docs/pl/intlayer_with_lynx+react.md +518 -0
  334. package/docs/pl/intlayer_with_nestjs.md +272 -0
  335. package/docs/pl/intlayer_with_next-i18next.md +628 -0
  336. package/docs/pl/intlayer_with_next-intl.md +446 -0
  337. package/docs/pl/intlayer_with_nextjs_14.md +1594 -0
  338. package/docs/pl/intlayer_with_nextjs_15.md +1701 -0
  339. package/docs/pl/intlayer_with_nextjs_16.md +21 -0
  340. package/docs/pl/intlayer_with_nextjs_page_router.md +1513 -0
  341. package/docs/pl/intlayer_with_nuxt.md +885 -0
  342. package/docs/pl/intlayer_with_react_native+expo.md +698 -0
  343. package/docs/pl/intlayer_with_react_router_v7.md +503 -0
  344. package/docs/pl/intlayer_with_tanstack.md +562 -0
  345. package/docs/pl/intlayer_with_vite+preact.md +1736 -0
  346. package/docs/pl/intlayer_with_vite+react.md +1438 -0
  347. package/docs/pl/intlayer_with_vite+solid.md +290 -0
  348. package/docs/pl/intlayer_with_vite+svelte.md +289 -0
  349. package/docs/pl/intlayer_with_vite+vue.md +1116 -0
  350. package/docs/pl/introduction.md +209 -0
  351. package/docs/pl/locale_mapper.md +242 -0
  352. package/docs/pl/mcp_server.md +211 -0
  353. package/docs/pl/packages/express-intlayer/t.md +458 -0
  354. package/docs/pl/packages/intlayer/getConfiguration.md +146 -0
  355. package/docs/pl/packages/intlayer/getEnumeration.md +160 -0
  356. package/docs/pl/packages/intlayer/getHTMLTextDir.md +121 -0
  357. package/docs/pl/packages/intlayer/getLocaleLang.md +81 -0
  358. package/docs/pl/packages/intlayer/getLocaleName.md +118 -0
  359. package/docs/pl/packages/intlayer/getLocalizedUrl.md +300 -0
  360. package/docs/pl/packages/intlayer/getMultilingualUrls.md +221 -0
  361. package/docs/pl/packages/intlayer/getPathWithoutLocale.md +75 -0
  362. package/docs/pl/packages/intlayer/getTranslation.md +190 -0
  363. package/docs/pl/packages/intlayer/getTranslationContent.md +189 -0
  364. package/docs/pl/packages/next-intlayer/t.md +353 -0
  365. package/docs/pl/packages/next-intlayer/useDictionary.md +270 -0
  366. package/docs/pl/packages/next-intlayer/useIntlayer.md +263 -0
  367. package/docs/pl/packages/next-intlayer/useLocale.md +166 -0
  368. package/docs/pl/packages/react-intlayer/t.md +303 -0
  369. package/docs/pl/packages/react-intlayer/useDictionary.md +289 -0
  370. package/docs/pl/packages/react-intlayer/useI18n.md +249 -0
  371. package/docs/pl/packages/react-intlayer/useIntlayer.md +256 -0
  372. package/docs/pl/packages/react-intlayer/useLocale.md +210 -0
  373. package/docs/pl/per_locale_file.md +321 -0
  374. package/docs/pl/readme.md +261 -0
  375. package/docs/pl/releases/v6.md +305 -0
  376. package/docs/pl/roadmap.md +362 -0
  377. package/docs/pl/testing.md +202 -0
  378. package/docs/pl/vs_code_extension.md +126 -0
  379. package/docs/pt/component_i18n.md +1 -1
  380. package/docs/pt/configuration.md +6 -0
  381. package/docs/pt/intlayer_cli.md +8 -3
  382. package/docs/pt/intlayer_with_next-i18next.md +627 -0
  383. package/docs/pt/intlayer_with_next-intl.md +446 -0
  384. package/docs/pt/intlayer_with_nextjs_16.md +21 -0
  385. package/docs/pt/intlayer_with_tanstack.md +4 -0
  386. package/docs/pt/intlayer_with_vite+react.md +4 -0
  387. package/docs/ru/component_i18n.md +1 -1
  388. package/docs/ru/configuration.md +6 -0
  389. package/docs/ru/intlayer_cli.md +301 -22
  390. package/docs/ru/intlayer_with_next-i18next.md +629 -0
  391. package/docs/ru/intlayer_with_next-intl.md +448 -0
  392. package/docs/ru/intlayer_with_nextjs_16.md +21 -0
  393. package/docs/ru/intlayer_with_tanstack.md +4 -0
  394. package/docs/ru/intlayer_with_vite+react.md +4 -0
  395. package/docs/tr/component_i18n.md +1 -1
  396. package/docs/tr/configuration.md +6 -0
  397. package/docs/tr/intlayer_cli.md +8 -0
  398. package/docs/tr/intlayer_with_next-i18next.md +627 -0
  399. package/docs/tr/intlayer_with_next-intl.md +446 -0
  400. package/docs/tr/intlayer_with_nextjs_16.md +21 -0
  401. package/docs/tr/intlayer_with_tanstack.md +4 -0
  402. package/docs/tr/intlayer_with_vite+react.md +4 -0
  403. package/docs/vi/CI_CD.md +198 -0
  404. package/docs/vi/autoFill.md +284 -0
  405. package/docs/vi/component_i18n.md +186 -0
  406. package/docs/vi/configuration.md +710 -0
  407. package/docs/vi/dictionary/condition.md +237 -0
  408. package/docs/vi/dictionary/content_file.md +1115 -0
  409. package/docs/vi/dictionary/enumeration.md +255 -0
  410. package/docs/vi/dictionary/file.md +234 -0
  411. package/docs/vi/dictionary/function_fetching.md +212 -0
  412. package/docs/vi/dictionary/gender.md +275 -0
  413. package/docs/vi/dictionary/insertion.md +191 -0
  414. package/docs/vi/dictionary/markdown.md +381 -0
  415. package/docs/vi/dictionary/nesting.md +273 -0
  416. package/docs/vi/dictionary/translation.md +309 -0
  417. package/docs/vi/formatters.md +595 -0
  418. package/docs/vi/how_works_intlayer.md +256 -0
  419. package/docs/vi/index.md +174 -0
  420. package/docs/vi/interest_of_intlayer.md +292 -0
  421. package/docs/vi/intlayer_CMS.md +549 -0
  422. package/docs/vi/intlayer_cli.md +850 -0
  423. package/docs/vi/intlayer_visual_editor.md +288 -0
  424. package/docs/vi/intlayer_with_angular.md +692 -0
  425. package/docs/vi/intlayer_with_astro.md +252 -0
  426. package/docs/vi/intlayer_with_create_react_app.md +1230 -0
  427. package/docs/vi/intlayer_with_express.md +409 -0
  428. package/docs/vi/intlayer_with_lynx+react.md +520 -0
  429. package/docs/vi/intlayer_with_nestjs.md +272 -0
  430. package/docs/vi/intlayer_with_next-i18next.md +628 -0
  431. package/docs/vi/intlayer_with_next-intl.md +446 -0
  432. package/docs/vi/intlayer_with_nextjs_14.md +1584 -0
  433. package/docs/vi/intlayer_with_nextjs_15.md +1738 -0
  434. package/docs/vi/intlayer_with_nextjs_16.md +21 -0
  435. package/docs/vi/intlayer_with_nextjs_page_router.md +1504 -0
  436. package/docs/vi/intlayer_with_nuxt.md +821 -0
  437. package/docs/vi/intlayer_with_react_native+expo.md +700 -0
  438. package/docs/vi/intlayer_with_react_router_v7.md +498 -0
  439. package/docs/vi/intlayer_with_tanstack.md +562 -0
  440. package/docs/vi/intlayer_with_vite+preact.md +1722 -0
  441. package/docs/vi/intlayer_with_vite+react.md +1407 -0
  442. package/docs/vi/intlayer_with_vite+solid.md +287 -0
  443. package/docs/vi/intlayer_with_vite+svelte.md +289 -0
  444. package/docs/vi/intlayer_with_vite+vue.md +1071 -0
  445. package/docs/vi/introduction.md +215 -0
  446. package/docs/vi/locale_mapper.md +242 -0
  447. package/docs/vi/mcp_server.md +211 -0
  448. package/docs/vi/packages/express-intlayer/t.md +457 -0
  449. package/docs/vi/packages/intlayer/getConfiguration.md +145 -0
  450. package/docs/vi/packages/intlayer/getEnumeration.md +162 -0
  451. package/docs/vi/packages/intlayer/getHTMLTextDir.md +121 -0
  452. package/docs/vi/packages/intlayer/getLocaleLang.md +81 -0
  453. package/docs/vi/packages/intlayer/getLocaleName.md +129 -0
  454. package/docs/vi/packages/intlayer/getLocalizedUrl.md +309 -0
  455. package/docs/vi/packages/intlayer/getMultilingualUrls.md +221 -0
  456. package/docs/vi/packages/intlayer/getPathWithoutLocale.md +75 -0
  457. package/docs/vi/packages/intlayer/getTranslation.md +201 -0
  458. package/docs/vi/packages/intlayer/getTranslationContent.md +188 -0
  459. package/docs/vi/packages/next-intlayer/t.md +352 -0
  460. package/docs/vi/packages/next-intlayer/useDictionary.md +273 -0
  461. package/docs/vi/packages/next-intlayer/useIntlayer.md +264 -0
  462. package/docs/vi/packages/next-intlayer/useLocale.md +166 -0
  463. package/docs/vi/packages/react-intlayer/t.md +304 -0
  464. package/docs/vi/packages/react-intlayer/useDictionary.md +288 -0
  465. package/docs/vi/packages/react-intlayer/useI18n.md +295 -0
  466. package/docs/vi/packages/react-intlayer/useIntlayer.md +256 -0
  467. package/docs/vi/packages/react-intlayer/useLocale.md +210 -0
  468. package/docs/vi/per_locale_file.md +326 -0
  469. package/docs/vi/readme.md +261 -0
  470. package/docs/vi/releases/v6.md +305 -0
  471. package/docs/vi/roadmap.md +346 -0
  472. package/docs/vi/testing.md +202 -0
  473. package/docs/vi/vs_code_extension.md +126 -0
  474. package/docs/zh/configuration.md +6 -0
  475. package/docs/zh/intlayer_cli.md +8 -3
  476. package/docs/zh/intlayer_with_next-i18next.md +628 -0
  477. package/docs/zh/intlayer_with_next-intl.md +448 -0
  478. package/docs/zh/intlayer_with_nextjs_16.md +21 -0
  479. package/docs/zh/intlayer_with_tanstack.md +4 -0
  480. package/docs/zh/intlayer_with_vite+react.md +4 -0
  481. package/frequent_questions/ar/SSR_Next_no_[locale].md +1 -2
  482. package/frequent_questions/ar/array_as_content_declaration.md +1 -2
  483. package/frequent_questions/ar/build_dictionaries.md +1 -2
  484. package/frequent_questions/ar/build_error_CI_CD.md +1 -2
  485. package/frequent_questions/ar/bun_set_up.md +1 -2
  486. package/frequent_questions/ar/customized_locale_list.md +1 -2
  487. package/frequent_questions/ar/domain_routing.md +1 -2
  488. package/frequent_questions/ar/esbuild_error.md +1 -2
  489. package/frequent_questions/ar/get_locale_cookie.md +1 -2
  490. package/frequent_questions/ar/intlayer_command_undefined.md +1 -2
  491. package/frequent_questions/ar/locale_incorect_in_url.md +1 -2
  492. package/frequent_questions/ar/static_rendering.md +1 -3
  493. package/frequent_questions/ar/translated_path_url.md +1 -2
  494. package/frequent_questions/ar/unknown_command.md +1 -2
  495. package/frequent_questions/de/SSR_Next_no_[locale].md +1 -2
  496. package/frequent_questions/de/array_as_content_declaration.md +1 -2
  497. package/frequent_questions/de/build_dictionaries.md +1 -2
  498. package/frequent_questions/de/build_error_CI_CD.md +1 -2
  499. package/frequent_questions/de/bun_set_up.md +1 -2
  500. package/frequent_questions/de/customized_locale_list.md +1 -2
  501. package/frequent_questions/de/domain_routing.md +1 -2
  502. package/frequent_questions/de/esbuild_error.md +1 -2
  503. package/frequent_questions/de/get_locale_cookie.md +1 -2
  504. package/frequent_questions/de/intlayer_command_undefined.md +1 -2
  505. package/frequent_questions/de/locale_incorect_in_url.md +1 -2
  506. package/frequent_questions/de/static_rendering.md +1 -3
  507. package/frequent_questions/de/translated_path_url.md +1 -2
  508. package/frequent_questions/de/unknown_command.md +1 -2
  509. package/frequent_questions/en/SSR_Next_no_[locale].md +1 -2
  510. package/frequent_questions/en/array_as_content_declaration.md +1 -2
  511. package/frequent_questions/en/build_dictionaries.md +1 -2
  512. package/frequent_questions/en/build_error_CI_CD.md +1 -2
  513. package/frequent_questions/en/bun_set_up.md +1 -2
  514. package/frequent_questions/en/customized_locale_list.md +1 -2
  515. package/frequent_questions/en/domain_routing.md +1 -2
  516. package/frequent_questions/en/esbuild_error.md +1 -2
  517. package/frequent_questions/en/get_locale_cookie.md +1 -2
  518. package/frequent_questions/en/intlayer_command_undefined.md +1 -2
  519. package/frequent_questions/en/locale_incorect_in_url.md +1 -2
  520. package/frequent_questions/en/static_rendering.md +1 -3
  521. package/frequent_questions/en/translated_path_url.md +1 -2
  522. package/frequent_questions/en/unknown_command.md +1 -2
  523. package/frequent_questions/en-GB/SSR_Next_no_[locale].md +1 -2
  524. package/frequent_questions/en-GB/array_as_content_declaration.md +1 -2
  525. package/frequent_questions/en-GB/build_dictionaries.md +1 -2
  526. package/frequent_questions/en-GB/build_error_CI_CD.md +1 -2
  527. package/frequent_questions/en-GB/bun_set_up.md +1 -2
  528. package/frequent_questions/en-GB/customized_locale_list.md +1 -2
  529. package/frequent_questions/en-GB/domain_routing.md +1 -2
  530. package/frequent_questions/en-GB/esbuild_error.md +1 -2
  531. package/frequent_questions/en-GB/get_locale_cookie.md +1 -2
  532. package/frequent_questions/en-GB/intlayer_command_undefined.md +1 -2
  533. package/frequent_questions/en-GB/locale_incorect_in_url.md +1 -2
  534. package/frequent_questions/en-GB/static_rendering.md +1 -3
  535. package/frequent_questions/en-GB/translated_path_url.md +1 -2
  536. package/frequent_questions/en-GB/unknown_command.md +1 -2
  537. package/frequent_questions/es/SSR_Next_no_[locale].md +1 -2
  538. package/frequent_questions/es/array_as_content_declaration.md +1 -2
  539. package/frequent_questions/es/build_dictionaries.md +1 -2
  540. package/frequent_questions/es/build_error_CI_CD.md +1 -2
  541. package/frequent_questions/es/bun_set_up.md +1 -2
  542. package/frequent_questions/es/customized_locale_list.md +1 -2
  543. package/frequent_questions/es/domain_routing.md +1 -2
  544. package/frequent_questions/es/esbuild_error.md +1 -2
  545. package/frequent_questions/es/get_locale_cookie.md +1 -2
  546. package/frequent_questions/es/intlayer_command_undefined.md +1 -2
  547. package/frequent_questions/es/locale_incorect_in_url.md +1 -2
  548. package/frequent_questions/es/static_rendering.md +1 -3
  549. package/frequent_questions/es/translated_path_url.md +1 -2
  550. package/frequent_questions/es/unknown_command.md +1 -2
  551. package/frequent_questions/fr/SSR_Next_no_[locale].md +1 -2
  552. package/frequent_questions/fr/array_as_content_declaration.md +1 -2
  553. package/frequent_questions/fr/build_dictionaries.md +1 -2
  554. package/frequent_questions/fr/build_error_CI_CD.md +1 -2
  555. package/frequent_questions/fr/bun_set_up.md +1 -2
  556. package/frequent_questions/fr/customized_locale_list.md +1 -2
  557. package/frequent_questions/fr/domain_routing.md +1 -2
  558. package/frequent_questions/fr/esbuild_error.md +1 -2
  559. package/frequent_questions/fr/get_locale_cookie.md +1 -2
  560. package/frequent_questions/fr/intlayer_command_undefined.md +1 -2
  561. package/frequent_questions/fr/locale_incorect_in_url.md +1 -2
  562. package/frequent_questions/fr/static_rendering.md +1 -3
  563. package/frequent_questions/fr/translated_path_url.md +1 -2
  564. package/frequent_questions/fr/unknown_command.md +1 -2
  565. package/frequent_questions/hi/SSR_Next_no_[locale].md +1 -2
  566. package/frequent_questions/hi/array_as_content_declaration.md +1 -2
  567. package/frequent_questions/hi/build_dictionaries.md +1 -2
  568. package/frequent_questions/hi/build_error_CI_CD.md +1 -2
  569. package/frequent_questions/hi/bun_set_up.md +1 -2
  570. package/frequent_questions/hi/customized_locale_list.md +1 -2
  571. package/frequent_questions/hi/domain_routing.md +1 -2
  572. package/frequent_questions/hi/esbuild_error.md +1 -2
  573. package/frequent_questions/hi/get_locale_cookie.md +1 -2
  574. package/frequent_questions/hi/intlayer_command_undefined.md +1 -2
  575. package/frequent_questions/hi/locale_incorect_in_url.md +1 -2
  576. package/frequent_questions/hi/static_rendering.md +1 -3
  577. package/frequent_questions/hi/translated_path_url.md +1 -2
  578. package/frequent_questions/hi/unknown_command.md +1 -2
  579. package/frequent_questions/id/SSR_Next_no_[locale].md +104 -0
  580. package/frequent_questions/id/array_as_content_declaration.md +71 -0
  581. package/frequent_questions/id/build_dictionaries.md +58 -0
  582. package/frequent_questions/id/build_error_CI_CD.md +74 -0
  583. package/frequent_questions/id/bun_set_up.md +53 -0
  584. package/frequent_questions/id/customized_locale_list.md +64 -0
  585. package/frequent_questions/id/domain_routing.md +113 -0
  586. package/frequent_questions/id/esbuild_error.md +29 -0
  587. package/frequent_questions/id/get_locale_cookie.md +142 -0
  588. package/frequent_questions/id/intlayer_command_undefined.md +155 -0
  589. package/frequent_questions/id/locale_incorect_in_url.md +73 -0
  590. package/frequent_questions/id/static_rendering.md +44 -0
  591. package/frequent_questions/id/translated_path_url.md +55 -0
  592. package/frequent_questions/id/unknown_command.md +97 -0
  593. package/frequent_questions/it/SSR_Next_no_[locale].md +1 -2
  594. package/frequent_questions/it/array_as_content_declaration.md +1 -2
  595. package/frequent_questions/it/build_dictionaries.md +1 -2
  596. package/frequent_questions/it/build_error_CI_CD.md +1 -2
  597. package/frequent_questions/it/bun_set_up.md +1 -2
  598. package/frequent_questions/it/customized_locale_list.md +1 -2
  599. package/frequent_questions/it/domain_routing.md +1 -2
  600. package/frequent_questions/it/esbuild_error.md +1 -2
  601. package/frequent_questions/it/get_locale_cookie.md +1 -2
  602. package/frequent_questions/it/intlayer_command_undefined.md +1 -2
  603. package/frequent_questions/it/locale_incorect_in_url.md +1 -2
  604. package/frequent_questions/it/static_rendering.md +1 -3
  605. package/frequent_questions/it/translated_path_url.md +1 -2
  606. package/frequent_questions/it/unknown_command.md +1 -2
  607. package/frequent_questions/ja/SSR_Next_no_[locale].md +1 -2
  608. package/frequent_questions/ja/array_as_content_declaration.md +1 -2
  609. package/frequent_questions/ja/build_dictionaries.md +1 -2
  610. package/frequent_questions/ja/build_error_CI_CD.md +1 -2
  611. package/frequent_questions/ja/bun_set_up.md +1 -2
  612. package/frequent_questions/ja/customized_locale_list.md +1 -2
  613. package/frequent_questions/ja/domain_routing.md +1 -2
  614. package/frequent_questions/ja/esbuild_error.md +1 -2
  615. package/frequent_questions/ja/get_locale_cookie.md +1 -2
  616. package/frequent_questions/ja/intlayer_command_undefined.md +1 -2
  617. package/frequent_questions/ja/locale_incorect_in_url.md +1 -2
  618. package/frequent_questions/ja/static_rendering.md +1 -3
  619. package/frequent_questions/ja/translated_path_url.md +1 -2
  620. package/frequent_questions/ja/unknown_command.md +1 -2
  621. package/frequent_questions/ko/SSR_Next_no_[locale].md +1 -2
  622. package/frequent_questions/ko/array_as_content_declaration.md +1 -2
  623. package/frequent_questions/ko/build_dictionaries.md +1 -2
  624. package/frequent_questions/ko/build_error_CI_CD.md +1 -2
  625. package/frequent_questions/ko/bun_set_up.md +1 -2
  626. package/frequent_questions/ko/customized_locale_list.md +1 -2
  627. package/frequent_questions/ko/domain_routing.md +1 -2
  628. package/frequent_questions/ko/esbuild_error.md +1 -2
  629. package/frequent_questions/ko/get_locale_cookie.md +1 -2
  630. package/frequent_questions/ko/intlayer_command_undefined.md +1 -2
  631. package/frequent_questions/ko/locale_incorect_in_url.md +1 -2
  632. package/frequent_questions/ko/static_rendering.md +1 -3
  633. package/frequent_questions/ko/translated_path_url.md +1 -2
  634. package/frequent_questions/ko/unknown_command.md +1 -2
  635. package/frequent_questions/pl/SSR_Next_no_[locale].md +104 -0
  636. package/frequent_questions/pl/array_as_content_declaration.md +71 -0
  637. package/frequent_questions/pl/build_dictionaries.md +58 -0
  638. package/frequent_questions/pl/build_error_CI_CD.md +74 -0
  639. package/frequent_questions/pl/bun_set_up.md +54 -0
  640. package/frequent_questions/pl/customized_locale_list.md +64 -0
  641. package/frequent_questions/pl/domain_routing.md +113 -0
  642. package/frequent_questions/pl/esbuild_error.md +29 -0
  643. package/frequent_questions/pl/get_locale_cookie.md +142 -0
  644. package/frequent_questions/pl/intlayer_command_undefined.md +155 -0
  645. package/frequent_questions/pl/locale_incorect_in_url.md +73 -0
  646. package/frequent_questions/pl/static_rendering.md +44 -0
  647. package/frequent_questions/pl/translated_path_url.md +55 -0
  648. package/frequent_questions/pl/unknown_command.md +97 -0
  649. package/frequent_questions/pt/SSR_Next_no_[locale].md +1 -2
  650. package/frequent_questions/pt/array_as_content_declaration.md +1 -2
  651. package/frequent_questions/pt/build_dictionaries.md +1 -2
  652. package/frequent_questions/pt/build_error_CI_CD.md +1 -2
  653. package/frequent_questions/pt/bun_set_up.md +1 -2
  654. package/frequent_questions/pt/customized_locale_list.md +1 -2
  655. package/frequent_questions/pt/domain_routing.md +1 -2
  656. package/frequent_questions/pt/esbuild_error.md +1 -2
  657. package/frequent_questions/pt/get_locale_cookie.md +1 -2
  658. package/frequent_questions/pt/intlayer_command_undefined.md +1 -2
  659. package/frequent_questions/pt/locale_incorect_in_url.md +1 -2
  660. package/frequent_questions/pt/static_rendering.md +1 -3
  661. package/frequent_questions/pt/translated_path_url.md +1 -2
  662. package/frequent_questions/pt/unknown_command.md +1 -2
  663. package/frequent_questions/ru/SSR_Next_no_[locale].md +1 -2
  664. package/frequent_questions/ru/array_as_content_declaration.md +1 -2
  665. package/frequent_questions/ru/build_dictionaries.md +1 -2
  666. package/frequent_questions/ru/build_error_CI_CD.md +1 -2
  667. package/frequent_questions/ru/bun_set_up.md +1 -2
  668. package/frequent_questions/ru/customized_locale_list.md +1 -2
  669. package/frequent_questions/ru/domain_routing.md +1 -2
  670. package/frequent_questions/ru/esbuild_error.md +1 -2
  671. package/frequent_questions/ru/get_locale_cookie.md +1 -2
  672. package/frequent_questions/ru/intlayer_command_undefined.md +1 -2
  673. package/frequent_questions/ru/locale_incorect_in_url.md +1 -2
  674. package/frequent_questions/ru/static_rendering.md +1 -2
  675. package/frequent_questions/ru/translated_path_url.md +1 -2
  676. package/frequent_questions/ru/unknown_command.md +1 -2
  677. package/frequent_questions/tr/SSR_Next_no_[locale].md +1 -2
  678. package/frequent_questions/tr/array_as_content_declaration.md +1 -2
  679. package/frequent_questions/tr/build_dictionaries.md +1 -2
  680. package/frequent_questions/tr/build_error_CI_CD.md +1 -2
  681. package/frequent_questions/tr/bun_set_up.md +1 -2
  682. package/frequent_questions/tr/customized_locale_list.md +1 -2
  683. package/frequent_questions/tr/domain_routing.md +1 -2
  684. package/frequent_questions/tr/esbuild_error.md +1 -2
  685. package/frequent_questions/tr/get_locale_cookie.md +1 -2
  686. package/frequent_questions/tr/intlayer_command_undefined.md +1 -2
  687. package/frequent_questions/tr/locale_incorect_in_url.md +1 -2
  688. package/frequent_questions/tr/static_rendering.md +1 -2
  689. package/frequent_questions/tr/translated_path_url.md +1 -2
  690. package/frequent_questions/tr/unknown_command.md +1 -2
  691. package/frequent_questions/vi/SSR_Next_no_[locale].md +106 -0
  692. package/frequent_questions/vi/array_as_content_declaration.md +71 -0
  693. package/frequent_questions/vi/build_dictionaries.md +58 -0
  694. package/frequent_questions/vi/build_error_CI_CD.md +74 -0
  695. package/frequent_questions/vi/bun_set_up.md +53 -0
  696. package/frequent_questions/vi/customized_locale_list.md +64 -0
  697. package/frequent_questions/vi/domain_routing.md +113 -0
  698. package/frequent_questions/vi/esbuild_error.md +29 -0
  699. package/frequent_questions/vi/get_locale_cookie.md +142 -0
  700. package/frequent_questions/vi/intlayer_command_undefined.md +155 -0
  701. package/frequent_questions/vi/locale_incorect_in_url.md +73 -0
  702. package/frequent_questions/vi/static_rendering.md +44 -0
  703. package/frequent_questions/vi/translated_path_url.md +55 -0
  704. package/frequent_questions/vi/unknown_command.md +97 -0
  705. package/frequent_questions/zh/SSR_Next_no_[locale].md +1 -2
  706. package/frequent_questions/zh/array_as_content_declaration.md +1 -2
  707. package/frequent_questions/zh/build_dictionaries.md +1 -2
  708. package/frequent_questions/zh/build_error_CI_CD.md +1 -2
  709. package/frequent_questions/zh/bun_set_up.md +1 -2
  710. package/frequent_questions/zh/customized_locale_list.md +1 -2
  711. package/frequent_questions/zh/domain_routing.md +1 -2
  712. package/frequent_questions/zh/esbuild_error.md +1 -2
  713. package/frequent_questions/zh/get_locale_cookie.md +1 -2
  714. package/frequent_questions/zh/intlayer_command_undefined.md +1 -2
  715. package/frequent_questions/zh/locale_incorect_in_url.md +1 -2
  716. package/frequent_questions/zh/static_rendering.md +1 -3
  717. package/frequent_questions/zh/translated_path_url.md +1 -2
  718. package/frequent_questions/zh/unknown_command.md +1 -2
  719. package/legal/id/privacy_notice.md +83 -0
  720. package/legal/id/terms_of_service.md +55 -0
  721. package/legal/pl/privacy_notice.md +83 -0
  722. package/legal/pl/terms_of_service.md +55 -0
  723. package/legal/vi/privacy_notice.md +83 -0
  724. package/legal/vi/terms_of_service.md +55 -0
  725. package/package.json +19 -18
  726. package/src/generated/blog.entry.ts +38 -0
@@ -0,0 +1,1520 @@
1
+ ---
2
+ createdAt: 2025-08-23
3
+ updatedAt: 2025-09-29
4
+ title: next-i18next vs next-intl vs Intlayer
5
+ description: So sánh next-i18next với next-intl và Intlayer cho việc quốc tế hóa (i18n) của ứng dụng Next.js
6
+ keywords:
7
+ - next-intl
8
+ - next-i18next
9
+ - Intlayer
10
+ - Quốc tế hóa
11
+ - Blog
12
+ - Next.js
13
+ - JavaScript
14
+ - React
15
+ slugs:
16
+ - blog
17
+ - next-i18next-vs-next-intl-vs-intlayer
18
+ ---
19
+
20
+ # next-i18next VS next-intl VS intlayer | Quốc tế hóa Next.js (i18n)
21
+
22
+ ![next-i18next VS next-intl VS intlayer](https://github.com/aymericzip/intlayer/blob/main/docs/assets/i18next-next-intl-intlayer.png?raw=true)
23
+
24
+ Hãy cùng xem xét những điểm tương đồng và khác biệt giữa ba lựa chọn i18n cho Next.js: next-i18next, next-intl và Intlayer.
25
+
26
+ Đây không phải là một hướng dẫn đầy đủ. Đây là một so sánh giúp bạn lựa chọn.
27
+
28
+ Chúng tôi tập trung vào **Next.js 13+ App Router** (với **React Server Components**) và đánh giá:
29
+
30
+ <TOC/>
31
+
32
+ > **tóm tắt**: Cả ba đều có thể địa phương hóa một ứng dụng Next.js. Nếu bạn muốn **nội dung theo phạm vi component**, **kiểu TypeScript nghiêm ngặt**, **kiểm tra khóa thiếu trong thời gian build**, **từ điển được tree-shaking**, và **App Router + trợ giúp SEO hàng đầu**, thì **Intlayer** là lựa chọn toàn diện và hiện đại nhất.
33
+
34
+ > Một sự nhầm lẫn thường gặp của các nhà phát triển là nghĩ rằng `next-intl` là phiên bản Next.js của `react-intl`. Không phải vậy, `next-intl` được duy trì bởi [Amann](https://github.com/amannn), trong khi `react-intl` được duy trì bởi [FormatJS](https://github.com/formatjs/formatjs).
35
+
36
+ ---
37
+
38
+ ## Tóm tắt ngắn gọn
39
+
40
+ - **next-intl** - Định dạng thông điệp nhẹ, đơn giản với hỗ trợ Next.js vững chắc. Các catalog tập trung là phổ biến; trải nghiệm nhà phát triển (DX) đơn giản, nhưng an toàn và bảo trì quy mô lớn phần lớn vẫn là trách nhiệm của bạn.
41
+ - **next-i18next** - i18next trong bộ dạng Next.js. Hệ sinh thái trưởng thành và các tính năng qua plugin (ví dụ: ICU), nhưng cấu hình có thể dài dòng và các catalog có xu hướng tập trung khi dự án phát triển.
42
+ - **Intlayer** - Mô hình nội dung tập trung vào component cho Next.js, **kiểu TypeScript nghiêm ngặt**, **kiểm tra trong thời gian build**, **tree-shaking**, **middleware tích hợp & trợ giúp SEO**, tùy chọn **Visual Editor/CMS**, và **dịch thuật hỗ trợ AI**.
43
+
44
+ ---
45
+
46
+ | Library | GitHub Stars | Total Commits | Last Commit | First Version | NPM Version | NPM Downloads |
47
+ | ---------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | ----------------------------------------------------------------------------------------------------------------------------------------------------- | ------------- | ------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------ |
48
+ | `aymericzip/intlayer` | [![GitHub Repo stars](https://img.shields.io/github/stars/aymericzip/intlayer?style=for-the-badge&label=%E2%AD%90%20stars)](https://github.com/aymericzip/intlayer/stargazers) | [![GitHub commit activity](https://img.shields.io/github/commit-activity/t/aymericzip/intlayer?style=for-the-badge&label=commits)](https://github.com/aymericzip/intlayer/commits) | [![Last Commit](https://img.shields.io/github/last-commit/aymericzip/intlayer?style=for-the-badge)](https://github.com/aymericzip/intlayer/commits) | April 2024 | [![npm](https://img.shields.io/npm/v/intlayer?style=for-the-badge)](https://www.npmjs.com/package/intlayer) | [![npm downloads](https://img.shields.io/npm/dm/intlayer?style=for-the-badge)](https://www.npmjs.com/package/intlayer) |
49
+ | `amannn/next-intl` | [![GitHub Repo stars](https://img.shields.io/github/stars/amannn/next-intl?style=for-the-badge&label=%E2%AD%90%20stars)](https://github.com/amannn/next-intl/stargazers) | [![GitHub commit activity](https://img.shields.io/github/commit-activity/t/amannn/next-intl?style=for-the-badge&label=commits)](https://github.com/amannn/next-intl/commits) | [![Last Commit](https://img.shields.io/github/last-commit/amannn/next-intl?style=for-the-badge)](https://github.com/amannn/next-intl/commits) | Nov 2020 | [![npm](https://img.shields.io/npm/v/next-intl?style=for-the-badge)](https://www.npmjs.com/package/next-intl) | [![npm downloads](https://img.shields.io/npm/dm/next-intl?style=for-the-badge)](https://www.npmjs.com/package/next-intl) |
50
+ | `i18next/i18next` | [![GitHub Repo stars](https://img.shields.io/github/stars/i18next/i18next?style=for-the-badge&label=%E2%AD%90%20stars)](https://github.com/i18next/i18next/stargazers) | [![GitHub commit activity](https://img.shields.io/github/commit-activity/t/i18next/i18next?style=for-the-badge&label=commits)](https://github.com/i18next/i18next/commits) | [![Last Commit](https://img.shields.io/github/last-commit/i18next/i18next?style=for-the-badge)](https://github.com/i18next/i18next/commits) | Jan 2012 | [![npm](https://img.shields.io/npm/v/i18next?style=for-the-badge)](https://www.npmjs.com/package/i18next) | [![npm downloads](https://img.shields.io/npm/dm/i18next?style=for-the-badge)](https://www.npmjs.com/package/i18next) |
51
+ | `i18next/next-i18next` | [![GitHub Repo stars](https://img.shields.io/github/stars/i18next/next-i18next?style=for-the-badge&label=%E2%AD%90%20stars)](https://github.com/i18next/next-i18next/stargazers) | [![GitHub commit activity](https://img.shields.io/github/commit-activity/t/i18next/next-i18next?style=for-the-badge&label=commits)](https://github.com/i18next/next-i18next/commits) | [![Last Commit](https://img.shields.io/github/last-commit/i18next/next-i18next?style=for-the-badge)](https://github.com/i18next/next-i18next/commits) | Nov 2018 | [![npm](https://img.shields.io/npm/v/next-i18next?style=for-the-badge)](https://www.npmjs.com/package/next-i18next) | [![npm downloads](https://img.shields.io/npm/dm/next-i18next?style=for-the-badge)](https://www.npmjs.com/package/next-i18next) |
52
+
53
+ > Các huy hiệu được cập nhật tự động. Các ảnh chụp nhanh sẽ thay đổi theo thời gian.
54
+
55
+ ---
56
+
57
+ ## So sánh Tính năng Song song (Tập trung vào Next.js)
58
+
59
+ | Tính năng | `next-intlayer` (Intlayer) | `next-intl` | `next-i18next` |
60
+ | ------------------------------------------------ | ----------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------------- |
61
+ | **Bản dịch Gần Thành phần** | ✅ Có, nội dung được đặt gần với từng thành phần | ❌ Không | ❌ Không |
62
+ | **Tích hợp TypeScript** | ✅ Nâng cao, tự động tạo kiểu nghiêm ngặt | ✅ Tốt | ⚠️ Cơ bản |
63
+ | **Phát hiện bản dịch thiếu** | ✅ Tô sáng lỗi TypeScript và cảnh báo/lỗi trong thời gian biên dịch | ⚠️ Dự phòng thời gian chạy | ⚠️ Dự phòng thời gian chạy |
64
+ | **Nội dung phong phú (JSX/Markdown/components)** | ✅ Hỗ trợ trực tiếp | ❌ Không thiết kế cho các node phong phú | ⚠️ Hạn chế |
65
+ | **Dịch thuật hỗ trợ AI** | ✅ Có, hỗ trợ nhiều nhà cung cấp AI. Có thể sử dụng bằng API key của bạn. Xem xét ngữ cảnh ứng dụng và phạm vi nội dung | ❌ Không | ❌ Không |
66
+ | **Trình chỉnh sửa trực quan** | ✅ Có, Trình chỉnh sửa trực quan cục bộ + CMS tùy chọn; có thể tách nội dung codebase ra ngoài; có thể nhúng | ❌ Không / có sẵn qua các nền tảng bản địa hóa bên ngoài | ❌ Không / có sẵn qua các nền tảng bản địa hóa bên ngoài |
67
+ | **Định tuyến bản địa hóa** | ✅ Có, hỗ trợ các đường dẫn bản địa hóa sẵn có (hoạt động với Next.js & Vite) | ✅ Tích hợp sẵn, App Router hỗ trợ phân đoạn `[locale]` | ✅ Tích hợp sẵn |
68
+ | **Tạo Đường Dẫn Động** | ✅ Có | ✅ Có | ✅ Có |
69
+ | **Phân số nhiều** | ✅ Mẫu dựa trên liệt kê | ✅ Tốt | ✅ Tốt |
70
+ | **Định dạng (ngày tháng, số, tiền tệ)** | ✅ Bộ định dạng tối ưu (Intl ở tầng dưới) | ✅ Tốt (trợ giúp Intl) | ✅ Tốt (trợ giúp Intl) |
71
+ | **Định dạng nội dung** | ✅ .tsx, .ts, .js, .json, .md, .txt, (.yaml đang phát triển) | ✅ .json, .js, .ts | ⚠️ .json |
72
+ | **Hỗ trợ ICU** | ⚠️ Đang phát triển | ✅ Có | ⚠️ Qua plugin (`i18next-icu`) |
73
+ | **Trợ giúp SEO (hreflang, sitemap)** | ✅ Công cụ tích hợp sẵn: trợ giúp cho sitemap, robots.txt, metadata | ✅ Tốt | ✅ Tốt |
74
+ | **Hệ sinh thái / Cộng đồng** | ⚠️ Nhỏ hơn nhưng đang phát triển nhanh và phản ứng tốt | ✅ Tốt | ✅ Tốt |
75
+ | **Kết xuất phía máy chủ & Thành phần máy chủ** | ✅ Có, tối ưu cho SSR / React Server Components | ⚠️ Hỗ trợ ở cấp trang nhưng cần truyền các hàm t trên cây thành phần cho các thành phần máy chủ con | ⚠️ Hỗ trợ ở cấp trang nhưng cần truyền các hàm t trên cây thành phần cho các thành phần máy chủ con |
76
+ | **Tree-shaking (chỉ tải nội dung được sử dụng)** | ✅ Có, theo từng component tại thời điểm build thông qua các plugin Babel/SWC | ⚠️ Một phần | ⚠️ Một phần |
77
+ | **Tải lười (Lazy loading)** | ✅ Có, theo từng locale / từng từ điển | ✅ Có (theo từng route/theo từng locale), cần quản lý namespace | ✅ Có (theo từng route/theo từng locale), cần quản lý namespace |
78
+ | **Loại bỏ nội dung không sử dụng** | ✅ Có, theo từ điển tại thời điểm build | ❌ Không, có thể quản lý thủ công bằng cách quản lý namespace | ❌ Không, có thể quản lý thủ công bằng cách quản lý namespace |
79
+ | **Quản lý dự án lớn** | ✅ Khuyến khích mô-đun, phù hợp với hệ thống thiết kế | ✅ Mô-đun với thiết lập | ✅ Mô-đun với thiết lập |
80
+ | **Kiểm tra bản dịch thiếu (CLI/CI)** | ✅ CLI: `npx intlayer content test` (kiểm tra thân thiện với CI) | ⚠️ Không tích hợp sẵn; tài liệu đề xuất `npx @lingual/i18n-check` | ⚠️ Không tích hợp sẵn; dựa vào công cụ i18next / runtime `saveMissing` |
81
+
82
+ ---
83
+
84
+ ## Giới thiệu
85
+
86
+ Next.js cung cấp hỗ trợ tích hợp cho routing quốc tế hóa (ví dụ: các đoạn locale). Nhưng tính năng đó không tự động thực hiện việc dịch thuật. Bạn vẫn cần một thư viện để hiển thị nội dung đã được bản địa hóa cho người dùng.
87
+
88
+ Có nhiều thư viện i18n tồn tại, nhưng trong thế giới Next.js hiện nay, có ba thư viện đang được ưa chuộng: next-i18next, next-intl và Intlayer.
89
+
90
+ ---
91
+
92
+ ## Kiến trúc & khả năng mở rộng
93
+
94
+ - **next-intl / next-i18next**: Mặc định sử dụng **danh mục tập trung** theo từng locale (cộng với **namespace** trong i18next). Hoạt động tốt ban đầu, nhưng thường trở thành một bề mặt chia sẻ lớn với sự phụ thuộc ngày càng tăng và sự thay đổi nhiều của các key.
95
+ - **Intlayer**: Khuyến khích sử dụng từ điển **theo từng component** (hoặc theo từng tính năng) **đặt cùng vị trí** với mã nguồn mà chúng phục vụ. Điều này giảm tải nhận thức, dễ dàng sao chép/di chuyển các phần UI, và giảm xung đột giữa các nhóm. Nội dung không sử dụng cũng dễ dàng được phát hiện và loại bỏ.
96
+
97
+ **Tại sao điều này quan trọng:** Trong các codebase lớn hoặc các thiết lập hệ thống thiết kế, **nội dung mô-đun** có khả năng mở rộng tốt hơn so với các danh mục đơn khối.
98
+
99
+ ---
100
+
101
+ ## Kích thước gói & phụ thuộc
102
+
103
+ Sau khi xây dựng ứng dụng, bundle là JavaScript mà trình duyệt sẽ tải để hiển thị trang. Do đó, kích thước bundle rất quan trọng đối với hiệu suất ứng dụng.
104
+
105
+ Có hai thành phần quan trọng trong bối cảnh bundle của ứng dụng đa ngôn ngữ:
106
+
107
+ - Mã ứng dụng
108
+ - Nội dung được trình duyệt tải
109
+
110
+ ## Mã ứng dụng
111
+
112
+ Tầm quan trọng của mã ứng dụng trong trường hợp này là rất nhỏ. Cả ba giải pháp đều hỗ trợ tree-shaking, nghĩa là các phần mã không sử dụng sẽ không được bao gồm trong bundle.
113
+
114
+ Dưới đây là so sánh kích thước bundle JavaScript được trình duyệt tải cho một ứng dụng đa ngôn ngữ với ba giải pháp.
115
+
116
+ Nếu chúng ta không cần bất kỳ bộ định dạng nào trong ứng dụng, danh sách các hàm được xuất sau khi tree-shaking sẽ là:
117
+
118
+ - **next-intlayer**: `useIntlayer`, `useLocale`, `NextIntlClientProvider`, (Kích thước bundle là 180.6 kB -> 78.6 kB (gzip))
119
+ - **next-intl**: `useTranslations`, `useLocale`, `NextIntlClientProvider`, (Kích thước bundle là 101.3 kB -> 31.4 kB (gzip))
120
+ - **next-i18next**: `useTranslation`, `useI18n`, `I18nextProvider`, (Kích thước bundle là 80.7 kB -> 25.5 kB (gzip))
121
+
122
+ Các hàm này chỉ là các wrapper quanh React context/state, vì vậy tổng ảnh hưởng của thư viện i18n lên kích thước bundle là rất nhỏ.
123
+
124
+ > Intlayer hơi lớn hơn một chút so với `next-intl` và `next-i18next` vì nó bao gồm nhiều logic hơn trong hàm `useIntlayer`. Điều này liên quan đến tích hợp markdown và `intlayer-editor`.
125
+
126
+ ## Nội dung và Bản dịch
127
+
128
+ Phần này thường bị các nhà phát triển bỏ qua, nhưng hãy xem xét trường hợp một ứng dụng gồm 10 trang với 10 ngôn ngữ. Giả sử mỗi trang chứa 100% nội dung duy nhất để đơn giản hóa phép tính (trong thực tế, nhiều nội dung bị trùng lặp giữa các trang, ví dụ: tiêu đề trang, đầu trang, chân trang, v.v.).
129
+
130
+ Một người dùng muốn truy cập trang `/fr/about` sẽ tải nội dung của một trang trong một ngôn ngữ nhất định. Bỏ qua việc tối ưu hóa nội dung có nghĩa là tải tới 8.200% `((1 + (((10 trang - 1) × (10 ngôn ngữ - 1)))) × 100)` nội dung của ứng dụng một cách không cần thiết. Bạn có thấy vấn đề không? Ngay cả khi nội dung này chỉ là văn bản, và trong khi bạn có thể ưu tiên tối ưu hóa hình ảnh trên trang web của mình, bạn đang gửi đi nội dung thừa khắp toàn cầu và khiến máy tính của người dùng phải xử lý nó một cách vô ích.
131
+
132
+ Hai vấn đề quan trọng:
133
+
134
+ - **Phân tách theo route:**
135
+
136
+ > Nếu tôi đang ở trang `/about`, tôi không muốn tải nội dung của trang `/home`
137
+
138
+ - **Phân tách theo locale:**
139
+
140
+ > Nếu tôi đang ở trang `/fr/about`, tôi không muốn tải nội dung của trang `/en/about`
141
+
142
+ Một lần nữa, cả ba giải pháp đều nhận thức được những vấn đề này và cho phép quản lý các tối ưu hóa này. Sự khác biệt giữa ba giải pháp là trải nghiệm nhà phát triển (DX).
143
+
144
+ `next-intl` và `next-i18next` sử dụng phương pháp tập trung để quản lý bản dịch, cho phép phân tách JSON theo locale và theo các tệp con. Trong `next-i18next`, chúng ta gọi các tệp JSON là 'namespaces'; `next-intl` cho phép khai báo các messages. Trong `intlayer`, chúng ta gọi các tệp JSON là 'dictionaries'.
145
+
146
+ - Trong trường hợp của `next-intl`, giống như `next-i18next`, nội dung được tải ở cấp độ trang/bố cục, sau đó nội dung này được tải vào một context provider. Điều này có nghĩa là nhà phát triển phải tự quản lý các file JSON sẽ được tải cho mỗi trang.
147
+
148
+ > Trong thực tế, điều này ngụ ý rằng các nhà phát triển thường bỏ qua tối ưu hóa này, ưu tiên tải toàn bộ nội dung trong context provider của trang để đơn giản.
149
+
150
+ - Trong trường hợp của `intlayer`, toàn bộ nội dung được tải trong ứng dụng. Sau đó một plugin (`@intlayer/babel` / `@intlayer/swc`) sẽ đảm nhiệm việc tối ưu gói bằng cách chỉ tải nội dung được sử dụng trên trang. Do đó, nhà phát triển không cần phải tự quản lý các từ điển sẽ được tải. Điều này cho phép tối ưu tốt hơn, dễ bảo trì hơn và giảm thời gian phát triển.
151
+
152
+ Khi ứng dụng phát triển (đặc biệt khi nhiều nhà phát triển cùng làm việc trên ứng dụng), việc quên xóa nội dung không còn sử dụng trong các tệp JSON là điều thường gặp.
153
+
154
+ > Lưu ý rằng tất cả JSON đều được tải trong mọi trường hợp (next-intl, next-i18next, intlayer).
155
+
156
+ Đây là lý do tại sao cách tiếp cận của Intlayer hiệu quả hơn: nếu một component không còn được sử dụng, từ điển của nó sẽ không được tải vào bundle.
157
+
158
+ Cách thư viện xử lý fallback cũng rất quan trọng. Giả sử ứng dụng mặc định là tiếng Anh, và người dùng truy cập trang `/fr/about`. Nếu bản dịch tiếng Pháp bị thiếu, chúng ta sẽ sử dụng fallback tiếng Anh.
159
+
160
+ Trong trường hợp của `next-intl` và `next-i18next`, thư viện yêu cầu tải JSON liên quan đến locale hiện tại, nhưng cũng phải tải JSON của locale dự phòng. Do đó, giả sử tất cả nội dung đã được dịch, mỗi trang sẽ tải 100% nội dung không cần thiết. **Ngược lại, `intlayer` xử lý fallback ngay trong thời gian xây dựng từ điển. Vì vậy, mỗi trang sẽ chỉ tải nội dung được sử dụng.**
161
+
162
+ > Lưu ý: Để tối ưu gói bundle sử dụng `intlayer`, bạn cần thiết lập tùy chọn `importMode: 'dynamic'` trong file `intlayer.config.ts` của bạn. Và đảm bảo plugin `@intlayer/babel` / `@intlayer/swc` đã được cài đặt (được cài đặt mặc định khi sử dụng `vite-intlayer`).
163
+
164
+ Dưới đây là ví dụ về tác động của việc tối ưu kích thước bundle sử dụng `intlayer` trong ứng dụng vite + react:
165
+
166
+ | Gói tối ưu hóa | Gói không tối ưu hóa |
167
+ | --------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------- |
168
+ | ![gói tối ưu hóa](https://github.com/aymericzip/intlayer/blob/main/docs/assets/bundle.png?raw=true) | ![gói không tối ưu hóa](https://github.com/aymericzip/intlayer/blob/main/docs/assets/bundle_no_optimization.png?raw=true) |
169
+
170
+ ---
171
+
172
+ ## TypeScript & an toàn
173
+
174
+ <Columns>
175
+ <Column>
176
+
177
+ **next-i18next**
178
+
179
+ - Kiểu cơ bản cho các hook. **kiểu khóa nghiêm ngặt yêu cầu công cụ/cấu hình bổ sung**.
180
+
181
+ </Column>
182
+ <Column>
183
+
184
+ **next-intl**
185
+
186
+ - Hỗ trợ TypeScript vững chắc, nhưng **các khóa không được kiểu nghiêm ngặt theo mặc định**. bạn sẽ duy trì các mẫu an toàn một cách thủ công.
187
+
188
+ </Column>
189
+ <Column>
190
+
191
+ **intlayer**
192
+
193
+ - **Tạo kiểu nghiêm ngặt** từ nội dung của bạn. **Tự động hoàn thành trong IDE** và **lỗi thời gian biên dịch** phát hiện lỗi chính tả và khóa thiếu trước khi triển khai.
194
+
195
+ </Column>
196
+ </Columns>
197
+
198
+ **Tại sao điều này quan trọng:** Kiểu mạnh giúp chuyển lỗi sang bên **trái** (CI/build) thay vì bên **phải** (runtime).
199
+
200
+ ---
201
+
202
+ ## Xử lý dịch thiếu
203
+
204
+ <Columns>
205
+ <Column>
206
+
207
+ **next-i18next**
208
+
209
+ - Dựa vào **fallback thời gian chạy**. Build không bị lỗi.
210
+
211
+ </Column>
212
+ <Column>
213
+
214
+ **next-intl**
215
+
216
+ - Dựa vào **fallback thời gian chạy**. Build không bị lỗi.
217
+
218
+ </Column>
219
+ <Column>
220
+
221
+ **intlayer**
222
+
223
+ - **Phát hiện trong thời gian build** với **cảnh báo/lỗi** cho các locale hoặc key bị thiếu.
224
+
225
+ </Column>
226
+ </Columns>
227
+
228
+ **Tại sao điều này quan trọng:** Phát hiện thiếu sót trong quá trình build giúp ngăn chặn các chuỗi 'undefined' xuất hiện trong môi trường production.
229
+
230
+ ---
231
+
232
+ ## Định tuyến, middleware & chiến lược URL
233
+
234
+ <Columns>
235
+ <Column>
236
+
237
+ **next-i18next**
238
+
239
+ - Cho phép định tuyến theo ngôn ngữ. Nhưng middleware không được tích hợp sẵn.
240
+
241
+ </Column>
242
+ <Column>
243
+
244
+ **next-intl**
245
+
246
+ - Cho phép định tuyến theo ngôn ngữ.
247
+ - Cung cấp middleware.
248
+
249
+ </Column>
250
+ <Column>
251
+
252
+ **intlayer**
253
+
254
+ - Cho phép định tuyến theo ngôn ngữ.
255
+ - Cung cấp middleware.
256
+
257
+ </Column>
258
+ </Columns>
259
+
260
+ **Tại sao điều này quan trọng:** Giúp cải thiện SEO và khả năng khám phá, cũng như trải nghiệm người dùng.
261
+
262
+ ---
263
+
264
+ ## Đồng bộ với Server Components (RSC)
265
+
266
+ <Columns>
267
+ <Column>
268
+
269
+ **next-i18next**
270
+
271
+ - Hỗ trợ các server component cho trang và layout.
272
+ - Không cung cấp API đồng bộ cho các thành phần server con.
273
+
274
+ </Column>
275
+ <Column>
276
+
277
+ **next-intl**
278
+
279
+ - Hỗ trợ các thành phần server trang và bố cục.
280
+ - Không cung cấp API đồng bộ cho các thành phần server con.
281
+
282
+ </Column>
283
+ <Column>
284
+
285
+ **intlayer**
286
+
287
+ - Hỗ trợ các thành phần server trang và bố cục.
288
+ - Cung cấp API đồng bộ cho các thành phần server con.
289
+
290
+ </Column>
291
+ </Columns>
292
+
293
+ **Tại sao điều này quan trọng:** Hỗ trợ thành phần server là một tính năng then chốt của Next.js 13+, giúp cải thiện hiệu suất. Việc truyền props như locale hoặc hàm `t` từ thành phần cha xuống các thành phần server con làm cho các thành phần của bạn kém tái sử dụng hơn.
294
+
295
+ ---
296
+
297
+ ## Tích hợp với các nền tảng bản địa hóa (TMS)
298
+
299
+ Các tổ chức lớn thường dựa vào Hệ thống Quản lý Dịch thuật (TMS) như **Crowdin**, **Phrase**, **Lokalise**, **Localizely**, hoặc **Localazy**.
300
+
301
+ - **Tại sao các công ty quan tâm**
302
+ - **Hợp tác & vai trò**: Có nhiều bên tham gia: nhà phát triển, quản lý sản phẩm, người dịch, người đánh giá, đội ngũ marketing.
303
+ - **Quy mô & hiệu quả**: dịch thuật liên tục, đánh giá trong ngữ cảnh.
304
+
305
+ - **next-intl / next-i18next**
306
+ - Thường sử dụng **danh mục JSON tập trung**, nên việc xuất/nhập với TMS rất đơn giản.
307
+ - Hệ sinh thái trưởng thành và có ví dụ/tích hợp cho các nền tảng trên.
308
+
309
+ - **Intlayer**
310
+ - Khuyến khích **từ điển phân tán, theo từng component** và hỗ trợ nội dung **TypeScript/TSX/JS/JSON/MD**.
311
+ - Điều này cải thiện tính mô-đun trong mã, nhưng có thể làm cho việc tích hợp TMS dạng plug-and-play trở nên khó khăn hơn khi một công cụ mong đợi các tệp JSON phẳng, tập trung.
312
+ - Intlayer cung cấp các lựa chọn thay thế: **dịch thuật hỗ trợ AI** (sử dụng khóa nhà cung cấp của bạn), một **Trình chỉnh sửa trực quan/CMS**, và các quy trình làm việc **CLI/CI** để phát hiện và điền trước các khoảng trống.
313
+
314
+ > Lưu ý: `next-intl` và `i18next` cũng chấp nhận các catalog TypeScript. Nếu nhóm của bạn lưu trữ các thông điệp trong các tệp `.ts` hoặc phân quyền chúng theo tính năng, bạn có thể gặp phải sự cản trở tương tự với TMS. Tuy nhiên, nhiều thiết lập `next-intl` vẫn tập trung trong thư mục `locales/`, điều này giúp việc chuyển đổi sang JSON cho TMS dễ dàng hơn một chút.
315
+
316
+ ---
317
+
318
+ ## Trải nghiệm nhà phát triển
319
+
320
+ Phần này thực hiện so sánh sâu giữa ba giải pháp. Thay vì xem xét các trường hợp đơn giản, như được mô tả trong tài liệu 'bắt đầu' cho mỗi giải pháp, chúng ta sẽ xem xét một trường hợp sử dụng thực tế, tương tự hơn với một dự án thực tế.
321
+
322
+ ### Cấu trúc ứng dụng
323
+
324
+ Cấu trúc ứng dụng rất quan trọng để đảm bảo khả năng bảo trì tốt cho codebase của bạn.
325
+
326
+ <Tab defaultTab="next-intl" group='techno'>
327
+
328
+ <TabItem label="next-i18next" value="next-i18next">
329
+
330
+ ```bash
331
+ .
332
+ ├── i18n.config.ts
333
+ └── src
334
+ ├── locales
335
+ │ ├── en
336
+ │ │ ├── common.json
337
+ │ │ └── about.json
338
+ │ └── fr
339
+ │ ├── common.json
340
+ │ └── about.json
341
+ ├── app
342
+ │ ├── i18n
343
+ │ │ └── server.ts
344
+ │ └── [locale]
345
+ │ ├── layout.tsx
346
+ │ └── about.tsx
347
+ └── components
348
+ ├── I18nProvider.tsx
349
+ ├── ClientComponent.tsx
350
+ └── ServerComponent.tsx
351
+ ```
352
+
353
+ </TabItem>
354
+ <TabItem label="next-intl" value="next-intl">
355
+
356
+ ```bash
357
+ .
358
+ ├── i18n.ts
359
+ ├── locales
360
+ │ ├── en
361
+ │ │ ├── home.json
362
+ │ │ └── navbar.json
363
+ │ ├── fr
364
+ │ │ ├── home.json
365
+ │ │ └── navbar.json
366
+ │ └── es
367
+ │ ├── home.json
368
+ │ └── navbar.json
369
+ └── src
370
+ ├── middleware.ts
371
+ ├── app
372
+ │ ├── i18n
373
+ │ │ └── server.ts
374
+ │ └── [locale]
375
+ │ └── home.tsx
376
+ └── components
377
+ └── Navbar
378
+ └── index.tsx
379
+ ```
380
+
381
+ </TabItem>
382
+ <TabItem label="intlayer" value="intlayer">
383
+
384
+ ```bash
385
+ .
386
+ ├── intlayer.config.ts
387
+ └── src
388
+ ├── middleware.ts
389
+ ├── app
390
+ │ └── [locale]
391
+ │ ├── layout.tsx
392
+ │ └── home
393
+ │ ├── index.tsx
394
+ │ └── index.content.ts
395
+ └── components
396
+ └── Navbar
397
+ ├── index.tsx
398
+ └── index.content.ts
399
+ ```
400
+
401
+ </TabItem>
402
+ </Tab>
403
+
404
+ #### So sánh
405
+
406
+ - **next-intl / next-i18next**: Danh mục tập trung (JSON; namespaces/messages). Cấu trúc rõ ràng, tích hợp tốt với các nền tảng dịch thuật, nhưng có thể dẫn đến nhiều chỉnh sửa chéo file khi ứng dụng phát triển.
407
+ - **Intlayer**: Từ điển `.content.{ts|js|json}` theo từng component, đặt cùng vị trí với component. Dễ dàng tái sử dụng component và suy luận cục bộ; thêm các file và dựa vào công cụ xây dựng thời gian biên dịch.
408
+
409
+ #### Cài đặt và Tải Nội dung
410
+
411
+ Như đã đề cập trước đó, bạn phải tối ưu cách mỗi file JSON được nhập vào code của bạn.
412
+ Cách thư viện xử lý việc tải nội dung rất quan trọng.
413
+
414
+ <Tab defaultTab="next-intl" group='techno'>
415
+ <TabItem label="next-i18next" value="next-i18next">
416
+
417
+ ```ts fileName="i18n.config.ts"
418
+ export const locales = ["en", "fr"] as const;
419
+ export type Locale = (typeof locales)[number];
420
+
421
+ export const defaultLocale: Locale = "en";
422
+
423
+ export const rtlLocales = ["ar", "he", "fa", "ur"] as const;
424
+ export const isRtl = (locale: string) =>
425
+ (rtlLocales as readonly string[]).includes(locale);
426
+
427
+ export function localizedPath(locale: string, path: string) {
428
+ // Trả về đường dẫn có locale nếu khác defaultLocale
429
+ return locale === defaultLocale ? path : "/" + locale + path;
430
+ }
431
+
432
+ const ORIGIN = "https://example.com";
433
+ export function abs(locale: string, path: string) {
434
+ // Trả về đường dẫn tuyệt đối với origin và đường dẫn đã được localize
435
+ return ORIGIN + localizedPath(locale, path);
436
+ }
437
+ ```
438
+
439
+ ```ts fileName="src/app/i18n/server.ts"
440
+ import { createInstance } from "i18next";
441
+ import { initReactI18next } from "react-i18next/initReactI18next";
442
+ import resourcesToBackend from "i18next-resources-to-backend";
443
+ import { defaultLocale } from "@/i18n.config";
444
+
445
+ // Tải tài nguyên JSON từ src/locales/<locale>/<namespace>.json
446
+ const backend = resourcesToBackend(
447
+ (locale: string, namespace: string) =>
448
+ import(`../../locales/${locale}/${namespace}.json`)
449
+ );
450
+
451
+ export async function initI18next(
452
+ locale: string,
453
+ namespaces: string[] = ["common"]
454
+ ) {
455
+ const i18n = createInstance();
456
+ await i18n
457
+ .use(initReactI18next)
458
+ .use(backend)
459
+ .init({
460
+ lng: locale,
461
+ fallbackLng: defaultLocale,
462
+ ns: namespaces,
463
+ defaultNS: "common",
464
+ interpolation: { escapeValue: false },
465
+ react: { useSuspense: false },
466
+ });
467
+ return i18n;
468
+ }
469
+ ```
470
+
471
+ ```tsx fileName="src/components/I18nProvider.tsx"
472
+ "use client";
473
+
474
+ import * as React from "react";
475
+ import { I18nextProvider } from "react-i18next";
476
+ import { createInstance } from "i18next";
477
+ import { initReactI18next } from "react-i18next/initReactI18next";
478
+ import resourcesToBackend from "i18next-resources-to-backend";
479
+ import { defaultLocale } from "@/i18n.config";
480
+
481
+ const backend = resourcesToBackend(
482
+ (locale: string, namespace: string) =>
483
+ import(`../../locales/${locale}/${namespace}.json`)
484
+ );
485
+
486
+ type Props = {
487
+ locale: string;
488
+ namespaces?: string[];
489
+ resources?: Record<string, any>; // { ns: bundle } // { ns: gói tài nguyên }
490
+ children: React.ReactNode;
491
+ };
492
+
493
+ export default function I18nProvider({
494
+ locale,
495
+ namespaces = ["common"],
496
+ resources,
497
+ children,
498
+ }: Props) {
499
+ const [i18n] = React.useState(() => {
500
+ const i = createInstance();
501
+
502
+ i.use(initReactI18next)
503
+ .use(backend)
504
+ .init({
505
+ lng: locale,
506
+ fallbackLng: defaultLocale,
507
+ ns: namespaces,
508
+ resources: resources ? { [locale]: resources } : undefined,
509
+ defaultNS: "common",
510
+ interpolation: { escapeValue: false },
511
+ react: { useSuspense: false },
512
+ });
513
+
514
+ return i;
515
+ });
516
+
517
+ return <I18nextProvider i18n={i18n}>{children}</I18nextProvider>;
518
+ }
519
+ ```
520
+
521
+ ```tsx fileName="src/app/[locale]/layout.tsx"
522
+ import type { ReactNode } from "react";
523
+ import { locales, defaultLocale, isRtl, type Locale } from "@/i18n.config";
524
+
525
+ export const dynamicParams = false;
526
+
527
+ export function generateStaticParams() {
528
+ return locales.map((locale) => ({ locale }));
529
+ }
530
+
531
+ export default function LocaleLayout({
532
+ children,
533
+ params,
534
+ }: {
535
+ children: ReactNode;
536
+ params: { locale: string };
537
+ }) {
538
+ const locale: Locale = (locales as readonly string[]).includes(params.locale)
539
+ ? (params.locale as any)
540
+ : defaultLocale;
541
+
542
+ const dir = isRtl(locale) ? "rtl" : "ltr";
543
+
544
+ return (
545
+ <html lang={locale} dir={dir}>
546
+ <body>{children}</body>
547
+ </html>
548
+ );
549
+ }
550
+ ```
551
+
552
+ ```tsx fileName="src/app/[locale]/about.tsx"
553
+ import I18nProvider from "@/components/I18nProvider";
554
+ import { initI18next } from "@/app/i18n/server";
555
+ import type { Locale } from "@/i18n.config";
556
+ import ClientComponent from "@/components/ClientComponent";
557
+ import ServerComponent from "@/components/ServerComponent";
558
+
559
+ // Ép buộc render tĩnh cho trang
560
+ export const dynamic = "force-static";
561
+
562
+ export default async function AboutPage({
563
+ params: { locale },
564
+ }: {
565
+ params: { locale: Locale };
566
+ }) {
567
+ const namespaces = ["common", "about"] as const;
568
+
569
+ const i18n = await initI18next(locale, [...namespaces]);
570
+ const tAbout = i18n.getFixedT(locale, "about");
571
+
572
+ return (
573
+ <I18nProvider locale={locale} namespaces={[...namespaces]}>
574
+ <main>
575
+ <h1>{tAbout("title")}</h1>
576
+
577
+ <ClientComponent />
578
+ <ServerComponent t={tAbout} locale={locale} count={0} />
579
+ </main>
580
+ </I18nProvider>
581
+ );
582
+ }
583
+ ```
584
+
585
+ </TabItem>
586
+ <TabItem label="next-intl" value="next-intl">
587
+
588
+ ```tsx fileName="src/i18n.ts"
589
+ import { getRequestConfig } from "next-intl/server";
590
+ import { notFound } from "next/navigation";
591
+
592
+ export const locales = ["en", "fr", "es"] as const;
593
+ export const defaultLocale = "en" as const;
594
+
595
+ async function loadMessages(locale: string) {
596
+ // Chỉ tải các namespace mà layout/trang của bạn cần
597
+ const [common, about] = await Promise.all([
598
+ import(`../locales/${locale}/common.json`).then((m) => m.default),
599
+ import(`../locales/${locale}/about.json`).then((m) => m.default),
600
+ ]);
601
+
602
+ return { common, about } as const;
603
+ }
604
+
605
+ export default getRequestConfig(async ({ locale }) => {
606
+ if (!locales.includes(locale as any)) notFound();
607
+
608
+ return {
609
+ messages: await loadMessages(locale),
610
+ };
611
+ });
612
+ ```
613
+
614
+ ```tsx fileName="src/app/[locale]/layout.tsx"
615
+ import type { ReactNode } from "react";
616
+ import { locales } from "@/i18n";
617
+ import {
618
+ getLocaleDirection,
619
+ unstable_setRequestLocale,
620
+ } from "next-intl/server";
621
+
622
+ export const dynamic = "force-static";
623
+
624
+ export function generateStaticParams() {
625
+ return locales.map((locale) => ({ locale }));
626
+ }
627
+
628
+ export default async function LocaleLayout({
629
+ children,
630
+ params,
631
+ }: {
632
+ children: ReactNode;
633
+ params: Promise<{ locale: string }>;
634
+ }) {
635
+ const { locale } = await params;
636
+
637
+ // Đặt locale yêu cầu đang hoạt động cho lần render server này (RSC)
638
+ unstable_setRequestLocale(locale);
639
+
640
+ const dir = getLocaleDirection(locale);
641
+
642
+ return (
643
+ <html lang={locale} dir={dir}>
644
+ <body>{children}</body>
645
+ </html>
646
+ );
647
+ }
648
+ ```
649
+
650
+ ```tsx fileName="src/app/[locale]/about/page.tsx"
651
+ import { getTranslations, getMessages, getFormatter } from "next-intl/server";
652
+ import { NextIntlClientProvider } from "next-intl";
653
+ import pick from "lodash/pick";
654
+ import ServerComponent from "@/components/ServerComponent";
655
+ import ClientComponentExample from "@/components/ClientComponentExample";
656
+
657
+ export const dynamic = "force-static";
658
+
659
+ export default async function AboutPage({
660
+ params,
661
+ }: {
662
+ params: Promise<{ locale: string }>;
663
+ }) {
664
+ const { locale } = await params;
665
+
666
+ // Các thông điệp được tải phía server. Chỉ đẩy những gì cần thiết cho client.
667
+ const messages = await getMessages();
668
+ const clientMessages = pick(messages, ["common", "about"]);
669
+
670
+ // Dịch/định dạng nghiêm ngặt phía server
671
+ const tAbout = await getTranslations("about");
672
+ const tCounter = await getTranslations("about.counter");
673
+ const format = await getFormatter();
674
+
675
+ const initialFormattedCount = format.number(0);
676
+
677
+ return (
678
+ <NextIntlClientProvider locale={locale} messages={clientMessages}>
679
+ <main>
680
+ <h1>{tAbout("title")}</h1>
681
+ <ClientComponentExample />
682
+ <ServerComponent
683
+ formattedCount={initialFormattedCount}
684
+ label={tCounter("label")}
685
+ increment={tCounter("increment")}
686
+ />
687
+ </main>
688
+ </NextIntlClientProvider>
689
+ );
690
+ }
691
+ ```
692
+
693
+ </TabItem>
694
+ <TabItem label="intlayer" value="intlayer">
695
+
696
+ ```tsx fileName="intlayer.config.ts"
697
+ import { type IntlayerConfig, Locales } from "intlayer";
698
+
699
+ const config: IntlayerConfig = {
700
+ internationalization: {
701
+ locales: [Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH],
702
+ defaultLocale: Locales.ENGLISH,
703
+ },
704
+ };
705
+
706
+ export default config;
707
+ ```
708
+
709
+ ```tsx fileName="src/app/[locale]/layout.tsx"
710
+ import { getHTMLTextDir } from "intlayer";
711
+ import {
712
+ IntlayerClientProvider,
713
+ generateStaticParams,
714
+ type NextLayoutIntlayer,
715
+ } from "next-intlayer";
716
+
717
+ export const dynamic = "force-static";
718
+
719
+ const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
720
+ const { locale } = await params;
721
+
722
+ return (
723
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
724
+ <body>
725
+ <IntlayerClientProvider locale={locale}>
726
+ {children}
727
+ </IntlayerClientProvider>
728
+ </body>
729
+ </html>
730
+ );
731
+ };
732
+
733
+ export default LandingLayout;
734
+ ```
735
+
736
+ ```tsx fileName="src/app/[locale]/about/page.tsx"
737
+ import { PageContent } from "@components/PageContent";
738
+ import type { NextPageIntlayer } from "next-intlayer";
739
+ import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
740
+ import { ClientComponent, ServerComponent } from "@components";
741
+
742
+ const LandingPage: NextPageIntlayer = async ({ params }) => {
743
+ const { locale } = await params;
744
+ const { title } = useIntlayer("about", locale);
745
+
746
+ return (
747
+ <IntlayerServerProvider locale={locale}>
748
+ <main>
749
+ <h1>{title}</h1>
750
+ <ClientComponent />
751
+ <ServerComponent />
752
+ </main>
753
+ </IntlayerServerProvider>
754
+ );
755
+ };
756
+
757
+ export default LandingPage;
758
+ ```
759
+
760
+ </TabItem>
761
+ </Tab>
762
+
763
+ #### So sánh
764
+
765
+ Cả ba đều hỗ trợ tải nội dung và providers theo từng locale.
766
+
767
+ - Với **next-intl/next-i18next**, bạn thường tải các messages/namespace được chọn theo từng route và đặt providers ở nơi cần thiết.
768
+
769
+ - Với **Intlayer**, thêm phân tích tại thời điểm build để suy luận việc sử dụng, điều này có thể giảm thiểu việc cấu hình thủ công và cho phép sử dụng một provider gốc duy nhất.
770
+
771
+ Chọn giữa kiểm soát rõ ràng và tự động hóa dựa trên sở thích của nhóm.
772
+
773
+ ### Sử dụng trong một component phía client
774
+
775
+ Hãy lấy ví dụ về một component phía client hiển thị bộ đếm.
776
+
777
+ <Tab defaultTab="next-intl" group='techno'>
778
+ <TabItem label="next-i18next" value="next-i18next">
779
+
780
+ **Bản dịch (mỗi namespace là một file JSON dưới `src/locales/...`)**
781
+
782
+ ```json fileName="src/locales/en/about.json"
783
+ {
784
+ "title": "About",
785
+ "description": "About page description",
786
+ "counter": {
787
+ "label": "Counter",
788
+ "increment": "Increment"
789
+ }
790
+ }
791
+ ```
792
+
793
+ ```json fileName="src/locales/fr/about.json"
794
+ {
795
+ "title": "À propos",
796
+ "description": "Description de la page À propos",
797
+ "counter": {
798
+ "label": "Compteur",
799
+ "increment": "Incrémenter"
800
+ }
801
+ }
802
+ ```
803
+
804
+ **Component phía client (chỉ tải namespace cần thiết)**
805
+
806
+ ```tsx fileName="src/components/ClientComponent.tsx"
807
+ "use client";
808
+
809
+ import React, { useState } from "react";
810
+ import { useTranslation } from "react-i18next";
811
+
812
+ const ClientComponent = () => {
813
+ const { t, i18n } = useTranslation("about");
814
+ const [count, setCount] = useState(0);
815
+
816
+ const numberFormat = new Intl.NumberFormat(i18n.language); // Định dạng số theo ngôn ngữ hiện tại
817
+
818
+ return (
819
+ <div>
820
+ <p>{numberFormat.format(count)}</p>
821
+ <button
822
+ aria-label={t("counter.label")}
823
+ onClick={() => setCount((c) => c + 1)}
824
+ >
825
+ {t("counter.increment")}
826
+ </button>
827
+ </div>
828
+ );
829
+ };
830
+
831
+ export default ClientComponent;
832
+ ```
833
+
834
+ > Đảm bảo trang/provider chỉ bao gồm các namespace bạn cần (ví dụ: `about`).
835
+ > Nếu bạn sử dụng React < 19, hãy ghi nhớ các formatter nặng như `Intl.NumberFormat`.
836
+
837
+ </TabItem>
838
+ <TabItem label="next-intl" value="next-intl">
839
+
840
+ **Bản dịch (dạng dữ liệu được tái sử dụng; tải chúng vào các thông điệp next-intl theo cách bạn muốn)**
841
+
842
+ ```json fileName="locales/vi/about.json"
843
+ {
844
+ "counter": {
845
+ "label": "Bộ đếm",
846
+ "increment": "Tăng"
847
+ }
848
+ }
849
+ ```
850
+
851
+ ```json fileName="locales/fr/about.json"
852
+ {
853
+ "counter": {
854
+ "label": "Compteur",
855
+ "increment": "Incrémenter"
856
+ }
857
+ }
858
+ ```
859
+
860
+ **Component phía client**
861
+
862
+ ```tsx fileName="src/components/ClientComponentExample.tsx"
863
+ "use client";
864
+
865
+ import React, { useState } from "react";
866
+ import { useTranslations, useFormatter } from "next-intl";
867
+
868
+ const ClientComponentExample = () => {
869
+ // Phạm vi trực tiếp đến đối tượng lồng nhau
870
+ const t = useTranslations("about.counter");
871
+ const format = useFormatter();
872
+ const [count, setCount] = useState(0);
873
+
874
+ return (
875
+ <div>
876
+ <p>{format.number(count)}</p>
877
+ <button
878
+ aria-label={t("label")}
879
+ onClick={() => setCount((count) => count + 1)}
880
+ >
881
+ {t("increment")}
882
+ </button>
883
+ </div>
884
+ );
885
+ };
886
+ ```
887
+
888
+ > Đừng quên thêm thông điệp "about" vào thông điệp client của trang
889
+
890
+ </TabItem>
891
+ <TabItem label="intlayer" value="intlayer">
892
+
893
+ **Nội dung**
894
+
895
+ ```ts fileName="src/components/ClientComponentExample/index.content.ts"
896
+ import { t, type Dictionary } from "intlayer";
897
+
898
+ const counterContent = {
899
+ key: "counter",
900
+ content: {
901
+ label: t({ en: "Counter", fr: "Compteur" }),
902
+ increment: t({ en: "Increment", fr: "Incrémenter" }),
903
+ },
904
+ } satisfies Dictionary;
905
+
906
+ export default counterContent;
907
+ ```
908
+
909
+ **Component phía client**
910
+
911
+ ```tsx fileName="src/components/ClientComponentExample/index.tsx"
912
+ "use client";
913
+
914
+ import React, { useState } from "react";
915
+ import { useNumber, useIntlayer } from "next-intlayer";
916
+
917
+ const ClientComponentExample = () => {
918
+ const [count, setCount] = useState(0);
919
+ const { label, increment } = useIntlayer("counter"); // trả về chuỗi
920
+ const { number } = useNumber();
921
+
922
+ return (
923
+ <div>
924
+ <p>{number(count)}</p>
925
+ <button aria-label={label} onClick={() => setCount((count) => count + 1)}>
926
+ {increment}
927
+ </button>
928
+ </div>
929
+ );
930
+ };
931
+ ```
932
+
933
+ </TabItem>
934
+ </Tab>
935
+
936
+ #### So sánh
937
+
938
+ - **Định dạng số**
939
+ - **next-i18next**: không có `useNumber`; sử dụng `Intl.NumberFormat` (hoặc i18next-icu).
940
+ - **next-intl**: `useFormatter().number(value)`.
941
+ - **Intlayer**: tích hợp sẵn `useNumber()`.
942
+
943
+ - **Khóa (Keys)**
944
+ - Giữ cấu trúc lồng nhau (`about.counter.label`) và phạm vi hook của bạn tương ứng (`useTranslation("about")` + `t("counter.label")` hoặc `useTranslations("about.counter")` + `t("label")`).
945
+
946
+ - **Vị trí file**
947
+ - **next-i18next** yêu cầu JSON ở `public/locales/{lng}/{ns}.json`.
948
+ - **next-intl** linh hoạt; tải thông điệp theo cách bạn cấu hình.
949
+ - **Intlayer** lưu nội dung trong các từ điển TS/JS và giải quyết theo key.
950
+
951
+ ---
952
+
953
+ ### Sử dụng trong một server component
954
+
955
+ Chúng ta sẽ lấy ví dụ về một component giao diện người dùng (UI). Component này là một server component, và nên có khả năng được chèn như một con của client component. (page (server component) -> client component -> server component). Vì component này có thể được chèn như một con của client component, nó không thể là async.
956
+
957
+ <Tab defaultTab="next-intl" group='techno'>
958
+ <TabItem label="next-i18next" value="next-i18next">
959
+
960
+ ```tsx fileName="src/components/ServerComponent.tsx"
961
+ type ServerComponentProps = {
962
+ t: (key: string) => string; // hàm dịch theo key
963
+ locale: string; // ngôn ngữ hiện tại
964
+ count: number; // số đếm
965
+ };
966
+
967
+ const ServerComponent = ({ t, locale, count }: ServerComponentProps) => {
968
+ const formatted = new Intl.NumberFormat(locale).format(count); // định dạng số theo locale
969
+
970
+ return (
971
+ <div>
972
+ <p>{formatted}</p>
973
+ <button aria-label={t("counter.label")}>{t("counter.increment")}</button>
974
+ </div>
975
+ );
976
+ };
977
+
978
+ export default ServerComponent;
979
+ ```
980
+
981
+ </TabItem>
982
+ <TabItem label="next-intl" value="next-intl">
983
+
984
+ ```tsx fileName="src/components/ServerComponent.tsx"
985
+ type ServerComponentProps = {
986
+ t: (key: string) => string; // hàm dịch theo key
987
+ locale: string; // ngôn ngữ hiện tại
988
+ count: number; // số đếm
989
+ formatter: Intl.NumberFormat;
990
+ };
991
+
992
+ const ServerComponent = ({
993
+ t,
994
+ locale,
995
+ count,
996
+ formatter,
997
+ }: ServerComponentProps) => {
998
+ const formatted = formatter.format(count);
999
+
1000
+ return (
1001
+ <div>
1002
+ <p>{formatted}</p>
1003
+ <button aria-label={t("counter.label")}>{t("counter.increment")}</button>
1004
+ </div>
1005
+ );
1006
+ };
1007
+
1008
+ export default ServerComponent;
1009
+ ```
1010
+
1011
+ > Vì component phía server không thể là async, bạn cần truyền các bản dịch và hàm formatter dưới dạng props.
1012
+ >
1013
+ > Trong trang / layout của bạn:
1014
+ >
1015
+ > - `import { getTranslations, getFormatter } from "next-intl/server";`
1016
+ > - `const t = await getTranslations("about.counter");`
1017
+ > - `const formatter = await getFormatter().then((formatter) => formatter.number());`
1018
+
1019
+ </TabItem>
1020
+ <TabItem label="intlayer" value="intlayer">
1021
+
1022
+ ```tsx fileName="src/components/ServerComponent.tsx"
1023
+ import { useIntlayer, useNumber } from "next-intlayer/server";
1024
+
1025
+ type ServerComponentProps = {
1026
+ count: number;
1027
+ };
1028
+
1029
+ const ServerComponent = ({ count }: ServerComponentProps) => {
1030
+ const { label, increment } = useIntlayer("counter");
1031
+ const { number } = useNumber();
1032
+
1033
+ return (
1034
+ <div>
1035
+ <p>{number(count)}</p>
1036
+ <button aria-label={label}>{increment}</button>
1037
+ </div>
1038
+ );
1039
+ };
1040
+ ```
1041
+
1042
+ </TabItem>
1043
+ </Tab>
1044
+
1045
+ > Intlayer cung cấp các hook **an toàn cho server** thông qua `next-intlayer/server`. Để hoạt động, `useIntlayer` và `useNumber` sử dụng cú pháp giống hook, tương tự như các hook phía client, nhưng dựa vào ngữ cảnh server (`IntlayerServerProvider`) ở bên dưới.
1046
+
1047
+ ### Metadata / Sitemap / Robots
1048
+
1049
+ Dịch nội dung là điều tuyệt vời. Nhưng mọi người thường quên rằng mục tiêu chính của quốc tế hóa là làm cho trang web của bạn trở nên dễ nhìn thấy hơn trên toàn thế giới. I18n là một đòn bẩy tuyệt vời để cải thiện khả năng hiển thị trang web của bạn.
1050
+
1051
+ Dưới đây là danh sách các thực hành tốt liên quan đến SEO đa ngôn ngữ.
1052
+
1053
+ - đặt thẻ meta hreflang trong thẻ `<head>`
1054
+ > Nó giúp các công cụ tìm kiếm hiểu được những ngôn ngữ nào có trên trang
1055
+ - liệt kê tất cả các bản dịch trang trong sitemap.xml sử dụng schema XML `http://www.w3.org/1999/xhtml`
1056
+ >
1057
+ - đừng quên loại trừ các trang có tiền tố khỏi robots.txt (ví dụ: `/dashboard`, và `/fr/dashboard`, `/es/dashboard`)
1058
+ >
1059
+ - sử dụng component Link tùy chỉnh để chuyển hướng đến trang được địa phương hóa nhất (ví dụ: bằng tiếng Pháp `<a href="/fr/about">A propos</a>`)
1060
+ >
1061
+
1062
+ Các nhà phát triển thường quên tham chiếu đúng các trang của họ theo từng ngôn ngữ.
1063
+
1064
+ <Tab defaultTab="next-intl" group='techno'>
1065
+
1066
+ <TabItem label="next-i18next" value="next-i18next">
1067
+
1068
+ ```ts fileName="i18n.config.ts"
1069
+ export const locales = ["en", "fr"] as const;
1070
+ export type Locale = (typeof locales)[number];
1071
+ export const defaultLocale: Locale = "en";
1072
+
1073
+ export function localizedPath(locale: string, path: string) {
1074
+ // Trả về đường dẫn có tiền tố ngôn ngữ nếu không phải ngôn ngữ mặc định
1075
+ return locale === defaultLocale ? path : "/" + locale + path;
1076
+ }
1077
+
1078
+ const ORIGIN = "https://example.com";
1079
+ export function abs(locale: string, path: string) {
1080
+ // Trả về URL tuyệt đối dựa trên đường dẫn đã địa phương hóa
1081
+ return ORIGIN + localizedPath(locale, path);
1082
+ }
1083
+ ```
1084
+
1085
+ ```tsx fileName="src/app/[locale]/about/layout.tsx"
1086
+ import type { Metadata } from "next";
1087
+ import { locales, defaultLocale, localizedPath } from "@/i18n.config";
1088
+
1089
+ type GenerateMetadataParams = {
1090
+ params: Promise<{
1091
+ locale: string;
1092
+ }>;
1093
+ };
1094
+
1095
+ export const generateMetadata = async ({
1096
+ params,
1097
+ }: GenerateMetadataParams): Promise<Metadata> => {
1098
+ const { locale } = await params;
1099
+
1100
+ // Nhập đúng gói JSON từ src/locales
1101
+ const messages = (await import("@/locales/" + locale + "/about.json"))
1102
+ .default;
1103
+
1104
+ const languages = Object.fromEntries(
1105
+ locales.map((locale) => [locale, localizedPath(locale, "/about")])
1106
+ );
1107
+
1108
+ return {
1109
+ title: messages.title,
1110
+ description: messages.description,
1111
+ alternates: {
1112
+ canonical: localizedPath(locale, "/about"),
1113
+ languages: { ...languages, "x-default": "/about" },
1114
+ },
1115
+ };
1116
+ };
1117
+
1118
+ export default async function AboutPage() {
1119
+ return <h1>About</h1>;
1120
+ }
1121
+ ```
1122
+
1123
+ ```ts fileName="src/app/sitemap.ts"
1124
+ import type { MetadataRoute } from "next";
1125
+ import { locales, defaultLocale, abs } from "@/i18n.config";
1126
+
1127
+ export const sitemap = (): MetadataRoute.Sitemap => {
1128
+ const languages = Object.fromEntries(
1129
+ locales.map((locale) => [locale, abs(locale, "/about")])
1130
+ );
1131
+ return [
1132
+ {
1133
+ url: abs(defaultLocale, "/about"),
1134
+ lastModified: new Date(),
1135
+ changeFrequency: "monthly",
1136
+ priority: 0.7,
1137
+ alternates: { languages },
1138
+ },
1139
+ ];
1140
+ };
1141
+ ```
1142
+
1143
+ ```ts fileName="src/app/robots.ts"
1144
+ import type { MetadataRoute } from "next";
1145
+ import { locales, defaultLocale, localizedPath } from "@/i18n.config";
1146
+
1147
+ const ORIGIN = "https://example.com";
1148
+
1149
+ const expandAllLocales = (path: string) => [
1150
+ localizedPath(defaultLocale, path),
1151
+ ...locales
1152
+ .filter((locale) => locale !== defaultLocale)
1153
+ .map((locale) => localizedPath(locale, path)),
1154
+ ];
1155
+
1156
+ export const robots = (): MetadataRoute.Robots => {
1157
+ const disallow = [
1158
+ ...expandAllLocales("/dashboard"),
1159
+ ...expandAllLocales("/admin"),
1160
+ ];
1161
+
1162
+ return {
1163
+ rules: { userAgent: "*", allow: ["/"], disallow },
1164
+ host: ORIGIN,
1165
+ sitemap: ORIGIN + "/sitemap.xml",
1166
+ };
1167
+ };
1168
+ ```
1169
+
1170
+ </TabItem>
1171
+ <TabItem label="next-intl" value="next-intl">
1172
+
1173
+ ```tsx fileName="src/app/[locale]/about/layout.tsx"
1174
+ import type { Metadata } from "next";
1175
+ import { locales, defaultLocale } from "@/i18n";
1176
+ import { getTranslations } from "next-intl/server";
1177
+
1178
+ const localizedPath = (locale: string, path: string) => {
1179
+ return locale === defaultLocale ? path : "/" + locale + path;
1180
+ };
1181
+
1182
+ type GenerateMetadataParams = {
1183
+ params: Promise<{
1184
+ locale: string;
1185
+ }>;
1186
+ };
1187
+
1188
+ export const generateMetadata = async ({
1189
+ params,
1190
+ }: GenerateMetadataParams): Promise<Metadata> => {
1191
+ const { locale } = await params;
1192
+ const t = await getTranslations({ locale, namespace: "about" });
1193
+
1194
+ const url = "/about";
1195
+ const languages = Object.fromEntries(
1196
+ locales.map((locale) => [locale, localizedPath(locale, url)])
1197
+ );
1198
+
1199
+ return {
1200
+ title: t("title"), // tiêu đề trang
1201
+ description: t("description"), // mô tả trang
1202
+ alternates: {
1203
+ canonical: localizedPath(locale, url), // đường dẫn chuẩn
1204
+ languages: { ...languages, "x-default": url }, // các ngôn ngữ thay thế, bao gồm mặc định
1205
+ },
1206
+ };
1207
+ };
1208
+
1209
+ // ... Phần còn lại của mã trang
1210
+ ```
1211
+
1212
+ ```tsx fileName="src/app/sitemap.ts"
1213
+ import type { MetadataRoute } from "next";
1214
+ import { locales, defaultLocale } from "@/i18n";
1215
+
1216
+ const origin = "https://example.com";
1217
+
1218
+ const formatterLocalizedPath = (locale: string, path: string) =>
1219
+ locale === defaultLocale ? origin + path : origin + "/" + locale + path;
1220
+
1221
+ export const sitemap = (): MetadataRoute.Sitemap => {
1222
+ const aboutLanguages = Object.fromEntries(
1223
+ locales.map((l) => [l, formatterLocalizedPath(l, "/about")])
1224
+ );
1225
+
1226
+ return [
1227
+ {
1228
+ url: formatterLocalizedPath(defaultLocale, "/about"),
1229
+ lastModified: new Date(),
1230
+ changeFrequency: "monthly",
1231
+ priority: 0.7,
1232
+ alternates: { languages: aboutLanguages },
1233
+ },
1234
+ ];
1235
+ };
1236
+ ```
1237
+
1238
+ ```tsx fileName="src/app/robots.ts"
1239
+ import type { MetadataRoute } from "next";
1240
+ import { locales, defaultLocale } from "@/i18n";
1241
+
1242
+ const origin = "https://example.com";
1243
+ const withAllLocales = (path: string) => [
1244
+ path,
1245
+ ...locales
1246
+ .filter((locale) => locale !== defaultLocale)
1247
+ .map((locale) => "/" + locale + path),
1248
+ ];
1249
+
1250
+ export const robots = (): MetadataRoute.Robots => {
1251
+ const disallow = [
1252
+ ...withAllLocales("/dashboard"),
1253
+ ...withAllLocales("/admin"),
1254
+ ];
1255
+
1256
+ return {
1257
+ rules: { userAgent: "*", allow: ["/"], disallow },
1258
+ host: origin,
1259
+ sitemap: origin + "/sitemap.xml",
1260
+ };
1261
+ };
1262
+ path,
1263
+ ...locales
1264
+ .filter((locale) => locale !== defaultLocale)
1265
+ .map((locale) => "/" + locale + path),
1266
+ ];
1267
+
1268
+ export const robots = (): MetadataRoute.Robots => {
1269
+ const disallow = [
1270
+ ...withAllLocales("/dashboard"),
1271
+ ...withAllLocales("/admin"),
1272
+ ];
1273
+
1274
+ return {
1275
+ rules: { userAgent: "*", allow: ["/"], disallow },
1276
+ host: origin,
1277
+ sitemap: origin + "/sitemap.xml",
1278
+ };
1279
+ };
1280
+ ```
1281
+
1282
+ </TabItem>
1283
+ <TabItem label="intlayer" value="intlayer">
1284
+
1285
+ ```typescript fileName="src/app/[locale]/about/layout.tsx"
1286
+ import { getIntlayer, getMultilingualUrls } from "intlayer";
1287
+ import type { Metadata } from "next";
1288
+ import type { LocalPromiseParams } from "next-intlayer";
1289
+
1290
+ export const generateMetadata = async ({
1291
+ params,
1292
+ }: LocalPromiseParams): Promise<Metadata> => {
1293
+ const { locale } = await params;
1294
+
1295
+ const metadata = getIntlayer("page-metadata", locale);
1296
+
1297
+ const multilingualUrls = getMultilingualUrls("/about");
1298
+
1299
+ return {
1300
+ ...metadata,
1301
+ alternates: {
1302
+ canonical: multilingualUrls[locale as keyof typeof multilingualUrls],
1303
+ languages: { ...multilingualUrls, "x-default": "/about" },
1304
+ },
1305
+ };
1306
+ };
1307
+
1308
+ // ... Phần còn lại của mã trang
1309
+ ```
1310
+
1311
+ ```tsx fileName="src/app/sitemap.ts"
1312
+ import { getMultilingualUrls } from "intlayer";
1313
+ import type { MetadataRoute } from "next";
1314
+
1315
+ const sitemap = (): MetadataRoute.Sitemap => [
1316
+ {
1317
+ url: "https://example.com/about",
1318
+ alternates: {
1319
+ languages: { ...getMultilingualUrls("https://example.com/about") },
1320
+ },
1321
+ },
1322
+ ];
1323
+ ```
1324
+
1325
+ ```tsx fileName="src/app/robots.ts"
1326
+ import { getMultilingualUrls } from "intlayer";
1327
+ import type { MetadataRoute } from "next";
1328
+
1329
+ const getAllMultilingualUrls = (urls: string[]) =>
1330
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)) as string[]);
1331
+
1332
+ const robots = (): MetadataRoute.Robots => ({
1333
+ rules: {
1334
+ userAgent: "*",
1335
+ allow: ["/"],
1336
+ disallow: getAllMultilingualUrls(["/dashboard"]), // không cho phép truy cập các URL đa ngôn ngữ của /dashboard
1337
+ },
1338
+ host: "https://example.com",
1339
+ sitemap: "https://example.com/sitemap.xml",
1340
+ });
1341
+
1342
+ export default robots;
1343
+ ```
1344
+
1345
+ </TabItem>
1346
+ </Tab>
1347
+
1348
+ > Intlayer cung cấp một hàm `getMultilingualUrls` để tạo các URL đa ngôn ngữ cho sitemap của bạn.
1349
+
1350
+ ### Middleware cho định tuyến locale
1351
+
1352
+ <Tab defaultTab="next-intl" group='techno'>
1353
+ <TabItem label="next-i18next" value="next-i18next">
1354
+
1355
+ Thêm middleware để xử lý phát hiện locale và định tuyến:
1356
+
1357
+ ```ts fileName="src/middleware.ts"
1358
+ import { NextResponse, type NextRequest } from "next/server";
1359
+ import { defaultLocale, locales } from "@/i18n.config";
1360
+
1361
+ const PUBLIC_FILE = /\.[^/]+$/; // loại trừ các tệp có phần mở rộng
1362
+
1363
+ export function middleware(request: NextRequest) {
1364
+ const { pathname } = request.nextUrl;
1365
+
1366
+ if (
1367
+ pathname.startsWith("/_next") ||
1368
+ pathname.startsWith("/api") ||
1369
+ pathname.startsWith("/static") ||
1370
+ PUBLIC_FILE.test(pathname)
1371
+ ) {
1372
+ return;
1373
+ }
1374
+
1375
+ const hasLocale = locales.some(
1376
+ (l) => pathname === "/" + l || pathname.startsWith("/" + l + "/")
1377
+ );
1378
+ if (!hasLocale) {
1379
+ const locale = defaultLocale;
1380
+ const url = request.nextUrl.clone();
1381
+ url.pathname = "/" + locale + (pathname === "/" ? "" : pathname);
1382
+ return NextResponse.redirect(url);
1383
+ }
1384
+ }
1385
+
1386
+ export const config = {
1387
+ matcher: [
1388
+ // Khớp với tất cả các đường dẫn ngoại trừ những đường dẫn bắt đầu bằng các từ này và các tệp có phần mở rộng
1389
+ "/((?!api|_next|static|.*\\..*).*)",
1390
+ ],
1391
+ };
1392
+ ```
1393
+
1394
+ </TabItem>
1395
+ <TabItem label="next-intl" value="next-intl">
1396
+
1397
+ Thêm middleware để xử lý phát hiện locale và định tuyến:
1398
+
1399
+ ```ts fileName="src/middleware.ts"
1400
+ import createMiddleware from "next-intl/middleware";
1401
+ import { locales, defaultLocale } from "@/i18n";
1402
+
1403
+ export default createMiddleware({
1404
+ locales: [...locales],
1405
+ defaultLocale,
1406
+ localeDetection: true,
1407
+ });
1408
+
1409
+ export const config = {
1410
+ // Bỏ qua API, các phần nội bộ của Next và tài nguyên tĩnh
1411
+ matcher: ["/((?!api|_next|.*\\..*).*)"],
1412
+ };
1413
+ ```
1414
+
1415
+ </TabItem>
1416
+ <TabItem label="intlayer" value="intlayer">
1417
+
1418
+ Intlayer cung cấp xử lý middleware tích hợp sẵn thông qua cấu hình gói `next-intlayer`.
1419
+
1420
+ ```ts fileName="src/middleware.ts"
1421
+ import { intlayerMiddleware } from "next-intlayer/middleware";
1422
+
1423
+ export const middleware = intlayerMiddleware();
1424
+
1425
+ // áp dụng middleware này chỉ cho các tệp trong thư mục app
1426
+ export const config = {
1427
+ matcher: "/((?!api|_next|static|.*\\..*).*)",
1428
+ };
1429
+ ```
1430
+
1431
+ Việc thiết lập middleware được tập trung trong tệp `intlayer.config.ts`.
1432
+
1433
+ </TabItem>
1434
+ </Tab>
1435
+
1436
+ ### Danh sách kiểm tra thiết lập và các thực hành tốt
1437
+
1438
+ <Tab defaultTab="next-intl" group='techno'>
1439
+ <TabItem label="next-i18next" value="next-i18next">
1440
+
1441
+ - Đảm bảo `lang` và `dir` được đặt trên thẻ `<html>` gốc trong `src/app/[locale]/layout.tsx`.
1442
+ - Chia bản dịch thành các namespace (ví dụ `common.json`, `about.json`) dưới thư mục `src/locales/<locale>/`.
1443
+ - Chỉ tải các namespace cần thiết trong các thành phần client bằng cách sử dụng `useTranslation('<ns>')` và giới hạn phạm vi `I18nProvider` với cùng các namespace đó.
1444
+ - Giữ các trang ở trạng thái tĩnh khi có thể: xuất `export const dynamic = 'force-static'` trên các trang; đặt `dynamicParams = false` và triển khai `generateStaticParams`.
1445
+ - Sử dụng các thành phần server đồng bộ lồng trong phạm vi client bằng cách truyền các chuỗi đã được tính toán hoặc hàm `t` cùng với `locale`.
1446
+ - Đối với SEO, thiết lập `alternates.languages` trong metadata, liệt kê các URL đã được địa phương hóa trong `sitemap.ts`, và không cho phép các tuyến đường địa phương hóa trùng lặp trong `robots.ts`.
1447
+ - Ưu tiên sử dụng các bộ định dạng nhận biết locale (ví dụ, `Intl.NumberFormat(locale)`) và ghi nhớ chúng trên client nếu sử dụng React < 19.
1448
+
1449
+ </TabItem>
1450
+ <TabItem label="next-intl" value="next-intl">
1451
+
1452
+ - **Thiết lập thuộc tính html `lang` và `dir`**: Trong `src/app/[locale]/layout.tsx`, tính toán `dir` thông qua `getLocaleDirection(locale)` và thiết lập `<html lang={locale} dir={dir}>`.
1453
+ - **Phân tách thông điệp theo namespace**: Tổ chức JSON theo từng locale và namespace (ví dụ, `common.json`, `about.json`).
1454
+ - **Giảm thiểu payload trên client**: Trên các trang, chỉ gửi các namespace cần thiết đến `NextIntlClientProvider` (ví dụ, `pick(messages, ['common', 'about'])`).
1455
+ - **Ưu tiên các trang tĩnh**: Xuất `export const dynamic = 'force-static'` và tạo các tham số tĩnh cho tất cả các `locales`.
1456
+ - **Các thành phần server đồng bộ**: Giữ cho các thành phần server đồng bộ bằng cách truyền các chuỗi đã được tính toán trước (nhãn đã dịch, số đã được định dạng) thay vì các cuộc gọi async hoặc các hàm không thể tuần tự hóa.
1457
+
1458
+ </TabItem>
1459
+ <TabItem label="intlayer" value="intlayer">
1460
+
1461
+ - **Nội dung mô-đun**: Đặt các từ điển nội dung cùng vị trí với các thành phần sử dụng các tệp `.content.{ts|js|json}`.
1462
+ - **An toàn kiểu**: Tận dụng tích hợp TypeScript để kiểm tra nội dung tại thời điểm biên dịch.
1463
+ - **Tối ưu hóa thời gian xây dựng**: Sử dụng công cụ xây dựng của Intlayer để tự động loại bỏ mã không dùng đến (tree-shaking) và tối ưu gói.
1464
+ - **Công cụ tích hợp**: Tận dụng các tính năng định tuyến tích hợp, trợ giúp SEO và hỗ trợ trình chỉnh sửa trực quan.
1465
+
1466
+ </TabItem>
1467
+ </Tab>
1468
+
1469
+ ---
1470
+
1471
+ ## Và người chiến thắng là…
1472
+
1473
+ Không đơn giản. Mỗi lựa chọn đều có những đánh đổi. Đây là cách tôi nhìn nhận:
1474
+
1475
+ <Columns>
1476
+ <Column>
1477
+
1478
+ **next-i18next**
1479
+
1480
+ - trưởng thành, đầy đủ tính năng, nhiều plugin cộng đồng, nhưng chi phí thiết lập cao hơn. Nếu bạn cần **hệ sinh thái plugin của i18next** (ví dụ: các quy tắc ICU nâng cao qua plugin) và đội ngũ của bạn đã quen với i18next, chấp nhận **cấu hình nhiều hơn** để có sự linh hoạt.
1481
+
1482
+ </Column>
1483
+ <Column>
1484
+
1485
+ **next-intl**
1486
+
1487
+ - đơn giản nhất, nhẹ, ít quyết định bắt buộc hơn. Nếu bạn muốn một giải pháp **tối giản**, bạn thoải mái với các danh mục tập trung, và ứng dụng của bạn có quy mô **nhỏ đến trung bình**.
1488
+
1489
+ </Column>
1490
+ <Column>
1491
+
1492
+ **Intlayer**
1493
+
1494
+ - được xây dựng cho Next.js hiện đại, với nội dung mô-đun, an toàn kiểu, công cụ hỗ trợ, và ít mã mẫu hơn. Nếu bạn đánh giá cao **nội dung phạm vi thành phần**, **TypeScript nghiêm ngặt**, **đảm bảo tại thời điểm xây dựng**, **tree-shaking**, và công cụ định tuyến/SEO/trình soạn thảo **đầy đủ tính năng** - đặc biệt cho **Next.js App Router**, hệ thống thiết kế và **các codebase lớn, mô-đun**.
1495
+
1496
+ </Column>
1497
+ </Columns>
1498
+
1499
+ Nếu bạn ưu tiên thiết lập tối giản và chấp nhận một số cấu hình thủ công, next-intl là lựa chọn tốt. Nếu bạn cần tất cả các tính năng và không ngại sự phức tạp, next-i18next sẽ phù hợp. Nhưng nếu bạn muốn một giải pháp hiện đại, có thể mở rộng, mô-đun với các công cụ tích hợp sẵn, Intlayer hướng đến việc cung cấp cho bạn điều đó ngay khi sử dụng.
1500
+
1501
+ > **Lựa chọn thay thế cho các nhóm doanh nghiệp**: Nếu bạn cần một giải pháp đã được chứng minh hoạt động hoàn hảo với các nền tảng bản địa hóa đã được thiết lập như **Crowdin**, **Phrase**, hoặc các hệ thống quản lý dịch thuật chuyên nghiệp khác, hãy cân nhắc **next-intl** hoặc **next-i18next** vì hệ sinh thái trưởng thành và các tích hợp đã được kiểm chứng của chúng.
1502
+
1503
+ > **Lộ trình tương lai**: Intlayer cũng dự định phát triển các plugin hoạt động trên nền tảng các giải pháp **i18next** và **next-intl**. Điều này sẽ mang lại cho bạn những lợi thế của Intlayer về tự động hóa, cú pháp và quản lý nội dung trong khi vẫn giữ được tính bảo mật và ổn định do các giải pháp đã được thiết lập này cung cấp trong mã ứng dụng của bạn.
1504
+
1505
+ ## GitHub STARs
1506
+
1507
+ Sao trên GitHub là một chỉ số mạnh mẽ cho thấy mức độ phổ biến của dự án, sự tin tưởng của cộng đồng và tính liên quan lâu dài. Mặc dù không phải là thước đo trực tiếp về chất lượng kỹ thuật, nhưng chúng phản ánh số lượng nhà phát triển thấy dự án hữu ích, theo dõi tiến trình của nó và có khả năng áp dụng nó. Để ước tính giá trị của một dự án, sao giúp so sánh mức độ thu hút giữa các lựa chọn thay thế và cung cấp cái nhìn sâu sắc về sự phát triển của hệ sinh thái.
1508
+
1509
+ [![Biểu đồ Lịch sử Sao](https://api.star-history.com/svg?repos=i18next/next-i18next&repos=amannn/next-intl&repos=aymericzip/intlayer&type=Date)](https://www.star-history.com/#i18next/next-i18next&amannn/next-intl&aymericzip/intlayer)
1510
+
1511
+ ---
1512
+
1513
+ ## Kết luận
1514
+
1515
+ Cả ba thư viện đều thành công trong việc cốt lõi hóa localization. Sự khác biệt là **bạn phải làm bao nhiêu công việc** để đạt được một thiết lập vững chắc, có thể mở rộng trong **Next.js hiện đại**:
1516
+
1517
+ - Với **Intlayer**, **nội dung mô-đun**, **TypeScript nghiêm ngặt**, **an toàn thời gian xây dựng**, **gói tree-shaken**, và **App Router + công cụ SEO hàng đầu** là **mặc định**, không phải là gánh nặng.
1518
+ - Nếu đội ngũ của bạn coi trọng **khả năng bảo trì và tốc độ** trong một ứng dụng đa ngôn ngữ, hướng thành phần, Intlayer cung cấp trải nghiệm **toàn diện nhất** hiện nay.
1519
+
1520
+ Tham khảo tài liệu ['Tại sao chọn Intlayer?'](https://intlayer.org/doc/why) để biết thêm chi tiết.