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 | * @param {boolean=} withFetchPriority use `fetchPriority` attribute
|
---|
48 | */
|
---|
49 | constructor(withCreateScriptUrl, withFetchPriority) {
|
---|
50 | super("load script");
|
---|
51 | this._withCreateScriptUrl = withCreateScriptUrl;
|
---|
52 | this._withFetchPriority = withFetchPriority;
|
---|
53 | }
|
---|
54 |
|
---|
55 | /**
|
---|
56 | * @returns {string | null} runtime code
|
---|
57 | */
|
---|
58 | generate() {
|
---|
59 | const compilation = /** @type {Compilation} */ (this.compilation);
|
---|
60 | const { runtimeTemplate, outputOptions } = compilation;
|
---|
61 | const {
|
---|
62 | scriptType,
|
---|
63 | chunkLoadTimeout: loadTimeout,
|
---|
64 | crossOriginLoading,
|
---|
65 | uniqueName,
|
---|
66 | charset
|
---|
67 | } = outputOptions;
|
---|
68 | const fn = RuntimeGlobals.loadScript;
|
---|
69 |
|
---|
70 | const { createScript } =
|
---|
71 | LoadScriptRuntimeModule.getCompilationHooks(compilation);
|
---|
72 |
|
---|
73 | const code = Template.asString([
|
---|
74 | "script = document.createElement('script');",
|
---|
75 | scriptType ? `script.type = ${JSON.stringify(scriptType)};` : "",
|
---|
76 | charset ? "script.charset = 'utf-8';" : "",
|
---|
77 | `script.timeout = ${/** @type {number} */ (loadTimeout) / 1000};`,
|
---|
78 | `if (${RuntimeGlobals.scriptNonce}) {`,
|
---|
79 | Template.indent(
|
---|
80 | `script.setAttribute("nonce", ${RuntimeGlobals.scriptNonce});`
|
---|
81 | ),
|
---|
82 | "}",
|
---|
83 | uniqueName
|
---|
84 | ? 'script.setAttribute("data-webpack", dataWebpackPrefix + key);'
|
---|
85 | : "",
|
---|
86 | this._withFetchPriority
|
---|
87 | ? Template.asString([
|
---|
88 | "if(fetchPriority) {",
|
---|
89 | Template.indent(
|
---|
90 | 'script.setAttribute("fetchpriority", fetchPriority);'
|
---|
91 | ),
|
---|
92 | "}"
|
---|
93 | ])
|
---|
94 | : "",
|
---|
95 | `script.src = ${
|
---|
96 | this._withCreateScriptUrl
|
---|
97 | ? `${RuntimeGlobals.createScriptUrl}(url)`
|
---|
98 | : "url"
|
---|
99 | };`,
|
---|
100 | crossOriginLoading
|
---|
101 | ? crossOriginLoading === "use-credentials"
|
---|
102 | ? 'script.crossOrigin = "use-credentials";'
|
---|
103 | : Template.asString([
|
---|
104 | "if (script.src.indexOf(window.location.origin + '/') !== 0) {",
|
---|
105 | Template.indent(
|
---|
106 | `script.crossOrigin = ${JSON.stringify(crossOriginLoading)};`
|
---|
107 | ),
|
---|
108 | "}"
|
---|
109 | ])
|
---|
110 | : ""
|
---|
111 | ]);
|
---|
112 |
|
---|
113 | return Template.asString([
|
---|
114 | "var inProgress = {};",
|
---|
115 | uniqueName
|
---|
116 | ? `var dataWebpackPrefix = ${JSON.stringify(`${uniqueName}:`)};`
|
---|
117 | : "// data-webpack is not used as build has no uniqueName",
|
---|
118 | "// loadScript function to load a script via script tag",
|
---|
119 | `${fn} = ${runtimeTemplate.basicFunction(
|
---|
120 | `url, done, key, chunkId${
|
---|
121 | this._withFetchPriority ? ", fetchPriority" : ""
|
---|
122 | }`,
|
---|
123 | [
|
---|
124 | "if(inProgress[url]) { inProgress[url].push(done); return; }",
|
---|
125 | "var script, needAttach;",
|
---|
126 | "if(key !== undefined) {",
|
---|
127 | Template.indent([
|
---|
128 | 'var scripts = document.getElementsByTagName("script");',
|
---|
129 | "for(var i = 0; i < scripts.length; i++) {",
|
---|
130 | Template.indent([
|
---|
131 | "var s = scripts[i];",
|
---|
132 | `if(s.getAttribute("src") == url${
|
---|
133 | uniqueName
|
---|
134 | ? ' || s.getAttribute("data-webpack") == dataWebpackPrefix + key'
|
---|
135 | : ""
|
---|
136 | }) { script = s; break; }`
|
---|
137 | ]),
|
---|
138 | "}"
|
---|
139 | ]),
|
---|
140 | "}",
|
---|
141 | "if(!script) {",
|
---|
142 | Template.indent([
|
---|
143 | "needAttach = true;",
|
---|
144 | createScript.call(code, /** @type {Chunk} */ (this.chunk))
|
---|
145 | ]),
|
---|
146 | "}",
|
---|
147 | "inProgress[url] = [done];",
|
---|
148 | `var onScriptComplete = ${runtimeTemplate.basicFunction(
|
---|
149 | "prev, event",
|
---|
150 | Template.asString([
|
---|
151 | "// avoid mem leaks in IE.",
|
---|
152 | "script.onerror = script.onload = null;",
|
---|
153 | "clearTimeout(timeout);",
|
---|
154 | "var doneFns = inProgress[url];",
|
---|
155 | "delete inProgress[url];",
|
---|
156 | "script.parentNode && script.parentNode.removeChild(script);",
|
---|
157 | `doneFns && doneFns.forEach(${runtimeTemplate.returningFunction(
|
---|
158 | "fn(event)",
|
---|
159 | "fn"
|
---|
160 | )});`,
|
---|
161 | "if(prev) return prev(event);"
|
---|
162 | ])
|
---|
163 | )}`,
|
---|
164 | `var timeout = setTimeout(onScriptComplete.bind(null, undefined, { type: 'timeout', target: script }), ${loadTimeout});`,
|
---|
165 | "script.onerror = onScriptComplete.bind(null, script.onerror);",
|
---|
166 | "script.onload = onScriptComplete.bind(null, script.onload);",
|
---|
167 | "needAttach && document.head.appendChild(script);"
|
---|
168 | ]
|
---|
169 | )};`
|
---|
170 | ]);
|
---|
171 | }
|
---|
172 | }
|
---|
173 |
|
---|
174 | module.exports = LoadScriptRuntimeModule;
|
---|