From 1c1916db06f281210739db004e2f2a9fb484b2f4 Mon Sep 17 00:00:00 2001 From: zumbiepig <121742281+zumbiepig@users.noreply.github.com> Date: Fri, 6 Sep 2024 14:49:58 -0700 Subject: [PATCH] new build --- build.ts | 32 +- bun.lockb | Bin 107482 -> 113007 bytes package.json | 2 + {src => public}/sw-full.ts | 0 {src => public}/sw.ts | 0 .../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 --------- 14 files changed, 27 insertions(+), 2623 deletions(-) rename {src => public}/sw-full.ts (100%) rename {src => public}/sw.ts (100%) delete mode 100644 src/resources/scripts/eagler-launch/1.5.2/main.ts delete mode 100644 src/resources/scripts/eagler-launch/1.8.8/eaglerforge.ts delete mode 100644 src/resources/scripts/eagler-launch/1.8.8/eaglermobile-ef.ts delete mode 100644 src/resources/scripts/eagler-launch/1.8.8/eaglerpocketmobile.ts delete mode 100644 src/resources/scripts/eagler-launch/1.8.8/main.ts delete mode 100644 src/resources/scripts/eagler-launch/1.9.4/main.ts delete mode 100644 src/resources/scripts/eagler-launch/b1.3/main.ts delete mode 100644 src/resources/scripts/google-tag.ts delete mode 100644 src/resources/scripts/main.ts diff --git a/build.ts b/build.ts index 71d307d..a728f77 100644 --- a/build.ts +++ b/build.ts @@ -1,14 +1,15 @@ import { $, build } from 'bun'; -import { readdirSync, statSync, writeFileSync } from 'fs'; -import { join } from 'path'; +import { readdirSync, statSync, writeFileSync, readFileSync } from 'fs'; +import { resolve } from 'path'; import chalk from 'chalk'; +import { minify } from 'html-minifier'; 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); + const name = resolve(dir, file); if (statSync(name).isDirectory()) { getFiles(baseDir, name, filesArr); } else { @@ -20,6 +21,10 @@ function getFiles(baseDir: string, dir?: string, filesArr?: string[]) { console.clear(); +const srcDir = resolve(import.meta.dir, 'src'); +const publicDir = resolve(import.meta.dir, 'public'); +const srcFilesArr = getFiles(resolve(import.meta.dir, 'src')); + console.log(chalk.cyan('Linting code...\n')); const lintOutput = await $`bunx eslint ./src/`.nothrow().text(); if (lintOutput) { @@ -38,9 +43,8 @@ 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, + entrypoints: srcFilesArr.filter((file) => file.endsWith('.ts')), outdir: './public/', target: 'browser', root: './src/', @@ -51,13 +55,27 @@ await build({ }, }); +console.log(chalk.cyan('Minifying HTML and CSS...\n')); +srcFilesArr + .filter((file) => !file.endsWith('.ts')) + .forEach((file) => { + writeFileSync( + file.replace(new RegExp(`^${srcDir}`), publicDir), + minify(readFileSync(file, 'utf8'), { + collapseWhitespace: true, + removeComments: true, + minifyCSS: true, + minifyJS: 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('Generating assets list...\n')); -const publicDir = join(import.meta.dir, 'public'); writeFileSync( - join(publicDir, 'resources/data/assets.json'), + resolve(publicDir, 'resources/data/assets.json'), JSON.stringify( getFiles(publicDir).map((asset) => { return asset.replace(new RegExp(`^${publicDir}`), '').replace(/\/index\.html$/, '/'); diff --git a/bun.lockb b/bun.lockb index 6b107c30593ed4dcb4c33620fc0958e55a38af2a..f031ae4b51d088ce9cc9042022023a75728f17ff 100755 GIT binary patch delta 18531 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�-oYb$saX>vg9HXA2J{c}@beh!(z0xjQVq(AgJX7>E!^7-#-(0;j0*ULq zy*{ha|Ky~^v=3b(4U2~pl%b}NfZ?rb4fGKIM+h&fLAb;hcHf2zkVZdtqqQhht4E4>^t)z) z)gfC&Id{8SF{KT3| zgP&!p778iO%h_S>g6Q1n8((z^Y?W5vo!7V1ld~sAEPH2*o^I?3>(%S*V!H}V6k+PS zQNknd&gO6z2DsLjI(mEnO=xi$HrjYI$u7#x_W7=4z9zCxU1Nk=yR$~j_4P4D$YK2 zOwQY%8)&`#Ww?+E4a~sjaq5~JF~)T&UOkp0!o_dY%Q+$^()nazZugj(o9~!8)+Mk@ zSqaclwJF4M8kQ?!qnu9_?z&^}-a|X5gu`qKwljPBZpalq%%m>XbxYk|^+YZLV>Na_ zUChN&zf*a}qC@9vh2|&oqy6iDe*vc0BRq^(%f_NS+hK`*7a4Q+LR~yM*e)71gTH*uwBRZws((5;-}nq;igNS(J{AcxSgc@#(7#S@6z}Can-7t@tB&Q?*dEU&*Mc8u~?lMFD6Ag-yJ0TH+5NhW19%ek`?TQ zYC87>y(Rr{hj6}X_~^04Yx3Wl_odaUKl9kuQ(sLG5m8r2F@1^Rq}Wd*H-j3OYHPuJ zunvDS5rM8!j{_p@k0S65*EOdvUAQgq1-k8v%)o>ss_!NuF3!ghK~wrpe7))B7omXb z2P(X%k|)8?<-;InZMXJ5i)%1!ZXB?+7g~E;JJox#aR0X(Vkc)q490~W7De67!6Ujf z+rG0qGr>L)C#ZFkaZ!+^V)KP3)%mL7udN1Dv~8cAg;NmM))n_IFE%C&fO}U(K9=QA z_b$D483DBc#>H&)Og^&G`5t5SjLa?R-9Fsz5)1{pFjlBeQxHw(Lyq;|%sx9|MB#1r z)l@Pj@i8@T3JmR0w@eW^DbAN6^TOXevHH#06{c$QTAa791reUzqcRsOecK4dsI^B5UKCO?zwosR*;uPq7fbz_uQ+aQX_|C5t z5T~DUBDud|nfC~*$FLyua=E}0>3pj)YVV-1-0IwX%cON8$WsTVq0gVWmO`J-=PjMu zJ-6=Oza2bnHMO$M?u_$yMf29FGC}1{N0@8Xz0)x{&Syd&oc@o9J#pV13=%i=NlInr zDpff{c+~0XxUO?PyW7+1qlZUsEB-`Z{3AftS*u^9^WEN}U7vg1=~*~4Temf-jv1T< zadzXWkw#rRN2)hw;)cmRO0~X5gr_>6zl_~Du-$(~z17Yo_}qnu{iO^(L+UwydCSM+ zuKvEJ-zYT$Z5=JrD)HIl6j({_^tH#aVyv3O*G4{nt*t wW!A_cQGF)ZW`=H5)y24kPAw6ApI%TR-Wc(8>&L|#L8@X0ZhAZI5MzS=3tCIW', '|>', '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 deleted file mode 100644 index 6ceeb4e..0000000 --- a/src/resources/scripts/eagler-launch/1.8.8/eaglerforge.ts +++ /dev/null @@ -1,45 +0,0 @@ -// @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 deleted file mode 100644 index deb8218..0000000 --- a/src/resources/scripts/eagler-launch/1.8.8/eaglermobile-ef.ts +++ /dev/null @@ -1,688 +0,0 @@ -/* 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 deleted file mode 100644 index 7eed39a..0000000 --- a/src/resources/scripts/eagler-launch/1.8.8/eaglerpocketmobile.ts +++ /dev/null @@ -1,1156 +0,0 @@ -/* 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 deleted file mode 100644 index 13e6d29..0000000 --- a/src/resources/scripts/eagler-launch/1.8.8/main.ts +++ /dev/null @@ -1,26 +0,0 @@ -// @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 deleted file mode 100644 index 3ef1ed2..0000000 --- a/src/resources/scripts/eagler-launch/1.9.4/main.ts +++ /dev/null @@ -1,21 +0,0 @@ -// @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 deleted file mode 100644 index 7ab2b53..0000000 --- a/src/resources/scripts/eagler-launch/b1.3/main.ts +++ /dev/null @@ -1,10 +0,0 @@ -// @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 deleted file mode 100644 index 1beca01..0000000 --- a/src/resources/scripts/google-tag.ts +++ /dev/null @@ -1,12 +0,0 @@ -// @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 deleted file mode 100644 index 9218342..0000000 --- a/src/resources/scripts/main.ts +++ /dev/null @@ -1,631 +0,0 @@ -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]); -}