webpack/lib/dependencies/CssUrlDependency.js

145 lines
3.8 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 Ivan Kopeykin @vankop
*/
"use strict";
const makeSerializable = require("../util/makeSerializable");
const memoize = require("../util/memoize");
const ModuleDependency = require("./ModuleDependency");
/** @typedef {import("webpack-sources").ReplaceSource} ReplaceSource */
/** @typedef {import("../ChunkGraph")} ChunkGraph */
/** @typedef {import("../Dependency")} Dependency */
/** @typedef {import("../Dependency").UpdateHashContext} UpdateHashContext */
/** @typedef {import("../DependencyTemplate").DependencyTemplateContext} DependencyTemplateContext */
/** @typedef {import("../Module")} Module */
/** @typedef {import("../ModuleGraph")} ModuleGraph */
/** @typedef {import("../ModuleGraphConnection")} ModuleGraphConnection */
/** @typedef {import("../ModuleGraphConnection").ConnectionState} ConnectionState */
/** @typedef {import("../util/Hash")} Hash */
/** @typedef {import("../util/runtime").RuntimeSpec} RuntimeSpec */
2021-12-01 21:15:19 +08:00
const getRawDataUrlModule = memoize(() => require("../asset/RawDataUrlModule"));
2021-11-30 19:55:51 +08:00
class CssUrlDependency extends ModuleDependency {
/**
* @param {string} request request
* @param {[number, number]} range range of the argument
2023-04-14 06:52:50 +08:00
* @param {"string" | "url"} urlType dependency type e.g. url() or string
2021-11-30 19:55:51 +08:00
*/
2023-04-14 06:52:50 +08:00
constructor(request, range, urlType) {
2021-11-30 19:55:51 +08:00
super(request);
this.range = range;
2023-04-14 06:52:50 +08:00
this.urlType = urlType;
2021-11-30 19:55:51 +08:00
}
get type() {
return "css url()";
}
get category() {
return "url";
}
/**
* @param {string} context context directory
* @returns {Module} a module
*/
createIgnoredModule(context) {
2021-12-01 21:15:19 +08:00
const RawDataUrlModule = getRawDataUrlModule();
return new RawDataUrlModule("data:,", `ignored-asset`, `(ignored asset)`);
2021-11-30 19:55:51 +08:00
}
2021-12-03 23:23:09 +08:00
serialize(context) {
const { write } = context;
2023-04-14 06:52:50 +08:00
write(this.urlType);
2021-12-03 23:23:09 +08:00
super.serialize(context);
}
deserialize(context) {
const { read } = context;
2023-04-14 06:52:50 +08:00
this.urlType = read();
2021-12-03 23:23:09 +08:00
super.deserialize(context);
}
2021-11-30 19:55:51 +08:00
}
2021-12-03 23:23:09 +08:00
const cssEscapeString = str => {
let countWhiteOrBracket = 0;
let countQuotation = 0;
let countApostrophe = 0;
for (let i = 0; i < str.length; i++) {
const cc = str.charCodeAt(i);
switch (cc) {
case 9: // tab
case 10: // nl
case 32: // space
case 40: // (
case 41: // )
countWhiteOrBracket++;
break;
case 34:
countQuotation++;
break;
case 39:
countApostrophe++;
break;
}
}
if (countWhiteOrBracket < 2) {
return str.replace(/[\n\t ()'"\\]/g, m => `\\${m}`);
} else if (countQuotation <= countApostrophe) {
return `"${str.replace(/[\n"\\]/g, m => `\\${m}`)}"`;
} else {
return `'${str.replace(/[\n'\\]/g, m => `\\${m}`)}'`;
}
};
2021-11-30 19:55:51 +08:00
CssUrlDependency.Template = class CssUrlDependencyTemplate extends (
ModuleDependency.Template
) {
/**
* @param {Dependency} dependency the dependency for which the template should be applied
* @param {ReplaceSource} source the current replace source which can be modified
* @param {DependencyTemplateContext} templateContext the context object
* @returns {void}
*/
2021-11-30 20:46:42 +08:00
apply(
dependency,
source,
{ moduleGraph, runtimeTemplate, codeGenerationResults }
2021-11-30 20:46:42 +08:00
) {
2021-11-30 19:55:51 +08:00
const dep = /** @type {CssUrlDependency} */ (dependency);
2023-04-14 06:52:50 +08:00
let newValue;
switch (dep.urlType) {
case "string":
newValue = cssEscapeString(
runtimeTemplate.assetUrl({
publicPath: "",
module: moduleGraph.getModule(dep),
codeGenerationResults
})
);
break;
case "url":
newValue = `url(${cssEscapeString(
runtimeTemplate.assetUrl({
publicPath: "",
module: moduleGraph.getModule(dep),
codeGenerationResults
})
)})`;
break;
}
source.replace(dep.range[0], dep.range[1] - 1, newValue);
2021-11-30 19:55:51 +08:00
}
};
makeSerializable(CssUrlDependency, "webpack/lib/dependencies/CssUrlDependency");
module.exports = CssUrlDependency;