这个问题要从一个想法说起。
D2Admin 是一个开源的,前端中后台集成方案,原先是基于 vue-cli2,大概是向 vue-cli3 过渡时,
作者老李,想在页面右下角加个 Toggle 点击,跳到当前页面源码对应的 github 页面。
确实很实用的功能,D2Admin 的 Demo 页面太多了,想看某个页面的源码,对于不熟悉项目目录结构的新手很不友好。
这些页面统一为 .vue 组件,那么转换一下:如何获取 vue 单文件自身源码路径?
目前经历了三个方案,最终目标是把自身路径赋值到 this.$options.__source 上。目前方案 3 是最新的。
方案 1 :node + __filename
直接使用 node 中的 __filename 变量:
<template> <h1>{{ $options.__source }}</h1> </template> <script> export default { mounted() { this.$options.__source = __filename } } </script>
因为 webpack 编译时,会把源码文件在内部转为 node 模块,.vue 文件中的 script 内容也被转换了,
其中的 __filename 在编译时被运行,直接得到当前文件自身路径。
使用这个变量还需要在 webpack 配置中启用 node.__filename:
/* vue.config.js */ module.exports = { // ... chainWebpack: config => { // ... config.node .set('__dirname', true) // 同理 .set('__filename', true) } };
缺点
- 要在每个组件里手动赋值,还不能用 mixin
- __filename 得到的路径在部分 .vue 文件下并不准确,路径可能还会带附带 querystring
一开始,坚强的老李用这个方式,给上百个组件手动挂上了路径,但总比手动写死每个路径要好
方案 2 :vue-loader + exposeFilename
在 loader 层面,其实 vue-loader 提供了一个 exposeFilename 选项,只要启用,
就会给每个 .vue 组件带上 this.$options.__file,上面有准确的路径。
这样只需要改 loader 配置:
/* vue.config.js */ module.exports = { // ... chainWebpack: config => { // ... config.module .rule('vue') .use('vue-loader') .loader('vue-loader') .tap(options => { options.exposeFilename = true return options }) } };
开发环境默认是开启 exposeFilename 的。
此时组件内应该直接取 this.$options.__file,内容大致为 src/foo/bar.vue。
缺点
为了安全,vue-loader 在生产环境将 __file 赋值为文件名,非路径名,详见文档
后来得知这个方法,老李就第一时间改代码,删了方案 1 中的所有附加代码,结果发现生产环境结果不一致,翻车了orz
方案 3 :loader + Custom Block
既然方案 2 不让在生产环境用,那就自己写 loader 去加上这个源码路径,这里采用了Custom Block。
这个方法是慢慢调试自定义的 loader 摸索出来的,先在 vue-loader 之前加自定义的 loader A,
去注入 Custom Block 代码,再在全局加入一个针对该 Custom Block 的 loader B。
这里将方案封装,在 chainWebpack 中调用即可:
/* vue.config.js */ const VueFilenameInjector = require('./path/to/vue-filename-injector') module.exports = { chainWebpack: config => { VueFilenameInjector(config, { propName: '__source' // default }) } }
源码参考:@d2-projects/d2-advance/tools/vue-filename-injector
. └── vue-filename-injector ├── README.md ├── index.js └── src ├── index.js └── lib ├── config.js ├── injector.js └── loader.js
vue-filename-injector/index.js:
const { blockName } = require('./lib/config.js') // for chainWebpack module.exports = function(config, options) { // 注入 config.module .rule('vue') .use('vue-filename-injector') .loader(require.resolve('./lib/injector.js')) .options(options) .after('vue-loader') // 不知为何 .before() 不是预期的结果,这里就绕了一下 .end() // 解析 config.module .rule('') .resourceQuery(new RegExp(`blockType=${blockName}`)) .use('vue-filename-injector-loader') .loader(require.resolve('./lib/loader.js')) .end() }
vue-filename-injector/lib/config.js:
const defaultPropName = '__source' const blockName = 'vue-filename-injector' module.exports = { defaultPropName, blockName }
vue-filename-injector/lib/injector.js,源码部分来自 vue-loader:
const path = require('path') const loaderUtils = require('loader-utils') const { blockName, defaultPropName } = require('./config.js') module.exports = function (content /*, map, meta */) { const loaderContext = this const { rootContext, resourcePath } = loaderContext const context = rootContext || process.cwd() const options = loaderUtils.getOptions(loaderContext) || {} const rawShortFilePath = path .relative(context, resourcePath) .replace(/^(\.\.[\/\\])+/, '') const propName = options.propName || defaultPropName content += ` <${blockName}> export default function (Component) { Component.options.${propName} = ${JSON.stringify(rawShortFilePath.replace(/\\/g, '/'))} } </${blockName}> ` return content }
vue-filename-injector/lib/loader.js:
module.exports = function(source, map) { this.callback(null, source, map) }
相关仓库
可进入预览页面查看效果,在右下角有 Toggle
github.com/d2-projects… (可能还在翻车中)
github.com/d2-projects…
总结
目前看来,用自定义 loader 去注入代码是最便捷的方案,不用在每个组件中手写重复的代码。
由于 vue-cli3 采用 chainWebpack,加上个人对 webpack 的理解更是不深,暂时采用方案 3。
以上所述是小编给大家介绍的如何获取vue单文件自身源码路径详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
P70系列延期,华为新旗舰将在下月发布
3月20日消息,近期博主@数码闲聊站 透露,原定三月份发布的华为新旗舰P70系列延期发布,预计4月份上市。
而博主@定焦数码 爆料,华为的P70系列在定位上已经超过了Mate60,成为了重要的旗舰系列之一。它肩负着重返影像领域顶尖的使命。那么这次P70会带来哪些令人惊艳的创新呢?
根据目前爆料的消息来看,华为P70系列将推出三个版本,其中P70和P70 Pro采用了三角形的摄像头模组设计,而P70 Art则采用了与上一代P60 Art相似的不规则形状设计。这样的外观是否好看见仁见智,但辨识度绝对拉满。
更新日志
- 魔兽世界wlk暗牧一键输出宏是什么 wlk暗牧一键输出宏介绍
- 群星.1996-红不让台语原唱2辑【福和唱片】【WAV+CUE】
- 郭书瑶.2009-爱的抱抱(EP)【种子音乐】【FLAC分轨】
- 郑瑞芬.1989-BE.MY.BABY【现代】【WAV+CUE】
- 花钱请人每周放30万只不咬人的蚊子 防治登革热传播
- 饭制《第一后裔》丧尸版弗蕾娜
- 贝克汉姆亲临!2024FC品类游戏嘉年华圆满落幕
- 「命轨爻错之翼」风之翼发放说明
- 《原神》前瞻特别节目回顾长图
- 米游币抽抽乐-原神专场现已开启!
- 黑鸭子2001《风情中国HQCD》[日本版][WAV+CUE]
- 陈杰洲1990-成人礼[滚石][WAV+CUE]
- MarkAanderud-HandsFree(2024)[24-44,1]FLAC
- 孙露《观心》1:1母盘直刻限量版[低速原抓WAV+CUE][361M]
- 钟志刚《汽车DJ玩主》[低速原抓WAV+CUE][1G]