我的编程空间,编程开发者的网络收藏夹
学习永远不晚

vscode+babel然后开发一个智能移除未使用变量的插件

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

vscode+babel然后开发一个智能移除未使用变量的插件

本篇文章给大家分享的是有关vscode+babel然后开发一个智能移除未使用变量的插件,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

vscode 已经成为前端不可缺失的开发工具之一,之所以 vscode 能够获得开发者的青睐,我想和它“无所不能”的插件体系有很大一部分关系。在工作中我们能用它来开发纯工具型的插件,也可以用它开发一些和公司业务相结合的功能插件。在这里我分享一个通过结合babel来实现一个能够智能移除未使用的变量插件。

正文

今天我们首先来熟悉一下 vscode 插件项目的搭建流程

1、使用官方提供的脚手架初始化一个项目

安装脚手架

# npm 形式npm install -g yo generator-code# yarn 形式yarn global add yo generator-code

运行脚手架

# 运行脚手架yo code

选择模板,考虑到有些开发者对 TypeScript 并不熟悉,所以我们这里选择 New Extension (JavaScript)

? What type of extension do you want to create? New Extension (JavaScript)? What's the name of your extension? rm-unuse-var? What's the identifier of your extension? rm-unuse-var? What's the description of your extension? 移除未使用的变量? Enable JavaScript type checking in 'jsconfig.json'? Yes? Initialize a git repository? Yes? Which package manager to use? yarn

这是我们最终生成的目录结构

我们先来运行一下这个插件试试

点击上面运行按钮,会打开一个新的 vscode 窗口,在新窗口中按下Ctrl+Shift+P输入Hello World,在窗口右下角会看到一个提示框,说明我们第一个 vscode 插件运行成功运行了。

2、自定义命令、快捷键、菜单

vscode 插件很多功能都是基于一个个命令实现的,我们可以自定义一些命令,这个命令将出现在按下Ctrl+Shift+P后的命令列表里面,同时可以给命令配置快捷键、配置资源管理器菜单、编辑器菜单、标题菜单、下拉菜单、右上角图标等。

3、如何添加命令列表

package.json 部分配置

