当前位置: 首页 > news >正文

VSCode 插件开发实战(十五):如何支持多语言

前言

在软件开发中,多语言支持(i18n)是一个非常重要的功能。无论是桌面应用、移动应用,还是浏览器插件,都需要考虑如何支持不同国家和地区的用户,软件应用的多语言支持(i18n)已经成为提升用户体验的关键因素之一。

那么如何为您的自定义 VSCode 插件添加多语言支持,以便更好地服务来自不同语言背景的开发者?本教程将详细介绍如何通过简单而高效的方法,为您的 VSCode 插件实现多语言支持,从而提升其国际化能力。

添加多语言支持

1. 创建语言包文件

在你的插件项目中,创建一个 i18n 目录,用于存放不同语言的翻译文件。每种语言会对应一个单独的 JSON 文件,比如 en.json(英文)和 zh-cn.json(中文)。

en.json

{"helloWorld": "Hello, World!","greeting": "Welcome to our VSCode extension!"
}

zh-cn.json

{"helloWorld": "你好,世界!","greeting": "欢迎使用我们的 VSCode 插件!"
}

完整示例
以下是一个完整的插件目录结构示例:
my-vscode-extension

├── .vscode
│   ├── tasks.json
│   └── launch.json
├── .gitignore
├── README.md
├── package.json
├── src
│   ├── extension.ts
│   └── i18n
│       ├── en.json
│       └── zh-cn.json
├── tsconfig.json
└── vsc-extension-quickstart.md

2. 加载语言包

接下来,我们需要在插件代码中加载这些语言包。可以在 extension.js 或 extension.ts 中实现这一功能。
extension.ts

