mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			179 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Sergey Melyukov @smelukov
 | |
| */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const { ReplaceSource, RawSource, ConcatSource } = require("webpack-sources");
 | |
| const { UsageState } = require("../ExportsInfo");
 | |
| const Generator = require("../Generator");
 | |
| const RuntimeGlobals = require("../RuntimeGlobals");
 | |
| const Template = require("../Template");
 | |
| const { cssExportConvention } = require("../util/conventions");
 | |
| 
 | |
| /** @typedef {import("webpack-sources").Source} Source */
 | |
| /** @typedef {import("../../declarations/WebpackOptions").CssGeneratorExportsConvention} CssGeneratorExportsConvention */
 | |
| /** @typedef {import("../../declarations/WebpackOptions").CssGeneratorLocalIdentName} CssGeneratorLocalIdentName */
 | |
| /** @typedef {import("../Dependency")} Dependency */
 | |
| /** @typedef {import("../Generator").GenerateContext} GenerateContext */
 | |
| /** @typedef {import("../Generator").UpdateHashContext} UpdateHashContext */
 | |
| /** @typedef {import("../Module").ConcatenationBailoutReasonContext} ConcatenationBailoutReasonContext */
 | |
| /** @typedef {import("../NormalModule")} NormalModule */
 | |
| /** @typedef {import("../util/Hash")} Hash */
 | |
| 
 | |
| /**
 | |
|  * @template T
 | |
|  * @typedef {import("../InitFragment")<T>} InitFragment
 | |
|  */
 | |
| 
 | |
| const TYPES = new Set(["javascript"]);
 | |
| 
 | |
