内容简介:element-ui的按需引入的配置:三步下来就能方便的使用按需引入的功能了。其中的原理是什么?babel-plugin-component在其中做了什么?
element-ui的按需引入的配置: 文档地址
npm install babel-plugin-component -D
{ "presets": [["es2015", { "modules": false }]], "plugins": [ [ "component", { "libraryName": "element-ui", "styleLibraryName": "theme-chalk" } ] ] }
import Vue from 'vue'; import { Button, Select } from 'element-ui'; import App from './App.vue'; Vue.component(Button.name, Button); Vue.component(Select.name, Select);
三步下来就能方便的使用按需引入的功能了。
其中的原理是什么?babel-plugin-component在其中做了什么?
探究处理过程
首先新建一个demo,使用最简化的配置, demo地址 。
demo中只用了四种钩子:
Program:第一个访问的节点,初始化数据。
ImportDeclaration:处理import import { Button, Select } from 'element-ui';
CallExpression:函数执行会访问到,处理 Vue.component(Button.name, Button);
MemberExpression:处理对象访问, Select.name
。
总结一下处理的过程:
第一步
在Program初始化specified等数据,在处理当前文件的过程中这些数据作为全局使用。
第二步
在 ImportDeclaration 里将收集import的变量,比如Button,Select等
import { Button, Select } from 'element-ui'
将变量存储到specified中,这个specified会作为后面处理AST的判断条件
specified[spec.local.name] = spec.imported.name
第三步
在CallExpression中,根据是否使用到Button等会在AST添加节点,这些节点会转换为下面的代码:
import button form "element-ui/lib/button"
添加节点这个环节使用到 @babel/helper-module-imports
中的helper方法addSideEffect,addDefault,简化了手动操作。
简单介绍一下helper-module-imports: 文档链接
调用addSideEffect方法能够生成类似 import "source"
的代码,适合添加css等资源。
调用addDefault方法能够生成类似 import _default from "source"
的代码,适合添加js。
上面三步之后,想要的AST就构建完成了。以demo为例,源代码:
import { Button } from 'element-ui'; Vue.component(Button.name,Button)
执行npm run build ,babel处理之后的代码是:
var _button = _interopRequireDefault(require("element-ui/lib/theme-chalk/button.css")); require("element-ui/lib/theme-chalk/base.css"); var _button2 = _interopRequireDefault(require("element-ui/lib/button")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } Vue.component(_button2["default"].name, _Button);
可以看到自动引入了css require("element-ui/lib/theme-chalk/base.css")
,引入element-ui不见了,增加了 require("element-ui/lib/button")
需要解释一下,上面的import变成了require是因为babel中presets-env的影响;同理_interopRequireDefault也是。
如果在babel.config.json设置 modules:false
结果将是下面的样子:
import _Button2 from "element-ui/lib/theme-chalk/button.css"; import "element-ui/lib/theme-chalk/base.css"; import _Button from "element-ui/lib/button"; Vue.component(_Button.name, _Button); // 看起来顺眼多了
版本问题
在自己检查代码时发现第一个demo的结果 Vue.component(_button2["default"].name, _Button);
中的_Button是一个错误,代码中没有这个引用,执行起来肯定是要报错的;仔细查看了plugin.js并没有发现问题。当换成直接引入babel-plugin-component的时候就没有了问题,通过对比终于发现@babel/helper-module-imports的版本不同,
- babel-plugin-component 内部node_modules中依赖的 @babel/helper-module-imports 版本7.0.0
- 跟随helper-module-transforms一起安装的是7.10.4
切换到版本7.0.0就可以了。
解决方案 一
版本问题能够通过修改plugin.js来解决么?看下面的代码:
function importMethod(methodName, file, opts) { if (!selectedMethods[methodName]) { .... selectedMethods[methodName] = addDefault(file.path, path, { nameHint: methodName }); .... } // .... return selectedMethods[methodName]; }
在对 Vue.component(Button.name, Button)
的访问中需要对参数Button做两次处理,都需要执行到importMethod方法,methodName的值就是 "Button"
,按照执行逻辑两次执行返回的是同一的对象:
{ type:"Identifier", name:"_Button" }
生成代码的时候应该是 Vue.component(_button2["default"].name, _button2["default"])
,这里却好像把第二个_Button给忘了,猜测难道此处的引用传值导致的么?
考虑到通过一个简单的对象能生成 _button2["default"]
,说明自己也可以创建一个对象生成对应的代码,于是就简单的deepClone一下selectedMethods[methodName],试过之后果然可以,此处并没有查找到真正的原因,只作为探索,代码如下:
function importMethod(methodName, file, opts) { if (!selectedMethods[methodName]) { .... selectedMethods[methodName] = addDefault(file.path, path, { nameHint: methodName }); .... } // .... // 此处的t是types,带有一个cloneDeep的方法 return t.cloneDeep(selectedMethods[methodName]); }
解决方案二:
其实在打断点的时候发现,最终生成生成的AST是正确的,错在代码生成的阶段,经过尝试发现直接把 modules:false
就可以避免问题。一般来说我们都要把babel的模块处理取消掉,由webpack来处理模块打包,所以这个方案更加合适。
结束
查看有哪些钩子 : 地址
babel中插件的执行顺序: 插件执行顺序 :
本文只介绍了四个钩子,原插件还使用了IfStatement,ConditionalExpression,LogicalExpression,VariableDeclarator,Property,ArrayExpression,AssignmentExpression七个钩子,这几个钩子主要是处理特殊的情况,暂时还未遇到。
最后如有错误之处,望指正
以上所述就是小编给大家介绍的《从element-ui按需引入去探索》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- 从探索到落地,手淘引入 Swift “历险记”
- Python算法引入
- Java内存泄漏的引入
- React 中引入 Angular 组件
- TensorFlow 引入 Swift 会怎样?
- 现有iOS工程引入Flutter
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
数据结构与算法分析
Frank.M.Carrano / 金名 / 清华大学出版社 / 2007-11 / 98.00元
“数据结构”是计算机专业的基础与核心课程之一,Java是现今一种热门的语言。本书在编写过程中特别考虑到了面向对象程序设计(OOP)的思想与Java语言的特性。它不是从基于另一种程序设计语言的数据结构教材简单地“改编”而来的,因此在数据结构的实现上更加“地道”地运用了Java语言,并且自始至终强调以面向对象的方式来思考、分析和解决问题。 本书是为数据结构入门课程(通常课号是CS-2)而编写的教......一起来看看 《数据结构与算法分析》 这本书的介绍吧!
Markdown 在线编辑器
Markdown 在线编辑器
HEX CMYK 转换工具
HEX CMYK 互转工具