@intlayer/docs 8.7.5-canary.0 → 8.7.5

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 (81) hide show
  1. package/blog/de/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  2. package/blog/en-GB/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  3. package/blog/es/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  4. package/blog/fr/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  5. package/blog/id/list_i18n_technologies/frameworks/svelte.md +0 -2
  6. package/blog/it/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  7. package/blog/ja/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  8. package/blog/ko/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  9. package/blog/pl/list_i18n_technologies/frameworks/svelte.md +0 -2
  10. package/blog/pt/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  11. package/blog/ru/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  12. package/blog/vi/list_i18n_technologies/frameworks/svelte.md +0 -2
  13. package/blog/zh/next-i18next_vs_next-intl_vs_intlayer.md +0 -2
  14. package/dist/cjs/generated/docs.entry.cjs +60 -0
  15. package/dist/cjs/generated/docs.entry.cjs.map +1 -1
  16. package/dist/esm/generated/docs.entry.mjs +60 -0
  17. package/dist/esm/generated/docs.entry.mjs.map +1 -1
  18. package/dist/types/generated/docs.entry.d.ts +3 -0
  19. package/dist/types/generated/docs.entry.d.ts.map +1 -1
  20. package/docs/ar/benchmark/index.md +29 -0
  21. package/docs/ar/benchmark/nextjs.md +227 -0
  22. package/docs/ar/benchmark/tanstack.md +193 -0
  23. package/docs/ar/intlayer_with_tanstack.md +0 -2
  24. package/docs/de/benchmark/index.md +29 -0
  25. package/docs/de/benchmark/nextjs.md +227 -0
  26. package/docs/de/benchmark/tanstack.md +193 -0
  27. package/docs/en/benchmark/___NOTE.md +82 -0
  28. package/docs/en/benchmark/___nextjs.md +195 -0
  29. package/docs/en/benchmark/___tanstack.md +187 -0
  30. package/docs/en/benchmark/index.md +29 -0
  31. package/docs/en/benchmark/nextjs.md +228 -0
  32. package/docs/en/benchmark/tanstack.md +217 -0
  33. package/docs/en-GB/benchmark/index.md +29 -0
  34. package/docs/en-GB/benchmark/nextjs.md +228 -0
  35. package/docs/en-GB/benchmark/tanstack.md +193 -0
  36. package/docs/es/benchmark/index.md +29 -0
  37. package/docs/es/benchmark/nextjs.md +226 -0
  38. package/docs/es/benchmark/tanstack.md +193 -0
  39. package/docs/fr/benchmark/index.md +29 -0
  40. package/docs/fr/benchmark/nextjs.md +227 -0
  41. package/docs/fr/benchmark/tanstack.md +193 -0
  42. package/docs/hi/benchmark/index.md +29 -0
  43. package/docs/hi/benchmark/nextjs.md +227 -0
  44. package/docs/hi/benchmark/tanstack.md +193 -0
  45. package/docs/id/benchmark/index.md +29 -0
  46. package/docs/id/benchmark/nextjs.md +227 -0
  47. package/docs/id/benchmark/tanstack.md +193 -0
  48. package/docs/id/intlayer_with_react_native+expo.md +0 -2
  49. package/docs/it/benchmark/index.md +29 -0
  50. package/docs/it/benchmark/nextjs.md +227 -0
  51. package/docs/it/benchmark/tanstack.md +193 -0
  52. package/docs/ja/benchmark/index.md +29 -0
  53. package/docs/ja/benchmark/nextjs.md +227 -0
  54. package/docs/ja/benchmark/tanstack.md +193 -0
  55. package/docs/ko/benchmark/index.md +29 -0
  56. package/docs/ko/benchmark/nextjs.md +227 -0
  57. package/docs/ko/benchmark/tanstack.md +193 -0
  58. package/docs/ko/intlayer_with_tanstack.md +0 -2
  59. package/docs/pl/benchmark/index.md +29 -0
  60. package/docs/pl/benchmark/nextjs.md +227 -0
  61. package/docs/pl/benchmark/tanstack.md +193 -0
  62. package/docs/pt/benchmark/index.md +29 -0
  63. package/docs/pt/benchmark/nextjs.md +227 -0
  64. package/docs/pt/benchmark/tanstack.md +193 -0
  65. package/docs/ru/benchmark/index.md +29 -0
  66. package/docs/ru/benchmark/nextjs.md +227 -0
  67. package/docs/ru/benchmark/tanstack.md +193 -0
  68. package/docs/tr/benchmark/index.md +29 -0
  69. package/docs/tr/benchmark/nextjs.md +227 -0
  70. package/docs/tr/benchmark/tanstack.md +193 -0
  71. package/docs/uk/benchmark/index.md +29 -0
  72. package/docs/uk/benchmark/nextjs.md +227 -0
  73. package/docs/uk/benchmark/tanstack.md +193 -0
  74. package/docs/vi/benchmark/index.md +29 -0
  75. package/docs/vi/benchmark/nextjs.md +227 -0
  76. package/docs/vi/benchmark/tanstack.md +193 -0
  77. package/docs/zh/benchmark/index.md +29 -0
  78. package/docs/zh/benchmark/nextjs.md +227 -0
  79. package/docs/zh/benchmark/tanstack.md +193 -0
  80. package/package.json +6 -6
  81. package/src/generated/docs.entry.ts +60 -0
