| 
									
										
										
										
											2019-10-02 22:59:27 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const { ConcatSource, OriginalSource } = require("webpack-sources"); | 
					
						
							|  |  |  | const ExternalModule = require("./ExternalModule"); | 
					
						
							|  |  |  | const RuntimeGlobals = require("./RuntimeGlobals"); | 
					
						
							|  |  |  | const Template = require("./Template"); | 
					
						
							| 
									
										
										
										
											2019-10-11 21:46:57 +08:00
										 |  |  | const JavascriptModulesPlugin = require("./javascript/JavascriptModulesPlugin"); | 
					
						
							| 
									
										
										
										
											2019-10-02 22:59:27 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** @typedef {import("webpack-sources").Source} Source */ | 
					
						
							|  |  |  | /** @typedef {import("../declarations/WebpackOptions").LibraryCustomUmdCommentObject} LibraryCustomUmdCommentObject */ | 
					
						
							|  |  |  | /** @typedef {import("../declarations/WebpackOptions").LibraryCustomUmdObject} LibraryCustomUmdObject */ | 
					
						
							|  |  |  | /** @typedef {import("./Chunk")} Chunk */ | 
					
						
							|  |  |  | /** @typedef {import("./Compiler")} Compiler */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @param {string[]} accessor the accessor to convert to path | 
					
						
							|  |  |  |  * @returns {string} the path | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const accessorToObjectAccess = accessor => { | 
					
						
							|  |  |  | 	return accessor.map(a => `[${JSON.stringify(a)}]`).join(""); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @param {string=} base the path prefix | 
					
						
							|  |  |  |  * @param {string|string[]} accessor the accessor | 
					
						
							|  |  |  |  * @param {string=} joinWith the element separator | 
					
						
							|  |  |  |  * @returns {string} the path | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const accessorAccess = (base, accessor, joinWith = ", ") => { | 
					
						
							|  |  |  | 	const accessors = Array.isArray(accessor) ? accessor : [accessor]; | 
					
						
							|  |  |  | 	return accessors | 
					
						
							|  |  |  | 		.map((_, idx) => { | 
					
						
							|  |  |  | 			const a = base | 
					
						
							|  |  |  | 				? base + accessorToObjectAccess(accessors.slice(0, idx + 1)) | 
					
						
							|  |  |  | 				: accessors[0] + accessorToObjectAccess(accessors.slice(1, idx + 1)); | 
					
						
							|  |  |  | 			if (idx === accessors.length - 1) return a; | 
					
						
							|  |  |  | 			if (idx === 0 && base === undefined) | 
					
						
							|  |  |  | 				return `${a} = typeof ${a} === "object" ? ${a} : {}`; | 
					
						
							|  |  |  | 			return `${a} = ${a} || {}`; | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 		.join(joinWith); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** @typedef {string | string[] | LibraryCustomUmdObject} UmdTemplatePluginName */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @typedef {Object} UmdTemplatePluginOption | 
					
						
							|  |  |  |  * @property {boolean=} optionalAmdExternalAsGlobal | 
					
						
							|  |  |  |  * @property {boolean} namedDefine | 
					
						
							|  |  |  |  * @property {string | LibraryCustomUmdCommentObject} auxiliaryComment | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class UmdTemplatePlugin { | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {UmdTemplatePluginName} name the name of the UMD library | 
					
						
							|  |  |  | 	 * @param {UmdTemplatePluginOption} options the plugin option | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	constructor(name, options) { | 
					
						
							|  |  |  | 		if (typeof name === "object" && !Array.isArray(name)) { | 
					
						
							|  |  |  | 			this.name = name.root || name.amd || name.commonjs; | 
					
						
							|  |  |  | 			this.names = name; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			this.name = name; | 
					
						
							|  |  |  | 			this.names = { | 
					
						
							|  |  |  | 				commonjs: name, | 
					
						
							|  |  |  | 				root: name, | 
					
						
							|  |  |  | 				amd: name | 
					
						
							|  |  |  | 			}; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		this.optionalAmdExternalAsGlobal = options.optionalAmdExternalAsGlobal; | 
					
						
							|  |  |  | 		this.namedDefine = options.namedDefine; | 
					
						
							|  |  |  | 		this.auxiliaryComment = options.auxiliaryComment; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Compiler} compiler the compiler instance | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	apply(compiler) { | 
					
						
							|  |  |  | 		compiler.hooks.thisCompilation.tap("UmdTemplatePlugin", compilation => { | 
					
						
							|  |  |  | 			compilation.hooks.additionalTreeRuntimeRequirements.tap( | 
					
						
							|  |  |  | 				"UmdTemplatePlugin", | 
					
						
							|  |  |  | 				(chunk, set) => { | 
					
						
							|  |  |  | 					set.add(RuntimeGlobals.returnExportsFromRuntime); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			const hooks = JavascriptModulesPlugin.getCompilationHooks(compilation); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-09 17:55:56 +08:00
										 |  |  | 			hooks.render.tap( | 
					
						
							| 
									
										
										
										
											2019-10-02 22:59:27 +08:00
										 |  |  | 				"UmdTemplatePlugin", | 
					
						
							|  |  |  | 				(source, { chunk, moduleGraph, chunkGraph, runtimeTemplate }) => { | 
					
						
							| 
									
										
										
										
											2019-10-09 17:55:56 +08:00
										 |  |  | 					if (chunkGraph.getNumberOfEntryModules(chunk) === 0) return source; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-02 22:59:27 +08:00
										 |  |  | 					const modules = chunkGraph | 
					
						
							|  |  |  | 						.getChunkModules(chunk) | 
					
						
							|  |  |  | 						.filter( | 
					
						
							|  |  |  | 							m => | 
					
						
							|  |  |  | 								m instanceof ExternalModule && | 
					
						
							|  |  |  | 								(m.externalType === "umd" || m.externalType === "umd2") | 
					
						
							|  |  |  | 						); | 
					
						
							|  |  |  | 					let externals = /** @type {ExternalModule[]} */ (modules); | 
					
						
							|  |  |  | 					/** @type {ExternalModule[]} */ | 
					
						
							|  |  |  | 					const optionalExternals = []; | 
					
						
							|  |  |  | 					/** @type {ExternalModule[]} */ | 
					
						
							|  |  |  | 					let requiredExternals = []; | 
					
						
							|  |  |  | 					if (this.optionalAmdExternalAsGlobal) { | 
					
						
							|  |  |  | 						for (const m of externals) { | 
					
						
							|  |  |  | 							if (m.isOptional(moduleGraph)) { | 
					
						
							|  |  |  | 								optionalExternals.push(m); | 
					
						
							|  |  |  | 							} else { | 
					
						
							|  |  |  | 								requiredExternals.push(m); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						externals = requiredExternals.concat(optionalExternals); | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						requiredExternals = externals; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const replaceKeys = str => { | 
					
						
							|  |  |  | 						return compilation.getPath(str, { | 
					
						
							|  |  |  | 							chunk | 
					
						
							|  |  |  | 						}); | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const externalsDepsArray = modules => { | 
					
						
							|  |  |  | 						return `[${replaceKeys( | 
					
						
							|  |  |  | 							modules | 
					
						
							|  |  |  | 								.map(m => | 
					
						
							|  |  |  | 									JSON.stringify( | 
					
						
							|  |  |  | 										typeof m.request === "object" ? m.request.amd : m.request | 
					
						
							|  |  |  | 									) | 
					
						
							|  |  |  | 								) | 
					
						
							|  |  |  | 								.join(", ") | 
					
						
							|  |  |  | 						)}]`;
 | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const externalsRootArray = modules => { | 
					
						
							|  |  |  | 						return replaceKeys( | 
					
						
							|  |  |  | 							modules | 
					
						
							|  |  |  | 								.map(m => { | 
					
						
							|  |  |  | 									let request = m.request; | 
					
						
							|  |  |  | 									if (typeof request === "object") request = request.root; | 
					
						
							|  |  |  | 									return `root${accessorToObjectAccess([].concat(request))}`; | 
					
						
							|  |  |  | 								}) | 
					
						
							|  |  |  | 								.join(", ") | 
					
						
							|  |  |  | 						); | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const externalsRequireArray = type => { | 
					
						
							|  |  |  | 						return replaceKeys( | 
					
						
							|  |  |  | 							externals | 
					
						
							|  |  |  | 								.map(m => { | 
					
						
							|  |  |  | 									let expr; | 
					
						
							|  |  |  | 									let request = m.request; | 
					
						
							|  |  |  | 									if (typeof request === "object") { | 
					
						
							|  |  |  | 										request = request[type]; | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 									if (request === undefined) { | 
					
						
							|  |  |  | 										throw new Error( | 
					
						
							|  |  |  | 											"Missing external configuration for type:" + type | 
					
						
							|  |  |  | 										); | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 									if (Array.isArray(request)) { | 
					
						
							|  |  |  | 										expr = `require(${JSON.stringify( | 
					
						
							|  |  |  | 											request[0] | 
					
						
							|  |  |  | 										)})${accessorToObjectAccess(request.slice(1))}`;
 | 
					
						
							|  |  |  | 									} else { | 
					
						
							|  |  |  | 										expr = `require(${JSON.stringify(request)})`; | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 									if (m.isOptional(moduleGraph)) { | 
					
						
							|  |  |  | 										expr = `(function webpackLoadOptionalExternalModule() { try { return ${expr}; } catch(e) {} }())`; | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 									return expr; | 
					
						
							|  |  |  | 								}) | 
					
						
							|  |  |  | 								.join(", ") | 
					
						
							|  |  |  | 						); | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const externalsArguments = modules => { | 
					
						
							|  |  |  | 						return modules | 
					
						
							|  |  |  | 							.map( | 
					
						
							|  |  |  | 								m => | 
					
						
							|  |  |  | 									`__WEBPACK_EXTERNAL_MODULE_${Template.toIdentifier( | 
					
						
							|  |  |  | 										`${chunkGraph.getModuleId(m)}` | 
					
						
							|  |  |  | 									)}__`
 | 
					
						
							|  |  |  | 							) | 
					
						
							|  |  |  | 							.join(", "); | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const libraryName = library => { | 
					
						
							|  |  |  | 						return JSON.stringify(replaceKeys([].concat(library).pop())); | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					let amdFactory; | 
					
						
							|  |  |  | 					if (optionalExternals.length > 0) { | 
					
						
							|  |  |  | 						const wrapperArguments = externalsArguments(requiredExternals); | 
					
						
							|  |  |  | 						const factoryArguments = | 
					
						
							|  |  |  | 							requiredExternals.length > 0 | 
					
						
							|  |  |  | 								? externalsArguments(requiredExternals) + | 
					
						
							|  |  |  | 								  ", " + | 
					
						
							|  |  |  | 								  externalsRootArray(optionalExternals) | 
					
						
							|  |  |  | 								: externalsRootArray(optionalExternals); | 
					
						
							|  |  |  | 						amdFactory = | 
					
						
							|  |  |  | 							`function webpackLoadOptionalExternalModuleAmd(${wrapperArguments}) {\n` + | 
					
						
							|  |  |  | 							`			return factory(${factoryArguments});\n` + | 
					
						
							|  |  |  | 							"		}"; | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						amdFactory = "factory"; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const auxiliaryComment = this.auxiliaryComment; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					const getAuxilaryComment = type => { | 
					
						
							|  |  |  | 						if (auxiliaryComment) { | 
					
						
							|  |  |  | 							if (typeof auxiliaryComment === "string") | 
					
						
							|  |  |  | 								return "\t//" + auxiliaryComment + "\n"; | 
					
						
							|  |  |  | 							if (auxiliaryComment[type]) | 
					
						
							|  |  |  | 								return "\t//" + auxiliaryComment[type] + "\n"; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						return ""; | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					return new ConcatSource( | 
					
						
							|  |  |  | 						new OriginalSource( | 
					
						
							|  |  |  | 							"(function webpackUniversalModuleDefinition(root, factory) {\n" + | 
					
						
							|  |  |  | 								getAuxilaryComment("commonjs2") + | 
					
						
							|  |  |  | 								"	if(typeof exports === 'object' && typeof module === 'object')\n" + | 
					
						
							|  |  |  | 								"		module.exports = factory(" + | 
					
						
							|  |  |  | 								externalsRequireArray("commonjs2") + | 
					
						
							|  |  |  | 								");\n" + | 
					
						
							|  |  |  | 								getAuxilaryComment("amd") + | 
					
						
							|  |  |  | 								"	else if(typeof define === 'function' && define.amd)\n" + | 
					
						
							|  |  |  | 								(requiredExternals.length > 0 | 
					
						
							|  |  |  | 									? this.names.amd && this.namedDefine === true | 
					
						
							|  |  |  | 										? "		define(" + | 
					
						
							|  |  |  | 										  libraryName(this.names.amd) + | 
					
						
							|  |  |  | 										  ", " + | 
					
						
							|  |  |  | 										  externalsDepsArray(requiredExternals) + | 
					
						
							|  |  |  | 										  ", " + | 
					
						
							|  |  |  | 										  amdFactory + | 
					
						
							|  |  |  | 										  ");\n" | 
					
						
							|  |  |  | 										: "		define(" + | 
					
						
							|  |  |  | 										  externalsDepsArray(requiredExternals) + | 
					
						
							|  |  |  | 										  ", " + | 
					
						
							|  |  |  | 										  amdFactory + | 
					
						
							|  |  |  | 										  ");\n" | 
					
						
							|  |  |  | 									: this.names.amd && this.namedDefine === true | 
					
						
							|  |  |  | 									? "		define(" + | 
					
						
							|  |  |  | 									  libraryName(this.names.amd) + | 
					
						
							|  |  |  | 									  ", [], " + | 
					
						
							|  |  |  | 									  amdFactory + | 
					
						
							|  |  |  | 									  ");\n" | 
					
						
							|  |  |  | 									: "		define([], " + amdFactory + ");\n") + | 
					
						
							|  |  |  | 								(this.names.root || this.names.commonjs | 
					
						
							|  |  |  | 									? getAuxilaryComment("commonjs") + | 
					
						
							|  |  |  | 									  "	else if(typeof exports === 'object')\n" + | 
					
						
							|  |  |  | 									  "		exports[" + | 
					
						
							|  |  |  | 									  libraryName(this.names.commonjs || this.names.root) + | 
					
						
							|  |  |  | 									  "] = factory(" + | 
					
						
							|  |  |  | 									  externalsRequireArray("commonjs") + | 
					
						
							|  |  |  | 									  ");\n" + | 
					
						
							|  |  |  | 									  getAuxilaryComment("root") + | 
					
						
							|  |  |  | 									  "	else\n" + | 
					
						
							|  |  |  | 									  "		" + | 
					
						
							|  |  |  | 									  replaceKeys( | 
					
						
							|  |  |  | 											accessorAccess( | 
					
						
							|  |  |  | 												"root", | 
					
						
							|  |  |  | 												this.names.root || this.names.commonjs | 
					
						
							|  |  |  | 											) | 
					
						
							|  |  |  | 									  ) + | 
					
						
							|  |  |  | 									  " = factory(" + | 
					
						
							|  |  |  | 									  externalsRootArray(externals) + | 
					
						
							|  |  |  | 									  ");\n" | 
					
						
							|  |  |  | 									: "	else {\n" + | 
					
						
							|  |  |  | 									  (externals.length > 0 | 
					
						
							|  |  |  | 											? "		var a = typeof exports === 'object' ? factory(" + | 
					
						
							|  |  |  | 											  externalsRequireArray("commonjs") + | 
					
						
							|  |  |  | 											  ") : factory(" + | 
					
						
							|  |  |  | 											  externalsRootArray(externals) + | 
					
						
							|  |  |  | 											  ");\n" | 
					
						
							|  |  |  | 											: "		var a = factory();\n") + | 
					
						
							|  |  |  | 									  "		for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];\n" + | 
					
						
							|  |  |  | 									  "	}\n") + | 
					
						
							|  |  |  | 								`})(${ | 
					
						
							|  |  |  | 									runtimeTemplate.outputOptions.globalObject | 
					
						
							|  |  |  | 								}, function(${externalsArguments(externals)}) {\nreturn `,
 | 
					
						
							|  |  |  | 							"webpack/universalModuleDefinition" | 
					
						
							|  |  |  | 						), | 
					
						
							|  |  |  | 						source, | 
					
						
							|  |  |  | 						";\n})" | 
					
						
							|  |  |  | 					); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-09 17:55:56 +08:00
										 |  |  | 			hooks.chunkHash.tap( | 
					
						
							|  |  |  | 				"UmdTemplatePlugin", | 
					
						
							|  |  |  | 				(chunk, hash, { chunkGraph }) => { | 
					
						
							|  |  |  | 					if (chunkGraph.getNumberOfEntryModules(chunk) === 0) return; | 
					
						
							|  |  |  | 					hash.update("umd"); | 
					
						
							|  |  |  | 					hash.update(`${this.names.root}`); | 
					
						
							|  |  |  | 					hash.update(`${this.names.amd}`); | 
					
						
							|  |  |  | 					hash.update(`${this.names.commonjs}`); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2019-10-02 22:59:27 +08:00
										 |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = UmdTemplatePlugin; |