@intlayer/docs 7.5.12 → 7.5.13

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 (197) hide show
  1. package/blog/uk/compiler_vs_declarative_i18n.md +224 -0
  2. package/blog/uk/i18n_using_next-i18next.md +1086 -0
  3. package/blog/uk/i18n_using_next-intl.md +760 -0
  4. package/blog/uk/index.md +69 -0
  5. package/blog/uk/internationalization_and_SEO.md +273 -0
  6. package/blog/uk/intlayer_with_i18next.md +211 -0
  7. package/blog/uk/intlayer_with_next-i18next.md +202 -0
  8. package/blog/uk/intlayer_with_next-intl.md +203 -0
  9. package/blog/uk/intlayer_with_react-i18next.md +200 -0
  10. package/blog/uk/intlayer_with_react-intl.md +202 -0
  11. package/blog/uk/intlayer_with_vue-i18n.md +206 -0
  12. package/blog/uk/l10n_platform_alternative/Lokalise.md +80 -0
  13. package/blog/uk/l10n_platform_alternative/crowdin.md +80 -0
  14. package/blog/uk/l10n_platform_alternative/phrase.md +78 -0
  15. package/blog/uk/list_i18n_technologies/CMS/drupal.md +143 -0
  16. package/blog/uk/list_i18n_technologies/CMS/wix.md +167 -0
  17. package/blog/uk/list_i18n_technologies/CMS/wordpress.md +189 -0
  18. package/blog/uk/list_i18n_technologies/frameworks/angular.md +125 -0
  19. package/blog/uk/list_i18n_technologies/frameworks/flutter.md +128 -0
  20. package/blog/uk/list_i18n_technologies/frameworks/react-native.md +217 -0
  21. package/blog/uk/list_i18n_technologies/frameworks/react.md +155 -0
  22. package/blog/uk/list_i18n_technologies/frameworks/svelte.md +145 -0
  23. package/blog/uk/list_i18n_technologies/frameworks/vue.md +144 -0
  24. package/blog/uk/next-i18next_vs_next-intl_vs_intlayer.md +1499 -0
  25. package/blog/uk/nextjs-multilingual-seo-comparison.md +360 -0
  26. package/blog/uk/rag_powered_documentation_assistant.md +288 -0
  27. package/blog/uk/react-i18next_vs_react-intl_vs_intlayer.md +164 -0
  28. package/blog/uk/vue-i18n_vs_intlayer.md +279 -0
  29. package/blog/uk/what_is_internationalization.md +167 -0
  30. package/dist/cjs/generated/frequentQuestions.entry.cjs +20 -0
  31. package/dist/cjs/generated/frequentQuestions.entry.cjs.map +1 -1
  32. package/dist/esm/generated/frequentQuestions.entry.mjs +20 -0
  33. package/dist/esm/generated/frequentQuestions.entry.mjs.map +1 -1
  34. package/dist/types/generated/frequentQuestions.entry.d.ts +1 -0
  35. package/dist/types/generated/frequentQuestions.entry.d.ts.map +1 -1
  36. package/docs/ar/configuration.md +6 -1
  37. package/docs/ar/dictionary/content_file.md +6 -1
  38. package/docs/de/configuration.md +6 -1
  39. package/docs/de/dictionary/content_file.md +6 -1
  40. package/docs/en/configuration.md +6 -1
  41. package/docs/en/dictionary/content_file.md +6 -1
  42. package/docs/en-GB/configuration.md +6 -1
  43. package/docs/en-GB/dictionary/content_file.md +3 -1
  44. package/docs/es/configuration.md +6 -1
  45. package/docs/es/dictionary/content_file.md +6 -1
  46. package/docs/fr/configuration.md +6 -1
  47. package/docs/fr/dictionary/content_file.md +3 -1
  48. package/docs/hi/configuration.md +6 -1
  49. package/docs/hi/dictionary/content_file.md +3 -1
  50. package/docs/id/configuration.md +6 -1
  51. package/docs/id/dictionary/content_file.md +3 -1
  52. package/docs/it/configuration.md +6 -1
  53. package/docs/it/dictionary/content_file.md +3 -1
  54. package/docs/ja/configuration.md +6 -1
  55. package/docs/ja/dictionary/content_file.md +3 -1
  56. package/docs/ko/configuration.md +6 -1
  57. package/docs/ko/dictionary/content_file.md +3 -1
  58. package/docs/pl/configuration.md +3 -1
  59. package/docs/pl/dictionary/content_file.md +3 -1
  60. package/docs/pt/configuration.md +6 -1
  61. package/docs/pt/dictionary/content_file.md +3 -1
  62. package/docs/ru/configuration.md +6 -1
  63. package/docs/ru/dictionary/content_file.md +6 -1
  64. package/docs/tr/configuration.md +6 -1
  65. package/docs/tr/dictionary/content_file.md +3 -1
  66. package/docs/uk/CI_CD.md +198 -0
  67. package/docs/uk/autoFill.md +307 -0
  68. package/docs/uk/bundle_optimization.md +185 -0
  69. package/docs/uk/cli/build.md +64 -0
  70. package/docs/uk/cli/ci.md +137 -0
  71. package/docs/uk/cli/configuration.md +63 -0
  72. package/docs/uk/cli/debug.md +46 -0
  73. package/docs/uk/cli/doc-review.md +43 -0
  74. package/docs/uk/cli/doc-translate.md +132 -0
  75. package/docs/uk/cli/editor.md +28 -0
  76. package/docs/uk/cli/fill.md +130 -0
  77. package/docs/uk/cli/index.md +190 -0
  78. package/docs/uk/cli/init.md +84 -0
  79. package/docs/uk/cli/list.md +90 -0
  80. package/docs/uk/cli/list_projects.md +128 -0
  81. package/docs/uk/cli/live.md +41 -0
  82. package/docs/uk/cli/login.md +157 -0
  83. package/docs/uk/cli/pull.md +78 -0
  84. package/docs/uk/cli/push.md +98 -0
  85. package/docs/uk/cli/sdk.md +71 -0
  86. package/docs/uk/cli/test.md +76 -0
  87. package/docs/uk/cli/transform.md +65 -0
  88. package/docs/uk/cli/version.md +24 -0
  89. package/docs/uk/cli/watch.md +37 -0
  90. package/docs/uk/configuration.md +742 -0
  91. package/docs/uk/dictionary/condition.md +237 -0
  92. package/docs/uk/dictionary/content_file.md +1134 -0
  93. package/docs/uk/dictionary/enumeration.md +245 -0
  94. package/docs/uk/dictionary/file.md +232 -0
  95. package/docs/uk/dictionary/function_fetching.md +212 -0
  96. package/docs/uk/dictionary/gender.md +273 -0
  97. package/docs/uk/dictionary/insertion.md +187 -0
  98. package/docs/uk/dictionary/markdown.md +383 -0
  99. package/docs/uk/dictionary/nesting.md +273 -0
  100. package/docs/uk/dictionary/translation.md +332 -0
  101. package/docs/uk/formatters.md +595 -0
  102. package/docs/uk/how_works_intlayer.md +256 -0
  103. package/docs/uk/index.md +175 -0
  104. package/docs/uk/interest_of_intlayer.md +297 -0
  105. package/docs/uk/intlayer_CMS.md +569 -0
  106. package/docs/uk/intlayer_visual_editor.md +292 -0
  107. package/docs/uk/intlayer_with_angular.md +710 -0
  108. package/docs/uk/intlayer_with_astro.md +256 -0
  109. package/docs/uk/intlayer_with_create_react_app.md +1258 -0
  110. package/docs/uk/intlayer_with_express.md +429 -0
  111. package/docs/uk/intlayer_with_fastify.md +446 -0
  112. package/docs/uk/intlayer_with_lynx+react.md +548 -0
  113. package/docs/uk/intlayer_with_nestjs.md +283 -0
  114. package/docs/uk/intlayer_with_next-i18next.md +640 -0
  115. package/docs/uk/intlayer_with_next-intl.md +456 -0
  116. package/docs/uk/intlayer_with_nextjs_page_router.md +1541 -0
  117. package/docs/uk/intlayer_with_nuxt.md +711 -0
  118. package/docs/uk/intlayer_with_react_router_v7.md +600 -0
  119. package/docs/uk/intlayer_with_react_router_v7_fs_routes.md +669 -0
  120. package/docs/uk/intlayer_with_svelte_kit.md +579 -0
  121. package/docs/uk/intlayer_with_tanstack.md +818 -0
  122. package/docs/uk/intlayer_with_vite+preact.md +1748 -0
  123. package/docs/uk/intlayer_with_vite+react.md +1449 -0
  124. package/docs/uk/intlayer_with_vite+solid.md +302 -0
  125. package/docs/uk/intlayer_with_vite+svelte.md +520 -0
  126. package/docs/uk/intlayer_with_vite+vue.md +1113 -0
  127. package/docs/uk/introduction.md +222 -0
  128. package/docs/uk/locale_mapper.md +242 -0
  129. package/docs/uk/mcp_server.md +211 -0
  130. package/docs/uk/packages/express-intlayer/t.md +465 -0
  131. package/docs/uk/packages/intlayer/getEnumeration.md +159 -0
  132. package/docs/uk/packages/intlayer/getHTMLTextDir.md +121 -0
  133. package/docs/uk/packages/intlayer/getLocaleLang.md +81 -0
  134. package/docs/uk/packages/intlayer/getLocaleName.md +135 -0
  135. package/docs/uk/packages/intlayer/getLocalizedUrl.md +338 -0
  136. package/docs/uk/packages/intlayer/getMultilingualUrls.md +359 -0
  137. package/docs/uk/packages/intlayer/getPathWithoutLocale.md +75 -0
  138. package/docs/uk/packages/intlayer/getPrefix.md +213 -0
  139. package/docs/uk/packages/intlayer/getTranslation.md +190 -0
  140. package/docs/uk/packages/intlayer/getTranslationContent.md +189 -0
  141. package/docs/uk/packages/next-intlayer/t.md +365 -0
  142. package/docs/uk/packages/next-intlayer/useDictionary.md +276 -0
  143. package/docs/uk/packages/next-intlayer/useIntlayer.md +263 -0
  144. package/docs/uk/packages/next-intlayer/useLocale.md +166 -0
  145. package/docs/uk/packages/react-intlayer/t.md +311 -0
  146. package/docs/uk/packages/react-intlayer/useDictionary.md +295 -0
  147. package/docs/uk/packages/react-intlayer/useI18n.md +250 -0
  148. package/docs/uk/packages/react-intlayer/useIntlayer.md +251 -0
  149. package/docs/uk/packages/react-intlayer/useLocale.md +210 -0
  150. package/docs/uk/per_locale_file.md +345 -0
  151. package/docs/uk/plugins/sync-json.md +398 -0
  152. package/docs/uk/readme.md +265 -0
  153. package/docs/uk/releases/v6.md +305 -0
  154. package/docs/uk/releases/v7.md +624 -0
  155. package/docs/uk/roadmap.md +346 -0
  156. package/docs/uk/testing.md +204 -0
  157. package/docs/vi/configuration.md +6 -1
  158. package/docs/vi/dictionary/content_file.md +6 -1
  159. package/docs/zh/configuration.md +6 -1
  160. package/docs/zh/dictionary/content_file.md +6 -1
  161. package/frequent_questions/ar/error-vite-env-only.md +77 -0
  162. package/frequent_questions/de/error-vite-env-only.md +77 -0
  163. package/frequent_questions/en/error-vite-env-only.md +77 -0
  164. package/frequent_questions/en-GB/error-vite-env-only.md +77 -0
  165. package/frequent_questions/es/error-vite-env-only.md +76 -0
  166. package/frequent_questions/fr/error-vite-env-only.md +77 -0
  167. package/frequent_questions/hi/error-vite-env-only.md +77 -0
  168. package/frequent_questions/id/error-vite-env-only.md +77 -0
  169. package/frequent_questions/it/error-vite-env-only.md +77 -0
  170. package/frequent_questions/ja/error-vite-env-only.md +77 -0
  171. package/frequent_questions/ko/error-vite-env-only.md +77 -0
  172. package/frequent_questions/pl/error-vite-env-only.md +77 -0
  173. package/frequent_questions/pt/error-vite-env-only.md +77 -0
  174. package/frequent_questions/ru/error-vite-env-only.md +77 -0
  175. package/frequent_questions/tr/error-vite-env-only.md +77 -0
  176. package/frequent_questions/uk/SSR_Next_no_[locale].md +104 -0
  177. package/frequent_questions/uk/array_as_content_declaration.md +72 -0
  178. package/frequent_questions/uk/build_dictionaries.md +58 -0
  179. package/frequent_questions/uk/build_error_CI_CD.md +74 -0
  180. package/frequent_questions/uk/bun_set_up.md +53 -0
  181. package/frequent_questions/uk/customized_locale_list.md +64 -0
  182. package/frequent_questions/uk/domain_routing.md +113 -0
  183. package/frequent_questions/uk/error-vite-env-only.md +77 -0
  184. package/frequent_questions/uk/esbuild_error.md +29 -0
  185. package/frequent_questions/uk/get_locale_cookie.md +142 -0
  186. package/frequent_questions/uk/intlayer_command_undefined.md +155 -0
  187. package/frequent_questions/uk/locale_incorect_in_url.md +73 -0
  188. package/frequent_questions/uk/package_version_error.md +181 -0
  189. package/frequent_questions/uk/static_rendering.md +44 -0
  190. package/frequent_questions/uk/translated_path_url.md +55 -0
  191. package/frequent_questions/uk/unknown_command.md +97 -0
  192. package/frequent_questions/vi/error-vite-env-only.md +77 -0
  193. package/frequent_questions/zh/error-vite-env-only.md +77 -0
  194. package/legal/uk/privacy_notice.md +83 -0
  195. package/legal/uk/terms_of_service.md +55 -0
  196. package/package.json +6 -6
  197. package/src/generated/frequentQuestions.entry.ts +20 -0
