1 | /*
|
---|
2 | MIT License http://www.opensource.org/licenses/mit-license.php
|
---|
3 | */
|
---|
4 |
|
---|
5 | "use strict";
|
---|
6 |
|
---|
7 | const { SyncWaterfallHook } = require("tapable");
|
---|
8 | const Compilation = require("../Compilation");
|
---|
9 | const RuntimeGlobals = require("../RuntimeGlobals");
|
---|
10 | const Template = require("../Template");
|
---|
11 | const HelperRuntimeModule = require("./HelperRuntimeModule");
|
---|
12 |
|
---|
13 | /** @typedef {import("../Chunk")} Chunk */
|
---|
14 | /** @typedef {import("../Compiler")} Compiler */
|
---|
15 |
|
---|
16 | /**
|
---|
17 | * @typedef {Object} LoadScriptCompilationHooks
|
---|
18 | * @property {SyncWaterfallHook<[string, Chunk]>} createScript
|
---|
19 | */
|
---|
20 |
|
---|
21 | /** @type {WeakMap<Compilation, LoadScriptCompilationHooks>} */
|
---|
22 | const compilationHooksMap = new WeakMap();
|
---|
23 |
|
---|
24 | class LoadScriptRuntimeModule extends HelperRuntimeModule {
|
---|
25 | /**
|
---|
26 | * @param {Compilation} compilation the compilation
|
---|
27 | * @returns {LoadScriptCompilationHooks} hooks
|
---|
28 | */
|
---|
29 | static getCompilationHooks(compilation) {
|
---|
30 | if (!(compilation instanceof Compilation)) {
|
---|
31 | throw new TypeError(
|
---|
32 | "The 'compilation' argument must be an instance of Compilation"
|
---|
33 | );
|
---|
34 | }
|
---|
35 | let hooks = compilationHooksMap.get(compilation);
|
---|
36 | if (hooks === undefined) {
|
---|
37 | hooks = {
|
---|
38 | createScript: new SyncWaterfallHook(["source", "chunk"])
|
---|
39 | };
|
---|
40 | compilationHooksMap.set(compilation, hooks);
|
---|
41 | }
|
---|
42 | return hooks;
|
---|
43 | }
|
---|
44 |
|
---|
45 | /**
|
---|
46 | * @param {boolean=} withCreateScriptUrl use create script url for trusted types
|
---|
47 | */
|
---|
48 | constructor(withCreateScriptUrl) {
|
---|
49 | super("load script");
|
---|
50 | this._withCreateScriptUrl = withCreateScriptUrl;
|
---|
51 | }
|
---|
52 |
|
---|
53 | /**
|
---|
54 | * @returns {string} runtime code
|
---|
55 | */
|
---|
56 | generate() {
|
---|
57 | const { compilation } = this;
|
---|
58 | const { runtimeTemplate, outputOptions } = compilation;
|
---|
59 | const {
|
---|
60 | scriptType,
|
---|
61 | chunkLoadTimeout: loadTimeout,
|
---|
62 | crossOriginLoading,
|
---|
63 | uniqueName,
|
---|
64 | charset
|
---|
65 | } = outputOptions;
|
---|
66 | const fn = RuntimeGlobals.loadScript;
|
---|
67 |
|
---|
68 | const { createScript } =
|
---|
69 | LoadScriptRuntimeModule.getCompilationHooks(compilation);
|
---|
70 |
|
---|
71 | const code = Template.asString([
|
---|
72 | "script = document.createElement('script');",
|
---|
73 | scriptType ? `script.type = ${JSON.stringify(scriptType)};` : "",
|
---|
74 | charset ? "script.charset = 'utf-8';" : "",
|
---|
75 | `script.timeout = ${loadTimeout / 1000};`,
|
---|
76 | `if (${RuntimeGlobals.scriptNonce}) {`,
|
---|
77 | Template.indent(
|
---|
78 | `script.setAttribute("nonce", ${RuntimeGlobals.scriptNonce});`
|
---|
79 | ),
|
---|
80 | "}",
|
---|
81 | uniqueName
|
---|
82 | ? 'script.setAttribute("data-webpack", dataWebpackPrefix + key);'
|
---|
83 | : "",
|
---|
84 | `script.src = ${
|
---|
85 | this._withCreateScriptUrl
|
---|
86 | ? `${RuntimeGlobals.createScriptUrl}(url)`
|
---|
87 | : "url"
|
---|
88 | };`,
|
---|
89 | crossOriginLoading
|
---|
90 | ? Template.asString([
|
---|
91 | "if (script.src.indexOf(window.location.origin + '/') !== 0) {",
|
---|
92 | Template.indent(
|
---|
93 | `script.crossOrigin = ${JSON.stringify(crossOriginLoading)};`
|
---|
94 | ),
|
---|
95 | "}"
|
---|
96 | ])
|
---|
97 | : ""
|
---|
98 | ]);
|
---|
99 |
|
---|
100 | return Template.asString([
|
---|
101 | "var inProgress = {};",
|
---|
102 | uniqueName
|
---|
103 | ? `var dataWebpackPrefix = ${JSON.stringify(uniqueName + ":")};`
|
---|
104 | : "// data-webpack is not used as build has no uniqueName",
|
---|
105 | "// loadScript function to load a script via script tag",
|
---|
106 | `${fn} = ${runtimeTemplate.basicFunction("url, done, key, chunkId", [
|
---|
107 | "if(inProgress[url]) { inProgress[url].push(done); return; }",
|
---|
108 | "var script, needAttach;",
|
---|
109 | "if(key !== undefined) {",
|
---|
110 | Template.indent([
|
---|
111 | 'var scripts = document.getElementsByTagName("script");',
|
---|
112 | "for(var i = 0; i < scripts.length; i++) {",
|
---|
113 | Template.indent([
|
---|
114 | "var s = scripts[i];",
|
---|
115 | `if(s.getAttribute("src") == url${
|
---|
116 | uniqueName
|
---|
117 | ? ' || s.getAttribute("data-webpack") == dataWebpackPrefix + key'
|
---|
118 | : ""
|
---|
119 | }) { script = s; break; }`
|
---|
120 | ]),
|
---|
121 | "}"
|
---|
122 | ]),
|
---|
123 | "}",
|
---|
124 | "if(!script) {",
|
---|
125 | Template.indent([
|
---|
126 | "needAttach = true;",
|
---|
127 | createScript.call(code, this.chunk)
|
---|
128 | ]),
|
---|
129 | "}",
|
---|
130 | "inProgress[url] = [done];",
|
---|
131 | "var onScriptComplete = " +
|
---|
132 | runtimeTemplate.basicFunction(
|
---|
133 | "prev, event",
|
---|
134 | Template.asString([
|
---|
135 | "// avoid mem leaks in IE.",
|
---|
136 | "script.onerror = script.onload = null;",
|
---|
137 | "clearTimeout(timeout);",
|
---|
138 | "var doneFns = inProgress[url];",
|
---|
139 | "delete inProgress[url];",
|
---|
140 | "script.parentNode && script.parentNode.removeChild(script);",
|
---|
141 | `doneFns && doneFns.forEach(${runtimeTemplate.returningFunction(
|
---|
142 | "fn(event)",
|
---|
143 | "fn"
|
---|
144 | )});`,
|
---|
145 | "if(prev) return prev(event);"
|
---|
146 | ])
|
---|
147 | ),
|
---|
148 | ";",
|
---|
149 | `var timeout = setTimeout(onScriptComplete.bind(null, undefined, { type: 'timeout', target: script }), ${loadTimeout});`,
|
---|
150 | "script.onerror = onScriptComplete.bind(null, script.onerror);",
|
---|
151 | "script.onload = onScriptComplete.bind(null, script.onload);",
|
---|
152 | "needAttach && document.head.appendChild(script);"
|
---|
153 | ])};`
|
---|
154 | ]);
|
---|
155 | }
|
---|
156 | }
|
---|
157 |
|
---|
158 | module.exports = LoadScriptRuntimeModule;
|
---|