@intlayer/docs 7.0.0-canary.2 → 7.0.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 (59) hide show
  1. package/dist/cjs/common.cjs.map +1 -1
  2. package/dist/esm/common.mjs.map +1 -1
  3. package/dist/types/common.d.ts +5 -0
  4. package/dist/types/common.d.ts.map +1 -1
  5. package/docs/ar/configuration.md +0 -24
  6. package/docs/ar/intlayer_with_nextjs_16.md +1652 -0
  7. package/docs/ar/releases/v7.md +485 -0
  8. package/docs/de/configuration.md +0 -24
  9. package/docs/de/intlayer_with_nextjs_16.md +1662 -0
  10. package/docs/de/releases/v7.md +502 -0
  11. package/docs/en/autoFill.md +3 -1
  12. package/docs/en/configuration.md +53 -58
  13. package/docs/en/intlayer_with_nextjs_15.md +5 -2
  14. package/docs/en/intlayer_with_nextjs_16.md +4 -4
  15. package/docs/en/releases/v7.md +142 -2
  16. package/docs/en-GB/configuration.md +9 -30
  17. package/docs/en-GB/intlayer_with_nextjs_16.md +1642 -0
  18. package/docs/en-GB/releases/v7.md +485 -0
  19. package/docs/es/configuration.md +0 -24
  20. package/docs/es/intlayer_with_nextjs_16.md +1670 -0
  21. package/docs/es/releases/v7.md +502 -0
  22. package/docs/fr/configuration.md +0 -24
  23. package/docs/fr/intlayer_with_nextjs_16.md +1692 -0
  24. package/docs/fr/releases/v7.md +503 -0
  25. package/docs/hi/configuration.md +0 -24
  26. package/docs/hi/intlayer_with_nextjs_16.md +1618 -0
  27. package/docs/hi/releases/v7.md +485 -0
  28. package/docs/id/intlayer_with_nextjs_16.md +1604 -0
  29. package/docs/id/releases/v7.md +502 -0
  30. package/docs/it/configuration.md +0 -24
  31. package/docs/it/intlayer_with_nextjs_16.md +1600 -0
  32. package/docs/it/releases/v7.md +504 -0
  33. package/docs/ja/configuration.md +0 -24
  34. package/docs/ja/intlayer_CMS.md +0 -9
  35. package/docs/ja/intlayer_with_nextjs_16.md +1788 -0
  36. package/docs/ja/releases/v7.md +503 -0
  37. package/docs/ko/configuration.md +0 -24
  38. package/docs/ko/intlayer_with_nextjs_16.md +1641 -0
  39. package/docs/ko/releases/v7.md +503 -0
  40. package/docs/pl/intlayer_with_nextjs_16.md +1645 -0
  41. package/docs/pl/releases/v7.md +485 -0
  42. package/docs/pt/configuration.md +0 -24
  43. package/docs/pt/intlayer_with_nextjs_16.md +1646 -0
  44. package/docs/pt/introduction.md +0 -15
  45. package/docs/pt/releases/v7.md +485 -0
  46. package/docs/ru/configuration.md +0 -24
  47. package/docs/ru/intlayer_with_nextjs_16.md +1610 -0
  48. package/docs/ru/releases/v7.md +485 -0
  49. package/docs/tr/configuration.md +0 -24
  50. package/docs/tr/intlayer_with_nextjs_16.md +1599 -0
  51. package/docs/tr/releases/v7.md +485 -0
  52. package/docs/vi/intlayer_with_nextjs_16.md +1597 -0
  53. package/docs/vi/releases/v7.md +485 -0
  54. package/docs/zh/configuration.md +0 -24
  55. package/docs/zh/intlayer_CMS.md +0 -23
  56. package/docs/zh/intlayer_with_nextjs_16.md +1628 -0
  57. package/docs/zh/releases/v7.md +486 -0
  58. package/package.json +14 -14
  59. package/src/common.ts +5 -0
