rui_orig/wasmBridge.go

264 lines
6.7 KiB
Go

//go:build wasm
package rui
import (
"fmt"
"strconv"
"sync"
"syscall/js"
)
type wasmBridge struct {
queue chan string
answer map[int]chan DataObject
answerID int
answerMutex sync.Mutex
closed bool
canvas js.Value
}
func createWasmBridge() webBridge {
bridge := new(wasmBridge)
bridge.queue = make(chan string, 1000)
bridge.answerID = 1
bridge.answer = make(map[int]chan DataObject)
bridge.closed = false
return bridge
}
func (bridge *wasmBridge) startUpdateScript(htmlID string) bool {
return false
}
func (bridge *wasmBridge) finishUpdateScript(htmlID string) {
}
func (bridge *wasmBridge) printFuncToLog(funcName string, args ...any) {
if ProtocolInDebugLog {
text := funcName + "("
for i, arg := range args {
if i > 0 {
text += fmt.Sprintf(", `%v`", arg)
} else {
text += fmt.Sprintf("`%v`", arg)
}
}
DebugLog(text + ")")
}
}
func (bridge *wasmBridge) callFunc(funcName string, args ...any) bool {
bridge.printFuncToLog(funcName, args...)
js.Global().Call(funcName, args...)
return true
}
func (bridge *wasmBridge) updateInnerHTML(htmlID, html string) {
bridge.updateProperty(htmlID, "innerHTML", html)
}
func (bridge *wasmBridge) appendToInnerHTML(htmlID, html string) {
if ProtocolInDebugLog {
DebugLog(fmt.Sprintf("%s.innerHTML += '%s'", htmlID, html))
}
element := js.Global().Get("document").Call("getElementById", htmlID)
if !element.IsUndefined() && !element.IsNull() {
oldHtml := element.Get("innerHTML").String()
element.Set("innerHTML", oldHtml+html)
js.Global().Call("scanElementsSize")
}
}
func (bridge *wasmBridge) updateCSSProperty(htmlID, property, value string) {
if ProtocolInDebugLog {
DebugLog(fmt.Sprintf("%s.style[%s] = '%s'", htmlID, property, value))
}
element := js.Global().Get("document").Call("getElementById", htmlID)
if !element.IsUndefined() && !element.IsNull() {
element.Get("style").Set(property, value)
js.Global().Call("scanElementsSize")
}
}
func (bridge *wasmBridge) updateProperty(htmlID, property string, value any) {
if ProtocolInDebugLog {
DebugLog(fmt.Sprintf("%s.%s = '%v'", htmlID, property, value))
}
element := js.Global().Get("document").Call("getElementById", htmlID)
if !element.IsUndefined() && !element.IsNull() {
element.Set(property, value)
js.Global().Call("scanElementsSize")
}
}
func (bridge *wasmBridge) removeProperty(htmlID, property string) {
bridge.callFunc("removeProperty", htmlID, property)
}
func (bridge *wasmBridge) close() {
}
func (bridge *wasmBridge) readMessage() (string, bool) {
return "", false
}
func (bridge *wasmBridge) writeMessage(script string) bool {
if ProtocolInDebugLog {
DebugLog("Run script:")
DebugLog(script)
}
window := js.Global().Get("window")
window.Call("execScript", script)
return true
}
func (bridge *wasmBridge) cavnasStart(htmlID string) {
if ProtocolInDebugLog {
DebugLog("const ctx = document.getElementById('" + htmlID + "'elementId').getContext('2d');\nctx.save();")
}
bridge.canvas = js.Global().Call("getCanvasContext", htmlID)
if !bridge.canvas.IsNull() {
bridge.canvas.Call("save")
}
}
func (bridge *wasmBridge) callCanvasFunc(funcName string, args ...any) {
if !bridge.canvas.IsNull() {
for i, arg := range args {
if array, ok := arg.([]float64); ok {
arr := make([]any, len(array))
for k, x := range array {
arr[k] = x
}
args[i] = js.ValueOf(arr)
}
}
bridge.printFuncToLog("ctx."+funcName, args...)
bridge.canvas.Call(funcName, args...)
}
}
func (bridge *wasmBridge) callCanvasVarFunc(v any, funcName string, args ...any) {
if jsVar, ok := v.(js.Value); ok && !jsVar.IsNull() {
bridge.printFuncToLog(jsVar.String()+"."+funcName, args...)
jsVar.Call(funcName, args...)
}
}
func (bridge *wasmBridge) callCanvasImageFunc(url string, property string, funcName string, args ...any) {
image := js.Global().Get("images").Call("get", url)
if !image.IsUndefined() && !image.IsNull() && !bridge.canvas.IsNull() {
args = append([]any{image}, args...)
result := bridge.canvas.Call(funcName, args...)
if property != "" {
bridge.printFuncToLog("ctx."+property+" = ctx."+funcName, args...)
bridge.canvas.Set(property, result)
} else {
bridge.printFuncToLog("ctx."+funcName, args...)
}
}
}
func (bridge *wasmBridge) createCanvasVar(funcName string, args ...any) any {
if bridge.canvas.IsNull() {
return bridge.canvas
}
result := bridge.canvas.Call(funcName, args...)
bridge.printFuncToLog("var "+result.String()+" = ctx."+funcName, args...)
return result
}
func (bridge *wasmBridge) updateCanvasProperty(property string, value any) {
if !bridge.canvas.IsNull() {
if ProtocolInDebugLog {
DebugLog(fmt.Sprintf("ctx.%s = '%v'", property, value))
}
bridge.canvas.Set(property, value)
}
}
func (bridge *wasmBridge) cavnasFinish() {
if !bridge.canvas.IsNull() {
DebugLog("ctx.restore()")
bridge.canvas.Call("restore")
}
}
func (bridge *wasmBridge) canvasTextMetrics(htmlID, font, text string) TextMetrics {
result := TextMetrics{}
canvas := js.Global().Get("document").Call("getElementById", htmlID)
if !canvas.IsUndefined() && !canvas.IsNull() {
context := canvas.Call("getContext", "2d")
if !context.IsUndefined() && !context.IsNull() {
context.Call("save")
context.Set("font", font)
context.Set("textBaseline", "alphabetic")
context.Set("textAlign", "start")
metrics := context.Call("measureText", text)
if !metrics.IsUndefined() && !metrics.IsNull() {
metricsValue := func(name string) float64 {
value := metrics.Get(name)
if !value.IsUndefined() && !value.IsNull() && value.Type() == js.TypeNumber {
return value.Float()
}
return 0
}
result.Width = metricsValue("width")
result.Ascent = metricsValue("actualBoundingBoxAscent")
result.Descent = metricsValue("actualBoundingBoxDescent")
result.Left = metricsValue("actualBoundingBoxLeft")
result.Right = metricsValue("actualBoundingBoxRight")
}
context.Call("restore")
}
}
return result
}
func (bridge *wasmBridge) htmlPropertyValue(htmlID, name string) string {
element := js.Global().Get("document").Call("getElementById", htmlID)
if !element.IsUndefined() && !element.IsNull() {
return element.Get(name).String()
}
return ""
}
func (bridge *wasmBridge) answerReceived(answer DataObject) {
if text, ok := answer.PropertyValue("answerID"); ok {
if id, err := strconv.Atoi(text); err == nil {
if chanel, ok := bridge.answer[id]; ok {
chanel <- answer
delete(bridge.answer, id)
} else {
ErrorLog("Bad answerID = " + text + " (chan not found)")
}
} else {
ErrorLog("Invalid answerID = " + text)
}
} else {
ErrorLog("answerID not found")
}
}
func (bridge *wasmBridge) remoteAddr() string {
return "localhost"
}