记一次前端Vue项目国际化解决方案

背景

有一个vue项目,要实现国际化功能,能够切换中英文显示,因为该项目系统的用户包括了国内和国外用户。

需求

1、页面表单上的所有中文标签要国际化,包括表单属性标签、表格列头标签等, title=“数量”;

2、输入框的提示内容需要国际化,如 placeholder=“选择日期”

3、js代码中的提示信息需要国际化,如 message(“请勾选批量设置”)、confirm(‘您确定要设置业务损耗吗?’)、title: ‘删除错误’ 等;

解决方案

1、开发流程,一开始开发过程中,我们不考虑国际化,等代码基本完成后,最后再进行国际化;

2、考虑日后还可能由其他语种,所以这里我们做国际化词语库时,国际化编码使用5位数字,对应多种语言值,即一对多;

3、前端我们重新封装一个全局方法 $lang(param1, param2) 来支持国际化,param1是国际化编码,param2是默认值(如果国际化编码没找到对应的语言单词,则默认用param2,且去掉左右两边的 ‘~’符号);

其实后来又分析了下,如果一开始前端开发人员把所有需要国际化的中文词语,都写成 $lang(‘中文词语’) , $lang方法逻辑再修改下,如果没有第二个参数并且第一个参数对应的国际化词语也没有,则直接显示第一个参数字符串,而且这样的话,到后面再提取代码中的需要国际化的内容时就会很精确了。

4、国际化流程:

  • 从前端代码文件中将所有的中文提取出来,形成一个数组放到一个json文件中,并且数组需要去重一下;
  • 使用第三方的翻译接口,来对导出的中文进行翻译,生成一个中英文对照键值对json文件;
  • 校对中英文对照表,因为有的翻译不一定准确;
  • 根据校对后的中英文对照表,生成国际化编码库,并创建两个国际化文件;
  • 根据校对后的中英文对照表,并分析代码规则,将程序代码中的中文进行国际化处理;

国际化流程实施

在国际化流程实施中,我使用编写js脚本代码来实现相关的处理,使用node环境来执行脚本;

1、提取中文

从前端代码文件中将所有的中文提取出来,形成一个数组放到一个json文件中,并且数组需要去重一下;

下面的代码,是用来提取文件代码中的中文的,我们可以将代码文件命名为extractChinese.js,使用node来执行该脚本;

代码中要国际化的路径设置的是当前目录下的src下的 components和pages文件夹

const fs = require('fs');  
const path = require('path');    
const chineseRegex = /[一-龥]+/g;  function extractChineseFromFile(filePath) {  const content = fs.readFileSync(filePath, 'utf-8');  const chineseWords = content.match(chineseRegex);  return chineseWords || [];  
}  function processDirectory(directoryPath) {  const files = fs.readdirSync(directoryPath);  const chineseSentences = [];  files.forEach((fileName) => {  const filePath = path.join(directoryPath, fileName);  const stats = fs.statSync(filePath);if (stats.isDirectory()) {  chineseSentences.push(...processDirectory(filePath));  } else if (stats.isFile() && ['.js', '.vue'].indexOf(path.extname(filePath)) > -1) {const chineseWords = extractChineseFromFile(filePath);  chineseSentences.push(...chineseWords);  }  });  return chineseSentences;  
}  function main() {  const srcDirectory = path.join(__dirname, 'src');  const componentsDirectory = path.join(srcDirectory, 'components');  const pagesDirectory = path.join(srcDirectory, 'pages');  const componentsChineseSentences = processDirectory(componentsDirectory);  const pagesChineseSentences = processDirectory(pagesDirectory);  const allChineseSentences = [...componentsChineseSentences, ...pagesChineseSentences];  //const allChineseSentences = componentsChineseSentences;  const outputPath = path.join(__dirname, 'output.json'); // 使用 Set 对象来去重  let backString = Array.from(new Set(allChineseSentences)); // 对去重后的数组进行排序  backString.sort();fs.writeFileSync(outputPath, JSON.stringify(backString, null, 2), 'utf-8');  console.log('提取到的中文单词或语句已保存到output.json文件中。');  
}  main();
2、翻译中文

