-
Notifications
You must be signed in to change notification settings - Fork 152
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
分享小技巧:实现在浏览器中import内联JS模块 #1142
Comments
分享小技巧:实现在浏览器中import内联JS模块 by 字节前端 ByteFE现代浏览器支持了ES Modules[1],也就是浏览器原生支持的 JavaScript 模块化方案。虽然考虑兼容性,我们还很少能够把 ES Modules 用于生产环境,但是在开发、测试、学习的场景中,ES Modules 发挥了越来越大的作用,比如构建工具Vite[2],就利用 ES Modules 来快速提供开发调试环境。React 和 Vue 框架的学习中,也都可以利用 ES Modules 不用安装本地构建工具,直接在浏览器上体验这些现代框架。 不过 ES Modules 有个局限性,就是它在浏览器里能够 import 指定 URL 的模块化 JS 代码,但是不能 import 自身 HTML 文件里的模块,比如:
我们没有办法做到下面这种:
但是如果能实现这种 inline-import,其实还挺有用的,这就意味着我们可以在像 CodePen 这样简单的 Playground 环境中使用多个 JavaScript 模块,而不用把它们先发布成在线的 JS 文件再 import。 不过要实现 inline-import,也不是那么容易。 思路上,我们可以借助Blob[3]对象来实现,Blob 对象有一些神奇的能力,我在前端冷知识系列中分享过一篇文章《超好用的 Blob 对象!》[4],有兴趣的同学可以去看一下。 言归正传,我们可以实现一个函数,将一段 JavaScript 文本创建成 Blob 对象,并返回 Blob 对象的 URL。
接着我们实现一个 inlineImport 函数:
上面这段代码不复杂,结合 getBlobURL,其核心就是从标签
这样实现可以解决大部分问题,但是用起来还是不爽,因为这样只能动态 import。事实上,我们希望也能够以静态的方式 import,比如 实际上这个也是可以实现的,要用到现代浏览器的另一个特性,importmap。 importmap 本来是为了解决 ES Modules 引入模块的别名问题,比如我们觉得下面的代码写得不爽,因为 import 的 URL 太长了。
可以改成:
也就是在前面加一个 不过要注意,importmap 使用有限制,首先页面上只能有一个 那么,我们接着就可以利用生成 importmap 的思路来实现静态的 inline-import 了:
这个函数的内容看起来稍微多一些,主要是处理 importmap 的规则,如果页面上已经有 importmap 标签,就不能再创建 importmap 了,要抛出异常,另外用户确实需要自己创建 importmap,我们可以让用户用 有了这个 setup 方法之后,我们已经可以用静态的 import 了,我在代码的最后,如果 script 标签上设置 setup 属性,那么就自动运行 这样我们就可以这么写:
或者要用到自定义的 importmap 的时候可以这么写:
只是需要注意的是, 参考资料ES Modules: https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Guide/Modules [2]Vite: https://vitejs.dev/ [3]Blob: https://developer.mozilla.org/zh-CN/docs/Web/API/Blob [4]《超好用的Blob对象!》: https://github.com/akira-cn/FE_You_dont_know/issues/12 [5]github.com/xitu/inline…: https://github.com/xitu/inline-module |
https://mp.weixin.qq.com/s?__biz=Mzg2ODQ1OTExOA==&mid=2247496539&idx=1&sn=b0972823804fb179c094d1f95ab13d3f
The text was updated successfully, but these errors were encountered: