2023-10-27 06:26:54 +02:00
|
|
|
'use-strict'; // Force strict mode for transpiled
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
/*
|
|
|
|
WebUI Bridge
|
2023-10-08 03:01:02 +02:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
http://webui.me
|
|
|
|
https://github.com/webui-dev/webui
|
|
|
|
Copyright (c) 2020-2023 Hassan Draga.
|
|
|
|
Licensed under MIT License.
|
|
|
|
All rights reserved.
|
|
|
|
Canada.
|
|
|
|
|
|
|
|
Converted from JavaScript to TypeScript
|
|
|
|
By Oculi Julien. Copyright (c) 2023.
|
|
|
|
*/
|
|
|
|
|
|
|
|
//@ts-ignore use *.ts import real extension
|
2023-10-27 06:26:54 +02:00
|
|
|
import { AsyncFunction, addRefreshableEventListener } from './utils.ts';
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-10-27 06:26:54 +02:00
|
|
|
type DataTypes = string | number | boolean | Uint8Array;
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-08-21 09:23:29 -04:00
|
|
|
class WebuiBridge {
|
2023-08-18 15:50:31 -04:00
|
|
|
// WebUI settings
|
2023-10-27 06:26:54 +02:00
|
|
|
#secure: boolean;
|
|
|
|
#token: number;
|
|
|
|
#port: number;
|
|
|
|
#winNum: number;
|
2023-10-27 17:44:17 -04:00
|
|
|
#bindList: string[] = [];
|
2023-10-27 06:26:54 +02:00
|
|
|
#log: boolean;
|
|
|
|
#winX: number;
|
|
|
|
#winY: number;
|
|
|
|
#winW: number;
|
|
|
|
#winH: number;
|
2023-08-18 15:50:31 -04:00
|
|
|
// Internals
|
2023-10-27 06:26:54 +02:00
|
|
|
#ws: WebSocket;
|
|
|
|
#wsStatus = false;
|
|
|
|
#wsStatusOnce = false;
|
|
|
|
#closeReason = 0;
|
|
|
|
#closeValue: string;
|
|
|
|
#hasEvents = false;
|
|
|
|
#callPromiseID = new Uint16Array(1);
|
|
|
|
#callPromiseResolve: (((data: string) => unknown) | undefined)[] = [];
|
|
|
|
#allowNavigation = false;
|
|
|
|
#sendQueue: Uint8Array[] = [];
|
|
|
|
#isSending = false;
|
2023-08-18 15:50:31 -04:00
|
|
|
// WebUI const
|
2023-10-27 06:26:54 +02:00
|
|
|
#WEBUI_SIGNATURE = 221;
|
|
|
|
#CMD_JS = 254;
|
|
|
|
#CMD_JS_QUICK = 253;
|
|
|
|
#CMD_CLICK = 252;
|
|
|
|
#CMD_NAVIGATION = 251;
|
|
|
|
#CMD_CLOSE = 250;
|
|
|
|
#CMD_CALL_FUNC = 249;
|
|
|
|
#CMD_SEND_RAW = 248;
|
|
|
|
#CMD_NEW_ID = 247;
|
|
|
|
#CMD_MULTI = 246;
|
|
|
|
#MULTI_CHUNK_SIZE = 65500;
|
|
|
|
#PROTOCOL_SIZE = 8; // Protocol header size in bytes
|
|
|
|
#PROTOCOL_SIGN = 0; // Protocol byte position: Signature (1 Byte)
|
|
|
|
#PROTOCOL_TOKEN = 1; // Protocol byte position: Token (4 Bytes)
|
|
|
|
#PROTOCOL_ID = 5; // Protocol byte position: ID (2 Bytes)
|
|
|
|
#PROTOCOL_CMD = 7; // Protocol byte position: Command (1 Byte)
|
|
|
|
#PROTOCOL_DATA = 8; // Protocol byte position: Data (n Byte)
|
|
|
|
#Token = new Uint32Array(1);
|
|
|
|
#Ping: Boolean = true;
|
2023-08-18 15:50:31 -04:00
|
|
|
constructor({
|
2023-10-21 22:55:30 -04:00
|
|
|
secure,
|
2023-10-04 19:22:40 -04:00
|
|
|
token,
|
2023-08-18 15:50:31 -04:00
|
|
|
port,
|
|
|
|
winNum,
|
|
|
|
bindList,
|
|
|
|
log = false,
|
2023-08-29 21:12:45 -04:00
|
|
|
winX,
|
|
|
|
winY,
|
|
|
|
winW,
|
|
|
|
winH,
|
2023-08-18 15:50:31 -04:00
|
|
|
}: {
|
2023-10-27 06:26:54 +02:00
|
|
|
secure: boolean;
|
|
|
|
token: number;
|
|
|
|
port: number;
|
|
|
|
winNum: number;
|
2023-10-27 17:44:17 -04:00
|
|
|
bindList: string[];
|
2023-10-27 06:26:54 +02:00
|
|
|
log?: boolean;
|
|
|
|
winX: number;
|
|
|
|
winY: number;
|
|
|
|
winW: number;
|
|
|
|
winH: number;
|
2023-08-18 15:50:31 -04:00
|
|
|
}) {
|
|
|
|
// Constructor arguments are injected by webui.c
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#secure = secure;
|
|
|
|
this.#token = token;
|
|
|
|
this.#port = port;
|
|
|
|
this.#winNum = winNum;
|
|
|
|
this.#bindList = bindList;
|
|
|
|
this.#log = log;
|
|
|
|
this.#winX = winX;
|
|
|
|
this.#winY = winY;
|
|
|
|
this.#winW = winW;
|
|
|
|
this.#winH = winH;
|
2023-10-04 19:22:40 -04:00
|
|
|
// Token
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#Token[0] = this.#token;
|
2023-08-29 21:12:45 -04:00
|
|
|
// Instance
|
2023-08-18 15:50:31 -04:00
|
|
|
if ('webui' in globalThis) {
|
2023-10-27 06:26:54 +02:00
|
|
|
throw new Error('Sorry. WebUI is already defined, only one instance is allowed.');
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-08-29 21:12:45 -04:00
|
|
|
// Positioning the current window
|
|
|
|
if (this.#winX !== undefined && this.#winY !== undefined) {
|
2023-10-27 06:26:54 +02:00
|
|
|
window.moveTo(this.#winX, this.#winY);
|
2023-08-29 21:12:45 -04:00
|
|
|
}
|
|
|
|
// Resize the current window
|
|
|
|
if (this.#winW !== undefined && this.#winH !== undefined) {
|
2023-10-27 06:26:54 +02:00
|
|
|
window.resizeTo(this.#winW, this.#winH);
|
2023-08-29 21:12:45 -04:00
|
|
|
}
|
|
|
|
// WebSocket
|
2023-08-18 15:50:31 -04:00
|
|
|
if (!('WebSocket' in window)) {
|
2023-10-27 06:26:54 +02:00
|
|
|
alert('Sorry. WebSocket is not supported by your web browser.');
|
|
|
|
if (!this.#log) globalThis.close();
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-08-29 21:12:45 -04:00
|
|
|
// Connect to the backend application
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#start();
|
2023-08-18 15:50:31 -04:00
|
|
|
// Handle navigation server side
|
|
|
|
if ('navigation' in globalThis) {
|
|
|
|
globalThis.navigation.addEventListener('navigate', (event) => {
|
2023-10-08 03:01:02 +02:00
|
|
|
if (!this.#allowNavigation) {
|
2023-10-27 06:26:54 +02:00
|
|
|
event.preventDefault();
|
|
|
|
const url = new URL(event.destination.url);
|
2023-09-26 18:59:25 -04:00
|
|
|
if (this.#hasEvents) {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) console.log(`WebUI -> DOM -> Navigation Event [${url.href}]`);
|
|
|
|
this.#sendEventNavigation(url.href);
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#close(this.#CMD_NAVIGATION, url.href);
|
2023-09-26 18:59:25 -04:00
|
|
|
}
|
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
});
|
2023-08-18 15:50:31 -04:00
|
|
|
} else {
|
|
|
|
// Handle all link click to prevent natural navigation
|
|
|
|
// Rebind listener if user inject new html
|
2023-10-08 03:01:02 +02:00
|
|
|
addRefreshableEventListener(document.body, 'a', 'click', (event) => {
|
|
|
|
if (!this.#allowNavigation) {
|
2023-10-27 06:26:54 +02:00
|
|
|
event.preventDefault();
|
|
|
|
const { href } = event.target as HTMLAnchorElement;
|
2023-10-08 03:01:02 +02:00
|
|
|
if (this.#hasEvents) {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) console.log(`WebUI -> DOM -> Navigation Click Event [${href}]`);
|
|
|
|
this.#sendEventNavigation(href);
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#close(this.#CMD_NAVIGATION, href);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
});
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
// Prevent F5 refresh
|
|
|
|
document.addEventListener('keydown', (event) => {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) return; // Allowed in debug mode
|
|
|
|
if (event.key === 'F5') event.preventDefault();
|
|
|
|
});
|
2023-08-18 15:50:31 -04:00
|
|
|
onbeforeunload = () => {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#close();
|
|
|
|
};
|
2023-08-18 15:50:31 -04:00
|
|
|
setTimeout(() => {
|
|
|
|
if (!this.#wsStatusOnce) {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#freezeUi();
|
|
|
|
alert('Sorry. WebUI failed to connect to the backend application. Please try again.');
|
|
|
|
if (!this.#log) globalThis.close();
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
}, 1500);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
#close(reason = 0, value = '') {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#wsStatus = false;
|
|
|
|
this.#closeReason = reason;
|
|
|
|
this.#closeValue = value;
|
|
|
|
this.#ws.close();
|
2023-10-04 19:22:40 -04:00
|
|
|
if (reason === this.#CMD_NAVIGATION) {
|
2023-10-01 20:41:52 -04:00
|
|
|
if (this.#log) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> Close -> Navigation to [${value}]`);
|
2023-10-01 20:41:52 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#allowNavigation = true;
|
|
|
|
globalThis.location.replace(this.#closeValue);
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
2023-10-01 20:41:52 -04:00
|
|
|
if (this.#log) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> Close.`);
|
2023-10-01 20:41:52 -04:00
|
|
|
}
|
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
#freezeUi() {
|
2023-10-27 06:26:54 +02:00
|
|
|
document.body.style.filter = 'contrast(1%)';
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-08-23 22:53:06 -04:00
|
|
|
#isTextBasedCommand(cmd: number): Boolean {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (cmd !== this.#CMD_SEND_RAW) return true;
|
|
|
|
return false;
|
2023-08-23 22:53:06 -04:00
|
|
|
}
|
2023-08-25 00:33:11 -04:00
|
|
|
#getDataStrFromPacket(buffer: Uint8Array, startIndex: number): string {
|
2023-10-27 06:26:54 +02:00
|
|
|
let stringBytes: number[] = [];
|
2023-08-18 15:50:31 -04:00
|
|
|
for (let i = startIndex; i < buffer.length; i++) {
|
2023-10-08 03:01:02 +02:00
|
|
|
if (buffer[i] === 0) {
|
|
|
|
// Check for null byte
|
2023-10-27 06:26:54 +02:00
|
|
|
break;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
stringBytes.push(buffer[i]);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
// Convert the array of bytes to a string
|
2023-10-27 06:26:54 +02:00
|
|
|
const stringText = new TextDecoder().decode(new Uint8Array(stringBytes));
|
|
|
|
return stringText;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-04 19:22:40 -04:00
|
|
|
#getID(buffer: Uint8Array, index: number): number {
|
|
|
|
if (index < 0 || index >= buffer.length - 1) {
|
2023-10-27 06:26:54 +02:00
|
|
|
throw new Error('Index out of bounds or insufficient data.');
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
const firstByte = buffer[index];
|
|
|
|
const secondByte = buffer[index + 1];
|
|
|
|
const combined = (secondByte << 8) | firstByte; // Works only for little-endian
|
|
|
|
return combined;
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
|
|
|
#addToken(buffer: Uint8Array, value: number, index: number): void {
|
2023-10-08 03:01:02 +02:00
|
|
|
if (value < 0 || value > 0xffffffff) {
|
2023-10-27 06:26:54 +02:00
|
|
|
throw new Error('Number is out of the range for 4 bytes representation.');
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
|
|
|
if (index < 0 || index > buffer.length - 4) {
|
2023-10-27 06:26:54 +02:00
|
|
|
throw new Error('Index out of bounds or insufficient space in buffer.');
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
|
|
|
// WebUI expect Little-endian (Work for Little/Big endian platforms)
|
2023-10-27 06:26:54 +02:00
|
|
|
buffer[index] = value & 0xff; // Least significant byte
|
|
|
|
buffer[index + 1] = (value >>> 8) & 0xff;
|
|
|
|
buffer[index + 2] = (value >>> 16) & 0xff;
|
|
|
|
buffer[index + 3] = (value >>> 24) & 0xff; // Most significant byte
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
|
|
|
#addID(buffer: Uint8Array, value: number, index: number): void {
|
2023-10-08 03:01:02 +02:00
|
|
|
if (value < 0 || value > 0xffff) {
|
2023-10-27 06:26:54 +02:00
|
|
|
throw new Error('Number is out of the range for 2 bytes representation.');
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
|
|
|
if (index < 0 || index > buffer.length - 2) {
|
2023-10-27 06:26:54 +02:00
|
|
|
throw new Error('Index out of bounds or insufficient space in buffer.');
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
|
|
|
// WebUI expect Little-endian (Work for Little/Big endian platforms)
|
2023-10-27 06:26:54 +02:00
|
|
|
buffer[index] = value & 0xff; // Least significant byte
|
|
|
|
buffer[index + 1] = (value >>> 8) & 0xff; // Most significant byte
|
2023-10-04 19:22:40 -04:00
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
#start() {
|
2023-10-27 17:44:17 -04:00
|
|
|
this.#generateCallObjects();
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#keepAlive();
|
|
|
|
this.#callPromiseID[0] = 0;
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#bindList.includes(this.#winNum + '/')) {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#hasEvents = true;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
const url = this.#secure ? 'wss://localhost' : 'ws://localhost';
|
|
|
|
this.#ws = new WebSocket(`${url}:${this.#port}/_webui_ws_connect`);
|
|
|
|
this.#ws.binaryType = 'arraybuffer';
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#ws.onopen = () => {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#wsStatus = true;
|
|
|
|
this.#wsStatusOnce = true;
|
|
|
|
if (this.#log) console.log('WebUI -> Connected');
|
|
|
|
this.#clicksListener();
|
|
|
|
};
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#ws.onerror = () => {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) console.log('WebUI -> Connection Failed');
|
|
|
|
this.#freezeUi();
|
|
|
|
};
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#ws.onclose = (event) => {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#wsStatus = false;
|
2023-10-04 19:22:40 -04:00
|
|
|
if (this.#closeReason === this.#CMD_NAVIGATION) {
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#log) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> Connection closed du to Navigation to [${this.#closeValue}]`);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (this.#log) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> Connection lost (${event.code})`);
|
|
|
|
this.#freezeUi();
|
2023-08-18 15:50:31 -04:00
|
|
|
} else {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#closeWindowTimer();
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
};
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#ws.onmessage = async (event) => {
|
2023-10-27 06:26:54 +02:00
|
|
|
const buffer8 = new Uint8Array(event.data);
|
|
|
|
if (buffer8.length < this.#PROTOCOL_SIZE) return;
|
|
|
|
if (buffer8[this.#PROTOCOL_SIGN] !== this.#WEBUI_SIGNATURE) return;
|
2023-10-08 03:01:02 +02:00
|
|
|
if (this.#isTextBasedCommand(buffer8[this.#PROTOCOL_CMD])) {
|
2023-08-18 15:50:31 -04:00
|
|
|
// UTF8 Text based commands
|
2023-10-27 06:26:54 +02:00
|
|
|
const callId = this.#getID(buffer8, this.#PROTOCOL_ID);
|
2023-08-18 15:50:31 -04:00
|
|
|
// Process Command
|
2023-10-04 19:22:40 -04:00
|
|
|
switch (buffer8[this.#PROTOCOL_CMD]) {
|
|
|
|
case this.#CMD_JS_QUICK:
|
|
|
|
case this.#CMD_JS:
|
2023-08-18 15:50:31 -04:00
|
|
|
{
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
2023-08-25 00:33:11 -04:00
|
|
|
// 2: [ID]
|
2023-10-04 19:22:40 -04:00
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Script]
|
2023-10-27 06:26:54 +02:00
|
|
|
const script = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA);
|
|
|
|
const scriptSanitize = script.replace(/(?:\r\n|\r|\n)/g, '\n');
|
|
|
|
if (this.#log) console.log(`WebUI -> CMD -> JS [${scriptSanitize}]`);
|
2023-08-18 15:50:31 -04:00
|
|
|
// Get callback result
|
2023-10-27 06:26:54 +02:00
|
|
|
let FunReturn = 'undefined';
|
|
|
|
let FunError = false;
|
2023-08-18 15:50:31 -04:00
|
|
|
try {
|
2023-10-27 06:26:54 +02:00
|
|
|
FunReturn = await AsyncFunction(scriptSanitize)();
|
2023-08-18 15:50:31 -04:00
|
|
|
} catch (e) {
|
2023-10-27 06:26:54 +02:00
|
|
|
FunError = true;
|
|
|
|
FunReturn = e.message;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-04 19:22:40 -04:00
|
|
|
// Stop if this is a quick call
|
2023-10-27 06:26:54 +02:00
|
|
|
if (buffer8[this.#PROTOCOL_CMD] === this.#CMD_JS_QUICK) return;
|
2023-10-04 19:22:40 -04:00
|
|
|
// Get the call return
|
2023-08-18 15:50:31 -04:00
|
|
|
if (FunReturn === undefined) {
|
2023-10-27 06:26:54 +02:00
|
|
|
FunReturn = 'undefined';
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
// Logging
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log && !FunError) console.log(`WebUI -> CMD -> JS -> Return Success [${FunReturn}]`);
|
|
|
|
if (this.#log && FunError) console.log(`WebUI -> CMD -> JS -> Return Error [${FunReturn}]`);
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
2023-08-25 00:33:11 -04:00
|
|
|
// 2: [ID]
|
2023-10-04 19:22:40 -04:00
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Error, Script Response]
|
2023-08-18 15:50:31 -04:00
|
|
|
const Return8 = Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#WEBUI_SIGNATURE,
|
2023-10-27 06:26:54 +02:00
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0, // Token (4 Bytes)
|
|
|
|
0,
|
|
|
|
0, // ID (2 Bytes)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#CMD_JS,
|
2023-10-03 14:35:44 -04:00
|
|
|
FunError ? 1 : 0,
|
2023-10-08 03:01:02 +02:00
|
|
|
...new TextEncoder().encode(FunReturn),
|
2023-10-27 06:26:54 +02:00
|
|
|
);
|
|
|
|
this.#addToken(Return8, this.#token, this.#PROTOCOL_TOKEN);
|
|
|
|
this.#addID(Return8, callId, this.#PROTOCOL_ID);
|
|
|
|
this.#sendData(Return8);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
break;
|
2023-10-05 22:08:50 -04:00
|
|
|
case this.#CMD_CALL_FUNC:
|
|
|
|
{
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Call Response]
|
2023-10-27 06:26:54 +02:00
|
|
|
const callResponse = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA);
|
2023-10-05 22:08:50 -04:00
|
|
|
if (this.#log) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> CMD -> Call Response [${callResponse}]`);
|
2023-10-05 22:08:50 -04:00
|
|
|
}
|
|
|
|
if (this.#callPromiseResolve[callId]) {
|
|
|
|
if (this.#log) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> CMD -> Resolving Response #${callId}...`);
|
2023-10-05 22:08:50 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#callPromiseResolve[callId]?.(callResponse);
|
|
|
|
this.#callPromiseResolve[callId] = undefined;
|
2023-10-05 22:08:50 -04:00
|
|
|
}
|
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
break;
|
2023-10-05 22:08:50 -04:00
|
|
|
|
|
|
|
case this.#CMD_NAVIGATION:
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [URL]
|
2023-10-27 06:26:54 +02:00
|
|
|
const url = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA);
|
|
|
|
console.log(`WebUI -> CMD -> Navigation [${url}]`);
|
|
|
|
this.#close(this.#CMD_NAVIGATION, url);
|
|
|
|
break;
|
2023-10-05 22:08:50 -04:00
|
|
|
case this.#CMD_NEW_ID:
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [New Element]
|
2023-10-27 06:26:54 +02:00
|
|
|
const newElement = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA);
|
|
|
|
console.log(`WebUI -> CMD -> New Bind ID [${newElement}]`);
|
|
|
|
if (!this.#bindList.includes(newElement)) this.#bindList.push(newElement);
|
|
|
|
break;
|
2023-10-04 19:22:40 -04:00
|
|
|
case this.#CMD_CLOSE:
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
2023-10-27 06:26:54 +02:00
|
|
|
if (!this.#log) globalThis.close();
|
2023-09-22 22:50:18 -04:00
|
|
|
else {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log(`WebUI -> CMD -> Close`);
|
|
|
|
this.#ws.close();
|
2023-09-22 22:50:18 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
break;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
2023-10-05 22:08:50 -04:00
|
|
|
// Raw-binary based commands
|
2023-10-04 19:22:40 -04:00
|
|
|
switch (buffer8[this.#PROTOCOL_CMD]) {
|
|
|
|
case this.#CMD_SEND_RAW:
|
2023-10-05 22:08:50 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Function,Null,Raw Data]
|
|
|
|
// Get function name
|
2023-10-27 06:26:54 +02:00
|
|
|
const functionName: string = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA);
|
2023-10-05 22:08:50 -04:00
|
|
|
// Get the raw data
|
2023-10-27 06:26:54 +02:00
|
|
|
const rawDataIndex: number = 2 + functionName.length + 1;
|
|
|
|
const userRawData = buffer8.subarray(rawDataIndex);
|
|
|
|
if (this.#log) console.log(`WebUI -> CMD -> Send Raw ${buffer8.length} bytes to [${functionName}()]`);
|
2023-10-05 22:08:50 -04:00
|
|
|
// Call the user function, and pass the raw data
|
2023-10-27 06:26:54 +02:00
|
|
|
if (typeof window[functionName] === 'function') window[functionName](userRawData);
|
|
|
|
else await AsyncFunction(functionName + '(userRawData)')();
|
|
|
|
break;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
};
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-08 03:01:02 +02:00
|
|
|
#keepAlive = async () => {
|
2023-10-05 23:44:41 -04:00
|
|
|
while (true) {
|
|
|
|
if (this.#Ping) {
|
|
|
|
// Some web browsers may close the connection
|
|
|
|
// let's send a void message to keep WS open
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#sendData(new TextEncoder().encode('ping'));
|
2023-10-05 23:44:41 -04:00
|
|
|
} else {
|
|
|
|
// There is an active communication
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#Ping = true;
|
2023-10-05 23:44:41 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
await new Promise((resolve) => setTimeout(resolve, 20000));
|
2023-10-05 23:44:41 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
};
|
2023-08-18 15:50:31 -04:00
|
|
|
#clicksListener() {
|
|
|
|
Object.keys(window).forEach((key) => {
|
|
|
|
if (/^on(click)/.test(key)) {
|
|
|
|
globalThis.addEventListener(key.slice(2), (event) => {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (!(event.target instanceof HTMLElement)) return;
|
2023-08-18 15:50:31 -04:00
|
|
|
if (
|
|
|
|
this.#hasEvents ||
|
2023-10-08 03:01:02 +02:00
|
|
|
(event.target.id !== '' && this.#bindList.includes(this.#winNum + '/' + event.target?.id))
|
2023-08-18 15:50:31 -04:00
|
|
|
) {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#sendClick(event.target.id);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
});
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
});
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-24 20:58:59 -04:00
|
|
|
async #sendData(packet: Uint8Array) {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#Ping = false;
|
|
|
|
if (!this.#wsStatus || packet === undefined) return;
|
|
|
|
// Enqueue the packet
|
|
|
|
this.#sendQueue.push(packet);
|
|
|
|
if (this.#isSending) return;
|
|
|
|
this.#isSending = true;
|
|
|
|
while (this.#sendQueue.length > 0) {
|
|
|
|
const currentPacket = this.#sendQueue.shift()!;
|
|
|
|
if (currentPacket.length < this.#MULTI_CHUNK_SIZE) {
|
|
|
|
this.#ws.send(currentPacket.buffer);
|
|
|
|
} else {
|
|
|
|
// Pre-packet to let WebUI be ready for multi packet
|
|
|
|
const pre_packet = Uint8Array.of(
|
|
|
|
this.#WEBUI_SIGNATURE,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0, // Token (4 Bytes)
|
|
|
|
0,
|
|
|
|
0, // ID (2 Bytes)
|
|
|
|
this.#CMD_MULTI,
|
|
|
|
...new TextEncoder().encode(currentPacket.length.toString()),
|
|
|
|
0,
|
|
|
|
);
|
|
|
|
this.#ws.send(pre_packet.buffer);
|
|
|
|
// Send chunks
|
|
|
|
let offset = 0;
|
|
|
|
const sendChunk = async () => {
|
|
|
|
if (offset < currentPacket.length) {
|
|
|
|
const chunkSize = Math.min(this.#MULTI_CHUNK_SIZE, currentPacket.length - offset);
|
|
|
|
const chunk = currentPacket.subarray(offset, offset + chunkSize);
|
|
|
|
this.#ws.send(chunk);
|
|
|
|
offset += chunkSize;
|
|
|
|
await sendChunk();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
await sendChunk();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.#isSending = false;
|
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
#sendClick(elem: string) {
|
|
|
|
if (this.#wsStatus) {
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Element]
|
2023-08-18 15:50:31 -04:00
|
|
|
const packet =
|
|
|
|
elem !== ''
|
|
|
|
? Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#WEBUI_SIGNATURE,
|
2023-10-08 03:01:02 +02:00
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0, // Token (4 Bytes)
|
|
|
|
0,
|
|
|
|
0, // ID (2 Bytes)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#CMD_CLICK,
|
|
|
|
...new TextEncoder().encode(elem),
|
2023-10-08 03:01:02 +02:00
|
|
|
0,
|
2023-08-18 15:50:31 -04:00
|
|
|
)
|
|
|
|
: Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#WEBUI_SIGNATURE,
|
2023-10-08 03:01:02 +02:00
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0, // Token (4 Bytes)
|
|
|
|
0,
|
|
|
|
0, // ID (2 Bytes)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#CMD_CLICK,
|
2023-10-08 03:01:02 +02:00
|
|
|
0,
|
2023-10-27 06:26:54 +02:00
|
|
|
);
|
|
|
|
this.#addToken(packet, this.#token, this.#PROTOCOL_TOKEN);
|
2023-10-04 19:22:40 -04:00
|
|
|
// this.#addID(packet, 0, this.#PROTOCOL_ID)
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#sendData(packet);
|
|
|
|
if (this.#log) console.log(`WebUI -> Send Click [${elem}]`);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#sendEventNavigation(url: string) {
|
2023-10-08 03:01:02 +02:00
|
|
|
if (url !== '') {
|
|
|
|
if (this.#hasEvents) {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) console.log(`WebUI -> Send Navigation Event [${url}]`);
|
2023-09-26 18:59:25 -04:00
|
|
|
if (this.#wsStatus && this.#hasEvents) {
|
|
|
|
const packet = Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [URL]
|
|
|
|
this.#WEBUI_SIGNATURE,
|
2023-10-27 06:26:54 +02:00
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0, // Token (4 Bytes)
|
|
|
|
0,
|
|
|
|
0, // ID (2 Bytes)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#CMD_NAVIGATION,
|
2023-10-08 03:01:02 +02:00
|
|
|
...new TextEncoder().encode(url),
|
2023-10-27 06:26:54 +02:00
|
|
|
);
|
|
|
|
this.#addToken(packet, this.#token, this.#PROTOCOL_TOKEN);
|
2023-10-04 19:22:40 -04:00
|
|
|
// this.#addID(packet, 0, this.#PROTOCOL_ID)
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#sendData(packet);
|
2023-09-26 18:59:25 -04:00
|
|
|
}
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) console.log(`WebUI -> Navigation To [${url}]`);
|
|
|
|
this.#allowNavigation = true;
|
|
|
|
globalThis.location.replace(url);
|
2023-09-26 18:59:25 -04:00
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#closeWindowTimer() {
|
|
|
|
setTimeout(function () {
|
2023-10-27 06:26:54 +02:00
|
|
|
globalThis.close();
|
|
|
|
}, 1000);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-10-04 19:22:40 -04:00
|
|
|
#toUint16(value: number): number {
|
2023-10-27 06:26:54 +02:00
|
|
|
return value & 0xffff;
|
2023-10-01 20:41:52 -04:00
|
|
|
}
|
2023-10-27 17:44:17 -04:00
|
|
|
#generateCallObjects() {
|
|
|
|
for (const bind of this.#bindList) {
|
|
|
|
if (bind.trim()) {
|
|
|
|
const fn = bind.replace(`${this.#winNum}/`, '');
|
|
|
|
if (fn.trim())
|
|
|
|
this[fn] = (...args: DataTypes[]) => this.call(fn, ...args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-05 20:03:12 -04:00
|
|
|
#callPromise(fn: string, ...args: DataTypes[]) {
|
2023-10-27 06:26:54 +02:00
|
|
|
--this.#callPromiseID[0];
|
|
|
|
const callId = this.#toUint16(this.#callPromiseID[0]);
|
2023-10-05 20:03:12 -04:00
|
|
|
// Combine lengths
|
2023-10-08 03:01:02 +02:00
|
|
|
let argsLengths = args
|
|
|
|
.map((arg) => {
|
|
|
|
if (typeof arg === 'object') {
|
|
|
|
// Uint8Array
|
2023-10-27 06:26:54 +02:00
|
|
|
return arg.length;
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
|
|
|
// string, number, boolean
|
2023-10-27 06:26:54 +02:00
|
|
|
return new TextEncoder().encode(arg.toString()).length;
|
2023-10-08 03:01:02 +02:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.join(';');
|
2023-10-05 20:03:12 -04:00
|
|
|
// Combine values
|
2023-10-27 06:26:54 +02:00
|
|
|
let argsValues: Uint8Array = new Uint8Array();
|
2023-10-05 20:03:12 -04:00
|
|
|
for (const arg of args) {
|
2023-10-27 06:26:54 +02:00
|
|
|
let buffer: Uint8Array;
|
2023-10-05 20:03:12 -04:00
|
|
|
if (typeof arg === 'object') {
|
2023-10-27 06:26:54 +02:00
|
|
|
buffer = arg; // Uint8Array
|
2023-10-08 03:01:02 +02:00
|
|
|
} else {
|
2023-10-05 20:03:12 -04:00
|
|
|
// string, number, boolean
|
2023-10-27 06:26:54 +02:00
|
|
|
buffer = new TextEncoder().encode(arg.toString());
|
2023-10-05 20:03:12 -04:00
|
|
|
}
|
2023-10-27 06:26:54 +02:00
|
|
|
const temp = new Uint8Array(argsValues.length + buffer.length + 1);
|
|
|
|
temp.set(argsValues, 0);
|
|
|
|
temp.set(buffer, argsValues.length);
|
|
|
|
temp[argsValues.length + buffer.length] = 0x00;
|
|
|
|
argsValues = temp;
|
2023-10-05 20:03:12 -04:00
|
|
|
}
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
2023-10-24 20:58:59 -04:00
|
|
|
// 4: [Fn, Null, {LenLen...}, Null, {Data,Null,Data,Null...}]
|
2023-08-18 15:50:31 -04:00
|
|
|
const packet = Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#WEBUI_SIGNATURE,
|
2023-10-27 06:26:54 +02:00
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0, // Token (4 Bytes)
|
|
|
|
0,
|
|
|
|
0, // ID (2 Bytes)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#CMD_CALL_FUNC,
|
2023-08-18 15:50:31 -04:00
|
|
|
...new TextEncoder().encode(fn),
|
|
|
|
0,
|
2023-10-05 20:03:12 -04:00
|
|
|
...new TextEncoder().encode(argsLengths),
|
2023-08-22 18:11:35 -04:00
|
|
|
0,
|
2023-10-08 03:01:02 +02:00
|
|
|
...argsValues,
|
2023-10-27 06:26:54 +02:00
|
|
|
);
|
|
|
|
this.#addToken(packet, this.#token, this.#PROTOCOL_TOKEN);
|
|
|
|
this.#addID(packet, callId, this.#PROTOCOL_ID);
|
2023-08-18 15:50:31 -04:00
|
|
|
return new Promise((resolve) => {
|
2023-10-27 06:26:54 +02:00
|
|
|
this.#callPromiseResolve[callId] = resolve;
|
|
|
|
this.#sendData(packet);
|
|
|
|
});
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
// -- APIs --------------------------
|
|
|
|
/**
|
2023-10-08 03:01:02 +02:00
|
|
|
* Call a backend function
|
|
|
|
*
|
|
|
|
* @param fn - binding name
|
|
|
|
* @param data - data to be send to the backend function
|
|
|
|
* @return - Response of the backend callback string
|
|
|
|
* @example - const res = await webui.call("myID", 123, true, "Hi", new Uint8Array([0x42, 0x43, 0x44]))
|
|
|
|
*/
|
2023-10-05 20:03:12 -04:00
|
|
|
async call(fn: string, ...args: DataTypes[]): Promise<DataTypes> {
|
2023-10-27 06:26:54 +02:00
|
|
|
if (!fn) return Promise.reject(new SyntaxError('No binding name is provided'));
|
2023-10-05 20:03:12 -04:00
|
|
|
|
2023-10-27 06:26:54 +02:00
|
|
|
if (!this.#wsStatus) return Promise.reject(new Error('WebSocket is not connected'));
|
2023-10-05 20:03:12 -04:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
// Check binding list
|
2023-10-05 20:03:12 -04:00
|
|
|
if (!this.#hasEvents && !this.#bindList.includes(`${this.#winNum}/${fn}`))
|
2023-10-27 06:26:54 +02:00
|
|
|
return Promise.reject(new ReferenceError(`No binding was found for "${fn}"`));
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-10-05 20:03:12 -04:00
|
|
|
// Call backend and wait for response
|
2023-10-27 06:26:54 +02:00
|
|
|
if (this.#log) console.log(`WebUI -> Calling [${fn}(...)]`);
|
|
|
|
const response = (await this.#callPromise(fn, ...args)) as string;
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-10-05 20:03:12 -04:00
|
|
|
// WebUI lib accept `DataTypes` but return only string
|
2023-10-27 06:26:54 +02:00
|
|
|
if (typeof response !== 'string') return '';
|
2023-10-05 20:03:12 -04:00
|
|
|
|
2023-10-27 06:26:54 +02:00
|
|
|
return response;
|
2023-10-05 20:03:12 -04:00
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
/**
|
2023-10-05 20:03:12 -04:00
|
|
|
* Active or deactivate webui debug logging
|
2023-10-08 03:01:02 +02:00
|
|
|
*
|
2023-10-05 20:03:12 -04:00
|
|
|
* @param status - log status to set
|
2023-08-18 15:50:31 -04:00
|
|
|
*/
|
|
|
|
setLogging(status: boolean) {
|
|
|
|
if (status) {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log('WebUI -> Log Enabled.');
|
|
|
|
this.#log = true;
|
2023-08-18 15:50:31 -04:00
|
|
|
} else {
|
2023-10-27 06:26:54 +02:00
|
|
|
console.log('WebUI -> Log Disabled.');
|
|
|
|
this.#log = false;
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
2023-10-05 20:03:12 -04:00
|
|
|
* Encode text into base64 string
|
2023-10-08 03:01:02 +02:00
|
|
|
*
|
2023-10-05 20:03:12 -04:00
|
|
|
* @param data - text string
|
2023-08-18 15:50:31 -04:00
|
|
|
*/
|
2023-10-05 20:03:12 -04:00
|
|
|
encode(data: string): string {
|
2023-10-27 06:26:54 +02:00
|
|
|
return btoa(data);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
/**
|
2023-10-05 20:03:12 -04:00
|
|
|
* Decode base64 string into text
|
2023-10-08 03:01:02 +02:00
|
|
|
*
|
2023-10-05 20:03:12 -04:00
|
|
|
* @param data - base64 string
|
2023-08-18 15:50:31 -04:00
|
|
|
*/
|
2023-10-05 20:03:12 -04:00
|
|
|
decode(data: string): string {
|
2023-10-27 06:26:54 +02:00
|
|
|
return atob(data);
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
2023-10-05 20:03:12 -04:00
|
|
|
// Export
|
2023-10-27 06:26:54 +02:00
|
|
|
type webui = WebuiBridge;
|
|
|
|
export default webui;
|
|
|
|
export type { WebuiBridge };
|
2023-08-18 15:50:31 -04:00
|
|
|
// Wait for the html to be parsed
|
|
|
|
addEventListener('load', () => {
|
2023-10-27 06:26:54 +02:00
|
|
|
document.body.addEventListener('contextmenu', (event) => event.preventDefault());
|
|
|
|
addRefreshableEventListener(document.body, 'input', 'contextmenu', (event) => event.stopPropagation());
|
|
|
|
});
|