前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Next.js 实战 (四):i18n 国际化的最优方案实践

Next.js 实战 (四):i18n 国际化的最优方案实践

原创
作者头像
白雾茫茫丶
发布2024-12-11 09:30:20
发布2024-12-11 09:30:20
8671
举报

前言

有关 Next.js 国际化的方案网上很多,而且各部相同,但大部分的方案都是在 /app 目录下添加动态路由 [lang] 这样的形式,这不是我想要的效果。

我希望国际化的实现不能破坏应用程序的目录结构和路由,在经过一段时间摸索后,发现 next-intl

有提供现成的方案:

更多详细文档:next-intl

如果官方文档打不开的伙伴,可以到 Github 上克隆代码,本地运行根目录的 docs 文件夹

具体步骤

1、 安装依赖

代码语言:powershell
复制
pnpm add next-intl

2、 根目录新建 messages 文件夹,并写入对应的国际化文件:

代码语言:json
复制
// en.json
{
  "Route":{
    "about":"About",
    "dashboard":"Dashboard",
    "system-manage":"System Manage",
    "internationalization":"Internationalization"
  }
}

// zh.json
{
  "Route":{
    "about":"关于",
    "dashboard":"仪表盘",
    "system-manage":"系统管理",
    "internationalization":"国际化"
  }
}

3、 根目录的 next.config.ts 文件设置插件:

代码语言:ts
复制
import type { NextConfig } from "next";
import createNextIntlPlugin from 'next-intl/plugin';

const withNextIntl = createNextIntlPlugin();

const nextConfig: NextConfig = {};

export default withNextIntl(nextConfig);

4、 新建 src/i18n/config.ts 文件,写入配置:

代码语言:ts
复制
export type Locale = (typeof locales)[number];

export const locales = ['zh', 'en'] as const;
export const defaultLocale: Locale = 'zh';

5、 新建 src/i18n/request.ts 文件,创建一个请求范围的配置对象:

代码语言:ts
复制
import { getRequestConfig } from 'next-intl/server';

import { getLocale } from '@/i18n';

export default getRequestConfig(async () => {
  const locale = await getLocale();

  return {
    locale,
    messages: (await import(`../../messages/${locale}.json`)).default,
  };
});

6、 新建 src/i18n/index.ts 文件,用于服务端获取和设置语言

代码语言:ts
复制
'use server';

import { cookies } from 'next/headers';

import { defaultLocale, Locale } from '@/i18n/config';

// In this example the locale is read from a cookie. You could alternatively
// also read it from a database, backend service, or any other source.
const COOKIE_NAME = 'NEXT_LOCALE';

export async function getLocale() {
  return (await cookies()).get(COOKIE_NAME)?.value || defaultLocale;
}

export async function setLocale(locale: Locale) {
  (await cookies()).set(COOKIE_NAME, locale);
}

7、 app/layout.tsx 文件配置 NextIntlClientProvider

代码语言:ts
复制
import {NextIntlClientProvider} from 'next-intl';
import {getLocale, getMessages} from 'next-intl/server';
 
export default async function RootLayout({
  children
}: {
  children: React.ReactNode;
}) {
  const locale = await getLocale();
 
  // Providing all messages to the client
  // side is the easiest way to get started
  const messages = await getMessages();
 
  return (
    <html lang={locale}>
      <body>
        <NextIntlClientProvider messages={messages}>
          {children}
        </NextIntlClientProvider>
      </body>
    </html>
  );
}

8、 在文件中使用:

代码语言:ts
复制
import { useTranslations } from 'next-intl';
export default function Dashboard() {
  const t = useTranslations('Route');
  return (
    <h1>
      {t('dashboard')}
    </h1>
  );
}

切换语言

1、 新建 src/components/LangSwitch/index.tsx 文件:

代码语言:html
复制
'use client';

import { useLocale } from 'next-intl';

import { Button } from '@/components/ui/button';
import { setLocale } from '@/i18n';
import { type Locale, locales } from '@/i18n/config';

export default function LangSwitch() {
  const [ZH, EN] = locales;
  const locale = useLocale();

  // 切换语言
  function onChangeLang(value: Locale) {
    const locale = value as Locale;
    setLocale(locale);
  }
  return (
    <Button variant="ghost" size="icon" onClick={() => onChangeLang(locale === ZH ? EN : ZH)}>
      {locale === ZH ? '中' : 'EN'}
      <span className="sr-only">Toggle Lang</span>
    </Button>
  );
}

2、 在需要的位置引入组件:

代码语言:html
复制
import LangSwitch from '@/components/LangSwitch';

<LangSwitch />

最终效果

总结

这样的国际化方案切换语言的时候,路由就不会发生变化,更好地保留应用程序的原样,并且将当前语言的 key 存储到浏览器 cookie 中,刷新浏览器当前语言并不会失效,可以达到我们想要的效果。

Github 仓库next-admin

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 具体步骤
  • 切换语言
  • 最终效果
  • 总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档