From 76c28a5de1e14d452df2ca533c0762524ec6906b Mon Sep 17 00:00:00 2001 From: zumbiepig <121742281+zumbiepig@users.noreply.github.com> Date: Fri, 6 Sep 2024 18:07:55 -0700 Subject: [PATCH] step 2 --- .gitignore | 5 +- build.ts | 84 +- build_fast.ts | 49 - bun.lockb | Bin 113007 -> 113083 bytes eslint.config.js | 1 - package.json | 12 +- index.ts => server.ts | 0 src/game/web/main/1.2.5/index.html | 13305 ---------------- src/home/game/index.html | 2 +- .../scripts/eagler-launch/1.5.2/main.ts | 27 + .../eagler-launch/1.8.8/eaglerforge.ts | 45 + .../eagler-launch/1.8.8/eaglermobile-ef.ts | 688 + .../eagler-launch/1.8.8/eaglerpocketmobile.ts | 1156 ++ .../scripts/eagler-launch/1.8.8/main.ts | 26 + .../scripts/eagler-launch/1.9.4/main.ts | 21 + .../scripts/eagler-launch/b1.3/main.ts | 10 + src/resources/scripts/google-tag.ts | 12 + src/resources/scripts/main.ts | 631 + tsconfig.json | 7 +- 19 files changed, 2678 insertions(+), 13403 deletions(-) delete mode 100644 build_fast.ts rename index.ts => server.ts (100%) delete mode 100644 src/game/web/main/1.2.5/index.html create mode 100644 src/resources/scripts/eagler-launch/1.5.2/main.ts create mode 100644 src/resources/scripts/eagler-launch/1.8.8/eaglerforge.ts create mode 100644 src/resources/scripts/eagler-launch/1.8.8/eaglermobile-ef.ts create mode 100644 src/resources/scripts/eagler-launch/1.8.8/eaglerpocketmobile.ts create mode 100644 src/resources/scripts/eagler-launch/1.8.8/main.ts create mode 100644 src/resources/scripts/eagler-launch/1.9.4/main.ts create mode 100644 src/resources/scripts/eagler-launch/b1.3/main.ts create mode 100644 src/resources/scripts/google-tag.ts create mode 100644 src/resources/scripts/main.ts diff --git a/.gitignore b/.gitignore index aba28fd..6292299 100644 --- a/.gitignore +++ b/.gitignore @@ -1,10 +1,7 @@ package-lock.json desktop.ini -public/resources/data/assets.json -public/resources/scripts/ -public/sw.js -public/sw-full.js +public/ # Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore diff --git a/build.ts b/build.ts index ea3a3ef..6a73e9f 100644 --- a/build.ts +++ b/build.ts @@ -1,8 +1,11 @@ -import { $, build } from 'bun'; -import { mkdirSync, readdirSync, statSync, writeFileSync, readFileSync } from 'fs'; -import { resolve, dirname } from 'path'; -import chalk from 'chalk'; +import { build } from 'bun'; +import { mkdirSync, readdirSync, statSync, writeFileSync, readFileSync, rmSync } from 'fs'; +import { resolve, dirname, basename } from 'path'; import { minify } from 'html-minifier'; +import javascriptObfuscator from 'javascript-obfuscator'; +import chalk from 'chalk'; + +console.clear(); function getFiles(baseDir: string, dir?: string, filesArr?: string[]) { dir = dir ?? baseDir; @@ -19,32 +22,31 @@ function getFiles(baseDir: string, dir?: string, filesArr?: string[]) { return filesArr; } -console.clear(); - +const isDev = process.env.NODE_ENV === 'development' ? true : false; const srcDir = resolve(import.meta.dir, 'src'); const publicDir = resolve(import.meta.dir, 'public'); -const srcFilesArr = getFiles(resolve(import.meta.dir, 'src')); +const srcFiles = getFiles(resolve(import.meta.dir, 'src')); +const bundleFiles: string[] = []; +const minifyFiles: string[] = []; +const copyFiles: string[] = []; +srcFiles.forEach((file) => { + const strippedPath = file.replace(new RegExp(`^${srcDir}`), ''); + if (file.endsWith('.ts')) bundleFiles.push(file); + else if (isDev) copyFiles.push(file); + else if (/\.(html|css|js|json)$/.test(strippedPath)) { + if (strippedPath.startsWith('/game/offline/') || basename(strippedPath) === 'classes.js') copyFiles.push(file); + else minifyFiles.push(file); + } else copyFiles.push(file); +}); -console.log(chalk.cyan('Linting code...\n')); -//const lintOutput = await $`bunx eslint ./src/`.nothrow().text(); -//if (lintOutput) { -// console.error(lintOutput); -// process.exit(1); -//} - -console.log(chalk.cyan('Type-checking code...\n')); -//const tscOutput = await $`bunx tsc`.nothrow().text(); -//if (tscOutput) { -// console.error(tscOutput); -// process.exit(1); -//} - -console.log(chalk.cyan('Removing old build artifacts...\n')); -await $`rm -rf ./public/resources/scripts/ ./public/resources/data/assets.json ./public/sw.js ./public/sw-full.js`.quiet(); +if (!isDev) { + console.log(chalk.cyan('Removing old build artifacts...\n')); + rmSync(publicDir, { force: true, recursive: true }); +} console.log(chalk.cyan('Bundling TypeScript and modules...\n')); await build({ - entrypoints: srcFilesArr.filter((file) => file.endsWith('.ts')), + entrypoints: srcFiles.filter((file) => file.endsWith('.ts')), outdir: './public/', target: 'browser', root: './src/', @@ -55,10 +57,9 @@ await build({ }, }); -console.log(chalk.cyan('Minifying HTML and CSS...\n')); -srcFilesArr - .filter((file) => !file.endsWith('.ts')) - .forEach((file) => { +if (!isDev) { + console.log(chalk.cyan('Minifying HTML, JS, CSS, and JSON...\n')); + minifyFiles.forEach((file) => { const outputPath = file.replace(new RegExp(`^${srcDir}`), publicDir); mkdirSync(dirname(outputPath), { recursive: true }); writeFileSync( @@ -68,16 +69,37 @@ srcFilesArr removeComments: true, minifyCSS: true, minifyJS: true, + continueOnParseError: true, }) ); }); -console.log(chalk.cyan('Obfuscating JavaScript...\n')); -await $`bunx javascript-obfuscator ./public/resources/scripts/ --output ./public/resources/scripts/ --options-preset high-obfuscation`.quiet(); + console.log(chalk.cyan('Obfuscating JavaScript...\n')); + getFiles(resolve(publicDir, 'resources/scripts')).forEach((file) => { + writeFileSync( + file, + javascriptObfuscator + .obfuscate(readFileSync(file, 'utf-8'), { + optionsPreset: 'high-obfuscation', + target: 'browser', + }) + .getObfuscatedCode() + ); + }); +} + +console.log(chalk.cyan('Copying other files...\n')); +copyFiles.forEach((file) => { + const outputPath = file.replace(new RegExp(`^${srcDir}`), publicDir); + mkdirSync(dirname(outputPath), { recursive: true }); + writeFileSync(outputPath, readFileSync(file)); +}); console.log(chalk.cyan('Generating assets list...\n')); +const assetsJsonPath = resolve(publicDir, 'resources/data/assets.json'); +mkdirSync(dirname(assetsJsonPath), { recursive: true }); writeFileSync( - resolve(publicDir, 'resources/data/assets.json'), + assetsJsonPath, JSON.stringify( getFiles(publicDir).map((asset) => { return asset.replace(new RegExp(`^${publicDir}`), '').replace(/\/index\.html$/, '/'); diff --git a/build_fast.ts b/build_fast.ts deleted file mode 100644 index 275721f..0000000 --- a/build_fast.ts +++ /dev/null @@ -1,49 +0,0 @@ -import { $, build } from 'bun'; -import { readdirSync, statSync, writeFileSync } from 'fs'; -import { join } from 'path'; -import chalk from 'chalk'; - -function getFiles(baseDir: string, dir?: string, filesArr?: string[]) { - dir = dir ?? baseDir; - filesArr = filesArr ?? []; - const files = readdirSync(dir); - for (const file of files) { - const name = join(dir, file); - if (statSync(name).isDirectory()) { - getFiles(baseDir, name, filesArr); - } else { - filesArr.push(name); - } - } - return filesArr; -} - -console.log(chalk.cyan('Removing old build artifacts...\n')); -await $`rm -rf ./public/resources/scripts/ ./public/resources/data/assets.json ./public/sw.js ./public/sw-full.js`.quiet(); - -console.log(chalk.cyan('Bundling TypeScript and modules...\n')); -const srcFilesArr = getFiles(join(import.meta.dir, 'src')); -await build({ - entrypoints: srcFilesArr, - outdir: './public/', - target: 'browser', - root: './src/', - minify: { - syntax: true, - whitespace: true, - identifiers: false, - }, -}); - -console.log(chalk.cyan('Generating assets list...\n')); -const publicDir = join(import.meta.dir, 'public'); -writeFileSync( - join(publicDir, 'resources/data/assets.json'), - JSON.stringify( - getFiles(publicDir).map((asset) => { - return asset.replace(new RegExp(`^${publicDir}`), '').replace(/\/index\.html$/, '/'); - }) - ) -); - -console.log(chalk.green('Build complete!\n')); diff --git a/bun.lockb b/bun.lockb index f031ae4b51d088ce9cc9042022023a75728f17ff..c3ae02ada466f968d674d2e9c76e8ff62b38632a 100755 GIT binary patch delta 12200 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 N%x8b{9@fqAe*k3jUEBZw delta 11903 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 z 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 V#vBiRwv1Ij$3hNH-OCaj{|ATUBG~`{ diff --git a/eslint.config.js b/eslint.config.js index 1eb2c7c..3258652 100644 --- a/eslint.config.js +++ b/eslint.config.js @@ -5,7 +5,6 @@ export default tseslint.config( eslint.configs.recommended, tseslint.configs.eslintRecommended, ...tseslint.configs.strict, - // ...tseslint.configs.strictTypeChecked, ...tseslint.configs.stylisticTypeChecked, { languageOptions: { diff --git a/package.json b/package.json index bc0a8a8..cdeda9b 100644 --- a/package.json +++ b/package.json @@ -3,12 +3,12 @@ "name": "minexlauncher", "type": "module", "scripts": { - "start": "NODE_ENV=production bun run ./index.ts", - "dev": "NODE_ENV=development DEBUG=app:* bun --hot run ./index.ts", - "lint": "eslint ./src/", - "lint:fix": "eslint --fix ./src/", - "build": "bun run ./build.ts", - "build:fast": "bun run ./build_fast.ts" + "start": "NODE_ENV=production bun run ./server.ts", + "dev": "NODE_ENV=development DEBUG=minexlauncher:* bun --hot run ./server.ts", + "lint": "eslint ./src/ && tsc", + "lint:fix": "eslint --fix ./src/ && tsc", + "build": "NODE_ENV=production bun run ./build.ts", + "build:fast": "NODE_ENV=development bun run ./build.ts" }, "dependencies": { "chalk": "^5.3.0", diff --git a/index.ts b/server.ts similarity index 100% rename from index.ts rename to server.ts diff --git a/src/game/web/main/1.2.5/index.html b/src/game/web/main/1.2.5/index.html deleted file mode 100644 index e099dfc..0000000 --- a/src/game/web/main/1.2.5/index.html +++ /dev/null @@ -1,13305 +0,0 @@ - - - - Eaglercraft 1.2.5 - - - - - - - - - - - \ No newline at end of file diff --git a/src/home/game/index.html b/src/home/game/index.html index ce86de3..59cd360 100644 --- a/src/home/game/index.html +++ b/src/home/game/index.html @@ -82,7 +82,7 @@ 1.5.2 -
+
1.2.5
diff --git a/src/resources/scripts/eagler-launch/1.5.2/main.ts b/src/resources/scripts/eagler-launch/1.5.2/main.ts new file mode 100644 index 0000000..49b42e0 --- /dev/null +++ b/src/resources/scripts/eagler-launch/1.5.2/main.ts @@ -0,0 +1,27 @@ +// @ts-nocheck +window.addEventListener('load', function () { + const relayId = Math.floor(Math.random() * 3); + window.eaglercraftOpts = { + container: 'game_frame', + assetsURI: `${window.location.pathname}/assets.epk`, + localesURI: `${window.location.pathname}/lang/`, + serverWorkerURI: `${window.location.pathname}/worker_bootstrap.js`, + worldsFolder: 'MAIN', + servers: [{ serverName: 'BrickMC', serverAddress: 'wss://play.brickmc.net', hideAddress: false }], + relays: [ + { addr: 'wss://relay.deev.is/', name: 'lax1dude relay #1', primary: relayId === 0 }, + { addr: 'wss://relay.lax1dude.net/', name: 'lax1dude relay #2', primary: relayId === 1 }, + { addr: 'wss://relay.shhnowisnottheti.me/', name: 'ayunami relay #1', primary: relayId === 2 }, + ], + mainMenu: { + splashes: ['Darviglet!', 'eaglerenophile!', 'You Eagler!', 'Yeeeeeee!', 'yeee', 'EEEEEEEEE!', 'You Darvig!', 'You Vigg!', ':>', '|>', 'You Yumpster!'], + eaglerLogo: false, + }, + }; + + const urlParams = new URLSearchParams(window.location.search); + window.eaglercraftOpts.joinServer = urlParams.get('server') ?? undefined; + + history.replaceState({}, '', '/play'); + main(); +}); diff --git a/src/resources/scripts/eagler-launch/1.8.8/eaglerforge.ts b/src/resources/scripts/eagler-launch/1.8.8/eaglerforge.ts new file mode 100644 index 0000000..6ceeb4e --- /dev/null +++ b/src/resources/scripts/eagler-launch/1.8.8/eaglerforge.ts @@ -0,0 +1,45 @@ +// @ts-nocheck +window.addEventListener('load', () => { + const relayId = Math.floor(Math.random() * 3); + window.eaglercraftXOpts = { + container: 'game_frame', + assetsURI: `${window.location.pathname}/assets.epk`, + localesURI: `${window.location.pathname}/lang/`, + servers: [ + { addr: 'wss://mc.ricenetwork.xyz', name: 'Rice Network' }, + { addr: 'wss://mc.lamplifesteal.xyz', name: 'LampLifesteal' }, + { addr: 'wss://electronmc.club', name: 'Electron Network' }, + { addr: 'wss://play.brickmc.net', name: 'BrickMC' }, + ], + relays: [ + { addr: 'wss://relay.deev.is/', comment: 'lax1dude relay #1', primary: relayId === 0 }, + { addr: 'wss://relay.lax1dude.net/', comment: 'lax1dude relay #2', primary: relayId === 1 }, + { addr: 'wss://relay.shhnowisnottheti.me/', comment: 'ayunami relay #1', primary: relayId === 2 }, + ], + }; + + const storage = { + local: { + get: function (key: string) { + const item = localStorage.getItem('minexlauncher'); + if (item !== null) { + const json = JSON.parse(item); + if (json[key] !== undefined) { + return json[key]; + } else { + return null; + } + } else { + return null; + } + }, + }, + }; + + const urlParams = new URLSearchParams(window.location.search); + window.eaglercraftXOpts.joinServer = urlParams.get('server') ?? undefined; + window.eaglercraftXOpts.Mods = storage.local.get('mods') ?? []; + + history.replaceState({}, '', '/play'); + main(); +}); diff --git a/src/resources/scripts/eagler-launch/1.8.8/eaglermobile-ef.ts b/src/resources/scripts/eagler-launch/1.8.8/eaglermobile-ef.ts new file mode 100644 index 0000000..deb8218 --- /dev/null +++ b/src/resources/scripts/eagler-launch/1.8.8/eaglermobile-ef.ts @@ -0,0 +1,688 @@ +/* eslint-disable */ +// @ts-nocheck +if (new URLSearchParams(window.location.search).get('mobile') === 'true') { + try { + unsafeWindow.console.warn("DANGER: This userscript is using unsafeWindow. Unsafe websites could potentially use this to gain access to data and other content that the browser normally wouldn't allow!"), Object.defineProperty(window, 'clientWindow', { value: unsafeWindow }); + } catch { + Object.defineProperty(window, 'clientWindow', { value: window }); + } + (function () { + try { + return document.createEvent('TouchEvent'), !0; + } catch (A) { + return !1; + } + })() || alert('WARNING: This script was created for mobile, and may break functionality in non-mobile browsers!'), + (clientWindow.crouchLock = !1), + (clientWindow.sprintLock = !1), + (clientWindow.keyboardFix = !1), + (clientWindow.inputFix = !1), + (clientWindow.blockNextInput = !1), + (clientWindow.hiddenInputFocused = !1); + var A = null, + I = null, + g = null; + String.prototype.toKeyCode = function () { + return { + 0: 48, + 1: 49, + 2: 50, + 3: 51, + 4: 52, + 5: 53, + 6: 54, + 7: 55, + 8: 56, + 9: 57, + backspace: 8, + tab: 9, + enter: 13, + shift: 16, + ctrl: 17, + alt: 18, + pause_break: 19, + caps_lock: 20, + escape: 27, + ' ': 32, + page_up: 33, + page_down: 34, + end: 35, + home: 36, + left_arrow: 37, + up_arrow: 38, + right_arrow: 39, + down_arrow: 40, + insert: 45, + delete: 46, + a: 65, + b: 66, + c: 67, + d: 68, + e: 69, + f: 70, + g: 71, + h: 72, + i: 73, + j: 74, + k: 75, + l: 76, + m: 77, + n: 78, + o: 79, + p: 80, + q: 81, + r: 82, + s: 83, + t: 84, + u: 85, + v: 86, + w: 87, + x: 88, + y: 89, + z: 90, + left_window_key: 91, + right_window_key: 92, + select_key: 93, + numpad_0: 96, + numpad_1: 97, + numpad_2: 98, + numpad_3: 99, + numpad_4: 100, + numpad_5: 101, + numpad_6: 102, + numpad_7: 103, + numpad_8: 104, + numpad_9: 105, + '*': 106, + '+': 107, + '-': 109, + '.': 110, + '/': 111, + f1: 112, + f2: 113, + f3: 114, + f4: 115, + f5: 116, + f6: 117, + f7: 118, + f8: 119, + f9: 120, + f10: 121, + f11: 122, + f12: 123, + num_lock: 144, + scroll_lock: 145, + ';': 186, + '=': 187, + ',': 188, + '-': 189, + '.': 190, + '/': 191, + '`': 192, + '[': 219, + '\\': 220, + ']': 221, + '"': 222, + }[this]; + }; + const i = EventTarget.prototype.addEventListener; + Object.defineProperty(EventTarget.prototype, 'addEventListener', { + value: function (A, I, ...g) { + 'keydown' == A + ? i.call( + this, + A, + function (...A) { + if (A[0].isValid || !clientWindow.keyboardFix) return I.apply(this, A); + }, + ...g + ) + : i.call(this, A, I, ...g); + }, + }); + const e = Event.prototype.preventDefault; + function C(A, I) { + const g = A.toKeyCode(); + let i = new KeyboardEvent(I, { key: A, keyCode: g, which: g }); + (i.isValid = !0), clientWindow.dispatchEvent(i); + } + function t(A, I, g) { + g.dispatchEvent(new PointerEvent(I, { button: A })); + } + function G(A, I) { + A.dispatchEvent(new WheelEvent('wheel', { wheelDeltaY: I })); + } + function d(A) { + let I = document.getElementById('inGameStyle'), + g = document.getElementById('inMenuStyle'); + (I.disabled = A), (g.disabled = !A); + } + (Event.prototype.preventDefault = function (A) { + ('hiddenInput' != document.activeElement.id || A) && ((this._preventDefault = e), this._preventDefault()); + }), + (clientWindow.fakelock = null), + Object.defineProperty(Element.prototype, 'requestPointerLock', { + value: function () { + return (clientWindow.fakelock = this), document.dispatchEvent(new Event('pointerlockchange')), d(!0), !0; + }, + }), + Object.defineProperty(Document.prototype, 'pointerLockElement', { + get: function () { + return clientWindow.fakelock; + }, + }), + Object.defineProperty(Document.prototype, 'exitPointerLock', { + value: function () { + return (clientWindow.fakelock = null), document.dispatchEvent(new Event('pointerlockchange')), d(!1), !0; + }, + }), + (clientWindow.fakefull = null), + Object.defineProperty(Element.prototype, 'requestFullscreen', { + value: function () { + return (clientWindow.fakefull = this), document.dispatchEvent(new Event('fullscreenchange')), !0; + }, + }), + Object.defineProperty(document, 'fullscreenElement', { + get: function () { + return clientWindow.fakefull; + }, + }), + Object.defineProperty(Document.prototype, 'exitFullscreen', { + value: function () { + return (clientWindow.fakefull = null), document.dispatchEvent(new Event('fullscreenchange')), !0; + }, + }); + const c = document.createElement; + document.createElement = function (A, I) { + this._createElement = c; + var g = this._createElement(A); + return ( + 'input' != A || + I || + (document.querySelectorAll('#fileUpload').forEach((A) => A.parentNode.removeChild(A)), + (g.id = 'fileUpload'), + g.addEventListener( + 'change', + function (A) { + (g.hidden = !0), (g.style.display = 'none'); + }, + { passive: !1, once: !0 } + ), + clientWindow.addEventListener( + 'focus', + function (A) { + setTimeout(() => { + (g.hidden = !0), (g.style.display = 'none'); + }, 300); + }, + { once: !0 } + ), + document.body.appendChild(g)), + g + ); + }; + let n = document.createElement('style'); + (n.id = 'inGameStyle'), (n.textContent = '\n.inGame {\ndisplay: none;\n}'), document.documentElement.appendChild(n); + let o = document.createElement('style'); + function Z(A, I, g) { + var i = document.createElement(g ?? 'button', !0); + return ( + i.classList.add(A), + i.classList.add(I), + i.classList.add('mobileControl'), + i.addEventListener( + 'touchmove', + function (A) { + A.preventDefault(); + }, + !1 + ), + i.addEventListener('contextmenu', function (A) { + A.preventDefault(); + }), + i + ); + } + var l; + (o.id = 'inMenuStyle'), + (o.textContent = '\n.inMenu {\ndisplay: none;\n}'), + document.documentElement.appendChild(o), + ((l = 'canvas'), + new Promise((A) => { + if (document.querySelector(l)) return A(document.querySelector(l)); + const I = new MutationObserver((g) => { + document.querySelector(l) && (I.disconnect(), A(document.querySelector(l))); + }); + I.observe(document.documentElement, { childList: !0, subtree: !0 }); + })).then(() => { + !(function () { + var i = document.querySelector('canvas'); + i.addEventListener( + 'touchmove', + function (g) { + g.preventDefault(); + const e = g.targetTouches[0]; + A || ((A = e.pageX), (I = e.pageY)), (g.movementX = e.pageX - A), (g.movementY = e.pageY - I); + var C = clientWindow.fakelock ? new MouseEvent('mousemove', { movementX: g.movementX, movementY: g.movementY }) : new WheelEvent('wheel', { wheelDeltaY: g.movementY }); + i.dispatchEvent(C), (A = e.pageX), (I = e.pageY); + }, + !1 + ), + i.addEventListener( + 'touchend', + function (g) { + (A = null), (I = null); + }, + !1 + ), + d(null != clientWindow.fakelock); + let e = Z('strafeRightButton', 'inGame', 'div'); + (e.style.cssText = 'left:20vh;bottom:20vh;'), document.body.appendChild(e); + let c = Z('strafeLeftButton', 'inGame', 'div'); + (c.style.cssText = 'left:0vh;bottom:20vh;'), document.body.appendChild(c); + let n = Z('forwardButton', 'inGame', 'div'); + (n.style.cssText = 'left:10vh;bottom:20vh;'), + n.addEventListener( + 'touchstart', + function (A) { + C('w', 'keydown'), e.classList.remove('hide'), c.classList.remove('hide'), n.classList.add('active'); + }, + !1 + ), + n.addEventListener( + 'touchmove', + function (A) { + A.preventDefault(); + const I = A.targetTouches[0]; + g || (g = I.pageX); + let i = I.pageX - g; + 10 * i > clientWindow.innerHeight ? (e.classList.add('active'), c.classList.remove('active'), C('d', 'keydown'), C('a', 'keyup')) : 10 * i < 0 - clientWindow.innerHeight ? (c.classList.add('active'), e.classList.remove('active'), C('a', 'keydown'), C('d', 'keyup')) : (e.classList.remove('active'), c.classList.remove('active')); + }, + !1 + ), + n.addEventListener( + 'touchend', + function (A) { + C('w', 'keyup'), C('d', 'keyup'), C('a', 'keyup'), e.classList.remove('active'), c.classList.remove('active'), e.classList.add('hide'), c.classList.add('hide'), n.classList.remove('active'), (g = null); + }, + !1 + ), + e.classList.add('hide'), + c.classList.add('hide'), + document.body.appendChild(n); + let o = Z('rightButton', 'inGame'); + (o.style.cssText = 'left:20vh;bottom:10vh;'), + o.addEventListener( + 'touchstart', + function (A) { + C('d', 'keydown'); + }, + !1 + ), + o.addEventListener( + 'touchend', + function (A) { + C('d', 'keyup'); + }, + !1 + ), + document.body.appendChild(o); + let l = Z('leftButton', 'inGame'); + (l.style.cssText = 'left: 0vh; bottom:10vh;'), + l.addEventListener( + 'touchstart', + function (A) { + C('a', 'keydown'); + }, + !1 + ), + l.addEventListener( + 'touchend', + function (A) { + C('a', 'keyup'); + }, + !1 + ), + document.body.appendChild(l); + let b = Z('backButton', 'inGame'); + (b.style.cssText = 'left:10vh;bottom:0vh;'), + b.addEventListener( + 'touchstart', + function (A) { + C('s', 'keydown'); + }, + !1 + ), + b.addEventListener( + 'touchend', + function (A) { + C('s', 'keyup'); + }, + !1 + ), + document.body.appendChild(b); + let w = Z('jumpButton', 'inGame'); + (w.style.cssText = 'right:10vh;bottom:10vh;'), + w.addEventListener( + 'touchstart', + function (A) { + C(' ', 'keydown'); + }, + !1 + ), + w.addEventListener( + 'touchend', + function (A) { + C(' ', 'keyup'); + }, + !1 + ), + document.body.appendChild(w); + let M = Z('crouchButton', 'inGame'); + (M.style.cssText = 'left:10vh;bottom:10vh;'), + M.addEventListener( + 'touchstart', + function (A) { + C('shift', 'keydown'), + (clientWindow.crouchLock = !!clientWindow.crouchLock && null), + (clientWindow.crouchTimer = setTimeout(function (A) { + (clientWindow.crouchLock = null != clientWindow.crouchLock), M.classList.toggle('active'); + }, 1e3)); + }, + !1 + ), + M.addEventListener( + 'touchend', + function (A) { + clientWindow.crouchLock || (C('shift', 'keyup'), M.classList.remove('active'), (clientWindow.crouchLock = !1)), clearTimeout(clientWindow.crouchTimer); + }, + !1 + ), + document.body.appendChild(M); + let v = Z('inventoryButton', 'inGame'); + (v.style.cssText = 'right:0vh;bottom:30vh;'), + v.addEventListener( + 'touchstart', + function (A) { + C('e', 'keydown'); + }, + !1 + ), + v.addEventListener( + 'touchend', + function (A) { + C('e', 'keyup'); + }, + !1 + ), + document.body.appendChild(v); + let m = Z('exitButton', 'inMenu'); + (m.style.cssText = 'top: 0vh; margin: auto; left: 0vh; right:8vh; width: 8vh; height: 8vh;'), + m.addEventListener( + 'touchstart', + function (A) { + C('`', 'keydown'); + }, + !1 + ), + m.addEventListener( + 'touchend', + function (A) { + C('`', 'keyup'); + }, + !1 + ), + document.body.appendChild(m); + let h = document.createElement('input', !0); + (h.id = 'hiddenInput'), + h.classList.add('inMenu'), + (h.style.cssText = 'position:absolute;top: 0vh; margin: auto; left: 8vh; right:0vh; width: 8vh; height: 8vh;font-size:20px;z-index: -10;color: transparent;text-shadow: 0 0 0 black;'), + h.addEventListener( + 'beforeinput', + function (A) { + A.stopImmediatePropagation(), A.preventDefault(!0); + let I = 'insertLineBreak' == A.inputType ? 'return' : null == A.data ? 'delete' : A.data.slice(-1); + if ((clientWindow.lastKey || (clientWindow.console.warn('Enabling blocking duplicate key events. Some functionality may be lost.'), (clientWindow.inputFix = !0)), clientWindow.keyboardFix)) + if ('insertLineBreak' == A.inputType) C('enter', 'keydown'), C('enter', 'keyup'); + else { + const g = A.inputType.slice(0, 1); + if ('i' == g && A.data) { + if (clientWindow.lastKey == I && clientWindow.blockNextInput && clientWindow.inputFix) clientWindow.blockNextInput = !1; + else { + I.toLowerCase() != I ? (C('shift', 'keydown'), C(I, 'keydown'), C(I, 'keyup'), C('shift', 'keyup')) : (C(I, 'keydown'), C(I, 'keyup')), (clientWindow.blockNextInput = !0); + } + } else ('d' != g && A.data) || (C('backspace', 'keydown'), C('backspace', 'keyup'), (clientWindow.blockNextInput = !1)); + } + (clientWindow.lastKey = I), (h.value = ' '); + }, + !1 + ), + h.addEventListener( + 'input', + function (A) { + ' ' != h.value && (h.value = ' '); + }, + !1 + ), + h.addEventListener( + 'keydown', + function (A) { + (229 != A.keyCode && 229 != A.which) || clientWindow.keyboardFix || (clientWindow.console.warn('Switching from keydown to input events due to invalid KeyboardEvent. Some functionality will be lost.'), (clientWindow.keyboardFix = !0), clientWindow.lastKey && (C(clientWindow.lastKey, 'keydown'), C(clientWindow.lastKey, 'keyup'))); + }, + !1 + ), + h.addEventListener('blur', function (A) { + clientWindow.hiddenInputFocused = !1; + }), + document.body.appendChild(h); + let a = Z('keyboardButton', 'inMenu'); + (a.style.cssText = 'top: 0vh; margin: auto; left: 8vh; right:0vh; width: 8vh; height: 8vh;'), + a.addEventListener( + 'touchstart', + function (A) { + A.preventDefault(); + }, + !1 + ), + a.addEventListener( + 'touchend', + function (A) { + A.preventDefault(), clientWindow.hiddenInputFocused ? h.blur() : (h.select(), (clientWindow.hiddenInputFocused = !0)); + }, + !1 + ), + document.body.appendChild(a); + let p = Z('placeButton', 'inGame'); + (p.style.cssText = 'right:0vh;bottom:20vh;'), + p.addEventListener( + 'touchstart', + function (A) { + t(2, 'mousedown', i); + }, + !1 + ), + p.addEventListener( + 'touchend', + function (A) { + t(2, 'mouseup', i); + }, + !1 + ), + document.body.appendChild(p); + let R = Z('breakButton', 'inGame'); + (R.style.cssText = 'right:10vh;bottom:20vh;'), + R.addEventListener( + 'touchstart', + function (A) { + t(0, 'mousedown', i); + }, + !1 + ), + R.addEventListener( + 'touchend', + function (A) { + t(0, 'mouseup', i); + }, + !1 + ), + document.body.appendChild(R); + let W = Z('selectButton', 'inGame'); + (W.style.cssText = 'right:20vh;bottom:20vh;'), + W.addEventListener( + 'touchstart', + function (A) { + t(1, 'mousedown', i); + }, + !1 + ), + W.addEventListener( + 'touchend', + function (A) { + t(1, 'mouseup', i); + }, + !1 + ), + document.body.appendChild(W); + let u = Z('scrollUpButton', 'inGame'); + (u.style.cssText = 'right:0vh;bottom:0vh;'), + u.addEventListener( + 'touchstart', + function (A) { + G(i, -10); + }, + !1 + ), + document.body.appendChild(u); + let Y = Z('scrollDownButton', 'inGame'); + (Y.style.cssText = 'right:10vh;bottom:0vh;'), + Y.addEventListener( + 'touchstart', + function (A) { + G(i, 10); + }, + !1 + ), + document.body.appendChild(Y); + let S = Z('throwButton', 'inGame'); + (S.style.cssText = 'right:10vh;bottom:30vh;'), + S.addEventListener( + 'touchstart', + function (A) { + C('q', 'keydown'); + }, + !1 + ), + S.addEventListener( + 'touchend', + function (A) { + C('q', 'keyup'); + }, + !1 + ), + document.body.appendChild(S); + let j = Z('sprintButton', 'inGame'); + (j.style.cssText = 'right:0vh;bottom:10vh;'), + j.addEventListener( + 'touchstart', + function (A) { + C('r', 'keydown'), + (clientWindow.sprintLock = !!clientWindow.sprintLock && null), + (clientWindow.sprintTimer = setTimeout(function (A) { + (clientWindow.sprintLock = null != clientWindow.sprintLock), j.classList.toggle('active'); + }, 1e3)); + }, + !1 + ), + j.addEventListener( + 'touchend', + function (A) { + clientWindow.sprintLock || (C('r', 'keyup'), j.classList.remove('active'), (clientWindow.sprintLock = !1)), clearTimeout(clientWindow.sprintTimer); + }, + !1 + ), + document.body.appendChild(j); + let y = Z('pauseButton', 'inGame'); + (y.style.cssText = 'top: 0vh; margin: auto; left: 0vh; right: 32vh; width: 8vh; height: 8vh;'), + y.addEventListener( + 'touchstart', + function (A) { + C('`', 'keydown'); + }, + !1 + ), + y.addEventListener( + 'touchend', + function (A) { + C('`', 'keyup'); + }, + !1 + ), + document.body.appendChild(y); + let L = Z('chatButton', 'inGame'); + (L.style.cssText = 'top: 0vh; margin: auto; left: 0vh; right: 16vh; width: 8vh; height: 8vh;'), + L.addEventListener( + 'touchstart', + function (A) { + C('t', 'keydown'); + }, + !1 + ), + document.body.appendChild(L); + let N = Z('perspectiveButton', 'inGame'); + (N.style.cssText = 'top: 0vh; margin: auto; left: 0vh; right: 0vh; width: 8vh; height: 8vh;'), + N.addEventListener( + 'touchstart', + function (A) { + C('f', 'keydown'), C('5', 'keydown'); + }, + !1 + ), + N.addEventListener( + 'touchend', + function (A) { + C('f', 'keyup'), C('5', 'keyup'); + }, + !1 + ), + document.body.appendChild(N); + let U = Z('screenshotButton', 'inGame'); + (U.style.cssText = 'top: 0vh; margin: auto; left: 16vh; right: 0vh; width: 8vh; height: 8vh;'), + U.addEventListener( + 'touchstart', + function (A) { + C('f', 'keydown'), C('2', 'keydown'); + }, + !1 + ), + U.addEventListener( + 'touchend', + function (A) { + C('f', 'keyup'), C('2', 'keyup'); + }, + !1 + ), + document.body.appendChild(U); + let z = Z('coordinatesButton', 'inGame'); + (z.style.cssText = 'top: 0vh; margin: auto; left: 32vh; right: 0vh; width: 8vh; height: 8vh;'), + z.addEventListener( + 'touchstart', + function (A) { + C('f', 'keydown'), C('3', 'keydown'); + }, + !1 + ), + z.addEventListener( + 'touchend', + function (A) { + C('f', 'keyup'), C('3', 'keyup'); + }, + !1 + ), + document.body.appendChild(z); + })(); + }); + let b = document.createElement('style'); + (b.textContent = + '\n.mobileControl, .mobileControl:active, .mobileControl.active{\nposition: absolute; \nwidth: 10vh;\nheight: 10vh;\nfont-size:4vh;\n-webkit-user-select: none;\n-ms-user-select: none;\nuser-select: none;\nline-height: 0px;\npadding:0px;\nbackground-color: transparent;\nbox-sizing: content-box;\nimage-rendering: pixelated;\nbackground-size: cover;\noutline:none;\nbox-shadow: none;\nborder: none;\n}\n.mobileControl:active, .mobileControl.active {\nposition: absolute; \nwidth: 10vh;\nheight: 10vh;\nfont-size:4vh;\n-webkit-user-select: none;\n-ms-user-select: none;\nuser-select: none;\nline-height: 0px;\npadding:0px;\nbackground-color: transparent;\ncolor: #ffffff;\ntext-shadow: 0.35vh 0.35vh #000000;\nbox-sizing: content-box;\nimage-rendering: pixelated;\nbackground-size: contain, cover;\noutline:none;\nbox-shadow: none;\nborder: none;\n}\nhtml, body, canvas {\nheight: 100svh !important;\nheight: -webkit-fill-available !important;\ntouch-action: pan-x pan-y;\n-webkit-touch-callout: none;\n-webkit-user-select: none;\n-khtml-user-select: none;\n-moz-user-select: none;\n-ms-user-select: none;\nuser-select: none;\noutline: none;\n-webkit-tap-highlight-color: rgba(255, 255, 255, 0);\n}\n.hide {\ndisplay: none;\n}\n#fileUpload {\nposition: absolute;\nleft: 0;\nright: 100vw;\ntop: 0; \nbottom: 100vh;\nwidth: 100vw;\nheight: 100vh;\nbackground-color:rgba(255,255,255,0.5);\n}\n.strafeRightButton {\nbackground-image: url("");\n}\n.strafeRightButton.active, .strafeRightButton:active {\nbackground-image: url("");\n}\n.strafeLeftButton {\nbackground-image: url("");\n}\n.strafeLeftButton.active, .strafeLeftButton:active {\nbackground-image: url("");\n}\n.forwardButton {\nbackground-image: url("");\n}\n.forwardButton.active, .forwardButton:active {\nbackground-image: url("");\n}\n.rightButton {\nbackground-image: url("");\n}\n.rightButton.active, .rightButton:active {\nbackground-image: url("");\n}\n.leftButton {\nbackground-image: url("");\n}\n.leftButton.active, .leftButton:active {\nbackground-image: url("");\n}\n.backButton {\nbackground-image: url("");\n}\n.backButton.active, .backButton:active {\nbackground-image: url("");\n}\n.jumpButton {\nbackground-image: url("");\n}\n.jumpButton.active, .jumpButton:active {\nbackground-image: url("");\n}\n.crouchButton {\nbackground-image: url("");\n}\n.crouchButton:active {\nbackground-image: url("");\n}\n.crouchButton.active {\nbackground-image: url("");\n}\n.inventoryButton {\nbackground-image: url("");\n}\n.inventoryButton.active, .inventoryButton:active {\nbackground-image: url("");\n}\n.chatButton {\nbackground-image: url("");\n}\n.chatButton.active, .chatButton:active {\nbackground-image: url("");\n}\n.pauseButton {\nbackground-image: url("");\n}\n.pauseButton.active, .pauseButton:active {\nbackground-image: url("");\n}\n.exitButton {\nbackground-image: url("");\n}\n.exitButton.active, .exitButton:active {\nbackground-image: url("");\n}\n.keyboardButton {\nbackground-image: url("")\n}\n.keyboardButton.active, .keyboardButton:active {\nbackground-image: url("")\n}\n.placeButton {\nbackground-image: url("");\n}\n.placeButton.active, .placeButton:active {\nbackground-image: url("");\n}\n.breakButton {\nbackground-image: url("");\n}\n.breakButton.active, .breakButton:active {\nbackground-image: url("");\n}\n.selectButton {\nbackground-image: url("");\n}\n.selectButton.active, .selectButton:active {\nbackground-image: url("");\n}\n.scrollUpButton {\nbackground-image: url("");\n}\n.scrollUpButton.active, .scrollUpButton:active {\nbackground-image: url("");\n}\n.scrollDownButton {\nbackground-image: url("");\n}\n.scrollDownButton.active, .scrollDownButton:active {\nbackground-image: url("");\n}\n.throwButton {\nbackground-image: url("");\n}\n.throwButton.active, .throwButton:active {\nbackground-image: url("");\n}\n.sprintButton {\nbackground-image: url("");\n}\n.sprintButton.active, .sprintButton:active {\nbackground-image: url("");\n}\n.perspectiveButton {\nbackground-image: url("");\n}\n.perspectiveButton.active, .perspectiveButton:active {\nbackground-image: url("");\n}\n.screenshotButton {\nbackground-image: url("");\n}\n.screenshotButton.active, .screenshotButton:active {\nbackground-image: url("");\n}\n.coordinatesButton {\nbackground-image: url("");\n}\n.coordinatesButton.active, .coordinatesButton:active {\nbackground-image: url("");\n}\n'), + document.documentElement.appendChild(b); +} diff --git a/src/resources/scripts/eagler-launch/1.8.8/eaglerpocketmobile.ts b/src/resources/scripts/eagler-launch/1.8.8/eaglerpocketmobile.ts new file mode 100644 index 0000000..7eed39a --- /dev/null +++ b/src/resources/scripts/eagler-launch/1.8.8/eaglerpocketmobile.ts @@ -0,0 +1,1156 @@ +/* eslint-disable */ +// @ts-nocheck +if (new URLSearchParams(window.location.search).get('mobile') === 'true') { + // Removed brainless unsafeWindow + console.log('Eagler Pocket Mobile v1.40'); + // TODO: remove the mobile check is implement the dynamic enabling and disabling of individual features + function isMobile() { + try { + document.createEvent('TouchEvent'); + return true; + } catch (e) { + return false; + } + } + if (!isMobile()) { + alert('WARNING: This script was created for mobile, and may break functionality in non-mobile browsers!'); + } + // TODO: consolidate all of these into a single object? + window.crouchLock = false; // Used for crouch mobile control + window.sprintLock = false; // Used for sprint mobile control + window.keyboardFix = false; // keyboardFix ? "Standard Keyboard" : "Compatibility Mode" + window.inputFix = false; // If true, Duplicate Mode + window.blockNextInput = false; // Used for Duplicate Mode + window.hiddenInputFocused = false; // Used for keyboard display on mobile + window.canvasTouchMode = 0; // Used for canvas touch handling + /* + 0 Idle + 1 Touch initiated + 2 Primary touch + 3 Secondary touch + 4 Scroll + 5 Finished +*/ + window.canvasTouchStartX = null; + window.canvasTouchStartY = null; + window.canvasTouchPreviousX = null; + window.canvasTouchPreviousY = null; + window.canvasPrimaryID = null; + window.buttonTouchStartX = null; + + // charCodeAt is designed for unicode characters, and doesn't match the behavior of the keyCodes used by KeyboardEvents, thus necessitating this function + String.prototype.toKeyCode = function () { + const keyCodeList = { + '0': 48, + '1': 49, + '2': 50, + '3': 51, + '4': 52, + '5': 53, + '6': 54, + '7': 55, + '8': 56, + '9': 57, + backspace: 8, + tab: 9, + enter: 13, + shift: 16, + ctrl: 17, + alt: 18, + pause_break: 19, + caps_lock: 20, + escape: 27, + ' ': 32, + page_up: 33, + page_down: 34, + end: 35, + home: 36, + left_arrow: 37, + up_arrow: 38, + right_arrow: 39, + down_arrow: 40, + insert: 45, + delete: 46, + a: 65, + b: 66, + c: 67, + d: 68, + e: 69, + f: 70, + g: 71, + h: 72, + i: 73, + j: 74, + k: 75, + l: 76, + m: 77, + n: 78, + o: 79, + p: 80, + q: 81, + r: 82, + s: 83, + t: 84, + u: 85, + v: 86, + w: 87, + x: 88, + y: 89, + z: 90, + left_window_key: 91, + right_window_key: 92, + select_key: 93, + numpad_0: 96, + numpad_1: 97, + numpad_2: 98, + numpad_3: 99, + numpad_4: 100, + numpad_5: 101, + numpad_6: 102, + numpad_7: 103, + numpad_8: 104, + numpad_9: 105, + '*': 106, + '+': 107, + '-': 109, + '.': 110, + '/': 111, + f1: 112, + f2: 113, + f3: 114, + f4: 115, + f5: 116, + f6: 117, + f7: 118, + f8: 119, + f9: 120, + f10: 121, + f11: 122, + f12: 123, + num_lock: 144, + scroll_lock: 145, + ';': 186, + '=': 187, + ',': 188, + '-': 189, + '.': 190, + '/': 191, + '\u0060': 192, + '[': 219, + '\u005C': 220, + ']': 221, + '\u0022': 222, + }; + return keyCodeList[this]; + }; + // Overrides the addEventListener behavior to all code injection on keydown event listeners. This function has thrown TypeErrors on some Android devices because fn is not recognized as a function + // This is used by Compatibility Mode to block invalid keyEvents + const _addEventListener = EventTarget.prototype.addEventListener; + Object.defineProperty(EventTarget.prototype, 'addEventListener', { + value: function (type, fn, ...rest) { + if (type == 'keydown') { + // Check if a keydown event is being added + _addEventListener.call( + this, + type, + function (...args) { + if (args[0].isTrusted && window.keyboardFix) { + // When we are in compatibility mode, we ignore all trusted keyboard events + return; + } + return fn.apply(this, args); // Appends the rest of the function specified by addEventListener + }, + ...rest + ); + } else { + // If it's not a keydown event, behave like normal (hopefully) + _addEventListener.call(this, type, fn, ...rest); + } + }, + }); + // Overrides preventDefault, because on some (Android) devices you couldn't type into hiddenInput + const _preventDefault = Event.prototype.preventDefault; + Event.prototype.preventDefault = function (shouldBypass) { + if (document.activeElement.id != 'hiddenInput' || shouldBypass) { + // activeElement is what element is currently focused + this._preventDefault = _preventDefault; + this._preventDefault(); + } + }; + // Key and mouse events + // Note: the client must have the key, keyCode, and which parameters defined or it will crash + // Note: for text inputs, the client only reads from the "key" paramater + // * an exception to this appears to be the shift and backspace key + // Note: for inGame inputs, the client only reads from the "keyCode character" + function keyEvent(name, state) { + const charCode = name.toKeyCode(); + const evt = new KeyboardEvent(state, { + key: name, + keyCode: charCode, + which: charCode, + }); + window.dispatchEvent(evt); + } + function mouseEvent(number, state, element, event = { clientX: 0, clientY: 0, screenX: 0, screenY: 0 }) { + element.dispatchEvent( + new PointerEvent(state, { + button: number, + buttons: number, + clientX: event.clientX, + clientY: event.clientY, + screenX: event.screenX, + screenY: event.screenY, + }) + ); + } + function wheelEvent(element, delta) { + element.dispatchEvent( + new WheelEvent('wheel', { + wheelDeltaY: delta, + }) + ); + } + function setButtonVisibility(pointerLocked) { + const inGameStyle = document.getElementById('inGameStyle'); + const inMenuStyle = document.getElementById('inMenuStyle'); + inGameStyle.disabled = pointerLocked; + inMenuStyle.disabled = !pointerLocked; + } + // POINTERLOCK + // When requestpointerlock is called, this dispatches an event, saves the requested element to window.fakelock, and unhides the touch controls + window.fakelock = null; + + Object.defineProperty(Element.prototype, 'requestPointerLock', { + value: function () { + window.fakelock = this; + document.dispatchEvent(new Event('pointerlockchange')); + setButtonVisibility(true); + return true; + }, + }); + + // Makes pointerLockElement return window.fakelock + Object.defineProperty(Document.prototype, 'pointerLockElement', { + get: function () { + return window.fakelock; + }, + }); + // When exitPointerLock is called, this dispatches an event, clears the + Object.defineProperty(Document.prototype, 'exitPointerLock', { + value: function () { + window.fakelock = null; + document.dispatchEvent(new Event('pointerlockchange')); + setButtonVisibility(false); + return true; + }, + }); + + // FULLSCREEN + window.fakefull = null; + // Stops the client from crashing when fullscreen is requested + Object.defineProperty(Element.prototype, 'requestFullscreen', { + value: function () { + window.fakefull = this; + document.dispatchEvent(new Event('fullscreenchange')); + return true; + }, + }); + Object.defineProperty(document, 'fullscreenElement', { + get: function () { + return window.fakefull; + }, + }); + Object.defineProperty(Document.prototype, 'exitFullscreen', { + value: function () { + window.fakefull = null; + document.dispatchEvent(new Event('fullscreenchange')); + return true; + }, + }); + + // FILE UPLOADING + // Safari doesn't recognize the element.click() used to display the file uploader as an action performed by the user, so it ignores it. + // This hijacks the element.createElement() function to add the file upload to the DOM, so the user can manually press the button again. + const _createElement = document.createElement; + document.createElement = function (type, ignore) { + this._createElement = _createElement; + const element = this._createElement(type); + if (type == 'input' && !ignore) { + // We set the ingore flag to true when we create the hiddenInput + document.querySelectorAll('#fileUpload').forEach((e) => e.parentNode.removeChild(e)); // Get rid of any left over fileUpload inputs + element.id = 'fileUpload'; + element.addEventListener( + 'change', + function (e) { + element.hidden = true; + element.style.display = 'none'; + }, + { passive: false, once: true } + ); + window.addEventListener( + 'focus', + function (e) { + setTimeout(() => { + element.hidden = true; + element.style.display = 'none'; + }, 300); + }, + { once: true } + ); + document.body.appendChild(element); + } + return element; + }; + + // Lazy way to hide touch controls through CSS. + const inGameStyle = document.createElement('style'); + inGameStyle.id = 'inGameStyle'; + inGameStyle.textContent = ` + .inGame { + display: none; + }`; + document.documentElement.appendChild(inGameStyle); + + const inMenuStyle = document.createElement('style'); + inMenuStyle.id = 'inMenuStyle'; + inMenuStyle.textContent = ` + .inMenu { + display: none; + }`; + document.documentElement.appendChild(inMenuStyle); + + // The canvas is created by the client after it finishes unzipping and loading. When the canvas is created, this applies any necessary event listeners and creates buttons + function waitForElm(selector) { + return new Promise((resolve) => { + if (document.querySelector(selector)) { + return resolve(document.querySelector(selector)); + } + const observer = new MutationObserver((mutations) => { + if (document.querySelector(selector)) { + observer.disconnect(); + resolve(document.querySelector(selector)); + } + }); + observer.observe(document.documentElement, { + childList: true, + subtree: true, + }); + }); + } + function createTouchButton(buttonClass, buttonDisplay, elementName) { + const touchButton = document.createElement(elementName ?? 'button', true); + touchButton.classList.add(buttonClass); + touchButton.classList.add(buttonDisplay); + touchButton.classList.add('mobileControl'); + touchButton.addEventListener( + 'touchmove', + function (e) { + e.preventDefault(); + }, + false + ); + touchButton.addEventListener('contextmenu', function (e) { + e.preventDefault(); + }); + return touchButton; + } + + waitForElm('canvas').then(() => { + insertCanvasElements(); + }); + function insertCanvasElements() { + // Translates touchmove events to mousemove events when inGame, and touchmove events to wheele events when inMenu + const canvas = document.querySelector('canvas'); + canvas.addEventListener( + 'touchstart', + function (e) { + if (window.canvasTouchMode < 2) { + // If a touch is initiated but not assigned + if (window.canvasPrimaryID == null) { + window.canvasTouchMode = 1; + const primaryTouch = e.changedTouches[0]; + window.canvasPrimaryID = primaryTouch.identifier; + canvasTouchStartX = primaryTouch.clientX; + canvasTouchStartY = primaryTouch.clientY; + canvasTouchPreviousX = canvasTouchStartX; + canvasTouchPreviousY = canvasTouchStartY; + + window.touchTimer = setTimeout(function (e) { + // If our touch is still set to initiaited, set it to secondary touch + if (window.canvasTouchMode == 1) { + window.canvasTouchMode = 3; + mouseEvent(2, 'mousedown', canvas, primaryTouch); + if (window.fakelock) { + // We only dispatch mouseup inGame because we want to be able to click + drag items in GUI's + mouseEvent(2, 'mouseup', canvas, primaryTouch); + } + } + }, 300); + } else if (window.canvasTouchMode == 1 && !window.fakelock) { + // If we already have a primary touch, it means we're using two fingers + window.canvasTouchMode = 4; + clearTimeout(window.crouchTimer); // TODO: Find out why this isn't redudnant + } + } + }, + false + ); + + canvas.addEventListener( + 'touchmove', + function (e) { + e.preventDefault(); // Prevents window zoom when using two fingers + let primaryTouch; + for (let touchIndex = 0; touchIndex < e.targetTouches.length; touchIndex++) { + // Iterate through our touches to find a touch event matching the primary touch ID + if (e.targetTouches[touchIndex].identifier == window.canvasPrimaryID) { + primaryTouch = e.targetTouches[touchIndex]; + break; + } + } + if (primaryTouch) { + primaryTouch.distanceX = primaryTouch.clientX - canvasTouchStartX; + primaryTouch.distanceY = primaryTouch.clientY - canvasTouchStartY; + primaryTouch.squaredNorm = primaryTouch.distanceX * primaryTouch.distanceX + primaryTouch.distanceY * primaryTouch.distanceY; + primaryTouch.movementX = primaryTouch.clientX - canvasTouchPreviousX; + primaryTouch.movementY = primaryTouch.clientY - canvasTouchPreviousY; + if (window.canvasTouchMode == 1) { + // If the primary touch is still only initiated + if (primaryTouch.squaredNorm > 25) { + // If our touch becomes a touch + drag + clearTimeout(window.crouchTimer); + window.canvasTouchMode = 2; + if (!window.fakelock) { + // When we're inGame, we don't want to be placing blocks when we are moving the camera around + mouseEvent(1, 'mousedown', canvas, primaryTouch); + } + } + } else { + // If our touch is primary, secondary, scroll or finished + if (window.canvasTouchMode == 4) { + // If our touch is scrolling + wheelEvent(canvas, primaryTouch.movementY); + } else { + canvas.dispatchEvent( + new MouseEvent('mousemove', { + clientX: primaryTouch.clientX, + clientY: primaryTouch.clientY, + screenX: primaryTouch.screenX, + screenY: primaryTouch.screenY, // The top four are used for item position when in GUI's, the bottom two are for moving the camera inGame + movementX: primaryTouch.movementX, + movementY: primaryTouch.movementY, + }) + ); + } + } + canvasTouchPreviousX = primaryTouch.clientX; + canvasTouchPreviousY = primaryTouch.clientY; + } + }, + false + ); + + function canvasTouchEnd(e) { + for (let touchIndex = 0; touchIndex < e.changedTouches.length; touchIndex++) { + // Iterate through changed touches to find primary touch + if (e.changedTouches[touchIndex].identifier == window.canvasPrimaryID) { + const primaryTouch = e.changedTouches[touchIndex]; + // When any of the controlling fingers go away, we want to wait until we aren't receiving any other touch events + if (window.canvasTouchMode == 2) { + mouseEvent(1, 'mouseup', canvas, primaryTouch); + } else if (window.canvasTouchMode == 3) { + e.preventDefault(); // This prevents some mobile devices from dispatching a mousedown + mouseup event after a touch is ended + mouseEvent(2, 'mouseup', canvas, primaryTouch); + } + window.canvasTouchMode = 5; + } + } + if (e.targetTouches.length == 0) { + // We want to wait until all fingers are off the canvas before we reset for the next cycle + window.canvasTouchMode = 0; + window.canvasPrimaryID = null; + } + } + + canvas.addEventListener('touchend', canvasTouchEnd, false); + canvas.addEventListener('touchcancel', canvasTouchEnd, false); // TODO: Find out why this is different than touchend + setButtonVisibility(window.fakelock != null); //Updates our mobile controls when the canvas finally loads + // All of the touch buttons + const strafeRightButton = createTouchButton('strafeRightButton', 'inGame', 'div'); + strafeRightButton.classList.add('strafeSize'); + strafeRightButton.style.cssText = 'left:24vh;bottom:22vh;'; + document.body.appendChild(strafeRightButton); + const strafeLeftButton = createTouchButton('strafeLeftButton', 'inGame', 'div'); + strafeLeftButton.classList.add('strafeSize'); + strafeLeftButton.style.cssText = 'left:5.5vh;bottom:22vh;'; + document.body.appendChild(strafeLeftButton); + + const forwardButton = createTouchButton('forwardButton', 'inGame', 'div'); // We use a div here so can use the targetTouches property of touchmove events. If we didn't it would require me to make an actual touch handler and I don't want to + forwardButton.style.cssText = 'left:14vh;bottom:22vh;'; + forwardButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('w', 'keydown'); + strafeRightButton.classList.remove('hide'); + strafeLeftButton.classList.remove('hide'); + forwardButton.classList.add('active'); + }, + false + ); + forwardButton.addEventListener( + 'touchmove', + function (e) { + e.preventDefault(); + const touch = e.targetTouches[0]; // We are just hoping that the user will only ever use one finger on the forward button + + if (!buttonTouchStartX) { + // TODO: move this to a touchstart event handler + buttonTouchStartX = touch.pageX; + } + const movementX = touch.pageX - buttonTouchStartX; + if (movementX * 10 > window.innerHeight) { + strafeRightButton.classList.add('active'); + strafeLeftButton.classList.remove('active'); + keyEvent('d', 'keydown'); + keyEvent('a', 'keyup'); + } else if (movementX * 10 < 0 - window.innerHeight) { + strafeLeftButton.classList.add('active'); + strafeRightButton.classList.remove('active'); + keyEvent('a', 'keydown'); + keyEvent('d', 'keyup'); + } else { + strafeRightButton.classList.remove('active'); + strafeLeftButton.classList.remove('active'); + } + }, + false + ); + forwardButton.addEventListener( + 'touchend', + function (e) { + keyEvent('w', 'keyup'); // Luckily, it doesn't seem like eagler cares if we dispatch extra keyup events, so we can get away with just dispatching all of them here + keyEvent('d', 'keyup'); + keyEvent('a', 'keyup'); + strafeRightButton.classList.remove('active'); + strafeLeftButton.classList.remove('active'); + strafeRightButton.classList.add('hide'); + strafeLeftButton.classList.add('hide'); + forwardButton.classList.remove('active'); + + buttonTouchStartX = null; + }, + false + ); + strafeRightButton.classList.add('hide'); + strafeLeftButton.classList.add('hide'); + document.body.appendChild(forwardButton); + + const rightButton = createTouchButton('rightButton', 'inGame'); + rightButton.style.cssText = 'left:24vh;bottom:12vh;'; + rightButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('d', 'keydown'); + }, + false + ); + rightButton.addEventListener( + 'touchend', + function (e) { + keyEvent('d', 'keyup'); + }, + false + ); + document.body.appendChild(rightButton); + const leftButton = createTouchButton('leftButton', 'inGame'); + leftButton.style.cssText = 'left: 4vh; bottom:12vh;'; + leftButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('a', 'keydown'); + }, + false + ); + leftButton.addEventListener( + 'touchend', + function (e) { + keyEvent('a', 'keyup'); + }, + false + ); + document.body.appendChild(leftButton); + const backButton = createTouchButton('backButton', 'inGame'); + backButton.style.cssText = 'left:14vh;bottom:2vh;'; + backButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('s', 'keydown'); + }, + false + ); + backButton.addEventListener( + 'touchend', + function (e) { + keyEvent('s', 'keyup'); + }, + false + ); + document.body.appendChild(backButton); + const jumpButton = createTouchButton('jumpButton', 'inGame'); + jumpButton.style.cssText = 'right:20vh;bottom:20vh;'; + jumpButton.addEventListener( + 'touchstart', + function (e) { + keyEvent(' ', 'keydown'); + }, + false + ); + jumpButton.addEventListener( + 'touchend', + function (e) { + keyEvent(' ', 'keyup'); + }, + false + ); + document.body.appendChild(jumpButton); + + const crouchButton = createTouchButton('crouchButton', 'inGame'); + crouchButton.style.cssText = 'left:14vh;bottom:12vh;'; + crouchButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('shift', 'keydown'); + window.crouchLock = window.crouchLock ? null : false; + window.crouchTimer = setTimeout(function (e) { + // Allows us to lock the button after a long press + window.crouchLock = window.crouchLock != null; + crouchButton.classList.toggle('active'); + }, 1000); + }, + false + ); + + crouchButton.addEventListener( + 'touchend', + function (e) { + if (!window.crouchLock) { + keyEvent('shift', 'keyup'); + crouchButton.classList.remove('active'); + window.crouchLock = false; + } + clearTimeout(window.crouchTimer); + }, + false + ); + document.body.appendChild(crouchButton); + const inventoryButton = createTouchButton('inventoryButton', 'inGame'); + inventoryButton.classList.add('smallMobileControl'); + inventoryButton.style.cssText = 'right:19.5vh;bottom:0vh;'; + inventoryButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('e', 'keydown'); + }, + false + ); + inventoryButton.addEventListener( + 'touchend', + function (e) { + keyEvent('shift', 'keydown'); // Sometimes shift gets stuck on, which interferes with item manipulation in GUI's + keyEvent('shift', 'keyup'); // Sometimes shift gets stuck on, which interferes with item manipulation in GUI's + keyEvent('e', 'keyup'); + }, + false + ); + document.body.appendChild(inventoryButton); + const exitButton = createTouchButton('exitButton', 'inMenu'); + exitButton.classList.add('smallMobileControl'); + exitButton.style.cssText = 'top: 0.5vh; margin: auto; left: 1vh; right:8vh;'; + exitButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('`', 'keydown'); + }, + false + ); + exitButton.addEventListener( + 'touchend', + function (e) { + keyEvent('`', 'keyup'); + }, + false + ); + document.body.appendChild(exitButton); + // ---Input Handling--- + // This code is a mess, specifically because Android is so so SO inconsistent with how it handles the keyboard + // Some keyboards dispatch key events, some directly append text, and none of them meet the most basic standards supported by most other devices + // This mess is my attempt at dealing with that, and it will most likely only ever be triggered by Android + // + // It has three main modes. + // 1) Standard keyboard mode: + // This mode keeps the hiddenInput empty, saves the last key press, and on every keypress checks if it the keys are being pressed incorrectly. + // If there is a problem, it switches to compatibility mode, using beforeinput and input events instead of keydown and keyup + // 2) Compatibility mode: + // This most is most likely going to be used by Android, because Android only every dispatches keyCode 229 for any keypress + // When we enter this mode, we grab the last known key press and redispatch it, and programatically dispatch key events by reading e.inputType and e.data from beforeinput + // Unfortunately, Android is weird with this as well. Sometimes it only dispatches insertCompositionText events, and sometimes it gives the correct inputTypes as well + // Additionally, programmatically setting the input's text contents (BECAUSE ANDROID IGNORES PREVENTDEFAULT AGHHHHH) dispatches a repeat of the previous event + // Luckily, we can check if this happens when we first create the input, which necessitates the third mode: + // 3) Duplicate mode: + // If we are getting duplicate inputs, this mode ignores every other input if it matches the state saved in window.previousKey + // If users make it to this mode and still are having issues, it may be best just to remove support for their device + // ---Input Handling--- + const hiddenInput = document.createElement('input', true); + hiddenInput.id = 'hiddenInput'; + hiddenInput.classList.add('inMenu'); + hiddenInput.style.cssText = 'position:absolute;top: 0vh; margin: auto; left: 8vh; right:0vh; width: 8vh; height: 8vh;font-size:20px;z-index: -10;color: transparent;text-shadow: 0 0 0 black;'; // We hide the input behind a key because display: none and opacity:0 causes issues + hiddenInput.addEventListener( + 'beforeinput', + function (e) { + // For some reason beforeinput doesn't have the same deletion problems that input has on Android + e.stopImmediatePropagation(); // Android ignores this and the prevent default, so this will probably be removed in the future + e.preventDefault(true); // We pass a value because we've hijacked the prevent default function to have a "should bypass" boolean value + const inputData = e.inputType == 'insertLineBreak' ? 'return' : e.data == null ? 'delete' : e.data.slice(-1); // Saves the last key press. + if (!window.lastKey) { + // When we first set hiddenInput's text contents to " " we can use this to check if Duplicate Mode is needed + window.console.warn('Enabling blocking duplicate key events. Some functionality may be lost.'); + window.inputFix = true; + } + if (window.keyboardFix) { + if (e.inputType == 'insertLineBreak') { + // Detects return key press + keyEvent('enter', 'keydown'); + keyEvent('enter', 'keyup'); + } else { + const sliceInputType = e.inputType.slice(0, 1); // Android doesn't constiently dispatch the correct inputType, but most of them either start with i for insert or d for delete, so this dumb solution should be good enough. + if (sliceInputType == 'i' && e.data) { + // Android sometimes always dispatches insertCompositionText inputTypes, so checking that e.data isn't null is necessary + const isDuplicate = window.lastKey == inputData && window.blockNextInput && window.inputFix; + if (isDuplicate) { + // If our key press matches the last unblocked key press and we are in duplicaye mode, ignore the input + window.blockNextInput = false; + } else { + const isShift = inputData.toLowerCase() != inputData; + if (isShift) { + // The Eaglerclient only uses e.key, e.keyCode and e.which, so we have to dispatch the shift key event separately + keyEvent('shift', 'keydown'); + keyEvent(inputData, 'keydown'); + keyEvent(inputData, 'keyup'); + keyEvent('shift', 'keyup'); + } else { + keyEvent(inputData, 'keydown'); + keyEvent(inputData, 'keyup'); + } + window.blockNextInput = true; + } + } else if (sliceInputType == 'd' || !e.data) { + keyEvent('backspace', 'keydown'); + keyEvent('backspace', 'keyup'); + window.blockNextInput = false; // If we delete a character, there couldn't be a duplicate of the previous key press + } + } + } + window.lastKey = inputData; // Saves the last key pressed + hiddenInput.value = ' '; //This previously allowed us to have a character to delete, but beforeinput doesn't require this. This does allow us to check wether Duplicate Mode is necessary though + }, + false + ); + hiddenInput.addEventListener( + 'input', + function (e) { + // Since we are using beforeInput for input detection, setting the text contents of hiddenInput causes weird behavior, so we use input instead + if (hiddenInput.value != ' ') { + // Avoid updating it if not needed so Duplicate Mode doesn't throw a fit + hiddenInput.value = ' '; + } + }, + false + ); + hiddenInput.addEventListener( + 'keydown', + function (e) { + // Enables Compatibility Mode if we receive an invalid key press event + if ((e.keyCode == 229 || e.which == 229) && !window.keyboardFix) { + window.console.warn('Switching from keydown to input events due to invalid KeyboardEvent. Some functionality will be lost.'); + window.keyboardFix = true; + if (window.lastKey) { + // Resend the last saved key press (which is being tracked by the beforeinput event listener) so the transition to Compatibility Mode isn't noticeable + keyEvent(window.lastKey, 'keydown'); + keyEvent(window.lastKey, 'keyup'); + } + } + }, + false + ); + hiddenInput.addEventListener('blur', function (e) { + // Updates window.hiddenInputFocused to reflect the actual state of the focus + window.hiddenInputFocused = false; + }); + document.body.appendChild(hiddenInput); + const keyboardButton = createTouchButton('keyboardButton', 'inMenu'); + keyboardButton.classList.add('smallMobileControl'); + keyboardButton.style.cssText = 'top: 0.5vh; margin: auto; left: 6vh; right:0vh;'; + keyboardButton.addEventListener( + 'touchstart', + function (e) { + e.preventDefault(); + }, + false + ); + keyboardButton.addEventListener( + 'touchend', + function (e) { + e.preventDefault(); + if (window.hiddenInputFocused) { + hiddenInput.blur(); + } else { + hiddenInput.select(); + window.hiddenInputFocused = true; + } + }, + false + ); + document.body.appendChild(keyboardButton); + const placeButton = createTouchButton('placeButton', 'inGame'); + placeButton.style.cssText = 'right:6vh;bottom:37vh;'; + placeButton.addEventListener( + 'touchstart', + function (e) { + mouseEvent(2, 'mousedown', canvas); + }, + false + ); + placeButton.addEventListener( + 'touchend', + function (e) { + mouseEvent(2, 'mouseup', canvas); + }, + false + ); + document.body.appendChild(placeButton); + const breakButton = createTouchButton('breakButton', 'inGame'); + breakButton.style.cssText = 'right:19vh;bottom:41vh;'; + breakButton.addEventListener( + 'touchstart', + function (e) { + mouseEvent(0, 'mousedown', canvas); + }, + false + ); + breakButton.addEventListener( + 'touchend', + function (e) { + mouseEvent(0, 'mouseup', canvas); + }, + false + ); + document.body.appendChild(breakButton); + const selectButton = createTouchButton('selectButton', 'inGame'); + selectButton.style.cssText = 'right:6vh;bottom:49vh;'; + selectButton.addEventListener( + 'touchstart', + function (e) { + mouseEvent(1, 'mousedown', canvas); + }, + false + ); + selectButton.addEventListener( + 'touchend', + function (e) { + mouseEvent(1, 'mouseup', canvas); + }, + false + ); + document.body.appendChild(selectButton); + const scrollUpButton = createTouchButton('scrollUpButton', 'inGame'); + scrollUpButton.classList.add('smallMobileControl'); + scrollUpButton.style.cssText = 'right:6.6vh;bottom:0vh;'; + scrollUpButton.addEventListener( + 'touchstart', + function (e) { + wheelEvent(canvas, -10); + }, + false + ); + document.body.appendChild(scrollUpButton); + const scrollDownButton = createTouchButton('scrollDownButton', 'inGame'); + scrollDownButton.classList.add('smallMobileControl'); + scrollDownButton.style.cssText = 'right:25.8vh;bottom:0vh;'; + scrollDownButton.addEventListener( + 'touchstart', + function (e) { + wheelEvent(canvas, 10); + }, + false + ); + document.body.appendChild(scrollDownButton); + const throwButton = createTouchButton('throwButton', 'inGame'); + throwButton.classList.add('smallMobileControl'); + throwButton.style.cssText = 'right:13vh;bottom:0vh;'; + throwButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('q', 'keydown'); + }, + false + ); + throwButton.addEventListener( + 'touchend', + function (e) { + keyEvent('q', 'keyup'); + }, + false + ); + document.body.appendChild(throwButton); + const sprintButton = createTouchButton('sprintButton', 'inGame'); + sprintButton.style.cssText = 'right:19vh;bottom:53vh;'; + sprintButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('r', 'keydown'); + window.sprintLock = window.sprintLock ? null : false; + window.sprintTimer = setTimeout(function (e) { + window.sprintLock = window.sprintLock != null; + sprintButton.classList.toggle('active'); + }, 1000); + }, + false + ); + + sprintButton.addEventListener( + 'touchend', + function (e) { + if (!window.sprintLock) { + keyEvent('r', 'keyup'); + sprintButton.classList.remove('active'); + window.sprintLock = false; + } + clearTimeout(window.sprintTimer); + }, + false + ); + document.body.appendChild(sprintButton); + const pauseButton = createTouchButton('pauseButton', 'inGame'); + pauseButton.classList.add('smallMobileControl'); + pauseButton.style.cssText = 'top: 0.5vh; margin: auto; left: 0vh; right: 0vh;'; + pauseButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('`', 'keydown'); + }, + false + ); + pauseButton.addEventListener( + 'touchend', + function (e) { + keyEvent('`', 'keyup'); + }, + false + ); + document.body.appendChild(pauseButton); + const chatButton = createTouchButton('chatButton', 'inGame'); + chatButton.classList.add('smallMobileControl'); + chatButton.style.cssText = 'top: 0.5vh; margin: auto; left: 0vh; right: 14vh;'; + chatButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('t', 'keydown'); + }, + false + ); // For some reason dispatching a keyup event for this closes the chat, which is really weird + document.body.appendChild(chatButton); + const perspectiveButton = createTouchButton('perspectiveButton', 'inGame'); + perspectiveButton.classList.add('smallMobileControl'); + perspectiveButton.style.cssText = 'top: 0.5vh; margin: auto; left: 0vh; right: 28vh;'; + perspectiveButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('f', 'keydown'); + keyEvent('5', 'keydown'); + }, + false + ); + perspectiveButton.addEventListener( + 'touchend', + function (e) { + keyEvent('f', 'keyup'); + keyEvent('5', 'keyup'); + }, + false + ); + document.body.appendChild(perspectiveButton); + const screenshotButton = createTouchButton('screenshotButton', 'inGame'); + screenshotButton.classList.add('smallMobileControl'); + screenshotButton.style.cssText = 'top: 0.5vh; margin: auto; left: 28vh; right: 0vh;'; + screenshotButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('f', 'keydown'); + keyEvent('2', 'keydown'); + }, + false + ); + screenshotButton.addEventListener( + 'touchend', + function (e) { + keyEvent('f', 'keyup'); + keyEvent('2', 'keyup'); + }, + false + ); + document.body.appendChild(screenshotButton); + const coordinatesButton = createTouchButton('coordinatesButton', 'inGame'); + coordinatesButton.classList.add('smallMobileControl'); + coordinatesButton.style.cssText = 'top: 0.5vh; margin: auto; left: 14vh; right: 0vh;'; + coordinatesButton.addEventListener( + 'touchstart', + function (e) { + keyEvent('f', 'keydown'); + keyEvent('3', 'keydown'); + }, + false + ); + coordinatesButton.addEventListener( + 'touchend', + function (e) { + keyEvent('f', 'keyup'); + keyEvent('3', 'keyup'); + }, + false + ); + document.body.appendChild(coordinatesButton); + } + // CSS for touch screen buttons, along with fixing iOS's issues with 100vh ignoring the naviagtion bar, and actually disabling zoom because safari ignores user-scalable=no :( + const customStyle = document.createElement('style'); + customStyle.textContent = ` + html, body, canvas { + height: 100svh !important; + height: -webkit-fill-available !important; + touch-action: pan-x pan-y; + -webkit-touch-callout: none; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + outline: none; + -webkit-tap-highlight-color: rgba(255, 255, 255, 0); + } + .mobileControl { + position: absolute; + width: 9vh; + height: 9vh; + -webkit-user-select: none; + -ms-user-select: none; + user-select: none; + padding:0px; + background-color: transparent; + box-sizing: content-box; + image-rendering: pixelated; + background-size: cover; + outline:none; + box-shadow: none; + border: none; + margin: 1vh; + opacity: 0.5; + } + .mobileControl:active { + opacity: 0.75; + } + .strafeSize { + width: 7.5vh; + height: 7.5vh; + } + .smallMobileControl { + width: 6vh; + height: 6vh; + margin: 1vh 0vh; + } + .hide { + display: none; + } + #fileUpload { + position: absolute; + left: 0; + right: 100vw; + top: 0; + bottom: 100vh; + width: 100vw; + height: 100vh; + background-color:rgba(255,255,255,0.5); + } + .strafeRightButton { + background-image: url(""); + } + .strafeLeftButton { + background-image: url(""); + } + .forwardButton { + background-image: url(""); + } + .rightButton { + background-image: url(""); + } + .leftButton { + background-image: url(""); + } + .backButton { + background-image: url(""); + } + .jumpButton { + background-image: url(""); + } + .crouchButton { + background-image: url(""); + } + .crouchButton:active { + background-image: url(""); + } + .inventoryButton { + background-image: url(""); + } + .chatButton { + background-image: url(""); + } + .pauseButton { + background-image: url(""); + } + .exitButton { + background-image: url(""); + } + .keyboardButton { + background-image: url(""); + } + .placeButton { + background-image: url(""); + } + .breakButton { + background-image: url(""); + } + .selectButton { + background-image: url(""); + } + .scrollUpButton { + background-image: url(""); + } + .scrollDownButton { + background-image: url(""); + } + .throwButton { + background-image: url(""); + } + .sprintButton { + background-image: url(""); + } + .perspectiveButton { + background-image: url(""); + } + .screenshotButton { + background-image: url(""); + } + .coordinatesButton { + background-image: url(""); + } + `; + document.documentElement.appendChild(customStyle); +} diff --git a/src/resources/scripts/eagler-launch/1.8.8/main.ts b/src/resources/scripts/eagler-launch/1.8.8/main.ts new file mode 100644 index 0000000..13e6d29 --- /dev/null +++ b/src/resources/scripts/eagler-launch/1.8.8/main.ts @@ -0,0 +1,26 @@ +// @ts-nocheck +window.addEventListener('load', () => { + const relayId = Math.floor(Math.random() * 3); + window.eaglercraftXOpts = { + container: 'game_frame', + assetsURI: `${window.location.pathname}/assets.epk`, + localesURI: `${window.location.pathname}/lang/`, + servers: [ + { addr: 'wss://mc.ricenetwork.xyz', name: 'Rice Network' }, + { addr: 'wss://mc.lamplifesteal.xyz', name: 'LampLifesteal' }, + { addr: 'wss://electronmc.club', name: 'Electron Network' }, + { addr: 'wss://play.brickmc.net', name: 'BrickMC' }, + ], + relays: [ + { addr: 'wss://relay.deev.is/', comment: 'lax1dude relay #1', primary: relayId === 0 }, + { addr: 'wss://relay.lax1dude.net/', comment: 'lax1dude relay #2', primary: relayId === 1 }, + { addr: 'wss://relay.shhnowisnottheti.me/', comment: 'ayunami relay #1', primary: relayId === 2 }, + ], + }; + + const urlParams = new URLSearchParams(window.location.search); + window.eaglercraftXOpts.joinServer = urlParams.get('server') ?? undefined; + + history.replaceState({}, '', '/play'); + main(); +}); diff --git a/src/resources/scripts/eagler-launch/1.9.4/main.ts b/src/resources/scripts/eagler-launch/1.9.4/main.ts new file mode 100644 index 0000000..3ef1ed2 --- /dev/null +++ b/src/resources/scripts/eagler-launch/1.9.4/main.ts @@ -0,0 +1,21 @@ +// @ts-nocheck +window.addEventListener('load', () => { + const relayId = Math.floor(Math.random() * 3); + window.eaglercraftXOpts = { + container: 'game_frame', + assetsURI: `${window.location.pathname}/assets.epk`, + localesURI: `${window.location.pathname}/lang/`, + servers: [{ addr: 'wss://eagler.xyz', name: 'TemuzX' }], + relays: [ + { addr: 'wss://relay.deev.is/', comment: 'lax1dude relay #1', primary: relayId === 0 }, + { addr: 'wss://relay.lax1dude.net/', comment: 'lax1dude relay #2', primary: relayId === 1 }, + { addr: 'wss://relay.shhnowisnottheti.me/', comment: 'ayunami relay #1', primary: relayId === 2 }, + ], + }; + + const urlParams = new URLSearchParams(window.location.search); + window.eaglercraftXOpts.joinServer = urlParams.get('server') ?? undefined; + + history.replaceState({}, '', '/play'); + main(); +}); diff --git a/src/resources/scripts/eagler-launch/b1.3/main.ts b/src/resources/scripts/eagler-launch/b1.3/main.ts new file mode 100644 index 0000000..7ab2b53 --- /dev/null +++ b/src/resources/scripts/eagler-launch/b1.3/main.ts @@ -0,0 +1,10 @@ +// @ts-nocheck +window.addEventListener('load', function () { + window.minecraftOpts = ['game_frame', `${window.location.pathname}/assets.epk`]; + + const urlParams = new URLSearchParams(window.location.search); + window.minecraftOpts.push(urlParams.get('server') ?? undefined); + + history.replaceState({}, '', '/play'); + main(); +}); diff --git a/src/resources/scripts/google-tag.ts b/src/resources/scripts/google-tag.ts new file mode 100644 index 0000000..1beca01 --- /dev/null +++ b/src/resources/scripts/google-tag.ts @@ -0,0 +1,12 @@ +// @ts-nocheck +const googleTag = document.createElement('script'); +googleTag.async = true; +googleTag.src = 'https://www.googletagmanager.com/gtag/js?id=G-GD4SJRCR7Z'; +document.head.appendChild(googleTag); + +window.dataLayer = window.dataLayer || []; +function gtag(...args: unknown[]) { + dataLayer.push(...args); +} +gtag('js', new Date()); +gtag('config', 'G-GD4SJRCR7Z'); diff --git a/src/resources/scripts/main.ts b/src/resources/scripts/main.ts new file mode 100644 index 0000000..9218342 --- /dev/null +++ b/src/resources/scripts/main.ts @@ -0,0 +1,631 @@ +import { gt, coerce } from 'semver'; +import { inflate, deflate } from 'pako'; + +let selectedVersion: string | undefined = undefined; +let articleAnimationLock = false; + +const theme = { + load: function (themeToLoad?: string) { + const themeElement = document.querySelector('#theme') as HTMLLinkElement | null; + if (themeElement) themeElement.href = themeToLoad ? `/resources/styles/themes/${themeToLoad}.css` : `/resources/styles/themes/${storage.local.get('theme') ?? 'default'}.css`; + }, + set: function (newTheme: string) { + storage.local.set('theme', newTheme); + theme.load(); + }, +}; + +const versionSelector = { + open: function () { + const customOptions = document.querySelector('.custom-options'); + const customSelect = document.querySelector('.custom-select'); + if (customOptions && customSelect) { + customOptions.classList.add('open'); + customSelect.classList.add('open'); + } + }, + close: function () { + const customOptions = document.querySelector('.custom-options'); + const customSelect = document.querySelector('.custom-select'); + if (customOptions && customSelect) { + customOptions.classList.remove('open'); + customSelect.classList.remove('open'); + } + }, + toggle: function () { + const customOptions = document.querySelector('.custom-options'); + const customSelect = document.querySelector('.custom-select'); + if (customOptions && customSelect) { + customOptions.classList.toggle('open'); + customSelect.classList.toggle('open'); + } + }, +}; + +const game = { + play: function (version?: string) { + if (version) { + document.body.style.display = 'none'; + storage.session.set('lastGame', version); + // @ts-expect-error + window.top.location.href = version; + } else if (selectedVersion) { + document.body.style.display = 'none'; + storage.session.set('lastGame', selectedVersion); + // @ts-expect-error + window.top.location.href = selectedVersion; + } else { + alert('Please select a version to play.'); + return; + } + }, + select: function (path: string, name?: string) { + selectedVersion = path; + const selector = document.querySelector('.custom-select'); + if (selector?.textContent) { + if (name) { + selector.textContent = `Selected: ${name}`; + } else { + selector.textContent = `Selected: ${path}`; + } + } + versionSelector.close(); + }, + archive: function (client: string) { + const clients: Record = { + '1.8': '18-client-version', + '1.5': '15-client-version', + 'b1.3': 'b13-client-version', + }; + const dropdown = clients[client] ? (document.querySelector(`#${clients[client]}`) as HTMLSelectElement | null) : null; + if (dropdown?.value) { + selectedVersion = `https://archive.eaglercraft.rip/Eaglercraft${client === '1.8' ? 'X_1.8' : `_${client}`}/client/${dropdown.value}/index.html`; + game.play(); + } + }, +}; + +const navigate = { + home: { + game: function () { + document.body.style.display = 'none'; + const navUrl = '/home/game/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + clients: function () { + document.body.style.display = 'none'; + const navUrl = '/home/clients/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + archive: function () { + document.body.style.display = 'none'; + const navUrl = '/home/archive/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + downloads: function () { + document.body.style.display = 'none'; + const navUrl = '/home/downloads/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + }, + mods: { + mods: function () { + document.body.style.display = 'none'; + const navUrl = '/mods/mods/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + resourcepacks: function () { + document.body.style.display = 'none'; + const navUrl = '/mods/resourcepacks/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + }, + articles: function () { + const navUrl = '/articles/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + mobile: function () { + const navUrl = '/mobile/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + updates: function () { + const navUrl = '/updates/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + servers: function () { + const navUrl = '/servers/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, + settings: function () { + const navUrl = '/settings/'; + storage.session.set('lastPage', navUrl); + window.location.href = navUrl; + }, +}; + +const article = { + open: function (articleId: string) { + const modal = document.querySelector(`#article-${articleId}`) as HTMLElement | null; + const modalContent = document.querySelector(`#article-${articleId} .article-content`) as HTMLElement | null; + if (!articleAnimationLock && modal && modalContent) { + articleAnimationLock = true; + modal.style.animation = '0.5s ease-in-out 1 normal article'; + modal.style.display = 'flex'; + modalContent.scroll({ top: 0, left: 0, behavior: 'instant' }); + modal.addEventListener( + 'animationend', + () => { + const closeArticleHandler = (event: Event) => { + if (event.target === modal) { + modal.removeEventListener('click', closeArticleHandler); + article.close(articleId); + } + }; + modal.addEventListener('click', closeArticleHandler); + articleAnimationLock = false; + }, + { once: true } + ); + } + }, + close: function (articleId: string) { + const modal = document.querySelector(`#article-${articleId}`) as HTMLElement | null; + if (!articleAnimationLock && modal) { + articleAnimationLock = true; + modal.style.animation = '0.5s ease-in-out 1 reverse article-tempfix'; + modal.addEventListener( + 'animationend', + () => { + modal.style.display = 'none'; + articleAnimationLock = false; + }, + { once: true } + ); + } + }, +}; + +/*const cookie = { + set: function (key: string, value: string | number | object | [] | boolean | null | undefined, days: number) { + let maxAge; + if (days) { + maxAge = days * 60 * 60 * 24; + } else { + maxAge = 31536000; + } + document.cookie = `${encodeURIComponent(key)}=${encodeURIComponent(value)}; Max-Age=${maxAge}; Path=/; SameSite=Lax; Secure`; + }, + get: function (key: string) { + for (const cookie of document.cookie.split('; ')) { + const cookiePair = cookie.split('='); + if (encodeURIComponent(key) === cookiePair[0]) { + return decodeURIComponent(cookiePair[1]); + } + } + return undefined; + }, + delete: function (key: string) { + document.cookie = `${encodeURIComponent(key)}=; Max-Age=0; Path=/`; + }, +};*/ + +const storage = { + local: { + get: function (key: string) { + const item = localStorage.getItem('minexlauncher'); + if (item !== null) { + const json = JSON.parse(item); + if (json[key] !== undefined) { + return json[key]; + } + return undefined; + } + return undefined; + }, + set: function (key: string, value: string | number | object | boolean | null | undefined) { + let item = localStorage.getItem('minexlauncher'); + if (item === null) { + item = '{}'; + } + const json = JSON.parse(item); + json[key] = value; + localStorage.setItem('minexlauncher', JSON.stringify(json)); + }, + }, + session: { + get: function (key: string) { + const item = sessionStorage.getItem('minexlauncher'); + if (item !== null) { + const json = JSON.parse(item); + if (json[key] !== undefined) { + return json[key]; + } + return undefined; + } + return undefined; + }, + set: function (key: string, value: string | number | object | boolean | null | undefined) { + let item = sessionStorage.getItem('minexlauncher'); + if (item === null) { + item = '{}'; + } + const json = JSON.parse(item); + json[key] = value; + sessionStorage.setItem('minexlauncher', JSON.stringify(json)); + }, + }, +}; + +const query = { + get: function (name: string) { + return new URLSearchParams(window.top?.location.search).get(name); + }, +}; + +const detect = { + mobile: function (): boolean { + try { + document.exitPointerLock(); + return /Mobi/i.test(window.navigator.userAgent); + } catch { + return true; + } + }, + landscape: function () { + return window.innerWidth > window.innerHeight; + }, +}; + +const mods = { + check: function (mod: string): boolean { + const mods: string[] = storage.local.get('mods') ?? []; + return mods.includes(mod); + }, + add: function (mod: string): void { + const mods: string[] = storage.local.get('mods') ?? []; + if (!mods.includes(mod)) { + mods.push(mod); + mods.sort(); + storage.local.set('mods', mods); + } + }, + remove: function (mod: string): void { + const mods: string[] = storage.local.get('mods') ?? []; + const modIndex = mods.indexOf(mod); + if (modIndex !== -1) { + mods.splice(modIndex, 1); + storage.local.set('mods', mods); + } + }, + toggle: function (modId: string): void { + const mod = `/resources/mods/downloads/${modId}.js`; + const mods: string[] = storage.local.get('mods') ?? []; + const modIndex = mods.indexOf(mod); + if (modIndex === -1) { + mods.push(mod); + mods.sort(); + storage.local.set('mods', mods); + const modInstallElem = document.querySelector(`#mod-install-${modId}`); + if (modInstallElem) { + modInstallElem.textContent = 'Uninstall'; + modInstallElem.classList.add('installed'); + } + } else { + mods.splice(modIndex, 1); + storage.local.set('mods', mods); + const modInstallElem = document.querySelector(`#mod-install-${modId}`); + if (modInstallElem) { + modInstallElem.textContent = 'Install'; + modInstallElem.classList.remove('installed'); + } + } + }, +}; + +const sw = { + register: function (url: string) { + if ('serviceWorker' in navigator) { + window.addEventListener('load', () => { + navigator.serviceWorker.register(url).then(() => { + navigator.serviceWorker.addEventListener('message', (event) => { + if (event.origin === window.location.origin) { + if (event.data.title === 'sw-install-complete') { + alert('MineXLauncher is now ready for offline use!'); + } + } + }); + }); + }); + } + }, +}; + +const base64Gzip = { + decode: function (base64: string) { + // Decode Base64 to binary string + const binaryString = atob(base64); + + // Convert binary string to Uint8Array + const len = binaryString.length; + const bytes = new Uint8Array(len); + for (let i = 0; i < len; i++) { + bytes[i] = binaryString.charCodeAt(i); + } + + // Use pako to decompress the Uint8Array + const decompressed = inflate(bytes, { to: 'string' }); + + return decompressed; + }, + encode: function (inputString: string) { + // Convert the input string to a Uint8Array + const encoder = new TextEncoder(); + const inputBytes = encoder.encode(inputString); + + // Use pako to compress the Uint8Array + const compressedBytes = deflate(inputBytes); + + // Convert the compressed Uint8Array to a binary string + let binaryString = ''; + for (const byte of compressedBytes) { + binaryString += String.fromCharCode(byte); + } + + // Encode the binary string to Base64 + const base64String = btoa(binaryString); + + return base64String; + }, +}; + +if (detect.mobile()) { + const link = document.createElement('link'); + link.rel = 'stylesheet'; + link.href = '/resources/styles/mobile.css'; + document.head.appendChild(link); +} + +if (window.location.pathname === '/') { + const lastPage = storage.session.get('lastPage'); + const isMobile = detect.mobile(); + const iframe = document.createElement('iframe'); + iframe.src = !storage.local.get('lastVersion') ? '/welcome/' : lastPage ? lastPage : isMobile ? '/mobile/' : '/home/game/'; + + document.addEventListener('DOMContentLoaded', () => document.body.appendChild(iframe)); + + /* document.addEventListener('load', () => { + if (storage.local.get('offlineCache')) { + sw.register('/sw-full.js'); + } else { + sw.register('/sw.js'); + } + }); */ + document.addEventListener('load', () => sw.register('/sw.js')); + + window.addEventListener('beforeinstallprompt', (event) => { + // @ts-expect-error + if (iframe.contentWindow) iframe.contentWindow.installPwaEvent = event; + }); +} else { + theme.load(); + + document.addEventListener('DOMContentLoaded', async () => { + const profileName = document.querySelector('.profile-name'); + const titleBarText = document.querySelector('.title-bar span'); + + const lastVersion = storage.local.get('lastVersion'); + const updateData = (await (await fetch('/resources/data/main.json')).json()).updates; + const currentVersion = updateData[0].version; + const changelog = updateData[0].changelog.map((change: string) => ` - ${change}`).join('\n'); + + if (profileName) profileName.textContent = storage.local.get('username'); + if (titleBarText) titleBarText.textContent += ` ${currentVersion}`; + + // @ts-expect-error + if (lastVersion && gt(coerce(currentVersion, { includePrerelease: true }), coerce(lastVersion, { includePrerelease: true }))) { + alert(`MineXLauncher has been updated to v${currentVersion}!\n\nChanges in v${currentVersion}:\n${changelog}`); + storage.local.set('lastVersion', currentVersion); + } + }); + + /* if (storage.local.get('showAds')) { + const googleAdsScript = document.createElement('script'); + googleAdsScript.async = true; + googleAdsScript.src = 'https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js?client=ca-pub-1132419379737567'; + googleAdsScript.crossOrigin = 'anonymous'; + document.head.appendChild(googleAdsScript); + + document.addEventListener('DOMContentLoaded', () => { + const googleAdsPush = document.createElement('script'); + googleAdsPush.text = '(adsbygoogle = window.adsbygoogle || []).push({});'; + document.body.appendChild(googleAdsPush); + + const adsContainers = document.querySelectorAll('.ads-container'); + adsContainers.forEach((adsContainer) => { + adsContainer.style.display = 'flex'; + }); + }); + } */ +} + +if (window.location.pathname === '/settings/') { + document.addEventListener('DOMContentLoaded', async () => { + const profileName = document.querySelector('.profile-name'); + const usernameInput = document.querySelector('#username-input') as HTMLInputElement | null; + const themeSelect = document.querySelector('#theme-select') as HTMLSelectElement | null; + const offlineCheckbox = document.querySelector('#offline-checkbox') as HTMLInputElement | null; + const adsCheckbox = document.querySelector('#ads-checkbox') as HTMLInputElement | null; + const themeData: { id: string; name: string }[] = (await (await fetch('/resources/data/main.json')).json()).themes; + + if (usernameInput) { + usernameInput.placeholder = storage.local.get('username') ?? ''; + usernameInput.addEventListener('input', () => { + let username = usernameInput.value.replace(/[^A-Za-z0-9]/g, '_').substring(0, 16); + usernameInput.value = username; + while (username.length < 3) username += '_'; + + storage.local.set('username', username); + if (profileName) profileName.textContent = username; + }); + } + + if (themeSelect) { + themeData.forEach((theme: { id: string; name: string }) => { + const option = document.createElement('option'); + option.value = theme.id; + option.textContent = theme.name; + themeSelect?.appendChild(option); + }); + themeSelect.value = storage.local.get('theme') ?? ''; + themeSelect.addEventListener('change', () => theme.set(themeSelect.value ?? 'default')); + } + + if (offlineCheckbox) { + offlineCheckbox.checked = storage.local.get('offlineCache') ?? false; + offlineCheckbox.addEventListener('change', () => { + storage.local.set('offlineCache', offlineCheckbox.checked); + if (offlineCheckbox.checked) { + sw.register('/sw-full.js'); + alert( + 'Offline cache is now downloading.\nThe download size is about 1GB, so it may take a while.\n\nPlease do not leave this page while the download is in progress.\nYou will be notified when the download is complete.' + ); + } else { + sw.register('/sw.js'); + alert('Offline cache has been deleted.'); + } + }); + } + + if (adsCheckbox) { + adsCheckbox.checked = storage.local.get('showAds'); + adsCheckbox.addEventListener('change', () => { + storage.local.set('showAds', adsCheckbox.checked); + const adsContainers = document.querySelectorAll('.ads-container'); + adsContainers.forEach((adsContainer: Element) => ((adsContainer as HTMLElement).style.display = 'none')); + }); + } + }); +} else if (window.location.pathname === '/welcome/') { + document.addEventListener('DOMContentLoaded', async () => { + const setupForm = document.querySelector('#setup-form') as HTMLFormElement | null; + const usernameInput = document.querySelector('#username-input') as HTMLInputElement | null; + const themeSelect = document.querySelector('#theme-select') as HTMLSelectElement | null; + const offlineCheckbox = document.querySelector('#offline-checkbox') as HTMLInputElement | null; + const themeData: { id: string; name: string }[] = (await (await fetch('/resources/data/main.json')).json()).themes; + + if (setupForm) { + if (usernameInput) { + usernameInput.addEventListener('input', () => { + const username = usernameInput.value.replace(/[^A-Za-z0-9]/g, '_').substring(0, 16); + usernameInput.value = username; + }); + } + + if (themeSelect) { + themeData.forEach((theme: { id: string; name: string }) => { + const option = document.createElement('option'); + option.value = theme.id; + option.textContent = theme.name; + themeSelect?.appendChild(option); + }); + themeSelect.addEventListener('change', () => theme.load(themeSelect.value ?? 'default')); + } + + setupForm.addEventListener('submit', async (event) => { + event.preventDefault(); + + let username = usernameInput?.value.replace(/[^A-Za-z0-9]/g, '_').substring(0, 16); + if (usernameInput) usernameInput.value = username ?? ''; + + if (!username) { + alert('Please type a username.'); + return; + } else { + while (username.length < 3) username += '_'; + + storage.local.set('username', username); + storage.local.set('theme', themeSelect?.value ?? 'default'); + // storage.local.set('offlineCache', offlineCheckbox?.checked ?? false); + // storage.local.set('showAds', true); + storage.local.set('mods', []); + storage.local.set('lastVersion', (await (await fetch('/resources/data/main.json')).json()).updates[0].version); + + if (offlineCheckbox?.checked) { + sw.register('/sw-full.js'); + alert( + 'Offline cache is now downloading.\nThe download size is about 1GB, so it may take a while.\n\nPlease do not leave this page while the download is in progress.\nYou will be notified when the download is complete.' + ); + } else sw.register('/sw.js'); + + // @ts-expect-error + window.top.location.href = '/'; + } + }); + } + }); +} else if (window.location.pathname === '/mods/mods/' || window.location.pathname === '/mods/resourcepacks/') { + document.addEventListener('DOMContentLoaded', async () => { + const addonType: 'mods' | 'resourcepacks' = window.location.pathname === '/mods/mods/' ? 'mods' : 'resourcepacks'; + const addonData: { id: string; name: string; description: string; author: string; authorLink: string; source: string }[] = (await (await fetch('/resources/data/main.json')).json()).addons; + const modList = document.querySelector('.mod-list'); + // @ts-expect-error + addonData[addonType].forEach((addon) => { + const modItem = document.createElement('div'); + modItem.classList.add('mod-item'); + modItem.innerHTML = `
${ + addon.name + }

