@intlayer/docs 7.0.6 → 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,1501 @@
1
+ ---
2
+ createdAt: 2025-08-23
3
+ updatedAt: 2025-09-29
4
+ title: next-i18next vs next-intl vs Intlayer
5
+ description: Porównanie next-i18next z next-intl i Intlayer pod kątem internacjonalizacji (i18n) aplikacji Next.js
6
+ keywords:
7
+ - next-intl
8
+ - next-i18next
9
+ - Intlayer
10
+ - Internacjonalizacja
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 | Internacjonalizacja (i18n) Next.js
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
+ Przyjrzyjmy się podobieństwom i różnicom między trzema opcjami i18n dla Next.js: next-i18next, next-intl oraz Intlayer.
25
+
26
+ To nie jest pełny poradnik. To porównanie, które ma pomóc Ci w wyborze.
27
+
28
+ Skupiamy się na **Next.js 13+ App Router** (z **React Server Components**) i oceniamy:
29
+
30
+ <TOC/>
31
+
32
+ > **w skrócie**: Wszystkie trzy rozwiązania mogą lokalizować aplikację Next.js. Jeśli chcesz **zawartość ograniczoną do komponentu**, **ścisłe typy TypeScript**, **sprawdzanie brakujących kluczy podczas kompilacji**, **słowniki poddane tree-shakingowi** oraz **pierwszorzędne wsparcie App Router + SEO**, **Intlayer** jest najbardziej kompletnym i nowoczesnym wyborem.
33
+
34
+ > Częstym nieporozumieniem wśród deweloperów jest myślenie, że `next-intl` to wersja Next.js `react-intl`. Tak nie jest, `next-intl` jest utrzymywany przez [Amann](https://github.com/amannn), podczas gdy `react-intl` jest utrzymywany przez [FormatJS](https://github.com/formatjs/formatjs).
35
+
36
+ ---
37
+
38
+ ## W skrócie
39
+
40
+ - **next-intl** - Lekki, prosty formatowanie wiadomości z solidnym wsparciem Next.js. Centralizowane katalogi są powszechne; DX jest prosty, ale bezpieczeństwo i utrzymanie na dużą skalę pozostają głównie Twoją odpowiedzialnością.
41
+ - **next-i18next** - i18next w oprawie Next.js. Dojrzały ekosystem i funkcje dostępne przez wtyczki (np. ICU), ale konfiguracja może być rozbudowana, a katalogi mają tendencję do centralizacji wraz z rozwojem projektów.
42
+ - **Intlayer** - Model zawartości skoncentrowany na komponentach dla Next.js, **ścisłe typowanie TS**, **sprawdzanie podczas kompilacji**, **tree-shaking**, **wbudowane middleware i pomocniki SEO**, opcjonalny **Visual Editor/CMS** oraz **tłumaczenia wspomagane przez 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
+ > Odznaki aktualizują się automatycznie. Zrzuty ekranu mogą się zmieniać w czasie.
54
+
55
+ ---
56
+
57
+ ## Porównanie funkcji obok siebie (skoncentrowane na Next.js)
58
+
59
+ | Funkcja | `next-intlayer` (Intlayer) | `next-intl` | `next-i18next` |
60
+ | ---------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------ | ------------------------------------------------------------------------------------------------------------------------------ |
61
+ | **Tłumaczenia blisko komponentów** | ✅ Tak, zawartość zlokalizowana razem z każdym komponentem | ❌ Nie | ❌ Nie |
62
+ | **Integracja z TypeScript** | ✅ Zaawansowana, automatycznie generowane ścisłe typy | ✅ Dobra | ⚠️ Podstawowa |
63
+ | **Wykrywanie brakujących tłumaczeń** | ✅ Podświetlanie błędów TypeScript oraz błędy/ostrzeżenia podczas kompilacji | ⚠️ Obsługa awaryjna w czasie wykonywania | ⚠️ Obsługa awaryjna w czasie wykonywania |
64
+ | **Bogata zawartość (JSX/Markdown/komponenty)** | ✅ Bezpośrednie wsparcie | ❌ Nie zaprojektowane dla bogatych węzłów | ⚠️ Ograniczone |
65
+ | **Tłumaczenie wspomagane AI** | ✅ Tak, obsługuje wielu dostawców AI. Można używać własnych kluczy API. Uwzględnia kontekst Twojej aplikacji i zakres treści | ❌ Nie | ❌ Nie |
66
+ | **Edytor wizualny** | ✅ Tak, lokalny Edytor wizualny + opcjonalny CMS; może eksternalizować zawartość codebase; osadzalny | ❌ Nie / dostępny przez zewnętrzne platformy lokalizacyjne | ❌ Nie / dostępny przez zewnętrzne platformy lokalizacyjne |
67
+ | **Lokalizowane routingi** | ✅ Tak, obsługuje lokalizowane ścieżki od razu (działa z Next.js i Vite) | ✅ Wbudowany, App Router obsługuje segment `[locale]` | ✅ Wbudowany |
68
+ | **Generowanie dynamicznych tras** | ✅ Tak | ✅ Tak | ✅ Tak |
69
+ | **Pluralizacja** | ✅ Wzorce oparte na enumeracji | ✅ Dobre | ✅ Dobre |
70
+ | **Formatowanie (daty, liczby, waluty)** | ✅ Zoptymalizowane formatery (Intl w tle) | ✅ Dobre (pomocniki Intl) | ✅ Dobre (pomocniki Intl) |
71
+ | **Format treści** | ✅ .tsx, .ts, .js, .json, .md, .txt, (.yaml WIP) | ✅ .json, .js, .ts | ⚠️ .json |
72
+ | **Wsparcie ICU** | ⚠️ W trakcie realizacji | ✅ Tak | ⚠️ Za pomocą wtyczki (`i18next-icu`) |
73
+ | **Narzędzia SEO (hreflang, sitemap)** | ✅ Wbudowane narzędzia: pomocniki do sitemap, robots.txt, metadanych | ✅ Dobre | ✅ Dobre |
74
+ | **Ekosystem / Społeczność** | ⚠️ Mniejsza, ale szybko rosnąca i reagująca | ✅ Dobra | ✅ Dobra |
75
+ | **Renderowanie po stronie serwera i komponenty serwerowe** | ✅ Tak, zoptymalizowane pod SSR / React Server Components | ⚠️ Obsługiwane na poziomie strony, ale wymaga przekazywania funkcji t w drzewie komponentów dla dzieci komponentów serwerowych | ⚠️ Obsługiwane na poziomie strony, ale wymaga przekazywania funkcji t w drzewie komponentów dla dzieci komponentów serwerowych |
76
+ | **Tree-shaking (ładowanie tylko używanej zawartości)** | ✅ Tak, per-komponent podczas budowania za pomocą wtyczek Babel/SWC | ⚠️ Częściowo | ⚠️ Częściowo |
77
+ | **Lazy loading (leniwe ładowanie)** | ✅ Tak, per-lokalizacja / per-słownik | ✅ Tak (per-trasa/per-lokalizacja), wymaga zarządzania przestrzeniami nazw | ✅ Tak (per-trasa/per-lokalizacja), wymaga zarządzania przestrzeniami nazw |
78
+ | **Usuwanie nieużywanych treści** | ✅ Tak, na poziomie słownika podczas budowania | ❌ Nie, można zarządzać ręcznie za pomocą zarządzania przestrzenią nazw | ❌ Nie, można zarządzać ręcznie za pomocą zarządzania przestrzenią nazw |
79
+ | **Zarządzanie dużymi projektami** | ✅ Zachęca do modularności, odpowiednie dla systemów projektowych | ✅ Modularne z konfiguracją | ✅ Modularne z konfiguracją |
80
+ | **Testowanie brakujących tłumaczeń (CLI/CI)** | ✅ CLI: `npx intlayer content test` (audyt przyjazny CI) | ⚠️ Nie wbudowane; dokumentacja sugeruje `npx @lingual/i18n-check` | ⚠️ Nie wbudowane; opiera się na narzędziach i18next / runtime `saveMissing` |
81
+
82
+ ---
83
+
84
+ ## Wprowadzenie
85
+
86
+ Next.js oferuje wbudowane wsparcie dla internacjonalizowanego routingu (np. segmenty lokalizacji). Jednak ta funkcja sama w sobie nie wykonuje tłumaczeń. Nadal potrzebujesz biblioteki do renderowania zlokalizowanych treści dla użytkowników.
87
+
88
+ Istnieje wiele bibliotek i18n, ale w świecie Next.js obecnie trzy zyskują na popularności: next-i18next, next-intl oraz Intlayer.
89
+
90
+ ---
91
+
92
+ ## Architektura i skalowalność
93
+
94
+ - **next-intl / next-i18next**: Domyślnie korzystają z **centralizowanych katalogów** dla każdego locale (oraz **namespaces** w i18next). Działa dobrze na początku, ale często staje się dużą wspólną powierzchnią z rosnącym sprzężeniem i zmianami kluczy.
95
+ - **Intlayer**: Zachęca do stosowania słowników **per-komponent** (lub per-funkcjonalność) **współlokowanych** z kodem, który obsługują. Zmniejsza to obciążenie poznawcze, ułatwia duplikację/migrację elementów UI oraz redukuje konflikty między zespołami. Nieużywana zawartość jest naturalnie łatwiejsza do wykrycia i usunięcia.
96
+
97
+ **Dlaczego to ważne:** W dużych bazach kodu lub systemach designu, **modułowa zawartość** skalują się lepiej niż monolityczne katalogi.
98
+
99
+ ---
100
+
101
+ ## Rozmiary pakietów i zależności
102
+
103
+ Po zbudowaniu aplikacji, bundle to JavaScript, który przeglądarka załaduje, aby wyrenderować stronę. Rozmiar bundle jest więc istotny dla wydajności aplikacji.
104
+
105
+ Dwa komponenty są ważne w kontekście bundle aplikacji wielojęzycznej:
106
+
107
+ - Kod aplikacji
108
+ - Zawartość ładowana przez przeglądarkę
109
+
110
+ ## Kod aplikacji
111
+
112
+ Znaczenie kodu aplikacji jest w tym przypadku minimalne. Wszystkie trzy rozwiązania są tree-shakable, co oznacza, że nieużywane części kodu nie są dołączane do bundle.
113
+
114
+ Oto porównanie rozmiaru bundle JavaScript ładowanego przez przeglądarkę dla aplikacji wielojęzycznej z trzema rozwiązaniami.
115
+
116
+ Jeśli w aplikacji nie potrzebujemy żadnego formatera, lista eksportowanych funkcji po tree-shakingu będzie następująca:
117
+
118
+ - **next-intlayer**: `useIntlayer`, `useLocale`, `NextIntlClientProvider`, (Rozmiar bundla to 180,6 kB -> 78,6 kB (gzip))
119
+ - **next-intl**: `useTranslations`, `useLocale`, `NextIntlClientProvider`, (Rozmiar bundla to 101,3 kB -> 31,4 kB (gzip))
120
+ - **next-i18next**: `useTranslation`, `useI18n`, `I18nextProvider`, (Rozmiar bundla to 80,7 kB -> 25,5 kB (gzip))
121
+
122
+ Te funkcje są jedynie wrapperami wokół kontekstu/stanu React, więc całkowity wpływ biblioteki i18n na rozmiar bundla jest minimalny.
123
+
124
+ > Intlayer jest nieco większy niż `next-intl` i `next-i18next`, ponieważ zawiera więcej logiki w funkcji `useIntlayer`. Jest to związane z integracją markdown i `intlayer-editor`.
125
+
126
+ ## Zawartość i tłumaczenia
127
+
128
+ Ta część jest często ignorowana przez deweloperów, ale rozważmy przypadek aplikacji składającej się z 10 stron w 10 językach. Załóżmy, że każda strona zawiera w 100% unikalną treść, aby uprościć obliczenia (w rzeczywistości wiele treści jest powtarzalnych między stronami, np. tytuł strony, nagłówek, stopka itp.).
129
+
130
+ Użytkownik chcący odwiedzić stronę `/fr/about` załaduje zawartość jednej strony w danym języku. Ignorowanie optymalizacji treści oznaczałoby niepotrzebne załadowanie 8 200% `((1 + (((10 stron - 1) × (10 języków - 1)))) × 100)` zawartości aplikacji. Widzisz problem? Nawet jeśli ta zawartość pozostaje tekstem, a Ty prawdopodobnie wolisz myśleć o optymalizacji obrazów na swojej stronie, wysyłasz niepotrzebną treść na cały świat i zmuszasz komputery użytkowników do jej przetwarzania bez powodu.
131
+
132
+ Dwa ważne zagadnienia:
133
+
134
+ - **Podział według ścieżki:**
135
+
136
+ > Jeśli jestem na stronie `/about`, nie chcę ładować zawartości strony `/home`
137
+
138
+ - **Podział według lokalizacji:**
139
+
140
+ > Jeśli jestem na stronie `/fr/about`, nie chcę ładować zawartości strony `/en/about`
141
+
142
+ Ponownie, wszystkie trzy rozwiązania zdają sobie sprawę z tych problemów i pozwalają zarządzać tymi optymalizacjami. Różnica między tymi trzema rozwiązaniami to DX (Developer Experience).
143
+
144
+ `next-intl` i `next-i18next` używają scentralizowanego podejścia do zarządzania tłumaczeniami, pozwalając na podział plików JSON według lokalizacji i podplików. W `next-i18next` nazywamy pliki JSON 'namespaces'; `next-intl` pozwala deklarować wiadomości. W `intlayer` nazywamy pliki JSON 'dictionaries'.
145
+
146
+ - W przypadku `next-intl`, podobnie jak w `next-i18next`, zawartość jest ładowana na poziomie strony/layoutu, a następnie ta zawartość jest ładowana do providera kontekstu. Oznacza to, że deweloper musi ręcznie zarządzać plikami JSON, które będą ładowane dla każdej strony.
147
+
148
+ > W praktyce oznacza to, że deweloperzy często pomijają tę optymalizację, woląc dla uproszczenia załadować całą zawartość w providerze kontekstu strony.
149
+
150
+ - W przypadku `intlayer` cała zawartość jest ładowana w aplikacji. Następnie wtyczka (`@intlayer/babel` / `@intlayer/swc`) zajmuje się optymalizacją bundla, ładując tylko zawartość używaną na stronie. Deweloper nie musi więc ręcznie zarządzać słownikami, które będą ładowane. Pozwala to na lepszą optymalizację, lepszą utrzymywalność oraz skraca czas developmentu.
151
+
152
+ W miarę rozwoju aplikacji (szczególnie gdy nad aplikacją pracuje wielu programistów), często zdarza się zapomnieć o usunięciu treści, które nie są już używane, z plików JSON.
153
+
154
+ > Należy zauważyć, że wszystkie pliki JSON są ładowane we wszystkich przypadkach (next-intl, next-i18next, intlayer).
155
+
156
+ Dlatego podejście Intlayer jest bardziej wydajne: jeśli komponent nie jest już używany, jego słownik nie jest ładowany do bundla.
157
+
158
+ Równie ważne jest, jak biblioteka obsługuje mechanizm fallbacków. Załóżmy, że aplikacja domyślnie jest w języku angielskim, a użytkownik odwiedza stronę `/fr/about`. Jeśli tłumaczenia we francuskim są niekompletne, zostanie zastosowany fallback na angielski.
159
+
160
+ W przypadku `next-intl` i `next-i18next` biblioteka wymaga załadowania pliku JSON związanego z bieżącym locale, ale także z locale zapasowym (fallback). W związku z tym, zakładając, że cały content został przetłumaczony, każda strona załaduje 100% niepotrzebnej zawartości. **Dla porównania, `intlayer` przetwarza fallback podczas budowania słownika. W ten sposób każda strona załaduje tylko używaną zawartość.**
161
+
162
+ > Uwaga: Aby zoptymalizować bundle za pomocą `intlayer`, musisz ustawić opcję `importMode: 'dynamic'` w pliku `intlayer.config.ts`. I upewnić się, że wtyczka `@intlayer/babel` / `@intlayer/swc` jest zainstalowana (domyślnie instalowana przy użyciu `vite-intlayer`).
163
+
164
+ Oto przykład wpływu optymalizacji rozmiaru bundle za pomocą `intlayer` w aplikacji vite + react:
165
+
166
+ | Zoptymalizowany pakiet | Pakiet niezoptymalizowany |
167
+ | ----------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------ |
168
+ | ![zoptymalizowany pakiet](https://github.com/aymericzip/intlayer/blob/main/docs/assets/bundle.png?raw=true) | ![niezoptymalizowany pakiet](https://github.com/aymericzip/intlayer/blob/main/docs/assets/bundle_no_optimization.png?raw=true) |
169
+
170
+ ---
171
+
172
+ ## TypeScript i bezpieczeństwo
173
+
174
+ <Columns>
175
+ <Column>
176
+
177
+ **next-i18next**
178
+
179
+ - Podstawowe typy dla hooków. **ścisłe typowanie kluczy wymaga dodatkowych narzędzi/konfiguracji**.
180
+
181
+ </Column>
182
+ <Column>
183
+
184
+ **next-intl**
185
+
186
+ - Solidne wsparcie TypeScript, ale **klucze nie są domyślnie ściśle typowane**. Bezpieczeństwo utrzymasz ręcznie.
187
+
188
+ </Column>
189
+ <Column>
190
+
191
+ **intlayer**
192
+
193
+ - **Generuje ścisłe typy** na podstawie twoich treści. **Autouzupełnianie w IDE** oraz **błędy podczas kompilacji** wychwytują literówki i brakujące klucze przed wdrożeniem.
194
+
195
+ </Column>
196
+ </Columns>
197
+
198
+ **Dlaczego to ważne:** Silne typowanie przesuwa błędy na **wcześniejszy etap** (CI/build), zamiast na **późniejszy** (czas działania).
199
+
200
+ ---
201
+
202
+ ## Obsługa brakujących tłumaczeń
203
+
204
+ <Columns>
205
+ <Column>
206
+
207
+ **next-i18next**
208
+
209
+ - Polega na **fallbackach w czasie działania**. Budowanie nie kończy się błędem.
210
+
211
+ </Column>
212
+ <Column>
213
+
214
+ **next-intl**
215
+
216
+ - Polega na **fallbackach w czasie działania**. Budowanie nie kończy się błędem.
217
+
218
+ </Column>
219
+ <Column>
220
+
221
+ **intlayer**
222
+
223
+ - **Wykrywanie podczas kompilacji** z **ostrzeżeniami/błędami** dla brakujących lokalizacji lub kluczy.
224
+
225
+ </Column>
226
+ </Columns>
227
+
228
+ **Dlaczego to ważne:** Wykrywanie braków podczas kompilacji zapobiega pojawianiu się 'undefined' w produkcji.
229
+
230
+ ---
231
+
232
+ ## Routing, middleware i strategia URL
233
+
234
+ <Columns>
235
+ <Column>
236
+
237
+ **next-i18next**
238
+
239
+ - Umożliwia lokalizowany routing. Jednak middleware nie jest wbudowane.
240
+
241
+ </Column>
242
+ <Column>
243
+
244
+ **next-intl**
245
+
246
+ - Umożliwia lokalizowany routing.
247
+ - Dostarcza middleware.
248
+
249
+ </Column>
250
+ <Column>
251
+
252
+ **intlayer**
253
+
254
+ - Umożliwia lokalizowany routing.
255
+ - Dostarcza middleware.
256
+
257
+ </Column>
258
+ </Columns>
259
+
260
+ **Dlaczego to ważne:** Pomaga w SEO i odkrywalności, a także poprawia doświadczenie użytkownika.
261
+
262
+ ---
263
+
264
+ ## Wsparcie dla Server Components (RSC)
265
+
266
+ <Columns>
267
+ <Column>
268
+
269
+ **next-i18next**
270
+
271
+ - Wspiera komponenty serwerowe stron i layoutów.
272
+ - Nie zapewnia synchronicznego API dla dziecięcych komponentów serwerowych.
273
+
274
+ </Column>
275
+ <Column>
276
+
277
+ **next-intl**
278
+
279
+ - Obsługuje komponenty serwerowe stron i układów.
280
+ - Nie zapewnia synchronicznego API dla dziecięcych komponentów serwerowych.
281
+
282
+ </Column>
283
+ <Column>
284
+
285
+ **intlayer**
286
+
287
+ - Obsługuje komponenty serwerowe stron i układów.
288
+ - Zapewnia synchroniczne API dla dziecięcych komponentów serwerowych.
289
+
290
+ </Column>
291
+ </Columns>
292
+
293
+ **Dlaczego to ważne:** Wsparcie komponentów serwerowych jest kluczową funkcją Next.js 13+, wspomagającą wydajność. Przekazywanie propsów takich jak locale lub funkcji `t` z komponentu nadrzędnego do dziecięcych komponentów serwerowych sprawia, że Twoje komponenty są mniej wielokrotnego użytku.
294
+
295
+ ---
296
+
297
+ ## Integracja z platformami lokalizacyjnymi (TMS)
298
+
299
+ Duże organizacje często polegają na Systemach Zarządzania Tłumaczeniami (TMS), takich jak **Crowdin**, **Phrase**, **Lokalise**, **Localizely** czy **Localazy**.
300
+
301
+ - **Dlaczego firmy zwracają na to uwagę**
302
+ - **Współpraca i role**: Zaangażowanych jest wielu uczestników: deweloperzy, menedżerowie produktu, tłumacze, recenzenci, zespoły marketingowe.
303
+ - **Skala i efektywność**: ciągła lokalizacja, przegląd w kontekście.
304
+
305
+ - **next-intl / next-i18next**
306
+ - Zazwyczaj używają **scentralizowanych katalogów JSON**, więc eksport/import z TMS jest prosty.
307
+ - Dojrzałe ekosystemy oraz przykłady/integracje dla wymienionych platform.
308
+
309
+ - **Intlayer**
310
+ - Zachęca do **zdecentralizowanych słowników per komponent** i obsługuje zawartość w formatach **TypeScript/TSX/JS/JSON/MD**.
311
+ - Poprawia to modularność w kodzie, ale może utrudniać integrację typu plug-and-play z TMS, gdy narzędzie oczekuje scentralizowanych, płaskich plików JSON.
312
+ - Intlayer oferuje alternatywy: **tłumaczenia wspomagane AI** (z użyciem własnych kluczy dostawcy), **Edytor Wizualny/CMS** oraz workflowy **CLI/CI** do wykrywania i wypełniania luk.
313
+
314
+ > Uwaga: `next-intl` i `i18next` również akceptują katalogi TypeScript. Jeśli Twój zespół przechowuje wiadomości w plikach `.ts` lub decentralizuje je według funkcji, możesz napotkać podobne problemy z TMS. Jednak wiele konfiguracji `next-intl` pozostaje scentralizowanych w folderze `locales/`, co jest nieco łatwiejsze do przekształcenia na JSON dla TMS.
315
+
316
+ ---
317
+
318
+ ## Doświadczenie dewelopera
319
+
320
+ Ta część dokonuje dogłębnego porównania trzech rozwiązań. Zamiast rozważać proste przypadki, opisane w dokumentacji „getting started” dla każdego rozwiązania, rozważymy rzeczywisty przypadek użycia, bardziej zbliżony do prawdziwego projektu.
321
+
322
+ ### Struktura aplikacji
323
+
324
+ Struktura aplikacji jest ważna, aby zapewnić dobrą utrzymywalność Twojej bazy kodu.
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
+ #### Porównanie
405
+
406
+ - **next-intl / next-i18next**: Centralizowane katalogi (JSON; przestrzenie nazw/wiadomości). Jasna struktura, dobrze integruje się z platformami tłumaczeniowymi, ale może prowadzić do większej liczby edycji między plikami w miarę rozwoju aplikacji.
407
+ - **Intlayer**: Słowniki `.content.{ts|js|json}` przypisane do komponentów i umieszczone razem z nimi. Ułatwia ponowne użycie komponentów i lokalne rozumowanie; dodaje pliki i opiera się na narzędziach działających podczas budowania.
408
+
409
+ #### Konfiguracja i ładowanie zawartości
410
+
411
+ Jak wspomniano wcześniej, musisz zoptymalizować sposób importowania każdego pliku JSON do swojego kodu.
412
+ Sposób, w jaki biblioteka obsługuje ładowanie zawartości, jest ważny.
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
+ // Funkcja zwracająca ścieżkę z uwzględnieniem lokalizacji
428
+ export function localizedPath(locale: string, path: string) {
429
+ return locale === defaultLocale ? path : "/" + locale + path;
430
+ }
431
+
432
+ const ORIGIN = "https://example.com";
433
+ // Funkcja zwracająca absolutny URL z lokalizacją
434
+ export function abs(locale: string, path: string) {
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
+ // Ładowanie zasobów JSON z 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: pakiet }
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
+ // Wymuś statyczne renderowanie strony
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
+ // Załaduj tylko przestrzenie nazw potrzebne dla twojego layoutu/stron
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
+ // Ustaw aktywny język żądania dla tego renderowania po stronie serwera (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
+ // Wiadomości są ładowane po stronie serwera. Przekaż tylko to, co jest potrzebne klientowi.
667
+ const messages = await getMessages();
668
+ const clientMessages = pick(messages, ["common", "about"]);
669
+
670
+ // Tłumaczenia/formatowanie wyłącznie po stronie serwera
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
+ #### Porównanie
764
+
765
+ Wszystkie trzy wspierają ładowanie treści i providerów dla poszczególnych lokalizacji.
766
+
767
+ - W przypadku **next-intl/next-i18next** zazwyczaj ładujesz wybrane wiadomości/przestrzenie nazw na trasę i umieszczasz providery tam, gdzie są potrzebne.
768
+
769
+ - W przypadku **Intlayer** dodaje analizę w czasie budowania, aby wywnioskować użycie, co może zmniejszyć ręczne łączenie i pozwolić na pojedynczego providera na poziomie root.
770
+
771
+ Wybierz między kontrolą eksplicytną a automatyzacją w zależności od preferencji zespołu.
772
+
773
+ ### Użycie w komponencie klienckim
774
+
775
+ Weźmy przykład komponentu klienckiego renderującego licznik.
776
+
777
+ <Tab defaultTab="next-intl" group='techno'>
778
+ <TabItem label="next-i18next" value="next-i18next">
779
+
780
+ **Tłumaczenia (po jednym pliku JSON na namespace w `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
+ **Komponent kliencki (ładuje tylko wymagany namespace)**
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);
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
+ > Upewnij się, że strona/dostawca zawiera tylko potrzebne przestrzenie nazw (np. `about`).
835
+ > Jeśli używasz React < 19, zapamiętuj ciężkie formatery, takie jak `Intl.NumberFormat`.
836
+
837
+ </TabItem>
838
+ <TabItem label="next-intl" value="next-intl">
839
+
840
+ **Tłumaczenia (zachowano strukturę; załaduj je do wiadomości next-intl według własnego uznania)**
841
+
842
+ ```json fileName="locales/en/about.json"
843
+ {
844
+ "counter": {
845
+ "label": "Counter",
846
+ "increment": "Increment"
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
+ **Komponent klienta**
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
+ // Bezpośredni zakres do zagnieżdżonego obiektu
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
+ > Nie zapomnij dodać komunikatu "about" na stronie klienta
889
+
890
+ </TabItem>
891
+ <TabItem label="intlayer" value="intlayer">
892
+
893
+ **Zawartość**
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({ pl: "Licznik", en: "Counter", fr: "Compteur" }),
902
+ increment: t({ pl: "Zwiększ", en: "Increment", fr: "Incrémenter" }),
903
+ },
904
+ } satisfies Dictionary;
905
+
906
+ export default counterContent;
907
+ ```
908
+
909
+ **Komponent klienta**
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"); // zwraca stringi
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
+ #### Porównanie
937
+
938
+ - **Formatowanie liczb**
939
+ - **next-i18next**: brak `useNumber`; używa `Intl.NumberFormat` (lub i18next-icu).
940
+ - **next-intl**: `useFormatter().number(value)`.
941
+ - **Intlayer**: wbudowane `useNumber()`.
942
+
943
+ - **Klucze**
944
+ - Zachowaj zagnieżdżoną strukturę (`about.counter.label`) i odpowiednio ustaw zakres hooka (`useTranslation("about")` + `t("counter.label")` lub `useTranslations("about.counter")` + `t("label")`).
945
+
946
+ - **Lokalizacje plików**
947
+ - **next-i18next** oczekuje plików JSON w `public/locales/{lng}/{ns}.json`.
948
+ - **next-intl** jest elastyczny; ładuj wiadomości według własnej konfiguracji.
949
+ - **Intlayer** przechowuje zawartość w słownikach TS/JS i rozwiązuje je za pomocą klucza.
950
+
951
+ ---
952
+
953
+ ### Użycie w komponencie serwerowym
954
+
955
+ Weźmiemy pod uwagę przypadek komponentu UI. Ten komponent jest komponentem serwerowym i powinien mieć możliwość bycia wstawionym jako dziecko komponentu klienta. (strona (komponent serwerowy) -> komponent klienta -> komponent serwerowy). Ponieważ ten komponent może być wstawiony jako dziecko komponentu klienta, nie może być asynchroniczny.
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;
963
+ locale: string;
964
+ count: number;
965
+ };
966
+
967
+ const ServerComponent = ({ t, locale, count }: ServerComponentProps) => {
968
+ const formatted = new Intl.NumberFormat(locale).format(count);
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;
987
+ locale: string;
988
+ count: number;
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
+ > Ponieważ komponent serwera nie może być asynchroniczny, musisz przekazać tłumaczenia i funkcję formatującą jako propsy.
1012
+ >
1013
+ > W twojej stronie / układzie:
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 udostępnia **bezpieczne dla serwera** hooki za pośrednictwem `next-intlayer/server`. Aby działały, `useIntlayer` i `useNumber` używają składni podobnej do hooków klienta, ale pod spodem zależą od kontekstu serwera (`IntlayerServerProvider`).
1046
+
1047
+ ### Metadane / Sitemap / Robots
1048
+
1049
+ Tłumaczenie treści jest świetne. Jednak ludzie często zapominają, że głównym celem internacjonalizacji jest uczynienie Twojej strony bardziej widoczną dla świata. I18n to niesamowita dźwignia do poprawy widoczności Twojej strony.
1050
+
1051
+ Oto lista dobrych praktyk dotyczących wielojęzycznego SEO.
1052
+
1053
+ - ustaw metatagi hreflang w tagu `<head>`
1054
+ > Pomaga to wyszukiwarkom zrozumieć, jakie języki są dostępne na stronie
1055
+ - wymień wszystkie tłumaczenia stron w pliku sitemap.xml, używając schematu XML `http://www.w3.org/1999/xhtml`
1056
+ >
1057
+ - nie zapomnij wykluczyć stron z prefiksami w pliku robots.txt (np. `/dashboard` oraz `/fr/dashboard`, `/es/dashboard`)
1058
+ >
1059
+ - używaj niestandardowego komponentu Link, aby przekierować do najbardziej zlokalizowanej strony (np. po francusku `<a href="/fr/about">A propos</a>`)
1060
+ >
1061
+
1062
+ Programiści często zapominają o prawidłowym referencjonowaniu swoich stron w różnych lokalizacjach.
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
+ return locale === defaultLocale ? path : "/" + locale + path;
1075
+ }
1076
+
1077
+ const ORIGIN = "https://example.com";
1078
+ export function abs(locale: string, path: string) {
1079
+ return ORIGIN + localizedPath(locale, path);
1080
+ }
1081
+ ```
1082
+
1083
+ ```tsx fileName="src/app/[locale]/about/layout.tsx"
1084
+ import type { Metadata } from "next";
1085
+ import { locales, defaultLocale, localizedPath } from "@/i18n.config";
1086
+
1087
+ type GenerateMetadataParams = {
1088
+ params: Promise<{
1089
+ locale: string;
1090
+ }>;
1091
+ };
1092
+
1093
+ export const generateMetadata = async ({
1094
+ params,
1095
+ }: GenerateMetadataParams): Promise<Metadata> => {
1096
+ const { locale } = await params;
1097
+
1098
+ // Importuj odpowiedni pakiet JSON z src/locales
1099
+ const messages = (await import("@/locales/" + locale + "/about.json"))
1100
+ .default;
1101
+
1102
+ const languages = Object.fromEntries(
1103
+ locales.map((locale) => [locale, localizedPath(locale, "/about")])
1104
+ );
1105
+
1106
+ return {
1107
+ title: messages.title,
1108
+ description: messages.description,
1109
+ alternates: {
1110
+ canonical: localizedPath(locale, "/about"),
1111
+ languages: { ...languages, "x-default": "/about" },
1112
+ },
1113
+ };
1114
+ };
1115
+
1116
+ export default async function AboutPage() {
1117
+ return <h1>O nas</h1>;
1118
+ }
1119
+ ```
1120
+
1121
+ ```ts fileName="src/app/sitemap.ts"
1122
+ import type { MetadataRoute } from "next";
1123
+ import { locales, defaultLocale, abs } from "@/i18n.config";
1124
+
1125
+ export const sitemap = (): MetadataRoute.Sitemap => {
1126
+ const languages = Object.fromEntries(
1127
+ locales.map((locale) => [locale, abs(locale, "/about")])
1128
+ );
1129
+ return [
1130
+ {
1131
+ url: abs(defaultLocale, "/about"),
1132
+ lastModified: new Date(),
1133
+ changeFrequency: "monthly",
1134
+ priority: 0.7,
1135
+ alternates: { languages },
1136
+ },
1137
+ ];
1138
+ };
1139
+ ```
1140
+
1141
+ ```ts fileName="src/app/robots.ts"
1142
+ import type { MetadataRoute } from "next";
1143
+ import { locales, defaultLocale, localizedPath } from "@/i18n.config";
1144
+
1145
+ const ORIGIN = "https://example.com";
1146
+
1147
+ const expandAllLocales = (path: string) => [
1148
+ localizedPath(defaultLocale, path),
1149
+ ...locales
1150
+ .filter((locale) => locale !== defaultLocale)
1151
+ .map((locale) => localizedPath(locale, path)),
1152
+ ];
1153
+
1154
+ export const robots = (): MetadataRoute.Robots => {
1155
+ const disallow = [
1156
+ ...expandAllLocales("/dashboard"),
1157
+ ...expandAllLocales("/admin"),
1158
+ ];
1159
+
1160
+ return {
1161
+ rules: { userAgent: "*", allow: ["/"], disallow },
1162
+ host: ORIGIN,
1163
+ sitemap: ORIGIN + "/sitemap.xml",
1164
+ };
1165
+ };
1166
+ ```
1167
+
1168
+ </TabItem>
1169
+ <TabItem label="next-intl" value="next-intl">
1170
+
1171
+ ```tsx fileName="src/app/[locale]/about/layout.tsx"
1172
+ import type { Metadata } from "next";
1173
+ import { locales, defaultLocale } from "@/i18n";
1174
+ import { getTranslations } from "next-intl/server";
1175
+
1176
+ const localizedPath = (locale: string, path: string) => {
1177
+ // Zwraca ścieżkę lokalizowaną, jeśli locale jest domyślne, zwraca oryginalną ścieżkę
1178
+ return locale === defaultLocale ? path : "/" + locale + path;
1179
+ };
1180
+
1181
+ type GenerateMetadataParams = {
1182
+ params: Promise<{
1183
+ locale: string;
1184
+ }>;
1185
+ };
1186
+
1187
+ export const generateMetadata = async ({
1188
+ params,
1189
+ }: GenerateMetadataParams): Promise<Metadata> => {
1190
+ const { locale } = await params;
1191
+ const t = await getTranslations({ locale, namespace: "about" });
1192
+
1193
+ const url = "/about";
1194
+ const languages = Object.fromEntries(
1195
+ locales.map((locale) => [locale, localizedPath(locale, url)])
1196
+ );
1197
+
1198
+ return {
1199
+ title: t("title"),
1200
+ description: t("description"),
1201
+ alternates: {
1202
+ canonical: localizedPath(locale, url),
1203
+ languages: { ...languages, "x-default": url },
1204
+ },
1205
+ };
1206
+ };
1207
+
1208
+ // ... Reszta kodu strony
1209
+ ```
1210
+
1211
+ ```tsx fileName="src/app/sitemap.ts"
1212
+ import type { MetadataRoute } from "next";
1213
+ import { locales, defaultLocale } from "@/i18n";
1214
+
1215
+ const origin = "https://example.com";
1216
+
1217
+ const formatterLocalizedPath = (locale: string, path: string) =>
1218
+ locale === defaultLocale ? origin + path : origin + "/" + locale + path;
1219
+
1220
+ export const sitemap = (): MetadataRoute.Sitemap => {
1221
+ const aboutLanguages = Object.fromEntries(
1222
+ locales.map((l) => [l, formatterLocalizedPath(l, "/about")])
1223
+ );
1224
+
1225
+ return [
1226
+ {
1227
+ url: formatterLocalizedPath(defaultLocale, "/about"),
1228
+ lastModified: new Date(),
1229
+ changeFrequency: "monthly", // częstotliwość zmian: miesięcznie
1230
+ priority: 0.7, // priorytet strony
1231
+ alternates: { languages: aboutLanguages }, // alternatywne wersje językowe
1232
+ },
1233
+ ];
1234
+ };
1235
+ ```
1236
+
1237
+ ```tsx fileName="src/app/robots.ts"
1238
+ import type { MetadataRoute } from "next";
1239
+ import { locales, defaultLocale } from "@/i18n";
1240
+
1241
+ const origin = "https://example.com";
1242
+ const withAllLocales = (path: string) => [
1243
+ path,
1244
+ ...locales
1245
+ .filter((locale) => locale !== defaultLocale)
1246
+ .map((locale) => "/" + locale + path),
1247
+ ];
1248
+
1249
+ export const robots = (): MetadataRoute.Robots => {
1250
+ const disallow = [
1251
+ ...withAllLocales("/dashboard"),
1252
+ ...withAllLocales("/admin"),
1253
+ ];
1254
+
1255
+ return {
1256
+ rules: { userAgent: "*", allow: ["/"], disallow },
1257
+ host: origin,
1258
+ sitemap: origin + "/sitemap.xml",
1259
+ };
1260
+ };
1261
+ ```
1262
+
1263
+ </TabItem>
1264
+ <TabItem label="intlayer" value="intlayer">
1265
+
1266
+ ```typescript fileName="src/app/[locale]/about/layout.tsx"
1267
+ import { getIntlayer, getMultilingualUrls } from "intlayer";
1268
+ import type { Metadata } from "next";
1269
+ import type { LocalPromiseParams } from "next-intlayer";
1270
+
1271
+ export const generateMetadata = async ({
1272
+ params,
1273
+ }: LocalPromiseParams): Promise<Metadata> => {
1274
+ const { locale } = await params;
1275
+
1276
+ const metadata = getIntlayer("page-metadata", locale);
1277
+
1278
+ const multilingualUrls = getMultilingualUrls("/about");
1279
+
1280
+ return {
1281
+ ...metadata,
1282
+ alternates: {
1283
+ canonical: multilingualUrls[locale as keyof typeof multilingualUrls],
1284
+ languages: { ...multilingualUrls, "x-default": "/about" },
1285
+ },
1286
+ };
1287
+ };
1288
+
1289
+ // ... Reszta kodu strony
1290
+ ```
1291
+
1292
+ ```tsx fileName="src/app/sitemap.ts"
1293
+ import { getMultilingualUrls } from "intlayer";
1294
+ import type { MetadataRoute } from "next";
1295
+
1296
+ const sitemap = (): MetadataRoute.Sitemap => [
1297
+ {
1298
+ url: "https://example.com/about",
1299
+ alternates: {
1300
+ languages: { ...getMultilingualUrls("https://example.com/about") },
1301
+ },
1302
+ },
1303
+ ];
1304
+ ```
1305
+
1306
+ ```tsx fileName="src/app/robots.ts"
1307
+ import { getMultilingualUrls } from "intlayer";
1308
+ import type { MetadataRoute } from "next";
1309
+
1310
+ const getAllMultilingualUrls = (urls: string[]) =>
1311
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)) as string[]);
1312
+
1313
+ const robots = (): MetadataRoute.Robots => ({
1314
+ rules: {
1315
+ userAgent: "*",
1316
+ allow: ["/"],
1317
+ disallow: getAllMultilingualUrls(["/dashboard"]),
1318
+ },
1319
+ host: "https://example.com",
1320
+ sitemap: "https://example.com/sitemap.xml",
1321
+ });
1322
+
1323
+ export default robots;
1324
+ ```
1325
+
1326
+ </TabItem>
1327
+ </Tab>
1328
+
1329
+ > Intlayer dostarcza funkcję `getMultilingualUrls` do generowania wielojęzycznych adresów URL dla Twojej mapy witryny.
1330
+
1331
+ ### Middleware do routingu lokalizacji
1332
+
1333
+ <Tab defaultTab="next-intl" group='techno'>
1334
+ <TabItem label="next-i18next" value="next-i18next">
1335
+
1336
+ Dodaj middleware do obsługi wykrywania lokalizacji i routingu:
1337
+
1338
+ ```ts fileName="src/middleware.ts"
1339
+ import { NextResponse, type NextRequest } from "next/server";
1340
+ import { defaultLocale, locales } from "@/i18n.config";
1341
+
1342
+ const PUBLIC_FILE = /\.[^/]+$/; // wyklucz pliki z rozszerzeniami
1343
+
1344
+ export function middleware(request: NextRequest) {
1345
+ const { pathname } = request.nextUrl;
1346
+
1347
+ if (
1348
+ pathname.startsWith("/_next") ||
1349
+ pathname.startsWith("/api") ||
1350
+ pathname.startsWith("/static") ||
1351
+ PUBLIC_FILE.test(pathname)
1352
+ ) {
1353
+ return;
1354
+ }
1355
+
1356
+ const hasLocale = locales.some(
1357
+ (l) => pathname === "/" + l || pathname.startsWith("/" + l + "/")
1358
+ );
1359
+ if (!hasLocale) {
1360
+ const locale = defaultLocale;
1361
+ const url = request.nextUrl.clone();
1362
+ url.pathname = "/" + locale + (pathname === "/" ? "" : pathname);
1363
+ return NextResponse.redirect(url);
1364
+ }
1365
+ }
1366
+
1367
+ export const config = {
1368
+ matcher: [
1369
+ // Dopasuj wszystkie ścieżki z wyjątkiem tych zaczynających się od podanych oraz plików z rozszerzeniem
1370
+ "/((?!api|_next|static|.*\\..*).*)",
1371
+ ],
1372
+ };
1373
+ ```
1374
+
1375
+ </TabItem>
1376
+ <TabItem label="next-intl" value="next-intl">
1377
+
1378
+ Dodaj middleware do obsługi wykrywania i routingu lokalizacji:
1379
+
1380
+ ```ts fileName="src/middleware.ts"
1381
+ import createMiddleware from "next-intl/middleware";
1382
+ import { locales, defaultLocale } from "@/i18n";
1383
+
1384
+ export default createMiddleware({
1385
+ locales: [...locales],
1386
+ defaultLocale,
1387
+ localeDetection: true,
1388
+ });
1389
+
1390
+ export const config = {
1391
+ // Pomijaj API, wewnętrzne elementy Next oraz zasoby statyczne
1392
+ matcher: ["/((?!api|_next|.*\\..*).*)"],
1393
+ };
1394
+ ```
1395
+
1396
+ </TabItem>
1397
+ <TabItem label="intlayer" value="intlayer">
1398
+
1399
+ Intlayer zapewnia wbudowane zarządzanie middleware za pomocą konfiguracji pakietu `next-intlayer`.
1400
+
1401
+ ```ts fileName="src/middleware.ts"
1402
+ import { intlayerMiddleware } from "next-intlayer/middleware";
1403
+
1404
+ export const middleware = intlayerMiddleware();
1405
+
1406
+ // stosuje ten middleware tylko do plików w katalogu app
1407
+ export const config = {
1408
+ matcher: "/((?!api|_next|static|.*\\..*).*)",
1409
+ };
1410
+ ```
1411
+
1412
+ Konfiguracja middleware jest scentralizowana w pliku `intlayer.config.ts`.
1413
+
1414
+ </TabItem>
1415
+ </Tab>
1416
+
1417
+ ### Lista kontrolna konfiguracji i dobre praktyki
1418
+
1419
+ <Tab defaultTab="next-intl" group='techno'>
1420
+ <TabItem label="next-i18next" value="next-i18next">
1421
+
1422
+ - Upewnij się, że `lang` i `dir` są ustawione na głównym elemencie `<html>` w `src/app/[locale]/layout.tsx`.
1423
+ - Podziel tłumaczenia na przestrzenie nazw (na przykład `common.json`, `about.json`) w katalogu `src/locales/<locale>/`.
1424
+ - Ładuj tylko wymagane przestrzenie nazw w komponentach klienckich, używając `useTranslation('<ns>')` oraz ograniczając `I18nProvider` do tych samych przestrzeni nazw.
1425
+ - Utrzymuj strony statyczne, gdy to możliwe: eksportuj `export const dynamic = 'force-static'` na stronach; ustaw `dynamicParams = false` i zaimplementuj `generateStaticParams`.
1426
+ - Używaj synchronicznych komponentów serwerowych zagnieżdżonych pod granicami klienta, przekazując już obliczone ciągi znaków lub funkcję `t` oraz `locale`.
1427
+ - Dla SEO ustaw `alternates.languages` w metadanych, wymień lokalizowane adresy URL w `sitemap.ts` i zabroń duplikowania lokalizowanych tras w `robots.ts`.
1428
+ - Preferuj formatery uwzględniające lokalizację (np. `Intl.NumberFormat(locale)`) i zapamiętuj je po stronie klienta, jeśli używasz React < 19.
1429
+
1430
+ </TabItem>
1431
+ <TabItem label="next-intl" value="next-intl">
1432
+
1433
+ - **Ustaw atrybuty html `lang` i `dir`**: W pliku `src/app/[locale]/layout.tsx` oblicz `dir` za pomocą `getLocaleDirection(locale)` i ustaw `<html lang={locale} dir={dir}>`.
1434
+ - **Podziel wiadomości na przestrzenie nazw**: Organizuj pliki JSON według lokalizacji i przestrzeni nazw (np. `common.json`, `about.json`).
1435
+ - **Minimalizuj obciążenie klienta**: Na stronach wysyłaj do `NextIntlClientProvider` tylko wymagane przestrzenie nazw (np. `pick(messages, ['common', 'about'])`).
1436
+ - **Preferuj strony statyczne**: Eksportuj `export const dynamic = 'force-static'` i generuj statyczne parametry dla wszystkich `locales`.
1437
+ - **Synchroniczne komponenty serwerowe**: Utrzymuj komponenty serwerowe synchroniczne, przekazując wcześniej obliczone ciągi znaków (przetłumaczone etykiety, sformatowane liczby) zamiast wywołań asynchronicznych lub funkcji nieserializowalnych.
1438
+
1439
+ </TabItem>
1440
+ <TabItem label="intlayer" value="intlayer">
1441
+
1442
+ - **Modularna zawartość**: Umieszczaj słowniki zawartości razem z komponentami, korzystając z plików `.content.{ts|js|json}`.
1443
+ - **Bezpieczeństwo typów**: Wykorzystaj integrację z TypeScript do walidacji zawartości na etapie kompilacji.
1444
+ - **Optymalizacja w czasie budowania**: Używaj narzędzi Intlayer do automatycznego tree-shakingu i optymalizacji pakietu.
1445
+ - **Zintegrowane narzędzia**: Wykorzystaj wbudowane routowanie, pomocników SEO oraz wsparcie edytora wizualnego.
1446
+
1447
+ </TabItem>
1448
+ </Tab>
1449
+
1450
+ ---
1451
+
1452
+ ## A zwycięzca to…
1453
+
1454
+ To nie jest proste. Każda opcja ma swoje kompromisy. Oto jak to widzę:
1455
+
1456
+ <Columns>
1457
+ <Column>
1458
+
1459
+ **next-i18next**
1460
+
1461
+ - dojrzały, pełen funkcji, wiele wtyczek społecznościowych, ale wyższy koszt konfiguracji. Jeśli potrzebujesz **ekosystemu wtyczek i18next** (np. zaawansowane reguły ICU przez wtyczki) i Twój zespół już zna i18next, akceptując **więcej konfiguracji** dla elastyczności.
1462
+
1463
+ </Column>
1464
+ <Column>
1465
+
1466
+ **next-intl**
1467
+
1468
+ - najprostszy, lekki, mniej narzuconych decyzji. Jeśli chcesz **minimalne** rozwiązanie, czujesz się komfortowo z centralizowanymi katalogami, a Twoja aplikacja jest **mała do średniej wielkości**.
1469
+
1470
+ </Column>
1471
+ <Column>
1472
+
1473
+ **Intlayer**
1474
+
1475
+ - zbudowany dla nowoczesnego Next.js, z modularną zawartością, bezpieczeństwem typów, narzędziami i mniejszą ilością boilerplate. Jeśli cenisz sobie **zawartość ograniczoną do komponentu**, **ścisły TypeScript**, **gwarancje w czasie kompilacji**, **tree-shaking** oraz **wbudowane** narzędzia do routingu/SEO/edytora – szczególnie dla **Next.js App Router**, systemów projektowych i **dużych, modularnych baz kodu**.
1476
+
1477
+ </Column>
1478
+ </Columns>
1479
+
1480
+ Jeśli wolisz minimalną konfigurację i akceptujesz trochę ręcznego łączenia, next-intl jest dobrym wyborem. Jeśli potrzebujesz wszystkich funkcji i nie przeszkadza Ci złożoność, next-i18next sprawdzi się. Ale jeśli chcesz nowoczesne, skalowalne, modularne rozwiązanie z wbudowanymi narzędziami, Intlayer ma na celu dostarczyć to od razu po wyjęciu z pudełka.
1481
+
1482
+ > **Alternatywa dla zespołów korporacyjnych**: Jeśli potrzebujesz sprawdzonego rozwiązania, które działa doskonale z uznanymi platformami lokalizacyjnymi, takimi jak **Crowdin**, **Phrase** lub innymi profesjonalnymi systemami zarządzania tłumaczeniami, rozważ **next-intl** lub **next-i18next** ze względu na ich dojrzały ekosystem i sprawdzone integracje.
1483
+
1484
+ > **Przyszła mapa drogowa**: Intlayer planuje również rozwijać wtyczki działające na bazie rozwiązań **i18next** i **next-intl**. Dzięki temu zyskasz zalety Intlayer w zakresie automatyzacji, składni i zarządzania treścią, zachowując jednocześnie bezpieczeństwo i stabilność zapewnianą przez te sprawdzone rozwiązania w kodzie aplikacji.
1485
+
1486
+ ## Gwiazdki GitHub (GitHub STARs)
1487
+
1488
+ Gwiazdy na GitHubie są silnym wskaźnikiem popularności projektu, zaufania społeczności oraz jego długoterminowej istotności. Choć nie są bezpośrednią miarą jakości technicznej, odzwierciedlają, ilu deweloperów uważa projekt za przydatny, śledzi jego rozwój i prawdopodobnie zdecyduje się go używać. Przy szacowaniu wartości projektu, gwiazdy pomagają porównać zainteresowanie różnymi alternatywami oraz dostarczają wglądu w rozwój ekosystemu.
1489
+
1490
+ [![Wykres historii gwiazd](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)
1491
+
1492
+ ---
1493
+
1494
+ ## Podsumowanie
1495
+
1496
+ Wszystkie trzy biblioteki odnoszą sukces w podstawowej lokalizacji. Różnica polega na tym, **ile pracy musisz włożyć**, aby osiągnąć solidną, skalowalną konfigurację w **nowoczesnym Next.js**:
1497
+
1498
+ - W przypadku **Intlayer** **modularna zawartość**, **ścisły TS**, **bezpieczeństwo w czasie kompilacji**, **tree-shaken bundles** oraz **pierwszorzędny App Router i narzędzia SEO** są **domyślne**, a nie obowiązkowe.
1499
+ - Jeśli Twój zespół ceni **łatwość utrzymania i szybkość** w aplikacji wielojęzycznej, opartej na komponentach, Intlayer oferuje dziś **najpełniejsze** doświadczenie.
1500
+
1501
+ Zapoznaj się z dokumentem ['Dlaczego Intlayer?'](https://intlayer.org/doc/why) po więcej szczegółów.