378 lines
14 KiB
JavaScript
378 lines
14 KiB
JavaScript
|
/*
|
||
|
* 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
|
||
|
* @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;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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);
|
||
|
|
||
|
/** @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) }));
|
||
|
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);
|
||
|
|
||
|
};
|
||
|
|