{  // 扩展的激活事件  "activationEvents": ["onCommand:rm-unuse-var.helloWorld"],  // 入口文件  "main": "./extension.js",  // 添加指令  "contributes": {    "commands": [      {        // 这里的值必须和activationEvents里面配置的一样        "command": "rm-unuse-var.helloWorld",        // 这个就是我们指令的名称,可以修改这里的值重新运行插件试试看        "title": "Hello World"      }    ]  }}

在开发中快捷键的使用方式是最便捷的,接下来我们修改一下配置,让插件支持快捷键的方式运行。

{  "contributes": {    "commands": [      {        // 这里的值必须和activationEvents里面配置的一样        "command": "rm-unuse-var.helloWorld",        // 这个就是我们指令的名称,可以修改这里的值重新运行插件试试看        "title": "Hello World"      }    ],    // 快捷键绑定    "keybindings": [      {        "command": "rm-unuse-var.helloWorld",        "key": "ctrl+6",        "mac": "cmd+6"      }    ]  }}

我们再重新运行一下,通过快捷键Ctrl+6看看我们的插件是否能够正常运行。没错就是这么简单,我们的插件已经能够支持快捷键的形式运行了。

4、叫 helloWorld 太土了,下一步我们来修改一下指令的名称

package.json

{  "activationEvents": ["onCommand:rm-unuse-var.rm-js-var"],  "main": "./extension.js",  "contributes": {    "commands": [      {        "command": "rm-unuse-var.rm-js-var",        "title": "Hello World"      }    ],    "keybindings": [      {        "command": "rm-unuse-var.rm-js-var",        "key": "ctrl+6",        "mac": "cmd+6"      }    ]  }}

因为我们在extension.js中注册了指令的名称,所以也要同步修改

let disposable = vscode.commands.registerCommand(  "rm-unuse-var.rm-js-var",  function () {    vscode.window.showInformationMessage("Hello World from rm-unuse-var!");  });

5、安装babel相关库

我们修改代码可以分为 3 个步骤

将代码解析成 AST 语法树2、遍历修改 AST 语法树3、根据修改过的 AST 语法树生成新的代码

这 3 个步骤 babel 都有对应的库来处理

  • @babel/parser生成 AST 语法树,文档地址(https://www.babeljs.cn/docs/babel-parser)

  • @babel/traverse遍历 AST 语法树,文档地址(https://www.babeljs.cn/docs/babel-traverse)

  • @babel/generator根据 AST 语法树生成代码,文档地址(https://www.babeljs.cn/docs/babel-generator)

  • @babel/types工具库,文档地址(https://www.babeljs.cn/docs/babel-types)

6、我们来看下这些库的基本用法,比如实现一个将 es6 的箭头函数转换成普通函数

转换前

const say = () => {  console.log("hello");};

转换后

function say() {  console.log("hello");}

代码实现,代码部分写死仅供学习参考

const t = require("@babel/types");const parser = require("@babel/parser");const traverse = require("@babel/traverse").default;const generate = require("@babel/generator").default;// 1、将代码解析成 AST 语法树const ast = parser.parse(`const say = () => {  console.log("hello");};`);// 2、遍历修改 AST 语法树traverse(ast, {  VariableDeclaration(path) {    const { node } = path;    // 写死找到第一个申明    const declaration = node.declarations[0];    // 定义的内容    const init = declaration.init;    // 判断是否是箭头函数    if (t.isArrowFunctionExpression(init)) {      // 将原来的表达式替换成新生成的函数      path.replaceWith(        t.functionDeclaration(          declaration.id,          init.params,          init.body,          init.generator,          init.async        )      );    }  },});// 3、根据修改过的 AST 语法树生成新的代码console.log(generate(ast).code);

很多同学肯定好奇现在我们的表达式比较简单还好,如果复杂的话定义嵌套会非常深和复杂,这个时候应该怎么知道去替换哪个节点?。其实这里可以借助astexplorer.net/ 这是一个在线转换 AST 的网站。我们可以打开两个窗口,把转换前的代码放到第一个窗口,把需要转换的接口放到第二个窗口。这个时候我们就可以对比一下转换前后的差异,来实现我们的代码了。

6、思考插件如何实现?

获取编辑器当前打开的 js 文件的代码2、将代码解析成 AST 语法树3、遍历 AST 语法树,删除未使用的定义4、根据修改过的 AST 语法树生成新的代码5、替换当前 js 文件的代码

其中 2、4 我们已经会了,接下来只需要看下 1、3、5 如何实现就行

1 和 5 我们可以通过 vscode 提供的方法

获取编辑器当前打开的 js 文件的代码

import * as vscode from "vscode";// 当前打开的文件const { activeTextEditor } = vscode.window;// 然后通过document下的getText就能轻松获取到我们的代码了const code = activeTextEditor.document.getText();

替换当前 js 文件的代码

activeTextEditor.edit((editBuilder) => {  editBuilder.replace(    // 因为我们要全文件替换,所以我们需要定义一个从头到位的区间    new vscode.Range(      new vscode.Position(0, 0),      new vscode.Position(activeTextEditor.document.lineCount + 1, 0)    ),    // 我们的新代码    generate(ast).code  );});

好了接下来我们就剩核心的第 3 步了。

遍历 AST 语法树,删除未使用的定义

我们先来分析一下,未使用的定义包含了哪些?

import vue from "vue";const a = { test1: 1, test2: 2 };const { test1, test2 } = a;function b() {}let c = () => {};var d = () => {};

然后在线 ast 转换网站,复制这些内容进去看看生成的语法树结构

vscode+babel然后开发一个智能移除未使用变量的插件

我们先来实现一个例子吧,比如把下面代码中没有用的变量移除掉

转换前

var a = 1;var b = 2;console.log(a);

转换后

var a = 1;console.log(a);
  • scope.getBinding(name) 获取当前所有绑定

  • scope.getBinding(name).referenced 绑定是否被引用

  • scope.getBinding(name).constantViolations 获取当前所有绑定修改

  • scope.getBinding(name).referencePaths 获取当前所有绑定路径

代码实现

const t = require("@babel/types");const parser = require("@babel/parser");const traverse = require("@babel/traverse").default;const generate = require("@babel/generator").default;const ast = parser.parse(`var a = 1;var b = 2;console.log(a);`);traverse(ast, {  VariableDeclaration(path) {    const { node } = path;    const { declarations } = node;    // 此处便利可以处理 const a = 1,b = 2; 这种场景    node.declarations = declarations.filter((declaration) => {      const { id } = declaration;      // const { b, c } = a;      if (t.isObjectPattern(id)) {        // path.scope.getBinding(name).referenced 判断变量是否被引用        // 通过filter移除掉没有使用的变量        id.properties = id.properties.filter((property) => {          const binding = path.scope.getBinding(property.key.name);          return !!binding?.referenced;        });        // 如果对象中所有变量都没有被应用,则该对象整个移除        return id.properties.length > 0;      } else {        // const a = 1;        const binding = path.scope.getBinding(id.name);        return !!binding?.referenced;      }    });    // 如果整个定义语句都没有被引用则整个移除    if (node.declarations.length === 0) {      path.remove();    }  },});console.log(generate(ast).code);

7、了解基本处理流程之后,我们就来看下最终的代码实现吧

const t = require("@babel/types");const parser = require("@babel/parser");const traverse = require("@babel/traverse").default;const generate = require("@babel/generator").default;const ast = parser.parse(  `import vue from 'vue';  var a = 1;var b = 2;var { test1, test2 } = { test1: 1, test2: 2 };function c(){}function d(){}d();console.log(a, test1);`,  {    sourceType: "module",  });traverse(ast, {  // 处理 const var let  VariableDeclaration(path) {    const { node } = path;    const { declarations } = node;    node.declarations = declarations.filter((declaration) => {      const { id } = declaration;      if (t.isObjectPattern(id)) {        id.properties = id.properties.filter((property) => {          const binding = path.scope.getBinding(property.key.name);          return !!binding?.referenced;        });        return id.properties.length > 0;      } else {        const binding = path.scope.getBinding(id.name);        return !!binding?.referenced;      }    });    if (node.declarations.length === 0) {      path.remove();    }  },  // 处理 import  ImportDeclaration(path) {    const { node } = path;    const { specifiers } = node;    if (!specifiers.length) {      return;    }    node.specifiers = specifiers.filter((specifier) => {      const { local } = specifier;      const binding = path.scope.getBinding(local.name);      return !!binding?.referenced;    });    if (node.specifiers.length === 0) {      path.remove();    }  },  // 处理 function  FunctionDeclaration(path) {    const { node } = path;    const { id } = node;    const binding = path.scope.getBinding(id.name);    if (!binding?.referenced) {      path.remove();    }  },});console.log(generate(ast).code);

8、vscode 设置我们的插件只支持 js 文件的限制

因为我们现在实现是针对 js 文件的,所以打开其他类型的文件我们可以让我们的快捷键失效。我们可以修改package.jsonpackage.json

{  "contributes": {    "commands": [      {        "command": "rm-unuse-var.remove",        "title": "Hello World"      }    ],    "keybindings": [      {        "command": "rm-unuse-var.remove",        "key": "ctrl+6",        "mac": "cmd+6",        "when": "resourceLangId == javascript"      }    ]  }}

9、整合到我们前面创建的项目中去

此处省略...相信看了上面这些介绍大家已经完全有能力自己整合了

10、打包发布插件

打包我们可以vsce工具

全局安装 vsce

# npmnpm i vsce -g# yarnyarn global add vsce

打包插件

打包前先修改 README.md 文件否则会报错

vsce package

执行完毕之后会生成一个.vsix 文件

如果要在本地 vscode 使用可以直接导入

如果要发布到市场的话,我们需要先注册账号 https://code.visualstudio.com/api/working-with-extensions/publishing-extension#publishing-extensions

# 登录账号vsce login your-publisher-name# 发布vsce publish

发布成功之后就能在我们的市场上看到了 marketplace.visualstudio.com/items?itemN… 也可以在 vscode 中搜索打我们的插件

以上就是vscode+babel然后开发一个智能移除未使用变量的插件,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

vscode+babel然后开发一个智能移除未使用变量的插件

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

vscode+babel然后开发一个智能移除未使用变量的插件

本篇文章给大家分享的是有关vscode+babel然后开发一个智能移除未使用变量的插件,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。vscode 已经成为前端不可缺失的开发工具
2023-06-22

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录