@@ -0,0 +1,1604 @@
1
+ ---
2
+ createdAt: 2024-12-06
3
+ updatedAt: 2025-10-09
4
+ title: Cara menerjemahkan aplikasi Next.js 16 Anda – panduan i18n 2025
5
+ description: Temukan cara membuat situs web Next.js 16 Anda menjadi multibahasa. Ikuti dokumentasi untuk melakukan internasionalisasi (i18n) dan menerjemahkannya.
6
+ keywords:
7
+ - Internasionalisasi
8
+ - Dokumentasi
9
+ - Intlayer
10
+ - Next.js 16
11
+ - JavaScript
12
+ - React
13
+ slugs:
14
+ - doc
15
+ - environment
16
+ - nextjs
17
+ applicationTemplate: https://github.com/aymericzip/intlayer-next-16-template
18
+ youtubeVideo: https://www.youtube.com/watch?v=e_PPG7PTqGU
19
+ history:
20
+ - version: 7.0.0
21
+ date: 2025-06-29
22
+ changes: Riwayat awal
23
+ ---
24
+
25
+ # Terjemahkan situs web Next.js 16 Anda menggunakan Intlayer | Internasionalisasi (i18n)
26
+
27
+ <iframe title="Solusi i18n terbaik untuk Next.js? Temukan Intlayer" class="m-auto aspect-[16/9] w-full overflow-hidden rounded-lg border-0" allow="autoplay; gyroscope;" loading="lazy" width="1080" height="auto" src="https://www.youtube.com/embed/e_PPG7PTqGU?autoplay=0&amp;origin=http://intlayer.org&amp;controls=0&amp;rel=1"/>
28
+
29
+ Lihat [Template Aplikasi](https://github.com/aymericzip/intlayer-next-16-template) di GitHub.
30
+
31
+ ## Apa itu Intlayer?
32
+
33
+ **Intlayer** adalah perpustakaan internasionalisasi (i18n) sumber terbuka yang inovatif, dirancang untuk menyederhanakan dukungan multibahasa dalam aplikasi web modern. Intlayer terintegrasi mulus dengan kerangka kerja terbaru **Next.js 16**, termasuk **App Router** yang kuat. Ini dioptimalkan untuk bekerja dengan **Server Components** agar rendering lebih efisien dan sepenuhnya kompatibel dengan [**Turbopack**](https://nextjs.org/docs/architecture/turbopack).
34
+
35
+ Dengan Intlayer, Anda dapat:
36
+
37
+ - **Mengelola terjemahan dengan mudah** menggunakan kamus deklaratif pada tingkat komponen.
38
+ - **Melokalkan metadata**, rute, dan konten secara dinamis.
39
+ - **Mengakses terjemahan di komponen sisi klien dan sisi server**.
40
+ - **Memastikan dukungan TypeScript** dengan tipe yang dihasilkan secara otomatis, meningkatkan autocompletion dan deteksi kesalahan.
41
+ - **Manfaatkan fitur canggih**, seperti deteksi dan pengalihan locale secara dinamis.
42
+
43
+ > Intlayer kompatibel dengan Next.js 12, 13, 14, dan 16. Jika Anda menggunakan Next.js Page Router, Anda dapat merujuk ke [panduan ini](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/intlayer_with_nextjs_page_router.md). Untuk Next.js 12, 13, 14 dengan App Router, lihat [panduan ini](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/intlayer_with_nextjs_14.md).
44
+
45
+ ---
46
+
47
+ ## Panduan Langkah demi Langkah untuk Mengatur Intlayer di Aplikasi Next.js
48
+
49
+ ### Langkah 1: Instalasi Dependensi
50
+
51
+ Instal paket yang diperlukan menggunakan npm:
52
+
53
+ ```bash packageManager="npm"
54
+ npm install intlayer next-intlayer
55
+ ```
56
+
57
+ ```bash packageManager="pnpm"
58
+ pnpm add intlayer next-intlayer
59
+ ```
60
+
61
+ ```bash packageManager="yarn"
62
+ yarn add intlayer next-intlayer
63
+ ```
64
+
65
+ - **intlayer**
66
+
67
+ Paket inti yang menyediakan alat internasionalisasi untuk manajemen konfigurasi, terjemahan, [deklarasi konten](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/dictionary/content_file.md), transpile, dan [perintah CLI](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/intlayer_cli.md).
68
+
69
+ - **next-intlayer**
70
+
71
+ Paket yang mengintegrasikan Intlayer dengan Next.js. Paket ini menyediakan context providers dan hooks untuk internasionalisasi Next.js. Selain itu, paket ini juga menyertakan plugin Next.js untuk mengintegrasikan Intlayer dengan [Webpack](https://webpack.js.org/) atau [Turbopack](https://nextjs.org/docs/app/api-reference/turbopack), serta proxy untuk mendeteksi locale pilihan pengguna, mengelola cookie, dan menangani pengalihan URL.
72
+
73
+ ### Langkah 2: Konfigurasikan Proyek Anda
74
+
75
+ Buat file konfigurasi untuk mengatur bahasa aplikasi Anda:
76
+
77
+ ```typescript fileName="intlayer.config.ts" codeFormat="typescript"
78
+ import { Locales, type IntlayerConfig } from "intlayer";
79
+
80
+ const config: IntlayerConfig = {
81
+ internationalization: {
82
+ locales: [
83
+ Locales.ENGLISH,
84
+ Locales.FRENCH,
85
+ Locales.SPANISH,
86
+ // Locale lain Anda
87
+ ],
88
+ defaultLocale: Locales.ENGLISH,
89
+ },
90
+ };
91
+
92
+ export default config;
93
+ ```
94
+
95
+ ```javascript fileName="intlayer.config.mjs" codeFormat="esm"
96
+ import { Locales } from "intlayer";
97
+
98
+ /** @type {import('intlayer').IntlayerConfig} */
99
+ const config = {
100
+ internationalization: {
101
+ locales: [
102
+ Locales.ENGLISH,
103
+ Locales.FRENCH,
104
+ Locales.SPANISH,
105
+ // Locale lain Anda
106
+ ],
107
+ defaultLocale: Locales.ENGLISH,
108
+ },
109
+ };
110
+
111
+ export default config;
112
+ ```
113
+
114
+ ```javascript fileName="intlayer.config.cjs" codeFormat="commonjs"
115
+ const { Locales } = require("intlayer");
116
+
117
+ /** @type {import('intlayer').IntlayerConfig} */
118
+ const config = {
119
+ internationalization: {
120
+ locales: [
121
+ Locales.ENGLISH,
122
+ Locales.FRENCH,
123
+ Locales.SPANISH,
124
+ // Locale lain Anda
125
+ ],
126
+ defaultLocale: Locales.ENGLISH,
127
+ },
128
+ };
129
+
130
+ module.exports = config;
131
+ ```
132
+
133
+ > Melalui file konfigurasi ini, Anda dapat mengatur URL yang dilokalkan, pengalihan proxy, nama cookie, lokasi dan ekstensi deklarasi konten Anda, menonaktifkan log Intlayer di konsol, dan lainnya. Untuk daftar lengkap parameter yang tersedia, lihat [dokumentasi konfigurasi](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/configuration.md).
134
+
135
+ ### Langkah 3: Integrasikan Intlayer dalam Konfigurasi Next.js Anda
136
+
137
+ Konfigurasikan setup Next.js Anda untuk menggunakan Intlayer:
138
+
139
+ ```typescript fileName="next.config.ts" codeFormat="typescript"
140
+ import type { NextConfig } from "next";
141
+ import { withIntlayer } from "next-intlayer/server";
142
+
143
+ const nextConfig: NextConfig = {
144
+ /* opsi konfigurasi di sini */
145
+ };
146
+
147
+ export default withIntlayer(nextConfig);
148
+ ```
149
+
150
+ ```typescript fileName="next.config.mjs" codeFormat="esm"
151
+ import { withIntlayer } from "next-intlayer/server";
152
+
153
+ /** @type {import('next').NextConfig} */
154
+ const nextConfig = {
155
+ /* opsi konfigurasi di sini */
156
+ };
157
+
158
+ export default withIntlayer(nextConfig);
159
+ ```
160
+
161
+ ```typescript fileName="next.config.cjs" codeFormat="commonjs"
162
+ const { withIntlayer } = require("next-intlayer/server");
163
+
164
+ /** @type {import('next').NextConfig} */
165
+ const nextConfig = {
166
+ /* opsi konfigurasi di sini */
167
+ };
168
+
169
+ module.exports = withIntlayer(nextConfig);
170
+ ```
171
+
172
+ > Plugin Next.js `withIntlayer()` digunakan untuk mengintegrasikan Intlayer dengan Next.js. Plugin ini memastikan pembuatan file deklarasi konten dan memantau file tersebut dalam mode pengembangan. Plugin ini mendefinisikan variabel lingkungan Intlayer dalam lingkungan [Webpack](https://webpack.js.org/) atau [Turbopack](https://nextjs.org/docs/app/api-reference/turbopack). Selain itu, plugin ini menyediakan alias untuk mengoptimalkan performa dan memastikan kompatibilitas dengan komponen server.
173
+
174
+ > Fungsi `withIntlayer()` adalah fungsi promise. Fungsi ini memungkinkan persiapan kamus intlayer sebelum proses build dimulai. Jika Anda ingin menggunakannya bersama plugin lain, Anda dapat menggunakan `await`. Contoh:
175
+ >
176
+ > ```tsx
177
+ > const nextConfig = await withIntlayer(nextConfig);
178
+ > const nextConfigWithOtherPlugins = withOtherPlugins(nextConfig);
179
+ >
180
+ > export default nextConfigWithOtherPlugins;
181
+ > ```
182
+ >
183
+ > Jika Anda ingin menggunakannya secara sinkron, Anda dapat menggunakan fungsi `withIntlayerSync()`. Contoh:
184
+ >
185
+ > ```tsx
186
+ > const nextConfig = withIntlayerSync(nextConfig);
187
+ > const nextConfigWithOtherPlugins = withOtherPlugins(nextConfig);
188
+ >
189
+ > export default nextConfigWithOtherPlugins;
190
+ > ```
191
+
192
+ ### Langkah 4: Definisikan Rute Locale Dinamis
193
+
194
+ Hapus semua dari `RootLayout` dan ganti dengan kode berikut:
195
+
196
+ ```tsx {3} fileName="src/app/layout.tsx" codeFormat="typescript"
197
+ import type { PropsWithChildren, FC } from "react";
198
+ import "./globals.css";
199
+
200
+ const RootLayout: FC<PropsWithChildren> = ({ children }) => (
201
+ // Anda masih dapat membungkus children dengan provider lain, seperti `next-themes`, `react-query`, `framer-motion`, dll.
202
+ <>{children}</>
203
+ );
204
+
205
+ export default RootLayout;
206
+ ```
207
+
208
+ ```jsx {3} fileName="src/app/layout.mjx" codeFormat="esm"
209
+ import "./globals.css";
210
+
211
+ const RootLayout = ({ children }) => (
212
+ // Anda masih dapat membungkus children dengan penyedia lain, seperti `next-themes`, `react-query`, `framer-motion`, dll.
213
+ <>{children}</>
214
+ );
215
+
216
+ export default RootLayout;
217
+ ```
218
+
219
+ ```jsx {1,8} fileName="src/app/layout.csx" codeFormat="commonjs"
220
+ require("./globals.css");
221
+
222
+ const RootLayout = ({ children }) => (
223
+ // Anda masih dapat membungkus children dengan penyedia lain, seperti `next-themes`, `react-query`, `framer-motion`, dll.
224
+ <>{children}</>
225
+ );
226
+
227
+ module.exports = {
228
+ default: RootLayout,
229
+ generateStaticParams,
230
+ };
231
+ ```
232
+
233
+ > Menjaga komponen `RootLayout` tetap kosong memungkinkan untuk mengatur atribut [`lang`](https://developer.mozilla.org/fr/docs/Web/HTML/Global_attributes/lang) dan [`dir`](https://developer.mozilla.org/fr/docs/Web/HTML/Global_attributes/dir) pada tag `<html>`.
234
+
235
+ Untuk mengimplementasikan routing dinamis, sediakan jalur untuk locale dengan menambahkan layout baru di direktori `[locale]` Anda:
236
+
237
+ ```tsx fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
238
+ import type { NextLayoutIntlayer } from "next-intlayer";
239
+ import { Inter } from "next/font/google";
240
+ import { getHTMLTextDir } from "intlayer";
241
+
242
+ const inter = Inter({ subsets: ["latin"] });
243
+
244
+ const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
245
+ const { locale } = await params;
246
+ return (
247
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
248
+ <body className={inter.className}>{children}</body>
249
+ </html>
250
+ );
251
+ };
252
+
253
+ export default LocaleLayout;
254
+ ```
255
+
256
+ ```jsx fileName="src/app/[locale]/layout.mjx" codeFormat="esm"
257
+ import { getHTMLTextDir } from "intlayer";
258
+
259
+ const inter = Inter({ subsets: ["latin"] });
260
+
261
+ const LocaleLayout = async ({ children, params: { locale } }) => {
262
+ const { locale } = await params;
263
+ return (
264
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
265
+ <body className={inter.className}>{children}</body>
266
+ </html>
267
+ );
268
+ };
269
+
270
+ export default LocaleLayout;
271
+ ```
272
+
273
+ ```jsx fileName="src/app/[locale]/layout.csx" codeFormat="commonjs"
274
+ const { Inter } = require("next/font/google");
275
+ const { getHTMLTextDir } = require("intlayer");
276
+
277
+ const inter = Inter({ subsets: ["latin"] });
278
+
279
+ const LocaleLayout = async ({ children, params: { locale } }) => {
280
+ const { locale } = await params;
281
+ return (
282
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
283
+ <body className={inter.className}>{children}</body>
284
+ </html>
285
+ );
286
+ };
287
+
288
+ module.exports = LocaleLayout;
289
+ ```
290
+
291
+ > Segmen jalur `[locale]` digunakan untuk menentukan locale. Contoh: `/en-US/about` akan merujuk ke `en-US` dan `/fr/about` ke `fr`.
292
+
293
+ > Pada tahap ini, Anda akan menemui error: `Error: Missing <html> and <body> tags in the root layout.`. Ini diharapkan karena file `/app/page.tsx` tidak lagi digunakan dan dapat dihapus. Sebagai gantinya, segmen path `[locale]` akan mengaktifkan halaman `/app/[locale]/page.tsx`. Akibatnya, halaman akan dapat diakses melalui path seperti `/en`, `/fr`, `/es` di browser Anda. Untuk mengatur locale default sebagai halaman root, lihat pengaturan `proxy` pada langkah 7.
294
+
295
+ Kemudian, implementasikan fungsi `generateStaticParams` di Layout aplikasi Anda.
296
+
297
+ ```tsx {1} fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
298
+ export { generateStaticParams } from "next-intlayer"; // Baris yang harus ditambahkan
299
+
300
+ const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
301
+ /*... Sisa kode */
302
+ };
303
+
304
+ export default LocaleLayout;
305
+ ```
306
+
307
+ ```jsx {1} fileName="src/app/[locale]/layout.mjx" codeFormat="esm"
308
+ export { generateStaticParams } from "next-intlayer"; // Baris untuk disisipkan
309
+
310
+ const LocaleLayout = async ({ children, params: { locale } }) => {
311
+ /*... Sisa kode*/
312
+ };
313
+
314
+ // ... Sisa kode
315
+ ```
316
+
317
+ ```jsx {1,7} fileName="src/app/[locale]/layout.csx" codeFormat="commonjs"
318
+ const { generateStaticParams } = require("next-intlayer"); // Baris untuk disisipkan
319
+
320
+ const LocaleLayout = async ({ children, params: { locale } }) => {
321
+ /*... Sisa kode*/
322
+ };
323
+
324
+ module.exports = { default: LocaleLayout, generateStaticParams };
325
+ ```
326
+
327
+ > `generateStaticParams` memastikan bahwa aplikasi Anda membangun terlebih dahulu halaman-halaman yang diperlukan untuk semua lokal, mengurangi komputasi saat runtime dan meningkatkan pengalaman pengguna. Untuk detail lebih lanjut, lihat [dokumentasi Next.js tentang generateStaticParams](https://nextjs.org/docs/app/building-your-application/rendering/static-and-dynamic-rendering#generate-static-params).
328
+
329
+ > Intlayer bekerja dengan `export const dynamic = 'force-static';` untuk memastikan bahwa halaman-halaman dibangun terlebih dahulu untuk semua lokal.
330
+
331
+ ### Langkah 5: Deklarasikan Konten Anda
332
+
333
+ Buat dan kelola deklarasi konten Anda untuk menyimpan terjemahan:
334
+
335
+ ```tsx fileName="src/app/[locale]/page.content.ts" contentDeclarationFormat="typescript"
336
+ import { t, type Dictionary } from "intlayer";
337
+
338
+ const pageContent = {
339
+ key: "page",
340
+ content: {
341
+ getStarted: {
342
+ main: t({
343
+ en: "Get started by editing",
344
+ fr: "Commencez par éditer",
345
+ es: "Comience por editar",
346
+ }),
347
+ pageLink: "src/app/page.tsx",
348
+ },
349
+ },
350
+ } satisfies Dictionary;
351
+
352
+ export default pageContent;
353
+ ```
354
+
355
+ ```javascript fileName="src/app/[locale]/page.content.mjs" contentDeclarationFormat="esm"
356
+ import { t } from "intlayer";
357
+
358
+ /** @type {import('intlayer').Dictionary} */
359
+ const pageContent = {
360
+ key: "page",
361
+ content: {
362
+ getStarted: {
363
+ main: t({
364
+ en: "Mulailah dengan mengedit",
365
+ fr: "Commencez par éditer",
366
+ es: "Comience por editar",
367
+ }),
368
+ pageLink: "src/app/page.tsx",
369
+ },
370
+ },
371
+ };
372
+
373
+ export default pageContent;
374
+ ```
375
+
376
+ ```javascript fileName="src/app/[locale]/page.content.cjs" contentDeclarationFormat="commonjs"
377
+ const { t } = require("intlayer");
378
+
379
+ /** @type {import('intlayer').Dictionary} */
380
+ const pageContent = {
381
+ key: "page",
382
+ content: {
383
+ getStarted: {
384
+ main: t({
385
+ en: "Get started by editing",
386
+ fr: "Commencez par éditer",
387
+ es: "Comience por editar",
388
+ id: "Mulai dengan mengedit",
389
+ }),
390
+ pageLink: "src/app/page.tsx",
391
+ },
392
+ },
393
+ };
394
+
395
+ module.exports = pageContent;
396
+ ```
397
+
398
+ ```json fileName="src/app/[locale]/page.content.json" contentDeclarationFormat="json"
399
+ {
400
+ "$schema": "https://intlayer.org/schema.json",
401
+ "key": "page",
402
+ "content": {
403
+ "getStarted": {
404
+ "nodeType": "translation",
405
+ "translation": {
406
+ "en": "Get started by editing",
407
+ "fr": "Commencez par éditer",
408
+ "es": "Comience por editar",
409
+ "id": "Mulai dengan mengedit"
410
+ }
411
+ },
412
+ "pageLink": "src/app/page.tsx"
413
+ }
414
+ }
415
+ ```
416
+
417
+ > Deklarasi konten Anda dapat didefinisikan di mana saja dalam aplikasi Anda selama sudah dimasukkan ke dalam direktori `contentDir` (secara default, `./src`). Dan sesuai dengan ekstensi file deklarasi konten (secara default, `.content.{json,ts,tsx,js,jsx,mjs,mjx,cjs,cjx}`).
418
+
419
+ > Untuk detail lebih lanjut, lihat [dokumentasi deklarasi konten](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/dictionary/content_file.md).
420
+
421
+ ### Langkah 6: Memanfaatkan Konten dalam Kode Anda
422
+
423
+ Akses kamus konten Anda di seluruh aplikasi:
424
+
425
+ ```tsx fileName="src/app/[locale]/page.tsx" codeFormat="typescript"
426
+ import type { FC } from "react";
427
+ import { ClientComponentExample } from "@components/ClientComponentExample";
428
+ import { ServerComponentExample } from "@components/ServerComponentExample";
429
+ import { type NextPageIntlayer, IntlayerClientProvider } from "next-intlayer";
430
+ import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
431
+
432
+ const PageContent: FC = () => {
433
+ const content = useIntlayer("page");
434
+
435
+ return (
436
+ <>
437
+ <p>{content.getStarted.main}</p> {/* Mengakses teks utama dari konten */}
438
+ <code>{content.getStarted.pageLink}</code>{" "}
439
+ {/* Menampilkan tautan halaman dari konten */}
440
+ </>
441
+ );
442
+ };
443
+
444
+ const Page: NextPageIntlayer = async ({ params }) => {
445
+ const { locale } = await params;
446
+
447
+ return (
448
+ <IntlayerServerProvider locale={locale}>
449
+ <PageContent />
450
+ <ServerComponentExample />
451
+
452
+ <IntlayerClientProvider locale={locale}>
453
+ <ClientComponentExample />
454
+ </IntlayerClientProvider>
455
+ </IntlayerServerProvider>
456
+ );
457
+ };
458
+
459
+ export default Page;
460
+ ```
461
+
462
+ ```jsx fileName="src/app/[locale]/page.mjx" codeFormat="esm"
463
+ import { ClientComponentExample } from "@components/ClientComponentExample";
464
+ import { ServerComponentExample } from "@components/ServerComponentExample";
465
+ import { IntlayerClientProvider } from "next-intlayer";
466
+ import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
467
+
468
+ const PageContent = () => {
469
+ const content = useIntlayer("page");
470
+
471
+ return (
472
+ <>
473
+ <p>{content.getStarted.main}</p>{" "}
474
+ {/* Menampilkan teks utama dari konten */}
475
+ <code>{content.getStarted.pageLink}</code>{" "}
476
+ {/* Menampilkan tautan halaman dari konten */}
477
+ </>
478
+ );
479
+ };
480
+
481
+ const Page = async ({ params }) => {
482
+ const { locale } = await params;
483
+
484
+ return (
485
+ <IntlayerServerProvider locale={locale}>
486
+ <PageContent />
487
+ <ServerComponentExample />
488
+
489
+ <IntlayerClientProvider locale={locale}>
490
+ <ClientComponentExample />
491
+ </IntlayerClientProvider>
492
+ </IntlayerServerProvider>
493
+ );
494
+ };
495
+
496
+ export default Page;
497
+ ```
498
+
499
+ ```jsx fileName="src/app/[locale]/page.csx" codeFormat="commonjs"
500
+ import { ClientComponentExample } from "@components/ClientComponentExample";
501
+ import { ServerComponentExample } from "@components/ServerComponentExample";
502
+ import { IntlayerClientProvider } from "next-intlayer";
503
+ import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
504
+
505
+ const PageContent = () => {
506
+ const content = useIntlayer("page");
507
+
508
+ return (
509
+ <>
510
+ <p>{content.getStarted.main}</p>
511
+ <code>{content.getStarted.pageLink}</code>
512
+ </>
513
+ );
514
+ };
515
+
516
+ const Page = async ({ params }) => {
517
+ const { locale } = await params;
518
+
519
+ return (
520
+ <IntlayerServerProvider locale={locale}>
521
+ <PageContent />
522
+ <ServerComponentExample />
523
+
524
+ <IntlayerClientProvider locale={locale}>
525
+ <ClientComponentExample />
526
+ </IntlayerClientProvider>
527
+ </IntlayerServerProvider>
528
+ );
529
+ };
530
+ ```
531
+
532
+ - **`IntlayerClientProvider`** digunakan untuk menyediakan locale ke komponen sisi klien. Ini dapat ditempatkan di komponen induk mana pun, termasuk layout. Namun, menempatkannya di layout sangat disarankan karena Next.js membagikan kode layout di seluruh halaman, sehingga menjadi lebih efisien. Dengan menggunakan `IntlayerClientProvider` di layout, Anda menghindari inisialisasi ulang untuk setiap halaman, meningkatkan kinerja dan menjaga konteks lokalisasi yang konsisten di seluruh aplikasi Anda.
533
+ - **`IntlayerServerProvider`** digunakan untuk menyediakan locale ke anak server. Ini tidak dapat ditempatkan di layout.
534
+
535
+ > Layout dan halaman tidak dapat berbagi konteks server yang sama karena sistem konteks server didasarkan pada penyimpanan data per permintaan (melalui mekanisme [cache React](https://react.dev/reference/react/cache)), yang menyebabkan setiap "konteks" dibuat ulang untuk segmen aplikasi yang berbeda. Menempatkan provider di layout bersama akan memecah isolasi ini, sehingga mencegah propagasi nilai konteks server yang benar ke komponen server Anda.
536
+
537
+ ```tsx {4,7} fileName="src/components/ClientComponentExample.tsx" codeFormat="typescript"
538
+ "use client";
539
+
540
+ import type { FC } from "react";
541
+ import { useIntlayer } from "next-intlayer";
542
+
543
+ export const ClientComponentExample: FC = () => {
544
+ const content = useIntlayer("client-component-example"); // Membuat deklarasi konten terkait
545
+
546
+ return (
547
+ <div>
548
+ <h2>{content.title}</h2>
549
+ <p>{content.content}</p>
550
+ </div>
551
+ );
552
+ };
553
+ ```
554
+
555
+ ```jsx {3,6} fileName="src/components/ClientComponentExample.mjx" codeFormat="esm"
556
+ "use client";
557
+
558
+ import { useIntlayer } from "next-intlayer";
559
+
560
+ const ClientComponentExample = () => {
561
+ const content = useIntlayer("client-component-example"); // Membuat deklarasi konten terkait
562
+
563
+ return (
564
+ <div>
565
+ <h2>{content.title}</h2>
566
+ <p>{content.content}</p>
567
+ </div>
568
+ );
569
+ };
570
+ ```
571
+
572
+ ```jsx {3,6} fileName="src/components/ClientComponentExample.csx" codeFormat="commonjs"
573
+ "use client";
574
+
575
+ const { useIntlayer } = require("next-intlayer");
576
+
577
+ const ClientComponentExample = () => {
578
+ const content = useIntlayer("client-component-example"); // Membuat deklarasi konten terkait
579
+
580
+ return (
581
+ <div>
582
+ <h2>{content.title}</h2>
583
+ <p>{content.content}</p>
584
+ </div>
585
+ );
586
+ };
587
+ ```
588
+
589
+ ```tsx {2} fileName="src/components/ServerComponentExample.tsx" codeFormat="typescript"
590
+ import type { FC } from "react";
591
+ import { useIntlayer } from "next-intlayer/server";
592
+
593
+ export const ServerComponentExample: FC = () => {
594
+ const content = useIntlayer("server-component-example"); // Membuat deklarasi konten terkait
595
+
596
+ return (
597
+ <div>
598
+ <h2>{content.title}</h2>
599
+ <p>{content.content}</p>
600
+ </div>
601
+ );
602
+ };
603
+ ```
604
+
605
+ ```jsx {1} fileName="src/components/ServerComponentExample.mjx" codeFormat="esm"
606
+ import { useIntlayer } from "next-intlayer/server";
607
+
608
+ const ServerComponentExample = () => {
609
+ const content = useIntlayer("server-component-example"); // Membuat deklarasi konten terkait
610
+
611
+ return (
612
+ <div>
613
+ <h2>{content.title}</h2>
614
+ <p>{content.content}</p>
615
+ </div>
616
+ );
617
+ };
618
+ ```
619
+
620
+ ```jsx {1} fileName="src/components/ServerComponentExample.csx" codeFormat="commonjs"
621
+ const { useIntlayer } = require("next-intlayer/server");
622
+
623
+ const ServerComponentExample = () => {
624
+ const content = useIntlayer("server-component-example"); // Buat deklarasi konten terkait
625
+
626
+ return (
627
+ <div>
628
+ <h2>{content.title}</h2>
629
+ <p>{content.content}</p>
630
+ </div>
631
+ );
632
+ };
633
+ ```
634
+
635
+ > Jika Anda ingin menggunakan konten Anda dalam atribut `string`, seperti `alt`, `title`, `href`, `aria-label`, dll., Anda harus memanggil nilai fungsi tersebut, seperti:
636
+
637
+ > ```jsx
638
+ > <img src={content.image.src.value} alt={content.image.value} />
639
+ > ```
640
+
641
+ > Untuk mempelajari lebih lanjut tentang hook `useIntlayer`, lihat [dokumentasi](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/packages/next-intlayer/useIntlayer.md).
642
+
643
+ ### (Opsional) Langkah 7: Konfigurasikan Proxy untuk Deteksi Locale
644
+
645
+ Atur proxy untuk mendeteksi locale yang dipilih pengguna:
646
+
647
+ ```typescript fileName="src/proxy.ts" codeFormat="typescript"
648
+ export { intlayerProxy as proxy } from "next-intlayer/proxy";
649
+
650
+ export const config = {
651
+ matcher:
652
+ "/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
653
+ };
654
+ ```
655
+
656
+ ```javascript fileName="src/proxy.mjs" codeFormat="esm"
657
+ export { intlayerProxy as proxy } from "next-intlayer/proxy";
658
+
659
+ export const config = {
660
+ matcher:
661
+ "/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
662
+ };
663
+ ```
664
+
665
+ ```javascript fileName="src/proxy.cjs" codeFormat="commonjs"
666
+ const { intlayerProxy } = require("next-intlayer/proxy");
667
+
668
+ const config = {
669
+ matcher:
670
+ "/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
671
+ };
672
+
673
+ module.exports = { proxy: intlayerProxy, config };
674
+ ```
675
+
676
+ > `intlayerProxy` digunakan untuk mendeteksi locale pilihan pengguna dan mengarahkan mereka ke URL yang sesuai seperti yang ditentukan dalam [konfigurasi](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/configuration.md). Selain itu, ini memungkinkan penyimpanan locale pilihan pengguna dalam cookie.
677
+
678
+ > Jika Anda perlu menggabungkan beberapa proxy bersama-sama (misalnya, `intlayerProxy` dengan otentikasi atau proxy kustom), Intlayer sekarang menyediakan helper yang disebut `multipleProxies`.
679
+
680
+ ```ts
681
+ import { multipleProxies, intlayerProxy } from "next-intlayer/proxy";
682
+ import { customProxy } from "@utils/customProxy";
683
+
684
+ export const proxy = multipleProxies([intlayerProxy, customProxy]);
685
+ ```
686
+
687
+ ### (Opsional) Langkah 8: Internasionalisasi metadata Anda
688
+
689
+ Jika Anda ingin menginternasionalisasi metadata Anda, seperti judul halaman Anda, Anda dapat menggunakan fungsi `generateMetadata` yang disediakan oleh Next.js. Di dalamnya, Anda dapat mengambil konten dari fungsi `getIntlayer` untuk menerjemahkan metadata Anda.
690
+
691
+ ```typescript fileName="src/app/[locale]/metadata.content.ts" contentDeclarationFormat="typescript"
692
+ import { type Dictionary, t } from "intlayer";
693
+ import { Metadata } from "next";
694
+
695
+ const metadataContent = {
696
+ key: "page-metadata",
697
+ content: {
698
+ title: t({
699
+ en: "Create Next App",
700
+ fr: "Créer une application Next.js",
701
+ es: "Crear una aplicación Next.js",
702
+ }),
703
+ description: t({
704
+ en: "Generated by create next app",
705
+ fr: "Généré par create next app",
706
+ es: "Generado por create next app",
707
+ }),
708
+ },
709
+ } satisfies Dictionary<Metadata>;
710
+
711
+ export default metadataContent;
712
+ ```
713
+
714
+ ```javascript fileName="src/app/[locale]/metadata.content.mjs" contentDeclarationFormat="esm"
715
+ import { t } from "intlayer";
716
+
717
+ /** @type {import('intlayer').Dictionary<import('next').Metadata>} */
718
+ const metadataContent = {
719
+ key: "page-metadata",
720
+ content: {
721
+ title: t({
722
+ en: "Create Next App",
723
+ fr: "Créer une application Next.js",
724
+ es: "Crear una aplicación Next.js",
725
+ }),
726
+ description: t({
727
+ en: "Dihasilkan oleh create next app",
728
+ fr: "Généré par create next app",
729
+ es: "Generado por create next app",
730
+ }),
731
+ },
732
+ };
733
+
734
+ export default metadataContent;
735
+ fr: "Dihasilkan oleh create next app",
736
+ es: "Generado por create next app",
737
+ }),
738
+ },
739
+ };
740
+
741
+ export default metadataContent;
742
+ ```
743
+
744
+ ```javascript fileName="src/app/[locale]/metadata.content.cjs" contentDeclarationFormat="commonjs"
745
+ const { t } = require("intlayer");
746
+
747
+ /** @type {import('intlayer').Dictionary<import('next').Metadata>} */
748
+ const metadataContent = {
749
+ key: "page-metadata",
750
+ content: {
751
+ title: t({
752
+ en: "Create Next App",
753
+ fr: "Créer une application Next.js",
754
+ es: "Crear una aplicación Next.js",
755
+ }),
756
+ description: t({
757
+ en: "Dihasilkan oleh create next app",
758
+ fr: "Généré par create next app",
759
+ es: "Generado por create next app",
760
+ }),
761
+ },
762
+ };
763
+
764
+ module.exports = metadataContent;
765
+ ```
766
+
767
+ ```json fileName="src/app/[locale]/metadata.content.json" contentDeclarationFormat="json"
768
+ {
769
+ "key": "page-metadata",
770
+ "content": {
771
+ "title": {
772
+ "nodeType": "translation",
773
+ "translation": {
774
+ "en": "Preact logo",
775
+ "fr": "Logo Preact",
776
+ "es": "Logo Preact",
777
+ "id": "Logo Preact"
778
+ },
779
+ },
780
+ "description": {
781
+ "nodeType": "translation",
782
+ "translation": {
783
+ "en": "Generated by create next app",
784
+ "fr": "Généré par create next app",
785
+ "es": "Generado por create next app",
786
+ "id": "Dihasilkan oleh create next app"
787
+ },
788
+ },
789
+ },
790
+ };
791
+ ```
792
+
793
+ ````typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
794
+ import { getIntlayer, getMultilingualUrls } from "intlayer";
795
+ import type { Metadata } from "next";
796
+ import type { LocalPromiseParams } from "next-intlayer";
797
+
798
+ export const generateMetadata = async ({
799
+ params,
800
+ }: LocalPromiseParams): Promise<Metadata> => {
801
+ const { locale } = await params;
802
+
803
+ const metadata = getIntlayer("page-metadata", locale);
804
+
805
+ /**
806
+ * Menghasilkan objek yang berisi semua url untuk setiap locale.
807
+ *
808
+ * Contoh:
809
+ * ```ts
810
+ * getMultilingualUrls('/about');
811
+ *
812
+ * // Mengembalikan
813
+ * // {
814
+ * // en: '/about',
815
+ * // fr: '/fr/about',
816
+ * // es: '/es/about',
817
+ * // }
818
+ * ```
819
+ */
820
+ const multilingualUrls = getMultilingualUrls("/");
821
+
822
+ return {
823
+ ...metadata,
824
+ alternates: {
825
+ canonical: multilingualUrls[locale as keyof typeof multilingualUrls],
826
+ languages: { ...multilingualUrls, "x-default": "/" },
827
+ },
828
+ openGraph: {
829
+ url: multilingualUrls[locale],
830
+ },
831
+ };
832
+ };
833
+
834
+ // ... Sisa kode
835
+ ````
836
+
837
+ ````javascript fileName="src/app/[locale]/layout.mjs or src/app/[locale]/page.mjs" codeFormat="esm"
838
+ import { getIntlayer, getMultilingualUrls } from "intlayer";
839
+
840
+ export const generateMetadata = async ({ params }) => {
841
+ const { locale } = await params;
842
+
843
+ const metadata = getIntlayer("page-metadata", locale);
844
+
845
+ /**
846
+ * Menghasilkan objek yang berisi semua url untuk setiap locale.
847
+ *
848
+ * Contoh:
849
+ * ```ts
850
+ * getMultilingualUrls('/about');
851
+ *
852
+ * // Mengembalikan
853
+ * // {
854
+ * // en: '/about',
855
+ * // fr: '/fr/about',
856
+ * // es: '/es/about'
857
+ * // }
858
+ * ```
859
+ */
860
+ const multilingualUrls = getMultilingualUrls("/");
861
+
862
+ return {
863
+ ...metadata,
864
+ alternates: {
865
+ canonical: multilingualUrls[locale],
866
+ languages: { ...multilingualUrls, "x-default": "/" },
867
+ },
868
+ openGraph: {
869
+ url: multilingualUrls[locale],
870
+ },
871
+ };
872
+ };
873
+
874
+ // ... Sisa kode
875
+ ````
876
+
877
+ ````javascript fileName="src/app/[locale]/layout.cjs or src/app/[locale]/page.cjs" codeFormat="commonjs"
878
+ const { getIntlayer, getMultilingualUrls } = require("intlayer");
879
+
880
+ const generateMetadata = async ({ params }) => {
881
+ const { locale } = await params;
882
+
883
+ const metadata = getIntlayer("page-metadata", locale);
884
+
885
+ /**
886
+ * Menghasilkan objek yang berisi semua url untuk setiap locale.
887
+ *
888
+ * Contoh:
889
+ * ```ts
890
+ * getMultilingualUrls('/about');
891
+ *
892
+ * // Mengembalikan
893
+ * // {
894
+ * // en: '/about',
895
+ * // fr: '/fr/about',
896
+ * // es: '/es/about'
897
+ * // }
898
+ * ```
899
+ */
900
+ const multilingualUrls = getMultilingualUrls("/");
901
+
902
+ return {
903
+ ...metadata,
904
+ alternates: {
905
+ canonical: multilingualUrls[locale],
906
+ languages: { ...multilingualUrls, "x-default": "/" },
907
+ },
908
+ openGraph: {
909
+ url: multilingualUrls[locale],
910
+ },
911
+ };
912
+ };
913
+
914
+ module.exports = { generateMetadata };
915
+
916
+ // ... Sisa kode
917
+ ````
918
+
919
+ > Perhatikan bahwa fungsi `getIntlayer` yang diimpor dari `next-intlayer` mengembalikan konten Anda yang dibungkus dalam sebuah `IntlayerNode`, memungkinkan integrasi dengan editor visual. Sebaliknya, fungsi `getIntlayer` yang diimpor dari `intlayer` mengembalikan konten Anda secara langsung tanpa properti tambahan.
920
+
921
+ Sebagai alternatif, Anda dapat menggunakan fungsi `getTranslation` untuk mendeklarasikan metadata Anda. Namun, menggunakan file deklarasi konten disarankan untuk mengotomatisasi terjemahan metadata Anda dan mengeksternalisasi konten pada suatu saat.
922
+
923
+ ```typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
924
+ import {
925
+ type IConfigLocales,
926
+ getTranslation,
927
+ getMultilingualUrls,
928
+ } from "intlayer";
929
+ import type { Metadata } from "next";
930
+ import type { LocalPromiseParams } from "next-intlayer";
931
+
932
+ export const generateMetadata = async ({
933
+ params,
934
+ }: LocalPromiseParams): Promise<Metadata> => {
935
+ const { locale } = await params;
936
+ const t = <T>(content: IConfigLocales<T>) => getTranslation(content, locale);
937
+
938
+ return {
939
+ title: t<string>({
940
+ en: "My title",
941
+ fr: "Mon titre",
942
+ es: "Mi título",
943
+ }),
944
+ description: t({
945
+ en: "Deskripsi saya",
946
+ fr: "Ma description",
947
+ es: "Mi descripción",
948
+ }),
949
+ };
950
+ };
951
+
952
+ // ... Sisa kode
953
+ ```
954
+
955
+ ```javascript fileName="src/app/[locale]/layout.mjs or src/app/[locale]/page.mjs" codeFormat="esm"
956
+ import { getTranslation, getMultilingualUrls } from "intlayer";
957
+
958
+ export const generateMetadata = async ({ params }) => {
959
+ const { locale } = await params;
960
+ const t = (content) => getTranslation(content, locale);
961
+
962
+ return {
963
+ title: t({
964
+ en: "Judul saya",
965
+ fr: "Mon titre",
966
+ es: "Mi título",
967
+ }),
968
+ description: t({
969
+ en: "Deskripsi saya",
970
+ fr: "Ma description",
971
+ es: "Mi descripción",
972
+ }),
973
+ };
974
+ };
975
+
976
+ // ... Sisa kode
977
+ ```
978
+
979
+ ```javascript fileName="src/app/[locale]/layout.cjs or src/app/[locale]/page.cjs" codeFormat="commonjs"
980
+ const { getTranslation, getMultilingualUrls } = require("intlayer");
981
+
982
+ const generateMetadata = async ({ params }) => {
983
+ const { locale } = await params;
984
+
985
+ const t = (content) => getTranslation(content, locale);
986
+
987
+ return {
988
+ title: t({
989
+ en: "My title",
990
+ fr: "Mon titre",
991
+ es: "Mi título",
992
+ }),
993
+ description: t({
994
+ en: "My description",
995
+ fr: "Ma description",
996
+ es: "Mi descripción",
997
+ }),
998
+ };
999
+ };
1000
+
1001
+ module.exports = { generateMetadata };
1002
+
1003
+ // ... Sisa kode
1004
+ ```
1005
+
1006
+ > Pelajari lebih lanjut tentang optimasi metadata [di dokumentasi resmi Next.js](https://nextjs.org/docs/app/building-your-application/optimizing/metadata).
1007
+
1008
+ ### (Opsional) Langkah 9: Internasionalisasi sitemap.xml dan robots.txt Anda
1009
+
1010
+ Untuk menginternasionalisasi `sitemap.xml` dan `robots.txt` Anda, Anda dapat menggunakan fungsi `getMultilingualUrls` yang disediakan oleh Intlayer. Fungsi ini memungkinkan Anda untuk menghasilkan URL multibahasa untuk sitemap Anda.
1011
+
1012
+ ```tsx fileName="src/app/sitemap.ts" codeFormat="typescript"
1013
+ import { getMultilingualUrls } from "intlayer";
1014
+ import type { MetadataRoute } from "next";
1015
+
1016
+ const sitemap = (): MetadataRoute.Sitemap => [
1017
+ {
1018
+ url: "https://example.com",
1019
+ alternates: {
1020
+ languages: { ...getMultilingualUrls("https://example.com") },
1021
+ },
1022
+ },
1023
+ {
1024
+ url: "https://example.com/login",
1025
+ alternates: {
1026
+ languages: { ...getMultilingualUrls("https://example.com/login") },
1027
+ },
1028
+ },
1029
+ {
1030
+ url: "https://example.com/register",
1031
+ alternates: {
1032
+ languages: { ...getMultilingualUrls("https://example.com/register") },
1033
+ },
1034
+ },
1035
+ ];
1036
+
1037
+ export default sitemap;
1038
+ ```
1039
+
1040
+ ```jsx fileName="src/app/sitemap.mjx" codeFormat="esm"
1041
+ import { getMultilingualUrls } from "intlayer";
1042
+
1043
+ const sitemap = () => [
1044
+ {
1045
+ url: "https://example.com",
1046
+ alternates: {
1047
+ languages: { ...getMultilingualUrls("https://example.com") },
1048
+ },
1049
+ },
1050
+ {
1051
+ url: "https://example.com/login",
1052
+ alternates: {
1053
+ languages: { ...getMultilingualUrls("https://example.com/login") },
1054
+ },
1055
+ },
1056
+ {
1057
+ url: "https://example.com/register",
1058
+ alternates: {
1059
+ languages: { ...getMultilingualUrls("https://example.com/register") },
1060
+ },
1061
+ },
1062
+ ];
1063
+
1064
+ export default sitemap;
1065
+ ```
1066
+
1067
+ ```jsx fileName="src/app/sitemap.csx" codeFormat="commonjs"
1068
+ const { getMultilingualUrls } = require("intlayer");
1069
+
1070
+ const sitemap = () => [
1071
+ {
1072
+ url: "https://example.com",
1073
+ alternates: {
1074
+ languages: { ...getMultilingualUrls("https://example.com") },
1075
+ },
1076
+ },
1077
+ {
1078
+ url: "https://example.com/login",
1079
+ alternates: {
1080
+ languages: { ...getMultilingualUrls("https://example.com/login") },
1081
+ },
1082
+ },
1083
+ {
1084
+ url: "https://example.com/register",
1085
+ alternates: {
1086
+ languages: { ...getMultilingualUrls("https://example.com/register") },
1087
+ },
1088
+ },
1089
+ ];
1090
+
1091
+ module.exports = sitemap;
1092
+ ```
1093
+
1094
+ ```tsx fileName="src/app/robots.ts" codeFormat="typescript"
1095
+ import type { MetadataRoute } from "next";
1096
+ import { getMultilingualUrls } from "intlayer";
1097
+
1098
+ const getAllMultilingualUrls = (urls: string[]) =>
1099
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)) as string[]);
1100
+
1101
+ // Mendefinisikan aturan untuk robots.txt
1102
+ const robots = (): MetadataRoute.Robots => ({
1103
+ rules: {
1104
+ userAgent: "*", // Mengizinkan semua user-agent
1105
+ allow: ["/"], // Mengizinkan akses ke root
1106
+ disallow: getAllMultilingualUrls(["/login", "/register"]), // Melarang akses ke halaman login dan register dalam semua bahasa
1107
+ },
1108
+ host: "https://example.com", // Host situs
1109
+ sitemap: `https://example.com/sitemap.xml`, // Lokasi sitemap
1110
+ });
1111
+
1112
+ export default robots;
1113
+ ```
1114
+
1115
+ ```jsx fileName="src/app/robots.mjx" codeFormat="esm"
1116
+ import { getMultilingualUrls } from "intlayer";
1117
+
1118
+ // Mendapatkan semua URL multibahasa dari daftar URL
1119
+ const getAllMultilingualUrls = (urls) =>
1120
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)));
1121
+
1122
+ const robots = () => ({
1123
+ rules: {
1124
+ userAgent: "*", // Mengizinkan semua user-agent
1125
+ allow: ["/"], // Mengizinkan akses ke root
1126
+ disallow: getAllMultilingualUrls(["/login", "/register"]), // Melarang akses ke halaman login dan register dalam semua bahasa
1127
+ },
1128
+ host: "https://example.com",
1129
+ sitemap: `https://example.com/sitemap.xml`,
1130
+ });
1131
+
1132
+ export default robots;
1133
+ ```
1134
+
1135
+ ```jsx fileName="src/app/robots.csx" codeFormat="commonjs"
1136
+ const { getMultilingualUrls } = require("intlayer");
1137
+
1138
+ const getAllMultilingualUrls = (urls) =>
1139
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)));
1140
+
1141
+ const robots = () => ({
1142
+ rules: {
1143
+ userAgent: "*",
1144
+ allow: ["/"],
1145
+ disallow: getAllMultilingualUrls(["/login", "/register"]),
1146
+ },
1147
+ host: "https://example.com",
1148
+ sitemap: `https://example.com/sitemap.xml`,
1149
+ });
1150
+
1151
+ module.exports = robots;
1152
+ ```
1153
+
1154
+ > Pelajari lebih lanjut tentang optimasi sitemap [di dokumentasi resmi Next.js](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/sitemap). Pelajari lebih lanjut tentang optimasi robots.txt [di dokumentasi resmi Next.js](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/robots).
1155
+
1156
+ ### (Opsional) Langkah 10: Ubah bahasa konten Anda
1157
+
1158
+ Untuk mengubah bahasa konten Anda di Next.js, cara yang direkomendasikan adalah menggunakan komponen `Link` untuk mengarahkan pengguna ke halaman yang sesuai dengan lokal yang diinginkan. Komponen `Link` memungkinkan prefetching halaman, yang membantu menghindari pemuatan ulang halaman secara penuh.
1159
+
1160
+ ```tsx fileName="src/components/LocaleSwitcher.tsx" codeFormat="typescript"
1161
+ "use client";
1162
+
1163
+ import type { FC } from "react";
1164
+ import {
1165
+ Locales,
1166
+ getHTMLTextDir,
1167
+ getLocaleName,
1168
+ getLocalizedUrl,
1169
+ } from "intlayer";
1170
+ import { useLocale } from "next-intlayer";
1171
+ import Link from "next/link";
1172
+
1173
+ export const LocaleSwitcher: FC = () => {
1174
+ const { locale, pathWithoutLocale, availableLocales, setLocale } =
1175
+ useLocale();
1176
+
1177
+ return (
1178
+ <div>
1179
+ <button popoverTarget="localePopover">{getLocaleName(locale)}</button>
1180
+ <div id="localePopover" popover="auto">
1181
+ {availableLocales.map((localeItem) => (
1182
+ <Link
1183
+ href={getLocalizedUrl(pathWithoutLocale, localeItem)}
1184
+ key={localeItem}
1185
+ aria-current={locale === localeItem ? "page" : undefined}
1186
+ onClick={() => setLocale(localeItem)}
1187
+ replace // Akan memastikan tombol "kembali" pada browser akan mengarahkan ke halaman sebelumnya
1188
+ >
1189
+ <span>
1190
+ {/* Lokal - misalnya FR */}
1191
+ {localeItem}
1192
+ </span>
1193
+ <span>
1194
+ {/* Bahasa dalam Lokal sendiri - misalnya Français */}
1195
+ {getLocaleName(localeItem, locale)}
1196
+ </span>
1197
+ <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
1198
+ {/* Bahasa dalam Lokal saat ini - misalnya Francés dengan lokal saat ini disetel ke Locales.SPANISH */}
1199
+ {getLocaleName(localeItem)}
1200
+ </span>
1201
+ <span dir="ltr" lang={Locales.ENGLISH}>
1202
+ {/* Bahasa dalam Bahasa Inggris - misalnya French */}
1203
+ {getLocaleName(localeItem, Locales.ENGLISH)}
1204
+ </span>
1205
+ </Link>
1206
+ ))}
1207
+ </div>
1208
+ </div>
1209
+ );
1210
+ };
1211
+ ```
1212
+
1213
+ ```jsx fileName="src/components/LocaleSwitcher.msx" codeFormat="esm"
1214
+ "use client";
1215
+
1216
+ import {
1217
+ Locales,
1218
+ getHTMLTextDir,
1219
+ getLocaleName,
1220
+ getLocalizedUrl,
1221
+ } from "intlayer";
1222
+ import { useLocale } from "next-intlayer";
1223
+ import Link from "next/link";
1224
+
1225
+ export const LocaleSwitcher = () => {
1226
+ const { locale, pathWithoutLocale, availableLocales, setLocale } =
1227
+ useLocale();
1228
+
1229
+ return (
1230
+ <div>
1231
+ <button popoverTarget="localePopover">{getLocaleName(locale)}</button>
1232
+ <div id="localePopover" popover="auto">
1233
+ {availableLocales.map((localeItem) => (
1234
+ <Link
1235
+ href={getLocalizedUrl(pathWithoutLocale, localeItem)}
1236
+ key={localeItem}
1237
+ aria-current={locale === localeItem ? "page" : undefined}
1238
+ onClick={() => setLocale(localeItem)}
1239
+ replace // Akan memastikan bahwa tombol "kembali" pada browser akan mengarahkan ke halaman sebelumnya
1240
+ >
1241
+ <span>
1242
+ {/* Lokal - misal FR */}
1243
+ {localeItem}
1244
+ </span>
1245
+ <span>
1246
+ {/* Bahasa dalam Lokal sendiri - misal Français */}
1247
+ {getLocaleName(localeItem, locale)}
1248
+ </span>
1249
+ <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
1250
+ {/* Bahasa dalam Lokal saat ini - misal Francés dengan lokal saat ini disetel ke Locales.SPANISH */}
1251
+ {getLocaleName(localeItem)}
1252
+ </span>
1253
+ <span dir="ltr" lang={Locales.ENGLISH}>
1254
+ {/* Bahasa dalam Bahasa Inggris - misal French */}
1255
+ {getLocaleName(localeItem, Locales.ENGLISH)}
1256
+ </span>
1257
+ </Link>
1258
+ ))}
1259
+ </div>
1260
+ </div>
1261
+ );
1262
+ };
1263
+ ```
1264
+
1265
+ ```jsx fileName="src/components/LocaleSwitcher.csx" codeFormat="commonjs"
1266
+ "use client";
1267
+
1268
+ const {
1269
+ Locales,
1270
+ getHTMLTextDir,
1271
+ getLocaleName,
1272
+ getLocalizedUrl,
1273
+ } = require("intlayer");
1274
+ const { useLocale } = require("next-intlayer");
1275
+ const Link = require("next/link");
1276
+
1277
+ export const LocaleSwitcher = () => {
1278
+ const { locale, pathWithoutLocale, availableLocales, setLocale } =
1279
+ useLocale();
1280
+
1281
+ return (
1282
+ <div>
1283
+ <button popoverTarget="localePopover">{getLocaleName(locale)}</button>
1284
+ <div id="localePopover" popover="auto">
1285
+ {availableLocales.map((localeItem) => (
1286
+ <Link
1287
+ href={getLocalizedUrl(pathWithoutLocale, localeItem)}
1288
+ key={localeItem}
1289
+ aria-current={locale === localeItem ? "page" : undefined}
1290
+ onClick={() => setLocale(localeItem)}
1291
+ replace // Akan memastikan tombol "kembali" pada browser akan mengarahkan ke halaman sebelumnya
1292
+ >
1293
+ <span>
1294
+ {/* Lokal - misalnya FR */}
1295
+ {localeItem}
1296
+ </span>
1297
+ <span>
1298
+ {/* Bahasa dalam Lokal sendiri - misalnya Français */}
1299
+ {getLocaleName(localeItem, locale)}
1300
+ </span>
1301
+ <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
1302
+ {/* Bahasa dalam Lokal saat ini - misalnya Francés dengan lokal saat ini disetel ke Locales.SPANISH */}
1303
+ {getLocaleName(localeItem)}
1304
+ </span>
1305
+ <span dir="ltr" lang={Locales.ENGLISH}>
1306
+ {/* Bahasa dalam Bahasa Inggris - misalnya French */}
1307
+ {getLocaleName(localeItem, Locales.ENGLISH)}
1308
+ </span>
1309
+ </Link>
1310
+ ))}
1311
+ </div>
1312
+ </div>
1313
+ );
1314
+ };
1315
+ ```
1316
+
1317
+ > Cara alternatif adalah menggunakan fungsi `setLocale` yang disediakan oleh hook `useLocale`. Fungsi ini tidak akan memungkinkan prefetching halaman. Lihat dokumentasi [`useLocale` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/packages/next-intlayer/useLocale.md) untuk detail lebih lanjut.
1318
+
1319
+ > Anda juga dapat mengatur fungsi pada opsi `onLocaleChange` untuk memicu fungsi kustom saat locale berubah.
1320
+
1321
+ ```tsx fileName="src/components/LocaleSwitcher.tsx"
1322
+ "use client";
1323
+
1324
+ import { useRouter } from "next/navigation";
1325
+ import { useLocale } from "next-intlayer";
1326
+ import { getLocalizedUrl } from "intlayer";
1327
+
1328
+ // ... Sisa kode
1329
+
1330
+ const router = useRouter();
1331
+ const { setLocale } = useLocale({
1332
+ onLocaleChange: (locale) => {
1333
+ router.push(getLocalizedUrl(pathWithoutLocale, locale));
1334
+ },
1335
+ });
1336
+
1337
+ return (
1338
+ <button onClick={() => setLocale(Locales.FRENCH)}>
1339
+ Ganti ke Bahasa Perancis
1340
+ </button>
1341
+ );
1342
+ ```
1343
+
1344
+ > Referensi dokumentasi:
1345
+ >
1346
+ > - [`useLocale` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/packages/next-intlayer/useLocale.md)
1347
+ > - [`getLocaleName` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/packages/intlayer/getLocaleName.md)
1348
+ > - [`getLocalizedUrl` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/packages/intlayer/getLocalizedUrl.md)
1349
+ > - [`getHTMLTextDir` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/packages/intlayer/getHTMLTextDir.md)
1350
+ > - atribut [`hrefLang`](https://developers.google.com/search/docs/specialty/international/localized-versions?hl=fr)
1351
+ > - atribut [`lang`](https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/lang)
1352
+ > - atribut [`dir`](https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/dir)
1353
+ > - atribut [`aria-current`](https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Attributes/aria-current)
1354
+
1355
+ ### (Opsional) Langkah 11: Membuat Komponen Link yang Dilokalkan
1356
+
1357
+ Untuk memastikan navigasi aplikasi Anda menghormati lokal saat ini, Anda dapat membuat komponen `Link` kustom. Komponen ini secara otomatis menambahkan awalan bahasa saat ini pada URL internal, sehingga. Misalnya, ketika pengguna berbahasa Prancis mengklik tautan ke halaman "Tentang", mereka akan diarahkan ke `/fr/about` alih-alih `/about`.
1358
+
1359
+ Perilaku ini berguna untuk beberapa alasan:
1360
+
1361
+ - **SEO dan Pengalaman Pengguna**: URL yang dilokalkan membantu mesin pencari mengindeks halaman spesifik bahasa dengan benar dan menyediakan konten kepada pengguna dalam bahasa pilihan mereka.
1362
+ - **Konsistensi**: Dengan menggunakan tautan yang dilokalkan di seluruh aplikasi Anda, Anda menjamin navigasi tetap dalam lokal saat ini, mencegah perubahan bahasa yang tidak diinginkan.
1363
+ - **Pemeliharaan**: Memusatkan logika lokalisasi dalam satu komponen menyederhanakan pengelolaan URL, sehingga basis kode Anda lebih mudah dipelihara dan diperluas seiring pertumbuhan aplikasi Anda.
1364
+
1365
+ Berikut adalah implementasi komponen `Link` yang dilokalisasi dalam TypeScript:
1366
+
1367
+ ```tsx fileName="src/components/Link.tsx" codeFormat="typescript"
1368
+ "use client";
1369
+
1370
+ import { getLocalizedUrl } from "intlayer";
1371
+ import NextLink, { type LinkProps as NextLinkProps } from "next/link";
1372
+ import { useLocale } from "next-intlayer";
1373
+ import type { PropsWithChildren, FC } from "react";
1374
+
1375
+ /**
1376
+ * Fungsi utilitas untuk memeriksa apakah URL yang diberikan bersifat eksternal.
1377
+ * Jika URL dimulai dengan http:// atau https://, maka dianggap eksternal.
1378
+ */
1379
+ export const checkIsExternalLink = (href?: string): boolean =>
1380
+ /^https?:\/\//.test(href ?? "");
1381
+
1382
+ /**
1383
+ * Komponen Link kustom yang menyesuaikan atribut href berdasarkan locale saat ini.
1384
+ * Untuk tautan internal, menggunakan `getLocalizedUrl` untuk menambahkan prefix locale pada URL (misalnya, /fr/about).
1385
+ * Ini memastikan navigasi tetap dalam konteks locale yang sama.
1386
+ */
1387
+ export const Link: FC<PropsWithChildren<NextLinkProps>> = ({
1388
+ href,
1389
+ children,
1390
+ ...props
1391
+ }) => {
1392
+ const { locale } = useLocale();
1393
+ const isExternalLink = checkIsExternalLink(href.toString());
1394
+
1395
+ // Jika tautan bersifat internal dan href valid diberikan, dapatkan URL yang sudah dilokalisasi.
1396
+ const hrefI18n: NextLinkProps["href"] =
1397
+ href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
1398
+
1399
+ return (
1400
+ <NextLink href={hrefI18n} {...props}>
1401
+ {children}
1402
+ </NextLink>
1403
+ );
1404
+ };
1405
+ ```
1406
+
1407
+ ```jsx fileName="src/components/Link.mjx" codeFormat="esm"
1408
+ "use client";
1409
+
1410
+ import { getLocalizedUrl } from "intlayer";
1411
+ import NextLink from "next/link";
1412
+ import { useLocale } from "next-intlayer";
1413
+
1414
+ /**
1415
+ * Fungsi utilitas untuk memeriksa apakah URL yang diberikan bersifat eksternal.
1416
+ * Jika URL dimulai dengan http:// atau https://, maka dianggap eksternal.
1417
+ */
1418
+ export const checkIsExternalLink = (href) => /^https?:\/\//.test(href ?? "");
1419
+
1420
+ /**
1421
+ * Komponen Link kustom yang menyesuaikan atribut href berdasarkan locale saat ini.
1422
+ * Untuk tautan internal, menggunakan `getLocalizedUrl` untuk menambahkan prefix locale pada URL (misalnya, /fr/about).
1423
+ * Ini memastikan navigasi tetap dalam konteks locale yang sama.
1424
+ */
1425
+ export const Link = ({ href, children, ...props }) => {
1426
+ const { locale } = useLocale();
1427
+ const isExternalLink = checkIsExternalLink(href.toString());
1428
+
1429
+ // Jika tautan bersifat internal dan href yang valid diberikan, dapatkan URL yang sudah dilokalkan.
1430
+ const hrefI18n =
1431
+ href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
1432
+
1433
+ return (
1434
+ <NextLink href={hrefI18n} {...props}>
1435
+ {children}
1436
+ </NextLink>
1437
+ );
1438
+ };
1439
+ ```
1440
+
1441
+ ```jsx fileName="src/components/Link.csx" codeFormat="commonjs"
1442
+ "use client";
1443
+
1444
+ const { getLocalizedUrl } = require("intlayer");
1445
+ const NextLink = require("next/link");
1446
+ const { useLocale } = require("next-intlayer");
1447
+
1448
+ /**
1449
+ * Fungsi utilitas untuk memeriksa apakah URL yang diberikan bersifat eksternal.
1450
+ * Jika URL dimulai dengan http:// atau https://, maka dianggap eksternal.
1451
+ */
1452
+ const checkIsExternalLink = (href) => /^https?:\/\//.test(href ?? "");
1453
+
1454
+ /**
1455
+ * Komponen Link kustom yang menyesuaikan atribut href berdasarkan locale saat ini.
1456
+ * Untuk tautan internal, menggunakan `getLocalizedUrl` untuk menambahkan prefix locale pada URL (misalnya, /fr/about).
1457
+ * Ini memastikan navigasi tetap dalam konteks locale yang sama.
1458
+ */
1459
+ const Link = ({ href, children, ...props }) => {
1460
+ const { locale } = useLocale();
1461
+ const isExternalLink = checkIsExternalLink(href.toString());
1462
+
1463
+ // Jika tautan bersifat internal dan href valid diberikan, dapatkan URL yang sudah dilokalkan.
1464
+ const hrefI18n =
1465
+ href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
1466
+
1467
+ return (
1468
+ <NextLink href={hrefI18n} {...props}>
1469
+ {children}
1470
+ </NextLink>
1471
+ );
1472
+ };
1473
+ ```
1474
+
1475
+ #### Cara Kerjanya
1476
+
1477
+ - **Mendeteksi Tautan Eksternal**:
1478
+ Fungsi pembantu `checkIsExternalLink` menentukan apakah sebuah URL adalah eksternal. Tautan eksternal dibiarkan tidak berubah karena tidak memerlukan lokalisasi.
1479
+
1480
+ - **Mengambil Locale Saat Ini**:
1481
+ Hook `useLocale` menyediakan locale saat ini (misalnya, `fr` untuk bahasa Perancis).
1482
+
1483
+ - **Melokalisasi URL**:
1484
+ Untuk tautan internal (yaitu, bukan eksternal), `getLocalizedUrl` digunakan untuk secara otomatis menambahkan prefix locale saat ini pada URL. Ini berarti jika pengguna Anda menggunakan bahasa Perancis, memberikan `/about` sebagai `href` akan mengubahnya menjadi `/fr/about`.
1485
+
1486
+ - **Mengembalikan Tautan**:
1487
+ Komponen mengembalikan elemen `<a>` dengan URL yang sudah dilokalisasi, memastikan navigasi konsisten dengan locale.
1488
+
1489
+ Dengan mengintegrasikan komponen `Link` ini di seluruh aplikasi Anda, Anda mempertahankan pengalaman pengguna yang koheren dan sadar bahasa sekaligus mendapatkan manfaat dari peningkatan SEO dan kegunaan.
1490
+
1491
+ ### (Opsional) Langkah 12: Mendapatkan locale saat ini di Server Actions
1492
+
1493
+ Jika Anda memerlukan locale aktif di dalam Server Action (misalnya, untuk melokalkan email atau menjalankan logika yang sadar locale), panggil `getLocale` dari `next-intlayer/server`:
1494
+
1495
+ ```tsx fileName="src/app/actions/getLocale.ts" codeFormat="typescript"
1496
+ "use server";
1497
+
1498
+ import { getLocale } from "next-intlayer/server";
1499
+
1500
+ export const myServerAction = async () => {
1501
+ const locale = await getLocale();
1502
+
1503
+ // Lakukan sesuatu dengan locale
1504
+ };
1505
+ ```
1506
+
1507
+ > Fungsi `getLocale` mengikuti strategi bertingkat untuk menentukan locale pengguna:
1508
+ >
1509
+ > 1. Pertama, memeriksa header permintaan untuk nilai locale yang mungkin telah diatur oleh proxy
1510
+ > 2. Jika tidak ditemukan locale di header, mencari locale yang disimpan dalam cookie
1511
+ > 3. Jika tidak ditemukan cookie, mencoba mendeteksi bahasa yang dipilih pengguna dari pengaturan browser mereka
1512
+ > 4. Sebagai upaya terakhir, menggunakan locale default yang dikonfigurasi dalam aplikasi
1513
+ >
1514
+ > Ini memastikan locale yang paling sesuai dipilih berdasarkan konteks yang tersedia.
1515
+
1516
+ ### (Opsional) Langkah 13: Optimalkan ukuran bundle Anda
1517
+
1518
+ Saat menggunakan `next-intlayer`, kamus disertakan dalam bundel untuk setiap halaman secara default. Untuk mengoptimalkan ukuran bundel, Intlayer menyediakan plugin SWC opsional yang secara cerdas menggantikan panggilan `useIntlayer` menggunakan makro. Ini memastikan kamus hanya disertakan dalam bundel untuk halaman yang benar-benar menggunakannya.
1519
+
1520
+ Untuk mengaktifkan optimasi ini, instal paket `@intlayer/swc`. Setelah terinstal, `next-intlayer` akan secara otomatis mendeteksi dan menggunakan plugin tersebut:
1521
+
1522
+ ```bash packageManager="npm"
1523
+ npm install @intlayer/swc --save-dev
1524
+ ```
1525
+
1526
+ ```bash packageManager="pnpm"
1527
+ pnpm add @intlayer/swc --save-dev
1528
+ ```
1529
+
1530
+ ```bash packageManager="yarn"
1531
+ yarn add @intlayer/swc --save-dev
1532
+ ```
1533
+
1534
+ > Catatan: Optimasi ini hanya tersedia untuk Next.js 13 ke atas.
1535
+
1536
+ > Catatan: Paket ini tidak diinstal secara default karena plugin SWC masih bersifat eksperimental di Next.js. Hal ini mungkin akan berubah di masa depan.
1537
+
1538
+ ### Memantau perubahan kamus pada Turbopack
1539
+
1540
+ Saat menggunakan Turbopack sebagai server pengembangan dengan perintah `next dev`, perubahan kamus tidak akan terdeteksi secara otomatis secara default.
1541
+
1542
+ Keterbatasan ini terjadi karena Turbopack tidak dapat menjalankan plugin webpack secara paralel untuk memantau perubahan pada file konten Anda. Untuk mengatasinya, Anda perlu menggunakan perintah `intlayer watch` untuk menjalankan server pengembangan dan pengawas build Intlayer secara bersamaan.
1543
+
1544
+ ```json5 fileName="package.json"
1545
+ {
1546
+ // ... Konfigurasi package.json Anda yang sudah ada
1547
+ "scripts": {
1548
+ // ... Konfigurasi skrip Anda yang sudah ada
1549
+ "dev": "intlayer watch --with 'next dev'",
1550
+ },
1551
+ }
1552
+ ```
1553
+
1554
+ > Jika Anda menggunakan next-intlayer@<=6.x.x, Anda perlu mempertahankan flag `--turbopack` agar aplikasi Next.js 16 dapat bekerja dengan benar menggunakan Turbopack. Kami menyarankan menggunakan next-intlayer@>=7.x.x untuk menghindari keterbatasan ini.
1555
+
1556
+ ### Konfigurasi TypeScript
1557
+
1558
+ Intlayer menggunakan augmentasi modul untuk mendapatkan manfaat dari TypeScript dan membuat basis kode Anda lebih kuat.
1559
+
1560
+ ![Autocompletion](https://github.com/aymericzip/intlayer/blob/main/docs/assets/autocompletion.png?raw=true)
1561
+
1562
+ ![Kesalahan Terjemahan](https://github.com/aymericzip/intlayer/blob/main/docs/assets/translation_error.png?raw=true)
1563
+
1564
+ Pastikan konfigurasi TypeScript Anda menyertakan tipe yang dihasilkan secara otomatis.
1565
+
1566
+ ```json5 fileName="tsconfig.json"
1567
+ {
1568
+ // ... Konfigurasi TypeScript Anda yang sudah ada
1569
+ "include": [
1570
+ // ... Konfigurasi TypeScript Anda yang sudah ada
1571
+ ".intlayer/**/*.ts", // Sertakan tipe yang dihasilkan secara otomatis
1572
+ ],
1573
+ }
1574
+ ```
1575
+
1576
+ ### Konfigurasi Git
1577
+
1578
+ Disarankan untuk mengabaikan file yang dihasilkan oleh Intlayer. Ini memungkinkan Anda untuk menghindari meng-commit file tersebut ke repositori Git Anda.
1579
+
1580
+ Untuk melakukan ini, Anda dapat menambahkan instruksi berikut ke file `.gitignore` Anda:
1581
+
1582
+ ```plaintext fileName=".gitignore"
1583
+ # Abaikan file yang dihasilkan oleh Intlayer
1584
+ .intlayer
1585
+ ```
1586
+
1587
+ ### Ekstensi VS Code
1588
+
1589
+ Untuk meningkatkan pengalaman pengembangan Anda dengan Intlayer, Anda dapat menginstal **Ekstensi VS Code Intlayer** resmi.
1590
+
1591
+ [Pasang dari VS Code Marketplace](https://marketplace.visualstudio.com/items?itemName=intlayer.intlayer-vs-code-extension)
1592
+
1593
+ Ekstensi ini menyediakan:
1594
+
1595
+ - **Autocompletion** untuk kunci terjemahan.
1596
+ - **Deteksi kesalahan waktu nyata** untuk terjemahan yang hilang.
1597
+ - **Pratinjau langsung** dari konten yang diterjemahkan.
1598
+ - **Tindakan cepat** untuk dengan mudah membuat dan memperbarui terjemahan.
1599
+
1600
+ Untuk detail lebih lanjut tentang cara menggunakan ekstensi ini, lihat [dokumentasi Ekstensi VS Code Intlayer](https://intlayer.org/doc/vs-code-extension).
1601
+
1602
+ ### Melangkah Lebih Jauh
1603
+
1604
+ Untuk melangkah lebih jauh, Anda dapat mengimplementasikan [editor visual](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/intlayer_visual_editor.md) atau mengeksternalisasi konten Anda menggunakan [CMS](https://github.com/aymericzip/intlayer/blob/main/docs/docs/id/intlayer_CMS.md).