import * as vscode from 'vscode';
import * as path from 'path';
import * as fs from 'fs';function loadMessageBundle(locale: string) {const filePath = path.join(__dirname, 'i18n', `${locale}.json`);if (fs.existsSync(filePath)) {return JSON.parse(fs.readFileSync(filePath, 'utf8'));} else {// Default to English if locale file is not foundreturn JSON.parse(fs.readFileSync(path.join(__dirname, 'i18n', 'en.json'), 'utf8'));}
}export function activate(context: vscode.ExtensionContext) {const locale = vscode.env.language; // Get the current language setting of VSCodeconst messages = loadMessageBundle(locale);let disposable = vscode.commands.registerCommand('extension.helloWorld', () => {vscode.window.showInformationMessage(messages['helloWorld']);});context.subscriptions.push(disposable);
}export function deactivate() {}

3. 更新 package.json

最后,我们需要更新 package.json 文件,声明插件的语言包配置。

{"contributes": {"localizations": [{"languageId": "en","languageName": "English","translations": [{"id": "en","path": "./i18n/en.json"}]},{"languageId": "zh-cn","languageName": "Chinese (Simplified)","translations": [{"id": "zh-cn","path": "./i18n/zh-cn.json"}]}]}
}

进阶操作

动态切换语言

有时候,用户可能希望在不重启 VSCode 的情况下切换语言。我们可以借助 VSCode API 实现这一功能。

修改 extension.ts

首先,我们需要修改 extension.ts 文件,以支持动态加载语言包。

import * as vscode from 'vscode';
import * as path from 'path';
import * as fs from 'fs';let currentLocale: string = vscode.env.language;
let messages: { [key: string]: string };function loadMessageBundle(locale: string) {const filePath = path.join(__dirname, 'i18n', `${locale}.json`);if (fs.existsSync(filePath)) {return JSON.parse(fs.readFileSync(filePath, 'utf8'));} else {return JSON.parse(fs.readFileSync(path.join(__dirname, 'i18n', 'en.json'), 'utf8'));}
}function refreshMessages() {messages = loadMessageBundle(currentLocale);
}export function activate(context: vscode.ExtensionContext) {refreshMessages();let disposable = vscode.commands.registerCommand('extension.helloWorld', () => {vscode.window.showInformationMessage(messages['helloWorld']);});let changeLocaleCommand = vscode.commands.registerCommand('extension.changeLocale', async () => {const picked = await vscode.window.showQuickPick(['en', 'zh-cn'], {placeHolder: 'Select a language'});if (picked) {currentLocale = picked;refreshMessages();vscode.window.showInformationMessage(messages['greeting']);}});context.subscriptions.push(disposable, changeLocaleCommand);
}export function deactivate() {}

更新 package.json

为了让用户能够通过命令面板切换语言,我们需要在 package.json 中添加相应的命令配置。

{"contributes": {"commands": [{"command": "extension.helloWorld","title": "Hello World"},{"command": "extension.changeLocale","title": "Change Language"}],"localizations": [{"languageId": "en","languageName": "English","translations": [{"id": "en","path": "./i18n/en.json"}]},{"languageId": "zh-cn","languageName": "Chinese (Simplified)","translations": [{"id": "zh-cn","path": "./i18n/zh-cn.json"}]}]}
}

使用 TypeScript 类型定义

为了编写更健壮的代码,我们可以为语言包定义一个类型,并在加载语言包时进行类型检查。

定义类型

interface Messages {helloWorld: string;greeting: string;
}修改 loadMessageBundle 函数
function loadMessageBundle(locale: string): Messages {const filePath = path.join(__dirname, 'i18n', `${locale}.json`);if (fs.existsSync(filePath)) {return JSON.parse(fs.readFileSync(filePath, 'utf8')) as Messages;} else {return JSON.parse(fs.readFileSync(path.join(__dirname, 'i18n', 'en.json'), 'utf8')) as Messages;}
}

这样,我们在使用 messages 对象时,TypeScript 会帮助我们进行类型检查,确保代码的可靠性。

处理复杂的翻译需求

在实际应用中,翻译内容可能不仅仅是简单的字符串,还会涉及变量和占位符。我们可以使用较为成熟的 i18n 库来处理这些复杂的翻译需求。例如,使用 i18n 或 i18next 库。

使用 i18n
首先,安装 i18n 库:

npm install i18n

配置 i18n

import * as i18n from 'i18n';
import * as path from 'path';i18n.configure({locales: ['en', 'zh-cn'],directory: path.join(__dirname, 'i18n'),defaultLocale: 'en',extension: '.json',register: global
});function setLocale(locale: string) {i18n.setLocale(locale);
}export function activate(context: vscode.ExtensionContext) {setLocale(vscode.env.language);let disposable = vscode.commands.registerCommand('extension.helloWorld', () => {vscode.window.showInformationMessage(__('helloWorld'));});let changeLocaleCommand = vscode.commands.registerCommand('extension.changeLocale', async () => {const picked = await vscode.window.showQuickPick(['en', 'zh-cn'], {placeHolder: 'Select a language'});if (picked) {setLocale(picked);vscode.window.showInformationMessage(__('greeting'));}});context.subscriptions.push(disposable, changeLocaleCommand);
}export function deactivate() {}

修改语言包格式
i18n 库要求语言包文件的格式与之前有所不同:

en.json

{"helloWorld": "Hello, World!","greeting": "Welcome to our VSCode extension!"
}

zh-cn.json

{"helloWorld": "你好,世界!","greeting": "欢迎使用我们的 VSCode 插件!"
}

总结

通过本文的详细步骤,我们深入探讨了如何为 VSCode 自定义插件添加多语言支持。我们从创建简单的语言文件开始,逐步实现了动态切换语言的功能,并结合 TypeScript 类型定义和第三方库来处理复杂的翻译需求。掌握这些技术,您不仅能提高插件的用户体验,还能扩大其用户群体,推动插件在国际化市场上的应用。

相关文章:

VSCode 插件开发实战(十五):如何支持多语言

前言 在软件开发中,多语言支持(i18n)是一个非常重要的功能。无论是桌面应用、移动应用,还是浏览器插件,都需要考虑如何支持不同国家和地区的用户,软件应用的多语言支持(i18n)已经成…...

面试241228

面试可参考 1、cas的概念 2、AQS的概念 3、redis的数据结构 使用场景 不熟 4、redis list 扩容流程 5、dubbo 怎么进行服务注册和调用,6、dubbo 预热 7如何解决cos上传的安全问题kafka的高并发高吞吐的原因ES倒排索引的原理 spring的 bean的 二级缓存和三级缓存 spr…...

​Python数据序列化模块pickle使用

pickle 是 Python 的一个标准库模块,它实现了基本的对象序列化和反序列化。序列化是指将对象转换为字节流的过程,这样对象就可以被保存到文件中或通过网络传输。反序列化是指从字节流中恢复对象的过程。 以下是 pickle 模块的基本使用方法: …...

Spring Boot对访问密钥加解密——HMAC-SHA256

HMAC-SHA256 简介 HMAC-SHA256 是一种基于 哈希函数 的消息认证码(Message Authentication Code, MAC),它结合了哈希算法(如 SHA-256)和一个密钥,用于验证消息的完整性和真实性。 HMAC 是 “Hash-based M…...

HTML 元素:网页构建的基础

HTML 元素:网页构建的基础 HTML(HyperText Markup Language,超文本标记语言)是构建网页的基石。它定义了网页的结构和内容,而HTML元素则是构成HTML文档的基石。在本篇文章中,我们将深入探讨HTML元素的概念、类型、用法,以及如何在网页设计中有效地使用它们。 什么是HT…...

HEIC 是什么图片格式?如何把 iPhone 中的 HEIC 转为 JPG?

在 iPhone 拍摄照片时,默认的图片格式为 HEIC。虽然 HEIC 格式具有高压缩比、高画质等优点,但在某些设备或软件上可能存在兼容性问题。因此,将 HEIC 格式转换为更为通用的 JPG 格式就显得很有必要。本教程将介绍如何使用简鹿格式工厂&#xf…...

爆肝1个月:DDR4 的信号完整性(万字长文SI)

前言: 大学里面,总有很多课程,很浪费时间,学了没点用处,问过老师,为什么信号完整性,示波器使用等课程不开呢,这种是对工作真实有帮助的? 老师:因为老师…...

前端js验证码插件

相关代码,在最上方的绑定资源...

关于easy-es对时间范围查询遇到的小bug

前言:在使用easy-es之前作为一个小白的我只有es原生查询的基础,在自己通过查看官方文档自学easy-es遇到了一个挫折,其他的还好语法和MybatisPlus差不多,正以为我觉得很快就能入手,在对时间范围的判断就给我当头一棒&am…...

Mask R-CNN

目录 摘要 Abstract Mask R-CNN 网络架构 Backbone RPN Proposal Layer ROIAlign bbox检测 Mask分割 损失计算 实验复现 总结 摘要 Mask R-CNN是在Faster R-CNN的基础上进行改进的目标检测和实例分割网络。Faster R-CNN主要用于目标检测,输出对象的边…...

大模型-Dify使用笔记

大模型-Dify使用笔记 0、调整docker镜像源1、安装1、Docker Compose方式部署 2、访问 Dify 0、调整docker镜像源 由于墙的存在,所以默认的docker镜像源很难拉取项目,需要调整相关的docker配置文件 vim /etc/docker/daemon.json添加如下docker镜像源 { …...

Suno Api V4模型无水印开发「综合实战开发自己的音乐网站」 —— 「Suno Api系列」第14篇

历史文章 Suno AI API接入 - 将AI音乐接入到自己的产品中,支持120并发任务 Suno Api V4模型无水印开发「灵感模式」 —— 「Suno Api系列」第1篇 Suno Api V4模型无水印开发「自定义模式」 —— 「Suno Api系列」第2篇 Suno Api V4模型无水印开发「AI生成歌词」…...

云原生架构中的中间件容器化:优劣势与实践探索

在云原生架构逐步推进的过程中,许多企业已经开始将应用和服务容器化,以充分利用云计算带来的弹性和自动化。随着容器技术的发展,容器化不仅仅限于应用层,越来越多的中间件也被考虑纳入容器化范畴,包括Redis、Kafka、Ra…...

如何测试模型推理性能:从零开始的Python指南

如何测试模型推理性能:从零开始的Python指南 什么是模型推理性能?测试模型推理性能的步骤1. 监测内存使用情况2. 测试模型吞吐量 运行测试总结 在机器学习和深度学习中,模型的推理性能是一个非常重要的指标。它可以帮助我们了解模型在实际应用…...

我们来学activiti -- bpmn

bpmn 题记bpmn结余 题记 在《Activiti很难学》提到学习知识点需要面对的思想钢印问题 按常见步骤,先展示下官方的客套话 BPMN(Business Process Model and Notation)是一种业务流程建模符号, 它是一种图形化的语言,用…...

【每日学点鸿蒙知识】节点析构问题、区分手机和pad、 Navigation路由问题、Tabs组件宽度、如何监听Map

1、HarmonyOS 只调用根节点的dispose,是否其下的子节点都能析构掉还是需要遍历子节点,都执行dispose才能正常析构? 前端持有引用关系的需要dispose,new出来的builderNode和FrameNode也需要dispose。只调用根节点的dispose,无法保证其下的子节…...

敏捷测试文化的转变

敏捷文化是敏捷测试转型的基础,只有具备敏捷文化的氛围,对组织架构、流程和相关测试实践的调整才能起作用。在前面的敏捷测试定义中,敏捷测试是遵从敏捷软件开发原则的一种测试实践,这意味着敏捷的价值观。 此外,从传…...

如何配置线程池参数,才能创建性能最好、最稳定的Spring异步线程池?

配置性能最好、最稳定的Spring异步线程池,需要综合考虑业务场景、硬件资源(CPU核心数、内存等)、并发量、任务特性(CPU密集型、IO密集型等)以及线程池参数。 以下是优化线程池配置的关键点及代码示例: 线程…...

【时间之外】IT人求职和创业应知【80】-特殊日子

目录 北京冬季招聘会 OpenAI CEO炮轰马斯克 英伟达推出全新AI芯片B300 莫欢喜,总成空。本周必须要谨行慎言。 感谢所有打开这个页面的朋友。人生不如意,开越野车去撒野,会害了自己,不如提升自己。提升自己的捷径就是学习和思考…...

Vue中接入萤石等直播视频(更新中ing)

一、萤石: 1. 萤石云开发文档: https://open.ys7.com/help/31 2、安装: npm install ezuikit-js --save 3、在文件中引用:import EZUIKit from ezuikit-js 4、具体代码: 获取accessToken:https://open.…...

HTML 列表、表格、表单

1 列表标签 作用:布局内容排列整齐的区域 列表分类:无序列表、有序列表、定义列表。 例如: 1.1 无序列表 标签:ul 嵌套 li,ul是无序列表,li是列表条目。 注意事项: ul 标签里面只能包裹 li…...

服务器硬防的应用场景都有哪些?

服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式,避免服务器受到各种恶意攻击和网络威胁,那么,服务器硬防通常都会应用在哪些场景当中呢? 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...

如何在网页里填写 PDF 表格?

有时候,你可能希望用户能在你的网站上填写 PDF 表单。然而,这件事并不简单,因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件,但原生并不支持编辑或填写它们。更糟的是,如果你想收集表单数据&#xff…...

USB Over IP专用硬件的5个特点

USB over IP技术通过将USB协议数据封装在标准TCP/IP网络数据包中,从根本上改变了USB连接。这允许客户端通过局域网或广域网远程访问和控制物理连接到服务器的USB设备(如专用硬件设备),从而消除了直接物理连接的需要。USB over IP的…...

HashMap中的put方法执行流程(流程图)

1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...

初探Service服务发现机制

1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源&#xf…...

【前端异常】JavaScript错误处理:分析 Uncaught (in promise) error

在前端开发中,JavaScript 异常是不可避免的。随着现代前端应用越来越多地使用异步操作(如 Promise、async/await 等),开发者常常会遇到 Uncaught (in promise) error 错误。这个错误是由于未正确处理 Promise 的拒绝(r…...

C++_哈希表

本篇文章是对C学习的哈希表部分的学习分享 相信一定会对你有所帮助~ 那咱们废话不多说,直接开始吧! 一、基础概念 1. 哈希核心思想: 哈希函数的作用:通过此函数建立一个Key与存储位置之间的映射关系。理想目标:实现…...

SQL注入篇-sqlmap的配置和使用

在之前的皮卡丘靶场第五期SQL注入的内容中我们谈到了sqlmap,但是由于很多朋友看不了解命令行格式,所以是纯手动获取数据库信息的 接下来我们就用sqlmap来进行皮卡丘靶场的sql注入学习,链接:https://wwhc.lanzoue.com/ifJY32ybh6vc…...

Oracle实用参考(13)——Oracle for Linux物理DG环境搭建(2)

13.2. Oracle for Linux物理DG环境搭建 Oracle 数据库的DataGuard技术方案,业界也称为DG,其在数据库高可用、容灾及负载分离等方面,都有着非常广泛的应用,对此,前面相关章节已做过较为详尽的讲解,此处不再赘述。 需要说明的是, DG方案又分为物理DG和逻辑DG,两者的搭建…...