使用第三方的翻译接口,来对导出的中文进行翻译,生成一个中英文对照键值对json文件;

翻译接口,这里我们用的是百度翻译,至于如何去使用百度翻译,这里就不再说了,自己去百度看吧;

该步骤需要用到第一步生成的 output.json 文件,然后翻译结果是存在 translated_zh_en.json 中。

const fs = require('fs');  
const axios = require('axios');  
const appId = '123456789'; // 替换成你的百度翻译的APP ID  
const secretKey = '999999999'; // 替换成你的百度翻译的密钥  const crypto = require('crypto');  
axios.defaults.headers.post["Content-Type"] = "application/x-www-form-urlencoded;charset=UTF-8";function md5Hash(input) {  // 创建一个哈希对象  const hash = crypto.createHash('md5');  // 更新哈希对象的内容  hash.update(input);  // 获取哈希值的二进制表示  const hashBuffer = hash.digest();  // 将二进制转换为十六进制表示  const hashHex = hashBuffer.toString('hex');  // 返回小写的哈希值  return hashHex.toLowerCase();  
}  // 使用百度翻译API进行翻译  
async function translateToEnglish(text) { const params = {  q: text,  appid: appId,  salt: Date.now(),  from: 'zh',  to: 'en',  sign: ''  };// 计算签名  params.sign = md5Hash(params.appid + params.q + params.salt + secretKey);// 请求翻译  const url = `http://api.fanyi.baidu.com/api/trans/vip/translate?q=${encodeURIComponent(params.q)}&from=zh&to=en&appid=${params.appid}&salt=${params.salt}&sign=${params.sign}`;const response = await axios.get(url); //console.log(url);//console.log(response.data)// 返回翻译结果  return response.data.trans_result[0].dst;  
}  function sleep(ms) {  return new Promise(resolve => setTimeout(resolve, ms));  
}  async function mysleep() {  console.log('休息1秒......................');  await sleep(1000); // 暂停 1 秒  console.log('休息完成...');  
} async function process() {  // 读取json文件  const data = JSON.parse(fs.readFileSync('output.json', 'utf8'));  // 存储翻译结果的对象  let translationData = {};  let execNumber = 1;// 遍历中文字符串数组,进行翻译for (let i = 0; i < data.length; i++) {  const chineseString = data[i];  const englishString = await translateToEnglish(chineseString);  // 将原中文字符串和英文字符串形成键值对存储到translationData对象中  translationData[chineseString] = englishString;if (execNumber >= 120) {   // 如果不想全部执行,则执行多少场退出break;} else if (i == execNumber*20) { // 每执行20次接口调用,就休息1秒execNumber++;await mysleep()}}// 将翻译结果写入translate.json文件中  fs.writeFileSync('translated_zh_en.json', JSON.stringify(translationData, null, 2));  
}  process().catch(error => {  console.error(error);  
});
3、校对翻译

校对中英文对照表,因为有的翻译不一定准确;(找个行业英语水平高点的人,自己去校对吧)

4、创建国际化库

根据校对后的中英文对照表,生成国际化编码库,并创建两个国际化文件;

const fs = require('fs');  // 读取原始 JSON 文件  
const data = JSON.parse(fs.readFileSync('translated_zh_en.json', 'utf8'));  // 中文和英文的 JSON 文件  
const chineseData = {};  
const englishData = {};  let serialNumber = 00001;  // 遍历原始数据,生成新的键值对  
for (let chinese in data) {  const english = data[chinese];  // 生成新的键值对,序号为 5 位数字  const key = `N${String(serialNumber).padStart(5, '0')}`;  chineseData[key] = chinese;  englishData[key] = english;  serialNumber++;  
}  // 将中文和英文的 JSON 数据写入文件  
fs.writeFileSync('cn.json', JSON.stringify(chineseData, null, 2));  
fs.writeFileSync('en.json', JSON.stringify(englishData, null, 2));
5、代码国际化处理

根据第4步生成的中文国际化文件 cn.json ,并分析代码规则,将程序代码中的中文进行国际化处理;