@@ -0,0 +1,227 @@
1
+ ---
2
+ createdAt: 2026-04-20
3
+ updatedAt: 2026-04-21
4
+ title: Solusi i18n terbaik untuk Next.js tahun 2026 - Laporan Benchmark
5
+ description: Bandingkan library internasionalisasi (i18n) Next.js seperti next-intl, next-i18next, dan Intlayer. Laporan performa terperinci tentang ukuran bundle, kebocoran, dan reaktivitas.
6
+ keywords:
7
+ - benchmark
8
+ - i18n
9
+ - intl
10
+ - nextjs
11
+ - performa
12
+ - intlayer
13
+ slugs:
14
+ - doc
15
+ - benchmark
16
+ - nextjs
17
+ author: Aymeric PINEAU
18
+ applicationTemplate: https://github.com/intlayer-org/benchmark-i18n
19
+ history:
20
+ - version: 8.7.5
21
+ date: 2026-01-06
22
+ changes: "Inisialisasi benchmark"
23
+ ---
24
+
25
+ # Library i18n Next.js — Laporan Benchmark 2026
26
+
27
+ Halaman ini adalah laporan benchmark untuk solusi i18n pada Next.js.
28
+
29
+ ## Daftar Isi
30
+
31
+ <Toc/>
32
+
33
+ ## Benchmark Interaktif
34
+
35
+ <I18nBenchmark framework="nextjs" vertical/>
36
+
37
+ ## Referensi hasil:
38
+
39
+ <iframe
40
+ src="https://intlayer.org/markdown?url=https%3A%2F%2Fraw.githubusercontent.com%2Fintlayer-org%2Fbenchmark-i18n%2Fmain%2Freport%2Fscripts%2Fsummarize-nextjs.md"
41
+ width="100%"
42
+ height="600px"
43
+ style="border:none;">
44
+ </iframe>
45
+
46
+ > https://intlayer.org/markdown?url=https%3A%2F%2Fraw.githubusercontent.com%2Fintlayer-org%2Fbenchmark-i18n%2Fmain%2Freport%2Fscripts%2Fsummarize-nextjs.md
47
+
48
+ Lihat repositori benchmark lengkap [di sini](https://github.com/intlayer-org/benchmark-i18n).
49
+
50
+ ## Pendahuluan
51
+
52
+ Library internasionalisasi memiliki dampak besar pada aplikasi Anda. Risiko utamanya adalah memuat konten untuk setiap halaman dan setiap bahasa padahal pengguna hanya mengunjungi satu halaman.
53
+
54
+ Seiring berkembangnya aplikasi Anda, ukuran bundle dapat tumbuh secara eksponensial, yang dapat mengganggu performa secara nyata.
55
+
56
+ Sebagai contoh, pada kasus terburuk, setelah diinternasionalisasi, halaman Anda bisa menjadi hampir 4 kali lebih besar.
57
+
58
+ Dampak lain dari library i18n adalah pengembangan yang lebih lambat. Mengubah komponen menjadi konten multibahasa di berbagai bahasa memakan waktu lama.
59
+
60
+ Karena masalah ini sulit, banyak solusi tersedia—beberapa berfokus pada DX (Developer Experience), yang lain pada performa atau skalabilitas, dan sebagainya.
61
+
62
+ Intlayer mencoba mengoptimalkan di semua dimensi ini.
63
+
64
+ ## Uji aplikasi Anda
65
+
66
+ Untuk mengungkap masalah ini, saya membangun pemindai gratis yang dapat Anda coba [di sini](https://intlayer.org/i18n-seo-scanner).
67
+
68
+ <iframe src="https://intlayer.org/i18n-seo-scanner" width="100%" height="600px" style="border:none;"/>
69
+
70
+ ## Masalah
71
+
72
+ Ada dua cara utama untuk membatasi dampak aplikasi multibahasa pada bundle Anda:
73
+
74
+ - Memisahkan JSON (atau konten) Anda ke berbagai file / variabel / namespace sehingga bundler dapat melakukan tree-shaking pada konten yang tidak digunakan untuk halaman tertentu.
75
+ - Memuat konten halaman Anda secara dinamis hanya dalam bahasa pengguna.
76
+
77
+ Batasan teknis untuk pendekatan ini:
78
+
79
+ **Pemuatan dinamis**
80
+
81
+ Bahkan saat Anda mendeklarasikan rute seperti `[locale]/page.tsx`, dengan Webpack atau Turbopack, dan bahkan jika `generateStaticParams` didefinisikan, bundler tidak memperlakukan `locale` sebagai konstanta statis. Itu berarti bundler mungkin menarik konten untuk semua bahasa ke dalam setiap halaman. Cara utama untuk membatasi ini adalah memuat konten melalui impor dinamis (misalnya `import('./locales/${locale}.json')`).
82
+
83
+ Apa yang terjadi pada saat build adalah Next.js memancarkan satu bundle JS per lokal (misalnya `./locales_fr_12345.js`). Setelah situs dikirim ke klien, saat halaman berjalan, browser melakukan permintaan HTTP ekstra untuk file JS yang dibutuhkan (misalnya `./locales_fr_12345.js`).
84
+
85
+ > Cara lain untuk mengatasi masalah yang sama adalah dengan menggunakan `fetch()` untuk memuat JSON secara dinamis. Itulah cara kerja `Tolgee` saat JSON berada di bawah `/public`, atau `next-translate`, yang mengandalkan `getStaticProps` untuk memuat konten. Alurnya sama: browser membuat permintaan HTTP ekstra untuk memuat aset tersebut.
86
+
87
+ **Pemisahan konten (Content splitting)**
88
+
89
+ Jika Anda menggunakan sintaks seperti `const t = useTranslation()` + `t('objek-saya.sub-objek-saya.kunci-saya')`, seluruh JSON biasanya harus ada dalam bundle sehingga library dapat mem-parsing-nya dan menyelesaikan kuncinya. Banyak dari konten tersebut kemudian dikirimkan bahkan saat tidak digunakan di halaman.
90
+
91
+ Untuk memitigasi hal ini, beberapa library meminta Anda untuk mendeklarasikan per halaman namespace mana yang akan dimuat—misalnya `next-i18next`, `next-intl`, `lingui`, `next-translate`, `next-international`.
92
+
93
+ Sebaliknya, `Paraglide` menambahkan langkah ekstra sebelum build untuk mengubah JSON menjadi simbol datar seperti `const en_my_var = () => 'nilai saya'`. Secara teori, hal itu memungkinkan tree-shaking konten yang tidak digunakan pada halaman. Seperti yang akan kita lihat, metode tersebut masih memiliki trade-off.
94
+
95
+ Terakhir, `Intlayer` menerapkan optimisasi waktu build sehingga `useIntlayer('kunci-saya')` diganti dengan konten yang sesuai secara langsung.
96
+
97
+ ## Metodologi
98
+
99
+ Untuk benchmark ini, kami membandingkan library berikut:
100
+
101
+ - `Base App` (Tanpa library i18n)
102
+ - `next-intlayer` (v8.7.5)
103
+ - `next-i18next` (v16.0.5)
104
+ - `next-intl` (v4.9.1)
105
+ - `@lingui/core` (v5.3.0)
106
+ - `next-translate` (v3.1.2)
107
+ - `next-international` (v1.3.1)
108
+ - `@inlang/paraglide-js` (v2.15.1)
109
+ - `tolgee` (v7.0.0)
110
+ - `@lingo.dev/compiler` (v0.4.0)
111
+ - `wuchale` (v0.22.11)
112
+ - `gt-next` (v6.16.5)
113
+
114
+ Saya menggunakan `Next.js` versi `16.2.4` dengan App Router.
115
+
116
+ Saya membangun aplikasi multibahasa dengan **10 halaman** dan **10 bahasa**.
117
+
118
+ Saya membandingkan **empat strategi pemuatan**:
119
+
120
+ | Strategi | Tanpa namespace (global) | Dengan namespace (terlingkup/scoped) |
121
+ | :------------------- | :------------------------------------------- | :---------------------------------------------------------------------- |
122
+ | **Pemuatan statis** | **Static**: Semuanya di memori saat startup. | **Scoped static**: Dipisah per namespace; semuanya dimuat saat startup. |
123
+ | **Pemuatan dinamis** | **Dynamic**: Pemuatan on-demand per lokal. | **Scoped dynamic**: Pemuatan granular per namespace dan lokal. |
124
+
125
+ ## Ringkasan strategi
126
+
127
+ - **Static**: Sederhana; tidak ada latensi jaringan setelah pemuatan awal. Kekurangannya: ukuran bundle yang besar.
128
+ - **Dynamic**: Mengurangi beban awal (lazy-loading). Ideal bila Anda memiliki banyak lokal.
129
+ - **Scoped static**: Menjaga kode tetap teratur (pemisahan logis) tanpa permintaan jaringan ekstra yang kompleks.
130
+ - **Scoped dynamic**: Pendekatan terbaik untuk _code splitting_ dan performa. Meminimalkan memori dengan memuat hanya apa yang dibutuhkan tampilan saat ini dan lokal yang aktif.
131
+
132
+ ### Apa yang saya ukur:
133
+
134
+ Saya menjalankan aplikasi multibahasa yang sama di browser asli untuk setiap stack, lalu mencatat apa yang sebenarnya muncul di jaringan dan berapa lama waktu yang dibutuhkan. Ukuran dilaporkan **setelah kompresi web normal**, karena itu lebih mendekati apa yang sebenarnya diunduh orang daripada jumlah sumber mentah.
135
+
136
+ - **Ukuran library internasionalisasi**: Setelah pemaketan, tree-shaking, dan minifikasi, ukuran library i18n adalah ukuran kode provider (misalnya `NextIntlClientProvider`) + hook (misalnya `useTranslations`) dalam komponen kosong. Ini tidak termasuk pemuatan file terjemahan. Ini menjawab seberapa mahal library tersebut sebelum konten masuk dalam perhitungan.
137
+
138
+ - **JavaScript per halaman**: Untuk setiap rute benchmark, berapa banyak skrip yang ditarik browser untuk kunjungan tersebut, dirata-ratakan di seluruh halaman dalam suite (dan di seluruh lokal di mana laporan menggabungkannya). Halaman yang berat adalah halaman yang lambat.
139
+
140
+ - **Kebocoran dari lokal lain (Leakage)**: Ini adalah konten dari halaman yang sama tetapi dalam bahasa lain yang tidak sengaja dimuat di halaman yang diaudit. Konten ini tidak perlu dan harus dihindari (misalnya konten halaman `/fr/about` dalam bundle halaman `/en/about`).
141
+
142
+ - **Kebocoran dari rute lain**: Ide yang sama untuk **layar lain** di aplikasi: apakah teks mereka ikut terbawa saat Anda hanya membuka satu halaman (misalnya konten halaman `/en/about` dalam bundle halaman `/en/contact`). Skor tinggi menandakan pemisahan yang lemah atau bundle yang terlalu luas.
143
+
144
+ - **Ukuran bundle komponen rata-rata**: Bagian UI umum diukur **satu per satu** daripada bersembunyi di dalam satu angka aplikasi yang besar. Ini menunjukkan apakah internasionalisasi secara diam-diam membesarkan komponen sehari-hari. Misalnya, jika komponen Anda merender ulang, ia akan memuat semua data itu dari memori. Melampirkan JSON raksasa ke komponen mana pun, seperti menghubungkan gudang besar data yang tidak digunakan yang akan memperlambat performa komponen Anda.
145
+
146
+ - **Responsivitas pergantian bahasa**: Saya mengganti bahasa menggunakan kontrol aplikasi itu sendiri dan mencatat berapa lama waktu yang dibutuhkan hingga halaman benar-benar berganti — apa yang akan diperhatikan pengunjung, bukan langkah mikro laboratorium.
147
+
148
+ - **Pekerjaan render setelah perubahan bahasa**: Tindak lanjut yang lebih sempit: seberapa besar upaya yang dilakukan antarmuka untuk menggambar ulang bahasa baru setelah pergantian dilakukan. Berguna ketika waktu yang "dirasakan" dan biaya framework berbeda.
149
+
150
+ - **Waktu pemuatan halaman awal**: Dari navigasi hingga browser menganggap halaman dimuat sepenuhnya untuk skenario yang saya uji. Bagus untuk membandingkan cold start.
151
+
152
+ - **Waktu hidrasi (Hydration)**: Ketika aplikasi mengeksposnya, berapa lama waktu yang dihabiskan klien untuk mengubah HTML server menjadi sesuatu yang benar-benar dapat Anda klik. Tanda strip dalam tabel berarti implementasi tersebut tidak memberikan angka hidrasi yang andal dalam benchmark ini.
153
+
154
+ ## Hasil secara terperinci
155
+
156
+ ### 1 — Solusi yang harus dihindari
157
+
158
+ Beberapa solusi, seperti `gt-next` atau `lingo.dev`, jelas paling baik dihindari. Mereka menggabungkan ketergantungan pada vendor (vendor lock-in) dengan pengotoran basis kode Anda. Meskipun menghabiskan banyak waktu mencoba menerapkannya, saya tidak pernah berhasil membuatnya bekerja—baik di TanStack Start maupun di Next.js.
159
+
160
+ Masalah yang ditemui:
161
+
162
+ **(General Translation)** (`gt-next@6.16.5`):
163
+
164
+ - Untuk aplikasi 110kb, `gt-react` menambahkan lebih dari 440kb ekstra.
165
+ - `Quota Exceeded, please upgrade your plan` pada build pertama dengan General Translation.
166
+ - Terjemahan tidak dirender; saya mendapatkan error `Error: <T> used on the client-side outside of <GTProvider>`, yang tampaknya merupakan bug pada library.
167
+ - Saat menerapkan **gt-tanstack-start-react**, saya juga menemukan [masalah](https://github.com/generaltranslation/gt/issues/1210#event-24510646961) dengan library tersebut: `does not provide an export named 'printAST' - @formatjs/icu-messageformat-parser`, yang membuat aplikasi rusak. Setelah melaporkan masalah ini, pengelola memperbaikinya dalam waktu 24 jam.
168
+ - Library ini memblokir rendering statis halaman Next.js.
169
+
170
+ **(Lingo.dev)** (`@lingo.dev/compiler@0.4.0`):
171
+
172
+ - Kuota AI terlampaui, memblokir seluruh build—sehingga Anda tidak dapat merilis ke produksi tanpa membayar.
173
+ - Kompilator melewatkan hampir 40% konten yang diterjemahkan. Saya harus menulis ulang semua `.map` menjadi blok komponen datar agar dapat berfungsi.
174
+ - CLI mereka penuh bug dan sering mereset file config tanpa alasan.
175
+ - Saat build, library ini menghapus total JSON yang dihasilkan ketika ada konten baru ditambahkan. Akibatnya, segelintir kunci dapat menghapus lebih dari 300 kunci yang ada.
176
+
177
+ ### 2 — Solusi eksperimental
178
+
179
+ **(Wuchale)** (`wuchale@0.22.11`):
180
+
181
+ Ide di balik `Wuchale` menarik tetapi belum layak. Saya menemui masalah reaktivitas dan harus memaksa perenderan ulang provider untuk menjalankan aplikasi. Dokumentasinya juga cukup tidak jelas, yang membuat adopsi lebih sulit.
182
+
183
+ **(Paraglide)** (`@inlang/paraglide-js@2.15.1`):
184
+
185
+ `Paraglide` menawarkan pendekatan yang inovatif dan terencana dengan baik. Meskipun demikian, dalam benchmark ini tree-shaking yang diiklankan perusahaan mereka tidak berfungsi untuk setup Next.js atau TanStack Start saya. Alur kerja dan DX lebih kompleks daripada opsi lainnya.
186
+ Secara pribadi saya tidak suka harus membuat ulang file JS sebelum setiap push, yang menciptakan risiko konflik merge konstan melalui PR. Alat ini juga tampak lebih fokus pada Vite daripada Next.js.
187
+ Terakhir, dibandingkan dengan solusi lain, Paraglide tidak menggunakan store (misalnya React context) untuk mengambil lokal saat ini untuk merender konten. Untuk setiap node yang di-parsing, ia akan meminta lokal dari localStorage / cookie dst. Ini mengarah pada eksekusi logika yang tidak perlu yang berdampak pada reaktivitas komponen.
188
+
189
+ ### 3 — Solusi yang dapat diterima
190
+
191
+ **(Tolgee)** (`tolgee@7.0.0`):
192
+
193
+ `Tolgee` mengatasi banyak masalah yang disebutkan sebelumnya. Saya merasa lebih sulit untuk mengadopsinya daripada alat serupa. Ia tidak memberikan type safety, yang juga membuat pendeteksian kunci yang hilang saat compile time lebih sulit. Saya harus membungkus fungsi Tolgee dengan fungsi saya sendiri untuk menambahkan deteksi kunci yang hilang.
194
+
195
+ **(Next Intl)** (`next-intl@4.9.1`):
196
+
197
+ `next-intl` adalah opsi yang paling tren dan yang paling didorong oleh agen AI, tetapi menurut pandangan saya itu salah. Memulainya memang mudah. Dalam praktiknya, mengoptimalkan untuk membatasi kebocoran itu kompleks. Menggabungkan pemuatan dinamis + namespacing + tipe TypeScript sangat memperlambat pengembangan. Paket ini juga cukup berat (~13kb untuk `NextIntlClientProvider` + `useTranslations`, yang berarti lebih dari 2x lipat `next-intlayer`). **next-intl** dulu memblokir rendering statis halaman Next.js. Ia menyediakan helper bernama `setRequestLocale()`. Itu tampaknya sudah diatasi sebagian untuk file tersentralisasi seperti `en.json` / `fr.json`, tetapi rendering statis tetap rusak ketika konten dipisah ke dalam namespace seperti `en/shared.json` / `fr/shared.json` / `es/shared.json`.
198
+
199
+ **(Next I18next)** (`next-i18next@16.0.5`):
200
+
201
+ `next-i18next` mungkin adalah opsi yang paling populer karena merupakan salah satu solusi i18n pertama untuk aplikasi JavaScript. Ia memiliki banyak plugin komunitas. Ia memiliki kelemahan utama yang sama dengan `next-intl`. Paketnya sangat berat (~18kb untuk `I18nProvider` + `useTranslation`, sekitar 3x lipat `next-intlayer`).
202
+
203
+ Format pesan juga berbeda: `next-intl` menggunakan ICU MessageFormat, sementara `i18next` menggunakan formatnya sendiri.
204
+
205
+ **(Next International)** (`next-international@1.3.1`):
206
+
207
+ `next-international` juga menangani masalah di atas tetapi tidak jauh berbeda dari `next-intl` atau `next-i18next`. Ia menyertakan `scopedT()` untuk terjemahan spesifik namespace, tetapi menggunakannya pada dasarnya tidak berdampak pada ukuran bundle.
208
+
209
+ **(Lingui)** (`@lingui/core@5.3.0`):
210
+
211
+ `Lingui` sering dipuji. Secara pribadi saya merasa alur kerja `lingui extract` / `lingui compile` lebih kompleks daripada alternatifnya, tanpa keunggulan yang jelas. Saya juga menyadari sintaksis yang tidak konsisten yang membingungkan AI (misalnya `t()`, `t''`, `i18n.t()`, `<Trans>`).
212
+
213
+ ### 4 — Rekomendasi
214
+
215
+ **(Next Translate)** (`next-translate@3.1.2`):
216
+
217
+ `next-translate` adalah rekomendasi utama saya jika Anda menyukai API bergaya `t()`. Ini elegan melalui `next-translate-plugin`, memuat namespace melalui `getStaticProps` dengan loader Webpack / Turbopack. Ini juga merupakan opsi teringan di sini (~2,5kb). Untuk namespacing, mendefinisikan namespace per halaman atau rute di config dipikirkan dengan matang dan lebih mudah dipelihara daripada alternatif utama seperti **next-intl** atau **next-i18next**. Di versi `3.1.2`, saya mencatat bahwa rendering statis tidak berfungsi; Next.js kembali ke rendering dinamis.
218
+
219
+ **(Intlayer)** (`next-intlayer@8.7.5`):
220
+
221
+ Saya tidak akan secara pribadi menilai `next-intlayer` demi objektivitas, karena itu adalah solusi saya sendiri.
222
+
223
+ ### Catatan pribadi
224
+
225
+ Catatan ini bersifat pribadi dan tidak memengaruhi hasil benchmark. Di dunia i18n Anda sering melihat konsensus seputar `const t = useTranslation('xx')` + `<>{t('xx.xx')}</>`.
226
+
227
+ Dalam aplikasi React, menyuntikkan fungsi sebagai `ReactNode` adalah, dalam pandangan saya, sebuah anti-pattern. Ini juga menambah kompleksitas yang dapat dihindari dan overhead eksekusi JavaScript (meskipun hampir tidak terlihat).
@@ -0,0 +1,193 @@
1
+ ---
2
+ createdAt: 2026-04-20
3
+ updatedAt: 2026-04-21
4
+ title: Solusi i18n terbaik untuk TanStack Start tahun 2026 - Laporan Benchmark
5
+ description: Bandingkan library internasionalisasi TanStack Start seperti react-i18next, use-intl, dan Intlayer. Laporan performa terperinci tentang ukuran bundle, kebocoran, dan reaktivitas.
6
+ keywords:
7
+ - benchmark
8
+ - i18n
9
+ - intl
10
+ - tanstack
11
+ - performa
12
+ - intlayer
13
+ slugs:
14
+ - doc
15
+ - benchmark
16
+ - tanstack
17
+ author: Aymeric PINEAU
18
+ applicationTemplate: https://github.com/intlayer-org/benchmark-i18n-tanstack-start-template
19
+ history:
20
+ - version: 8.7.5
21
+ date: 2026-01-06
22
+ changes: "Inisialisasi benchmark"
23
+ ---
24
+
25
+ # Library i18n TanStack Start — Laporan Benchmark 2026
26
+
27
+ Halaman ini adalah laporan benchmark untuk solusi i18n pada TanStack Start.
28
+
29
+ ## Daftar Isi
30
+
31
+ <Toc/>
32
+
33
+ ## Benchmark Interaktif
34
+
35
+ <I18nBenchmark framework="tanstack" vertical/>
36
+
37
+ ## Referensi hasil:
38
+
39
+ <iframe
40
+ src="https://intlayer.org/markdown?url=https%3A%2F%2Fraw.githubusercontent.com%2Fintlayer-org%2Fbenchmark-i18n%2Fmain%2Freport%2Fscripts%2Fsummarize-tanstack.md"
41
+ width="100%"
42
+ height="600px"
43
+ style="border:none;">
44
+ </iframe>
45
+
46
+ > https://intlayer.org/markdown?url=https%3A%2F%2Fraw.githubusercontent.com%2Fintlayer-org%2Fbenchmark-i18n%2Fmain%2Freport%2Fscripts%2Fsummarize-tanstack.md
47
+
48
+ Lihat repositori benchmark lengkap [di sini](https://github.com/intlayer-org/benchmark-i18n/tree/main).
49
+
50
+ ## Pendahuluan
51
+
52
+ Solusi internasionalisasi adalah salah satu dependensi terberat dalam aplikasi React. Pada TanStack Start, risiko utamanya adalah mengirimkan konten yang tidak perlu: terjemahan untuk halaman lain dan lokal lain dalam bundle rute tunggal.
53
+
54
+ Seiring berkembangnya aplikasi Anda, masalah tersebut dapat dengan cepat meledakkan JavaScript yang dikirim ke klien dan memperlambat navigasi.
55
+
56
+ Dalam praktiknya, untuk implementasi yang paling tidak dioptimalkan, halaman yang diinternasionalisasi bisa menjadi beberapa kali lebih berat daripada versi tanpa i18n.
57
+
58
+ Dampak lainnya adalah pada pengalaman pengembang (DX): bagaimana Anda mendeklarasikan konten, tipe, organisasi namespace, pemuatan dinamis, dan reaktivitas saat lokal berubah.
59
+
60
+ ## Uji aplikasi Anda
61
+
62
+ Untuk mendeteksi masalah kebocoran i18n dengan cepat, saya menyiapkan pemindai gratis yang tersedia [di sini](https://intlayer.org/i18n-seo-scanner).
63
+
64
+ <iframe src="https://intlayer.org/i18n-seo-scanner" width="100%" height="600px" style="border:none;"/>
65
+
66
+ ## Masalah
67
+
68
+ Dua pengungkit sangat penting untuk membatasi biaya aplikasi multibahasa:
69
+
70
+ - Memisahkan konten per halaman / namespace sehingga Anda tidak memuat seluruh kamus saat tidak membutuhkannya.
71
+ - Memuat lokal yang tepat secara dinamis, hanya saat dibutuhkan.
72
+
73
+ Memahami batasan teknis dari pendekatan ini:
74
+
75
+ **Pemuatan dinamis**
76
+
77
+ Tanpa pemuatan dinamis, sebagian besar solusi menyimpan pesan dalam memori sejak render pertama, yang menambah overhead signifikan untuk aplikasi dengan banyak rute dan lokal.
78
+
79
+ Dengan pemuatan dinamis, Anda menerima trade-off: JS awal yang lebih sedikit, tetapi terkadang permintaan ekstra saat mengganti bahasa.
80
+
81
+ **Pemisahan konten (Content splitting)**
82
+
83
+ Sintaksis yang dibangun di sekitar `const t = useTranslation()` + `t('a.b.c')` sangat nyaman tetapi seringkali mendorong penyimpanan objek JSON besar pada saat runtime. Model tersebut membuat tree-shaking sulit kecuali library menawarkan strategi pemisahan per halaman yang nyata.
84
+
85
+ ## Metodologi
86
+
87
+ Untuk benchmark ini, kami membandingkan library berikut:
88
+
89
+ - `Base App` (Tanpa library i18n)
90
+ - `react-intlayer` (v8.7.5-canary.0)
91
+ - `react-i18next` (v17.0.2)
92
+ - `use-intl` (v4.9.1)
93
+ - `@lingui/core` (v5.3.0)
94
+ - `@inlang/paraglide-js` (v2.15.1)
95
+ - `tolgee` (v7.0.0)
96
+ - `react-intl` (v10.1.1)
97
+ - `wuchale` (v0.22.11)
98
+ - `gt-react` (vlatest)
99
+ - `lingo.dev` (v0.133.9)
100
+
101
+ Framework yang digunakan adalah `TanStack Start` dengan aplikasi multibahasa yang terdiri dari **10 halaman** dan **10 bahasa**.
102
+
103
+ Kami membandingkan **empat strategi pemuatan**:
104
+
105
+ | Strategi | Tanpa namespace (global) | Dengan namespace (terlingkup/scoped) |
106
+ | :------------------- | :------------------------------------------- | :---------------------------------------------------------------------- |
107
+ | **Pemuatu statis** | **Static**: Semuanya di memori saat startup. | **Scoped static**: Dipisah per namespace; semuanya dimuat saat startup. |
108
+ | **Pemuatan dinamis** | **Dynamic**: Pemuatan on-demand per lokal. | **Scoped dynamic**: Pemuatan granular per namespace dan lokal. |
109
+
110
+ ## Ringkasan strategi
111
+
112
+ - **Static**: Sederhana; tidak ada latensi jaringan setelah pemuatan awal. Kekurangannya: ukuran bundle yang besar.
113
+ - **Dynamic**: Mengurangi beban awal (lazy-loading). Ideal bila Anda memiliki banyak lokal.
114
+ - **Scoped static**: Menjaga kode tetap teratur (pemisahan logis) tanpa permintaan jaringan ekstra yang kompleks.
115
+ - **Scoped dynamic**: Pendekatan terbaik untuk _code splitting_ dan performa. Meminimalkan memori dengan memuat hanya apa yang dibutuhkan tampilan saat ini dan lokal yang aktif.
116
+
117
+ ## Hasil secara terperinci
118
+
119
+ ### 1 — Solusi yang harus dihindari
120
+
121
+ Beberapa solusi, seperti `gt-react` atau `lingo.dev`, jelas merupakan solusi yang sebaiknya dihindari. Mereka menggabungkan ketergantungan pada vendor (vendor lock-in) dengan pengotoran basis kode Anda. Lebih buruk lagi: meskipun menghabiskan banyak waktu mencoba menerapkannya, saya tidak pernah berhasil membuatnya bekerja dengan benar di TanStack Start (mirip dengan Next.js dengan `gt-next`).
122
+
123
+ Masalah yang ditemui:
124
+
125
+ **(General Translation)** (`gt-react@latest`):
126
+
127
+ - Untuk aplikasi sekitar 110kb, `gt-react` dapat menambahkan lebih dari 440kb ekstra (besaran yang sama seperti yang terlihat pada implementasi Next.js dalam benchmark yang sama).
128
+ - `Quota Exceeded, please upgrade your plan` pada build pertama dengan General Translation.
129
+ - Terjemahan tidak dirender; saya mendapatkan error `Error: <T> used on the client-side outside of <GTProvider>`, yang tampaknya merupakan bug pada library tersebut.
130
+ - Saat menerapkan **gt-tanstack-start-react**, saya juga menemukan [masalah](https://github.com/generaltranslation/gt/issues/1210#event-24510646961) dengan library tersebut: `does not provide an export named 'printAST' - @formatjs/icu-messageformat-parser`, yang membuat aplikasi rusak. Setelah melaporkan masalah ini, pengelola memperbaikinya dalam waktu 24 jam.
131
+ - Library ini menggunakan anti-pattern melalui fungsi `initializeGT()`, yang memblokir bundle dari tree-shaking secara bersih.
132
+
133
+ **(Lingo.dev)** (`lingo.dev@0.133.9`):
134
+
135
+ - Kuota AI terlampaui (atau memblokir dependensi server), membuat build / produksi berisiko tanpa membayar.
136
+ - Kompilator melewatkan hampir 40% konten yang diterjemahkan. Saya harus menulis ulang semua `.map` menjadi blok komponen datar agar dapat berfungsi.
137
+ - CLI mereka penuh bug dan sering mereset file config tanpa alasan.
138
+ - Saat build, library ini menghapus total JSON yang dihasilkan ketika ada konten baru ditambahkan. Akibatnya, Anda bisa berakhir dengan hanya beberapa kunci yang menghapus ratusan kunci yang ada.
139
+ - Saya menemui masalah reaktivitas dengan library tersebut di TanStack Start: saat lokal berubah saya harus memaksa perenderan ulang provider agar berfungsi.
140
+
141
+ ### 2 — Solusi eksperimental
142
+
143
+ **(Wuchale)** (`wuchale@0.22.11`):
144
+
145
+ Ide di balik `Wuchale` menarik tetapi belum menjadi solusi yang layak. Saya menemui masalah reaktivitas dengan library tersebut dan harus memaksa perenderan ulang provider untuk menjalankan aplikasi di TanStack Start. Dokumentasinya juga cukup tidak jelas, yang membuat adopsi lebih sulit.
146
+
147
+ ### 3 — Solusi yang dapat diterima
148
+
149
+ **(Paraglide)** (`@inlang/paraglide-js@2.15.1`):
150
+
151
+ `Paraglide` menawarkan pendekatan yang inovatif dan terencana dengan baik. Meskipun demikian, dalam benchmark ini tree-shaking yang diiklankan perusahaan mereka tidak berfungsi untuk implementasi Next.js saya atau untuk TanStack Start. Alur kerja dan DX-nya juga lebih kompleks daripada opsi lainnya. Secara pribadi saya bukan penggemar keharusan untuk membuat ulang file JS sebelum setiap push, yang menciptakan risiko konflik merge yang konstan bagi pengembang melalui PR.
152
+
153
+ **(Tolgee)** (`tolgee@7.0.0`):
154
+
155
+ `Tolgee` mengatasi banyak masalah yang disebutkan sebelumnya. Saya merasa lebih sulit untuk memulai adopsi Tolgee dibandingkan alat lain dengan pendekatan serupa. Ia tidak memberikan type safety, yang juga membuat pendeteksian kunci yang hilang saat compile time jauh lebih sulit. Saya harus membungkus API Tolgee dengan API saya sendiri untuk menambahkan deteksi kunci yang hilang.
156
+
157
+ Pada TanStack Start saya juga memiliki masalah reaktivitas: saat lokal berubah, saya harus memaksa provider untuk me-render ulang dan berlangganan ke event perubahan lokal sehingga pemuatan dalam bahasa lain berperilaku dengan benar.
158
+
159
+ **(use-intl)** (`use-intl@4.9.1`):
160
+
161
+ `use-intl` adalah bagian "intl" paling modis di ekosistem React (keluarga yang sama dengan `next-intl`) dan sering didorong oleh agen AI, tetapi menurut pandangan saya itu salah dalam pengaturan yang mementingkan performa. Memulainya cukup sederhana. Dalam praktiknya, proses untuk mengoptimalkan dan membatasi kebocoran cukup kompleks. Demikian juga, menggabungkan pemuatan dinamis + namespacing + tipe TypeScript sangat memperlambat pengembangan.
162
+
163
+ Pada TanStack Start Anda menghindari jebakan khusus Next.js (`setRequestLocale`, rendering statis), tetapi masalah intinya sama: tanpa disiplin yang ketat, bundle dengan cepat membawa terlalu banyak pesan dan pemeliharaan namespace per rute menjadi menyakitkan.
164
+
165
+ **(react-i18next)** (`react-i18next@17.0.2`):
166
+
167
+ `react-i18next` mungkin adalah opsi yang paling populer karena merupakan salah satu yang pertama melayani kebutuhan i18n aplikasi JavaScript. Ia juga memiliki serangkaian plugin komunitas yang luas untuk masalah tertentu.
168
+
169
+ Namun, ia memiliki kelemahan utama yang sama dengan stack yang dibangun di atas `t('a.b.c')`: optimasi dimungkinkan tetapi sangat memakan waktu, dan proyek besar berisiko jatuh ke dalam praktik buruk (namespace + pemuatan dinamis + tipe).
170
+
171
+ Format pesan juga berbeda: `use-intl` menggunakan ICU MessageFormat, sementara `i18next` menggunakan formatnya sendiri—yang memperumit tooling atau migrasi jika Anda mencampurnya.
172
+
173
+ **(Lingui)** (`@lingui/core@5.3.0`):
174
+
175
+ `Lingui` sering dipuji. Secara pribadi saya merasa alur kerja di sekitar `lingui extract` / `lingui compile` lebih kompleks daripada pendekatan lain, tanpa keunggulan yang jelas dalam benchmark TanStack Start ini. Saya juga menyadari sintaksis yang tidak konsisten yang membingungkan AI (misalnya `t()`, `t''`, `i18n.t()`, `<Trans>`).
176
+
177
+ **(react-intl)** (`react-intl@10.1.1`):
178
+
179
+ `react-intl` adalah implementasi performa dari tim Format.js. DX-nya tetap verbose: `const intl = useIntl()` + `intl.formatMessage({ id: "xx.xx" })` menambah kompleksitas, kerja JavaScript ekstra, dan mengikat instance i18n global ke banyak node di tree React.
180
+
181
+ ### 4 — Rekomendasi
182
+
183
+ Benchmark TanStack Start ini tidak memiliki padanan langsung untuk `next-translate` (plugin Next.js + `getStaticProps`). Bagi tim yang benar-benar menginginkan API `t()` dengan ekosistem yang matang, `react-i18next` dan `use-intl` tetap menjadi pilihan yang "masuk akal", tetapi bersiaplah untuk menghabiskan banyak waktu mengoptimalkan untuk menghindari kebocoran.
184
+
185
+ **(Intlayer)** (`react-intlayer@8.7.5-canary.0`):
186
+
187
+ Saya tidak akan secara pribadi menilai `react-intlayer` demi objektivitas, karena itu adalah solusi saya sendiri.
188
+
189
+ ### Catatan pribadi
190
+
191
+ Catatan ini bersifat pribadi dan tidak memengaruhi hasil benchmark. Namun, di dunia i18n Anda sering melihat konsensus seputar pola seperti `const t = useTranslation('xx')` + `<>{t('xx.xx')}</>` untuk konten terjemahan.
192
+
193
+ Dalam aplikasi React, menyuntikkan fungsi sebagai `ReactNode` adalah, dalam pandangan saya, sebuah anti-pattern. Ini juga menambah kompleksitas yang dapat dihindari dan overhead eksekusi JavaScript (meskipun hampir tidak terlihat).
@@ -709,5 +709,3 @@ import "@formatjs/intl-datetimeformat/polyfill";
709
709
  - Periksa konfigurasi Metro Anda (alias resolver, plugin aset, jalur `tsconfig`) jika modul gagal di-resolve.
710
710
 
711
711
  ---
712
-
713
- ---
@@ -0,0 +1,29 @@
1
+ ---
2
+ createdAt: 2026-04-20
3
+ updatedAt: 2026-04-20
4
+ title: Benchmark delle librerie i18n
5
+ description: Scopri come Intlayer si confronta con altre librerie i18n in termini di prestazioni e dimensioni del bundle.
6
+ keywords:
7
+ - benchmark
8
+ - i18n
9
+ - intl
10
+ - nextjs
11
+ - tanstack
12
+ - intlayer
13
+ slugs:
14
+ - doc
15
+ - benchmark
16
+ history:
17
+ - version: 8.7.5
18
+ date: 2026-01-06
19
+ changes: "Inizializzazione benchmark"
20
+ ---
21
+
22
+ # Benchmark - Rapporto
23
+
24
+ Benchmark Bloom è una suite di benchmark sulle prestazioni che misura l'impatto reale delle librerie i18n (internazionalizzazione) su più framework React e strategie di caricamento.
25
+
26
+ Di seguito trovi i rapporti dettagliati e la documentazione tecnica per ciascun framework:
27
+
28
+ - [**Rapporto di benchmark Next.js**](https://github.com/aymericzip/intlayer/blob/main/docs/docs/it/benchmark/nextjs.md)
29
+ - [**Rapporto di benchmark TanStack Start**](https://github.com/aymericzip/intlayer/blob/main/docs/docs/it/benchmark/tanstack.md)