深色模式
国际化(i18n)不只是翻译文本那么简单。日期格式、数字格式、复数规则、文本方向(RTL),这些都需要一套完整的解决方案。vue-i18n 是 Vue 生态中最成熟的国际化库,配合 Composition API 可以做到优雅且类型安全的多语言支持。
基础集成
安装和初始化 vue-i18n。
javascript
// i18n/index.js
import { createI18n } from 'vue-i18n'
import zhCN from './locales/zh-CN'
import enUS from './locales/en-US'
const i18n = createI18n({
legacy: false, // 启用 Composition API 模式
locale: 'zh-CN', // 默认语言
fallbackLocale: 'en-US', // 回退语言
messages: {
'zh-CN': zhCN,
'en-US': enUS
}
})
export default i18n
// main.js
import { createApp } from 'vue'
import App from './App.vue'
import i18n from './i18n'
const app = createApp(App)
app.use(i18n)
app.mount('#app')翻译文件结构
按功能模块组织翻译文件,避免单文件过大。
javascript
// i18n/locales/zh-CN.js
export default {
common: {
confirm: '确定',
cancel: '取消',
save: '保存',
delete: '删除',
search: '搜索',
loading: '加载中...',
noData: '暂无数据'
},
user: {
profile: '个人资料',
settings: '设置',
logout: '退出登录',
welcomeBack: '欢迎回来,{name}',
unreadMessages: '{count} 条未读消息'
},
validation: {
required: '{field}不能为空',
minLength: '{field}最少{min}个字符',
maxLength: '{field}最多{max}个字符',
email: '请输入有效的邮箱地址'
}
}javascript
// i18n/locales/en-US.js
export default {
common: {
confirm: 'Confirm',
cancel: 'Cancel',
save: 'Save',
delete: 'Delete',
search: 'Search',
loading: 'Loading...',
noData: 'No data'
},
user: {
profile: 'Profile',
settings: 'Settings',
logout: 'Logout',
welcomeBack: 'Welcome back, {name}',
unreadMessages: '{count} unread messages'
},
validation: {
required: '{field} is required',
minLength: '{field} must be at least {min} characters',
maxLength: '{field} must be at most {max} characters',
email: 'Please enter a valid email address'
}
}Composition API 用法
在 <script> 中使用 useI18n 钩子。
vue
{% raw %}
<template>
<div class="user-page">
<!-- 基础翻译 -->
<h1>{{ t('user.profile') }}</h1>
<!-- 带参数的翻译 -->
<p>{{ t('user.welcomeBack', { name: username }) }}</p>
<!-- 复数处理 -->
<p>{{ t('user.unreadMessages', { count: messageCount }) }}</p>
<!-- 语言切换 -->
<select v-model="locale">
<option value="zh-CN">中文</option>
<option value="en-US">English</option>
</select>
<button @click="logout">{{ t('user.logout') }}</button>
</div>
</template>
<script>
import { ref } from 'vue'
import { useI18n } from 'vue-i18n'
const { t, locale, d, n } = useI18n()
const username = ref('张三')
const messageCount = ref(5)
const logout = () => {
// 退出逻辑
}
</script>
{% endraw %}日期和数字格式化
vue-i18n 支持基于 locale 的日期和数字格式化,统一处理本地化差异。
javascript
// i18n/index.js
const i18n = createI18n({
legacy: false,
locale: 'zh-CN',
fallbackLocale: 'en-US',
messages: { 'zh-CN': zhCN, 'en-US': enUS },
// 日期格式化预设
datetimeFormats: {
'zh-CN': {
short: { year: 'numeric', month: '2-digit', day: '2-digit' },
long: {
year: 'numeric', month: 'long', day: '2-digit',
hour: '2-digit', minute: '2-digit'
}
},
'en-US': {
short: { year: 'numeric', month: 'short', day: 'numeric' },
long: {
year: 'numeric', month: 'long', day: 'numeric',
hour: '2-digit', minute: '2-digit'
}
}
},
// 数字格式化预设
numberFormats: {
'zh-CN': {
currency: { style: 'currency', currency: 'CNY' },
percent: { style: 'percent', minimumFractionDigits: 2 }
},
'en-US': {
currency: { style: 'currency', currency: 'USD' },
percent: { style: 'percent', minimumFractionDigits: 2 }
}
}
})vue
{% raw %}
<template>
<div>
<!-- 日期格式化 -->
<p>短日期: {{ d(new Date(), 'short') }}</p>
<p>长日期: {{ d(new Date(), 'long') }}</p>
<!-- 数字格式化 -->
<p>价格: {{ n(9999, 'currency') }}</p>
<p>比例: {{ n(0.856, 'percent') }}</p>
</div>
</template>
<script>
import { useI18n } from 'vue-i18n'
const { t, d, n } = useI18n()
</script>
{% endraw %}按需加载语言包
大型项目的翻译文件可能很大,推荐按语言懒加载。
javascript
// i18n/index.js
import { createI18n } from 'vue-i18n'
const i18n = createI18n({
legacy: false,
locale: localStorage.getItem('locale') || 'zh-CN',
fallbackLocale: 'en-US',
messages: {} // 初始为空
})
// 动态加载语言包
const loadedLanguages = []
async function loadLocaleMessages(locale) {
if (loadedLanguages.includes(locale)) {
i18n.global.locale.value = locale
return
}
const messages = await import(
/* webpackChunkName: "locale-[request]" */
`./locales/${locale}.js`
)
i18n.global.setLocaleMessage(locale, messages.default)
loadedLanguages.push(locale)
i18n.global.locale.value = locale
}
export { i18n, loadLocaleMessages }
// 切换语言时调用
// await loadLocaleMessages('en-US')
// localStorage.setItem('locale', 'en-US')小结
legacy: false启用 Composition API 模式,用useI18n()获取t、d、n方法- 翻译文件按功能模块组织,避免单文件膨胀
d()和n()分别处理日期和数字的本地化格式化- 按需加载语言包,减少首屏 bundle 体积
- 语言切换后记得同步
localStorage,保持用户偏好