首先要分析程序需要国际化的代码规则,因为这个替换不是简单的去就把中文替换,可能代码都由变化,我们分析项目代码中目前的规则如下:

场 景

**代码示例****

**查找内容****

**替换内容****

作为组件元素内容的

<vxe-button @click="closeModel">取消</vxe-button> <span style="color: red;">如调整了颜色尺码,保存后请务必核对检查数量和配色数据!</span> <div class="title">尺码信息</div>

>取消<

>{{$lang(‘10000’, ‘取消’)}}<

作为组件元素属性值的

<vxe-table-column field="odgc_pcs" title="数量" width="100" header-align="center" align="right"> <el-date-picker v-if="row.type == 'date'" type="date" placeholder="选择日期" v-model="row.value">

title="数量"placeholder=“选择日期”

:title=“ l a n g ( ′ 1000 1 ′ , ′ 数量 ′ ) " : p l a c e h o l d e r = " lang(‘10001’, ‘数量’)“:placeholder=” lang(′10001′,′数量′)":placeholder="lang(‘Ph_select_data’, ‘选择日期’)”

组件模板代码中三元运算结果

<el-button size="mini" @click="alterConsumption(row)">{{onlyShow?'查看':'修改'}}</el-button>

‘查看’:‘查看’ ::‘修改’: ‘修改’

l a n g ( ′ 1000 2 ′ , ′ 查看 ′ ) 同上 lang(‘10002’, ‘查看’)同上 lang(′10002′,′查看′)同上lang(‘10003’, ‘修改’)同上

js 中方法参数值

this.$XModal.message("请勾选批量设置", "error"); this.$XModal.confirm('您确定要设置吗?') this.$confirm("确定要删除此记录吗 ?", "提示", { confirmButtonText: "确定", cancelButtonText: "取消", type: "warning", })