@@ -0,0 +1,224 @@
1
+ ---
2
+ createdAt: 2025-11-24
3
+ updatedAt: 2025-11-24
4
+ title: Компіляторне i18n vs декларативне i18n
5
+ description: Вивчення архітектурних компромісів між «магічним» компіляторним підходом до інтернаціоналізації та явним декларативним управлінням контентом.
6
+ keywords:
7
+ - Intlayer
8
+ - Інтернаціоналізація (Internationalization)
9
+ - Блог
10
+ - Next.js
11
+ - JavaScript
12
+ - React
13
+ - i18n
14
+ - Компілятор
15
+ - Декларативний
16
+ slugs:
17
+ - blog
18
+ - compiler-vs-declarative-i18n
19
+ ---
20
+
21
+ # Аргументи за та проти компіляторного i18n
22
+
23
+ Якщо ви будуєте вебдодатки більше ніж десятиліття, ви знаєте, що інтернаціоналізація (i18n) завжди була точкою тертя. Зазвичай це завдання, яке ніхто не хоче виконувати — витягування рядків, керування JSON-файлами та турботи про правила множини.
24
+
25
+ Нещодавно з'явилася нова хвиля **"Compiler-based" i18n tools**, які обіцяють зробити цей біль минулим. Пропозиція приваблива: **Просто вписуйте текст у ваші компоненти, а build tool зробить усе інше.** Жодних ключів, жодних імпортів — лише магія.
26
+
27
+ Але, як і з усіма абстракціями в software engineering, магія має свою ціну.
28
+
29
+ У цьому дописі в блозі ми розглянемо перехід від декларативних бібліотек до компіляторних підходів, приховані архітектурні борги, які вони породжують, і чому «нудний» підхід може досі бути найкращим для професійних застосунків.
30
+
31
+ ## Зміст
32
+
33
+ <TOC/>
34
+
35
+ ## Коротка історія інтернаціоналізації
36
+
37
+ Щоб зрозуміти, де ми зараз, потрібно поглянути назад, до того, з чого ми починали.
38
+
39
+ Приблизно в 2011–2012 роках ландшафт JavaScript був значно іншим. Bundlers, як ми їх знаємо (Webpack, Vite), не існували або були на ранніх стадіях розвитку. Ми з'єднували скрипти прямо в браузері. У цю епоху з'явилися бібліотеки на кшталт **i18next**.
40
+
41
+ Вони вирішували проблему єдиним можливим тоді способом: **Runtime Dictionaries**. Ви завантажували величезний об'єкт JSON у пам'ять, а функція шукала ключі на льоту. Це було надійно, явно і працювало скрізь.
42
+
43
+ Перенесімося в сьогодення. У нас є потужні компілятори (SWC, Rust-based bundlers), які можуть розбирати абстрактні синтаксичні дерева (AST) за мілісекунди. Ця міць породила нову ідею: _Навіщо нам вручну керувати ключами? Чому компілятор не може просто побачити текст "Hello World" і замінити його за нас?_
44
+
45
+ Отже, з'явилося Compiler-based i18n.
46
+
47
+ > **Приклад компіляторного i18n:**
48
+ >
49
+ > - Paraglide (tree-shaken модулі, які компілюють кожне повідомлення в маленьку ESM-функцію, щоб bundler-и могли автоматично видаляти невикористані локалі та ключі. Ви імпортуєте повідомлення як функції замість пошуку за рядковими ключами.)
50
+ > - LinguiJS (компілятор макросів у функції, який переписує макроси повідомлень типу `<Trans>` у звичайні виклики JS-функцій під час збірки. Ви отримуєте синтаксис ICU/MessageFormat з невеликим runtime-відбитком.)
51
+ > - Lingo.dev (зосереджується на автоматизації пайплайна локалізації шляхом інʼєкції перекладеного вмісту безпосередньо під час збірки вашого React-додатка. Може автоматично генерувати переклади з використанням AI та інтегруватися безпосередньо в CI/CD.)
52
+ > - Wuchale (Svelte-first preprocessor that extracts inline text in .svelte files and compiles it into zero-wrapper translation functions. It avoids string keys, and separates the content extraction logic completely from the main application runtime.)
53
+ > - Intlayer (Compiler / Extract CLI that parses your components, generates typed dictionaries, and can optionally rewrite code to use explicit Intlayer content. The goal is to use the compiler for velocity while keeping a declarative, framework-agnostic core.)
54
+ >
55
+ > **Приклад декларативного i18n:**
56
+ >
57
+ > - i18next / react-i18next / next-i18next (визнаний галузевий стандарт, що використовує runtime JSON-словники та має широку екосистему плагінів)
58
+ > - react-intl (Частина бібліотеки FormatJS, зосереджена на стандартному синтаксисі повідомлень ICU та суворому форматуванні даних)
59
+ > - next-intl (Оптимізований спеціально для Next.js з інтеграцією для App Router та React Server Components)
60
+ > - vue-i18n / @nuxt/i18n (Стандартне рішення для екосистеми Vue, яке пропонує блоки перекладу на рівні компонентів та тісну інтеграцію з реактивністю)
61
+ > - svelte-i18n (Легка обгортка над Svelte stores для реактивних перекладів під час виконання)
62
+ > - angular-translate (Застаріла динамічна бібліотека перекладів, яка покладається на пошук ключів під час виконання замість злиття під час збірки)
63
+ > - angular-i18n (Рідний підхід Angular з попередньою компіляцією (ahead-of-time), який зливає файли XLIFF безпосередньо в шаблони під час збірки)
64
+ > - Tolgee (Поєднує декларативний код з in-context SDK для редагування "click-to-translate" безпосередньо в UI)
65
+ > - Intlayer (Підхід на рівні компонентів, із файлами декларацій контенту, які дозволяють нативний tree-shaking і валідацію TypeScript)
66
+
67
+ ## Компілятор Intlayer
68
+
69
+ Хоча **Intlayer** є рішенням, яке в основі заохочує **декларативний підхід** до вашого контенту, воно включає компілятор, щоб пришвидшити розробку або полегшити швидке прототипування.
70
+
71
+ Компілятор Intlayer обходить AST (Abstract Syntax Tree) ваших компонентів React, Vue або Svelte, а також інші файли JavaScript/TypeScript. Його роль — виявляти жорстко закодовані рядки та витягувати їх у спеціальні декларації `.content`.
72
+
73
+ > Для детальнішої інформації перегляньте документацію: [Документація компілятора Intlayer](https://github.com/aymericzip/intlayer/blob/main/docs/docs/uk/compiler.md)
74
+
75
+ ## Принада компілятора (підхід «магія»)
76
+
77
+ Існує причина, чому цей новий підхід набирає популярності. Для розробника досвід відчувається неймовірно.
78
+
79
+ ### 1. Швидкість і «потік»
80
+
81
+ Коли ви в роботі й повністю зосереджені, зупинка, щоб придумати семантичну назву змінної (`home_hero_title_v2`), порушує ваш потік. За підходу з компілятором ви просто вводите `<p>Welcome back</p>` і продовжуєте рух. Тертя відсутнє.
82
+
83
+ ### 2. Місія порятунку для legacy-коду
84
+
85
+ Уявіть, що ви успадкували величезну codebase з 5 000 компонентів і жодного перекладу. Оснащення цього вручну на основі ключів — це кошмар на кілька місяців роботи. Інструмент на базі компілятора виступає як стратегія порятунку, миттєво витягуючи тисячі рядків, не вимагаючи від вас жодного ручного редагування файлів.
86
+
87
+ ### 3. Ера AI
88
+
89
+ Це сучасна перевага, яку не варто ігнорувати. AI coding assistants (наприклад, Copilot або ChatGPT) природно генерують стандартний JSX/HTML. Вони не знають вашої конкретної схеми ключів для перекладів.
90
+
91
+ - **Declarative:** Вам доведеться переписувати вихідний код, згенерований AI, замінюючи текст на ключі.
92
+ - **Compiler:** Ви копіюєте та вставляєте код від AI — і все працює одразу.
93
+
94
+ ## Перевірка реальності: чому «магія» небезпечна
95
+
96
+ Хоча "магія" приваблює, абстракція просочується. Залежність від інструменту збірки, який має зрозуміти людський намір, призводить до архітектурної крихкості.
97
+
98
+ ### Евристична крихкість (гра вгадування)
99
+
100
+ Компілятор має вгадувати, що є контентом, а що — кодом. Це призводить до крайніх випадків, де ви в кінцевому підсумку "боретеся" з інструментом.
101
+
102
+ Розгляньте ці сценарії:
103
+
104
+ - Чи буде витягнуто `<span className="active"></span>`? (Це рядок, але скоріше за все — клас).
105
+ - Чи буде витягнуто `<span status="pending"></span>`? (Це значення пропса).
106
+ - Чи буде витягнуто `<span>{"Hello World"}</span>`? (Це JS-вираз).
107
+ - Чи буде витягнуто `<span>Hello {name}. How are you?</span>`? (Інтерполяція складна).
108
+ - Чи буде витягнуто `<span aria-label="Image of cat"></span>`? (Атрибути доступності потребують перекладу).
109
+ - Чи буде витягнуто `<span data-testid="my-element"></span>`? (Ідентифікатори тестів НЕ повинні перекладатися).
110
+ - Чи буде витягнуто `<MyComponent errorMessage="An error occurred" />`?
111
+ - Чи буде витягнуто `<p>This is a paragraph{" "}\n containing multiple lines</p>`?
112
+ - Чи буде витягнуто результат функції в `<p>{getStatusMessage()}</p>`?
113
+ - Чи буде витягнуто `<div>{isLoading ? "The page is loading" : <MyComponent/>} </div>`?
114
+ - Чи буде витягнуто ідентифікатор продукту на кшталт `<span>AX-99</span>`?
115
+
116
+ Ви неминуче додаєте спеціальні коментарі (наприклад `// ігнорувати-переклад`, або певні пропси, як-от `data-compiler-ignore="true"`) щоб запобігти порушенню логіки застосунку.
117
+
118
+ ### Як Intlayer справляється з цією складністю?
119
+
120
+ Intlayer використовує комбінований підхід для визначення, чи слід витягувати поле для перекладу, намагаючись мінімізувати false positives:
121
+
122
+ 1. **Аналіз AST:** Він перевіряє тип елемента (наприклад, розрізняє `reactNode`, `label` або проп `title`).
123
+ 2. **Розпізнавання шаблонів:** Воно визначає, чи рядок починається з великої літери або містить пробіли, що вказує на те, що це, ймовірно, текст, читабельний людиною, а не ідентифікатор коду.
124
+
125
+ ### Жорстке обмеження динамічних даних
126
+
127
+ Видобування компілятором покладається на **статичний аналіз**. Він повинен бачити буквальний рядок у вашому коді, щоб згенерувати стабільний ідентифікатор.
128
+ Якщо ваш API повертає рядок коду помилки на кшталт `server_error`, ви не зможете перекласти його за допомогою компілятора, оскільки компілятор не знає про існування цього рядка під час збірки. Ви змушені створити вторинну систему, що працює тільки під час виконання ("runtime-only"), тільки для динамічних даних.
129
+
130
+ ### Відсутність чанкування
131
+
132
+ Деякі компілятори не розбивають переклади на чанки по сторінках. Якщо ваш компілятор генерує великий JSON-файл для кожної мови (наприклад, `./lang/en.json`, `./lang/fr.json` тощо), ймовірно, ви в кінцевому підсумку завантажите контент усіх ваших сторінок для однієї відвідуваної сторінки. Також кожен компонент, що використовує цей контент, ймовірно буде гідратований значно більшим обсягом даних, ніж необхідно, що потенційно може призвести до проблем з продуктивністю.
133
+
134
+ Також будьте уважні з динамічним завантаженням ваших перекладів. Якщо цього не робити, ви завантажите вміст для всіх мов поряд із поточною.
135
+
136
+ > Щоб проілюструвати проблему, уявіть сайт з 10 сторінками й 10 мовами (усі повністю унікальні). Ви завантажите вміст для ще 99 сторінок (10 × 10 - 1).
137
+
138
+ ### «Вибух чанків» та мережеві водоспади
139
+
140
+ Щоб вирішити проблему чанкінгу, деякі рішення пропонують розбивку на чанки на рівні компонентів або навіть ключів. Проте проблема вирішується лише частково. Основний аргумент на користь таких рішень часто звучить як «Ваш контент буде tree-shaken».
141
+
142
+ Дійсно, якщо ви завантажуєте контент статично, ваше рішення видалить невикористовуваний вміст через tree-shaking, але в підсумку ви все одно опинитеся з вмістом усіх мов, завантаженим разом із вашим застосунком.
143
+
144
+ Тож чому б не завантажувати його динамічно? Так, у такому випадку ви завантажите більше, ніж потрібно, але це не обходиться без компромісів.
145
+
146
+ Динамічне завантаження контенту ізолює кожен фрагмент у власний chunk, який буде завантажений лише тоді, коли компонент відрендериться. Це означає, що ви здійсните один HTTP-запит на кожний текстовий блок. 1,000 текстових блоків на вашій сторінці? → 1,000 HTTP-запитів до ваших серверів. І щоб зменшити шкоду та оптимізувати час першого рендерингу вашого додатка, вам доведеться вставити кілька Suspense boundaries або Skeleton Loaders.
147
+
148
+ > Note: Навіть з Next.js та SSR ваші компоненти все одно будуть гідратовані після завантаження, тож HTTP-запити все одно будуть здійснені.
149
+
150
+ > Note: Щоб відстежити кожен запит чанка, ви можете перевірити вкладку `network` у інструментах розробника вашого браузера.
151
+
152
+ Рішення? Використати підхід, що дозволяє оголошувати контент з обмеженою областю дії, як це роблять `i18next`, `next-intl` або `intlayer`.
153
+
154
+ > Примітка: `i18next` та `next-intl` вимагають від вас вручну керувати імпортами namespace / messages для кожної сторінки, щоб оптимізувати розмір бандла. Варто використовувати bundle analyzer, такий як `rollup-plugin-visualizer` (vite), `@next/bundle-analyzer` (next.js) або `webpack-bundle-analyzer` (React CRA / Angular / etc), щоб виявити, чи ви забруднюєте бандл невикористаними перекладами.
155
+
156
+ ### Навантаження на продуктивність під час виконання
157
+
158
+ Щоб зробити переклади реактивними (щоб вони оновлювалися миттєво при зміні мови), компілятор часто вставляє хуки для керування станом у кожен компонент.
159
+
160
+ Якщо ви рендерите список з 5 000 елементів, ви ініціалізуєте 5 000 хуків `useState` і `useEffect` виключно для тексту. React має ідентифікувати та повторно відрендерити всіх 5 000 споживачів одночасно. Це спричиняє масштабне блокування "Main Thread", яке заморожує UI під час переключення. Це витрачає пам'ять і процесорні цикли, які заощаджують декларативні бібліотеки (які зазвичай використовують один провайдер `Context`).
161
+
162
+ > Зауважте, що ця проблема подібна і для інших фреймворків, не лише React.
163
+
164
+ ## Пастка: Vendor Lock-in
165
+
166
+ Будьте обережні при виборі i18n-рішення, яке дозволяє витягувати або мігрувати ключі перекладу.
167
+
168
+ У випадку декларативної бібліотеки ваш вихідний код явно містить наміри перекладу: це ваші ключі, і ви їх контролюєте. Якщо ви хочете змінити бібліотеку, зазвичай потрібно лише оновити імпорт.
169
+
170
+ З компіляторним підходом ваш вихідний код може бути просто англійським текстом, без жодних слідів логіки перекладу: усе приховано в конфігурації інструмента збірки. Якщо цей плагін перестане підтримуватися або ви захочете змінити рішення, ви можете опинитися заблокованими. Немає простого способу «eject»: у вашому коді немає придатних ключів, і, можливо, доведеться заново згенерувати всі переклади для нової бібліотеки.
171
+
172
+ Деякі рішення також пропонують сервіси генерації перекладів. Закінчилися кредити? Більше немає перекладів.
173
+
174
+ Компілери часто хешують текст (наприклад, `"Hello World"` -> `x7f2a`). Ваші файли перекладів виглядають як `{ "x7f2a": "Привіт, світ" }`. Пастка: якщо ви зміните бібліотеку, нова бібліотека побачить `"Hello World"` і шукатиме цей ключ. Вона його не знайде, бо ваш файл перекладів заповнений хешами (`x7f2a`).
175
+
176
+ ### Прив'язка до платформи
177
+
178
+ Вибираючи підхід на основі компілятора, ви прив'язуєте себе до базової платформи. Наприклад, певні компілятори недоступні для всіх бандлерів (таких як Vite, Turbopack або Metro). Це може ускладнити майбутні міграції, і можливо вам доведеться використовувати кілька рішень, щоб охопити всі ваші додатки.
179
+
180
+ Але ширше кажучи, навіть без обмежень конкретного bundler'а, кожне i18n-рішення вводить власні конвенції для оголошення та управління контентом. Внаслідок цього будь-який підхід до i18n апріорі створює певний рівень залежності від постачальника (vendor lock-in) саме цього рішення. Крім того, кілька провідних i18n-бібліотек спонсоруються, фінансуються або підтримуються компаніями, що працюють у галузі локалізації.
181
+
182
+ ## Інша сторона: ризики декларативного підходу
183
+
184
+ Чесно кажучи, традиційний декларативний підхід теж не ідеальний. У нього є свої «підводні камені».
185
+
186
+ 1. **Пекло неймспейсів:** Часто доводиться вручну керувати тим, які JSON-файли завантажувати (`common.json`, `dashboard.json`, `footer.json`). Якщо забути один — користувач побачить сирі ключі.
187
+ 2. **Надмірне завантаження:** Без ретельної конфігурації дуже легко випадково завантажити _усі_ ваші ключі перекладу для _всіх_ сторінок при початковому завантаженні, що роздутує розмір вашого бандла.
188
+ 3. **Дрейф синхронізації:** Часто ключі залишаються в JSON-файлі задовго після того, як компонент, який їх використовував, був видалений. Ваші файли перекладів ростуть нескінченно, заповнені "zombie keys."
189
+
190
+ ## Проміжний підхід Intlayer
191
+
192
+ Саме тут інструменти на кшталт **Intlayer** намагаються інновувати. Intlayer розуміє, що хоч компілятори й потужні, неявна магія є небезпечною.
193
+
194
+ Intlayer пропонує змішаний підхід, який дозволяє скористатися перевагами обох підходів: декларативного управління контентом, сумісного з його компілятором, щоб заощадити час розробки.
195
+
196
+ Навіть якщо ви не використовуєте компілятор Intlayer, Intlayer пропонує команду `transform` (також доступну через розширення для VSCode). Замість того, щоб робити магію в прихованому кроці збірки, вона фактично може **перезаписати код ваших компонентів**. Вона сканує ваш текст і замінює його на явні декларації контенту у вашому кодовому сховищі.
197
+
198
+ Це дає вам найкраще з обох підходів:
199
+
200
+ 1. **Гранулярність:** Ви зберігаєте переклади поруч із компонентами (підвищуючи модульність і ефективність tree-shaking).
201
+ 2. **Безпека:** Переклад стає явним кодом, а не прихованою магією на етапі збірки.
202
+ 3. **Відсутність lock-in:** Оскільки код перетворюється в декларативну структуру у вашому репозиторії, ви можете легко натиснути Tab або скористатися Copilot вашого IDE, щоб згенерувати декларації контенту — ви не ховаєте логіку в плагіні webpack.
203
+
204
+ ## Висновок
205
+
206
+ Отже, що обрати?
207
+
208
+ **Якщо ви створюєте MVP або хочете рухатися швидко:**
209
+ Підхід на основі компілятора — цілком прийнятний вибір. Він дозволяє працювати надзвичайно швидко. Вам не потрібно хвилюватися про структури файлів чи ключі. Ви просто розробляєте. Технічний борг — це проблема «майбутнього вас».
210
+
211
+ **Якщо ви Junior Developer або вам не важлива оптимізація:**
212
+ Якщо ви прагнете мінімального ручного управління, підхід на основі компілятора, ймовірно, найкращий. Вам не доведеться самостійно працювати з ключами чи файлами перекладів — просто пишіть текст, а компілятор автоматизує решту. Це зменшує зусилля налаштування й типові помилки i18n, пов'язані з ручними кроками.
213
+
214
+ **Якщо ви інтернаціоналізуєте існуючий проєкт, який уже містить тисячі компонентів для рефакторингу:**
215
+ Підхід, заснований на компіляторі, може бути прагматичним вибором у цій ситуації. Початкова фаза витягування може заощадити тижні або місяці ручної роботи. Однак розгляньте використання інструмента, такого як команда Intlayer `transform`, яка може витягувати рядки й перетворювати їх на явні декларативні оголошення контенту. Це дає вам швидкість автоматизації, одночасно зберігаючи безпеку та портативність декларативного підходу. Ви отримуєте найкраще з обох світів: швидку початкову міграцію без довгострокового архітектурного боргу.
216
+
217
+ **Якщо ви розробляєте професійний додаток корпоративного рівня:**
218
+ «Магія» зазвичай — погана ідея. Вам потрібен контроль.
219
+
220
+ - Потрібно обробляти динамічні дані з бекендів.
221
+ - Потрібно забезпечити продуктивність на слабких пристроях (уникати hook explosions).
222
+ - Необхідно переконатися, що ви не залишитеся прив'язаними до конкретного інструмента збірки назавжди.
223
+
224
+ Для професійних додатків **декларативне управління контентом** (наприклад, Intlayer або перевірені бібліотеки) залишається золотим стандартом. Воно відокремлює ваші обов'язки, зберігає архітектуру чистою і гарантує, що здатність вашого застосунку підтримувати кілька мов не залежить від «чорної скриньки» компілятора, який вгадує ваші наміри.