Next.js Internationalization (i18n) with next-intl و Intlayer
كلا من next-intl و Intlayer هما إطاران مفتوحا المصدر للتعريب (i18n) مصممان لتطبيقات Next.js. يتم استخدامهما بشكل واسع لإدارة الترجمات، والمحلية، وتبديل اللغات في مشاريع البرمجيات.
يتشاركان ثلاث مفاهيم رئيسية:
إعلان المحتوى: الطريقة لتحديد المحتوى القابل للترجمة في تطبيقك.
- يُسمى ملف إعلان المحتوى في Intlayer، والذي يمكن أن يكون ملف JSON أو JS أو TS يقوم بتصدير البيانات المنظمة. راجع وثائق Intlayer لمزيد من المعلومات.
- يُسمى الرسائل أو رسائل اللغة في next-intl، عادةً في ملفات JSON. راجع وثائق next-intl لمزيد من المعلومات.
الأدوات: أدوات لبناء وتفسير إعلانات المحتوى في التطبيق، مثل useIntlayer() أو useLocale() لـ Intlayer، و useTranslations() لـ next-intl.
الإضافات والبرمجيات الوسيطة: ميزات لإدارة إعادة توجيه عنوان URL، وتحسين التجميع، وأكثر—على سبيل المثال، intlayerMiddleware لـ Intlayer أو createMiddleware لـ next-intl.
Intlayer مقابل next-intl: الاختلافات الرئيسية
لتحليل أعمق حول كيفية مقارنة Intlayer بمكتبات i18n الأخرى لـ Next.js (مثل next-intl)، تحقق من منشور المدونة next-i18next vs. next-intl vs. Intlayer.
كيفية إنشاء رسائل next-intl مع Intlayer
لماذا تستخدم Intlayer مع next-intl؟
تقدم ملفات إعلان المحتوى في Intlayer بشكل عام تجربة أفضل للمطورين. إنها أكثر مرونة وقابلة للصيانة بسبب ميزتين رئيسيتين:
مكان مرن: يمكنك وضع ملف إعلان المحتوى في Intlayer في أي مكان في شجرة ملفات تطبيقك. يجعل ذلك من السهل إعادة تسمية أو حذف المكونات دون ترك ملفات رسائل غير مستعملة أو معلقة.
هياكل الملفات النموذجية:
bash.└── src └── components └── MyComponent ├── index.content.ts # ملف إعلان المحتوى └── index.tsx
ترجمات مركزة: تُخزن Intlayer جميع الترجمات في إعلان محتوى واحد، مما يضمن عدم فقدان أي ترجمة. في مشاريع TypeScript، تُعلم الترجمات المفقودة تلقائياً كأخطاء نوع، مما يوفر تغذية راجعة فورية للمطورين.
التثبيت
لاستخدام Intlayer و next-intl معًا، قم بتثبيت المكتبتين:
npm install intlayer next-intl
تكوين Intlayer لتصدير رسائل next-intl
ملاحظة: يمكن أن يؤدي تصدير الرسائل من Intlayer لـ next-intl إلى إدخال اختلافات طفيفة في الهيكل. إذا كان ذلك ممكنًا، احتفظ بتدفق يعمل بIntlayer فقط أو next-intl فقط لتبسيط التكامل. إذا كنت بحاجة إلى إنشاء رسائل next-intl من Intlayer، اتبع الخطوات أدناه.
قم بإنشاء أو تحديث ملف intlayer.config.ts (أو .mjs / .cjs) في جذر مشروعك:
import { Locales, type IntlayerConfig } from "intlayer";const config: IntlayerConfig = { internationalization: { locales: [Locales.ENGLISH, Locales.FRENCH, Locales.SPANISH], defaultLocale: Locales.ENGLISH, }, content: { dictionaryOutput: ["next-intl"], // استخدم ناتج next-intl nextIntlMessagesDir: "./intl/messages", // حيث يتم حفظ رسائل next-intl },};export default config;
إعلان المحتوى
فيما يلي أمثلة على ملفات إعلان المحتوى بعدة تنسيقات. ستقوم Intlayer بتجميع هذه الملفات في ملفات رسائل يمكن أن تستهلكها next-intl.
import { t, type DeclarationContent } from "intlayer";const content = { key: "my-component", content: { helloWorld: t({ en: "Hello World", fr: "Bonjour le monde", es: "Hola Mundo", }), },} satisfies DeclarationContent;export default content;
بناء رسائل next-intl
لبناء ملفات الرسائل لـ next-intl، قم بتشغيل:
npx intlayer build
سيقوم هذا بإنشاء الموارد في دليل ./intl/messages (كما هو مُكون في intlayer.config.*). الناتج المتوقع:
.└── intl └── messages └── ar └── my-content.json └── fr └── my-content.json └── es └── my-content.json
تتضمن كل ملف رسائل مجمعة من جميع إعلانات محتوى Intlayer. عادةً ما تتطابق المفاتيح في المستوى الأعلى مع حقول content.key الخاصة بك.
استخدام next-intl في تطبيق Next.js الخاص بك
لمزيد من التفاصيل، راجع وثائق استخدام next-intl الرسمية.
إنشاء برمجية وسيطة (اختياري):
إذا كنت ترغب في إدارة الكشف التلقائي عن اللغة أو إعادة التوجيه، استخدم createMiddleware لـ next-intl.middleware.tsimport createMiddleware from "next-intl/middleware";import { NextResponse } from "next/server";export default createMiddleware({ locales: ["en", "fr", "es"], defaultLocale: "en",});export const config = { matcher: ["/((?!api|_next|.*\\..*).*)"],};
إنشاء layout.tsx أو _app.tsx لتحميل الرسائل:
إذا كنت تستخدم App Router (Next.js 13+)، أنشئ تخطيطًا:app/[locale]/layout.tsximport { NextIntlClientProvider } from 'next-intl';import { notFound } from 'next/navigation';import React, { ReactNode } from 'react';export const dynamic = 'force-dynamic';export default async function RootLayout({ children, params}: { children: ReactNode; params: { locale: string };}) { let messages; try { messages = (await import(`../../intl/messages/${params.locale}.json`)).default; } catch (error) { notFound(); } return ( <html lang={params.locale}> <body> <NextIntlClientProvider locale={params.locale} messages={messages}> {children} </NextIntlClientProvider> </body> </html> );}
إذا كنت تستخدم Pages Router (Next.js 12 أو أقل)، قم بتحميل الرسائل في _app.tsx:
pages/_app.tsximport type { AppProps } from 'next/app';import { NextIntlProvider } from 'next-intl';function MyApp({ Component, pageProps }: AppProps) { return ( <NextIntlProvider locale={pageProps.locale} messages={pageProps.messages}> <Component {...pageProps} /> </NextIntlProvider> );}export default MyApp;
جلب الرسائل من جانب الخادم (مثال Pages Router):
pages/index.tsximport { GetServerSideProps } from "next";import HomePage from "../components/HomePage";export default HomePage;export const getServerSideProps: GetServerSideProps = async ({ locale }) => { const messages = (await import(`../intl/messages/${locale}.json`)).default; return { props: { locale, messages, }, };};
استخدام المحتوى في مكونات Next.js
بمجرد تحميل الرسائل في next-intl، يمكنك استخدامها في مكوناتك عبر useTranslations():
import type { FC } from "react";import { useTranslations } from 'next-intl';const MyComponent: FC = () => { const t = useTranslations('my-component'); // 'my-component' تتوافق مع مفتاح المحتوى في Intlayer return ( <div> <h1>{t('helloWorld')}</h1> </div> );};export default MyComponent;
هذا كل شيء! كلما قمت بتحديث أو إضافة ملفات إعلان محتوى جديدة من Intlayer، أعد تشغيل أمر intlayer build لإعادة توليد رسائل JSON الخاصة بك لـ next-intl. ستقوم next-intl بالتقاط المحتوى المحدث تلقائياً.
إذا كان لديك فكرة لتحسين هذه الوثيقة، فلا تتردد في المساهمة من خلال تقديم طلب سحب على GitHub.
رابط GitHub للمدونة