| class CssExportsGenerator extends Generator {
 | |
| 	/**
 | |
| 	 * @param {CssGeneratorExportsConvention} convention the convention of the exports name
 | |
| 	 * @param {CssGeneratorLocalIdentName | undefined} localIdentName css export local ident name
 | |
| 	 */
 | |
| 	constructor(convention, localIdentName) {
 | |
| 		super();
 | |
| 		/** @type {CssGeneratorExportsConvention} */
 | |
| 		this.convention = convention;
 | |
| 		/** @type {CssGeneratorLocalIdentName | undefined} */
 | |
| 		this.localIdentName = localIdentName;
 | |
| 	}
 | |
| 
 | |
| 	// TODO add getConcatenationBailoutReason to allow concatenation
 | |
| 	// but how to make it have a module id
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {NormalModule} module module for which the code should be generated
 | |
| 	 * @param {GenerateContext} generateContext context for generate
 | |
| 	 * @returns {Source} generated code
 | |
| 	 */
 | |
| 	generate(module, generateContext) {
 | |
| 		const source = new ReplaceSource(new RawSource(""));
 | |
| 		/** @type {InitFragment<TODO>[]} */
 | |
| 		const initFragments = [];
 | |
| 		/** @type {Map<string, string>} */
 | |
| 		const cssExports = new Map();
 | |
| 
 | |
| 		generateContext.runtimeRequirements.add(RuntimeGlobals.module);
 | |
| 
 | |
| 		let chunkInitFragments;
 | |
| 		const runtimeRequirements = new Set();
 | |
| 
 | |
| 		const templateContext = {
 | |
| 			runtimeTemplate: generateContext.runtimeTemplate,
 | |
| 			dependencyTemplates: generateContext.dependencyTemplates,
 | |
| 			moduleGraph: generateContext.moduleGraph,
 | |
| 			chunkGraph: generateContext.chunkGraph,
 | |
| 			module,
 | |
| 			runtime: generateContext.runtime,
 | |
| 			runtimeRequirements: runtimeRequirements,
 | |
| 			concatenationScope: generateContext.concatenationScope,
 | |
| 			codeGenerationResults: generateContext.codeGenerationResults,
 | |
| 			initFragments,
 | |
| 			cssExports,
 | |
| 			get chunkInitFragments() {
 | |
| 				if (!chunkInitFragments) {
 | |
| 					const data = generateContext.getData();
 | |
| 					chunkInitFragments = data.get("chunkInitFragments");
 | |
| 					if (!chunkInitFragments) {
 | |
| 						chunkInitFragments = [];
 | |
| 						data.set("chunkInitFragments", chunkInitFragments);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				return chunkInitFragments;
 | |
| 			}
 | |
| 		};
 | |
| 
 | |
| 		/**
 | |
| 		 * @param {Dependency} dependency the dependency
 | |
| 		 */
 | |
| 		const handleDependency = dependency => {
 | |
| 			const constructor = /** @type {new (...args: any[]) => Dependency} */ (
 | |
| 				dependency.constructor
 | |
| 			);
 | |
| 			const template = generateContext.dependencyTemplates.get(constructor);
 | |
| 			if (!template) {
 | |
| 				throw new Error(
 | |
| 					"No template for dependency: " + dependency.constructor.name
 | |
| 				);
 | |
| 			}
 | |
| 
 | |
| 			template.apply(dependency, source, templateContext);
 | |
| 		};
 | |
| 		module.dependencies.forEach(handleDependency);
 | |
| 
 | |
| 		if (generateContext.concatenationScope) {
 | |
| 			const source = new ConcatSource();
 | |
| 			const usedIdentifiers = new Set();
 | |
| 			for (const [name, v] of cssExports) {
 | |
| 				for (let k of cssExportConvention(name, this.convention)) {
 | |
| 					let identifier = Template.toIdentifier(k);
 | |
| 					let i = 0;
 | |
| 					while (usedIdentifiers.has(identifier)) {
 | |
| 						identifier = Template.toIdentifier(k + i);
 | |
| 					}
 | |
| 					usedIdentifiers.add(identifier);
 | |
| 					generateContext.concatenationScope.registerExport(k, identifier);
 | |
| 					source.add(
 | |
| 						`${
 | |
| 							generateContext.runtimeTemplate.supportsConst() ? "const" : "var"
 | |
| 						} ${identifier} = ${JSON.stringify(v)};\n`
 | |
| 					);
 | |
| 				}
 | |
| 			}
 | |
| 			return source;
 | |
| 		} else {
 | |
| 			const otherUsed =
 | |
| 				generateContext.moduleGraph
 | |
| 					.getExportsInfo(module)
 | |
| 					.otherExportsInfo.getUsed(generateContext.runtime) !==
 | |
| 				UsageState.Unused;
 | |
| 			if (otherUsed) {
 | |
| 				generateContext.runtimeRequirements.add(
 | |
| 					RuntimeGlobals.makeNamespaceObject
 | |
| 				);
 | |
| 			}
 | |
| 			const newCssExports = [];
 | |
| 			for (let [k, v] of cssExports) {
 | |
| 				for (let name of cssExportConvention(k, this.convention)) {
 | |
| 					newCssExports.push(`\t${JSON.stringify(name)}: ${JSON.stringify(v)}`);
 | |
| 				}
 | |
| 			}
 | |
| 			return new RawSource(
 | |
| 				`${otherUsed ? `${RuntimeGlobals.makeNamespaceObject}(` : ""}${
 | |
| 					module.moduleArgument
 | |
| 				}.exports = {\n${newCssExports.join(",\n")}\n}${otherUsed ? ")" : ""};`
 | |
| 			);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {NormalModule} module fresh module
 | |
| 	 * @returns {Set<string>} available types (do not mutate)
 | |
| 	 */
 | |
| 	getTypes(module) {
 | |
| 		return TYPES;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {NormalModule} module the module
 | |
| 	 * @param {string=} type source type
 | |
| 	 * @returns {number} estimate size of the module
 | |
| 	 */
 | |
| 	getSize(module, type) {
 | |
| 		return 42;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {Hash} hash hash that will be modified
 | |
| 	 * @param {UpdateHashContext} updateHashContext context for updating hash
 | |
| 	 */
 | |
| 	updateHash(hash, { module }) {}
 | |
| }
 | |
| 
 | |
| module.exports = CssExportsGenerator;
 |