@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,1652 @@
1
+ ---
2
+ createdAt: 2025-10-25
3
+ updatedAt: 2025-10-25
4
+ title: كيفية ترجمة تطبيق Next.js 16 الخاص بك – دليل i18n لعام 2025
5
+ description: اكتشف كيفية جعل موقع Next.js 16 الخاص بك متعدد اللغات. اتبع الوثائق لتدويل (i18n) وترجمته.
6
+ keywords:
7
+ - التدويل
8
+ - الوثائق
9
+ - Intlayer
10
+ - Next.js 16
11
+ - جافا سكريبت
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: بداية السجل
23
+ ---
24
+
25
+ # ترجمة موقع Next.js 16 الخاص بك باستخدام Intlayer | التدويل (i18n)
26
+
27
+ <iframe title="أفضل حل i18n لـ Next.js؟ اكتشف 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
+ شاهد [قالب التطبيق](https://github.com/aymericzip/intlayer-next-16-template) على GitHub.
30
+
31
+ ## ما هو Intlayer؟
32
+
33
+ **Intlayer** هي مكتبة دولية (i18n) مبتكرة ومفتوحة المصدر مصممة لتبسيط دعم اللغات المتعددة في تطبيقات الويب الحديثة. يتكامل Intlayer بسلاسة مع أحدث إطار عمل **Next.js 16**، بما في ذلك **App Router** القوي الخاص به. تم تحسينه للعمل مع **مكونات الخادم (Server Components)** لتحقيق عرض فعال وهو متوافق تمامًا مع [**Turbopack**](https://nextjs.org/docs/architecture/turbopack).
34
+
35
+ مع Intlayer، يمكنك:
36
+
37
+ - **إدارة الترجمات بسهولة** باستخدام قواميس إعلانية على مستوى المكون.
38
+ - **توطين البيانات الوصفية، والمسارات، والمحتوى بشكل ديناميكي**.
39
+ - **الوصول إلى الترجمات في كل من مكونات جانب العميل وجانب الخادم**.
40
+ - **ضمان دعم TypeScript** مع أنواع مولدة تلقائيًا، مما يحسن الإكمال التلقائي واكتشاف الأخطاء.
41
+ - **الاستفادة من الميزات المتقدمة**، مثل الكشف الديناميكي عن اللغة وتبديلها.
42
+
43
+ > يتوافق Intlayer مع Next.js 12 و 13 و 14 و 16. إذا كنت تستخدم Next.js Page Router، يمكنك الرجوع إلى هذا [الدليل](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/intlayer_with_nextjs_page_router.md). بالنسبة لـ Next.js 12 و 13 و 14 مع App Router، راجع هذا [الدليل](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/intlayer_with_nextjs_14.md).
44
+
45
+ ---
46
+
47
+ ## دليل خطوة بخطوة لإعداد Intlayer في تطبيق Next.js
48
+
49
+ ### الخطوة 1: تثبيت التبعيات
50
+
51
+ قم بتثبيت الحزم اللازمة باستخدام 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
+ الحزمة الأساسية التي توفر أدوات التدويل لإدارة التكوين، والترجمة، و[إعلان المحتوى](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/dictionary/content_file.md)، والترجمة البرمجية، و[أوامر CLI](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/intlayer_cli.md).
68
+
69
+ - **next-intlayer**
70
+
71
+ الحزمة التي تدمج Intlayer مع Next.js. توفر موفري السياق وخطافات (hooks) للتدويل في Next.js. بالإضافة إلى ذلك، تتضمن إضافة Next.js لدمج Intlayer مع [Webpack](https://webpack.js.org/) أو [Turbopack](https://nextjs.org/docs/app/api-reference/turbopack)، بالإضافة إلى وكيل لاكتشاف اللغة المفضلة للمستخدم، وإدارة ملفات تعريف الارتباط، والتعامل مع إعادة توجيه عناوين URL.
72
+
73
+ ### الخطوة 2: تكوين مشروعك
74
+
75
+ أنشئ ملف تكوين لتحديد لغات تطبيقك:
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
+ // لغات أخرى خاصة بك
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
+ // لغات أخرى خاصة بك
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
+ // لغات أخرى خاصة بك
125
+ ],
126
+ defaultLocale: Locales.ENGLISH,
127
+ },
128
+ };
129
+
130
+ module.exports = config;
131
+ ```
132
+
133
+ > من خلال ملف التهيئة هذا، يمكنك إعداد عناوين URL محلية، إعادة توجيه البروكسي، أسماء ملفات تعريف الارتباط، موقع وامتداد إعلانات المحتوى الخاصة بك، تعطيل سجلات Intlayer في وحدة التحكم، والمزيد. للحصول على قائمة كاملة بالمعلمات المتاحة، راجع [توثيق التهيئة](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/configuration.md).
134
+
135
+ ### الخطوة 3: دمج Intlayer في تهيئة Next.js الخاصة بك
136
+
137
+ قم بتكوين إعداد Next.js الخاص بك لاستخدام 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
+ /* خيارات التهيئة هنا */
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
+ /* خيارات التهيئة هنا */
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
+ /* خيارات التهيئة هنا */
167
+ };
168
+
169
+ module.exports = withIntlayer(nextConfig);
170
+ ```
171
+
172
+ > يتم استخدام مكون Next.js الإضافي `withIntlayer()` لدمج Intlayer مع Next.js. يضمن بناء ملفات إعلان المحتوى ومراقبتها في وضع التطوير. كما يحدد متغيرات بيئة Intlayer ضمن بيئات [Webpack](https://webpack.js.org/) أو [Turbopack](https://nextjs.org/docs/app/api-reference/turbopack). بالإضافة إلى ذلك، يوفر أسماء مستعارة لتحسين الأداء ويضمن التوافق مع مكونات الخادم.
173
+
174
+ > دالة `withIntlayer()` هي دالة وعد (Promise). تسمح بتحضير قواميس intlayer قبل بدء البناء. إذا كنت ترغب في استخدامها مع مكونات إضافية أخرى، يمكنك انتظارها باستخدام await. مثال:
175
+ >
176
+ > ```tsx
177
+ > const nextConfig = await withIntlayer(nextConfig);
178
+ > const nextConfigWithOtherPlugins = withOtherPlugins(nextConfig);
179
+ >
180
+ > export default nextConfigWithOtherPlugins;
181
+ > ```
182
+ >
183
+ > إذا كنت ترغب في استخدامه بشكل متزامن، يمكنك استخدام دالة `withIntlayerSync()`. مثال:
184
+ >
185
+ > ```tsx
186
+ > const nextConfig = withIntlayerSync(nextConfig);
187
+ > const nextConfigWithOtherPlugins = withOtherPlugins(nextConfig);
188
+ >
189
+ > export default nextConfigWithOtherPlugins;
190
+ > ```
191
+ >
192
+ > ### الخطوة 4: تعريف مسارات اللغة الديناميكية
193
+ >
194
+ > قم بإزالة كل شيء من `RootLayout` واستبداله بالكود التالي:
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
+ > // لا يزال بإمكانك تغليف العناصر الفرعية بمزودين آخرين، مثل `next-themes`، `react-query`، `framer-motion`، إلخ.
202
+ > <>{children}</>
203
+ > );
204
+ > ```
205
+
206
+ export default RootLayout;
207
+
208
+ ````
209
+
210
+ ```jsx {3} fileName="src/app/layout.mjx" codeFormat="esm"
211
+ import "./globals.css";
212
+
213
+ const RootLayout = ({ children }) => (
214
+ // لا يزال بإمكانك تغليف العناصر الفرعية بمزودين آخرين، مثل `next-themes`، `react-query`، `framer-motion`، إلخ.
215
+ <>{children}</>
216
+ );
217
+
218
+ export default RootLayout;
219
+ ````
220
+
221
+ ```jsx {1,8} fileName="src/app/layout.csx" codeFormat="commonjs"
222
+ require("./globals.css");
223
+
224
+ const RootLayout = ({ children }) => (
225
+ // لا يزال بإمكانك تغليف العناصر الفرعية بمزودين آخرين، مثل `next-themes`، `react-query`، `framer-motion`، إلخ.
226
+ <>{children}</>
227
+ );
228
+
229
+ module.exports = {
230
+ default: RootLayout,
231
+ generateStaticParams,
232
+ };
233
+ ```
234
+
235
+ > يتيح ترك مكون `RootLayout` فارغًا تعيين سمات [`lang`](https://developer.mozilla.org/fr/docs/Web/HTML/Global_attributes/lang) و [`dir`](https://developer.mozilla.org/fr/docs/Web/HTML/Global_attributes/dir) إلى وسم `<html>`.
236
+
237
+ لتنفيذ التوجيه الديناميكي، قم بتوفير المسار للغة بإضافة تخطيط جديد في دليل `[locale]` الخاص بك:
238
+
239
+ ```tsx fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
240
+ import type { NextLayoutIntlayer } from "next-intlayer";
241
+ import { Inter } from "next/font/google";
242
+ import { getHTMLTextDir } from "intlayer";
243
+
244
+ const inter = Inter({ subsets: ["latin"] });
245
+
246
+ const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
247
+ const { locale } = await params;
248
+ return (
249
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
250
+ <body className={inter.className}>{children}</body>
251
+ </html>
252
+ );
253
+ };
254
+
255
+ export default LocaleLayout;
256
+ ```
257
+
258
+ ```jsx fileName="src/app/[locale]/layout.mjx" codeFormat="esm"
259
+ import { getHTMLTextDir } from "intlayer";
260
+
261
+ const inter = Inter({ subsets: ["latin"] });
262
+
263
+ const LocaleLayout = async ({ children, params: { locale } }) => {
264
+ const { locale } = await params;
265
+ return (
266
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
267
+ <body className={inter.className}>{children}</body>
268
+ </html>
269
+ );
270
+ };
271
+
272
+ export default LocaleLayout;
273
+ ```
274
+
275
+ ```jsx fileName="src/app/[locale]/layout.csx" codeFormat="commonjs"
276
+ const { Inter } = require("next/font/google");
277
+ const { getHTMLTextDir } = require("intlayer");
278
+
279
+ const inter = Inter({ subsets: ["latin"] });
280
+
281
+ const LocaleLayout = async ({ children, params: { locale } }) => {
282
+ const { locale } = await params;
283
+ return (
284
+ <html lang={locale} dir={getHTMLTextDir(locale)}>
285
+ <body className={inter.className}>{children}</body>
286
+ </html>
287
+ );
288
+ };
289
+
290
+ module.exports = LocaleLayout;
291
+ ```
292
+
293
+ > يتم استخدام جزء المسار `[locale]` لتحديد اللغة المحلية. مثال: `/en-US/about` سيشير إلى `en-US` و `/fr/about` إلى `fr`.
294
+
295
+ > في هذه المرحلة، ستواجه الخطأ: `Error: Missing <html> and <body> tags in the root layout.`. هذا متوقع لأن ملف `/app/page.tsx` لم يعد قيد الاستخدام ويمكن حذفه. بدلاً من ذلك، سيقوم جزء المسار `[locale]` بتنشيط صفحة `/app/[locale]/page.tsx`. ونتيجة لذلك، ستكون الصفحات متاحة عبر مسارات مثل `/en`، `/fr`، `/es` في متصفحك. لتعيين اللغة الافتراضية كصفحة الجذر، راجع إعداد `proxy` في الخطوة 7.
296
+
297
+ ثم، قم بتنفيذ دالة `generateStaticParams` في تخطيط التطبيق الخاص بك.
298
+
299
+ ```tsx {1} fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
300
+ export { generateStaticParams } from "next-intlayer"; // السطر الذي يجب إدراجه
301
+
302
+ const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
303
+ /*... بقية الكود */
304
+ };
305
+
306
+ export default LocaleLayout;
307
+ ```
308
+
309
+ ```jsx {1} fileName="src/app/[locale]/layout.mjx" codeFormat="esm"
310
+ export { generateStaticParams } from "next-intlayer"; // السطر الذي يجب إدراجه
311
+
312
+ const LocaleLayout = async ({ children, params: { locale } }) => {
313
+ /*... بقية الكود */
314
+ };
315
+
316
+ // ... بقية الكود
317
+ ```
318
+
319
+ ```jsx {1,7} fileName="src/app/[locale]/layout.csx" codeFormat="commonjs"
320
+ const { generateStaticParams } = require("next-intlayer"); // السطر الذي يجب إدراجه
321
+
322
+ const LocaleLayout = async ({ children, params: { locale } }) => {
323
+ /*... بقية الكود */
324
+ };
325
+
326
+ module.exports = { default: LocaleLayout, generateStaticParams };
327
+ ```
328
+
329
+ > يضمن `generateStaticParams` أن يقوم تطبيقك ببناء الصفحات اللازمة مسبقًا لجميع اللغات، مما يقلل من حسابات وقت التشغيل ويحسن تجربة المستخدم. لمزيد من التفاصيل، راجع [توثيق Next.js حول generateStaticParams](https://nextjs.org/docs/app/building-your-application/rendering/static-and-dynamic-rendering#generate-static-params).
330
+
331
+ > يعمل Intlayer مع `export const dynamic = 'force-static';` لضمان بناء الصفحات مسبقًا لجميع اللغات.
332
+
333
+ ### الخطوة 5: إعلان المحتوى الخاص بك
334
+
335
+ قم بإنشاء وإدارة إعلانات المحتوى الخاصة بك لتخزين الترجمات:
336
+
337
+ ```tsx fileName="src/app/[locale]/page.content.ts" contentDeclarationFormat="typescript"
338
+ import { t, type Dictionary } from "intlayer";
339
+
340
+ const pageContent = {
341
+ key: "page",
342
+ content: {
343
+ getStarted: {
344
+ main: t({
345
+ en: "Get started by editing",
346
+ fr: "Commencez par éditer",
347
+ es: "Comience por editar",
348
+ }),
349
+ pageLink: "src/app/page.tsx",
350
+ },
351
+ },
352
+ } satisfies Dictionary;
353
+
354
+ export default pageContent;
355
+ ```
356
+
357
+ ```javascript fileName="src/app/[locale]/page.content.mjs" contentDeclarationFormat="esm"
358
+ import { t } from "intlayer";
359
+
360
+ /** @type {import('intlayer').Dictionary} */
361
+ const pageContent = {
362
+ key: "page",
363
+ content: {
364
+ getStarted: {
365
+ main: t({
366
+ en: "ابدأ بالتعديل",
367
+ fr: "Commencez par éditer",
368
+ es: "Comience por editar",
369
+ }),
370
+ pageLink: "src/app/page.tsx",
371
+ },
372
+ },
373
+ };
374
+
375
+ export default pageContent;
376
+ ```
377
+
378
+ ```javascript fileName="src/app/[locale]/page.content.cjs" contentDeclarationFormat="commonjs"
379
+ const { t } = require("intlayer");
380
+
381
+ /** @type {import('intlayer').Dictionary} */
382
+ const pageContent = {
383
+ key: "page",
384
+ content: {
385
+ getStarted: {
386
+ main: t({
387
+ en: "Get started by editing",
388
+ fr: "Commencez par éditer",
389
+ es: "Comience por editar",
390
+ ar: "ابدأ بالتعديل",
391
+ }),
392
+ pageLink: "src/app/page.tsx",
393
+ },
394
+ },
395
+ };
396
+
397
+ module.exports = pageContent;
398
+ ```
399
+
400
+ ```json fileName="src/app/[locale]/page.content.json" contentDeclarationFormat="json"
401
+ {
402
+ "$schema": "https://intlayer.org/schema.json",
403
+ "key": "page",
404
+ "content": {
405
+ "getStarted": {
406
+ "nodeType": "translation",
407
+ "translation": {
408
+ "en": "Get started by editing",
409
+ "fr": "Commencez par éditer",
410
+ "es": "Comience por editar",
411
+ "ar": "ابدأ بالتعديل"
412
+ }
413
+ },
414
+ "pageLink": "src/app/page.tsx"
415
+ }
416
+ }
417
+ ```
418
+
419
+ > يمكن تعريف إعلانات المحتوى الخاصة بك في أي مكان في تطبيقك بمجرد تضمينها في دليل `contentDir` (افتراضيًا، `./src`). ويجب أن تتطابق مع امتداد ملف إعلان المحتوى (افتراضيًا، `.content.{json,ts,tsx,js,jsx,mjs,mjx,cjs,cjx}`).
420
+
421
+ > لمزيد من التفاصيل، راجع [توثيق إعلان المحتوى](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/dictionary/content_file.md).
422
+
423
+ ### الخطوة 6: استخدام المحتوى في كودك
424
+
425
+ يمكنك الوصول إلى قواميس المحتوى الخاصة بك في جميع أنحاء تطبيقك:
426
+
427
+ ```tsx fileName="src/app/[locale]/page.tsx" codeFormat="typescript"
428
+ import type { FC } from "react";
429
+ import { ClientComponentExample } from "@components/ClientComponentExample";
430
+ import { ServerComponentExample } from "@components/ServerComponentExample";
431
+ import { type NextPageIntlayer, IntlayerClientProvider } from "next-intlayer";
432
+ import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
433
+
434
+ const PageContent: FC = () => {
435
+ const content = useIntlayer("page");
436
+
437
+ return (
438
+ <>
439
+ <p>{content.getStarted.main}</p> {/* المحتوى الرئيسي لبدء الاستخدام */}
440
+ <code>{content.getStarted.pageLink}</code>{" "}
441
+ {/* رابط الصفحة لبدء الاستخدام */}
442
+ </>
443
+ );
444
+ };
445
+
446
+ const Page: NextPageIntlayer = async ({ params }) => {
447
+ const { locale } = await params;
448
+
449
+ return (
450
+ <IntlayerServerProvider locale={locale}>
451
+ <PageContent />
452
+ <ServerComponentExample />
453
+
454
+ <IntlayerClientProvider locale={locale}>
455
+ <ClientComponentExample />
456
+ </IntlayerClientProvider>
457
+ </IntlayerServerProvider>
458
+ );
459
+ };
460
+
461
+ export default Page;
462
+ ```
463
+
464
+ ```jsx fileName="src/app/[locale]/page.mjx" codeFormat="esm"
465
+ import { ClientComponentExample } from "@components/ClientComponentExample";
466
+ import { ServerComponentExample } from "@components/ServerComponentExample";
467
+ import { IntlayerClientProvider } from "next-intlayer";
468
+ import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
469
+
470
+ const PageContent = () => {
471
+ const content = useIntlayer("page");
472
+
473
+ return (
474
+ <>
475
+ <p>{content.getStarted.main}</p> {/* // المحتوى الرئيسي للبدء */}
476
+ <code>{content.getStarted.pageLink}</code> {/* // رابط الصفحة للبدء */}
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`** يُستخدم لتوفير اللغة للمكونات على جانب العميل. يمكن وضعه في أي مكون أبوي، بما في ذلك التخطيط (layout). ومع ذلك، يُنصح بوضعه في التخطيط لأن Next.js يشارك كود التخطيط عبر الصفحات، مما يجعله أكثر كفاءة. باستخدام `IntlayerClientProvider` في التخطيط، تتجنب إعادة تهيئته لكل صفحة، مما يحسن الأداء ويحافظ على سياق تعريب متسق في جميع أنحاء تطبيقك.
533
+ - **`IntlayerServerProvider`** يُستخدم لتوفير اللغة لأطفال الخادم (server children). لا يمكن تعيينه في التخطيط.
534
+
535
+ > لا يمكن للتخطيط والصفحة مشاركة سياق خادم مشترك لأن نظام سياق الخادم يعتمد على مخزن بيانات لكل طلب (عبر آلية [React's cache](https://react.dev/reference/react/cache))، مما يؤدي إلى إعادة إنشاء كل "سياق" لأجزاء مختلفة من التطبيق. وضع المزود في تخطيط مشترك سيكسر هذا العزل، مما يمنع الانتشار الصحيح لقيم سياق الخادم إلى مكونات الخادم الخاصة بك.
536
+
537
+ > لا يمكن للتخطيط والصفحة مشاركة سياق خادم مشترك لأن نظام سياق الخادم يعتمد على مخزن بيانات لكل طلب (عبر آلية [ذاكرة التخزين المؤقت لـ React](https://react.dev/reference/react/cache))، مما يؤدي إلى إعادة إنشاء كل "سياق" لأجزاء مختلفة من التطبيق. وضع المزود في تخطيط مشترك سيكسر هذا العزل، مما يمنع الانتشار الصحيح لقيم سياق الخادم إلى مكونات الخادم الخاصة بك.
538
+
539
+ ```tsx {4,7} fileName="src/components/ClientComponentExample.tsx" codeFormat="typescript"
540
+ "use client";
541
+
542
+ import type { FC } from "react";
543
+ import { useIntlayer } from "next-intlayer";
544
+
545
+ export const ClientComponentExample: FC = () => {
546
+ const content = useIntlayer("client-component-example"); // إنشاء إعلان المحتوى المرتبط
547
+
548
+ return (
549
+ <div>
550
+ <h2>{content.title}</h2>
551
+ <p>{content.content}</p>
552
+ </div>
553
+ );
554
+ };
555
+ ```
556
+
557
+ ```jsx {3,6} fileName="src/components/ClientComponentExample.mjx" codeFormat="esm"
558
+ "use client";
559
+
560
+ import { useIntlayer } from "next-intlayer";
561
+
562
+ const ClientComponentExample = () => {
563
+ const content = useIntlayer("client-component-example"); // إنشاء إعلان المحتوى المرتبط
564
+
565
+ return (
566
+ <div>
567
+ <h2>{content.title}</h2>
568
+ <p>{content.content}</p>
569
+ </div>
570
+ );
571
+ };
572
+ ```
573
+
574
+ ```jsx {3,6} fileName="src/components/ClientComponentExample.csx" codeFormat="commonjs"
575
+ "use client";
576
+
577
+ const { useIntlayer } = require("next-intlayer");
578
+
579
+ const ClientComponentExample = () => {
580
+ const content = useIntlayer("client-component-example"); // إنشاء إعلان المحتوى المرتبط
581
+
582
+ return (
583
+ <div>
584
+ <h2>{content.title}</h2>
585
+ <p>{content.content}</p>
586
+ </div>
587
+ );
588
+ };
589
+ ```
590
+
591
+ ```tsx {2} fileName="src/components/ServerComponentExample.tsx" codeFormat="typescript"
592
+ import type { FC } from "react";
593
+ import { useIntlayer } from "next-intlayer/server";
594
+
595
+ export const ServerComponentExample: FC = () => {
596
+ const content = useIntlayer("server-component-example"); // إنشاء إعلان المحتوى المرتبط
597
+
598
+ return (
599
+ <div>
600
+ <h2>{content.title}</h2>
601
+ <p>{content.content}</p>
602
+ </div>
603
+ );
604
+ };
605
+ ```
606
+
607
+ ```jsx {1} fileName="src/components/ServerComponentExample.mjx" codeFormat="esm"
608
+ import { useIntlayer } from "next-intlayer/server";
609
+
610
+ const ServerComponentExample = () => {
611
+ const content = useIntlayer("server-component-example"); // إنشاء إعلان المحتوى المرتبط
612
+
613
+ return (
614
+ <div>
615
+ <h2>{content.title}</h2>
616
+ <p>{content.content}</p>
617
+ </div>
618
+ );
619
+ };
620
+ ```
621
+
622
+ ```jsx {1} fileName="src/components/ServerComponentExample.csx" codeFormat="commonjs"
623
+ const { useIntlayer } = require("next-intlayer/server");
624
+
625
+ const ServerComponentExample = () => {
626
+ const content = useIntlayer("server-component-example"); // إنشاء إعلان المحتوى المرتبط
627
+
628
+ return (
629
+ <div>
630
+ <h2>{content.title}</h2>
631
+ <p>{content.content}</p>
632
+ </div>
633
+ );
634
+ };
635
+ ```
636
+
637
+ > إذا كنت تريد استخدام المحتوى الخاص بك في خاصية من نوع `string`، مثل `alt`، `title`، `href`، `aria-label`، إلخ، يجب عليك استدعاء قيمة الدالة، مثل:
638
+
639
+ > ```jsx
640
+ > <img src={content.image.src.value} alt={content.image.value} />
641
+ > ```
642
+
643
+ > لمعرفة المزيد عن الخطاف `useIntlayer`، راجع [التوثيق](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/packages/next-intlayer/useIntlayer.md).
644
+
645
+ ### (اختياري) الخطوة 7: تكوين الوكيل لاكتشاف اللغة
646
+
647
+ قم بإعداد وكيل لاكتشاف اللغة المفضلة للمستخدم:
648
+
649
+ ```typescript fileName="src/proxy.ts" codeFormat="typescript"
650
+ export { intlayerProxy as proxy } from "next-intlayer/proxy";
651
+
652
+ export const config = {
653
+ matcher:
654
+ "/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
655
+ };
656
+ ```
657
+
658
+ ```javascript fileName="src/proxy.mjs" codeFormat="esm"
659
+ export { intlayerProxy as proxy } from "next-intlayer/proxy";
660
+
661
+ export const config = {
662
+ matcher:
663
+ "/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
664
+ };
665
+ ```
666
+
667
+ ```javascript fileName="src/proxy.cjs" codeFormat="commonjs"
668
+ const { intlayerProxy } = require("next-intlayer/proxy");
669
+
670
+ const config = {
671
+ matcher:
672
+ "/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
673
+ };
674
+
675
+ module.exports = { proxy: intlayerProxy, config };
676
+ ```
677
+
678
+ > يُستخدم `intlayerProxy` لاكتشاف اللغة المفضلة للمستخدم وإعادة توجيهه إلى عنوان URL المناسب كما هو محدد في [التكوين](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/configuration.md). بالإضافة إلى ذلك، يتيح حفظ اللغة المفضلة للمستخدم في ملف تعريف الارتباط (كوكي).
679
+
680
+ > إذا كنت بحاجة إلى ربط عدة بروكسيات معًا (على سبيل المثال، `intlayerProxy` مع المصادقة أو بروكسيات مخصصة)، يوفر Intlayer الآن أداة مساعدة تسمى `multipleProxies`.
681
+
682
+ ```ts
683
+ import { multipleProxies, intlayerProxy } from "next-intlayer/proxy";
684
+ import { customProxy } from "@utils/customProxy";
685
+
686
+ export const proxy = multipleProxies([intlayerProxy, customProxy]);
687
+ ```
688
+
689
+ ### (اختياري) الخطوة 8: تدويل بيانات التعريف الخاصة بك
690
+
691
+ في حال رغبتك في تدويل بيانات التعريف الخاصة بك، مثل عنوان الصفحة، يمكنك استخدام دالة `generateMetadata` المقدمة من Next.js. بداخلها، يمكنك استرجاع المحتوى من دالة `getIntlayer` لترجمة بيانات التعريف الخاصة بك.
692
+
693
+ ```typescript fileName="src/app/[locale]/metadata.content.ts" contentDeclarationFormat="typescript"
694
+ import { type Dictionary, t } from "intlayer";
695
+ import { Metadata } from "next";
696
+
697
+ const metadataContent = {
698
+ key: "page-metadata",
699
+ content: {
700
+ title: t({
701
+ en: "Create Next App",
702
+ fr: "Créer une application Next.js",
703
+ es: "Crear una aplicación Next.js",
704
+ }),
705
+ description: t({
706
+ en: "Generated by create next app",
707
+ fr: "Généré par create next app",
708
+ es: "Generado por create next app",
709
+ }),
710
+ },
711
+ } satisfies Dictionary<Metadata>;
712
+
713
+ export default metadataContent;
714
+ ```
715
+
716
+ ```javascript fileName="src/app/[locale]/metadata.content.mjs" contentDeclarationFormat="esm"
717
+ import { t } from "intlayer";
718
+
719
+ /** @type {import('intlayer').Dictionary<import('next').Metadata>} */
720
+ const metadataContent = {
721
+ key: "page-metadata",
722
+ content: {
723
+ title: t({
724
+ en: "Create Next App",
725
+ fr: "Créer une application Next.js",
726
+ es: "Crear una aplicación Next.js",
727
+ }),
728
+ description: t({
729
+ en: "Generated by create next app",
730
+ fr: "Généré par create next app",
731
+ es: "Generado por create next app",
732
+ }),
733
+ },
734
+ };
735
+
736
+ export default metadataContent;
737
+ ```
738
+
739
+ ```javascript fileName="src/app/[locale]/metadata.content.cjs" contentDeclarationFormat="commonjs"
740
+ const { t } = require("intlayer");
741
+
742
+ /** @type {import('intlayer').Dictionary<import('next').Metadata>} */
743
+ const metadataContent = {
744
+ key: "page-metadata",
745
+ content: {
746
+ title: t({
747
+ en: "Create Next App",
748
+ fr: "Créer une application Next.js",
749
+ es: "Crear una aplicación Next.js",
750
+ }),
751
+ description: t({
752
+ en: "Generated by create next app",
753
+ fr: "Généré par create next app",
754
+ es: "Generado por create next app",
755
+ }),
756
+ },
757
+ };
758
+
759
+ module.exports = metadataContent;
760
+ fr: "تم إنشاؤه بواسطة create next app",
761
+ es: "تم إنشاؤه بواسطة create next app",
762
+ }),
763
+ },
764
+ };
765
+
766
+ export default metadataContent;
767
+ ```
768
+
769
+ ```javascript fileName="src/app/[locale]/metadata.content.cjs" contentDeclarationFormat="commonjs"
770
+ const { t } = require("intlayer");
771
+
772
+ /** @type {import('intlayer').Dictionary<import('next').Metadata>} */
773
+ const metadataContent = {
774
+ key: "page-metadata",
775
+ content: {
776
+ title: t({
777
+ en: "Create Next App",
778
+ fr: "Créer une application Next.js",
779
+ es: "Crear una aplicación Next.js",
780
+ }),
781
+ description: t({
782
+ en: "Generated by create next app",
783
+ fr: "تم إنشاؤه بواسطة create next app",
784
+ es: "تم إنشاؤه بواسطة create next app",
785
+ }),
786
+ },
787
+ };
788
+
789
+ module.exports = metadataContent;
790
+ ```
791
+
792
+ ```json fileName="src/app/[locale]/metadata.content.json" contentDeclarationFormat="json"
793
+ {
794
+ "key": "page-metadata",
795
+ "content": {
796
+ "title": {
797
+ "nodeType": "translation",
798
+ "translation": {
799
+ "en": "Preact logo",
800
+ "ar": "شعار Preact",
801
+ "fr": "Logo Preact",
802
+ "es": "Logo Preact"
803
+ }
804
+ },
805
+ "description": {
806
+ "nodeType": "translation",
807
+ "translation": {
808
+ "en": "Generated by create next app",
809
+ "ar": "تم إنشاؤه بواسطة create next app",
810
+ "fr": "Généré par create next app",
811
+ "es": "Generado por create next app"
812
+ }
813
+ }
814
+ }
815
+ }
816
+ ```
817
+
818
+ ````typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
819
+ import { getIntlayer, getMultilingualUrls } from "intlayer";
820
+ import type { Metadata } from "next";
821
+ import type { LocalPromiseParams } from "next-intlayer";
822
+
823
+ export const generateMetadata = async ({
824
+ params,
825
+ }: LocalPromiseParams): Promise<Metadata> => {
826
+ const { locale } = await params;
827
+
828
+ const metadata = getIntlayer("page-metadata", locale);
829
+
830
+ /**
831
+ * ينشئ كائن يحتوي على جميع الروابط لكل لغة.
832
+ *
833
+ * مثال:
834
+ * ```ts
835
+ * getMultilingualUrls('/about');
836
+ *
837
+ * // يعيد
838
+ * // {
839
+ * // en: '/about',
840
+ * // fr: '/fr/about',
841
+ * // es: '/es/about',
842
+ * // }
843
+ * ```
844
+ */
845
+ const multilingualUrls = getMultilingualUrls("/");
846
+
847
+ return {
848
+ ...metadata,
849
+ alternates: {
850
+ canonical: multilingualUrls[locale as keyof typeof multilingualUrls],
851
+ languages: { ...multilingualUrls, "x-default": "/" },
852
+ },
853
+ openGraph: {
854
+ url: multilingualUrls[locale],
855
+ },
856
+ };
857
+ };
858
+
859
+ // ... بقية الكود
860
+ ````
861
+
862
+ ````javascript fileName="src/app/[locale]/layout.mjs or src/app/[locale]/page.mjs" codeFormat="esm"
863
+ import { getIntlayer, getMultilingualUrls } from "intlayer";
864
+
865
+ export const generateMetadata = async ({ params }) => {
866
+ const { locale } = await params;
867
+
868
+ const metadata = getIntlayer("page-metadata", locale);
869
+
870
+ /**
871
+ * ينشئ كائن يحتوي على جميع عناوين URL لكل لغة.
872
+ *
873
+ * مثال:
874
+ * ```ts
875
+ * getMultilingualUrls('/about');
876
+ *
877
+ * // يعيد
878
+ * // {
879
+ * // en: '/about',
880
+ * // fr: '/fr/about',
881
+ * // es: '/es/about'
882
+ * // }
883
+ * ```
884
+ */
885
+ const multilingualUrls = getMultilingualUrls("/");
886
+
887
+ return {
888
+ ...metadata,
889
+ alternates: {
890
+ canonical: multilingualUrls[locale],
891
+ languages: { ...multilingualUrls, "x-default": "/" },
892
+ },
893
+ openGraph: {
894
+ url: multilingualUrls[locale],
895
+ },
896
+ };
897
+ };
898
+
899
+ // ... بقية الكود
900
+ ````
901
+
902
+ ````javascript fileName="src/app/[locale]/layout.cjs or src/app/[locale]/page.cjs" codeFormat="commonjs"
903
+ const { getIntlayer, getMultilingualUrls } = require("intlayer");
904
+
905
+ const generateMetadata = async ({ params }) => {
906
+ const { locale } = await params;
907
+
908
+ const metadata = getIntlayer("page-metadata", locale);
909
+
910
+ /**
911
+ * ينشئ كائن يحتوي على جميع الروابط لكل لغة.
912
+ *
913
+ * مثال:
914
+ * ```ts
915
+ * getMultilingualUrls('/about');
916
+ *
917
+ * // يعيد
918
+ * // {
919
+ * // en: '/about',
920
+ * // fr: '/fr/about',
921
+ * // es: '/es/about'
922
+ * // }
923
+ * ```
924
+ */
925
+ const multilingualUrls = getMultilingualUrls("/");
926
+
927
+ return {
928
+ ...metadata,
929
+ alternates: {
930
+ canonical: multilingualUrls[locale],
931
+ languages: { ...multilingualUrls, "x-default": "/" },
932
+ },
933
+ openGraph: {
934
+ url: multilingualUrls[locale],
935
+ },
936
+ };
937
+ };
938
+
939
+ module.exports = { generateMetadata };
940
+
941
+ // ... بقية الكود
942
+ ````
943
+
944
+ > لاحظ أن الدالة `getIntlayer` المستوردة من `next-intlayer` تُعيد المحتوى الخاص بك مغلفًا في `IntlayerNode`، مما يسمح بالتكامل مع المحرر البصري. بالمقابل، الدالة `getIntlayer` المستوردة من `intlayer` تُعيد المحتوى الخاص بك مباشرةً بدون خصائص إضافية.
945
+
946
+ بدلاً من ذلك، يمكنك استخدام دالة `getTranslation` لإعلان بيانات التعريف الخاصة بك. ومع ذلك، يُنصح باستخدام ملفات إعلان المحتوى لأتمتة ترجمة بيانات التعريف الخاصة بك وفصل المحتوى في مرحلة ما.
947
+
948
+ ````typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
949
+ import {
950
+ type IConfigLocales,
951
+ getTranslation,
952
+ getMultilingualUrls,
953
+ } from "intlayer";
954
+ import type { Metadata } from "next";
955
+ import type { LocalPromiseParams } from "next-intlayer";
956
+
957
+ export const generateMetadata = async ({
958
+ params,
959
+ }: LocalPromiseParams): Promise<Metadata> => {
960
+ const { locale } = await params;
961
+ const t = <T>(content: IConfigLocales<T>) => getTranslation(content, locale);
962
+
963
+ return {
964
+ title: t<string>({
965
+ en: "My title",
966
+ بدلاً من ذلك، يمكنك استخدام دالة `getTranslation` لإعلان بيانات التعريف الخاصة بك. ومع ذلك، يُوصى باستخدام ملفات إعلان المحتوى لأتمتة ترجمة بيانات التعريف الخاصة بك وفصل المحتوى في مرحلة ما.
967
+
968
+ ```typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
969
+ import {
970
+ type IConfigLocales,
971
+ getTranslation,
972
+ getMultilingualUrls,
973
+ } from "intlayer";
974
+ import type { Metadata } from "next";
975
+ import type { LocalPromiseParams } from "next-intlayer";
976
+
977
+ export const generateMetadata = async ({
978
+ params,
979
+ }: LocalPromiseParams): Promise<Metadata> => {
980
+ const { locale } = await params;
981
+ const t = <T>(content: IConfigLocales<T>) => getTranslation(content, locale);
982
+
983
+ return {
984
+ title: t<string>({
985
+ en: "My title",
986
+ fr: "Mon titre",
987
+ es: "Mi título",
988
+ }),
989
+ description: t({
990
+ en: "My description",
991
+ fr: "Ma description",
992
+ es: "Mi descripción",
993
+ }),
994
+ };
995
+ };
996
+
997
+ // ... بقية الكود
998
+ ````
999
+
1000
+ ```javascript fileName="src/app/[locale]/layout.mjs or src/app/[locale]/page.mjs" codeFormat="esm"
1001
+ import { getTranslation, getMultilingualUrls } from "intlayer";
1002
+
1003
+ export const generateMetadata = async ({ params }) => {
1004
+ const { locale } = await params;
1005
+ const t = (content) => getTranslation(content, locale);
1006
+
1007
+ return {
1008
+ title: t({
1009
+ en: "My title",
1010
+ fr: "Mon titre",
1011
+ es: "Mi título",
1012
+ }),
1013
+ description: t({
1014
+ en: "My description",
1015
+ fr: "Ma description",
1016
+ es: "Mi descripción",
1017
+ }),
1018
+ };
1019
+ };
1020
+
1021
+ // ... بقية الكود
1022
+ ```
1023
+
1024
+ ```javascript fileName="src/app/[locale]/layout.cjs or src/app/[locale]/page.cjs" codeFormat="commonjs"
1025
+ const { getTranslation, getMultilingualUrls } = require("intlayer");
1026
+
1027
+ const generateMetadata = async ({ params }) => {
1028
+ const { locale } = await params;
1029
+
1030
+ const t = (content) => getTranslation(content, locale);
1031
+
1032
+ return {
1033
+ title: t({
1034
+ en: "My title",
1035
+ fr: "Mon titre",
1036
+ es: "Mi título",
1037
+ }),
1038
+ description: t({
1039
+ en: "My description",
1040
+ fr: "Ma description",
1041
+ es: "Mi descripción",
1042
+ }),
1043
+ };
1044
+ };
1045
+
1046
+ module.exports = { generateMetadata };
1047
+
1048
+ // ... بقية الكود
1049
+ ```
1050
+
1051
+ > تعرّف على المزيد حول تحسين بيانات التعريف [في الوثائق الرسمية لـ Next.js](https://nextjs.org/docs/app/building-your-application/optimizing/metadata).
1052
+
1053
+ ### (اختياري) الخطوة 9: تعريب ملف sitemap.xml و robots.txt الخاص بك
1054
+
1055
+ لتعريب ملفي `sitemap.xml` و `robots.txt`، يمكنك استخدام دالة `getMultilingualUrls` المقدمة من Intlayer. تتيح لك هذه الدالة إنشاء روابط متعددة اللغات لخريطة الموقع الخاصة بك.
1056
+
1057
+ ```tsx fileName="src/app/sitemap.ts" codeFormat="typescript"
1058
+ import { getMultilingualUrls } from "intlayer";
1059
+ import type { MetadataRoute } from "next";
1060
+
1061
+ const sitemap = (): MetadataRoute.Sitemap => [
1062
+ {
1063
+ url: "https://example.com",
1064
+ alternates: {
1065
+ languages: { ...getMultilingualUrls("https://example.com") },
1066
+ },
1067
+ },
1068
+ {
1069
+ url: "https://example.com/login",
1070
+ alternates: {
1071
+ languages: { ...getMultilingualUrls("https://example.com/login") },
1072
+ },
1073
+ },
1074
+ {
1075
+ url: "https://example.com/register",
1076
+ alternates: {
1077
+ languages: { ...getMultilingualUrls("https://example.com/register") },
1078
+ },
1079
+ },
1080
+ ];
1081
+
1082
+ export default sitemap;
1083
+ ```
1084
+
1085
+ ```jsx fileName="src/app/sitemap.mjx" codeFormat="esm"
1086
+ import { getMultilingualUrls } from "intlayer";
1087
+
1088
+ const sitemap = () => [
1089
+ {
1090
+ url: "https://example.com",
1091
+ alternates: {
1092
+ languages: { ...getMultilingualUrls("https://example.com") },
1093
+ },
1094
+ },
1095
+ {
1096
+ url: "https://example.com/login",
1097
+ alternates: {
1098
+ languages: { ...getMultilingualUrls("https://example.com/login") },
1099
+ },
1100
+ },
1101
+ {
1102
+ url: "https://example.com/register",
1103
+ alternates: {
1104
+ languages: { ...getMultilingualUrls("https://example.com/register") },
1105
+ },
1106
+ },
1107
+ ];
1108
+
1109
+ export default sitemap;
1110
+ ```
1111
+
1112
+ ```jsx fileName="src/app/sitemap.csx" codeFormat="commonjs"
1113
+ const { getMultilingualUrls } = require("intlayer");
1114
+
1115
+ // خريطة الموقع مع روابط متعددة اللغات لكل صفحة
1116
+ const sitemap = () => [
1117
+ {
1118
+ url: "https://example.com",
1119
+ alternates: {
1120
+ languages: { ...getMultilingualUrls("https://example.com") },
1121
+ },
1122
+ },
1123
+ {
1124
+ url: "https://example.com/login",
1125
+ alternates: {
1126
+ languages: { ...getMultilingualUrls("https://example.com/login") },
1127
+ },
1128
+ },
1129
+ {
1130
+ url: "https://example.com/register",
1131
+ alternates: {
1132
+ languages: { ...getMultilingualUrls("https://example.com/register") },
1133
+ },
1134
+ },
1135
+ ];
1136
+
1137
+ module.exports = sitemap;
1138
+ ```
1139
+
1140
+ ```tsx fileName="src/app/robots.ts" codeFormat="typescript"
1141
+ import type { MetadataRoute } from "next";
1142
+ import { getMultilingualUrls } from "intlayer";
1143
+
1144
+ const getAllMultilingualUrls = (urls: string[]) =>
1145
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)) as string[]);
1146
+
1147
+ // دالة للحصول على جميع عناوين URL متعددة اللغات
1148
+ const robots = (): MetadataRoute.Robots => ({
1149
+ rules: {
1150
+ userAgent: "*", // السماح لجميع وكلاء المستخدم
1151
+ allow: ["/"], // السماح بالوصول إلى الصفحة الرئيسية
1152
+ disallow: getAllMultilingualUrls(["/login", "/register"]), // منع الوصول إلى صفحات تسجيل الدخول والتسجيل
1153
+ },
1154
+ host: "https://example.com", // المضيف الأساسي للموقع
1155
+ sitemap: `https://example.com/sitemap.xml`, // رابط خريطة الموقع
1156
+ });
1157
+
1158
+ export default robots;
1159
+ ```
1160
+
1161
+ ```jsx fileName="src/app/robots.mjx" codeFormat="esm"
1162
+ import { getMultilingualUrls } from "intlayer";
1163
+
1164
+ // دالة للحصول على جميع عناوين URL متعددة اللغات
1165
+ const getAllMultilingualUrls = (urls) =>
1166
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)));
1167
+
1168
+ const robots = () => ({
1169
+ rules: {
1170
+ userAgent: "*", // السماح لجميع وكلاء المستخدم
1171
+ allow: ["/"], // السماح بالوصول إلى الصفحة الرئيسية
1172
+ disallow: getAllMultilingualUrls(["/login", "/register"]), // منع الوصول إلى صفحات تسجيل الدخول والتسجيل
1173
+ },
1174
+ host: "https://example.com",
1175
+ sitemap: `https://example.com/sitemap.xml`,
1176
+ });
1177
+
1178
+ export default robots;
1179
+ ```
1180
+
1181
+ ```jsx fileName="src/app/robots.csx" codeFormat="commonjs"
1182
+ const { getMultilingualUrls } = require("intlayer");
1183
+
1184
+ // دالة للحصول على جميع الروابط متعددة اللغات
1185
+ const getAllMultilingualUrls = (urls) =>
1186
+ urls.flatMap((url) => Object.values(getMultilingualUrls(url)));
1187
+
1188
+ const robots = () => ({
1189
+ rules: {
1190
+ userAgent: "*", // السماح لجميع عناكب البحث
1191
+ allow: ["/"], // السماح بالوصول إلى الصفحة الرئيسية
1192
+ disallow: getAllMultilingualUrls(["/login", "/register"]), // منع الوصول إلى صفحات تسجيل الدخول والتسجيل بجميع اللغات
1193
+ },
1194
+ host: "https://example.com",
1195
+ sitemap: `https://example.com/sitemap.xml`,
1196
+ });
1197
+
1198
+ module.exports = robots;
1199
+ ```
1200
+
1201
+ > تعرّف على المزيد حول تحسين خريطة الموقع [في الوثائق الرسمية لـ Next.js](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/sitemap). تعرّف على المزيد حول تحسين ملف robots.txt [في الوثائق الرسمية لـ Next.js](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/robots).
1202
+
1203
+ ### (اختياري) الخطوة 10: تغيير لغة المحتوى الخاص بك
1204
+
1205
+ لتغيير لغة المحتوى الخاص بك في Next.js، الطريقة الموصى بها هي استخدام مكون `Link` لإعادة توجيه المستخدمين إلى الصفحة المحلية المناسبة. يتيح مكون `Link` التحميل المسبق للصفحة، مما يساعد على تجنب إعادة تحميل الصفحة بالكامل.
1206
+
1207
+ ```tsx fileName="src/components/LocaleSwitcher.tsx" codeFormat="typescript"
1208
+ "use client";
1209
+
1210
+ import type { FC } from "react";
1211
+ import {
1212
+ Locales,
1213
+ getHTMLTextDir,
1214
+ getLocaleName,
1215
+ getLocalizedUrl,
1216
+ } from "intlayer";
1217
+ import { useLocale } from "next-intlayer";
1218
+ import Link from "next/link";
1219
+
1220
+ export const LocaleSwitcher: FC = () => {
1221
+ const { locale, pathWithoutLocale, availableLocales, setLocale } =
1222
+ useLocale();
1223
+
1224
+ return (
1225
+ <div>
1226
+ <button popoverTarget="localePopover">{getLocaleName(locale)}</button>
1227
+ <div id="localePopover" popover="auto">
1228
+ {availableLocales.map((localeItem) => (
1229
+ <Link
1230
+ href={getLocalizedUrl(pathWithoutLocale, localeItem)}
1231
+ key={localeItem}
1232
+ aria-current={locale === localeItem ? "page" : undefined}
1233
+ onClick={() => setLocale(localeItem)}
1234
+ replace // سيضمن أن زر "العودة" في المتصفح يعيد التوجيه إلى الصفحة السابقة
1235
+ >
1236
+ <span>
1237
+ {/* اللغة المحلية - على سبيل المثال FR */}
1238
+ {localeItem}
1239
+ </span>
1240
+ <span>
1241
+ {/* اللغة بلغتها المحلية - على سبيل المثال Français */}
1242
+ {getLocaleName(localeItem, locale)}
1243
+ </span>
1244
+ <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
1245
+ {/* اللغة باللغة المحلية الحالية - على سبيل المثال Francés مع تعيين اللغة المحلية الحالية إلى Locales.SPANISH */}
1246
+ {getLocaleName(localeItem)}
1247
+ </span>
1248
+ <span dir="ltr" lang={Locales.ENGLISH}>
1249
+ {/* اللغة باللغة الإنجليزية - على سبيل المثال French */}
1250
+ {getLocaleName(localeItem, Locales.ENGLISH)}
1251
+ </span>
1252
+ </Link>
1253
+ ))}
1254
+ </div>
1255
+ </div>
1256
+ );
1257
+ };
1258
+ ```
1259
+
1260
+ ```jsx fileName="src/components/LocaleSwitcher.msx" codeFormat="esm"
1261
+ "use client";
1262
+
1263
+ import {
1264
+ Locales,
1265
+ getHTMLTextDir,
1266
+ getLocaleName,
1267
+ getLocalizedUrl,
1268
+ } from "intlayer";
1269
+ import { useLocale } from "next-intlayer";
1270
+ import Link from "next/link";
1271
+
1272
+ export const LocaleSwitcher = () => {
1273
+ const { locale, pathWithoutLocale, availableLocales, setLocale } =
1274
+ useLocale();
1275
+
1276
+ return (
1277
+ <div>
1278
+ <button popoverTarget="localePopover">{getLocaleName(locale)}</button>
1279
+ <div id="localePopover" popover="auto">
1280
+ {availableLocales.map((localeItem) => (
1281
+ <Link
1282
+ href={getLocalizedUrl(pathWithoutLocale, localeItem)}
1283
+ key={localeItem}
1284
+ aria-current={locale === localeItem ? "page" : undefined}
1285
+ onClick={() => setLocale(localeItem)}
1286
+ replace // سيضمن أن زر "العودة" في المتصفح يعيد التوجيه إلى الصفحة السابقة
1287
+ >
1288
+ <span>
1289
+ {/* اللغة - على سبيل المثال FR */}
1290
+ {localeItem}
1291
+ </span>
1292
+ <span>
1293
+ {/* اللغة بلغتها الخاصة - على سبيل المثال Français */}
1294
+ {getLocaleName(localeItem, locale)}
1295
+ </span>
1296
+ <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
1297
+ {/* اللغة في اللغة الحالية - على سبيل المثال Francés مع تعيين اللغة الحالية إلى Locales.SPANISH */}
1298
+ {getLocaleName(localeItem)}
1299
+ </span>
1300
+ <span dir="ltr" lang={Locales.ENGLISH}>
1301
+ {/* اللغة بالإنجليزية - على سبيل المثال French */}
1302
+ {getLocaleName(localeItem, Locales.ENGLISH)}
1303
+ </span>
1304
+ </Link>
1305
+ ))}
1306
+ </div>
1307
+ </div>
1308
+ );
1309
+ };
1310
+ ```
1311
+
1312
+ ```jsx fileName="src/components/LocaleSwitcher.csx" codeFormat="commonjs"
1313
+ "use client";
1314
+
1315
+ const {
1316
+ Locales,
1317
+ getHTMLTextDir,
1318
+ getLocaleName,
1319
+ getLocalizedUrl,
1320
+ } = require("intlayer");
1321
+ const { useLocale } = require("next-intlayer");
1322
+ const Link = require("next/link");
1323
+
1324
+ export const LocaleSwitcher = () => {
1325
+ const { locale, pathWithoutLocale, availableLocales, setLocale } =
1326
+ useLocale();
1327
+
1328
+ return (
1329
+ <div>
1330
+ <button popoverTarget="localePopover">{getLocaleName(locale)}</button>
1331
+ <div id="localePopover" popover="auto">
1332
+ {availableLocales.map((localeItem) => (
1333
+ <Link
1334
+ href={getLocalizedUrl(pathWithoutLocale, localeItem)}
1335
+ key={localeItem}
1336
+ aria-current={locale === localeItem ? "page" : undefined}
1337
+ onClick={() => setLocale(localeItem)}
1338
+ replace // سيضمن أن زر "العودة" في المتصفح يعيد التوجيه إلى الصفحة السابقة
1339
+ >
1340
+ <span>
1341
+ {/* اللغة - على سبيل المثال FR */}
1342
+ {localeItem}
1343
+ </span>
1344
+ <span>
1345
+ {/* اللغة في لغتها الأصلية - على سبيل المثال Français */}
1346
+ {getLocaleName(localeItem, locale)}
1347
+ </span>
1348
+ <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
1349
+ {/* اللغة في اللغة الحالية - على سبيل المثال Francés مع تعيين اللغة الحالية إلى Locales.SPANISH */}
1350
+ {getLocaleName(localeItem)}
1351
+ </span>
1352
+ <span dir="ltr" lang={Locales.ENGLISH}>
1353
+ {/* اللغة بالإنجليزية - على سبيل المثال French */}
1354
+ {getLocaleName(localeItem, Locales.ENGLISH)}
1355
+ </span>
1356
+ </Link>
1357
+ ))}
1358
+ </div>
1359
+ </div>
1360
+ );
1361
+ };
1362
+ ```
1363
+
1364
+ > طريقة بديلة هي استخدام دالة `setLocale` المقدمة من الخطاف `useLocale`. هذه الدالة لن تسمح بتحميل الصفحة مسبقًا. راجع [توثيق الخطاف `useLocale`](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/packages/next-intlayer/useLocale.md) لمزيد من التفاصيل.
1365
+
1366
+ > يمكنك أيضًا تعيين دالة في خيار `onLocaleChange` لتشغيل دالة مخصصة عند تغيير اللغة.
1367
+
1368
+ ```tsx fileName="src/components/LocaleSwitcher.tsx"
1369
+ "use client";
1370
+
1371
+ import { useRouter } from "next/navigation";
1372
+ import { useLocale } from "next-intlayer";
1373
+ import { getLocalizedUrl } from "intlayer";
1374
+
1375
+ // ... بقية الكود
1376
+
1377
+ const router = useRouter();
1378
+ const { setLocale } = useLocale({
1379
+ onLocaleChange: (locale) => {
1380
+ router.push(getLocalizedUrl(pathWithoutLocale, locale));
1381
+ },
1382
+ });
1383
+
1384
+ return (
1385
+ <button onClick={() => setLocale(Locales.FRENCH)}>
1386
+ التغيير إلى الفرنسية
1387
+ </button>
1388
+ );
1389
+ ```
1390
+
1391
+ > مراجع التوثيق:
1392
+ >
1393
+ > - [`useLocale` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/packages/next-intlayer/useLocale.md)
1394
+ > - [`getLocaleName` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/packages/intlayer/getLocaleName.md)
1395
+ > - [`getLocalizedUrl` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/packages/intlayer/getLocalizedUrl.md)
1396
+ > - [`getHTMLTextDir` hook](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/packages/intlayer/getHTMLTextDir.md)
1397
+ > - [`hrefLang` attribute](https://developers.google.com/search/docs/specialty/international/localized-versions?hl=fr)
1398
+ > - [`lang` attribute](https://developer.mozilla.org/ar/docs/Web/HTML/Global_attributes/lang)
1399
+ > - [`dir` attribute](https://developer.mozilla.org/ar/docs/Web/HTML/Global_attributes/dir)
1400
+ > - [`aria-current` attribute](https://developer.mozilla.org/ar/docs/Web/Accessibility/ARIA/Attributes/aria-current)
1401
+
1402
+ ### (اختياري) الخطوة 11: إنشاء مكون رابط محلي
1403
+
1404
+ لضمان أن تنقل تطبيقك يحترم اللغة الحالية، يمكنك إنشاء مكون `Link` مخصص. يقوم هذا المكون تلقائيًا بإضافة بادئة اللغة الحالية إلى عناوين URL الداخلية، بحيث عندما ينقر المستخدم الناطق بالفرنسية على رابط إلى صفحة "حول"، يتم توجيهه إلى `/fr/about` بدلاً من `/about`.
1405
+
1406
+ هذا السلوك مفيد لعدة أسباب:
1407
+
1408
+ - **تحسين محركات البحث وتجربة المستخدم**: تساعد عناوين URL المترجمة محركات البحث على فهرسة الصفحات الخاصة بكل لغة بشكل صحيح وتوفر للمستخدمين محتوى بلغتهم المفضلة.
1409
+ - **الاتساق**: باستخدام رابط مترجم في جميع أنحاء تطبيقك، تضمن أن التنقل يبقى ضمن اللغة الحالية، مما يمنع التبديلات غير المتوقعة في اللغة.
1410
+ - **قابلية الصيانة**: تركيز منطق الترجمة في مكون واحد يبسط إدارة عناوين URL، مما يجعل قاعدة الشيفرة الخاصة بك أسهل في الصيانة والتوسيع مع نمو تطبيقك.
1411
+
1412
+ فيما يلي تنفيذ لمكون `Link` محلي في TypeScript:
1413
+
1414
+ ```tsx fileName="src/components/Link.tsx" codeFormat="typescript"
1415
+ "use client";
1416
+
1417
+ import { getLocalizedUrl } from "intlayer";
1418
+ import NextLink, { type LinkProps as NextLinkProps } from "next/link";
1419
+ import { useLocale } from "next-intlayer";
1420
+ import type { PropsWithChildren, FC } from "react";
1421
+
1422
+ /**
1423
+ * دالة مساعدة للتحقق مما إذا كان عنوان URL معين خارجيًا.
1424
+ * إذا كان عنوان URL يبدأ بـ http:// أو https://، فإنه يعتبر خارجيًا.
1425
+ */
1426
+ export const checkIsExternalLink = (href?: string): boolean =>
1427
+ /^https?:\/\//.test(href ?? "");
1428
+
1429
+ /**
1430
+ * مكون رابط مخصص يقوم بتعديل خاصية href بناءً على اللغة الحالية.
1431
+ * بالنسبة للروابط الداخلية، يستخدم `getLocalizedUrl` لإضافة بادئة اللغة إلى الرابط (مثلاً: /fr/about).
1432
+ * هذا يضمن أن التنقل يبقى ضمن نفس سياق اللغة.
1433
+ */
1434
+ export const Link: FC<PropsWithChildren<NextLinkProps>> = ({
1435
+ href,
1436
+ children,
1437
+ ...props
1438
+ }) => {
1439
+ const { locale } = useLocale();
1440
+ const isExternalLink = checkIsExternalLink(href.toString());
1441
+
1442
+ // إذا كان الرابط داخليًا وتم توفير href صالح، احصل على الرابط المحلي.
1443
+ const hrefI18n: NextLinkProps["href"] =
1444
+ href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
1445
+
1446
+ return (
1447
+ <NextLink href={hrefI18n} {...props}>
1448
+ {children}
1449
+ </NextLink>
1450
+ );
1451
+ };
1452
+ ```
1453
+
1454
+ ```jsx fileName="src/components/Link.mjx" codeFormat="esm"
1455
+ "use client";
1456
+
1457
+ import { getLocalizedUrl } from "intlayer";
1458
+ import NextLink from "next/link";
1459
+ import { useLocale } from "next-intlayer";
1460
+
1461
+ /**
1462
+ * دالة مساعدة للتحقق مما إذا كان عنوان URL معين خارجيًا.
1463
+ * إذا بدأ عنوان URL بـ http:// أو https://، يعتبر خارجيًا.
1464
+ */
1465
+ export const checkIsExternalLink = (href) => /^https?:\/\//.test(href ?? "");
1466
+
1467
+ /**
1468
+ * مكون رابط مخصص يقوم بتكييف خاصية href بناءً على اللغة الحالية.
1469
+ * بالنسبة للروابط الداخلية، يستخدم `getLocalizedUrl` لإضافة بادئة اللغة إلى عنوان URL (مثلاً /fr/about).
1470
+ * هذا يضمن بقاء التنقل ضمن نفس سياق اللغة.
1471
+ */
1472
+ export const Link = ({ href, children, ...props }) => {
1473
+ const { locale } = useLocale();
1474
+ const isExternalLink = checkIsExternalLink(href.toString());
1475
+
1476
+ // إذا كان الرابط داخليًا وتم توفير href صالح، احصل على الرابط المحلي.
1477
+ const hrefI18n =
1478
+ href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
1479
+
1480
+ return (
1481
+ <NextLink href={hrefI18n} {...props}>
1482
+ {children}
1483
+ </NextLink>
1484
+ );
1485
+ };
1486
+ ```
1487
+
1488
+ ```jsx fileName="src/components/Link.csx" codeFormat="commonjs"
1489
+ "use client";
1490
+
1491
+ const { getLocalizedUrl } = require("intlayer");
1492
+ const NextLink = require("next/link");
1493
+ const { useLocale } = require("next-intlayer");
1494
+
1495
+ /**
1496
+ * دالة مساعدة للتحقق مما إذا كان الرابط المعطى خارجيًا.
1497
+ * إذا بدأ الرابط بـ http:// أو https://، يُعتبر خارجيًا.
1498
+ */
1499
+ const checkIsExternalLink = (href) => /^https?:\/\//.test(href ?? "");
1500
+
1501
+ /**
1502
+ * مكون Link مخصص يقوم بتكييف خاصية href بناءً على اللغة الحالية.
1503
+ * بالنسبة للروابط الداخلية، يستخدم `getLocalizedUrl` لإضافة بادئة اللغة إلى عنوان URL (مثلاً: /fr/about).
1504
+ * هذا يضمن أن التنقل يبقى ضمن نفس سياق اللغة.
1505
+ */
1506
+ const Link = ({ href, children, ...props }) => {
1507
+ const { locale } = useLocale();
1508
+ const isExternalLink = checkIsExternalLink(href.toString());
1509
+
1510
+ // إذا كانت الرابط داخلي وتم توفير href صالح، احصل على عنوان URL المحلي.
1511
+ const hrefI18n =
1512
+ href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
1513
+
1514
+ return (
1515
+ <NextLink href={hrefI18n} {...props}>
1516
+ {children}
1517
+ </NextLink>
1518
+ );
1519
+ };
1520
+ ```
1521
+
1522
+ #### كيف يعمل
1523
+
1524
+ - **كشف الروابط الخارجية**:
1525
+ - **تحديد الروابط الخارجية**:
1526
+ تقوم الدالة المساعدة `checkIsExternalLink` بتحديد ما إذا كان الرابط URL خارجيًا. تُترك الروابط الخارجية دون تغيير لأنها لا تحتاج إلى تعريب.
1527
+
1528
+ - **استرجاع اللغة الحالية**:
1529
+ يوفر الخطاف `useLocale` اللغة الحالية (مثلًا `fr` للفرنسية).
1530
+
1531
+ - **تعريب الرابط**:
1532
+ بالنسبة للروابط الداخلية (أي غير الخارجية)، يتم استخدام `getLocalizedUrl` لإضافة بادئة اللغة الحالية تلقائيًا إلى الرابط. هذا يعني أنه إذا كان المستخدم يستخدم اللغة الفرنسية، فإن تمرير `/about` كـ `href` سيُحوَّل إلى `/fr/about`.
1533
+
1534
+ - **إرجاع الرابط**:
1535
+ يُرجع المكون عنصر `<a>` بالرابط المعرب، مما يضمن أن التنقل يتوافق مع اللغة الحالية.
1536
+
1537
+ من خلال دمج مكون `Link` هذا في تطبيقك، تحافظ على تجربة مستخدم متماسكة وواعية للغة، مع الاستفادة أيضًا من تحسين محركات البحث وسهولة الاستخدام.
1538
+
1539
+ ### (اختياري) الخطوة 12: الحصول على اللغة الحالية في إجراءات الخادم
1540
+
1541
+ إذا كنت بحاجة إلى اللغة النشطة داخل إجراء خادم (مثلًا، لتوطين رسائل البريد الإلكتروني أو تنفيذ منطق يعتمد على اللغة)، استدعِ `getLocale` من `next-intlayer/server`:
1542
+
1543
+ ```tsx fileName="src/app/actions/getLocale.ts" codeFormat="typescript"
1544
+ "use server";
1545
+
1546
+ import { getLocale } from "next-intlayer/server";
1547
+
1548
+ export const myServerAction = async () => {
1549
+ const locale = await getLocale();
1550
+
1551
+ // قم بشيء ما باستخدام اللغة
1552
+ };
1553
+ ```
1554
+
1555
+ > تتبع دالة `getLocale` استراتيجية متدرجة لتحديد لغة المستخدم:
1556
+ >
1557
+ > 1. أولاً، يتحقق من رؤوس الطلب عن قيمة لغة قد تم تعيينها بواسطة البروكسي
1558
+ > 2. إذا لم يتم العثور على لغة في الرؤوس، يبحث عن لغة مخزنة في الكوكيز
1559
+ > 3. إذا لم يتم العثور على كوكيز، يحاول اكتشاف اللغة المفضلة للمستخدم من إعدادات المتصفح
1560
+ > 4. كخيار أخير، يعود إلى اللغة الافتراضية المكونة في التطبيق
1561
+ >
1562
+ > هذا يضمن اختيار اللغة الأنسب بناءً على السياق المتاح.
1563
+
1564
+ ### (اختياري) الخطوة 13: تحسين حجم الحزمة الخاصة بك
1565
+
1566
+ عند استخدام `next-intlayer`، يتم تضمين القواميس في الحزمة لكل صفحة بشكل افتراضي. لتحسين حجم الحزمة، يوفر Intlayer إضافة SWC اختيارية تستبدل بشكل ذكي استدعاءات `useIntlayer` باستخدام الماكروز. هذا يضمن تضمين القواميس فقط في الحزم الخاصة بالصفحات التي تستخدمها فعليًا.
1567
+
1568
+ لتمكين هذا التحسين، قم بتثبيت حزمة `@intlayer/swc`. بمجرد التثبيت، سيقوم `next-intlayer` بالكشف تلقائيًا عن الإضافة واستخدامها:
1569
+
1570
+ ```bash packageManager="npm"
1571
+ npm install @intlayer/swc --save-dev
1572
+ ```
1573
+
1574
+ ```bash packageManager="pnpm"
1575
+ pnpm add @intlayer/swc --save-dev
1576
+ ```
1577
+
1578
+ ```bash packageManager="yarn"
1579
+ yarn add @intlayer/swc --save-dev
1580
+ ```
1581
+
1582
+ > ملاحظة: هذا التحسين متاح فقط لـ Next.js 13 وما فوق.
1583
+
1584
+ > ملاحظة: هذه الحزمة غير مثبتة بشكل افتراضي لأن ملحقات SWC لا تزال تجريبية في Next.js. قد يتغير هذا في المستقبل.
1585
+
1586
+ ### مراقبة تغييرات القواميس على Turbopack
1587
+
1588
+ عند استخدام Turbopack كخادم تطوير مع أمر `next dev`، لن يتم اكتشاف تغييرات القواميس تلقائيًا بشكل افتراضي.
1589
+
1590
+ يحدث هذا القيد لأن Turbopack لا يمكنه تشغيل ملحقات webpack بالتوازي لمراقبة التغييرات في ملفات المحتوى الخاصة بك. للتغلب على هذا، ستحتاج إلى استخدام أمر `intlayer watch` لتشغيل كل من خادم التطوير ومراقب بناء Intlayer في نفس الوقت.
1591
+
1592
+ ```json5 fileName="package.json"
1593
+ {
1594
+ // ... تكوينات package.json الحالية الخاصة بك
1595
+ "scripts": {
1596
+ // ... تكوينات السكربتات الحالية الخاصة بك
1597
+ "dev": "intlayer watch --with 'next dev'",
1598
+ },
1599
+ }
1600
+ ```
1601
+
1602
+ > إذا كنت تستخدم next-intlayer@<=6.x.x، تحتاج إلى الاحتفاظ بعلم `--turbopack` لجعل تطبيق Next.js 16 يعمل بشكل صحيح مع Turbopack. نوصي باستخدام next-intlayer@>=7.x.x لتجنب هذا القيد.
1603
+
1604
+ ### تكوين TypeScript
1605
+
1606
+ يستخدم Intlayer تعزيز الوحدات (module augmentation) للاستفادة من TypeScript وجعل قاعدة الكود الخاصة بك أقوى.
1607
+
1608
+ ![الإكمال التلقائي](https://github.com/aymericzip/intlayer/blob/main/docs/assets/autocompletion.png?raw=true)
1609
+
1610
+ ![خطأ في الترجمة](https://github.com/aymericzip/intlayer/blob/main/docs/assets/translation_error.png?raw=true)
1611
+
1612
+ تأكد من أن تكوين TypeScript الخاص بك يتضمن الأنواع التي تم إنشاؤها تلقائيًا.
1613
+
1614
+ ```json5 fileName="tsconfig.json"
1615
+ {
1616
+ // ... تكوينات TypeScript الحالية الخاصة بك
1617
+ "include": [
1618
+ // ... تكوينات TypeScript الحالية الخاصة بك
1619
+ ".intlayer/**/*.ts", // Include the auto-generated types
1620
+ ],
1621
+ }
1622
+ ```
1623
+
1624
+ ### تكوين Git
1625
+
1626
+ يوصى بتجاهل الملفات التي يتم إنشاؤها بواسطة Intlayer. هذا يسمح لك بتجنب الالتزام بها في مستودع Git الخاص بك.
1627
+
1628
+ للقيام بذلك، يمكنك إضافة التعليمات التالية إلى ملف `.gitignore` الخاص بك:
1629
+
1630
+ ```plaintext fileName=".gitignore"
1631
+ # تجاهل الملفات التي تم إنشاؤها بواسطة Intlayer
1632
+ .intlayer
1633
+ ```
1634
+
1635
+ ### إضافة VS Code
1636
+
1637
+ لتحسين تجربة التطوير الخاصة بك مع Intlayer، يمكنك تثبيت **إضافة Intlayer الرسمية لـ VS Code**.
1638
+
1639
+ [التثبيت من سوق VS Code](https://marketplace.visualstudio.com/items?itemName=intlayer.intlayer-vs-code-extension)
1640
+
1641
+ تقدم هذه الإضافة:
1642
+
1643
+ - **الإكمال التلقائي** لمفاتيح الترجمة.
1644
+ - **كشف الأخطاء في الوقت الحقيقي** للترجمات المفقودة.
1645
+ - **معاينات داخلية** للمحتوى المترجم.
1646
+ - **إجراءات سريعة** لإنشاء الترجمات وتحديثها بسهولة.
1647
+
1648
+ لمزيد من التفاصيل حول كيفية استخدام الإضافة، راجع [توثيق إضافة Intlayer لـ VS Code](https://intlayer.org/doc/vs-code-extension).
1649
+
1650
+ ### التقدم أكثر
1651
+
1652
+ للتقدم أكثر، يمكنك تنفيذ [المحرر المرئي](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/intlayer_visual_editor.md) أو إخراج محتواك باستخدام [نظام إدارة المحتوى (CMS)](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ar/intlayer_CMS.md).