2023-08-18 15:50:31 -04:00
|
|
|
'use-strict' // Force strict mode for transpiled
|
|
|
|
|
|
|
|
/*
|
|
|
|
WebUI Bridge
|
|
|
|
|
|
|
|
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
|
|
|
|
import { AsyncFunction, addRefreshableEventListener } from './utils.ts'
|
|
|
|
|
|
|
|
type B64string = string
|
|
|
|
type JSONValue =
|
|
|
|
| string
|
|
|
|
| number
|
|
|
|
| boolean
|
|
|
|
| { [x: string]: JSONValue | undefined }
|
|
|
|
| JSONValue[]
|
2023-08-22 18:11:35 -04:00
|
|
|
| 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-04 19:22:40 -04:00
|
|
|
#token: number
|
2023-08-18 15:50:31 -04:00
|
|
|
#port: number
|
|
|
|
#winNum: number
|
|
|
|
#bindList: unknown[] = []
|
2023-08-29 21:12:45 -04:00
|
|
|
#log: boolean
|
|
|
|
#winX: number
|
|
|
|
#winY: number
|
|
|
|
#winW: number
|
|
|
|
#winH: number
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Internals
|
|
|
|
#ws: WebSocket
|
|
|
|
#wsStatus = false
|
|
|
|
#wsStatusOnce = false
|
|
|
|
#closeReason = 0
|
|
|
|
#closeValue: string
|
|
|
|
#hasEvents = false
|
2023-10-04 19:22:40 -04:00
|
|
|
#callPromiseID = new Uint16Array(1)
|
2023-09-30 23:15:55 -04:00
|
|
|
#callPromiseResolve: (((data: string) => unknown) | undefined)[] = []
|
2023-09-26 18:59:25 -04:00
|
|
|
#allowNavigation = false
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// WebUI const
|
2023-10-04 19:22:40 -04: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
|
|
|
|
#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);
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
constructor({
|
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-04 19:22:40 -04:00
|
|
|
token: number
|
2023-08-18 15:50:31 -04:00
|
|
|
port: number
|
|
|
|
winNum: number
|
|
|
|
bindList: unknown[]
|
|
|
|
log?: boolean
|
2023-08-29 21:12:45 -04:00
|
|
|
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-04 19:22:40 -04:00
|
|
|
this.#token = token
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#port = port
|
|
|
|
this.#winNum = winNum
|
|
|
|
this.#bindList = bindList
|
|
|
|
this.#log = log
|
2023-08-29 21:12:45 -04:00
|
|
|
this.#winX = winX
|
|
|
|
this.#winY = winY
|
|
|
|
this.#winW = winW
|
|
|
|
this.#winH = winH
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
// Token
|
|
|
|
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) {
|
|
|
|
throw new Error(
|
|
|
|
'Sorry. WebUI is already defined, only one instance is allowed.'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2023-08-29 21:12:45 -04:00
|
|
|
// Positioning the current window
|
|
|
|
if (this.#winX !== undefined && this.#winY !== undefined) {
|
|
|
|
window.moveTo(this.#winX, this.#winY)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resize the current window
|
|
|
|
if (this.#winW !== undefined && this.#winH !== undefined) {
|
|
|
|
window.resizeTo(this.#winW, this.#winH)
|
|
|
|
}
|
|
|
|
|
|
|
|
// WebSocket
|
2023-08-18 15:50:31 -04:00
|
|
|
if (!('WebSocket' in window)) {
|
|
|
|
alert('Sorry. WebSocket is not supported by your web browser.')
|
|
|
|
if (!this.#log) globalThis.close()
|
|
|
|
}
|
|
|
|
|
2023-08-29 21:12:45 -04:00
|
|
|
// Connect to the backend application
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#start()
|
|
|
|
|
|
|
|
// Handle navigation server side
|
|
|
|
if ('navigation' in globalThis) {
|
|
|
|
globalThis.navigation.addEventListener('navigate', (event) => {
|
2023-09-26 18:59:25 -04:00
|
|
|
if(!this.#allowNavigation) {
|
|
|
|
event.preventDefault()
|
|
|
|
const url = new URL(event.destination.url)
|
|
|
|
if (this.#hasEvents) {
|
|
|
|
if (this.#log) console.log(`WebUI -> DOM -> Navigation Event [${url.href}]`)
|
|
|
|
this.#sendEventNavigation(url.href)
|
|
|
|
}
|
|
|
|
else {
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#close(this.#CMD_NAVIGATION, url.href)
|
2023-09-26 18:59:25 -04: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
|
|
|
|
addRefreshableEventListener(
|
|
|
|
document.body,
|
|
|
|
'a',
|
|
|
|
'click',
|
|
|
|
(event) => {
|
2023-09-26 18:59:25 -04:00
|
|
|
if(!this.#allowNavigation) {
|
|
|
|
event.preventDefault()
|
|
|
|
const { href } = event.target as HTMLAnchorElement
|
|
|
|
if (this.#hasEvents) {
|
|
|
|
if (this.#log) console.log(`WebUI -> DOM -> Navigation Click Event [${href}]`)
|
|
|
|
// if (this.#isExternalLink(href)) {
|
2023-10-04 19:22:40 -04:00
|
|
|
// this.#close(this.#CMD_NAVIGATION, href)
|
2023-09-26 18:59:25 -04:00
|
|
|
// } else {
|
|
|
|
// this.#sendEventNavigation(href)
|
|
|
|
// }
|
|
|
|
this.#sendEventNavigation(href)
|
|
|
|
}
|
|
|
|
else {
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#close(this.#CMD_NAVIGATION, href)
|
2023-09-26 18:59:25 -04:00
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prevent F5 refresh
|
|
|
|
document.addEventListener('keydown', (event) => {
|
|
|
|
// Disable F5
|
|
|
|
if (this.#log) return
|
|
|
|
if (event.key === 'F5') event.preventDefault()
|
|
|
|
})
|
|
|
|
|
|
|
|
onbeforeunload = () => {
|
|
|
|
this.#close()
|
|
|
|
}
|
2023-08-22 18:11:35 -04:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
setTimeout(() => {
|
|
|
|
if (!this.#wsStatusOnce) {
|
|
|
|
this.#freezeUi()
|
|
|
|
alert(
|
2023-08-29 21:12:45 -04:00
|
|
|
'Sorry. WebUI failed to connect to the backend application. Please try again.'
|
2023-08-18 15:50:31 -04:00
|
|
|
)
|
|
|
|
if (!this.#log) globalThis.close()
|
|
|
|
}
|
|
|
|
}, 1500)
|
|
|
|
}
|
|
|
|
|
|
|
|
#close(reason = 0, value = '') {
|
2023-10-04 19:22:40 -04:00
|
|
|
// if (reason === this.#CMD_NAVIGATION) this.#sendEventNavigation(value)
|
2023-08-18 15:50:31 -04: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) {
|
|
|
|
console.log(
|
|
|
|
`WebUI -> Close -> Navigation to [${value}]`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
this.#allowNavigation = true
|
|
|
|
globalThis.location.replace(this.#closeValue)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (this.#log) {
|
|
|
|
console.log(
|
|
|
|
`WebUI -> Close.`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#freezeUi() {
|
|
|
|
document.body.style.filter = 'contrast(1%)'
|
|
|
|
}
|
|
|
|
|
2023-08-23 22:53:06 -04:00
|
|
|
#isTextBasedCommand(cmd: number): Boolean {
|
2023-10-04 19:22:40 -04:00
|
|
|
if(cmd !== this.#CMD_SEND_RAW)
|
2023-08-23 22:53:06 -04:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-08-25 00:33:11 -04:00
|
|
|
#getDataStrFromPacket(buffer: Uint8Array, startIndex: number): string {
|
2023-08-18 15:50:31 -04:00
|
|
|
let stringBytes: number[] = [];
|
|
|
|
for (let i = startIndex; i < buffer.length; i++) {
|
|
|
|
if (buffer[i] === 0) { // Check for null byte
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
stringBytes.push(buffer[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert the array of bytes to a string
|
2023-08-25 00:33:11 -04:00
|
|
|
const stringText = new TextDecoder().decode(new Uint8Array(stringBytes));
|
2023-08-18 15:50:31 -04:00
|
|
|
return stringText;
|
|
|
|
}
|
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
#getID(buffer: Uint8Array, index: number): number {
|
|
|
|
if (index < 0 || index >= buffer.length - 1) {
|
|
|
|
throw new Error('Index out of bounds or insufficient data.');
|
|
|
|
}
|
|
|
|
const firstByte = buffer[index];
|
|
|
|
const secondByte = buffer[index + 1];
|
|
|
|
const combined = (secondByte << 8) | firstByte; // Works only for little-endian
|
|
|
|
return combined;
|
|
|
|
}
|
|
|
|
|
|
|
|
#addToken(buffer: Uint8Array, value: number, index: number): void {
|
|
|
|
if (value < 0 || value > 0xFFFFFFFF) {
|
|
|
|
throw new Error('Number is out of the range for 4 bytes representation.');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index < 0 || index > buffer.length - 4) {
|
|
|
|
throw new Error('Index out of bounds or insufficient space in buffer.');
|
|
|
|
}
|
|
|
|
|
|
|
|
// WebUI expect Little-endian (Work for Little/Big endian platforms)
|
|
|
|
buffer[index] = value & 0xFF;
|
|
|
|
buffer[index + 1] = (value >>> 8) & 0xFF;
|
|
|
|
buffer[index + 2] = (value >>> 16) & 0xFF;
|
|
|
|
buffer[index + 3] = (value >>> 24) & 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
#addID(buffer: Uint8Array, value: number, index: number): void {
|
|
|
|
if (value < 0 || value > 0xFFFF) {
|
|
|
|
throw new Error('Number is out of the range for 2 bytes representation.');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index < 0 || index > buffer.length - 2) {
|
|
|
|
throw new Error('Index out of bounds or insufficient space in buffer.');
|
|
|
|
}
|
|
|
|
|
|
|
|
// WebUI expect Little-endian (Work for Little/Big endian platforms)
|
|
|
|
buffer[index] = value & 0xFF; // Least significant byte
|
|
|
|
buffer[index + 1] = (value >>> 8) & 0xFF; // Most significant byte
|
|
|
|
}
|
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
#start() {
|
2023-10-01 20:41:52 -04:00
|
|
|
this.#callPromiseID[0] = 0
|
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#bindList.includes(this.#winNum + '/')) {
|
|
|
|
this.#hasEvents = true
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#ws = new WebSocket(
|
|
|
|
`ws://localhost:${this.#port}/_webui_ws_connect`
|
|
|
|
)
|
|
|
|
this.#ws.binaryType = 'arraybuffer'
|
|
|
|
|
|
|
|
this.#ws.onopen = () => {
|
|
|
|
this.#wsStatus = true
|
|
|
|
this.#wsStatusOnce = true
|
|
|
|
if (this.#log) console.log('WebUI -> Connected')
|
|
|
|
this.#clicksListener()
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#ws.onerror = () => {
|
|
|
|
if (this.#log) console.log('WebUI -> Connection Failed')
|
|
|
|
this.#freezeUi()
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#ws.onclose = (event) => {
|
|
|
|
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) {
|
|
|
|
console.log(
|
2023-10-01 20:41:52 -04:00
|
|
|
`WebUI -> Connection closed du to Navigation to [${this.#closeValue}]`
|
2023-08-18 15:50:31 -04:00
|
|
|
)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (this.#log) {
|
|
|
|
console.log(`WebUI -> Connection lost (${event.code})`)
|
|
|
|
this.#freezeUi()
|
|
|
|
} else {
|
|
|
|
this.#closeWindowTimer()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#ws.onmessage = async (event) => {
|
|
|
|
const buffer8 = new Uint8Array(event.data)
|
2023-10-04 19:22:40 -04:00
|
|
|
if (buffer8.length < this.#PROTOCOL_SIZE) return
|
|
|
|
if (buffer8[this.#PROTOCOL_SIGN] !== this.#WEBUI_SIGNATURE) return
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
if(this.#isTextBasedCommand(buffer8[this.#PROTOCOL_CMD])) {
|
2023-08-18 15:50:31 -04:00
|
|
|
// UTF8 Text based commands
|
|
|
|
|
2023-10-04 19:22:40 -04: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_CALL_FUNC:
|
2023-08-18 15:50:31 -04:00
|
|
|
{
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Call Response]
|
2023-08-25 00:33:11 -04:00
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
const callResponse = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA)
|
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#log) {
|
2023-09-26 18:59:25 -04:00
|
|
|
console.log(`WebUI -> CMD -> Call Response [${callResponse}]`)
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
2023-09-30 23:15:55 -04:00
|
|
|
if (this.#callPromiseResolve[callId]) {
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#log) {
|
|
|
|
console.log(
|
2023-09-26 18:59:25 -04:00
|
|
|
`WebUI -> CMD -> Resolving Response #${callId}...`
|
2023-08-18 15:50:31 -04:00
|
|
|
)
|
|
|
|
}
|
2023-09-30 23:15:55 -04:00
|
|
|
this.#callPromiseResolve[callId]?.(callResponse)
|
|
|
|
this.#callPromiseResolve[callId] = undefined
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break
|
2023-10-04 19:22:40 -04:00
|
|
|
case this.#CMD_NAVIGATION:
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [URL]
|
|
|
|
|
|
|
|
const url = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA)
|
2023-09-26 18:59:25 -04:00
|
|
|
console.log(`WebUI -> CMD -> Navigation [${url}]`)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#close(this.#CMD_NAVIGATION, url)
|
2023-08-18 15:50:31 -04:00
|
|
|
break
|
2023-10-04 19:22:40 -04:00
|
|
|
case this.#CMD_NEW_ID:
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [New Element]
|
|
|
|
|
|
|
|
const newElement = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA)
|
2023-09-26 18:59:25 -04:00
|
|
|
console.log(`WebUI -> CMD -> New Bind ID [${newElement}]`)
|
2023-08-25 00:33:11 -04:00
|
|
|
if(!this.#bindList.includes(newElement))
|
|
|
|
this.#bindList.push(newElement)
|
2023-08-23 22:53:06 -04:00
|
|
|
break
|
2023-10-04 19:22:40 -04:00
|
|
|
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-08-25 00:33:11 -04:00
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
const script = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA)
|
2023-08-25 00:33:11 -04:00
|
|
|
const scriptSanitize = script.replace(
|
2023-08-18 15:50:31 -04:00
|
|
|
/(?:\r\n|\r|\n)/g,
|
|
|
|
'\n'
|
|
|
|
)
|
2023-08-25 00:33:11 -04:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#log)
|
2023-09-26 18:59:25 -04:00
|
|
|
console.log(`WebUI -> CMD -> JS [${scriptSanitize}]`)
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Get callback result
|
|
|
|
let FunReturn = 'undefined'
|
|
|
|
let FunError = false
|
|
|
|
try {
|
2023-08-25 00:33:11 -04:00
|
|
|
FunReturn = await AsyncFunction(scriptSanitize)()
|
2023-08-18 15:50:31 -04:00
|
|
|
} catch (e) {
|
|
|
|
FunError = true
|
|
|
|
FunReturn = e.message
|
|
|
|
}
|
2023-10-04 19:22:40 -04:00
|
|
|
|
|
|
|
// Stop if this is a quick call
|
|
|
|
if (buffer8[this.#PROTOCOL_CMD] === this.#CMD_JS_QUICK) return
|
|
|
|
|
|
|
|
// Get the call return
|
2023-08-18 15:50:31 -04:00
|
|
|
if (FunReturn === undefined) {
|
|
|
|
FunReturn = 'undefined'
|
|
|
|
}
|
|
|
|
|
|
|
|
// Logging
|
|
|
|
if (this.#log && !FunError)
|
2023-10-03 14:35:44 -04:00
|
|
|
console.log(`WebUI -> CMD -> JS -> Return Success [${FunReturn}]`)
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#log && FunError)
|
2023-10-03 14:35:44 -04:00
|
|
|
console.log(`WebUI -> CMD -> JS -> Return Error [${FunReturn}]`)
|
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: [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,
|
|
|
|
0, 0, 0, 0, // Token (4 Bytes)
|
|
|
|
0, 0, // ID (2 Bytes)
|
|
|
|
this.#CMD_JS,
|
2023-10-03 14:35:44 -04:00
|
|
|
FunError ? 1 : 0,
|
2023-08-18 15:50:31 -04:00
|
|
|
...new TextEncoder().encode(FunReturn)
|
|
|
|
)
|
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#addToken(Return8, this.#token, this.#PROTOCOL_TOKEN)
|
|
|
|
this.#addID(Return8, callId, this.#PROTOCOL_ID)
|
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#wsStatus) this.#ws.send(Return8.buffer)
|
|
|
|
}
|
|
|
|
break
|
2023-10-04 19:22:40 -04:00
|
|
|
case this.#CMD_CLOSE:
|
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
2023-09-22 22:50:18 -04:00
|
|
|
|
|
|
|
if (!this.#log)
|
|
|
|
globalThis.close()
|
|
|
|
else {
|
2023-09-26 18:59:25 -04:00
|
|
|
console.log(`WebUI -> CMD -> Close`)
|
2023-09-22 22:50:18 -04:00
|
|
|
this.#ws.close()
|
|
|
|
}
|
|
|
|
break
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Raw binary commands
|
|
|
|
|
|
|
|
// Process Command
|
2023-10-04 19:22:40 -04:00
|
|
|
switch (buffer8[this.#PROTOCOL_CMD]) {
|
|
|
|
case this.#CMD_SEND_RAW:
|
2023-08-18 15:50:31 -04:00
|
|
|
{
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Function,Null,Raw Data]
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Get function name
|
2023-10-04 19:22:40 -04:00
|
|
|
const functionName: string = this.#getDataStrFromPacket(buffer8, this.#PROTOCOL_DATA)
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Get the raw data
|
2023-08-25 00:33:11 -04:00
|
|
|
const rawDataIndex: number = 2 + functionName.length + 1
|
2023-08-18 15:50:31 -04:00
|
|
|
const userRawData = buffer8.subarray(rawDataIndex);
|
2023-09-26 18:59:25 -04:00
|
|
|
|
|
|
|
if (this.#log)
|
|
|
|
console.log(`WebUI -> CMD -> Send Raw ${buffer8.length} bytes to [${functionName}()]`)
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Call the user function, and pass the raw data
|
|
|
|
if (typeof window[functionName] === 'function')
|
|
|
|
window[functionName](userRawData);
|
|
|
|
else
|
|
|
|
await AsyncFunction(functionName + '(userRawData);')()
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#clicksListener() {
|
|
|
|
Object.keys(window).forEach((key) => {
|
|
|
|
if (/^on(click)/.test(key)) {
|
|
|
|
globalThis.addEventListener(key.slice(2), (event) => {
|
|
|
|
if (!(event.target instanceof HTMLElement)) return
|
|
|
|
if (
|
|
|
|
this.#hasEvents ||
|
|
|
|
(event.target.id !== '' &&
|
|
|
|
this.#bindList.includes(
|
|
|
|
this.#winNum + '/' + event.target?.id
|
|
|
|
))
|
|
|
|
) {
|
|
|
|
this.#sendClick(event.target.id)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#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,
|
|
|
|
0, 0, 0, 0, // Token (4 Bytes)
|
|
|
|
0, 0, // ID (2 Bytes)
|
|
|
|
this.#CMD_CLICK,
|
|
|
|
...new TextEncoder().encode(elem),
|
|
|
|
0
|
2023-08-18 15:50:31 -04:00
|
|
|
)
|
|
|
|
: Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#WEBUI_SIGNATURE,
|
|
|
|
0, 0, 0, 0, // Token (4 Bytes)
|
|
|
|
0, 0, // ID (2 Bytes)
|
|
|
|
this.#CMD_CLICK,
|
2023-08-18 15:50:31 -04:00
|
|
|
0
|
|
|
|
)
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#addToken(packet, this.#token, this.#PROTOCOL_TOKEN)
|
|
|
|
// this.#addID(packet, 0, this.#PROTOCOL_ID)
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#ws.send(packet.buffer)
|
2023-09-26 18:59:25 -04:00
|
|
|
if (this.#log) console.log(`WebUI -> Send Click [${elem}]`)
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#sendEventNavigation(url: string) {
|
2023-09-26 18:59:25 -04:00
|
|
|
if(url !== '') {
|
|
|
|
if(this.#hasEvents) {
|
|
|
|
if (this.#log) console.log(`WebUI -> Send Navigation Event [${url}]`)
|
|
|
|
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,
|
|
|
|
0, 0, 0, 0, // Token (4 Bytes)
|
|
|
|
0, 0, // ID (2 Bytes)
|
|
|
|
this.#CMD_NAVIGATION,
|
2023-09-26 18:59:25 -04:00
|
|
|
...new TextEncoder().encode(url)
|
|
|
|
)
|
2023-10-04 19:22:40 -04:00
|
|
|
|
|
|
|
this.#addToken(packet, this.#token, this.#PROTOCOL_TOKEN)
|
|
|
|
// this.#addID(packet, 0, this.#PROTOCOL_ID)
|
|
|
|
|
2023-09-26 18:59:25 -04:00
|
|
|
this.#ws.send(packet.buffer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (this.#log) console.log(`WebUI -> Navigation To [${url}]`)
|
|
|
|
this.#allowNavigation = true
|
|
|
|
globalThis.location.replace(url)
|
|
|
|
}
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#isExternalLink(url: string) {
|
|
|
|
return new URL(url).host === globalThis.location.host
|
|
|
|
}
|
2023-10-01 20:41:52 -04:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
#closeWindowTimer() {
|
|
|
|
setTimeout(function () {
|
|
|
|
globalThis.close()
|
|
|
|
}, 1000)
|
|
|
|
}
|
2023-10-01 20:41:52 -04:00
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
#toUint16(value: number): number {
|
|
|
|
return value & 0xFFFF;
|
2023-10-01 20:41:52 -04:00
|
|
|
}
|
|
|
|
|
2023-09-30 23:15:55 -04:00
|
|
|
#callPromise(fn: string, value: any) {
|
2023-10-01 20:41:52 -04:00
|
|
|
--this.#callPromiseID[0]
|
2023-10-04 19:22:40 -04:00
|
|
|
const callId = this.#toUint16(this.#callPromiseID[0])
|
2023-09-30 23:15:55 -04:00
|
|
|
if (this.#log) console.log(`WebUI -> Call [${fn}](${value}) (ID:${this.#callPromiseID[0]})`)
|
2023-08-18 15:50:31 -04:00
|
|
|
|
2023-10-04 19:22:40 -04:00
|
|
|
// Protocol
|
|
|
|
// 0: [SIGNATURE]
|
|
|
|
// 1: [TOKEN]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [CMD]
|
|
|
|
// 4: [Element ID, Null, Len, Null, Data, Null]
|
2023-08-25 00:33:11 -04:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
const packet = Uint8Array.of(
|
2023-10-04 19:22:40 -04:00
|
|
|
this.#WEBUI_SIGNATURE,
|
|
|
|
0, 0, 0, 0, // Token (4 Bytes)
|
|
|
|
0, 0, // ID (2 Bytes)
|
|
|
|
this.#CMD_CALL_FUNC,
|
2023-08-18 15:50:31 -04:00
|
|
|
...new TextEncoder().encode(fn),
|
|
|
|
0,
|
2023-08-22 18:11:35 -04:00
|
|
|
...new TextEncoder().encode(String(value.length)),
|
|
|
|
0,
|
|
|
|
...typeof value === 'object' ? value : new TextEncoder().encode(value),
|
2023-08-18 15:50:31 -04:00
|
|
|
0
|
|
|
|
)
|
|
|
|
|
2023-10-04 19:22:40 -04: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-09-30 23:15:55 -04:00
|
|
|
this.#callPromiseResolve[callId] = resolve
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#ws.send(packet.buffer)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// -- APIs --------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call a backend binding from the frontend.
|
|
|
|
* @param bindingName - Backend bind name.
|
|
|
|
* @param payload - Payload to send to the binding, accept any JSON valid data
|
|
|
|
* (string, number, array, object, boolean, undefined).
|
|
|
|
* @return - Response of the backend callback as JSON compatible value.
|
|
|
|
* @example
|
|
|
|
* __Backend (C)__
|
|
|
|
* ```c
|
|
|
|
* webui_bind(window, "get_cwd", get_current_working_directory);
|
|
|
|
* ```
|
|
|
|
* __Frontend (JS)__
|
|
|
|
* ```js
|
|
|
|
* const cwd = await webui.call("get_cwd");
|
|
|
|
* ```
|
|
|
|
* @example
|
|
|
|
* __Backend (C)__
|
|
|
|
* ```c
|
|
|
|
* webui_bind(window, "write_file", write_file);
|
|
|
|
* ```
|
|
|
|
* __Frontend (JS)__
|
|
|
|
* ```js
|
|
|
|
* webui.call("write_file", "content to write")
|
|
|
|
* .then(() => console.log("Success"))
|
|
|
|
* .catch(() => console.error("Can't write the file"))
|
|
|
|
* ```
|
|
|
|
* @example
|
|
|
|
* __Backend (C)__
|
|
|
|
* ```c
|
|
|
|
* //complex_datas() returns the following JSON string
|
|
|
|
* //'{ "count": 1, "list": [ 1, 2, 3 ], "isGood": true }'
|
|
|
|
* webui_bind(window, "complex_datas", complex_datas);
|
|
|
|
* ```
|
|
|
|
* __Frontend (JS)__
|
|
|
|
* ```js
|
|
|
|
* type ComplexDatas = {
|
|
|
|
* count: number;
|
|
|
|
* list: number[];
|
|
|
|
* isGood: boolean;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* const { count, list, isGood } = await webui
|
|
|
|
* .call<ComplexDatas>("complex_datas");
|
|
|
|
*
|
|
|
|
* //count = 1;
|
|
|
|
* //list = [ 1, 2, 3 ];
|
|
|
|
* //isGood = true;
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
async call<
|
|
|
|
Response extends JSONValue = string,
|
|
|
|
Payload extends JSONValue = JSONValue
|
|
|
|
>(bindingName: string, payload?: Payload): Promise<Response | void> {
|
|
|
|
if (!bindingName)
|
2023-08-22 18:11:35 -04:00
|
|
|
return Promise.reject(new SyntaxError('No binding name is provided'))
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
if (!this.#wsStatus)
|
2023-08-22 18:11:35 -04:00
|
|
|
return Promise.reject(new Error('WebSocket is not connected'))
|
2023-08-18 15:50:31 -04:00
|
|
|
// Check binding list
|
|
|
|
if (
|
|
|
|
!this.#hasEvents &&
|
|
|
|
!this.#bindList.includes(`${this.#winNum}/${bindingName}`)
|
|
|
|
)
|
|
|
|
return Promise.reject(
|
2023-08-22 18:11:35 -04:00
|
|
|
new ReferenceError(`No binding was found for "${bindingName}"`)
|
2023-08-18 15:50:31 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// Get the binding response
|
2023-09-30 23:15:55 -04:00
|
|
|
const response = (await this.#callPromise(
|
2023-08-18 15:50:31 -04:00
|
|
|
bindingName,
|
|
|
|
payload === undefined
|
|
|
|
? ''
|
2023-08-22 18:11:35 -04:00
|
|
|
: typeof payload === 'string' // String()
|
|
|
|
? payload
|
|
|
|
: typeof payload === 'object' // Uint8Array()
|
2023-08-18 15:50:31 -04:00
|
|
|
? payload
|
2023-08-22 18:11:35 -04:00
|
|
|
: JSON.stringify(payload) // JSON String
|
2023-08-18 15:50:31 -04:00
|
|
|
)) as string | void
|
|
|
|
|
|
|
|
// Handle response type (void, string or JSON value)
|
|
|
|
if (response === undefined) return
|
|
|
|
|
|
|
|
try {
|
|
|
|
return JSON.parse(response)
|
|
|
|
} catch {
|
|
|
|
//@ts-ignore string is a valid JSON value
|
|
|
|
return response
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Active or deactivate webui debug logging.
|
|
|
|
* @param status - log status to set.
|
|
|
|
*/
|
|
|
|
setLogging(status: boolean) {
|
|
|
|
if (status) {
|
|
|
|
console.log('WebUI -> Log Enabled.')
|
|
|
|
this.#log = true
|
|
|
|
} else {
|
|
|
|
console.log('WebUI -> Log Disabled.')
|
|
|
|
this.#log = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Encode datas into base64 string.
|
|
|
|
* @param datas - string or JSON value.
|
|
|
|
*/
|
|
|
|
encode(datas: JSONValue): B64string {
|
|
|
|
if (typeof datas === 'string') return btoa(datas)
|
|
|
|
return btoa(JSON.stringify(datas))
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decode a base64 string into any JSON valid format
|
|
|
|
* (string, number, array, object, boolean).
|
|
|
|
* @param b64 - base64 string to decode.
|
|
|
|
*/
|
|
|
|
decode<T extends JSONValue>(b64: B64string): T {
|
|
|
|
try {
|
|
|
|
return JSON.parse(atob(b64))
|
|
|
|
} catch {
|
|
|
|
//@ts-ignore string a valid JSON value
|
|
|
|
return atob(b64)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-21 09:23:29 -04:00
|
|
|
type webui = WebuiBridge
|
2023-08-18 15:50:31 -04:00
|
|
|
export default webui
|
2023-08-21 09:23:29 -04:00
|
|
|
export type { WebuiBridge }
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Wait for the html to be parsed
|
|
|
|
addEventListener('load', () => {
|
|
|
|
document.body.addEventListener('contextmenu', (event) =>
|
|
|
|
event.preventDefault()
|
|
|
|
)
|
|
|
|
addRefreshableEventListener(
|
|
|
|
document.body,
|
|
|
|
'input',
|
|
|
|
'contextmenu',
|
|
|
|
(event) => event.stopPropagation()
|
|
|
|
)
|
|
|
|
})
|