eaglercraft-1.8/sources/wasm-gc-teavm-bootstrap/js/main.js

378 lines
14 KiB
JavaScript
Raw Permalink Normal View History

/*
* 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);
};