Spaces:
Build error
Build error
/* | |
* Copyright (c) 2024 lax1dude. All Rights Reserved. | |
* | |
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND | |
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | |
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. | |
* IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, | |
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | |
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | |
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, | |
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | |
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | |
* POSSIBILITY OF SUCH DAMAGE. | |
* | |
*/ | |
/** | |
* @param {*} msg | |
*/ | |
function logInfo(msg) { | |
console.log("LoaderBootstrap: [INFO] " + msg); | |
} | |
/** | |
* @param {*} msg | |
*/ | |
function logWarn(msg) { | |
console.log("LoaderBootstrap: [WARN] " + msg); | |
} | |
/** | |
* @param {*} msg | |
*/ | |
function logError(msg) { | |
console.error("LoaderBootstrap: [ERROR] " + msg); | |
} | |
/** @type {function(string,number):ArrayBuffer|null} */ | |
var decodeBase64Impl = null; | |
/** | |
* @return {function(string,number):ArrayBuffer} | |
*/ | |
function createBase64Decoder() { | |
const revLookup = []; | |
const code = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | |
for (var i = 0, len = code.length; i < len; ++i) { | |
revLookup[code.charCodeAt(i)] = i; | |
} | |
revLookup["-".charCodeAt(0)] = 62; | |
revLookup["_".charCodeAt(0)] = 63; | |
/** | |
* @param {string} b64 | |
* @param {number} start | |
* @return {!Array<number>} | |
*/ | |
function getLens(b64, start) { | |
const len = b64.length - start; | |
if (len % 4 > 0) { | |
throw new Error("Invalid string. Length must be a multiple of 4"); | |
} | |
var validLen = b64.indexOf("=", start); | |
if (validLen === -1) { | |
validLen = len; | |
}else { | |
validLen -= start; | |
} | |
const placeHoldersLen = validLen === len ? 0 : 4 - (validLen % 4); | |
return [validLen, placeHoldersLen]; | |
} | |
/** | |
* @param {string} b64 | |
* @param {number} start | |
* @return {ArrayBuffer} | |
*/ | |
function decodeImpl(b64, start) { | |
var tmp; | |
const lens = getLens(b64, start); | |
const validLen = lens[0]; | |
const placeHoldersLen = lens[1]; | |
const arr = new Uint8Array(((validLen + placeHoldersLen) * 3 / 4) - placeHoldersLen); | |
var curByte = 0; | |
const len = (placeHoldersLen > 0 ? validLen - 4 : validLen) + start; | |
var i; | |
for (i = start; i < len; i += 4) { | |
tmp = (revLookup[b64.charCodeAt(i)] << 18) | | |
(revLookup[b64.charCodeAt(i + 1)] << 12) | | |
(revLookup[b64.charCodeAt(i + 2)] << 6) | | |
revLookup[b64.charCodeAt(i + 3)] | |
arr[curByte++] = (tmp >> 16) & 0xFF | |
arr[curByte++] = (tmp >> 8) & 0xFF | |
arr[curByte++] = tmp & 0xFF | |
} | |
if (placeHoldersLen === 2) { | |
tmp = (revLookup[b64.charCodeAt(i)] << 2) | | |
(revLookup[b64.charCodeAt(i + 1)] >> 4) | |
arr[curByte++] = tmp & 0xFF | |
}else if (placeHoldersLen === 1) { | |
tmp = (revLookup[b64.charCodeAt(i)] << 10) | | |
(revLookup[b64.charCodeAt(i + 1)] << 4) | | |
(revLookup[b64.charCodeAt(i + 2)] >> 2) | |
arr[curByte++] = (tmp >> 8) & 0xFF | |
arr[curByte++] = tmp & 0xFF | |
} | |
return arr.buffer; | |
} | |
return decodeImpl; | |
} | |
/** | |
* @param {string} url | |
* @param {number} start | |
* @return {ArrayBuffer} | |
*/ | |
function decodeBase64(url, start) { | |
if(!decodeBase64Impl) { | |
decodeBase64Impl = createBase64Decoder(); | |
} | |
return decodeBase64Impl(url, start); | |
} | |
/** | |
* @param {number} ms | |
* @return {!Promise} | |
*/ | |
function asyncSleep(ms) { | |
return new Promise(function(resolve) { | |
setTimeout(resolve, ms); | |
}); | |
} | |
/** | |
* @param {string} url | |
* @param {number} ms | |
* @return {!Promise} | |
*/ | |
function preloadImage(url, ms) { | |
return new Promise(function(resolve) { | |
const imgObj = new Image(); | |
imgObj.addEventListener("load", resolve); | |
imgObj.addEventListener("error", function() { | |
logWarn("Failed to preload image: " + url); | |
resolve(); | |
}); | |
imgObj.src = url; | |
setTimeout(resolve, ms); | |
}); | |
} | |
/** | |
* @param {string} url | |
* @return {!Promise<ArrayBuffer>} | |
*/ | |
function downloadURL(url) { | |
return new Promise(function(resolve) { | |
fetch(url, { "cache": "force-cache" }) | |
.then(function(res) { | |
return res.arrayBuffer(); | |
}) | |
.then(resolve) | |
.catch(function(ex) { | |
logError("Failed to fetch URL! " + ex); | |
resolve(null); | |
}); | |
}); | |
} | |
/** | |
* @param {string} url | |
* @return {!Promise<ArrayBuffer>} | |
*/ | |
function downloadDataURL(url) { | |
if(!url.startsWith("data:application/octet-stream;base64,")) { | |
return downloadURL(url); | |
}else { | |
return new Promise(function(resolve) { | |
downloadURL(url).then(function(res) { | |
if(res) { | |
resolve(res); | |
}else { | |
logWarn("Failed to decode base64 via fetch, doing it the slow way instead..."); | |
try { | |
resolve(decodeBase64(url, 37)); | |
}catch(ex) { | |
logError("Failed to decode base64! " + ex); | |
resolve(null); | |
} | |
} | |
}); | |
}); | |
} | |
} | |
/** | |
* @param {HTMLElement} rootElement | |
* @param {string} msg | |
*/ | |
function displayInvalidEPW(rootElement, msg) { | |
const downloadFailureMsg = /** @type {HTMLElement} */ (document.createElement("h2")); | |
downloadFailureMsg.style.color = "#AA0000"; | |
downloadFailureMsg.style.padding = "25px"; | |
downloadFailureMsg.style.fontFamily = "sans-serif"; | |
downloadFailureMsg.style["marginBlock"] = "0px"; | |
downloadFailureMsg.appendChild(document.createTextNode(msg)); | |
rootElement.appendChild(downloadFailureMsg); | |
const downloadFailureMsg2 = /** @type {HTMLElement} */ (document.createElement("h4")); | |
downloadFailureMsg2.style.color = "#AA0000"; | |
downloadFailureMsg2.style.padding = "25px"; | |
downloadFailureMsg2.style.fontFamily = "sans-serif"; | |
downloadFailureMsg2.style["marginBlock"] = "0px"; | |
downloadFailureMsg2.appendChild(document.createTextNode("Try again later")); | |
rootElement.style.backgroundColor = "white"; | |
rootElement.appendChild(downloadFailureMsg2); | |
} | |
window.main = async function() { | |
if(typeof window.eaglercraftXOpts === "undefined") { | |
const msg = "window.eaglercraftXOpts is not defined!"; | |
logError(msg); | |
alert(msg); | |
return; | |
} | |
const containerId = window.eaglercraftXOpts.container; | |
if(typeof containerId !== "string") { | |
const msg = "window.eaglercraftXOpts.container is not a string!"; | |
logError(msg); | |
alert(msg); | |
return; | |
} | |
var assetsURI = window.eaglercraftXOpts.assetsURI; | |
if(typeof assetsURI !== "string") { | |
if((typeof assetsURI === "object") && (typeof assetsURI[0] === "object") && (typeof assetsURI[0]["url"] === "string")) { | |
assetsURI = assetsURI[0]["url"]; | |
}else { | |
const msg = "window.eaglercraftXOpts.assetsURI is not a string!"; | |
logError(msg); | |
alert(msg); | |
return; | |
} | |
} | |
if(assetsURI.startsWith("data:")) { | |
delete window.eaglercraftXOpts.assetsURI; | |
} | |
const rootElement = /** @type {HTMLElement} */ (document.getElementById(containerId)); | |
if(!rootElement) { | |
const msg = "window.eaglercraftXOpts.container \"" + containerId + "\" is not a known element id!"; | |
logError(msg); | |
alert(msg); | |
return; | |
} | |
var node; | |
while(node = rootElement.lastChild) { | |
rootElement.removeChild(node); | |
} | |
const splashElement = /** @type {HTMLElement} */ (document.createElement("div")); | |
splashElement.style.width = "100%"; | |
splashElement.style.height = "100%"; | |
splashElement.style.setProperty("image-rendering", "pixelated"); | |
splashElement.style.background = "center / contain no-repeat url(\"\") white"; | |
rootElement.appendChild(splashElement); | |
// allow the screen to update | |
await asyncSleep(20); | |
/** @type {ArrayBuffer} */ | |
var theEPWFileBuffer; | |
if(assetsURI.startsWith("data:")) { | |
logInfo("Downloading EPW file \"<data: " + assetsURI.length + " chars>\"..."); | |
theEPWFileBuffer = await downloadDataURL(assetsURI); | |
}else { | |
logInfo("Downloading EPW file \"" + assetsURI + "\"..."); | |
theEPWFileBuffer = await downloadURL(assetsURI); | |
} | |
var isInvalid = false; | |
if(!theEPWFileBuffer) { | |
isInvalid = true; | |
}else if(theEPWFileBuffer.byteLength < 384) { | |
logError("The EPW file is too short"); | |
isInvalid = true; | |
} | |
if(isInvalid) { | |
rootElement.removeChild(splashElement); | |
const msg = "Failed to download EPW file!"; | |
displayInvalidEPW(rootElement, msg); | |
logError(msg); | |
return; | |
} | |
const dataView = new DataView(theEPWFileBuffer); | |
if(dataView.getUint32(0, true) !== 608649541 || dataView.getUint32(4, true) !== 1297301847) { | |
logError("The file is not an EPW file"); | |
isInvalid = true; | |
} | |
const phileLength = theEPWFileBuffer.byteLength; | |
if(dataView.getUint32(8, true) !== phileLength) { | |
logError("The EPW file is the wrong length"); | |
isInvalid = true; | |
} | |
if(isInvalid) { | |
rootElement.removeChild(splashElement); | |
const msg = "EPW file is invalid!"; | |
displayInvalidEPW(rootElement, msg); | |
logError(msg); | |
return; | |
} | |
const textDecoder = new TextDecoder("utf-8"); | |
const splashDataOffset = dataView.getUint32(100, true); | |
const splashDataLength = dataView.getUint32(104, true); | |
const splashMIMEOffset = dataView.getUint32(108, true); | |
const splashMIMELength = dataView.getUint32(112, true); | |
if(splashDataOffset < 0 || splashDataOffset + splashDataLength > phileLength | |
|| splashMIMEOffset < 0 || splashMIMEOffset + splashMIMELength > phileLength) { | |
logError("The EPW file contains an invalid offset (component: splash)"); | |
isInvalid = true; | |
} | |
if(isInvalid) { | |
rootElement.removeChild(splashElement); | |
const msg = "EPW file is invalid!"; | |
displayInvalidEPW(rootElement, msg); | |
logError(msg); | |
return; | |
} | |
const splashBinSlice = new Uint8Array(theEPWFileBuffer, splashDataOffset, splashDataLength); | |
const splashMIMESlice = new Uint8Array(theEPWFileBuffer, splashMIMEOffset, splashMIMELength); | |
const splashURL = URL.createObjectURL(new Blob([ splashBinSlice ], { "type": textDecoder.decode(splashMIMESlice) })); | |
await preloadImage(splashURL, 50); | |
logInfo("Loaded splash img: " + splashURL); | |
splashElement.style.background = "center / contain no-repeat url(\"" + splashURL + "\"), 0px 0px / 1000000% 1000000% no-repeat url(\"" + splashURL + "\") white"; | |
// allow the screen to update | |
await asyncSleep(20); | |
const loaderJSOffset = dataView.getUint32(164, true); | |
const loaderJSLength = dataView.getUint32(168, true); | |
const loaderWASMOffset = dataView.getUint32(180, true); | |
const loaderWASMLength = dataView.getUint32(184, true); | |
if(loaderJSOffset < 0 || loaderJSOffset + loaderJSLength > phileLength | |
|| loaderWASMOffset < 0 || loaderWASMOffset + loaderWASMLength > phileLength) { | |
logError("The EPW file contains an invalid offset (component: loader)"); | |
isInvalid = true; | |
} | |
if(isInvalid) { | |
rootElement.removeChild(splashElement); | |
const msg = "EPW file is invalid!"; | |
displayInvalidEPW(rootElement, msg); | |
logError(msg); | |
return; | |
} | |
const loaderJSSlice = new Uint8Array(theEPWFileBuffer, loaderJSOffset, loaderJSLength); | |
const loaderJSURL = URL.createObjectURL(new Blob([ loaderJSSlice ], { "type": "text/javascript;charset=utf-8" })); | |
logInfo("Loaded loader.js: " + splashURL); | |
const loaderWASMSlice = new Uint8Array(theEPWFileBuffer, loaderWASMOffset, loaderWASMLength); | |
const loaderWASMURL = URL.createObjectURL(new Blob([ loaderWASMSlice ], { "type": "application/wasm" })); | |
logInfo("Loaded loader.wasm: " + loaderWASMURL); | |
const optsObj = {}; | |
for(const [key, value] of Object.entries(window.eaglercraftXOpts)) { | |
if(key !== "container" && key !== "assetsURI") { | |
optsObj[key] = value; | |
} | |
} | |
window.__eaglercraftXLoaderContextPre = { | |
"rootElement": rootElement, | |
"eaglercraftXOpts": optsObj, | |
"theEPWFileBuffer": theEPWFileBuffer, | |
"loaderWASMURL": loaderWASMURL, | |
"splashURL": splashURL | |
}; | |
logInfo("Appending loader.js to document..."); | |
const scriptElement = /** @type {HTMLScriptElement} */ (document.createElement("script")); | |
scriptElement.type = "text/javascript"; | |
scriptElement.src = loaderJSURL; | |
document.head.appendChild(scriptElement); | |
}; | |