在 Vue 中使用 i18n 国际化

首先新建 src\locales 目录,编写两个国际化文件。

en-US.json

json 复制代码
{
  "greeting": "Hello World",
  "table": {
    "title": "Table Example",
    "description": "This is an application"
  },
  "buttons": {
    "toggleLanguage": "Toggle Language"
  },
  "currentLanguage": "Current Language"
}

zh-CN.json

json 复制代码
{
  "greeting": "你好,世界",
  "table": {
    "title": "表格示例",
    "description": "这是一个应用"
  },
  "buttons": {
    "toggleLanguage": "切换语言"
  },
  "currentLanguage": "当前语言"
}

新建 src\utils\i18n.ts 文件,编写国际化配置核心代码。

typescript 复制代码
import { match } from "@formatjs/intl-localematcher";
import { createI18n } from "vue-i18n";

export const i18n = createI18n({
  legacy: false,
  locale: "zh-CN",
});

/**
 * 支持的语言列表
 */
export const SUPPORTED_LOCALES = ["zh-CN", "en-US"];

/**
 * localStorage 中存储语言设置的键名
 */
const LOCALE_STORAGE_KEY = "locale";

/**
 * 切换全局语言
 * @param locale 目标语言区域设置,必须是支持的语言之一
 */
export const switchLanguage = async (locale: string) => {
  if (!SUPPORTED_LOCALES.includes(locale)) {
    throw new Error(`Unsupported locale: ${locale}`);
  }
  const { global } = i18n;
  switch (locale) {
    case "zh-CN":
      global.setLocaleMessage(locale, await import("../locales/zh-CN.json"));
      break;
    case "en-US":
      global.setLocaleMessage(locale, await import("../locales/en-US.json"));
      break;
  }
  global.locale.value = locale;
  const { documentElement } = document;
  documentElement.lang = locale;
  localStorage.setItem(LOCALE_STORAGE_KEY, locale);
};

/**
 * 根据浏览器语言设置切换到默认语言
 * 使用 @formatjs/intl-localematcher 进行标准化的语言匹配
 * 优先级:localStorage > 浏览器首选语言 > 浏览器其他语言 > 默认中文
 * 支持常见语言变体的智能匹配(如 zh-Hans → zh-CN,en-GB → en-US)
 */
export const switchDefaultLanguage = async () => {
  const languages = [...navigator.languages];
  const saved = localStorage.getItem(LOCALE_STORAGE_KEY);
  if (saved) languages.unshift(saved);
  const matched = match(languages, SUPPORTED_LOCALES, "zh-CN");
  await switchLanguage(matched);
};

随后即可在 src\main.ts 中使用 switchDefaultLanguage 函数进行初始语言设置。如果用户浏览器默认语言环境是英文,那么初次进入时即切换到英文。

如果使用了 element-plus,那么还需要根据语言环境加载组件库的语言包。

html 复制代码
<script setup lang="ts">
import { watchImmediate } from "@vueuse/core";
import dayjs from "dayjs";
import type { Language } from "element-plus/es/locale";
import elLocaleEn from "element-plus/es/locale/lang/en";
import { useI18n } from "vue-i18n";

const { locale } = useI18n();

const elLocale = ref<Language>();
const setElLocale = async (data: Promise<{ default: Language }>) => {
  elLocale.value = (await data).default;
};

watchImmediate(locale, async (locale) => {
  switch (locale) {
    case "zh-CN":
      await import("dayjs/locale/zh-cn");
      dayjs.locale("zh-cn");
      setElLocale(import("element-plus/es/locale/lang/zh-cn"));
      break;
    case "en-US":
      await import("dayjs/locale/en");
      dayjs.locale("en");
      elLocale.value = elLocaleEn;
      break;
  }
});
</script>

<template>
  <ElConfigProvider :locale="elLocale">
    <RouterView />
  </ElConfigProvider>
</template>
相关推荐
Lan.W35 分钟前
vue3-element-admin里新增mock接口一直没有生成,不生效
前端·javascript·vue.js·mock
仙古.梦回~39 分钟前
vue-skills
前端·javascript·vue.js
清灵xmf1 小时前
JS 原生深拷贝的终极方案——structuredClone
前端·javascript·vue.js·json.stringify·structuredclone
前端 贾公子1 小时前
响应式系统基础:依赖追踪的基础 —— 发布订阅模式(前端应用最广的设计模式)上
javascript·vue.js
M ? A2 小时前
VuReact:Vue转React的增量编译利器
前端·vue.js·后端·react.js·面试·开源·vureact
前端那点事3 小时前
Vue3 新趋势:10个高阶实用操作|性能优化+开发提效+避坑指南
前端·vue.js
前端那点事3 小时前
90%前端只会皮毛!JSON.parse/stringify高阶用法、数据规则、避坑终极指南
前端·vue.js
需要坚持的人3 小时前
让 SVG 不再“丢字变形”:一次思维导图导出文字转 Path 的实战优化
前端·vue.js·svg
蜡台3 小时前
idea 配置 vue 运行命令时, scripts 一栏始终为空
前端·vue.js·intellij-idea
ZC跨境爬虫4 小时前
跟着 MDN 学 HTML day_43:(DocumentFragment 接口详解)
前端·javascript·vue.js·ui·html·音视频