message("请勾选批量设置"message('请勾选要批量设置’confirm("您确定要设置吗?"confirm(‘您确定要设置吗?’“提示”,confirmButtonText: “确定”,cancelButtonText: “取消”,

message(this. l a n g ( ′ 1000 4 ′ , ′ 请勾选批量设置 ′ ) 同上 c o n f i r m ( t h i s . lang(‘10004’, ‘请勾选批量设置’)同上confirm(this. lang(′10004′,′请勾选批量设置′)同上confirm(this.lang(‘10005’, ‘您确定要设置吗?’)同上this. l a n g ( ′ 1000 6 ′ , ′ 提示 ′ ) , c o n f i r m B u t t o n T e x t : t h i s . lang(‘10006’, ‘提示’),confirmButtonText: this. lang(′10006′,′提示′),confirmButtonText:this.lang(‘10007’, ‘确定’)cancelButtonText: this.$lang(‘10008’, ‘取消’)

js 中对象属性赋值

`this. X M o d a l . m e s s a g e ( m e s s a g e : " 保存失败 " , s t a t u s : " e r r o r " ) ; t h i s . XModal.message({ message: “保存失败”, status: “error” }); this. XModal.message(message:“保存失败”,status:“error”);this.message({ message: ‘请选择要设置的物料!’, type: ‘warning’ }); this. X M o d a l . a l e r t ( m e s s a g e : " 请选择附件分类 " , s t a t u s : " w a r n i n g " , ) ; t h i s . XModal.alert({ message: “请选择附件分类”, status: “warning”, }); this. XModal.alert(message:“请选择附件分类”,status:“warning”,);this.XModal.alert({ status: “error”, title: “删除错误”, message: response.msg

“服务器删除发生错误”, });`

js 中 || 赋值

`this.$XModal.alert({ status: “error”, title: “删除错误”, message: response.msg

“服务器删除发生错误”, });`

替换的脚本代码如下:

const fs = require('fs');
const path = require('path');// 读取 cn.json 文件并解析 JSON 数据  
function loadTranslations() {const cnJsonPath = path.join(__dirname, 'src', 'lang', 'cn.json');const content = fs.readFileSync(cnJsonPath, 'utf-8');return JSON.parse(content);
}// 判断字符串是否以指定前缀开头  
function startsWith(str, prefix) {return str.startsWith(prefix);
}/*** 每个键值对的场景匹配* @param {String} fileContent 文件内容* @param {String} key 国际化变量名* @param {String} value 中文字符串*/
function replaceAllScene(fileContent, key, value) {// 场景:>取消<let searchValue = `>${value}<`;let replaceValue = `>{{$lang('${key}', '~${value}~')}}<`;fileContent = fileContent.split(searchValue).join(replaceValue);  // 场景:title="数量"searchValue = `title="${value}"`;replaceValue = `:title="$lang('${key}', '~${value}~')"`;fileContent = fileContent.split(searchValue).join(replaceValue);// 场景:placeholder="选择日期"searchValue = `placeholder="${value}"`;replaceValue = `:placeholder="$lang('${key}', '~${value}~')"`;fileContent = fileContent.split(searchValue).join(replaceValue);// 场景:message("请勾选批量设置"searchValue = `message("${value}"`;replaceValue = `message(this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// 场景:message('请勾选批量设置'searchValue = `message('${value}'`;replaceValue = `message(this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// 场景:confirm("您确定要设置业务损耗吗?"searchValue = `confirm("${value}"`;replaceValue = `confirm(this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// 场景:confirm('您确定要设置业务损耗吗?'searchValue = `confirm('${value}'`;replaceValue = `confirm(this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// confirmButtonText: "确定",searchValue = `confirmButtonText: "${value}",`;replaceValue = `confirmButtonText: this.$lang('${key}', '~${value}~'),`;fileContent = fileContent.split(searchValue).join(replaceValue);// cancelButtonText: "取消",searchValue = `cancelButtonText: "${value}",`;replaceValue = `cancelButtonText: this.$lang('${key}', '~${value}~'),`;fileContent = fileContent.split(searchValue).join(replaceValue);// message: "保存失败"searchValue = `message: "${value}"`;replaceValue = `message: this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// message: '保存失败''searchValue = `message: '${value}'`;replaceValue = `message: this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// title: "删除错误"searchValue = `title: "${value}"`;replaceValue = `title: this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);// title: '删除错误'searchValue = `title: '${value}'`;replaceValue = `title: this.$lang('${key}', '~${value}~')`;fileContent = fileContent.split(searchValue).join(replaceValue);return fileContent;
}// 在给定文件中替换指定的字符串  
function replaceStringsInFile(filePath, replacements) {const content = fs.readFileSync(filePath, 'utf-8');let newContent = content;for (const [key, value] of Object.entries(replacements)) {// 如果匹配到的字符串前面存在 "message(",则去掉左右两边的双引号  //const searchValue = startsWith(value, 'message("') ? value.slice(8, -1) : value;newContent = replaceAllScene(newContent, key, value);//newContent = newContent.split(searchValue).join("$lang('" + key + "',')" + searchValue + "'");}if (newContent !== content) {fs.writeFileSync(filePath, newContent, 'utf-8');console.log(`Replaced strings in ${filePath}`);}
}// 在指定目录下处理所有文件  
function processDirectory(directoryPath, replacements) {const files = fs.readdirSync(directoryPath);files.forEach((fileName) => {const filePath = path.join(directoryPath, fileName);const stats = fs.statSync(filePath);if (stats.isDirectory()) {processDirectory(filePath, replacements);} else if (stats.isFile()) {replaceStringsInFile(filePath, replacements);}});
}function main() {const translations = loadTranslations();const componentsDirectory = path.join(__dirname, 'src', 'components');const pagesDirPath = path.join(__dirname, 'src', 'pages');processDirectory(componentsDirectory, translations);processDirectory(pagesDirPath, translations);
}main();

到此,我们就完成了前端代码的国际化实现;

我们为什么要把原中文作为 国际化方法 $lang 的第二个参数呢?

因为,如果代码文件中看不到中文,修改代码的时候太难找了,你只能看到国际化数字编码。

建议

建议是在前端一开始开发的时候,就把需要国际化的地方都写成 $lang(‘中文’),包括模板代码和js代码中,

这样后期替换更精确,而且一开始开发人员也不用去管国际化,

并且,我们在提取代码中文时,就可以按 $lang(‘中文’) 这个格式精确提取了,国际化处理后就变成 $lang(‘国际化编码’,‘中文’) ,这样我们在第二次再提取时,就不会重复提取已经国际化处理后的代码中文了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/495489.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AIA - IMSIC之二(附IMSIC处理流程图)

本文属于《 RISC-V指令集基础系列教程》之一,欢迎查看其它文章。 1 ​​​​​​​通过IMSIC接收外部中断的CSR 软件通过《AIA - 新增的CSR》描述的CSR来访问IMSIC。 machine level 的 CSR 与 IMSIC 的 machine level interrupt file 可相互互动;而 supervisor level 的 CSR…

光谱相机的工作原理

光谱相机的工作原理主要基于不同物质对不同波长光的吸收、反射和透射特性存在差异&#xff0c;以下是其具体工作过程&#xff1a; 一、光的收集 目标物体在光源照射下&#xff0c;其表面会对光产生吸收、反射和透射等相互作用。光谱相机的光学系统&#xff08;如透镜、反射镜…

Kafka可视化工具 Offset Explorer (以前叫Kafka Tool)

数据的存储是基于 主题&#xff08;Topic&#xff09; 和 分区&#xff08;Partition&#xff09; 的 Kafka是一个高可靠性的分布式消息系统&#xff0c;广泛应用于大规模数据处理和实时, 为了更方便地管理和监控Kafka集群&#xff0c;开发人员和运维人员经常需要使用可视化工具…

TLDR:终端命令的简洁百科全书

TLDR&#xff0c;全称 “Too Long, Don’t Read”&#xff0c;是一款特别实用的终端命令百科全书工具。通过 TLDR&#xff0c;您可以快速查找到常用命令的使用方法&#xff0c;避免繁琐冗长的官方文档&#xff0c;让日常工作更加高效。 为什么选择 TLDR&#xff1f; 简单易用&…

2024-12-25-sklearn学习(20)无监督学习-双聚类 料峭春风吹酒醒,微冷,山头斜照却相迎。

文章目录 sklearn学习(20) 无监督学习-双聚类1 Spectral Co-Clustering1.1 数学公式 2 Spectral Biclustering2.1 数学表示 3 Biclustering 评价 sklearn学习(20) 无监督学习-双聚类 文章参考网站&#xff1a; https://sklearn.apachecn.org/ 和 https://scikit-learn.org/sta…

数据结构(Java版)第六期:LinkedList与链表(一)

目录 一、链表 1.1. 链表的概念及结构 1.2. 链表的实现 专栏&#xff1a;数据结构(Java版) 个人主页&#xff1a;手握风云 一、链表 1.1. 链表的概念及结构 链表是⼀种物理存储结构上⾮连续存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的引⽤链接次序实现的。与火车…

《Java核心技术I》Swing的网格包布局

复杂的布局管理 网格包布局 行列大小可改变&#xff0c;先建立表格&#xff0c;合并相邻单元格&#xff0c;组件指定在格内的对齐方式。 字体选择器组件&#xff1a; 另个指定字体和字体大小的组合框两个组合框标签两个选择粗体和斜体的复选框一个显示示例字符串的文本区 将容…

Python——day09

os模块 sys模块 time模块 logging模块

IIC驱动EEPROM

代码参考正点原子 i2c_dri:主要是三段式状态机的编写 module iic_dri#(parameter SLAVE_ADDR 7b1010000 , //EEPROM从机地址parameter CLK_FREQ 26d50_000_000, //模块输入的时钟频率parameter I2C_FREQ 18d250_000 //IIC_SCL的时钟频率)( …

《计算机组成及汇编语言原理》阅读笔记:p86-p115

《计算机组成及汇编语言原理》学习第 6 天&#xff0c;p86-p115 总结&#xff0c;总计 20 页。 一、技术总结 1.if statement 2.loop 在许多编程语言中&#xff0c;有类种循环&#xff1a;一种是在程序开头检测条件(test the condition),另一种是在程序末尾检测条件。 3.C…

(带源码)宠物主题商场系统 计算机项目 P10083

项目说明 本号所发布的项目均由我部署运行验证&#xff0c;可保证项目系统正常运行&#xff0c;以及提供完整源码。 如需要远程部署/定制/讲解系统&#xff0c;可以联系我。定制项目未经同意不会上传&#xff01; 项目源码获取方式放在文章末尾处 注&#xff1a;项目仅供学…

目标检测——基于yolov8和pyqt的螺栓松动检测系统

目录 1.项目克隆和环境配置1.1 我这里使用的是v8.0.6版本1.2 项目代码结构介绍 2.数据集介绍2.1 数据集采集2.2采集结果介绍 3.模型训练4.pyqt界面设计4.1 界面内容介绍4.2 界面实现 5.操作中的逻辑实现5.1 图片检测5.2 文件夹检测5.3 视频检测和摄像头检测 6. 效果展示 1.项目…

宠物行业的出路:在爱与陪伴中寻找增长新机遇

在当下的消费市场中&#xff0c;如果说有什么领域能够逆势而上&#xff0c;宠物行业无疑是一个亮点。当人们越来越注重生活品质和精神寄托时&#xff0c;宠物成为了许多人的重要伴侣。它们不仅仅是家庭的一员&#xff0c;更是情感的寄托和生活的调剂。然而&#xff0c;随着行业…

原点安全再次入选信通院 2024 大数据“星河”案例

近日&#xff0c;中国信息通信研究院和中国通信标准化协会大数据技术标准推进委员会&#xff08;CCSA TC601&#xff09;共同组织开展的 2024 大数据“星河&#xff08;Galaxy&#xff09;”案例征集活动结果正式公布。由工银瑞信基金管理有限公司、北京原点数安科技有限公司联…

【0x001D】HCI_Read_Remote_Version_Information命令详解

目录 一、命令概述 二、命令格式及参数说明 2.12. HCI_Read_Remote_Version_Information 命令格式 2.2. Connection_Handle 三、生成事件 3.1. HCI_Command_Status 事件 3.2. HCI_Read_Remote_Version_Information_Complete 事件 四、命令执行流程 4.1. 命令发起阶段(…

C语言-结构体内存大小

#include <stdio.h> #include <string.h> struct S1 { char a;//1 int b;//4 char c;//1 }; //分析 默认对齐数 成员对齐数 对齐数(前两个最小值) 最大对齐数 // 8 1 …

直流电源如何输出恒压源和恒流源

输出电流达到预定值时&#xff0c;变成稳流特性。 输出电压达到预定值时&#xff0c;变成稳压特性。 电流变大&#xff0c;成稳压。 电压变大&#xff0c;成稳流。

【软考高级】系统架构设计师复习笔记-精华版

文章目录 前言0 系统架构设计师0.1 考架构还是考系分0.2 架构核心知识0.3 架构教材变化 1 计算机操作系统1.1 cpu 组成1.2 内核的五大功能1.3 流水线技术1.4 段页式存储1.5 I/O 软件1.6 文件管理1.7 系统工程相关 2 嵌入式2.1 嵌入式技术2.2 板级支持包&#xff08;BSP&#xf…

如何识别钓鱼邮件和诈骗网站?(附网络安全意识培训PPT资料)

识别钓鱼邮件和诈骗网站是网络安全中的一个重要环节。以下是一些识别钓鱼邮件和诈骗网站的方法&#xff1a; 识别钓鱼邮件&#xff1a; 检查发件人地址&#xff1a; 仔细查看发件人的电子邮件地址&#xff0c;看是否与官方域名一致。 检查邮件内容&#xff1a; 留意邮件中是否…

查询 MySQL 默认的存储引擎(SELECT @@default_storage_engine;)

要查询 MySQL 默认的存储引擎&#xff0c;可以使用以下 SQL 查询语句&#xff1a; SELECT default_storage_engine;解释&#xff1a; SELECT: 表示你要执行一个查询。default_storage_engine: 这是一个 MySQL 系统变量&#xff0c;它存储着当前 MySQL 服务器的默认存储引擎。…