0.9.5-beta • Published 5 months ago
translate-compiler v0.9.5-beta
translate-compiler
介绍
Vue 国际化 i18n 编译器,自动提取项目(.vue)文件中的”中文“并替换为多语言($t(key)),并导出全部翻译到指定文件。
安装
npm install translate-compiler --save-dev [-g]
命令行参数
translate <env> options
Arguments:
env 指定环境:dev/test/prod
Options:
-d, --dir <dir> 翻译的文件夹
-l, --locals <value...> 需要生成的地区语言,参考下方常见语种列表
-t, --translate <value...> 需要翻译的语言,对应locals语种代码
-c, --channel <value> 翻译渠道 baidu百度,youdao有道
--appKey <value> 翻译api 应用key
--secretKey <value> 翻译api 密钥
-s, --suffix <value> 翻译的文件名后缀,默认:.vue
-e, --excludes <value...> 排除翻译的文件,可指定多个
-o, --output <file> 翻译输出文件写入的文件(夹)路径,默认:locals/local.js
--strict 是否严格模式, 严格模式只提取$t()内中文,否则提取全局
env
参数详解:- dev 仅仅导出翻译文件。
- test 为每个翻译文件生成后缀前加
.lang
的新文件,以便查看翻译后对照,并导出翻译文件。 - prod 翻译 key 直接替换原文件,用于生产打包部署前命令,并导出翻译文件。
自动翻译:
- 设置了
channel
、appKey
、secretKey
、translate
(locals
参数含有该语言 key),才会执行翻译,默认填充中文。 - 原翻译文件存在的 key 不会再做翻译,如需翻译可以删改该 key 或直接删除翻译文件重新生成。
- 设置了
其他备注:
test
生成的文件后缀前加.lang
的新文件,如需删除可执行dev
/prod
。- 指定
locals
参数将在输出output
文件(夹)的同时也会输出对应语种的文件,如en
。 output
支持输出到文件或文件夹,文件支持.js
和.json
格式,指定文件夹则默认文件名为local.js
- 指定
locals
生成的语种文件中已做过翻译时,新生成的翻译不会覆盖原有已经翻译过的 key,output
指定的文件则会覆盖。 - 生成的翻译 key 默认为
文件名_[处理过的中文拼音]
,长度 < 50,中文修改后 key 会改变,这点需要注意
常见语种列表
名称 | 百度翻译代码 | 有道翻译代码 | (Element Plus) | 本项目中使用 |
---|---|---|---|---|
简体中文 | zh | zh-CHS | zh-cn | zh |
英语 | en | en | en | en |
繁体中文 | cht | zh-CHT | zh-tw | zh-tw |
日语 | jp | ja | ja | ja |
韩语 | kor | ko | ko | ko |
德语 | de | de | de | de |
法语 | fra | fr | fr | fr |
自定义翻译字典
在根目录创建translate.json
文件,添加自定义翻译对照,内包含的中文则会使用指定的 key 作为键,适用于项目高频率出现的中文,防止生成过多重复多语言(不同文件相同中文会生成不同的 key),如下:
{
"com_yes": "是",
"com_no": "否",
"com_success": "成功",
"com_fail": "失败",
"com_enable": "启用",
"com_disable": "禁用",
"com_confirm": "确定",
"com_cancel": "取消",
"com_add": "新增",
"com_edit": "编辑",
"com_modify": "修改",
"com_delete": "删除",
"com_save": "保存",
"com_handle": "操作",
"com_add_success": "新增成功",
"com_edit_success": "编辑成功",
"com_modify_success": "修改成功",
"com_delete_success": "删除成功",
"com_save_success": "保存成功",
"com_handle_success": "操作成功",
"com_query": "查询",
"com_close": "关闭",
"com_reset": "重置",
"com_back": "返回",
"com_reminder": "提示",
"com_copy": "拷贝",
"com_export": "导出",
"com_detail": "查看",
"com_details": "详情",
"com_lookup": "查找",
"com_set": "设置",
"com_all": "全部",
"com_sel_all": "全选",
"com_clear": "清空",
"com_finish": "完成",
"com_no_data": "暂无数据",
"com_more_action": "更多操作",
"com_please_select": "请选择",
"com_please_input": "请输入",
"com_required": "必填项",
"com_prev_step": "上一步",
"com_next_step": "上一步",
"com_rules_length_max": "长度最长{0}个字符",
"com_rules_length_range": "长度在{0}到{1}个字符"
}
执行翻译
全局安装
- 帮助文档
translate --help
- 执行翻译
translate dev --dir src --excludes app.vue -o locals/local.json -l en cn --strict
本地安装
- 在
package.json
中添加
"scripts": {
"translate": "translate dev --dir src",
"translate:test": "translate test --dir src -l cn en --strict",
"translate:prod": "translate prod --dir src -l cn en"
},
示例
test.vue
<template>
<div>
<!-- 严格模式 -->
<div>
<p>{{ $t("使用") }}</p>
<p>{{ isOk ? $t("确定") : $t("取消") }}</p>
<p>{{ $t("清洁时长(min)") }}</p>
<p>{{ $t("密码8-16位,必须包含数字、符号、字母任意两种") }}</p>
</div>
<div>
<el-form-item :label="$t('用户名')"></el-form-item>
<el-input :placeholder="$t('请输入')"></el-input>
</div>
<div>
<p>{{ $t(`最多上传${limit}个文件`) }}</p>
<p>{{ $t(`最多上传${limit ? "0" : "1个"}个文件`) }}</p>
<p>{{ $t(`您确定要删除${user}的${num}个文件吗?`) }}</p>
<el-form-item :label="$t(`${platform}账号:`)"></el-form-item>
<el-form-item
:label="$t(`${platform ? '谷歌' : '微信'}账号:`)"
></el-form-item>
</div>
<!-- 非严格模式 -->
<div>
<p>基础信息</p>
<p>新增成功</p>
<p>长度要在7~13个字符</p>
<p>机身号、项目名称or设备名称</p>
<p>测试文本{{ userName }}</p>
<p>开始{{ isOk ? "篮球" : "兵乒球" }}比赛</p>
</div>
<div>
<el-form-item label="持卡人"></el-form-item>
<el-input per-label="权限" placeholder="请输入"></el-input>
</div>
<div>
<p>{{ "使用记录" }}</p>
<p>{{ isOk ? "使用记录" : "历史记录" }}</p>
<p>{{ isOk ? "开始" : `结束` }}</p>
<p>{{ `员工姓名` }}</p>
<p>{{ `长度最长${max}个字符` }}</p>
<p>{{ `确定要删除${type ? "自动清洗" : "手动清洗"}路线吗` }}</p>
<p>{{ `您确定要选择${type}路线${state}吗` }}</p>
<el-form-item :label="`${area}负责人`"></el-form-item>
</div>
</div>
</template>
<script>
export default {
computed: {
weekday() {
return this.$t("星期日");
},
timeDay() {
return "早上好";
},
},
methods: {
onTable: function () {
const test11 = this.$t("初始日期");
const test22 = this.$t("kpi结束时间");
const test33 = this.$t(`kpi结束${this.label}`);
const test44 = this.$t(`标签名称${this.label == "today" ? "王" : "二"}`);
},
onExport: function () {
const t1 = "转账信息";
const t2 = "pary时间";
const t3 = `标签名称${this.timeDay == "today" ? "李" : "林"}`;
const t4 = `这是(${this.user})的账号`;
const t5 = `${this.timeDay},${this.user}先生`;
},
},
};
</script>
执行结果
<template>
<div>
<div>
<p>{{ $t("test_shi_yong") }}</p>
<p>{{ isOk ? $t("com_confirm") : $t("com_cancel") }}</p>
<p>{{ $t("test_qjscmin") }}</p>
<p>{{ $t("test_mm816wbxbhszfhzmrylz") }}</p>
</div>
<div>
<el-form-item :label="$t('test_yong_hu_ming')"></el-form-item>
<el-input :placeholder="$t('com_please_input')"></el-input>
</div>
<div>
<p>{{ $t("test_zdscogwj", [limit]) }}</p>
<p>{{ $t("test_zdscogwj", [limit ? "0" : $t("test_1ge")]) }}</p>
<p>{{ $t("test_nqdyscodogwjm", [user, num]) }}</p>
<el-form-item :label="$t('test_ozhang_hao', [platform])"></el-form-item>
<el-form-item
:label="
$t('test_ozhang_hao', [
platform ? $t('test_gu_ge') : $t('test_wei_xin'),
])
"
></el-form-item>
</div>
<div>
<p>{{ $t("test_ji_chu_xin_xi") }}</p>
<p>{{ $t("com_add_success") }}</p>
<p>{{ $t("test_cdyz713gzf") }}</p>
<p>{{ $t("test_jshxmmcorsbmc") }}</p>
<p>{{ $t("test_ce_shi_wen_ben") }}{{ userName }}</p>
<p>
{{ $t("test_kai_shi")
}}{{ isOk ? $t("test_lan_qiu") : $t("test_bing_ping_qiu")
}}{{ $t("test_bi_sai") }}
</p>
</div>
<div>
<el-form-item :label="$t('test_chi_ka_ren')"></el-form-item>
<el-input
:per-label="$t('test_quan_xian')"
:placeholder="$t('com_please_input')"
></el-input>
</div>
<div>
<p>{{ $t("test_shi_yong_ji_lu") }}</p>
<p>{{ isOk ? $t("test_shi_yong_ji_lu") : $t("test_li_shi_ji_lu") }}</p>
<p>{{ isOk ? $t("test_kai_shi") : $t("test_jie_shu") }}</p>
<p>{{ $t("test_yuan_gong_xing_ming") }}</p>
<p>{{ $t("com_rules_length_max", [max]) }}</p>
<p>
{{
$t("test_qdyscolxm", [
type ? $t("test_zi_dong_qing_xi") : $t("test_shou_dong_qing_xi"),
])
}}
</p>
<p>{{ $t("test_nqdyxzolxom", [type, state]) }}</p>
<el-form-item :label="$t('test_ofu_ze_ren', [area])"></el-form-item>
</div>
</div>
</template>
<script>
export default {
computed: {
weekday() {
return this.$t("test_xing_qi_ri");
},
timeDay() {
return this.$t("test_zao_shang_hao");
},
},
methods: {
onTable: function () {
const test11 = this.$t("test_chu_shi_ri_qi");
const test22 = this.$t("test_kpijie_shu_shi_jian");
const test33 = this.$t("test_kpijie_shuo", [this.label]);
const test44 = this.$t("test_bqmco", [
this.label == "today" ? this.$t("test_wang") : this.$t("test_er"),
]);
},
onExport: function () {
const t1 = this.$t("test_zhuan_zhang_xin_xi");
const t2 = this.$t("test_paryshi_jian");
const t3 = this.$t("test_bqmco", [
this.timeDay == "today" ? this.$t("test_li") : this.$t("test_lin"),
]);
const t4 = this.$t("test_zhe_shiode_zhang_hao", [this.user]);
const t5 = this.$t("test_ooxian_sheng", [this.timeDay, this.user]);
},
},
};
</script>
翻译文件 (locals./local.js)
exports.__esModule = true;
exports.default = {
"test_shi_yong": "使用",
"com_confirm": "确定",
"com_cancel": "取消",
"test_qjscmin": "清洁时长(min)",
"test_mm816wbxbhszfhzmrylz": "密码8-16位,必须包含数字、符号、字母任意两种",
"test_yong_hu_ming": "用户名",
"com_please_input": "请输入",
"test_xing_qi_ri": "星期日",
"test_chu_shi_ri_qi": "初始日期",
"test_kpijie_shu_shi_jian": "kpi结束时间",
"test_zdscogwj": "最多上传{0}个文件",
"test_1ge": "1个",
"test_nqdyscodogwjm": "您确定要删除{0}的{1}个文件吗?",
"test_ozhang_hao": "{0}账号:",
"test_gu_ge": "谷歌",
"test_wei_xin": "微信",
"test_jie_shu": "结束",
"test_yuan_gong_xing_ming": "员工姓名",
"test_zi_dong_qing_xi": "自动清洗",
"test_shou_dong_qing_xi": "手动清洗",
"test_qdyscolxm": "确定要删除{0}路线吗",
"test_nqdyxzolxom": "您确定要选择{0}路线{1}吗",
"test_ofu_ze_ren": "{0}负责人",
"test_ji_chu_xin_xi": "基础信息",
"com_add_success": "新增成功",
"test_cdyz713gzf": "长度要在7~13个字符",
"test_jshxmmcorsbmc": "机身号、项目名称or设备名称",
"test_ce_shi_wen_ben": "测试文本",
"test_kai_shi": "开始",
"test_bi_sai": "比赛",
"test_lan_qiu": "篮球",
"test_bing_ping_qiu": "兵乒球",
"test_shi_yong_ji_lu": "使用记录",
"test_li_shi_ji_lu": "历史记录",
"test_chi_ka_ren": "持卡人",
"test_quan_xian": "权限",
"test_kpijie_shuo": "kpi结束{0}",
"test_wang": "王",
"test_er": "二",
"test_bqmco": "标签名称{0}",
"test_li": "李",
"test_lin": "林",
"test_zhe_shiode_zhang_hao": "这是({0})的账号",
"test_ooxian_sheng": "{0},{1}先生",
"test_zao_shang_hao": "早上好",
"test_zhuan_zhang_xin_xi": "转账信息",
"test_paryshi_jian": "pary时间"
}
项目中使用
import zhLocale from './locals/zh';
import enLocale from './locals/en';
import jaLocale from './locals/ja';
// ...
Vue.use(VueI18n);
const i18n = new VueI18n({
locale: 'CN',
silentTranslationWarn: true,
messages: {
ZH: { ...elementZhLocale, ...zhLocale },
EN: { ...elementEnLocale, ...enLocale },
JA: { ...elementJaLocale, ...jaLocale },
},
});
// ...
new Vue({
router,
store,
i18n,
render: h => h(App),
}).$mount('#app');
多语言切换
// 方案一:本地切换,对接口层含翻译的(字典,表格列等)做统一事件触发更新
this.$i18n.locale = item.value;
// 方案二:页面强制刷新,适用于有较多接口层翻译的情况
Cookies.set('langCode', item.value, { path: '/' }); // VueI18n初始化取的地方
this.$router.go(0); //强制刷新
0.8.5-beta
5 months ago
0.9.2-beta
5 months ago
0.8.2-beta
5 months ago
0.9.5-beta
5 months ago
0.8.8-beta
5 months ago
0.8.6-beta
5 months ago
0.9.0-beta
5 months ago
0.9.3-beta
5 months ago
0.8.3-beta
5 months ago
0.8.7-beta
5 months ago
0.9.4-beta
5 months ago
0.9.1-beta
5 months ago
0.2.6-beta
5 months ago
0.8.1-beta
5 months ago
0.8.0-beta
5 months ago
0.7.0-beta
5 months ago
0.6.0-beta
5 months ago
0.2.5-beta
5 months ago
0.2.4-beta
5 months ago
0.2.3-beta
5 months ago
0.2.2-beta
5 months ago
0.2.1-beta
5 months ago
0.2.0-beta
5 months ago
0.1.1-beta
5 months ago
0.1.0-beta
5 months ago
0.1.0
5 months ago
0.0.1-beta.2
5 months ago
0.0.1-beta.1
5 months ago