By ${addon.author}

${addon.description}

`; + modList?.appendChild(modItem); + }); + + if (addonType === 'mods') { + const installedMods = storage.local.get('mods') ?? []; + const modElements = document.querySelectorAll('.mod-install'); + modElements.forEach((modElement) => { + const modId = /^mod-install-(.*)$/.exec(modElement.id)?.[1]; + if (installedMods.includes(`/resources/mods/downloads/${modId}.js`)) { + modElement.textContent = 'Uninstall'; + modElement.classList.add('installed'); + } + }); + } + }); +} else if (window.location.pathname === '/updates/') { + document.addEventListener('DOMContentLoaded', async () => { + const updatesContainer = document.querySelector('.updates-container'); + const updateData: { version: string; changelog: string[] }[] = (await (await fetch('/resources/data/main.json')).json()).updates; + updateData.forEach((update) => { + const versionHeader = document.createElement('strong'); + versionHeader.textContent = `MineXLauncher ${update.version}`; + updatesContainer?.appendChild(versionHeader); + + const changelog = document.createElement('ul'); + update.changelog.forEach((change) => { + const changelogItem = document.createElement('li'); + changelogItem.textContent = change; + changelog.appendChild(changelogItem); + }); + + updatesContainer?.appendChild(changelog); + }); + }); +} + +if (window.location.hostname === null) { + // Stop the minifier from removing these functions + console.debug([navigate, query, versionSelector, game, mods, base64Gzip, article]); +} diff --git a/tsconfig.json b/tsconfig.json index 08a6d4a..2895e1d 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,9 +1,4 @@ { "extends": ["@tsconfig/bun", "@tsconfig/strictest"], - "exclude": ["node_modules", "public"], - "compilerOptions": { - //"outDir": "dist", - //"noEmit": false, - //"allowImportingTsExtensions": false - } + "exclude": ["node_modules", "public"] }