You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
225 lines
7.0 KiB
225 lines
7.0 KiB
3 years ago
|
const qs = require('querystring')
|
||
|
const id = 'vue-loader-plugin'
|
||
|
const NS = 'vue-loader'
|
||
|
const BasicEffectRulePlugin = require('webpack/lib/rules/BasicEffectRulePlugin')
|
||
|
const BasicMatcherRulePlugin = require('webpack/lib/rules/BasicMatcherRulePlugin')
|
||
|
const RuleSetCompiler = require('webpack/lib/rules/RuleSetCompiler')
|
||
|
const UseEffectRulePlugin = require('webpack/lib/rules/UseEffectRulePlugin')
|
||
|
|
||
|
const objectMatcherRulePlugins = []
|
||
|
try {
|
||
|
const ObjectMatcherRulePlugin = require('webpack/lib/rules/ObjectMatcherRulePlugin')
|
||
|
objectMatcherRulePlugins.push(
|
||
|
new ObjectMatcherRulePlugin('assert', 'assertions'),
|
||
|
new ObjectMatcherRulePlugin('descriptionData')
|
||
|
)
|
||
|
} catch (e) {
|
||
|
const DescriptionDataMatcherRulePlugin = require('webpack/lib/rules/DescriptionDataMatcherRulePlugin')
|
||
|
objectMatcherRulePlugins.push(new DescriptionDataMatcherRulePlugin())
|
||
|
}
|
||
|
|
||
|
const ruleSetCompiler = new RuleSetCompiler([
|
||
|
new BasicMatcherRulePlugin('test', 'resource'),
|
||
|
new BasicMatcherRulePlugin('mimetype'),
|
||
|
new BasicMatcherRulePlugin('dependency'),
|
||
|
new BasicMatcherRulePlugin('include', 'resource'),
|
||
|
new BasicMatcherRulePlugin('exclude', 'resource', true),
|
||
|
new BasicMatcherRulePlugin('conditions'),
|
||
|
new BasicMatcherRulePlugin('resource'),
|
||
|
new BasicMatcherRulePlugin('resourceQuery'),
|
||
|
new BasicMatcherRulePlugin('resourceFragment'),
|
||
|
new BasicMatcherRulePlugin('realResource'),
|
||
|
new BasicMatcherRulePlugin('issuer'),
|
||
|
new BasicMatcherRulePlugin('compiler'),
|
||
|
...objectMatcherRulePlugins,
|
||
|
new BasicEffectRulePlugin('type'),
|
||
|
new BasicEffectRulePlugin('sideEffects'),
|
||
|
new BasicEffectRulePlugin('parser'),
|
||
|
new BasicEffectRulePlugin('resolve'),
|
||
|
new BasicEffectRulePlugin('generator'),
|
||
|
new UseEffectRulePlugin()
|
||
|
])
|
||
|
|
||
|
class VueLoaderPlugin {
|
||
|
apply (compiler) {
|
||
|
const normalModule = compiler.webpack
|
||
|
? compiler.webpack.NormalModule
|
||
|
: require('webpack/lib/NormalModule')
|
||
|
// add NS marker so that the loader can detect and report missing plugin
|
||
|
compiler.hooks.compilation.tap(id, compilation => {
|
||
|
const normalModuleLoader = normalModule.getCompilationHooks(compilation).loader
|
||
|
normalModuleLoader.tap(id, loaderContext => {
|
||
|
loaderContext[NS] = true
|
||
|
})
|
||
|
})
|
||
|
|
||
|
const rules = compiler.options.module.rules
|
||
|
let rawVueRules
|
||
|
let vueRules = []
|
||
|
|
||
|
for (const rawRule of rules) {
|
||
|
// skip rules with 'enforce'. eg. rule for eslint-loader
|
||
|
if (rawRule.enforce) {
|
||
|
continue
|
||
|
}
|
||
|
// skip the `include` check when locating the vue rule
|
||
|
const clonedRawRule = Object.assign({}, rawRule)
|
||
|
delete clonedRawRule.include
|
||
|
|
||
|
const ruleSet = ruleSetCompiler.compile([{
|
||
|
rules: [clonedRawRule]
|
||
|
}])
|
||
|
vueRules = ruleSet.exec({
|
||
|
resource: 'foo.vue'
|
||
|
})
|
||
|
|
||
|
if (!vueRules.length) {
|
||
|
vueRules = ruleSet.exec({
|
||
|
resource: 'foo.vue.html'
|
||
|
})
|
||
|
}
|
||
|
if (vueRules.length > 0) {
|
||
|
if (rawRule.oneOf) {
|
||
|
throw new Error(
|
||
|
`[VueLoaderPlugin Error] vue-loader 15 currently does not support vue rules with oneOf.`
|
||
|
)
|
||
|
}
|
||
|
rawVueRules = rawRule
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
if (!vueRules.length) {
|
||
|
throw new Error(
|
||
|
`[VueLoaderPlugin Error] No matching rule for .vue files found.\n` +
|
||
|
`Make sure there is at least one root-level rule that matches .vue or .vue.html files.`
|
||
|
)
|
||
|
}
|
||
|
|
||
|
// get the normlized "use" for vue files
|
||
|
const vueUse = vueRules.filter(rule => rule.type === 'use').map(rule => rule.value)
|
||
|
|
||
|
// get vue-loader options
|
||
|
const vueLoaderUseIndex = vueUse.findIndex(u => {
|
||
|
return /^vue-loader|(\/|\\|@)vue-loader/.test(u.loader)
|
||
|
})
|
||
|
|
||
|
if (vueLoaderUseIndex < 0) {
|
||
|
throw new Error(
|
||
|
`[VueLoaderPlugin Error] No matching use for vue-loader is found.\n` +
|
||
|
`Make sure the rule matching .vue files include vue-loader in its use.`
|
||
|
)
|
||
|
}
|
||
|
|
||
|
// make sure vue-loader options has a known ident so that we can share
|
||
|
// options by reference in the template-loader by using a ref query like
|
||
|
// template-loader??vue-loader-options
|
||
|
const vueLoaderUse = vueUse[vueLoaderUseIndex]
|
||
|
vueLoaderUse.ident = 'vue-loader-options'
|
||
|
vueLoaderUse.options = vueLoaderUse.options || {}
|
||
|
|
||
|
// for each user rule (expect the vue rule), create a cloned rule
|
||
|
// that targets the corresponding language blocks in *.vue files.
|
||
|
const refs = new Map()
|
||
|
const clonedRules = rules
|
||
|
.filter(r => r !== rawVueRules)
|
||
|
.map((rawRule) => cloneRule(rawRule, refs))
|
||
|
|
||
|
// fix conflict with config.loader and config.options when using config.use
|
||
|
delete rawVueRules.loader
|
||
|
delete rawVueRules.options
|
||
|
rawVueRules.use = vueUse
|
||
|
|
||
|
// global pitcher (responsible for injecting template compiler loader & CSS
|
||
|
// post loader)
|
||
|
const pitcher = {
|
||
|
loader: require.resolve('./loaders/pitcher'),
|
||
|
resourceQuery: query => {
|
||
|
if (!query) { return false }
|
||
|
const parsed = qs.parse(query.slice(1))
|
||
|
return parsed.vue != null
|
||
|
},
|
||
|
options: {
|
||
|
cacheDirectory: vueLoaderUse.options.cacheDirectory,
|
||
|
cacheIdentifier: vueLoaderUse.options.cacheIdentifier
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// replace original rules
|
||
|
compiler.options.module.rules = [
|
||
|
pitcher,
|
||
|
...clonedRules,
|
||
|
...rules
|
||
|
]
|
||
|
}
|
||
|
}
|
||
|
|
||
|
let uid = 0
|
||
|
function cloneRule (rawRule, refs) {
|
||
|
const rules = ruleSetCompiler.compileRules(`clonedRuleSet-${++uid}`, [{
|
||
|
rules: [rawRule]
|
||
|
}], refs)
|
||
|
let currentResource
|
||
|
|
||
|
const conditions = rules[0].rules
|
||
|
.map(rule => rule.conditions)
|
||
|
// shallow flat
|
||
|
.reduce((prev, next) => prev.concat(next), [])
|
||
|
|
||
|
// do not process rule with enforce
|
||
|
if (!rawRule.enforce) {
|
||
|
const ruleUse = rules[0].rules
|
||
|
.map(rule => rule.effects
|
||
|
.filter(effect => effect.type === 'use')
|
||
|
.map(effect => effect.value)
|
||
|
)
|
||
|
// shallow flat
|
||
|
.reduce((prev, next) => prev.concat(next), [])
|
||
|
|
||
|
// fix conflict with config.loader and config.options when using config.use
|
||
|
delete rawRule.loader
|
||
|
delete rawRule.options
|
||
|
rawRule.use = ruleUse
|
||
|
}
|
||
|
|
||
|
const res = Object.assign({}, rawRule, {
|
||
|
resource: resources => {
|
||
|
currentResource = resources
|
||
|
return true
|
||
|
},
|
||
|
resourceQuery: query => {
|
||
|
if (!query) { return false }
|
||
|
const parsed = qs.parse(query.slice(1))
|
||
|
if (parsed.vue == null) {
|
||
|
return false
|
||
|
}
|
||
|
if (!conditions) {
|
||
|
return false
|
||
|
}
|
||
|
const fakeResourcePath = `${currentResource}.${parsed.lang}`
|
||
|
for (const condition of conditions) {
|
||
|
// add support for resourceQuery
|
||
|
const request = condition.property === 'resourceQuery' ? query : fakeResourcePath
|
||
|
if (condition && !condition.fn(request)) {
|
||
|
return false
|
||
|
}
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
|
})
|
||
|
|
||
|
delete res.test
|
||
|
|
||
|
if (rawRule.rules) {
|
||
|
res.rules = rawRule.rules.map(rule => cloneRule(rule, refs))
|
||
|
}
|
||
|
|
||
|
if (rawRule.oneOf) {
|
||
|
res.oneOf = rawRule.oneOf.map(rule => cloneRule(rule, refs))
|
||
|
}
|
||
|
|
||
|
return res
|
||
|
}
|
||
|
|
||
|
VueLoaderPlugin.NS = NS
|
||
|
module.exports = VueLoaderPlugin
|