preface
I think we are familiar with the Scope CSS of Vue, but speaking of the principle of Vue Scope CSS implementation, many people should say is not to add attributes to HTML, CSS ️?
That’s true, but that’s just what the final Scope CSS renders. And how does that work? I don’t think there are many people who can answer that.
So, back to today’s article, I will focus on the following three points, starting from the final presentation results of Vue’s Scope CSS, and explain the underlying principle of its implementation in a simple way:
- What is Scope CSS
- Vue-loader processing components (.vue files)
- ScopeId is applied to generate HTML attributes in Patch phase
1 What is Scope CSS
Scope CSS is an integral part of componentization. Scope CSS allows us to define CSS in components without contamination. For example, we define a component in Vue:
<! -- App.vue -->
<template>
<div class="box">scoped css</div>
</template>
<script>
export default {};
</script>
<style scoped>
.box {
width: 200px;
height: 200px;
background: #aff;
}
</style>
Copy the code
Normally, in the development environment, our components would be processed by vue-Loader and rendered to the page with the framework code at runtime. Accordingly, their corresponding HTML and CSS would look like this:
HTML part:
<div data-v-992092a6>scoped css</div>
Copy the code
The CSS part:
.box[data-v-992092a6] {
width: 200px;
height: 200px;
background: #aff;
}
Copy the code
It can be seen that the nature of Scope CSS is based on the attributes of HTML and CSS selector, by adding data-V-XXXX attributes to THE HTML tag and CSS selector respectively.
2 Vue-loader processing component (.vue file)
Earlier, we also mentioned that a component (.vue file) is handled first by vue-loader in a development environment. So, for Scope CSS, vue-Loader does three things:
- Parse the component and extract it
template
,script
,style
Corresponding code block - Construct and derive
export
Component instance, binding ScopId to the options of the component instance - right
style
ScopId is applied to generate selector properties
Vue loader is used to process.vue files. Vue loader is used to process.vue files
However, vue-loader has so many capabilities mainly because the underlying vue-loader uses the official vUE package @vue/component-compiler-utils. It provides three capabilities: parsing components (.vue files), compiling templates, and compiling styles.
Let’s see how vue-loader uses @vue/component-compiler-utils to parse components to extract templates, scripts, and styles.
2.1 Extracting template, script and style
The vue-loader extracts template, script, and style using the parse method of the @vue/component-compiler-utils package. The corresponding code (pseudo-code) will look like this:
// vue-loader/lib/index.js
const { parse } = require("@vue/component-compiler-utils");
module.exports = function (source) {
const loaderContext = this;
const { sourceMap, rootContext, resourcePath } = loaderContext;
const sourceRoot = path.dirname(path.relative(context, resourcePath));
const descriptor = parse({
source,
compiler: require("vue-template-compiler"),
filename,
sourceRoot,
needMap: sourceMap,
});
};
Copy the code
Let’s go through this code point by point. First, we get the current loaderContext, which contains compiler, compilation, and so on, the core object of the WebPack packaging process.
Next, build the file resource entry sourceRoot, which in general refers to the SRC file directory and is mainly used to build source-Map.
Finally, the source (component code) is parsed using the parse method provided by @vue/component-compiler-utils. Here, let’s look at a few arguments to the parse method:
soruce
The source code block, where the component corresponds to the code, containstemplate
,style
,script
compiler
Compile the core object, which is a CommonJS module (vue-template-compiler),parse
The method is used internallyparseComponent
Method to parse the componentfilename
The file name of the current component, for exampleApp.vue
sourceRoot
File resource entry for buildingsource-map
useneedMap
If you needsource-map
.parse
Methods are internally based onneedMap
The value of (true
或false
By default,true
) to determine whetherscript
,style
The correspondingsource-map
The parse method returns an object containing the template, style, and script blocks to Desciptor.
What you can see, then, is that the vue-loader is almost outsourced to the vUe-supplied package. And, I think this time will certainly have students ask: these and Vue Scope CSS how much money ️?
It has a lot to do with it! Vue’s Scope CSS is not a piece of paper. It is implemented only after the components are parsed, and then the template and style sections are processed separately.
So, obviously we’ve finished parsing the component at this point. Next, you need to construct and export the component instance ~
2.2 Construct and export component instances
After parsing components, vue-loader generates template, script, and style import statements, invoks normalizer to normalize components, and splices them into code strings:
let templateImport = `var render, staticRenderFns`;
if (descriptor.template) {
// Construct the import statement for template
}
let scriptImport = `var script = {}`;
if (descriptor.script) {
// Construct a script import statement
}
let stylesCode = ` `;
if (descriptor.styles.length) {
// Construct the style import statement
}
let code =
`
${templateImport}
${scriptImport}
${stylesCode}
import normalizer from ${stringifyRequest(`!${componentNormalizerPath}`)}
var component = normalizer(
script,
render,
staticRenderFns,
${hasFunctional ? `true` : `false`}.The ${/injectStyles/.test(stylesCode) ? `injectStyles` : `null`}.${hasScoped ? JSON.stringify(id) : `null`}.${isServer ? JSON.stringify(hash(request)) : `null`}
${isShadow ? `,true` : ` `}
)
`.trim() + `\n`;
Copy the code
TemplateImport, scriptImport, stylesCode, stylesCode, templateImport, scriptImport, stylesCode look like this:
import {
render,
staticRenderFns,
} from "./App.vue? vue&type=template&id=7ba5bd90&scoped=true&";
import script from "./App.vue? vue&type=script&lang=js&";
// Export compatible naming methods
export * from "./App.vue? vue&type=script&lang=js&";
import style0 from "./App.vue? vue&type=style&index=0&id=7ba5bd90&scoped=true&lang=css&";
Copy the code
: template and style import import statements have a common section id=7ba5bd90&scoped=true, This means that the template and style of the component need Scope CSS and the scopeId is 7ba5bd90.
Of course, this is just the first step in generating Scope CSS for subsequent template and style compilations! The normalizer method is then called to normalize the component:
import normalizer from ! "" . /node_modules/vue-loader/lib/runtime/componentNormalizer.js";
var component = normalizer(
script,
render,
staticRenderFns,
false.null."7ba5bd90".null
);
export default component.exports;
Copy the code
NormalizeComponent < normalizeComponent > normalizeComponent < normalizeComponent > normalizeComponent < normalizeComponent
I’m sure you’ve noticed that scopeId is passed as an argument to normalizeComponent to bind scopeId to the options of the component instance. So, let’s look at the normalizeComponent method (pseudocode) :
function normalizeComponent (
scriptExports,
render,
staticRenderFns,
functionalTemplate,
injectStyles,
scopeId,
moduleIdentifier, /* server only */
shadowMode /* vue-cli only */
) {...var options = typeof scriptExports === 'function'
? scriptExports.options
: scriptExports
// scopedId
if (scopeId) {
options._scopeId = 'data-v-' + scopeId
}
...
}
Copy the code
Options. _scopeId equals data-V-7ba5bd90, which is used to add data-V-7ba5bd90 to the HTML tag of the current component during patch. Therefore, this is why the template forms the real place with the scopeId!
2.3 Compile Style Style, apply ScopId to generate selector properties
After constructing the import statement corresponding to the Style, the Pitching loader is processed by the vuE-Loader internal because the query in the import statement contains the VUE. The Pitching Loader would rewrite the import statement to concatenate the inline Loader and this would look like this:
export * from '" -! . /node_modules/vue-style-loader/index.js?? ref--6-oneOf-1-0 ! . /node_modules/css-loader/dist/cjs.js?? ref--6-oneOf-1-1 ! . /node_modules/vue-loader/lib/loaders/stylePostLoader.js ! . /node_modules/postcss-loader/src/index.js?? ref--6-oneOf-1-2 ! . /node_modules/cache-loader/dist/cjs.js?? ref--0-0 ! . /node_modules/vue-loader/lib/index.js?? vue-loader-options! ./App.vue? vue&type=style&index=0&id=7ba5bd90&scoped=true&lang=css&" '
Copy the code
Webpack then resolves the loaders required by the module, which apparently resolves six:
[{loader: "vue-style-loader".options: "? ref--6-oneOf-1-0" },
{ loader: "css-loader".options: "? ref--6-oneOf-1-1" },
{ loader: "stylePostLoader".options: undefined },
{ loader: "postcss-loader".options: "? ref--6-oneOf-1-2" },
{ loader: "cache-loader".options: "? ref--0-0" },
{ loader: "vue-loader".options: "? vue-loader-options"}]Copy the code
At this point, WebPack will execute the six Loaders (and of course the parsing module itself). Also, Normal loaders in webpack.config.js are ignored (vue-style-loader also ignores pre-loaders).
If you don’t know about inline Loader, you can check out this article “Webpack Advanced”. 10 q – loader
For Scope CSS, the core is stylePostLoader. Let’s take a look at the stylePostLoader definition:
const { compileStyle } = require("@vue/component-compiler-utils");
module.exports = function (source, inMap) {
const query = qs.parse(this.resourceQuery.slice(1));
const { code, map, errors } = compileStyle({
source,
filename: this.resourcePath,
id: `data-v-${query.id}`.map: inMap,
scoped:!!!!! query.scoped,trim: true});if (errors.length) {
this.callback(errors[0]);
} else {
this.callback(null, code, map); }};Copy the code
From the stylePostLoader definition, we know that it uses the method compileStyle provided by @vue/component-compiler-utils to compile component style. Data-v -${query.id}, data-v-7ba5bd90, is the scopeId attribute of the selector declared in style.
Inside compileStyle, we use the familiar PostCSS to compile the style code and construct the scopeId property of the selector. As for how to use postCSS to complete this process, I won’t go into details here
ScopeId was applied to generate HTML attributes in Patch stage
If you remember from 3.2 constructing and exporting a component instance, we explained that binding _scopeId to the options of the component instance is the key to implementing the Scope of the template. However, what we didn’t say at the time was how the _scopeId was applied to the element on the template.
If you’re looking for the answer in vue-loader or @vue/component-compiler-utils code, I can tell you that you won’t find it in ten thousand years! Because the actual application of _scopeId takes place in the framework code of the Vue runtime (surprise ).
For those of you who have seen the Vue template compilation process, you probably know that the template is compiled into the render function, which then creates the corresponding VNode, and finally renders the DOM into the actual page:
The process from VNode to real DOM is completed by patch method. Let’s say we’re rendering DOM for the first time, which hits isUndef(oldVnode) true in patch:
function patch (oldVnode, vnode, hydrating, removeOnly) {
if (isUndef(oldVnode)) {
// empty mount (likely as component), create new root element
isInitialPatch = true
createElm(vnode, insertedVnodeQueue)
}
}
Copy the code
Because the DOM is rendered for the first time, there is no oldVnode
As you can see, the createElm method is executed. The createElm method creates the actual DOM corresponding to the VNode, and it does one important thing: it calls the setScope method and applies _scopeId to generate the data-V-xxx property on the DOM! Corresponding code (pseudocode) :
// packages/src/core/vdom/patch.js
function createElm(vnode, insertedVnodeQueue, parentElm, refElm, nested, ownerArray, index) {... setScope(vnode); . }Copy the code
In the setScope method, the component instance’s options._scopeId is added to the DOM as a property, generating a property named data-v-xxx on the HTML tag in the template. In addition, this process is accomplished by the Vue wrapped utility function nodeops.setStylescope, which essentially calls the DOM object setAttribute method:
// src/platforms/web/runtime/node-ops.js
export function setStyleScope (node: Element, scopeId: string) {
node.setAttribute(scopeId, ' ')}Copy the code
conclusion
If anyone has looked up articles on vue-Loader and Scope CSS on the web, Many articles have said that applying scopeId to the compilerTemplate method of the @vue/component-compiler-utils package generates the attributes of the HTML tag in the template. However, reading this article, we will find that there is no relationship between the two at all (except in the case of SSR)!
Also, I think students have noticed that the code for the Vue runtime framework mentioned in this article is Vue 2.x (not Vue3). Therefore, interested students can use the route provided in Vue3 Scope CSS process, I believe you will gain full . Finally, if there is any improper expression or mistake in this article, please raise Issue ~
Thumb up
If you get something from this post, please give me a “like”. This will be my motivation to keep sharing. Thank you
My name is Wu Liu. I like innovation and source Code manipulation, and I focus on source Code (Vue 3, Vite), front-end engineering, cross-end and other technology learning and sharing. Welcome to follow my wechat official account: Code Center.