webpack/lib/css/CssModulesPlugin.js

304 lines
8.9 KiB
JavaScript
Raw Normal View History

2021-11-30 19:55:51 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
const { ConcatSource } = require("webpack-sources");
const HotUpdateChunk = require("../HotUpdateChunk");
const RuntimeGlobals = require("../RuntimeGlobals");
2021-12-14 23:02:26 +08:00
const CssExportDependency = require("../dependencies/CssExportDependency");
2021-12-01 21:15:19 +08:00
const CssImportDependency = require("../dependencies/CssImportDependency");
2021-12-15 15:34:31 +08:00
const CssLocalIdentifierDependency = require("../dependencies/CssLocalIdentifierDependency");
2021-11-30 20:46:42 +08:00
const CssUrlDependency = require("../dependencies/CssUrlDependency");
const StaticExportsDependency = require("../dependencies/StaticExportsDependency");
2021-11-30 19:55:51 +08:00
const {
compareModulesByPostOrderIndexOrIdentifier
} = require("../util/comparators");
const createSchemaValidation = require("../util/create-schema-validation");
const createHash = require("../util/createHash");
2021-12-03 23:23:09 +08:00
const memoize = require("../util/memoize");
2021-11-30 19:55:51 +08:00
const CssGenerator = require("./CssGenerator");
const CssParser = require("./CssParser");
/** @typedef {import("webpack-sources").Source} Source */
/** @typedef {import("../Chunk")} Chunk */
/** @typedef {import("../Compiler")} Compiler */
/** @typedef {import("../Module")} Module */
2021-12-03 23:23:09 +08:00
const getCssLoadingRuntimeModule = memoize(() =>
require("./CssLoadingRuntimeModule")
);
2021-11-30 19:55:51 +08:00
const getSchema = name => {
const { definitions } = require("../../schemas/WebpackOptions.json");
return {
definitions,
oneOf: [{ $ref: `#/definitions/${name}` }]
};
};
const validateGeneratorOptions = createSchemaValidation(
require("../../schemas/plugins/css/CssGeneratorOptions.check.js"),
() => getSchema("CssGeneratorOptions"),
{
name: "Css Modules Plugin",
baseDataPath: "parser"
}
);
const validateParserOptions = createSchemaValidation(
require("../../schemas/plugins/css/CssParserOptions.check.js"),
() => getSchema("CssParserOptions"),
{
name: "Css Modules Plugin",
baseDataPath: "parser"
}
);
const escapeCss = (str, omitOptionalUnderscore) => {
const escaped = `${str}`.replace(
// cspell:word uffff
/[^a-zA-Z0-9_\u0081-\uffff-]/g,
s => `\\${s}`
);
return !omitOptionalUnderscore && /^[0-9_-]/.test(escaped)
? `_${escaped}`
: escaped;
2021-11-30 19:55:51 +08:00
};
const plugin = "CssModulesPlugin";
class CssModulesPlugin {
/**
* Apply the plugin
* @param {Compiler} compiler the compiler instance
* @returns {void}
*/
apply(compiler) {
compiler.hooks.compilation.tap(
plugin,
(compilation, { normalModuleFactory }) => {
2021-11-30 20:46:42 +08:00
compilation.dependencyFactories.set(
CssUrlDependency,
normalModuleFactory
);
compilation.dependencyTemplates.set(
CssUrlDependency,
new CssUrlDependency.Template()
);
2021-12-15 15:34:31 +08:00
compilation.dependencyTemplates.set(
CssLocalIdentifierDependency,
new CssLocalIdentifierDependency.Template()
);
2021-12-14 23:02:26 +08:00
compilation.dependencyTemplates.set(
CssExportDependency,
new CssExportDependency.Template()
);
2021-12-01 20:27:00 +08:00
compilation.dependencyFactories.set(
CssImportDependency,
normalModuleFactory
);
compilation.dependencyTemplates.set(
CssImportDependency,
new CssImportDependency.Template()
);
2021-11-30 20:46:42 +08:00
compilation.dependencyTemplates.set(
StaticExportsDependency,
new StaticExportsDependency.Template()
);
2021-11-30 19:55:51 +08:00
normalModuleFactory.hooks.createParser
.for("css")
.tap(plugin, parserOptions => {
validateParserOptions(parserOptions);
return new CssParser();
});
2021-12-14 23:02:26 +08:00
normalModuleFactory.hooks.createParser
.for("css/global")
.tap(plugin, parserOptions => {
validateParserOptions(parserOptions);
2021-12-15 15:34:31 +08:00
return new CssParser({
allowPseudoBlocks: false,
allowModeSwitch: false
});
2021-12-14 23:02:26 +08:00
});
normalModuleFactory.hooks.createParser
.for("css/module")
.tap(plugin, parserOptions => {
validateParserOptions(parserOptions);
2021-12-15 15:34:31 +08:00
return new CssParser({
defaultMode: "local"
});
2021-12-14 23:02:26 +08:00
});
2021-11-30 19:55:51 +08:00
normalModuleFactory.hooks.createGenerator
.for("css")
.tap(plugin, generatorOptions => {
validateGeneratorOptions(generatorOptions);
return new CssGenerator();
});
2021-12-14 23:02:26 +08:00
normalModuleFactory.hooks.createGenerator
.for("css/global")
.tap(plugin, generatorOptions => {
validateGeneratorOptions(generatorOptions);
return new CssGenerator();
});
normalModuleFactory.hooks.createGenerator
.for("css/module")
.tap(plugin, generatorOptions => {
validateGeneratorOptions(generatorOptions);
return new CssGenerator();
});
2021-11-30 19:55:51 +08:00
compilation.hooks.contentHash.tap("JavascriptModulesPlugin", chunk => {
const {
chunkGraph,
outputOptions: {
hashSalt,
hashDigest,
hashDigestLength,
hashFunction
}
} = compilation;
if (!CssModulesPlugin.chunkHasCss(chunk, chunkGraph)) return;
const hash = createHash(hashFunction);
if (hashSalt) hash.update(hashSalt);
const modules = this.getOrderedChunkCssModules(chunk, chunkGraph);
for (const module of modules) {
hash.update(chunkGraph.getModuleHash(module, chunk.runtime));
}
const digest = /** @type {string} */ (hash.digest(hashDigest));
chunk.contentHash.css = digest.substr(0, hashDigestLength);
});
compilation.hooks.renderManifest.tap(plugin, (result, options) => {
const { chunkGraph } = compilation;
const { hash, chunk, codeGenerationResults } = options;
if (chunk instanceof HotUpdateChunk) return result;
if (CssModulesPlugin.chunkHasCss(chunk, chunkGraph)) {
result.push({
render: () =>
this.renderChunk({
chunk,
chunkGraph,
2021-12-15 23:46:13 +08:00
codeGenerationResults,
uniqueName: compilation.outputOptions.uniqueName
2021-11-30 19:55:51 +08:00
}),
filenameTemplate: CssModulesPlugin.getChunkFilenameTemplate(
chunk,
compilation.outputOptions
),
pathOptions: {
hash,
runtime: chunk.runtime,
chunk,
contentHashType: "css"
},
identifier: `css${chunk.id}`,
hash: chunk.contentHash.css
});
}
return result;
});
const enabledChunks = new WeakSet();
const handler = (chunk, set) => {
if (enabledChunks.has(chunk)) {
return;
}
enabledChunks.add(chunk);
set.add(RuntimeGlobals.publicPath);
set.add(RuntimeGlobals.getChunkCssFilename);
set.add(RuntimeGlobals.hasOwnProperty);
set.add(RuntimeGlobals.moduleFactoriesAddOnly);
2021-12-01 16:50:13 +08:00
set.add(RuntimeGlobals.makeNamespaceObject);
2021-11-30 19:55:51 +08:00
2021-12-03 23:23:09 +08:00
const CssLoadingRuntimeModule = getCssLoadingRuntimeModule();
2021-11-30 19:55:51 +08:00
compilation.addRuntimeModule(chunk, new CssLoadingRuntimeModule(set));
};
compilation.hooks.runtimeRequirementInTree
.for(RuntimeGlobals.ensureChunkHandlers)
.tap(plugin, handler);
compilation.hooks.runtimeRequirementInTree
.for(RuntimeGlobals.hmrDownloadUpdateHandlers)
.tap(plugin, handler);
}
);
}
getOrderedChunkCssModules(chunk, chunkGraph) {
2021-12-03 02:29:55 +08:00
const cssImports = chunkGraph.getOrderedChunkModulesIterableBySourceType(
chunk,
"css-import",
// TODO improve order function
compareModulesByPostOrderIndexOrIdentifier(chunkGraph.moduleGraph)
);
const cssContent = chunkGraph.getOrderedChunkModulesIterableBySourceType(
2021-11-30 19:55:51 +08:00
chunk,
"css",
// TODO improve order function
compareModulesByPostOrderIndexOrIdentifier(chunkGraph.moduleGraph)
);
2021-12-03 02:29:55 +08:00
return [...(cssImports || []), ...(cssContent || [])];
2021-11-30 19:55:51 +08:00
}
2021-12-15 23:46:13 +08:00
renderChunk({ uniqueName, chunk, chunkGraph, codeGenerationResults }) {
2021-11-30 19:55:51 +08:00
const modules = this.getOrderedChunkCssModules(chunk, chunkGraph);
const source = new ConcatSource();
2021-12-14 23:02:26 +08:00
const metaData = [];
2021-11-30 19:55:51 +08:00
for (const module of modules) {
try {
2021-12-14 23:02:26 +08:00
const codeGenResult = codeGenerationResults.get(module, chunk.runtime);
2021-12-03 02:29:55 +08:00
const s =
2021-12-14 23:02:26 +08:00
codeGenResult.sources.get("css") ||
codeGenResult.sources.get("css-import");
2021-11-30 20:46:42 +08:00
if (s) {
source.add(s);
source.add("\n");
}
2021-12-14 23:02:26 +08:00
const exports =
codeGenResult.data && codeGenResult.data.get("css-exports");
const moduleId = chunkGraph.getModuleId(module) + "";
2021-12-14 23:02:26 +08:00
metaData.push(
`${
exports
? Array.from(exports, ([n, v]) =>
2021-12-15 23:46:13 +08:00
v === `${uniqueName ? uniqueName + "-" : ""}${moduleId}-${n}`
? `${escapeCss(n)}/`
: `${escapeCss(n)}(${escapeCss(v)})`
2021-12-14 23:02:26 +08:00
).join("")
: ""
}${escapeCss(moduleId)}`
2021-12-14 23:02:26 +08:00
);
2021-11-30 19:55:51 +08:00
} catch (e) {
e.message += `\nduring rendering of css ${module.identifier()}`;
throw e;
}
}
source.add(
2021-12-15 23:46:13 +08:00
`head{--webpack-${escapeCss(
(uniqueName ? uniqueName + "-" : "") + chunk.id,
true
)}:${metaData.join(",")};}`
2021-11-30 19:55:51 +08:00
);
return source;
}
static getChunkFilenameTemplate(chunk, outputOptions) {
if (chunk.cssFilenameTemplate) {
return chunk.cssFilenameTemplate;
} else {
return outputOptions.cssFilename;
}
}
static chunkHasCss(chunk, chunkGraph) {
2021-12-03 02:29:55 +08:00
return (
!!chunkGraph.getChunkModulesIterableBySourceType(chunk, "css") ||
!!chunkGraph.getChunkModulesIterableBySourceType(chunk, "css-import")
);
2021-11-30 19:55:51 +08:00
}
}
module.exports = CssModulesPlugin;