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
|
|
|
|
#port: number
|
|
|
|
#winNum: number
|
|
|
|
#log: boolean
|
|
|
|
#bindList: unknown[] = []
|
|
|
|
|
|
|
|
// Internals
|
|
|
|
#ws: WebSocket
|
|
|
|
#wsStatus = false
|
|
|
|
#wsStatusOnce = false
|
|
|
|
#closeReason = 0
|
|
|
|
#closeValue: string
|
|
|
|
#hasEvents = false
|
|
|
|
#fnId = 1
|
|
|
|
#fnPromiseResolve: (((data: string) => unknown) | undefined)[] = []
|
|
|
|
|
|
|
|
// WebUI const
|
|
|
|
#HEADER_SIGNATURE = 221
|
|
|
|
#HEADER_JS = 254
|
|
|
|
#HEADER_JS_QUICK = 253
|
|
|
|
#HEADER_CLICK = 252
|
|
|
|
#HEADER_SWITCH = 251
|
|
|
|
#HEADER_CLOSE = 250
|
|
|
|
#HEADER_CALL_FUNC = 249
|
|
|
|
#HEADER_SEND_RAW = 248
|
2023-08-23 22:53:06 -04:00
|
|
|
#HEADER_NEW_ID = 247
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
constructor({
|
|
|
|
port,
|
|
|
|
winNum,
|
|
|
|
bindList,
|
|
|
|
log = false,
|
|
|
|
}: {
|
|
|
|
port: number
|
|
|
|
winNum: number
|
|
|
|
bindList: unknown[]
|
|
|
|
log?: boolean
|
|
|
|
}) {
|
|
|
|
// Constructor arguments are injected by webui.c
|
|
|
|
this.#port = port
|
|
|
|
this.#winNum = winNum
|
|
|
|
this.#bindList = bindList
|
|
|
|
this.#log = log
|
|
|
|
|
|
|
|
if ('webui' in globalThis) {
|
|
|
|
throw new Error(
|
|
|
|
'Sorry. WebUI is already defined, only one instance is allowed.'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!('WebSocket' in window)) {
|
|
|
|
alert('Sorry. WebSocket is not supported by your web browser.')
|
|
|
|
if (!this.#log) globalThis.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#start()
|
|
|
|
|
|
|
|
// Handle navigation server side
|
|
|
|
if ('navigation' in globalThis) {
|
|
|
|
globalThis.navigation.addEventListener('navigate', (event) => {
|
|
|
|
const url = new URL(event.destination.url)
|
|
|
|
this.#sendEventNavigation(url.href)
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
// Handle all link click to prevent natural navigation
|
|
|
|
// Rebind listener if user inject new html
|
|
|
|
addRefreshableEventListener(
|
|
|
|
document.body,
|
|
|
|
'a',
|
|
|
|
'click',
|
|
|
|
(event) => {
|
|
|
|
event.preventDefault()
|
|
|
|
const { href } = event.target as HTMLAnchorElement
|
|
|
|
if (this.#isExternalLink(href)) {
|
|
|
|
this.#close(this.#HEADER_SWITCH, href)
|
|
|
|
} else {
|
|
|
|
this.#sendEventNavigation(href)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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(
|
|
|
|
'Sorry. WebUI failed to connect to the background application. Please try again.'
|
|
|
|
)
|
|
|
|
if (!this.#log) globalThis.close()
|
|
|
|
}
|
|
|
|
}, 1500)
|
|
|
|
}
|
|
|
|
|
|
|
|
#close(reason = 0, value = '') {
|
|
|
|
if (reason === this.#HEADER_SWITCH) this.#sendEventNavigation(value)
|
|
|
|
this.#wsStatus = false
|
|
|
|
this.#closeReason = reason
|
|
|
|
this.#closeValue = value
|
|
|
|
this.#ws.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
#freezeUi() {
|
|
|
|
document.body.style.filter = 'contrast(1%)'
|
|
|
|
}
|
|
|
|
|
2023-08-23 22:53:06 -04:00
|
|
|
#isTextBasedCommand(cmd: number): Boolean {
|
|
|
|
if(cmd !== this.#HEADER_SEND_RAW)
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
#start() {
|
|
|
|
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
|
|
|
|
this.#fnId = 1
|
|
|
|
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
|
|
|
|
if (this.#closeReason === this.#HEADER_SWITCH) {
|
|
|
|
if (this.#log) {
|
|
|
|
console.log(
|
|
|
|
`WebUI -> Connection lost -> Navigation to [${
|
|
|
|
this.#closeValue
|
|
|
|
}]`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
globalThis.location.replace(this.#closeValue)
|
|
|
|
} 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-08-25 10:24:41 -04:00
|
|
|
if (buffer8.length < 2) return
|
2023-08-18 15:50:31 -04:00
|
|
|
if (buffer8[0] !== this.#HEADER_SIGNATURE) return
|
|
|
|
|
2023-08-23 22:53:06 -04:00
|
|
|
if(this.#isTextBasedCommand(buffer8[1])) {
|
2023-08-18 15:50:31 -04:00
|
|
|
// UTF8 Text based commands
|
|
|
|
|
|
|
|
// Process Command
|
|
|
|
switch (buffer8[1]) {
|
|
|
|
case this.#HEADER_CALL_FUNC:
|
|
|
|
{
|
2023-08-25 00:33:11 -04:00
|
|
|
// Command Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [Call ID]
|
|
|
|
// 3: [Call Response]
|
|
|
|
|
|
|
|
const callResponse = this.#getDataStrFromPacket(buffer8, 3)
|
2023-08-18 15:50:31 -04:00
|
|
|
const callId = buffer8[2]
|
2023-08-25 00:33:11 -04:00
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
if (this.#log) {
|
2023-08-25 00:33:11 -04:00
|
|
|
console.log(`WebUI -> Call Response [${callResponse}]`)
|
2023-08-18 15:50:31 -04:00
|
|
|
}
|
|
|
|
if (this.#fnPromiseResolve[callId]) {
|
|
|
|
if (this.#log) {
|
|
|
|
console.log(
|
|
|
|
`WebUI -> Resolving Response #${callId}...`
|
|
|
|
)
|
|
|
|
}
|
2023-08-25 00:33:11 -04:00
|
|
|
this.#fnPromiseResolve[callId]?.(callResponse)
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#fnPromiseResolve[callId] = undefined
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break
|
|
|
|
case this.#HEADER_SWITCH:
|
2023-08-25 00:33:11 -04:00
|
|
|
// Command Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [URL]
|
|
|
|
|
|
|
|
const url = this.#getDataStrFromPacket(buffer8, 2)
|
|
|
|
console.log(`WebUI -> Switch [${url}]`)
|
|
|
|
this.#close(this.#HEADER_SWITCH, url)
|
2023-08-18 15:50:31 -04:00
|
|
|
break
|
2023-08-23 22:53:06 -04:00
|
|
|
case this.#HEADER_NEW_ID:
|
2023-08-25 00:33:11 -04:00
|
|
|
// Command Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [New Element]
|
|
|
|
|
|
|
|
const newElement = this.#getDataStrFromPacket(buffer8, 2)
|
|
|
|
console.log(`WebUI -> New Bind ID [${newElement}]`)
|
|
|
|
if(!this.#bindList.includes(newElement))
|
|
|
|
this.#bindList.push(newElement)
|
2023-08-23 22:53:06 -04:00
|
|
|
break
|
2023-08-18 15:50:31 -04:00
|
|
|
case this.#HEADER_CLOSE:
|
2023-08-25 00:33:11 -04:00
|
|
|
// Command Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
|
2023-08-23 22:53:06 -04:00
|
|
|
console.log(`WebUI -> Close`)
|
2023-08-25 10:24:41 -04:00
|
|
|
this.#ws.close()
|
2023-08-18 15:50:31 -04:00
|
|
|
break
|
|
|
|
case this.#HEADER_JS_QUICK:
|
|
|
|
case this.#HEADER_JS:
|
|
|
|
{
|
2023-08-25 00:33:11 -04:00
|
|
|
// Command Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [Script]
|
|
|
|
|
|
|
|
const script = this.#getDataStrFromPacket(buffer8, 3)
|
|
|
|
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-08-25 00:33:11 -04:00
|
|
|
console.log(`WebUI -> 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
|
|
|
|
}
|
|
|
|
if (buffer8[1] === this.#HEADER_JS_QUICK) return
|
|
|
|
if (FunReturn === undefined) {
|
|
|
|
FunReturn = 'undefined'
|
|
|
|
}
|
|
|
|
|
|
|
|
// Logging
|
|
|
|
if (this.#log && !FunError)
|
|
|
|
console.log(`WebUI -> JS -> Return [${FunReturn}]`)
|
|
|
|
if (this.#log && FunError)
|
|
|
|
console.log(`WebUI -> JS -> Error [${FunReturn}]`)
|
|
|
|
|
2023-08-25 00:33:11 -04:00
|
|
|
// Response Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [ID]
|
|
|
|
// 3: [Error]
|
|
|
|
// 4: [Script Response]
|
2023-08-18 15:50:31 -04:00
|
|
|
const Return8 = Uint8Array.of(
|
|
|
|
this.#HEADER_SIGNATURE,
|
|
|
|
this.#HEADER_JS,
|
|
|
|
buffer8[2],
|
|
|
|
FunError ? 0 : 1,
|
|
|
|
...new TextEncoder().encode(FunReturn)
|
|
|
|
)
|
|
|
|
|
|
|
|
if (this.#wsStatus) this.#ws.send(Return8.buffer)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Raw binary commands
|
|
|
|
|
|
|
|
// Process Command
|
|
|
|
switch (buffer8[1]) {
|
|
|
|
case this.#HEADER_SEND_RAW:
|
|
|
|
{
|
2023-08-25 00:33:11 -04:00
|
|
|
// Command Packet
|
2023-08-18 15:50:31 -04:00
|
|
|
// 0: [Signature]
|
2023-08-25 00:33:11 -04:00
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [Function]
|
|
|
|
// 3: [Null]
|
|
|
|
// 4: [Raw Data]
|
2023-08-18 15:50:31 -04:00
|
|
|
|
|
|
|
// Get function name
|
2023-08-25 00:33:11 -04:00
|
|
|
const functionName: string = this.#getDataStrFromPacket(buffer8, 2)
|
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);
|
|
|
|
|
|
|
|
// 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-08-25 00:33:11 -04:00
|
|
|
// Response Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [Element]
|
2023-08-18 15:50:31 -04:00
|
|
|
const packet =
|
|
|
|
elem !== ''
|
|
|
|
? Uint8Array.of(
|
|
|
|
this.#HEADER_SIGNATURE,
|
|
|
|
this.#HEADER_CLICK,
|
|
|
|
...new TextEncoder().encode(elem)
|
|
|
|
)
|
|
|
|
: Uint8Array.of(
|
|
|
|
this.#HEADER_SIGNATURE,
|
|
|
|
this.#HEADER_CLICK,
|
|
|
|
0
|
|
|
|
)
|
|
|
|
this.#ws.send(packet.buffer)
|
|
|
|
if (this.#log) console.log(`WebUI -> Click [${elem}]`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#sendEventNavigation(url: string) {
|
|
|
|
if (this.#hasEvents && this.#wsStatus && url !== '') {
|
|
|
|
const packet = Uint8Array.of(
|
2023-08-25 00:33:11 -04:00
|
|
|
// Response Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [URL]
|
2023-08-18 15:50:31 -04:00
|
|
|
this.#HEADER_SIGNATURE,
|
|
|
|
this.#HEADER_SWITCH,
|
|
|
|
...new TextEncoder().encode(url)
|
|
|
|
)
|
|
|
|
this.#ws.send(packet.buffer)
|
|
|
|
if (this.#log) console.log(`WebUI -> Navigation [${url}]`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#isExternalLink(url: string) {
|
|
|
|
return new URL(url).host === globalThis.location.host
|
|
|
|
}
|
|
|
|
#closeWindowTimer() {
|
|
|
|
setTimeout(function () {
|
|
|
|
globalThis.close()
|
|
|
|
}, 1000)
|
|
|
|
}
|
2023-08-22 18:11:35 -04:00
|
|
|
#fnPromise(fn: string, value: any) {
|
|
|
|
if (this.#log) console.log(`WebUI -> Call [${fn}](${value})`)
|
2023-08-18 15:50:31 -04:00
|
|
|
const callId = this.#fnId++
|
|
|
|
|
2023-08-25 00:33:11 -04:00
|
|
|
// Response Packet
|
|
|
|
// 0: [Signature]
|
|
|
|
// 1: [CMD]
|
|
|
|
// 2: [Call ID]
|
|
|
|
// 3: [Element ID, Null, Len, Null, Data, Null]
|
|
|
|
|
2023-08-18 15:50:31 -04:00
|
|
|
const packet = Uint8Array.of(
|
|
|
|
this.#HEADER_SIGNATURE,
|
|
|
|
this.#HEADER_CALL_FUNC,
|
|
|
|
callId,
|
|
|
|
...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
|
|
|
|
)
|
|
|
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
this.#fnPromiseResolve[callId] = resolve
|
|
|
|
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
|
|
|
|
const response = (await this.#fnPromise(
|
|
|
|
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()
|
|
|
|
)
|
|
|
|
})
|