5
0
mirror of https://github.com/wailsapp/wails.git synced 2025-05-02 04:59:38 +08:00

Feature: AssetServer Runtime (#2335)

* Tidy up runtime JS

* Initial implementation of runtime over http

* Update runtime deps. Fix test task.

* Support Clipboard.
Message Processor refactor.

* Add `Window.Screen()`
Clipboard `GetText` -> `Text`

* Support most dialogs
Better JS->Go object mapping
Implement Go->JS callback mechanism
Rename `window.runtime` -> `window.wails` to better reflect the Go API

* Support SaveFile dialog

* Remove go.work

* Tidy up

* Event->CustomEvent to prevent potential clash with native JS Event object
Support Eventing

* Support application calls

* Support logging

* Support named windows
Remove debug info

* Update v3 changes
This commit is contained in:
Lea Anthony 2023-02-06 20:50:11 +11:00 committed by GitHub
parent 88e79f2598
commit 5dbda4aead
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
70 changed files with 10176 additions and 3634 deletions

View File

@ -6,16 +6,19 @@ github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keL
github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I= github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I=
github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/gofuzz v1.0.0 h1:A8PeW59pxE9IoFRqBp37U+mSNaQoZ46F1f0f863XSXw=
github.com/google/renameio/v2 v2.0.0 h1:UifI23ZTGY8Tt29JbYFiuyIU3eX+RNFtUwefq9qAhxg= github.com/google/renameio/v2 v2.0.0 h1:UifI23ZTGY8Tt29JbYFiuyIU3eX+RNFtUwefq9qAhxg=
github.com/jessevdk/go-flags v1.5.0 h1:1jKYvbxEjfUl0fmqTCOfonvskHHXMjBySTLW4y9LFvc= github.com/jessevdk/go-flags v1.5.0 h1:1jKYvbxEjfUl0fmqTCOfonvskHHXMjBySTLW4y9LFvc=
github.com/klauspost/cpuid/v2 v2.2.0/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= github.com/klauspost/cpuid/v2 v2.2.0/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY=
github.com/konsorten/go-windows-terminal-sequences v1.0.1 h1:mweAR1A6xJ3oS2pRaGiHgQ4OO8tzTaLawm8vnODuwDk= github.com/konsorten/go-windows-terminal-sequences v1.0.1 h1:mweAR1A6xJ3oS2pRaGiHgQ4OO8tzTaLawm8vnODuwDk=
github.com/kr/fs v0.1.0 h1:Jskdu9ieNAYnjxsi0LbQp1ulIKZV1LAFgK1tWhpZgl8= github.com/kr/fs v0.1.0 h1:Jskdu9ieNAYnjxsi0LbQp1ulIKZV1LAFgK1tWhpZgl8=
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
github.com/kr/pty v1.1.1 h1:VkoXIwSboBpnk99O/KFauAEILuNHv5DVFKZMBN/gUgw=
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e h1:aoZm08cpOy4WuID//EZDgcC4zIxODThtZNPirFr42+A= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e h1:aoZm08cpOy4WuID//EZDgcC4zIxODThtZNPirFr42+A=
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
github.com/pkg/sftp v1.10.1 h1:VasscCm72135zRysgrJDKsntdmPN+OuU3+nnHYA9wyc= github.com/pkg/sftp v1.10.1 h1:VasscCm72135zRysgrJDKsntdmPN+OuU3+nnHYA9wyc=
github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs=
github.com/sirupsen/logrus v1.4.1 h1:GL2rEmy6nsikmW0r8opw9JIRScdMF5hA8cOYLH7In1k=
github.com/spf13/afero v1.6.0 h1:xoax2sJ2DT8S8xA2paPFjDCScCNeWsg75VG0DLRreiY= github.com/spf13/afero v1.6.0 h1:xoax2sJ2DT8S8xA2paPFjDCScCNeWsg75VG0DLRreiY=
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c= github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c=

View File

@ -14,6 +14,7 @@ import (
const ( const (
runtimeJSPath = "/wails/runtime.js" runtimeJSPath = "/wails/runtime.js"
ipcJSPath = "/wails/ipc.js" ipcJSPath = "/wails/ipc.js"
runtimePath = "/wails/runtime"
) )
type RuntimeAssets interface { type RuntimeAssets interface {
@ -22,6 +23,10 @@ type RuntimeAssets interface {
RuntimeDesktopJS() []byte RuntimeDesktopJS() []byte
} }
type RuntimeHandler interface {
HandleRuntimeCall(w http.ResponseWriter, r *http.Request)
}
type AssetServer struct { type AssetServer struct {
handler http.Handler handler http.Handler
wsHandler http.Handler wsHandler http.Handler
@ -34,6 +39,9 @@ type AssetServer struct {
servingFromDisk bool servingFromDisk bool
appendSpinnerToBody bool appendSpinnerToBody bool
// Use http based runtime
runtimeHandler RuntimeHandler
assetServerWebView assetServerWebView
} }
@ -77,6 +85,10 @@ func NewAssetServerWithHandler(handler http.Handler, bindingsJSON string, servin
return result, nil return result, nil
} }
func (d *AssetServer) UseRuntimeHandler(handler RuntimeHandler) {
d.runtimeHandler = handler
}
func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) { func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
if isWebSocket(req) { if isWebSocket(req) {
// Forward WebSockets to the distinct websocket handler if it exists // Forward WebSockets to the distinct websocket handler if it exists
@ -122,6 +134,13 @@ func (d *AssetServer) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
case runtimeJSPath: case runtimeJSPath:
d.writeBlob(rw, path, d.runtimeJS) d.writeBlob(rw, path, d.runtimeJS)
case runtimePath:
if d.runtimeHandler != nil {
d.runtimeHandler.HandleRuntimeCall(rw, req)
} else {
d.handler.ServeHTTP(rw, req)
}
case ipcJSPath: case ipcJSPath:
content := d.runtime.DesktopIPC() content := d.runtime.DesktopIPC()
if d.ipcJS != nil { if d.ipcJS != nil {

View File

@ -19,10 +19,20 @@ Informal and incomplete list of things needed in v3.
- [ ] Implement alias for `window` in JS - [ ] Implement alias for `window` in JS
- [ ] Implement runtime dispatcher - [ ] Implement runtime dispatcher
- [ ] Log - [ ] Log
- [ ] Same Window - [x] Same Window
- [ ] Other Window - [x] Other Window
- [ ] Dialogs - [x] Dialogs
- [ ] Events - [x] Info
- [x] Warning
- [x] Error
- [x] Question
- [x] OpenFile
- [x] SaveFile
- [x] Events
- [ ] Screens
- [x] Clipboard
- [ ] Application
- [ ] Create `.d.ts` file
## Templates ## Templates

View File

@ -87,32 +87,15 @@ tasks:
cmds: cmds:
- npm install - npm install
build-runtime-dev:
dir: internal/runtime/dev
deps:
- install-runtime-dev-deps
sources:
- ./*.js
generates:
- ../ipc_websocket.js
cmds:
- node build.js
build-runtime-ipc:
dir: internal/runtime
deps:
- install-runtime-dev-deps
sources:
- ./desktop/ipc.js
generates:
- ipc.js
cmds:
- npx esbuild desktop/ipc.js --bundle --minify --outfile=ipc.js
test-runtime: test-runtime:
dir: internal/runtime dir: internal/runtime
cmds: cmds:
- npx vitest - npx vitest run
update-runtime:
dir: internal/runtime
cmds:
- npx npm-check-updates -u
build-runtime-all: build-runtime-all:
dir: internal/runtime dir: internal/runtime
@ -123,10 +106,9 @@ tasks:
- build-runtime-debug-darwin - build-runtime-debug-darwin
- build-runtime-debug-windows - build-runtime-debug-windows
- build-runtime-debug-linux - build-runtime-debug-linux
- build-runtime-dev
- build-runtime-ipc
cmds: cmds:
- task: test-runtime - cmd: echo "build complete"
build-runtime: build-runtime:
dir: internal/runtime dir: internal/runtime

61
v3/V3 Changes.md Normal file
View File

@ -0,0 +1,61 @@
# Changes for v3
## Events
In v3, there are 3 types of events:
- Application Events
- Window Events
- Custom Events
### Application Events
Application events are events that are emitted by the application. These events include native events such as `ApplicationDidFinishLaunching` on macOS.
### Window Events
Window events are events that are emitted by a window. These events include native events such as `WindowDidBecomeMain` on macOS.
### Custom Events
Events that the user defines are called `CustomEvents`. This is to differentiate them from the `Event` object that is used to communicate with the browser. CustomEvents are now objects that encapsulate all the details of an event. This includes the event name, the data, and the source of the event.
The data associated with a CustomEvent is now a single value. If multiple values are required, then a struct can be used.
### Event callbacks and `Emit` function signature
The signatures events callbacks (as used by `On`, `Once` & `OnMultiple`) have changed. In v2, the callback function received optional data. In v3, the callback function receives a `CustomEvent` object that contains all data related to the event.
Similarly, the `Emit` function has changed. Instead of taking a name and optional data, it now takes a single `CustomEvent` object that it will emit.
### `Off` and `OffAll`
In v2, `Off` and `OffAll` calls would remove events in both JS and Go. Due to the multi-window nature of v3, this has been changed so that these methods only apply to the context they are called in. For example, if you call `Off` in a window, it will only remove events for that window. If you use `Off` in Go, it will only remove events for Go.
### Logging
There was a lot of requests for different types of logging in v2 so for v3 we have simplified things to make it as customisable as you want. There is now a single call `Log` that takes a LogMessage object. This object contains the message, the level, and the source, the log message and any data to be printed out. The default logger is the Console logger, however any number of outputs to log to can be added. Simply add custom loggers to the `options.Application.Logger.CustomLoggers` slice. The default logger does not have log level filtering, however custom loggers can be added that do.
### Developer notes
When emitting an event in Go, it will dispatch the event to local Go listeners and also each window in the application.
When emitting an event in JS, it now sends the event to the application. This will be processed as if it was emitted in Go, however the sender ID will be that of the window.
## Window
The Window API has largely remained the same, however there are a few changes to note.
- Windows now have a Name that identifies them. This is used to identify the window when emitting events.
## ClipBoard
The clipboard API has been simplified. There is now a single `Clipboard` object that can be used to read and write to the clipboard. The `Clipboard` object is available in both Go and JS. `SetText()` to set the text and `Text()` to get the text.
## Bindings
TBD
## Dialogs
Dialogs are now available in JavaScript!

View File

@ -0,0 +1,21 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<style>body{ text-align: center; color: white; background-color: rgba(0,0,0,0); user-select: none; -ms-user-select: none; -webkit-user-select: none; }</style>
</head>
<body>
<h1>Events Demo</h1>
<br/>
<div id="results"></div>
</body>
<script>
wails.Events.On("myevent", function(data) {
let currentHTML = document.getElementById("results").innerHTML;
document.getElementById("results").innerHTML = currentHTML + "<br/>" + JSON.stringify(data);;
})
</script>
</html>

View File

@ -0,0 +1,72 @@
package main
import (
"embed"
_ "embed"
"log"
"time"
"github.com/wailsapp/wails/v3/pkg/events"
"github.com/wailsapp/wails/v3/pkg/options"
"github.com/wailsapp/wails/v3/pkg/application"
)
//go:embed assets
var assets embed.FS
func main() {
app := application.New(options.Application{
Name: "Events Demo",
Description: "A demo of the Events API",
Mac: options.Mac{
ApplicationShouldTerminateAfterLastWindowClosed: true,
},
})
app.Events.On("myevent", func(e *application.CustomEvent) {
log.Printf("[Go] CustomEvent received: %+v\n", e)
})
app.On(events.Mac.ApplicationDidFinishLaunching, func() {
for {
log.Println("Sending event")
app.Events.Emit(&application.CustomEvent{
Name: "myevent",
Data: "hello",
})
time.Sleep(10 * time.Second)
}
})
app.NewWebviewWindowWithOptions(&options.WebviewWindow{
Title: "Events Demo",
Assets: options.Assets{
FS: assets,
},
Mac: options.MacWindow{
Backdrop: options.MacBackdropTranslucent,
TitleBar: options.TitleBarHiddenInsetUnified,
InvisibleTitleBarHeight: 50,
},
})
app.NewWebviewWindowWithOptions(&options.WebviewWindow{
Title: "Events Demo",
Assets: options.Assets{
FS: assets,
},
Mac: options.MacWindow{
Backdrop: options.MacBackdropTranslucent,
TitleBar: options.TitleBarHiddenInsetUnified,
InvisibleTitleBarHeight: 50,
},
})
err := app.Run()
if err != nil {
log.Fatal(err.Error())
}
}

View File

@ -5,9 +5,11 @@ go 1.19
require ( require (
github.com/go-task/task/v3 v3.20.0 github.com/go-task/task/v3 v3.20.0
github.com/jackmordaunt/icns/v2 v2.2.1 github.com/jackmordaunt/icns/v2 v2.2.1
github.com/json-iterator/go v1.1.12
github.com/leaanthony/clir v1.6.0 github.com/leaanthony/clir v1.6.0
github.com/leaanthony/gosod v1.0.3 github.com/leaanthony/gosod v1.0.3
github.com/leaanthony/winicon v1.0.0 github.com/leaanthony/winicon v1.0.0
github.com/matryer/is v1.4.0
github.com/pterm/pterm v0.12.51 github.com/pterm/pterm v0.12.51
github.com/samber/lo v1.37.0 github.com/samber/lo v1.37.0
github.com/stretchr/testify v1.8.1 github.com/stretchr/testify v1.8.1
@ -33,6 +35,8 @@ require (
github.com/mattn/go-runewidth v0.0.14 // indirect github.com/mattn/go-runewidth v0.0.14 // indirect
github.com/mattn/go-zglob v0.0.4 // indirect github.com/mattn/go-zglob v0.0.4 // indirect
github.com/mitchellh/hashstructure/v2 v2.0.2 // indirect github.com/mitchellh/hashstructure/v2 v2.0.2 // indirect
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 // indirect
github.com/modern-go/reflect2 v1.0.2 // indirect
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 // indirect github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 // indirect
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e // indirect github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e // indirect
github.com/pmezard/go-difflib v1.0.0 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect
@ -53,3 +57,5 @@ require (
gopkg.in/yaml.v3 v3.0.1 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect
mvdan.cc/sh/v3 v3.6.0 // indirect mvdan.cc/sh/v3 v3.6.0 // indirect
) )
replace github.com/wailsapp/wails/v2 => ../v2

View File

@ -26,6 +26,7 @@ github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg78
github.com/go-task/task/v3 v3.20.0 h1:pTavuhP+AiEpKLzh5I6Lja9Ux7ypYO5QMsEPTbhYEDc= github.com/go-task/task/v3 v3.20.0 h1:pTavuhP+AiEpKLzh5I6Lja9Ux7ypYO5QMsEPTbhYEDc=
github.com/go-task/task/v3 v3.20.0/go.mod h1:y7rWakbLR5gFElGgo6rA2dyr6vU/zNIDVfn3S4Of6OI= github.com/go-task/task/v3 v3.20.0/go.mod h1:y7rWakbLR5gFElGgo6rA2dyr6vU/zNIDVfn3S4Of6OI=
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
github.com/gookit/color v1.4.2/go.mod h1:fqRyamkC1W8uxl+lxCQxOT09l/vYfZ+QeiX3rKQHCoQ= github.com/gookit/color v1.4.2/go.mod h1:fqRyamkC1W8uxl+lxCQxOT09l/vYfZ+QeiX3rKQHCoQ=
github.com/gookit/color v1.5.0/go.mod h1:43aQb+Zerm/BWh2GnrgOQm7ffz7tvQXEKV6BFMl7wAo= github.com/gookit/color v1.5.0/go.mod h1:43aQb+Zerm/BWh2GnrgOQm7ffz7tvQXEKV6BFMl7wAo=
github.com/gookit/color v1.5.2 h1:uLnfXcaFjlrDnQDT+NCBcfhrXqYTx/rcCa6xn01Y8yI= github.com/gookit/color v1.5.2 h1:uLnfXcaFjlrDnQDT+NCBcfhrXqYTx/rcCa6xn01Y8yI=
@ -36,6 +37,8 @@ github.com/jackmordaunt/icns/v2 v2.2.1 h1:MGklwYP2yohKn2Bw7XxlF69LZe98S1vUfl5OvA
github.com/jackmordaunt/icns/v2 v2.2.1/go.mod h1:6aYIB9eSzyfHHMKqDf17Xrs1zetQPReAkiUSHzdw4cI= github.com/jackmordaunt/icns/v2 v2.2.1/go.mod h1:6aYIB9eSzyfHHMKqDf17Xrs1zetQPReAkiUSHzdw4cI=
github.com/joho/godotenv v1.4.0 h1:3l4+N6zfMWnkbPEXKng2o2/MR5mSwTrBih4ZEkkz1lg= github.com/joho/godotenv v1.4.0 h1:3l4+N6zfMWnkbPEXKng2o2/MR5mSwTrBih4ZEkkz1lg=
github.com/joho/godotenv v1.4.0/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= github.com/joho/godotenv v1.4.0/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
github.com/klauspost/cpuid/v2 v2.0.10/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= github.com/klauspost/cpuid/v2 v2.0.10/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c=
github.com/klauspost/cpuid/v2 v2.0.12/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= github.com/klauspost/cpuid/v2 v2.0.12/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c=
@ -74,6 +77,10 @@ github.com/mattn/go-zglob v0.0.4 h1:LQi2iOm0/fGgu80AioIJ/1j9w9Oh+9DZ39J4VAGzHQM=
github.com/mattn/go-zglob v0.0.4/go.mod h1:MxxjyoXXnMxfIpxTK2GAkw1w8glPsQILx3N5wrKakiY= github.com/mattn/go-zglob v0.0.4/go.mod h1:MxxjyoXXnMxfIpxTK2GAkw1w8glPsQILx3N5wrKakiY=
github.com/mitchellh/hashstructure/v2 v2.0.2 h1:vGKWl0YJqUNxE8d+h8f6NJLcCJrgbhC4NcD46KavDd4= github.com/mitchellh/hashstructure/v2 v2.0.2 h1:vGKWl0YJqUNxE8d+h8f6NJLcCJrgbhC4NcD46KavDd4=
github.com/mitchellh/hashstructure/v2 v2.0.2/go.mod h1:MG3aRVU/N29oo/V/IhBX8GR/zz4kQkprJgF2EVszyDE= github.com/mitchellh/hashstructure/v2 v2.0.2/go.mod h1:MG3aRVU/N29oo/V/IhBX8GR/zz4kQkprJgF2EVszyDE=
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc=
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ= github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ=
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8= github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8=
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs=
@ -107,6 +114,7 @@ github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
@ -119,8 +127,6 @@ github.com/tc-hib/winres v0.1.6 h1:qgsYHze+BxQPEYilxIz/KCQGaClvI2+yLBAZs+3+0B8=
github.com/tc-hib/winres v0.1.6/go.mod h1:pe6dOR40VOrGz8PkzreVKNvEKnlE8t4yR8A8naL+t7A= github.com/tc-hib/winres v0.1.6/go.mod h1:pe6dOR40VOrGz8PkzreVKNvEKnlE8t4yR8A8naL+t7A=
github.com/wailsapp/mimetype v1.4.1 h1:pQN9ycO7uo4vsUUuPeHEYoUkLVkaRntMnHJxVwYhwHs= github.com/wailsapp/mimetype v1.4.1 h1:pQN9ycO7uo4vsUUuPeHEYoUkLVkaRntMnHJxVwYhwHs=
github.com/wailsapp/mimetype v1.4.1/go.mod h1:9aV5k31bBOv5z6u+QP8TltzvNGJPmNJD4XlAL3U+j3o= github.com/wailsapp/mimetype v1.4.1/go.mod h1:9aV5k31bBOv5z6u+QP8TltzvNGJPmNJD4XlAL3U+j3o=
github.com/wailsapp/wails/v2 v2.3.2-0.20230117193915-45c3a501d9e6 h1:Wn+nhnS+VytzE0PegUzSh4T3hXJCtggKGD/4U5H9+wQ=
github.com/wailsapp/wails/v2 v2.3.2-0.20230117193915-45c3a501d9e6/go.mod h1:zlNLI0E2c2qA6miiuAHtp0Bac8FaGH0tlhA19OssR/8=
github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778 h1:QldyIu/L63oPpyvQmHgvgickp1Yw510KJOqX7H24mg8= github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778 h1:QldyIu/L63oPpyvQmHgvgickp1Yw510KJOqX7H24mg8=
github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778/go.mod h1:2MuV+tbUrU1zIOPMxZ5EncGwgmMJsa+9ucAQZXxsObs= github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778/go.mod h1:2MuV+tbUrU1zIOPMxZ5EncGwgmMJsa+9ucAQZXxsObs=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=

View File

@ -37,6 +37,6 @@ func GenerateBindings(options *GenerateBindingsOptions) error {
if err != nil { if err != nil {
return fmt.Errorf("error writing models file: %v", err) return fmt.Errorf("error writing models file: %v", err)
} }
println("Generated models file '" + options.ModelsFilename + "'")
return nil return nil
} }

View File

@ -3,22 +3,19 @@
package runtime package runtime
var RuntimeAssetsBundle = &RuntimeAssets{ var RuntimeAssetsBundle = &RuntimeAssets{
desktopIPC: DesktopIPC,
runtimeDesktopJS: DesktopRuntime, runtimeDesktopJS: DesktopRuntime,
} }
type RuntimeAssets struct { type RuntimeAssets struct {
desktopIPC []byte
websocketIPC []byte
runtimeDesktopJS []byte runtimeDesktopJS []byte
} }
func (r *RuntimeAssets) DesktopIPC() []byte { func (r *RuntimeAssets) DesktopIPC() []byte {
return r.desktopIPC return []byte("")
} }
func (r *RuntimeAssets) WebsocketIPC() []byte { func (r *RuntimeAssets) WebsocketIPC() []byte {
return r.websocketIPC return []byte("")
} }
func (r *RuntimeAssets) RuntimeDesktopJS() []byte { func (r *RuntimeAssets) RuntimeDesktopJS() []byte {

View File

@ -3,23 +3,19 @@
package runtime package runtime
var RuntimeAssetsBundle = &RuntimeAssets{ var RuntimeAssetsBundle = &RuntimeAssets{
desktopIPC: DesktopIPC,
websocketIPC: WebsocketIPC,
runtimeDesktopJS: DesktopRuntime, runtimeDesktopJS: DesktopRuntime,
} }
type RuntimeAssets struct { type RuntimeAssets struct {
desktopIPC []byte
websocketIPC []byte
runtimeDesktopJS []byte runtimeDesktopJS []byte
} }
func (r *RuntimeAssets) DesktopIPC() []byte { func (r *RuntimeAssets) DesktopIPC() []byte {
return r.desktopIPC return []byte("")
} }
func (r *RuntimeAssets) WebsocketIPC() []byte { func (r *RuntimeAssets) WebsocketIPC() []byte {
return r.websocketIPC return []byte("")
} }
func (r *RuntimeAssets) RuntimeDesktopJS() []byte { func (r *RuntimeAssets) RuntimeDesktopJS() []byte {

View File

@ -0,0 +1,3 @@
# README
After updating any files in this directory, you must run `wails task build-runtime` to regenerate the compiled JS.

View File

@ -0,0 +1,27 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 9 */
import {newRuntimeCaller} from "./runtime";
let call = newRuntimeCaller("application");
export function Hide() {
return call("Hide");
}
export function Show() {
return call("Show");
}
export function Quit() {
return call("Quit");
}

View File

@ -1,67 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 6 */
import {Call} from './calls';
// This is where we bind go method wrappers
window.go = {};
export function SetBindings(bindingsMap) {
try {
bindingsMap = JSON.parse(bindingsMap);
} catch (e) {
console.error(e);
}
// Initialise the bindings map
window.go = window.go || {};
// Iterate package names
Object.keys(bindingsMap).forEach((packageName) => {
// Create inner map if it doesn't exist
window.go[packageName] = window.go[packageName] || {};
// Iterate struct names
Object.keys(bindingsMap[packageName]).forEach((structName) => {
// Create inner map if it doesn't exist
window.go[packageName][structName] = window.go[packageName][structName] || {};
Object.keys(bindingsMap[packageName][structName]).forEach((methodName) => {
window.go[packageName][structName][methodName] = function () {
// No timeout by default
let timeout = 0;
// Actual function
function dynamic() {
const args = [].slice.call(arguments);
return Call([packageName, structName, methodName].join('.'), args, timeout);
}
// Allow setting timeout to function
dynamic.setTimeout = function (newTimeout) {
timeout = newTimeout;
};
// Allow getting timeout to function
dynamic.getTimeout = function () {
return timeout;
};
return dynamic;
}();
});
});
});
}

View File

@ -1,8 +0,0 @@
/**
* @description: Use the system default browser to open the url
* @param {string} url
* @return {void}
*/
export function BrowserOpenURL(url) {
window.WailsInvoke('BO:' + url);
}

View File

@ -1,189 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 6 */
export const callbacks = {};
/**
* Returns a number from the native browser random function
*
* @returns number
*/
function cryptoRandom() {
var array = new Uint32Array(1);
return window.crypto.getRandomValues(array)[0];
}
/**
* Returns a number using da old-skool Math.Random
* I likes to call it LOLRandom
*
* @returns number
*/
function basicRandom() {
return Math.random() * 9007199254740991;
}
// Pick a random number function based on browser capability
var randomFunc;
if (window.crypto) {
randomFunc = cryptoRandom;
} else {
randomFunc = basicRandom;
}
/**
* Call sends a message to the backend to call the binding with the
* given data. A promise is returned and will be completed when the
* backend responds. This will be resolved when the call was successful
* or rejected if an error is passed back.
* There is a timeout mechanism. If the call doesn't respond in the given
* time (in milliseconds) then the promise is rejected.
*
* @export
* @param {string} name
* @param {any=} args
* @param {number=} timeout
* @returns
*/
export function Call(name, args, timeout) {
// Timeout infinite by default
if (timeout == null) {
timeout = 0;
}
let windowID = window.wails.window.ID();
// Create a promise
return new Promise(function (resolve, reject) {
// Create a unique callbackID
var callbackID;
do {
callbackID = name + '-' + randomFunc();
} while (callbacks[callbackID]);
var timeoutHandle;
// Set timeout
if (timeout > 0) {
timeoutHandle = setTimeout(function () {
reject(Error('Call to ' + name + ' timed out. Request ID: ' + callbackID));
}, timeout);
}
// Store callback
callbacks[callbackID] = {
timeoutHandle: timeoutHandle,
reject: reject,
resolve: resolve
};
try {
const payload = {
name,
args,
callbackID,
windowID,
};
// Make the call
window.WailsInvoke('C' + JSON.stringify(payload));
} catch (e) {
// eslint-disable-next-line
console.error(e);
}
});
}
window.ObfuscatedCall = (id, args, timeout) => {
// Timeout infinite by default
if (timeout == null) {
timeout = 0;
}
// Create a promise
return new Promise(function (resolve, reject) {
// Create a unique callbackID
var callbackID;
do {
callbackID = id + '-' + randomFunc();
} while (callbacks[callbackID]);
var timeoutHandle;
// Set timeout
if (timeout > 0) {
timeoutHandle = setTimeout(function () {
reject(Error('Call to method ' + id + ' timed out. Request ID: ' + callbackID));
}, timeout);
}
// Store callback
callbacks[callbackID] = {
timeoutHandle: timeoutHandle,
reject: reject,
resolve: resolve
};
try {
const payload = {
id,
args,
callbackID,
windowID: window.wails.window.ID(),
};
// Make the call
window.WailsInvoke('c' + JSON.stringify(payload));
} catch (e) {
// eslint-disable-next-line
console.error(e);
}
});
};
/**
* Called by the backend to return data to a previously called
* binding invocation
*
* @export
* @param {string} incomingMessage
*/
export function Callback(incomingMessage) {
// Parse the message
let message;
try {
message = JSON.parse(incomingMessage);
} catch (e) {
const error = `Invalid JSON passed to callback: ${e.message}. Message: ${incomingMessage}`;
runtime.LogDebug(error);
throw new Error(error);
}
let callbackID = message.callbackid;
let callbackData = callbacks[callbackID];
if (!callbackData) {
const error = `Callback '${callbackID}' not registered!!!`;
console.error(error); // eslint-disable-line
throw new Error(error);
}
clearTimeout(callbackData.timeoutHandle);
delete callbacks[callbackID];
if (message.error) {
callbackData.reject(message.error);
} else {
callbackData.resolve(message.result);
}
}

View File

@ -0,0 +1,23 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 9 */
import {newRuntimeCaller} from "./runtime";
let call = newRuntimeCaller("clipboard");
export function SetText(text) {
return call("SetText", {text});
}
export function Text() {
return call("Text");
}

View File

@ -0,0 +1,85 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 9 */
import {newRuntimeCaller} from "./runtime";
import { nanoid } from 'nanoid/non-secure'
let call = newRuntimeCaller("dialog");
let dialogResponses = new Map();
function generateID() {
let result;
do {
result = nanoid();
} while (dialogResponses.has(result));
return result;
}
export function dialogCallback(id, data, isJSON) {
let p = dialogResponses.get(id);
if (p) {
if (isJSON) {
p.resolve(JSON.parse(data));
} else {
p.resolve(data);
}
dialogResponses.delete(id);
}
}
export function dialogErrorCallback(id, message) {
let p = dialogResponses.get(id);
if (p) {
p.reject(message);
dialogResponses.delete(id);
}
}
function dialog(type, options) {
return new Promise((resolve, reject) => {
let id = generateID();
options = options || {};
options["dialog-id"] = id;
dialogResponses.set(id, {resolve, reject});
call(type, options).catch((error) => {
reject(error);
dialogResponses.delete(id);
})
});
}
export function Info(options) {
return dialog("Info", options);
}
export function Warning(options) {
return dialog("Warning", options);
}
export function Error(options) {
return dialog("Error", options);
}
export function Question(options) {
return dialog("Question", options);
}
export function OpenFile(options) {
return dialog("OpenFile", options);
}
export function SaveFile(options) {
return dialog("SaveFile", options);
}

View File

@ -1,15 +1,18 @@
/* /*
_ __ _ __ _ __ _ __
| | / /___ _(_) /____ | | / /___ _(_) /____
| | /| / / __ `/ / / ___/ | | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ ) | |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/ |__/|__/\__,_/_/_/____/
The electron alternative for Go The electron alternative for Go
(c) Lea Anthony 2019-present (c) Lea Anthony 2019-present
*/ */
/* jshint esversion: 6 */
// Defines a single listener with a maximum number of times to callback /* jshint esversion: 9 */
import {newRuntimeCaller} from "./runtime";
let call = newRuntimeCaller("events");
/** /**
* The Listener class defines a listener! :-) * The Listener class defines a listener! :-)
@ -31,7 +34,7 @@ class Listener {
// Callback invokes the callback with the given data // Callback invokes the callback with the given data
// Returns true if this listener should be destroyed // Returns true if this listener should be destroyed
this.Callback = (data) => { this.Callback = (data) => {
callback.apply(null, data); callback(data);
// If maxCallbacks is infinite, return false (do not destroy) // If maxCallbacks is infinite, return false (do not destroy)
if (this.maxCallbacks === -1) { if (this.maxCallbacks === -1) {
return false; return false;
@ -43,21 +46,41 @@ class Listener {
} }
} }
export const eventListeners = {};
/**
* CustomEvent defines a custom event. It is passed to event listeners.
*
* @class CustomEvent
*/
export class CustomEvent {
/**
* Creates an instance of CustomEvent.
* @param {string} name - Name of the event
* @param {any} data - Data associated with the event
* @memberof CustomEvent
*/
constructor(name, data) {
this.name = name;
this.data = data;
}
}
export const eventListeners = new Map();
/** /**
* Registers an event listener that will be invoked `maxCallbacks` times before being destroyed * Registers an event listener that will be invoked `maxCallbacks` times before being destroyed
* *
* @export * @export
* @param {string} eventName * @param {string} eventName
* @param {function} callback * @param {function(CustomEvent): void} callback
* @param {number} maxCallbacks * @param {number} maxCallbacks
* @returns {function} A function to cancel the listener * @returns {function} A function to cancel the listener
*/ */
export function EventsOnMultiple(eventName, callback, maxCallbacks) { export function OnMultiple(eventName, callback, maxCallbacks) {
eventListeners[eventName] = eventListeners[eventName] || []; let listeners = eventListeners.get(eventName) || [];
const thisListener = new Listener(eventName, callback, maxCallbacks); const thisListener = new Listener(eventName, callback, maxCallbacks);
eventListeners[eventName].push(thisListener); listeners.push(thisListener);
eventListeners.set(eventName, listeners);
return () => listenerOff(thisListener); return () => listenerOff(thisListener);
} }
@ -66,11 +89,11 @@ export function EventsOnMultiple(eventName, callback, maxCallbacks) {
* *
* @export * @export
* @param {string} eventName * @param {string} eventName
* @param {function} callback * @param {function(CustomEvent): void} callback
* @returns {function} A function to cancel the listener * @returns {function} A function to cancel the listener
*/ */
export function EventsOn(eventName, callback) { export function On(eventName, callback) {
return EventsOnMultiple(eventName, callback, -1); return OnMultiple(eventName, callback, -1);
} }
/** /**
@ -78,135 +101,87 @@ export function EventsOn(eventName, callback) {
* *
* @export * @export
* @param {string} eventName * @param {string} eventName
* @param {function} callback * @param {function(CustomEvent): void} callback
* @returns {function} A function to cancel the listener * @returns {function} A function to cancel the listener
*/ */
export function EventsOnce(eventName, callback) { export function Once(eventName, callback) {
return EventsOnMultiple(eventName, callback, 1); return OnMultiple(eventName, callback, 1);
} }
function notifyListeners(eventData) { /**
* listenerOff unregisters a listener previously registered with On
*
* @param {Listener} listener
*/
function listenerOff(listener) {
const eventName = listener.eventName;
// Remove local listener
let listeners = eventListeners.get(eventName).filter(l => l !== listener);
if (listeners.length === 0) {
eventListeners.delete(eventName);
} else {
eventListeners.set(eventName, listeners);
}
}
// Get the event name /**
let eventName = eventData.name; * dispatches an event to all listeners
*
// Check if we have any listeners for this event * @export
if (eventListeners[eventName]) { * @param {CustomEvent} event
*/
// Keep a list of listener indexes to destroy export function dispatchCustomEvent(event) {
const newEventListenerList = eventListeners[eventName].slice(); console.log("dispatching event: ", {event});
let listeners = eventListeners.get(event.name);
// Iterate listeners if (listeners) {
for (let count = 0; count < eventListeners[eventName].length; count += 1) { // iterate listeners and call callback. If callback returns true, remove listener
let toRemove = [];
// Get next listener listeners.forEach(listener => {
const listener = eventListeners[eventName][count]; let remove = listener.Callback(event)
if (remove) {
let data = eventData.data; toRemove.push(listener);
}
// Do the callback });
const destroy = listener.Callback(data); // remove listeners
if (destroy) { if (toRemove.length > 0) {
// if the listener indicated to destroy itself, add it to the destroy list listeners = listeners.filter(l => !toRemove.includes(l));
newEventListenerList.splice(count, 1); if (listeners.length === 0) {
eventListeners.delete(event.name);
} else {
eventListeners.set(event.name, listeners);
} }
} }
// Update callbacks with new list of listeners
if (newEventListenerList.length === 0) {
removeListener(eventName);
} else {
eventListeners[eventName] = newEventListenerList;
}
} }
} }
/**
* Notify informs frontend listeners that an event was emitted with the given data
*
* @export
* @param {string} notifyMessage - encoded notification message
*/
export function EventsNotify(notifyMessage) {
// Parse the message
let message;
try {
message = JSON.parse(notifyMessage);
} catch (e) {
const error = 'Invalid JSON passed to Notify: ' + notifyMessage;
throw new Error(error);
}
notifyListeners(message);
}
/**
* Emit an event with the given name and data
*
* @export
* @param {string} eventName
*/
export function EventsEmit(eventName) {
const payload = {
name: eventName,
data: [].slice.apply(arguments).slice(1),
};
// Notify JS listeners
notifyListeners(payload);
// Notify Go listeners
window.WailsInvoke('EE' + JSON.stringify(payload));
}
function removeListener(eventName) {
// Remove local listeners
delete eventListeners[eventName];
// Notify Go listeners
window.WailsInvoke('EX' + eventName);
}
/** /**
* Off unregisters a listener previously registered with On, * Off unregisters a listener previously registered with On,
* optionally multiple listeneres can be unregistered via `additionalEventNames` * optionally multiple listeners can be unregistered via `additionalEventNames`
*
[v3 CHANGE] Off only unregisters listeners within the current window
* *
* @param {string} eventName * @param {string} eventName
* @param {...string} additionalEventNames * @param {...string} additionalEventNames
*/ */
export function EventsOff(eventName, ...additionalEventNames) { export function Off(eventName, ...additionalEventNames) {
removeListener(eventName) let eventsToRemove = [eventName, ...additionalEventNames];
eventsToRemove.forEach(eventName => {
if (additionalEventNames.length > 0) { eventListeners.delete(eventName);
additionalEventNames.forEach(eventName => { })
removeListener(eventName)
})
}
} }
/** /**
* Off unregisters all event listeners previously registered with On * OffAll unregisters all listeners
*/ * [v3 CHANGE] OffAll only unregisters listeners within the current window
export function EventsOffAll() {
const eventNames = Object.keys(eventListeners);
for (let i = 0; i !== eventNames.length; i++) {
removeListener(eventNames[i]);
}
}
/**
* listenerOff unregisters a listener previously registered with EventsOn
* *
* @param {Listener} listener
*/ */
function listenerOff(listener) { export function OffAll() {
const eventName = listener.eventName; eventListeners.clear();
// Remove local listener
eventListeners[eventName] = eventListeners[eventName].filter(l => l !== listener);
// Clean up if there are no event listeners left
if (eventListeners[eventName].length === 0) {
removeListener(eventName);
}
} }
/*
Emit emits an event to all listeners
*/
export function Emit(event) {
return call("Emit", event);
}

View File

@ -1,132 +1,115 @@
import { EventsOnMultiple, EventsNotify, eventListeners, EventsOn, EventsEmit, EventsOffAll, EventsOnce, EventsOff } from './events' import { On, Off, OffAll, OnMultiple, CustomEvent, dispatchCustomEvent, eventListeners, Once } from './events'
import { expect, describe, it, beforeAll, vi, afterEach, beforeEach } from 'vitest' import { expect, describe, it, vi, afterEach, beforeEach } from 'vitest'
// Edit an assertion and save to see HMR in action
beforeAll(() => {
window.WailsInvoke = vi.fn(() => {})
})
afterEach(() => { afterEach(() => {
EventsOffAll(); OffAll();
vi.resetAllMocks() vi.resetAllMocks()
}) })
describe('EventsOnMultiple', () => { describe('OnMultiple', () => {
let testEvent = new CustomEvent('a', {})
it('should stop after a specified number of times', () => { it('should stop after a specified number of times', () => {
const cb = vi.fn() const cb = vi.fn()
EventsOnMultiple('a', cb, 5) OnMultiple('a', cb, 5)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
expect(cb).toBeCalledTimes(5); expect(cb).toBeCalledTimes(5);
expect(window.WailsInvoke).toBeCalledTimes(1);
expect(window.WailsInvoke).toHaveBeenLastCalledWith('EXa');
}) })
it('should return a cancel fn', () => { it('should return a cancel fn', () => {
const cb = vi.fn() const cb = vi.fn()
const cancel = EventsOnMultiple('a', cb, 5) const cancel = OnMultiple('a', cb, 5)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
cancel() cancel()
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
EventsNotify(JSON.stringify({name: 'a', data: {}})) dispatchCustomEvent(testEvent)
expect(cb).toBeCalledTimes(2) expect(cb).toBeCalledTimes(2)
expect(window.WailsInvoke).toBeCalledTimes(1);
expect(window.WailsInvoke).toHaveBeenLastCalledWith('EXa');
}) })
}) })
describe('EventsOn', () => { describe('On', () => {
it('should create a listener with a count of -1', () => { it('should create a listener with a count of -1', () => {
EventsOn('a', () => {}) On('a', () => {})
expect(eventListeners['a'][0].maxCallbacks).toBe(-1) expect(eventListeners.get("a")[0].maxCallbacks).toBe(-1)
}) })
it('should return a cancel fn', () => { it('should return a cancel fn', () => {
const cancel = EventsOn('a', () => {}) const cancel = On('a', () => {})
cancel(); cancel();
expect(window.WailsInvoke).toBeCalledTimes(1);
expect(window.WailsInvoke).toHaveBeenLastCalledWith('EXa');
}) })
}) })
describe('EventsOnce', () => { describe('Once', () => {
it('should create a listener with a count of 1', () => { it('should create a listener with a count of 1', () => {
EventsOnce('a', () => {}) Once('a', () => {})
expect(eventListeners['a'][0].maxCallbacks).toBe(1) expect(eventListeners.get("a")[0].maxCallbacks).toBe(1)
}) })
it('should return a cancel fn', () => { it('should return a cancel fn', () => {
const cancel = EventsOn('a', () => {}) const cancel = EventsOn('a', () => {})
cancel(); cancel();
expect(window.WailsInvoke).toBeCalledTimes(1);
expect(window.WailsInvoke).toHaveBeenLastCalledWith('EXa');
}) })
}) })
//
describe('EventsNotify', () => { // describe('EventsNotify', () => {
it('should inform a listener', () => { // it('should inform a listener', () => {
const cb = vi.fn() // const cb = vi.fn()
EventsOn('a', cb) // EventsOn('a', cb)
EventsNotify(JSON.stringify({name: 'a', data: ["one", "two", "three"]})) // EventsNotify(JSON.stringify({name: 'a', data: ["one", "two", "three"]}))
expect(cb).toBeCalledTimes(1); // expect(cb).toBeCalledTimes(1);
expect(cb).toHaveBeenLastCalledWith("one", "two", "three"); // expect(cb).toHaveBeenLastCalledWith("one", "two", "three");
expect(window.WailsInvoke).toBeCalledTimes(0); // expect(window.WailsInvoke).toBeCalledTimes(0);
}) // })
}) // })
//
describe('EventsEmit', () => { // describe('EventsEmit', () => {
it('should emit an event', () => { // it('should emit an event', () => {
EventsEmit('a', 'one', 'two', 'three') // EventsEmit('a', 'one', 'two', 'three')
expect(window.WailsInvoke).toBeCalledTimes(1); // expect(window.WailsInvoke).toBeCalledTimes(1);
const calledWith = window.WailsInvoke.calls[0][0]; // const calledWith = window.WailsInvoke.calls[0][0];
expect(calledWith.slice(0, 2)).toBe('EE') // expect(calledWith.slice(0, 2)).toBe('EE')
expect(JSON.parse(calledWith.slice(2))).toStrictEqual({data: ["one", "two", "three"], name: "a"}) // expect(JSON.parse(calledWith.slice(2))).toStrictEqual({data: ["one", "two", "three"], name: "a"})
}) // })
}) // })
//
describe('EventsOff', () => { describe('Off', () => {
beforeEach(() => { beforeEach(() => {
EventsOn('a', () => {}) On('a', () => {})
EventsOn('a', () => {}) On('a', () => {})
EventsOn('a', () => {}) On('a', () => {})
EventsOn('b', () => {}) On('b', () => {})
EventsOn('c', () => {}) On('c', () => {})
}) })
it('should cancel all event listeners for a single type', () => { it('should cancel all event listeners for a single type', () => {
EventsOff('a') Off('a')
expect(eventListeners['a']).toBeUndefined() expect(eventListeners.get('a')).toBeUndefined()
expect(eventListeners['b']).not.toBeUndefined() expect(eventListeners.get('b')).not.toBeUndefined()
expect(eventListeners['c']).not.toBeUndefined() expect(eventListeners.get('c')).not.toBeUndefined()
expect(window.WailsInvoke).toBeCalledTimes(1);
expect(window.WailsInvoke).toHaveBeenLastCalledWith('EXa');
}) })
it('should cancel all event listeners for multiple types', () => { it('should cancel all event listeners for multiple types', () => {
EventsOff('a', 'b') Off('a', 'b')
expect(eventListeners['a']).toBeUndefined() expect(eventListeners.get('a')).toBeUndefined()
expect(eventListeners['b']).toBeUndefined() expect(eventListeners.get('b')).toBeUndefined()
expect(eventListeners['c']).not.toBeUndefined() expect(eventListeners.get('c')).not.toBeUndefined()
expect(window.WailsInvoke).toBeCalledTimes(2);
expect(window.WailsInvoke.calls).toStrictEqual([['EXa'], ['EXb']]);
}) })
}) })
describe('EventsOffAll', () => { describe('OffAll', () => {
it('should cancel all event listeners', () => { it('should cancel all event listeners', () => {
EventsOn('a', () => {}) On('a', () => {})
EventsOn('a', () => {}) On('a', () => {})
EventsOn('a', () => {}) On('a', () => {})
EventsOn('b', () => {}) On('b', () => {})
EventsOn('c', () => {}) On('c', () => {})
EventsOffAll() OffAll()
expect(eventListeners).toStrictEqual({}) expect(eventListeners.size).toBe(0)
expect(window.WailsInvoke).toBeCalledTimes(3);
expect(window.WailsInvoke.calls).toStrictEqual([['EXa'], ['EXb'], ['EXc']]);
}) })
}) })

View File

@ -1,45 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 9 */
let postMessage = null;
(function () {
// Credit: https://stackoverflow.com/a/2631521
let _deeptest = function (s) {
var obj = window[s.shift()];
while (obj && s.length) obj = obj[s.shift()];
return obj;
};
let windows = _deeptest(["chrome", "webview", "postMessage"]);
let mac_linux = _deeptest(["webkit", "messageHandlers", "external", "postMessage"]);
if (!windows && !mac_linux) {
console.error("Unsupported Platform");
return;
}
if (windows) {
postMessage = (message) => window.chrome.webview.postMessage(message);
}
if (mac_linux) {
postMessage = (message) => window.webkit.messageHandlers.external.postMessage(message);
}
})();
export function invoke(message, id) {
if( id && id !== -1) {
postMessage("WINDOWID:"+ id + ":" + message);
} else {
postMessage(message);
}
}

View File

@ -1,6 +1,6 @@
/* /*
_ __ _ __ _ __ _ __
| | / /___ _(_) /____ | | / /___ _(_) /____
| | /| / / __ `/ / / ___/ | | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ ) | |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/ |__/|__/\__,_/_/_/____/
@ -8,106 +8,16 @@ The electron alternative for Go
(c) Lea Anthony 2019-present (c) Lea Anthony 2019-present
*/ */
/* jshint esversion: 6 */ /* jshint esversion: 9 */
import {newRuntimeCaller} from "./runtime";
let call = newRuntimeCaller("log");
/** /**
* Sends a log message to the backend with the given level + message * Logs a message.
* * @param {message} Message to log
* @param {string} level
* @param {string} message
*/ */
function sendLogMessage(level, message) { export function Log(message) {
return call("Log", message);
// Log Message format:
// l[type][message]
window.WailsInvoke('L' + level + message);
} }
/**
* Log the given trace message with the backend
*
* @export
* @param {string} message
*/
export function LogTrace(message) {
sendLogMessage('T', message);
}
/**
* Log the given message with the backend
*
* @export
* @param {string} message
*/
export function LogPrint(message) {
sendLogMessage('P', message);
}
/**
* Log the given debug message with the backend
*
* @export
* @param {string} message
*/
export function LogDebug(message) {
sendLogMessage('D', message);
}
/**
* Log the given info message with the backend
*
* @export
* @param {string} message
*/
export function LogInfo(message) {
sendLogMessage('I', message);
}
/**
* Log the given warning message with the backend
*
* @export
* @param {string} message
*/
export function LogWarning(message) {
sendLogMessage('W', message);
}
/**
* Log the given error message with the backend
*
* @export
* @param {string} message
*/
export function LogError(message) {
sendLogMessage('E', message);
}
/**
* Log the given fatal message with the backend
*
* @export
* @param {string} message
*/
export function LogFatal(message) {
sendLogMessage('F', message);
}
/**
* Sets the Log level to the given log level
*
* @export
* @param {number} loglevel
*/
export function SetLogLevel(loglevel) {
sendLogMessage('S', loglevel);
}
// Log levels
export const LogLevel = {
TRACE: 1,
DEBUG: 2,
INFO: 3,
WARNING: 4,
ERROR: 5,
};

View File

@ -9,48 +9,56 @@ The electron alternative for Go
*/ */
/* jshint esversion: 9 */ /* jshint esversion: 9 */
import {invoke} from "./ipc.js"; import {dialogCallback, dialogErrorCallback, Error, Info, OpenFile, Question, SaveFile, Warning,} from "./dialogs";
import {Callback, callbacks} from './calls';
import {EventsNotify, eventListeners} from "./events"; import * as Clipboard from './clipboard';
import {SetBindings} from "./bindings"; import * as Application from './application';
import * as Log from './log';
import {newWindow} from "./window"; import {newWindow} from "./window";
import {dispatchCustomEvent, Emit, Off, OffAll, On, Once, OnMultiple} from "./events";
// export function Environment() {
// return Call(":wails:Environment");
// }
// Internal wails endpoints // Internal wails endpoints
window.wails = { window.wails = {
Callback, ...newRuntime(-1),
callbacks,
EventsNotify,
eventListeners,
SetBindings,
}; };
window._wails = {
dialogCallback,
dialogErrorCallback,
dispatchCustomEvent,
}
export function newRuntime(id) { export function newRuntime(id) {
return { return {
// Log: newLog(id), Clipboard: {
// Browser: newBrowser(id), ...Clipboard
// Screen: newScreen(id), },
// Events: newEvents(id), Application: {
...Application
},
Log,
Dialog: {
Info,
Warning,
Error,
Question,
OpenFile,
SaveFile,
},
Events: {
Emit,
On,
Once,
OnMultiple,
Off,
OffAll,
},
Window: newWindow(id), Window: newWindow(id),
Show: () => invoke("S"),
Hide: () => invoke("H"),
Quit: () => invoke("Q"),
// GetWindow: function (windowID) {
// if (!windowID) {
// return this.Window;
// }
// return newWindow(windowID);
// }
} }
} }
window.runtime = newRuntime(-1);
if (DEBUG) { if (DEBUG) {
console.log("Wails v3.0.0 Debug Mode Enabled"); console.log("Wails v3.0.0 Debug Mode Enabled");
} }

View File

@ -0,0 +1,50 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 9 */
const runtimeURL = window.location.origin + "/wails/runtime";
function runtimeCall(method, args) {
let url = new URL(runtimeURL);
url.searchParams.append("method", method);
if(args) {
url.searchParams.append("args", JSON.stringify(args));
}
return new Promise((resolve, reject) => {
fetch(url)
.then(response => {
if (response.ok) {
// check content type
if (response.headers.get("content-type") && response.headers.get("content-type").indexOf("application/json") !== -1) {
return response.json();
} else {
return response.text();
}
}
reject(Error(response.statusText));
})
.then(data => resolve(data))
.catch(error => reject(error));
});
}
export function newRuntimeCaller(object, id) {
if (!id || id === -1) {
return function (method, args) {
return runtimeCall(object + "." + method, args);
};
}
return function (method, args) {
args = args || {};
args["windowID"] = id;
return runtimeCall(object + "." + method, args);
}
}

View File

@ -1,25 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 9 */
import {Call} from "./calls";
/**
* Gets the all screens. Call this anew each time you want to refresh data from the underlying windowing system.
* @export
* @typedef {import('../wrapper/runtime').Screen} Screen
* @return {Promise<{Screen[]}>} The screens
*/
export function ScreenGetAll() {
return Call(":wails:ScreenGetAll");
}

View File

@ -10,58 +10,39 @@ The electron alternative for Go
/* jshint esversion: 9 */ /* jshint esversion: 9 */
import {newRuntimeCaller} from "./runtime";
import {Call} from "./calls";
import {invoke} from "./ipc";
export function newWindow(id) { export function newWindow(id) {
let call = newRuntimeCaller("window", id);
return { return {
// Reload: () => invoke('WR', id), // Reload: () => call('WR'),
// ReloadApp: () => invoke('WR', id), // ReloadApp: () => call('WR'),
// SetSystemDefaultTheme: () => invoke('WASDT', id), // SetSystemDefaultTheme: () => call('WASDT'),
// SetLightTheme: () => invoke('WALT', id), // SetLightTheme: () => call('WALT'),
// SetDarkTheme: () => invoke('WADT', id), // SetDarkTheme: () => call('WADT'),
Center: () => invoke('Wc', id), // IsFullscreen: () => call('WIF'),
SetTitle: (title) => invoke('WT' + title, id), // IsMaximized: () => call('WIM'),
Fullscreen: () => invoke('WF', id), // IsMinimized: () => call('WIMN'),
UnFullscreen: () => invoke('Wf', id), // IsWindowed: () => call('WIF'),
SetSize: (width, height) => invoke('WS' + width + ',' + height, id), Center: () => call('Center'),
GetSize: () => { SetTitle: (title) => call('SetTitle', {title}),
return Call(":wails:WindowGetSize") Fullscreen: () => call('Fullscreen'),
}, UnFullscreen: () => call('UnFullscreen'),
SetMaxSize: (width, height) => invoke('WZ:' + width + ':' + height, id), SetSize: (width, height) => call('SetSize', {width,height}),
SetMinSize: (width, height) => invoke('Wz:' + width + ':' + height, id), Size: () => { return call('Size') },
SetAlwaysOnTop: (b) => invoke('WATP:' + (b ? '1' : '0'), id), SetMaxSize: (width, height) => call('SetMaxSize', {width,height}),
SetPosition: (x, y) => invoke('Wp:' + x + ':' + y, id), SetMinSize: (width, height) => call('SetMinSize', {width,height}),
GetPosition: () => { SetAlwaysOnTop: (b) => call('SetAlwaysOnTop', {alwaysOnTop:b}),
return Call(":wails:WindowGetPos") SetPosition: (x, y) => call('SetPosition', {x,y}),
}, Position: () => { return call('Position') },
Hide: () => invoke('WH', id), Screen: () => { return call('Screen') },
Maximise: () => invoke('WM', id), Hide: () => call('Hide'),
Show: () => invoke('WS', id), Maximise: () => call('Maximise'),
ToggleMaximise: () => invoke('Wt', id), Show: () => call('Show'),
UnMaximise: () => invoke('WU', id), ToggleMaximise: () => call('ToggleMaximise'),
Minimise: () => invoke('Wm', id), UnMaximise: () => call('UnMaximise'),
UnMinimise: () => invoke('Wu', id), Minimise: () => call('Minimise'),
SetBackgroundColour: (R, G, B, A) => UnMinimise: () => call('UnMinimise'),
invoke('Wr:' + JSON.stringify({ SetBackgroundColour: (r, g, b, a) => call('SetBackgroundColour', {r, g, b, a}),
r: R || 0,
g: G || 0,
b: B || 0,
a: A || 255}, id)
),
} }
} }
// export function IsFullscreen: ()=> // return Call(":wails:WindowIsFullscreen"),
//
// export function IsMaximised: ()=> // return Call(":wails:WindowIsMaximised"),
//
// export function IsMinimised: ()=> // return Call(":wails:WindowIsMinimised"),
//
// export function IsNormal: ()=> // return Call(":wails:WindowIsNormal"),
//

View File

@ -1,54 +0,0 @@
<script>
import {overlayVisible} from './store'
import {fade,} from 'svelte/transition';
</script>
{#if $overlayVisible }
<div class="wails-reconnect-overlay" transition:fade="{{ duration: 300 }}">
<div class="wails-reconnect-overlay-content">
<div class="wails-reconnect-overlay-loadingspinner"></div>
</div>
</div>
{/if}
<style>
.wails-reconnect-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
backdrop-filter: blur(2px) saturate(0%) contrast(50%) brightness(25%);
z-index: 999999
}
.wails-reconnect-overlay-content {
position: relative;
top: 50%;
transform: translateY(-50%);
margin: 0;
background-image: url();
background-repeat: no-repeat;
background-position: center
}
.wails-reconnect-overlay-loadingspinner {
pointer-events: none;
width: 2.5em;
height: 2.5em;
border: .4em solid transparent;
border-color: #f00 #eee0 #f00 #eee0;
border-radius: 50%;
animation: loadingspin 1s linear infinite;
margin: auto;
padding: 2.5em
}
@keyframes loadingspin {
100% {
transform: rotate(360deg);
}
}
</style>

View File

@ -1,15 +0,0 @@
/* jshint esversion: 8 */
const esbuild = require("esbuild");
const sveltePlugin = require("esbuild-svelte");
esbuild
.build({
entryPoints: ["main.js"],
bundle: true,
minify: true,
outfile: "../ipc_websocket.js",
plugins: [sveltePlugin({compileOptions: {css: true}})],
logLevel: "info",
sourcemap: "inline",
})
.catch(() => process.exit(1));

View File

@ -1,8 +0,0 @@
export function log(message) {
// eslint-disable-next-line
console.log(
'%c wails dev %c ' + message + ' ',
'background: #aa0000; color: #fff; border-radius: 3px 0px 0px 3px; padding: 1px; font-size: 0.7rem',
'background: #009900; color: #fff; border-radius: 0px 3px 3px 0px; padding: 1px; font-size: 0.7rem'
);
}

View File

@ -1,125 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
/* jshint esversion: 6 */
import {log} from "./log";
import Overlay from "./Overlay.svelte";
import {hideOverlay, showOverlay} from "./store";
let components = {};
let wailsInvokeInternal = null;
let messageQueue = [];
window.WailsInvoke = (message) => {
if (!wailsInvokeInternal) {
console.log("Queueing: " + message);
messageQueue.push(message);
return;
}
wailsInvokeInternal(message);
};
window.addEventListener('DOMContentLoaded', () => {
components.overlay = new Overlay({
target: document.body,
anchor: document.querySelector('#wails-spinner'),
});
});
let websocket = null;
let connectTimer;
window.onbeforeunload = function () {
if (websocket) {
websocket.onclose = function () {
};
websocket.close();
websocket = null;
}
};
// ...and attempt to connect
connect();
function setupIPCBridge() {
wailsInvokeInternal = (message) => {
websocket.send(message);
};
for (let i = 0; i < messageQueue.length; i++) {
console.log("sending queued message: " + messageQueue[i]);
window.WailsInvoke(messageQueue[i]);
}
messageQueue = [];
}
// Handles incoming websocket connections
function handleConnect() {
log('Connected to backend');
hideOverlay();
setupIPCBridge();
clearInterval(connectTimer);
websocket.onclose = handleDisconnect;
websocket.onmessage = handleMessage;
}
// Handles websocket disconnects
function handleDisconnect() {
log('Disconnected from backend');
websocket = null;
showOverlay();
connect();
}
function _connect() {
if (websocket == null) {
websocket = new WebSocket('ws://' + window.location.host + '/wails/ipc');
websocket.onopen = handleConnect;
websocket.onerror = function (e) {
e.stopImmediatePropagation();
e.stopPropagation();
e.preventDefault();
websocket = null;
return false;
};
}
}
// Try to connect to the backend every .5s
function connect() {
_connect();
connectTimer = setInterval(_connect, 500);
}
function handleMessage(message) {
if (message.data === "reload") {
window.runtime.WindowReload();
return;
}
if (message.data === "reloadapp") {
window.runtime.WindowReloadApp()
return;
}
// As a bridge we ignore js and css injections
switch (message.data[0]) {
// Notifications
case 'n':
window.wails.EventsNotify(message.data.slice(1));
break;
case 'c':
const callbackData = message.data.slice(1);
window.wails.Callback(callbackData);
break;
default:
log('Unknown message: ' + message.data);
}
}

File diff suppressed because it is too large Load Diff

View File

@ -1,18 +0,0 @@
{
"name": "dev",
"version": "3.0.0",
"description": "Wails JS Dev",
"main": "main.js",
"scripts": {
"build": "run-p build:*",
"build:dev": "node build.js"
},
"author": "Lea Anthony <lea.anthony@gmail.com>",
"license": "ISC",
"devDependencies": {
"esbuild": "^0.12.17",
"esbuild-svelte": "^0.5.6",
"npm-run-all": "^4.1.5",
"svelte": "^3.49.0"
}
}

View File

@ -1,12 +0,0 @@
import {writable} from 'svelte/store';
/** Overlay */
export const overlayVisible = writable(false);
export function showOverlay() {
overlayVisible.set(true);
}
export function hideOverlay() {
overlayVisible.set(false);
}

View File

@ -1,9 +0,0 @@
package runtime
import _ "embed"
//go:embed ipc_websocket.js
var WebsocketIPC []byte
//go:embed ipc.js
var DesktopIPC []byte

View File

@ -1 +0,0 @@
(()=>{var o=null;(function(){let s=function(e){for(var n=window[e.shift()];n&&e.length;)n=n[e.shift()];return n},t=s(["chrome","webview","postMessage"]),i=s(["webkit","messageHandlers","external","postMessage"]);if(!t&&!i){console.error("Unsupported Platform");return}t&&(o=e=>window.chrome.webview.postMessage(e)),i&&(o=e=>window.webkit.messageHandlers.external.postMessage(e))})();function r(s,t){o(t&&t!==-1?"WINDOWID:"+t+":"+s:s)}})();

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

View File

@ -7,8 +7,10 @@
"author": "Lea Anthony <lea.anthony@gmail.com>", "author": "Lea Anthony <lea.anthony@gmail.com>",
"license": "ISC", "license": "ISC",
"devDependencies": { "devDependencies": {
"esbuild": "^0.15.6", "esbuild": "^0.17.5",
"happy-dom": "^8.1.3", "happy-dom": "^8.1.5",
"vitest": "^0.24.3" "nanoid": "^4.0.0",
"npm-check-updates": "^16.6.3",
"vitest": "^0.28.3"
} }
} }

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1 +1 @@
(()=>{var u=null;(function(){let e=function(n){for(var r=window[n.shift()];r&&n.length;)r=r[n.shift()];return r},t=e(["chrome","webview","postMessage"]),o=e(["webkit","messageHandlers","external","postMessage"]);if(!t&&!o){console.error("Unsupported Platform");return}t&&(u=n=>window.chrome.webview.postMessage(n)),o&&(u=n=>window.webkit.messageHandlers.external.postMessage(n))})();function i(e,t){u(t&&t!==-1?"WINDOWID:"+t+":"+e:e)}var c={};function y(){var e=new Uint32Array(1);return window.crypto.getRandomValues(e)[0]}function W(){return Math.random()*9007199254740991}var d;window.crypto?d=y:d=W;function f(e,t,o){o==null&&(o=0);let n=window.wails.window.ID();return new Promise(function(r,l){var s;do s=e+"-"+d();while(c[s]);var w;o>0&&(w=setTimeout(function(){l(Error("Call to "+e+" timed out. Request ID: "+s))},o)),c[s]={timeoutHandle:w,reject:l,resolve:r};try{let m={name:e,args:t,callbackID:s,windowID:n};window.WailsInvoke("C"+JSON.stringify(m))}catch(m){console.error(m)}})}window.ObfuscatedCall=(e,t,o)=>(o==null&&(o=0),new Promise(function(n,r){var l;do l=e+"-"+d();while(c[l]);var s;o>0&&(s=setTimeout(function(){r(Error("Call to method "+e+" timed out. Request ID: "+l))},o)),c[l]={timeoutHandle:s,reject:r,resolve:n};try{let w={id:e,args:t,callbackID:l,windowID:window.wails.window.ID()};window.WailsInvoke("c"+JSON.stringify(w))}catch(w){console.error(w)}}));function p(e){let t;try{t=JSON.parse(e)}catch(r){let l=`Invalid JSON passed to callback: ${r.message}. Message: ${e}`;throw runtime.LogDebug(l),new Error(l)}let o=t.callbackid,n=c[o];if(!n){let r=`Callback '${o}' not registered!!!`;throw console.error(r),new Error(r)}clearTimeout(n.timeoutHandle),delete c[o],t.error?n.reject(t.error):n.resolve(t.result)}var a={};function b(e){let t=e.name;if(a[t]){let o=a[t].slice();for(let n=0;n<a[t].length;n+=1){let r=a[t][n],l=e.data;r.Callback(l)&&o.splice(n,1)}o.length===0?S(t):a[t]=o}}function h(e){let t;try{t=JSON.parse(e)}catch{let n="Invalid JSON passed to Notify: "+e;throw new Error(n)}b(t)}function S(e){delete a[e],window.WailsInvoke("EX"+e)}window.go={};function v(e){try{e=JSON.parse(e)}catch(t){console.error(t)}window.go=window.go||{},Object.keys(e).forEach(t=>{window.go[t]=window.go[t]||{},Object.keys(e[t]).forEach(o=>{window.go[t][o]=window.go[t][o]||{},Object.keys(e[t][o]).forEach(n=>{window.go[t][o][n]=function(){let r=0;function l(){let s=[].slice.call(arguments);return f([t,o,n].join("."),s,r)}return l.setTimeout=function(s){r=s},l.getTimeout=function(){return r},l}()})})})}function g(e){return{Center:()=>i("Wc",e),SetTitle:t=>i("WT"+t,e),Fullscreen:()=>i("WF",e),UnFullscreen:()=>i("Wf",e),SetSize:(t,o)=>i("WS"+t+","+o,e),GetSize:()=>f(":wails:WindowGetSize"),SetMaxSize:(t,o)=>i("WZ:"+t+":"+o,e),SetMinSize:(t,o)=>i("Wz:"+t+":"+o,e),SetAlwaysOnTop:t=>i("WATP:"+(t?"1":"0"),e),SetPosition:(t,o)=>i("Wp:"+t+":"+o,e),GetPosition:()=>f(":wails:WindowGetPos"),Hide:()=>i("WH",e),Maximise:()=>i("WM",e),Show:()=>i("WS",e),ToggleMaximise:()=>i("Wt",e),UnMaximise:()=>i("WU",e),Minimise:()=>i("Wm",e),UnMinimise:()=>i("Wu",e),SetBackgroundColour:(t,o,n,r)=>i("Wr:"+JSON.stringify({r:t||0,g:o||0,b:n||0,a:r||255},e))}}window.wails={Callback:p,callbacks:c,EventsNotify:h,eventListeners:a,SetBindings:v};function x(e){return{Window:g(e),Show:()=>i("S"),Hide:()=>i("H"),Quit:()=>i("Q")}}window.runtime=x(-1);console.log("Wails v3.0.0 Debug Mode Enabled");})(); (()=>{var y=Object.defineProperty;var f=(t,e)=>{for(var n in e)y(t,n,{get:e[n],enumerable:!0})};var L=window.location.origin+"/wails/runtime";function h(t,e){let n=new URL(L);return n.searchParams.append("method",t),e&&n.searchParams.append("args",JSON.stringify(e)),new Promise((i,r)=>{fetch(n).then(o=>{if(o.ok)return o.headers.get("content-type")&&o.headers.get("content-type").indexOf("application/json")!==-1?o.json():o.text();r(Error(o.statusText))}).then(o=>i(o)).catch(o=>r(o))})}function l(t,e){return!e||e===-1?function(n,i){return h(t+"."+n,i)}:function(n,i){return i=i||{},i.windowID=e,h(t+"."+n,i)}}var D="useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict";var g=(t=21)=>{let e="",n=t;for(;n--;)e+=D[Math.random()*64|0];return e};var I=l("dialog"),u=new Map;function Q(){let t;do t=g();while(u.has(t));return t}function S(t,e,n){let i=u.get(t);i&&(n?i.resolve(JSON.parse(e)):i.resolve(e),u.delete(t))}function C(t,e){let n=u.get(t);n&&(n.reject(e),u.delete(t))}function c(t,e){return new Promise((n,i)=>{let r=Q();e=e||{},e["dialog-id"]=r,u.set(r,{resolve:n,reject:i}),I(t,e).catch(o=>{i(o),u.delete(r)})})}function M(t){return c("Info",t)}function O(t){return c("Warning",t)}function b(t){return c("Error",t)}function v(t){return c("Question",t)}function E(t){return c("OpenFile",t)}function R(t){return c("SaveFile",t)}var m={};f(m,{SetText:()=>H,Text:()=>B});var T=l("clipboard");function H(t){return T("SetText",{text:t})}function B(){return T("Text")}var d={};f(d,{Hide:()=>N,Quit:()=>G,Show:()=>J});var p=l("application");function N(){return p("Hide")}function J(){return p("Show")}function G(){return p("Quit")}var x={};f(x,{Log:()=>j});var _=l("log");function j(t){return _("Log",t)}function k(t){let e=l("window",t);return{Center:()=>e("Center"),SetTitle:n=>e("SetTitle",{title:n}),Fullscreen:()=>e("Fullscreen"),UnFullscreen:()=>e("UnFullscreen"),SetSize:(n,i)=>e("SetSize",{width:n,height:i}),Size:()=>e("Size"),SetMaxSize:(n,i)=>e("SetMaxSize",{width:n,height:i}),SetMinSize:(n,i)=>e("SetMinSize",{width:n,height:i}),SetAlwaysOnTop:n=>e("SetAlwaysOnTop",{alwaysOnTop:n}),SetPosition:(n,i)=>e("SetPosition",{x:n,y:i}),Position:()=>e("Position"),Screen:()=>e("Screen"),Hide:()=>e("Hide"),Maximise:()=>e("Maximise"),Show:()=>e("Show"),ToggleMaximise:()=>e("ToggleMaximise"),UnMaximise:()=>e("UnMaximise"),Minimise:()=>e("Minimise"),UnMinimise:()=>e("UnMinimise"),SetBackgroundColour:(n,i,r,o)=>e("SetBackgroundColour",{r:n,g:i,b:r,a:o})}}var q=l("events"),w=class{constructor(e,n,i){this.eventName=e,this.maxCallbacks=i||-1,this.Callback=r=>(n(r),this.maxCallbacks===-1?!1:(this.maxCallbacks-=1,this.maxCallbacks===0))}};var a=new Map;function s(t,e,n){let i=a.get(t)||[],r=new w(t,e,n);return i.push(r),a.set(t,i),()=>K(r)}function F(t,e){return s(t,e,-1)}function U(t,e){return s(t,e,1)}function K(t){let e=t.eventName,n=a.get(e).filter(i=>i!==t);n.length===0?a.delete(e):a.set(e,n)}function z(t){console.log("dispatching event: ",{event:t});let e=a.get(t.name);if(e){let n=[];e.forEach(i=>{i.Callback(t)&&n.push(i)}),n.length>0&&(e=e.filter(i=>!n.includes(i)),e.length===0?a.delete(t.name):a.set(t.name,e))}}function A(t,...e){[t,...e].forEach(i=>{a.delete(i)})}function P(){a.clear()}function W(t){return q("Emit",t)}window.wails={...V(-1)};window._wails={dialogCallback:S,dialogErrorCallback:C,dispatchCustomEvent:z};function V(t){return{Clipboard:{...m},Application:{...d},Log:x,Dialog:{Info:M,Warning:O,Error:b,Question:v,OpenFile:E,SaveFile:R},Events:{Emit:W,On:F,Once:U,OnMultiple:s,Off:A,OffAll:P},Window:k(t)}}console.log("Wails v3.0.0 Debug Mode Enabled");})();

View File

@ -1 +1 @@
(()=>{var u=null;(function(){let e=function(n){for(var r=window[n.shift()];r&&n.length;)r=r[n.shift()];return r},t=e(["chrome","webview","postMessage"]),o=e(["webkit","messageHandlers","external","postMessage"]);if(!t&&!o){console.error("Unsupported Platform");return}t&&(u=n=>window.chrome.webview.postMessage(n)),o&&(u=n=>window.webkit.messageHandlers.external.postMessage(n))})();function i(e,t){u(t&&t!==-1?"WINDOWID:"+t+":"+e:e)}var c={};function y(){var e=new Uint32Array(1);return window.crypto.getRandomValues(e)[0]}function W(){return Math.random()*9007199254740991}var d;window.crypto?d=y:d=W;function f(e,t,o){o==null&&(o=0);let n=window.wails.window.ID();return new Promise(function(r,l){var s;do s=e+"-"+d();while(c[s]);var w;o>0&&(w=setTimeout(function(){l(Error("Call to "+e+" timed out. Request ID: "+s))},o)),c[s]={timeoutHandle:w,reject:l,resolve:r};try{let m={name:e,args:t,callbackID:s,windowID:n};window.WailsInvoke("C"+JSON.stringify(m))}catch(m){console.error(m)}})}window.ObfuscatedCall=(e,t,o)=>(o==null&&(o=0),new Promise(function(n,r){var l;do l=e+"-"+d();while(c[l]);var s;o>0&&(s=setTimeout(function(){r(Error("Call to method "+e+" timed out. Request ID: "+l))},o)),c[l]={timeoutHandle:s,reject:r,resolve:n};try{let w={id:e,args:t,callbackID:l,windowID:window.wails.window.ID()};window.WailsInvoke("c"+JSON.stringify(w))}catch(w){console.error(w)}}));function p(e){let t;try{t=JSON.parse(e)}catch(r){let l=`Invalid JSON passed to callback: ${r.message}. Message: ${e}`;throw runtime.LogDebug(l),new Error(l)}let o=t.callbackid,n=c[o];if(!n){let r=`Callback '${o}' not registered!!!`;throw console.error(r),new Error(r)}clearTimeout(n.timeoutHandle),delete c[o],t.error?n.reject(t.error):n.resolve(t.result)}var a={};function b(e){let t=e.name;if(a[t]){let o=a[t].slice();for(let n=0;n<a[t].length;n+=1){let r=a[t][n],l=e.data;r.Callback(l)&&o.splice(n,1)}o.length===0?S(t):a[t]=o}}function h(e){let t;try{t=JSON.parse(e)}catch{let n="Invalid JSON passed to Notify: "+e;throw new Error(n)}b(t)}function S(e){delete a[e],window.WailsInvoke("EX"+e)}window.go={};function v(e){try{e=JSON.parse(e)}catch(t){console.error(t)}window.go=window.go||{},Object.keys(e).forEach(t=>{window.go[t]=window.go[t]||{},Object.keys(e[t]).forEach(o=>{window.go[t][o]=window.go[t][o]||{},Object.keys(e[t][o]).forEach(n=>{window.go[t][o][n]=function(){let r=0;function l(){let s=[].slice.call(arguments);return f([t,o,n].join("."),s,r)}return l.setTimeout=function(s){r=s},l.getTimeout=function(){return r},l}()})})})}function g(e){return{Center:()=>i("Wc",e),SetTitle:t=>i("WT"+t,e),Fullscreen:()=>i("WF",e),UnFullscreen:()=>i("Wf",e),SetSize:(t,o)=>i("WS"+t+","+o,e),GetSize:()=>f(":wails:WindowGetSize"),SetMaxSize:(t,o)=>i("WZ:"+t+":"+o,e),SetMinSize:(t,o)=>i("Wz:"+t+":"+o,e),SetAlwaysOnTop:t=>i("WATP:"+(t?"1":"0"),e),SetPosition:(t,o)=>i("Wp:"+t+":"+o,e),GetPosition:()=>f(":wails:WindowGetPos"),Hide:()=>i("WH",e),Maximise:()=>i("WM",e),Show:()=>i("WS",e),ToggleMaximise:()=>i("Wt",e),UnMaximise:()=>i("WU",e),Minimise:()=>i("Wm",e),UnMinimise:()=>i("Wu",e),SetBackgroundColour:(t,o,n,r)=>i("Wr:"+JSON.stringify({r:t||0,g:o||0,b:n||0,a:r||255},e))}}window.wails={Callback:p,callbacks:c,EventsNotify:h,eventListeners:a,SetBindings:v};function x(e){return{Window:g(e),Show:()=>i("S"),Hide:()=>i("H"),Quit:()=>i("Q")}}window.runtime=x(-1);console.log("Wails v3.0.0 Debug Mode Enabled");})(); (()=>{var y=Object.defineProperty;var f=(t,e)=>{for(var n in e)y(t,n,{get:e[n],enumerable:!0})};var L=window.location.origin+"/wails/runtime";function h(t,e){let n=new URL(L);return n.searchParams.append("method",t),e&&n.searchParams.append("args",JSON.stringify(e)),new Promise((i,r)=>{fetch(n).then(o=>{if(o.ok)return o.headers.get("content-type")&&o.headers.get("content-type").indexOf("application/json")!==-1?o.json():o.text();r(Error(o.statusText))}).then(o=>i(o)).catch(o=>r(o))})}function l(t,e){return!e||e===-1?function(n,i){return h(t+"."+n,i)}:function(n,i){return i=i||{},i.windowID=e,h(t+"."+n,i)}}var D="useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict";var g=(t=21)=>{let e="",n=t;for(;n--;)e+=D[Math.random()*64|0];return e};var I=l("dialog"),u=new Map;function Q(){let t;do t=g();while(u.has(t));return t}function S(t,e,n){let i=u.get(t);i&&(n?i.resolve(JSON.parse(e)):i.resolve(e),u.delete(t))}function C(t,e){let n=u.get(t);n&&(n.reject(e),u.delete(t))}function c(t,e){return new Promise((n,i)=>{let r=Q();e=e||{},e["dialog-id"]=r,u.set(r,{resolve:n,reject:i}),I(t,e).catch(o=>{i(o),u.delete(r)})})}function M(t){return c("Info",t)}function O(t){return c("Warning",t)}function b(t){return c("Error",t)}function v(t){return c("Question",t)}function E(t){return c("OpenFile",t)}function R(t){return c("SaveFile",t)}var m={};f(m,{SetText:()=>H,Text:()=>B});var T=l("clipboard");function H(t){return T("SetText",{text:t})}function B(){return T("Text")}var d={};f(d,{Hide:()=>N,Quit:()=>G,Show:()=>J});var p=l("application");function N(){return p("Hide")}function J(){return p("Show")}function G(){return p("Quit")}var x={};f(x,{Log:()=>j});var _=l("log");function j(t){return _("Log",t)}function k(t){let e=l("window",t);return{Center:()=>e("Center"),SetTitle:n=>e("SetTitle",{title:n}),Fullscreen:()=>e("Fullscreen"),UnFullscreen:()=>e("UnFullscreen"),SetSize:(n,i)=>e("SetSize",{width:n,height:i}),Size:()=>e("Size"),SetMaxSize:(n,i)=>e("SetMaxSize",{width:n,height:i}),SetMinSize:(n,i)=>e("SetMinSize",{width:n,height:i}),SetAlwaysOnTop:n=>e("SetAlwaysOnTop",{alwaysOnTop:n}),SetPosition:(n,i)=>e("SetPosition",{x:n,y:i}),Position:()=>e("Position"),Screen:()=>e("Screen"),Hide:()=>e("Hide"),Maximise:()=>e("Maximise"),Show:()=>e("Show"),ToggleMaximise:()=>e("ToggleMaximise"),UnMaximise:()=>e("UnMaximise"),Minimise:()=>e("Minimise"),UnMinimise:()=>e("UnMinimise"),SetBackgroundColour:(n,i,r,o)=>e("SetBackgroundColour",{r:n,g:i,b:r,a:o})}}var q=l("events"),w=class{constructor(e,n,i){this.eventName=e,this.maxCallbacks=i||-1,this.Callback=r=>(n(r),this.maxCallbacks===-1?!1:(this.maxCallbacks-=1,this.maxCallbacks===0))}};var a=new Map;function s(t,e,n){let i=a.get(t)||[],r=new w(t,e,n);return i.push(r),a.set(t,i),()=>K(r)}function F(t,e){return s(t,e,-1)}function U(t,e){return s(t,e,1)}function K(t){let e=t.eventName,n=a.get(e).filter(i=>i!==t);n.length===0?a.delete(e):a.set(e,n)}function z(t){console.log("dispatching event: ",{event:t});let e=a.get(t.name);if(e){let n=[];e.forEach(i=>{i.Callback(t)&&n.push(i)}),n.length>0&&(e=e.filter(i=>!n.includes(i)),e.length===0?a.delete(t.name):a.set(t.name,e))}}function A(t,...e){[t,...e].forEach(i=>{a.delete(i)})}function P(){a.clear()}function W(t){return q("Emit",t)}window.wails={...V(-1)};window._wails={dialogCallback:S,dialogErrorCallback:C,dispatchCustomEvent:z};function V(t){return{Clipboard:{...m},Application:{...d},Log:x,Dialog:{Info:M,Warning:O,Error:b,Question:v,OpenFile:E,SaveFile:R},Events:{Emit:W,On:F,Once:U,OnMultiple:s,Off:A,OffAll:P},Window:k(t)}}console.log("Wails v3.0.0 Debug Mode Enabled");})();

View File

@ -1 +1 @@
(()=>{var u=null;(function(){let e=function(n){for(var r=window[n.shift()];r&&n.length;)r=r[n.shift()];return r},t=e(["chrome","webview","postMessage"]),o=e(["webkit","messageHandlers","external","postMessage"]);if(!t&&!o){console.error("Unsupported Platform");return}t&&(u=n=>window.chrome.webview.postMessage(n)),o&&(u=n=>window.webkit.messageHandlers.external.postMessage(n))})();function i(e,t){u(t&&t!==-1?"WINDOWID:"+t+":"+e:e)}var c={};function y(){var e=new Uint32Array(1);return window.crypto.getRandomValues(e)[0]}function W(){return Math.random()*9007199254740991}var d;window.crypto?d=y:d=W;function f(e,t,o){o==null&&(o=0);let n=window.wails.window.ID();return new Promise(function(r,l){var s;do s=e+"-"+d();while(c[s]);var w;o>0&&(w=setTimeout(function(){l(Error("Call to "+e+" timed out. Request ID: "+s))},o)),c[s]={timeoutHandle:w,reject:l,resolve:r};try{let m={name:e,args:t,callbackID:s,windowID:n};window.WailsInvoke("C"+JSON.stringify(m))}catch(m){console.error(m)}})}window.ObfuscatedCall=(e,t,o)=>(o==null&&(o=0),new Promise(function(n,r){var l;do l=e+"-"+d();while(c[l]);var s;o>0&&(s=setTimeout(function(){r(Error("Call to method "+e+" timed out. Request ID: "+l))},o)),c[l]={timeoutHandle:s,reject:r,resolve:n};try{let w={id:e,args:t,callbackID:l,windowID:window.wails.window.ID()};window.WailsInvoke("c"+JSON.stringify(w))}catch(w){console.error(w)}}));function p(e){let t;try{t=JSON.parse(e)}catch(r){let l=`Invalid JSON passed to callback: ${r.message}. Message: ${e}`;throw runtime.LogDebug(l),new Error(l)}let o=t.callbackid,n=c[o];if(!n){let r=`Callback '${o}' not registered!!!`;throw console.error(r),new Error(r)}clearTimeout(n.timeoutHandle),delete c[o],t.error?n.reject(t.error):n.resolve(t.result)}var a={};function b(e){let t=e.name;if(a[t]){let o=a[t].slice();for(let n=0;n<a[t].length;n+=1){let r=a[t][n],l=e.data;r.Callback(l)&&o.splice(n,1)}o.length===0?S(t):a[t]=o}}function h(e){let t;try{t=JSON.parse(e)}catch{let n="Invalid JSON passed to Notify: "+e;throw new Error(n)}b(t)}function S(e){delete a[e],window.WailsInvoke("EX"+e)}window.go={};function v(e){try{e=JSON.parse(e)}catch(t){console.error(t)}window.go=window.go||{},Object.keys(e).forEach(t=>{window.go[t]=window.go[t]||{},Object.keys(e[t]).forEach(o=>{window.go[t][o]=window.go[t][o]||{},Object.keys(e[t][o]).forEach(n=>{window.go[t][o][n]=function(){let r=0;function l(){let s=[].slice.call(arguments);return f([t,o,n].join("."),s,r)}return l.setTimeout=function(s){r=s},l.getTimeout=function(){return r},l}()})})})}function g(e){return{Center:()=>i("Wc",e),SetTitle:t=>i("WT"+t,e),Fullscreen:()=>i("WF",e),UnFullscreen:()=>i("Wf",e),SetSize:(t,o)=>i("WS"+t+","+o,e),GetSize:()=>f(":wails:WindowGetSize"),SetMaxSize:(t,o)=>i("WZ:"+t+":"+o,e),SetMinSize:(t,o)=>i("Wz:"+t+":"+o,e),SetAlwaysOnTop:t=>i("WATP:"+(t?"1":"0"),e),SetPosition:(t,o)=>i("Wp:"+t+":"+o,e),GetPosition:()=>f(":wails:WindowGetPos"),Hide:()=>i("WH",e),Maximise:()=>i("WM",e),Show:()=>i("WS",e),ToggleMaximise:()=>i("Wt",e),UnMaximise:()=>i("WU",e),Minimise:()=>i("Wm",e),UnMinimise:()=>i("Wu",e),SetBackgroundColour:(t,o,n,r)=>i("Wr:"+JSON.stringify({r:t||0,g:o||0,b:n||0,a:r||255},e))}}window.wails={Callback:p,callbacks:c,EventsNotify:h,eventListeners:a,SetBindings:v};function x(e){return{Window:g(e),Show:()=>i("S"),Hide:()=>i("H"),Quit:()=>i("Q")}}window.runtime=x(-1);console.log("Wails v3.0.0 Debug Mode Enabled");})(); (()=>{var y=Object.defineProperty;var f=(t,e)=>{for(var n in e)y(t,n,{get:e[n],enumerable:!0})};var L=window.location.origin+"/wails/runtime";function h(t,e){let n=new URL(L);return n.searchParams.append("method",t),e&&n.searchParams.append("args",JSON.stringify(e)),new Promise((i,r)=>{fetch(n).then(o=>{if(o.ok)return o.headers.get("content-type")&&o.headers.get("content-type").indexOf("application/json")!==-1?o.json():o.text();r(Error(o.statusText))}).then(o=>i(o)).catch(o=>r(o))})}function l(t,e){return!e||e===-1?function(n,i){return h(t+"."+n,i)}:function(n,i){return i=i||{},i.windowID=e,h(t+"."+n,i)}}var D="useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict";var g=(t=21)=>{let e="",n=t;for(;n--;)e+=D[Math.random()*64|0];return e};var I=l("dialog"),u=new Map;function Q(){let t;do t=g();while(u.has(t));return t}function S(t,e,n){let i=u.get(t);i&&(n?i.resolve(JSON.parse(e)):i.resolve(e),u.delete(t))}function C(t,e){let n=u.get(t);n&&(n.reject(e),u.delete(t))}function c(t,e){return new Promise((n,i)=>{let r=Q();e=e||{},e["dialog-id"]=r,u.set(r,{resolve:n,reject:i}),I(t,e).catch(o=>{i(o),u.delete(r)})})}function M(t){return c("Info",t)}function O(t){return c("Warning",t)}function b(t){return c("Error",t)}function v(t){return c("Question",t)}function E(t){return c("OpenFile",t)}function R(t){return c("SaveFile",t)}var m={};f(m,{SetText:()=>H,Text:()=>B});var T=l("clipboard");function H(t){return T("SetText",{text:t})}function B(){return T("Text")}var d={};f(d,{Hide:()=>N,Quit:()=>G,Show:()=>J});var p=l("application");function N(){return p("Hide")}function J(){return p("Show")}function G(){return p("Quit")}var x={};f(x,{Log:()=>j});var _=l("log");function j(t){return _("Log",t)}function k(t){let e=l("window",t);return{Center:()=>e("Center"),SetTitle:n=>e("SetTitle",{title:n}),Fullscreen:()=>e("Fullscreen"),UnFullscreen:()=>e("UnFullscreen"),SetSize:(n,i)=>e("SetSize",{width:n,height:i}),Size:()=>e("Size"),SetMaxSize:(n,i)=>e("SetMaxSize",{width:n,height:i}),SetMinSize:(n,i)=>e("SetMinSize",{width:n,height:i}),SetAlwaysOnTop:n=>e("SetAlwaysOnTop",{alwaysOnTop:n}),SetPosition:(n,i)=>e("SetPosition",{x:n,y:i}),Position:()=>e("Position"),Screen:()=>e("Screen"),Hide:()=>e("Hide"),Maximise:()=>e("Maximise"),Show:()=>e("Show"),ToggleMaximise:()=>e("ToggleMaximise"),UnMaximise:()=>e("UnMaximise"),Minimise:()=>e("Minimise"),UnMinimise:()=>e("UnMinimise"),SetBackgroundColour:(n,i,r,o)=>e("SetBackgroundColour",{r:n,g:i,b:r,a:o})}}var q=l("events"),w=class{constructor(e,n,i){this.eventName=e,this.maxCallbacks=i||-1,this.Callback=r=>(n(r),this.maxCallbacks===-1?!1:(this.maxCallbacks-=1,this.maxCallbacks===0))}};var a=new Map;function s(t,e,n){let i=a.get(t)||[],r=new w(t,e,n);return i.push(r),a.set(t,i),()=>K(r)}function F(t,e){return s(t,e,-1)}function U(t,e){return s(t,e,1)}function K(t){let e=t.eventName,n=a.get(e).filter(i=>i!==t);n.length===0?a.delete(e):a.set(e,n)}function z(t){console.log("dispatching event: ",{event:t});let e=a.get(t.name);if(e){let n=[];e.forEach(i=>{i.Callback(t)&&n.push(i)}),n.length>0&&(e=e.filter(i=>!n.includes(i)),e.length===0?a.delete(t.name):a.set(t.name,e))}}function A(t,...e){[t,...e].forEach(i=>{a.delete(i)})}function P(){a.clear()}function W(t){return q("Emit",t)}window.wails={...V(-1)};window._wails={dialogCallback:S,dialogErrorCallback:C,dispatchCustomEvent:z};function V(t){return{Clipboard:{...m},Application:{...d},Log:x,Dialog:{Info:M,Warning:O,Error:b,Question:v,OpenFile:E,SaveFile:R},Events:{Emit:W,On:F,Once:U,OnMultiple:s,Off:A,OffAll:P},Window:k(t)}}console.log("Wails v3.0.0 Debug Mode Enabled");})();

View File

@ -3,9 +3,12 @@ package application
import "C" import "C"
import ( import (
"log" "log"
"os"
"runtime" "runtime"
"sync" "sync"
"github.com/wailsapp/wails/v3/pkg/logger"
"github.com/wailsapp/wails/v3/pkg/events" "github.com/wailsapp/wails/v3/pkg/events"
"github.com/wailsapp/wails/v3/pkg/options" "github.com/wailsapp/wails/v3/pkg/options"
@ -29,7 +32,14 @@ func New(appOptions options.Application) *App {
options: appOptions, options: appOptions,
applicationEventListeners: make(map[uint][]func()), applicationEventListeners: make(map[uint][]func()),
systemTrays: make(map[uint]*SystemTray), systemTrays: make(map[uint]*SystemTray),
log: logger.New(appOptions.Logger.CustomLoggers...),
} }
if !appOptions.Logger.Silent {
result.log.AddOutput(&logger.Console{})
}
result.Events = NewCustomEventProcessor(result.dispatchEventToWindows)
globalApplication = result globalApplication = result
return result return result
} }
@ -57,6 +67,8 @@ type platformApp interface {
setIcon(icon []byte) setIcon(icon []byte)
on(id uint) on(id uint)
dispatchOnMainThread(id uint) dispatchOnMainThread(id uint)
hide()
show()
} }
// Messages sent from javascript get routed here // Messages sent from javascript get routed here
@ -80,10 +92,8 @@ type App struct {
applicationEventListenersLock sync.RWMutex applicationEventListenersLock sync.RWMutex
// Windows // Windows
windows map[uint]*WebviewWindow windows map[uint]*WebviewWindow
windowsLock sync.Mutex windowsLock sync.Mutex
windowAliases map[string]uint
windowAliasesLock sync.Mutex
// System Trays // System Trays
systemTrays map[uint]*SystemTray systemTrays map[uint]*SystemTray
@ -104,8 +114,9 @@ type App struct {
// The main application menu // The main application menu
ApplicationMenu *Menu ApplicationMenu *Menu
// About MessageDialog
clipboard *Clipboard clipboard *Clipboard
Events *EventProcessor
log *logger.Logger
} }
func (a *App) getSystemTrayID() uint { func (a *App) getSystemTrayID() uint {
@ -114,19 +125,60 @@ func (a *App) getSystemTrayID() uint {
a.systemTrayID++ a.systemTrayID++
return a.systemTrayID return a.systemTrayID
} }
func (a *App) getWindowForID(id uint) *WebviewWindow {
a.windowsLock.Lock()
defer a.windowsLock.Unlock()
return a.windows[id]
}
func (a *App) On(eventType events.ApplicationEventType, callback func()) { func (a *App) On(eventType events.ApplicationEventType, callback func()) {
eventID := uint(eventType) eventID := uint(eventType)
a.applicationEventListenersLock.Lock() a.applicationEventListenersLock.Lock()
defer a.applicationEventListenersLock.Unlock() defer a.applicationEventListenersLock.Unlock()
a.applicationEventListeners[eventID] = append(a.applicationEventListeners[eventID], callback) a.applicationEventListeners[eventID] = append(a.applicationEventListeners[eventID], callback)
if a.impl != nil { if a.impl != nil {
a.impl.on(eventID) go a.impl.on(eventID)
} }
} }
func (a *App) NewWebviewWindow() *WebviewWindow { func (a *App) NewWebviewWindow() *WebviewWindow {
return a.NewWebviewWindowWithOptions(nil) return a.NewWebviewWindowWithOptions(nil)
} }
func (a *App) info(message string, args ...any) {
a.Log(&logger.Message{
Level: "INFO",
Message: message,
Data: args,
Sender: "Wails",
})
}
func (a *App) fatal(message string, args ...any) {
msg := "************** FATAL **************\n"
msg += message
msg += "***********************************\n"
a.Log(&logger.Message{
Level: "FATAL",
Message: msg,
Data: args,
Sender: "Wails",
})
a.log.Flush()
os.Exit(1)
}
func (a *App) error(message string, args ...any) {
a.Log(&logger.Message{
Level: "ERROR",
Message: message,
Data: args,
Sender: "Wails",
})
}
func (a *App) NewWebviewWindowWithOptions(windowOptions *options.WebviewWindow) *WebviewWindow { func (a *App) NewWebviewWindowWithOptions(windowOptions *options.WebviewWindow) *WebviewWindow {
// Ensure we have sane defaults // Ensure we have sane defaults
if windowOptions == nil { if windowOptions == nil {
@ -142,14 +194,6 @@ func (a *App) NewWebviewWindowWithOptions(windowOptions *options.WebviewWindow)
a.windows[id] = newWindow a.windows[id] = newWindow
a.windowsLock.Unlock() a.windowsLock.Unlock()
if windowOptions.Alias != "" {
if a.windowAliases == nil {
a.windowAliases = make(map[string]uint)
}
a.windowAliasesLock.Lock()
a.windowAliases[windowOptions.Alias] = id
a.windowAliasesLock.Unlock()
}
if a.running { if a.running {
newWindow.run() newWindow.run()
} }
@ -158,7 +202,6 @@ func (a *App) NewWebviewWindowWithOptions(windowOptions *options.WebviewWindow)
} }
func (a *App) NewSystemTray() *SystemTray { func (a *App) NewSystemTray() *SystemTray {
id := a.getSystemTrayID() id := a.getSystemTrayID()
newSystemTray := NewSystemTray(id) newSystemTray := NewSystemTray(id)
a.systemTraysLock.Lock() a.systemTraysLock.Lock()
@ -172,6 +215,7 @@ func (a *App) NewSystemTray() *SystemTray {
} }
func (a *App) Run() error { func (a *App) Run() error {
a.info("Starting application")
a.impl = newPlatformApp(a) a.impl = newPlatformApp(a)
a.running = true a.running = true
@ -191,7 +235,10 @@ func (a *App) Run() error {
for { for {
event := <-webviewRequests event := <-webviewRequests
a.handleWebViewRequest(event) a.handleWebViewRequest(event)
event.request.Release() err := event.request.Release()
if err != nil {
a.error("Failed to release webview request: %s", err.Error())
}
} }
}() }()
go func() { go func() {
@ -221,7 +268,7 @@ func (a *App) Run() error {
// set the application menu // set the application menu
a.impl.setApplicationMenu(a.ApplicationMenu) a.impl.setApplicationMenu(a.ApplicationMenu)
// set the application icon // set the application Icon
a.impl.setIcon(a.options.Icon) a.impl.setIcon(a.options.Icon)
return a.impl.run() return a.impl.run()
@ -382,3 +429,37 @@ func (a *App) dispatchOnMainThread(fn func()) {
// Call platform specific dispatch function // Call platform specific dispatch function
a.impl.dispatchOnMainThread(id) a.impl.dispatchOnMainThread(id)
} }
func (a *App) OpenFileDialogWithOptions(options *OpenFileDialogOptions) *OpenFileDialog {
result := a.OpenFileDialog()
result.SetOptions(options)
return result
}
func (a *App) SaveFileDialogWithOptions(s *SaveFileDialogOptions) *SaveFileDialog {
result := a.SaveFileDialog()
result.SetOptions(s)
return result
}
func (a *App) dispatchEventToWindows(event *CustomEvent) {
for _, window := range a.windows {
window.dispatchCustomEvent(event)
}
}
func (a *App) Hide() {
if a.impl != nil {
a.impl.hide()
}
}
func (a *App) Show() {
if a.impl != nil {
a.impl.show()
}
}
func (a *App) Log(message *logger.Message) {
a.log.Log(message)
}

View File

@ -113,6 +113,16 @@ static void setApplicationIcon(void *icon, int length) {
}); });
} }
// Hide the application
static void hide(void) {
[NSApp hide:nil];
}
// Show the application
static void show(void) {
[NSApp unhide:nil];
}
*/ */
import "C" import "C"
import ( import (
@ -127,6 +137,14 @@ type macosApp struct {
parent *App parent *App
} }
func (m *macosApp) hide() {
C.hide()
}
func (m *macosApp) show() {
C.show()
}
func (m *macosApp) on(eventID uint) { func (m *macosApp) on(eventID uint) {
C.registerListener(C.uint(eventID)) C.registerListener(C.uint(eventID))
} }

View File

@ -46,9 +46,9 @@ const (
) )
type Button struct { type Button struct {
label string Label string
isCancel bool IsCancel bool
isDefault bool IsDefault bool
callback func() callback func()
} }
@ -60,12 +60,16 @@ type messageDialogImpl interface {
show() show()
} }
type MessageDialogOptions struct {
DialogType DialogType
Title string
Message string
Buttons []*Button
Icon []byte
}
type MessageDialog struct { type MessageDialog struct {
dialogType DialogType MessageDialogOptions
title string
message string
buttons []*Button
icon []byte
// platform independent // platform independent
impl messageDialogImpl impl messageDialogImpl
@ -80,13 +84,16 @@ var defaultTitles = map[DialogType]string{
func newMessageDialog(dialogType DialogType) *MessageDialog { func newMessageDialog(dialogType DialogType) *MessageDialog {
return &MessageDialog{ return &MessageDialog{
dialogType: dialogType, MessageDialogOptions: MessageDialogOptions{
title: defaultTitles[dialogType], DialogType: dialogType,
Title: defaultTitles[dialogType],
},
impl: nil,
} }
} }
func (d *MessageDialog) SetTitle(title string) *MessageDialog { func (d *MessageDialog) SetTitle(title string) *MessageDialog {
d.title = title d.Title = title
return d return d
} }
@ -98,36 +105,41 @@ func (d *MessageDialog) Show() {
} }
func (d *MessageDialog) SetIcon(icon []byte) *MessageDialog { func (d *MessageDialog) SetIcon(icon []byte) *MessageDialog {
d.icon = icon d.Icon = icon
return d return d
} }
func (d *MessageDialog) AddButton(s string) *Button { func (d *MessageDialog) AddButton(s string) *Button {
result := &Button{ result := &Button{
label: s, Label: s,
} }
d.buttons = append(d.buttons, result) d.Buttons = append(d.Buttons, result)
return result return result
} }
func (d *MessageDialog) AddButtons(buttons []*Button) *MessageDialog {
d.Buttons = buttons
return d
}
func (d *MessageDialog) SetDefaultButton(button *Button) *MessageDialog { func (d *MessageDialog) SetDefaultButton(button *Button) *MessageDialog {
for _, b := range d.buttons { for _, b := range d.Buttons {
b.isDefault = false b.IsDefault = false
} }
button.isDefault = true button.IsDefault = true
return d return d
} }
func (d *MessageDialog) SetCancelButton(button *Button) *MessageDialog { func (d *MessageDialog) SetCancelButton(button *Button) *MessageDialog {
for _, b := range d.buttons { for _, b := range d.Buttons {
b.isCancel = false b.IsCancel = false
} }
button.isCancel = true button.IsCancel = true
return d return d
} }
func (d *MessageDialog) SetMessage(title string) *MessageDialog { func (d *MessageDialog) SetMessage(message string) *MessageDialog {
d.title = title d.Message = message
return d return d
} }
@ -135,9 +147,28 @@ type openFileDialogImpl interface {
show() ([]string, error) show() ([]string, error)
} }
type fileFilter struct { type FileFilter struct {
displayName string // Filter information EG: "Image Files (*.jpg, *.png)" DisplayName string // Filter information EG: "Image Files (*.jpg, *.png)"
pattern string // semicolon separated list of extensions, EG: "*.jpg;*.png" Pattern string // semicolon separated list of extensions, EG: "*.jpg;*.png"
}
type OpenFileDialogOptions struct {
CanChooseDirectories bool
CanChooseFiles bool
CanCreateDirectories bool
ShowHiddenFiles bool
ResolvesAliases bool
AllowsMultipleSelection bool
HideExtension bool
CanSelectHiddenExtension bool
TreatsFilePackagesAsDirectories bool
AllowsOtherFileTypes bool
Filters []FileFilter
Title string
Message string
ButtonText string
Directory string
} }
type OpenFileDialog struct { type OpenFileDialog struct {
@ -152,7 +183,7 @@ type OpenFileDialog struct {
canSelectHiddenExtension bool canSelectHiddenExtension bool
treatsFilePackagesAsDirectories bool treatsFilePackagesAsDirectories bool
allowsOtherFileTypes bool allowsOtherFileTypes bool
filters []fileFilter filters []FileFilter
title string title string
message string message string
@ -230,9 +261,9 @@ func (d *OpenFileDialog) PromptForSingleSelection() (string, error) {
// AddFilter adds a filter to the dialog. The filter is a display name and a semicolon separated list of extensions. // AddFilter adds a filter to the dialog. The filter is a display name and a semicolon separated list of extensions.
// EG: AddFilter("Image Files", "*.jpg;*.png") // EG: AddFilter("Image Files", "*.jpg;*.png")
func (d *OpenFileDialog) AddFilter(displayName, pattern string) *OpenFileDialog { func (d *OpenFileDialog) AddFilter(displayName, pattern string) *OpenFileDialog {
d.filters = append(d.filters, fileFilter{ d.filters = append(d.filters, FileFilter{
displayName: strings.TrimSpace(displayName), DisplayName: strings.TrimSpace(displayName),
pattern: strings.TrimSpace(pattern), Pattern: strings.TrimSpace(pattern),
}) })
return d return d
} }
@ -265,6 +296,24 @@ func (d *OpenFileDialog) CanSelectHiddenExtension(canSelectHiddenExtension bool)
return d return d
} }
func (d *OpenFileDialog) SetOptions(options *OpenFileDialogOptions) {
d.title = options.Title
d.message = options.Message
d.buttonText = options.ButtonText
d.directory = options.Directory
d.canChooseDirectories = options.CanChooseDirectories
d.canChooseFiles = options.CanChooseFiles
d.canCreateDirectories = options.CanCreateDirectories
d.showHiddenFiles = options.ShowHiddenFiles
d.resolvesAliases = options.ResolvesAliases
d.allowsMultipleSelection = options.AllowsMultipleSelection
d.hideExtension = options.HideExtension
d.canSelectHiddenExtension = options.CanSelectHiddenExtension
d.treatsFilePackagesAsDirectories = options.TreatsFilePackagesAsDirectories
d.allowsOtherFileTypes = options.AllowsOtherFileTypes
d.filters = options.Filters
}
func newOpenFileDialog() *OpenFileDialog { func newOpenFileDialog() *OpenFileDialog {
return &OpenFileDialog{ return &OpenFileDialog{
id: getDialogID(), id: getDialogID(),
@ -282,6 +331,19 @@ func newSaveFileDialog() *SaveFileDialog {
} }
} }
type SaveFileDialogOptions struct {
CanCreateDirectories bool
ShowHiddenFiles bool
CanSelectHiddenExtension bool
AllowOtherFileTypes bool
HideExtension bool
TreatsFilePackagesAsDirectories bool
Message string
Directory string
Filename string
ButtonText string
}
type SaveFileDialog struct { type SaveFileDialog struct {
id uint id uint
canCreateDirectories bool canCreateDirectories bool
@ -304,6 +366,19 @@ type saveFileDialogImpl interface {
show() (string, error) show() (string, error)
} }
func (d *SaveFileDialog) SetOptions(options *SaveFileDialogOptions) {
d.canCreateDirectories = options.CanCreateDirectories
d.showHiddenFiles = options.ShowHiddenFiles
d.canSelectHiddenExtension = options.CanSelectHiddenExtension
d.allowOtherFileTypes = options.AllowOtherFileTypes
d.hideExtension = options.HideExtension
d.treatsFilePackagesAsDirectories = options.TreatsFilePackagesAsDirectories
d.message = options.Message
d.directory = options.Directory
d.filename = options.Filename
d.buttonText = options.ButtonText
}
func (d *SaveFileDialog) CanCreateDirectories(canCreateDirectories bool) *SaveFileDialog { func (d *SaveFileDialog) CanCreateDirectories(canCreateDirectories bool) *SaveFileDialog {
d.canCreateDirectories = canCreateDirectories d.canCreateDirectories = canCreateDirectories
return d return d

View File

@ -61,7 +61,7 @@ static void* createAlert(int alertType, char* title, char *message, void *icon,
NSImage *image = [NSImage imageNamed:NSImageNameInfo]; NSImage *image = [NSImage imageNamed:NSImageNameInfo];
[alert setIcon:image]; [alert setIcon:image];
} }
} }
return alert; return alert;
} }
@ -315,54 +315,54 @@ type macosDialog struct {
func (m *macosDialog) show() { func (m *macosDialog) show() {
globalApplication.dispatchOnMainThread(func() { globalApplication.dispatchOnMainThread(func() {
// Mac can only have 4 buttons on a dialog // Mac can only have 4 Buttons on a dialog
if len(m.dialog.buttons) > 4 { if len(m.dialog.Buttons) > 4 {
m.dialog.buttons = m.dialog.buttons[:4] m.dialog.Buttons = m.dialog.Buttons[:4]
} }
if m.nsDialog != nil { if m.nsDialog != nil {
C.releaseDialog(m.nsDialog) C.releaseDialog(m.nsDialog)
} }
var title *C.char var title *C.char
if m.dialog.title != "" { if m.dialog.Title != "" {
title = C.CString(m.dialog.title) title = C.CString(m.dialog.Title)
} }
var message *C.char var message *C.char
if m.dialog.message != "" { if m.dialog.Message != "" {
message = C.CString(m.dialog.message) message = C.CString(m.dialog.Message)
} }
var iconData unsafe.Pointer var iconData unsafe.Pointer
var iconLength C.int var iconLength C.int
if m.dialog.icon != nil { if m.dialog.Icon != nil {
iconData = unsafe.Pointer(&m.dialog.icon[0]) iconData = unsafe.Pointer(&m.dialog.Icon[0])
iconLength = C.int(len(m.dialog.icon)) iconLength = C.int(len(m.dialog.Icon))
} else { } else {
// if it's an error, use the application icon // if it's an error, use the application Icon
if m.dialog.dialogType == ErrorDialog { if m.dialog.DialogType == ErrorDialog {
iconData = unsafe.Pointer(&globalApplication.options.Icon[0]) iconData = unsafe.Pointer(&globalApplication.options.Icon[0])
iconLength = C.int(len(globalApplication.options.Icon)) iconLength = C.int(len(globalApplication.options.Icon))
} }
} }
alertType, ok := alertTypeMap[m.dialog.dialogType] alertType, ok := alertTypeMap[m.dialog.DialogType]
if !ok { if !ok {
alertType = C.NSAlertStyleInformational alertType = C.NSAlertStyleInformational
} }
m.nsDialog = C.createAlert(alertType, title, message, iconData, iconLength) m.nsDialog = C.createAlert(alertType, title, message, iconData, iconLength)
// Reverse the buttons so that the default is on the right // Reverse the Buttons so that the default is on the right
reversedButtons := make([]*Button, len(m.dialog.buttons)) reversedButtons := make([]*Button, len(m.dialog.Buttons))
var count = 0 var count = 0
for i := len(m.dialog.buttons) - 1; i >= 0; i-- { for i := len(m.dialog.Buttons) - 1; i >= 0; i-- {
button := m.dialog.buttons[i] button := m.dialog.Buttons[i]
C.alertAddButton(m.nsDialog, C.CString(button.label), C.bool(button.isDefault), C.bool(button.isCancel)) C.alertAddButton(m.nsDialog, C.CString(button.Label), C.bool(button.IsDefault), C.bool(button.IsCancel))
reversedButtons[count] = m.dialog.buttons[i] reversedButtons[count] = m.dialog.Buttons[i]
count++ count++
} }
buttonPressed := int(C.dialogRunModal(m.nsDialog)) buttonPressed := int(C.dialogRunModal(m.nsDialog))
if len(m.dialog.buttons) > buttonPressed { if len(m.dialog.Buttons) > buttonPressed {
button := reversedButtons[buttonPressed] button := reversedButtons[buttonPressed]
if button.callback != nil { if button.callback != nil {
button.callback() button.callback()
@ -410,7 +410,7 @@ func (m *macosOpenFileDialog) show() ([]string, error) {
if len(m.dialog.filters) > 0 { if len(m.dialog.filters) > 0 {
var allPatterns []string var allPatterns []string
for _, filter := range m.dialog.filters { for _, filter := range m.dialog.filters {
patternComponents := strings.Split(filter.pattern, ";") patternComponents := strings.Split(filter.Pattern, ";")
for i, component := range patternComponents { for i, component := range patternComponents {
filterPattern := strings.TrimSpace(component) filterPattern := strings.TrimSpace(component)
filterPattern = strings.TrimPrefix(filterPattern, "*.") filterPattern = strings.TrimPrefix(filterPattern, "*.")

View File

@ -1,5 +1,12 @@
package application package application
import (
"encoding/json"
"sync"
"github.com/samber/lo"
)
var applicationEvents = make(chan uint) var applicationEvents = make(chan uint)
type WindowEvent struct { type WindowEvent struct {
@ -10,3 +17,144 @@ type WindowEvent struct {
var windowEvents = make(chan *WindowEvent) var windowEvents = make(chan *WindowEvent)
var menuItemClicked = make(chan uint) var menuItemClicked = make(chan uint)
type CustomEvent struct {
Name string `json:"name"`
Data any `json:"data"`
Sender string `json:"sender"`
}
func (e CustomEvent) ToJSON() string {
marshal, err := json.Marshal(&e)
if err != nil {
// TODO: Fatal error? log?
return ""
}
return string(marshal)
}
// eventListener holds a callback function which is invoked when
// the event listened for is emitted. It has a counter which indicates
// how the total number of events it is interested in. A value of zero
// means it does not expire (default).
type eventListener struct {
callback func(*CustomEvent) // Function to call with emitted event data
counter int // The number of times this callback may be called. -1 = infinite
delete bool // Flag to indicate that this listener should be deleted
}
// EventProcessor handles custom events
type EventProcessor struct {
// Go event listeners
listeners map[string][]*eventListener
notifyLock sync.RWMutex
dispatchEventToWindows func(*CustomEvent)
}
func NewCustomEventProcessor(dispatchEventToWindows func(*CustomEvent)) *EventProcessor {
return &EventProcessor{
listeners: make(map[string][]*eventListener),
dispatchEventToWindows: dispatchEventToWindows,
}
}
// On is the equivalent of Javascript's `addEventListener`
func (e *EventProcessor) On(eventName string, callback func(event *CustomEvent)) func() {
return e.registerListener(eventName, callback, -1)
}
// OnMultiple is the same as `On` but will unregister after `count` events
func (e *EventProcessor) OnMultiple(eventName string, callback func(event *CustomEvent), counter int) func() {
return e.registerListener(eventName, callback, counter)
}
// Once is the same as `On` but will unregister after the first event
func (e *EventProcessor) Once(eventName string, callback func(event *CustomEvent)) func() {
return e.registerListener(eventName, callback, 1)
}
// Emit sends an event to all listeners
func (e *EventProcessor) Emit(thisEvent *CustomEvent) {
if thisEvent == nil {
return
}
go e.dispatchEventToListeners(thisEvent)
go e.dispatchEventToWindows(thisEvent)
}
func (e *EventProcessor) Off(eventName string) {
e.unRegisterListener(eventName)
}
func (e *EventProcessor) OffAll() {
e.notifyLock.Lock()
defer e.notifyLock.Unlock()
e.listeners = make(map[string][]*eventListener)
}
// registerListener provides a means of subscribing to events of type "eventName"
func (e *EventProcessor) registerListener(eventName string, callback func(*CustomEvent), counter int) func() {
// Create new eventListener
thisListener := &eventListener{
callback: callback,
counter: counter,
delete: false,
}
e.notifyLock.Lock()
// Append the new listener to the listeners slice
e.listeners[eventName] = append(e.listeners[eventName], thisListener)
e.notifyLock.Unlock()
return func() {
e.notifyLock.Lock()
defer e.notifyLock.Unlock()
if _, ok := e.listeners[eventName]; !ok {
return
}
e.listeners[eventName] = lo.Filter(e.listeners[eventName], func(l *eventListener, i int) bool {
return l != thisListener
})
}
}
// unRegisterListener provides a means of unsubscribing to events of type "eventName"
func (e *EventProcessor) unRegisterListener(eventName string) {
e.notifyLock.Lock()
defer e.notifyLock.Unlock()
delete(e.listeners, eventName)
}
// dispatchEventToListeners calls all registered listeners event name
func (e *EventProcessor) dispatchEventToListeners(event *CustomEvent) {
e.notifyLock.Lock()
defer e.notifyLock.Unlock()
listeners := e.listeners[event.Name]
if listeners == nil {
return
}
// We have a dirty flag to indicate that there are items to delete
itemsToDelete := false
// Callback in goroutine
for _, listener := range listeners {
if listener.counter > 0 {
listener.counter--
}
go listener.callback(event)
if listener.counter == 0 {
listener.delete = true
itemsToDelete = true
}
}
// Do we have items to delete?
if itemsToDelete == true {
e.listeners[event.Name] = lo.Filter(listeners, func(l *eventListener, i int) bool {
return l.delete == false
})
}
}

View File

@ -0,0 +1,135 @@
package application_test
import (
"sync"
"testing"
"github.com/wailsapp/wails/v3/pkg/application"
"github.com/matryer/is"
)
type mockNotifier struct {
Events []*application.CustomEvent
}
func (m *mockNotifier) dispatchEventToWindows(event *application.CustomEvent) {
m.Events = append(m.Events, event)
}
func (m *mockNotifier) Reset() {
m.Events = []*application.CustomEvent{}
}
func Test_EventsOn(t *testing.T) {
i := is.New(t)
notifier := &mockNotifier{}
eventProcessor := application.NewCustomEventProcessor(notifier.dispatchEventToWindows)
// Test On
eventName := "test"
counter := 0
var wg sync.WaitGroup
wg.Add(1)
unregisterFn := eventProcessor.On(eventName, func(event *application.CustomEvent) {
// This is called in a goroutine
counter++
wg.Done()
})
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
wg.Wait()
i.Equal(1, counter)
// Unregister
notifier.Reset()
unregisterFn()
counter = 0
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
i.Equal(0, counter)
}
func Test_EventsOnce(t *testing.T) {
i := is.New(t)
notifier := &mockNotifier{}
eventProcessor := application.NewCustomEventProcessor(notifier.dispatchEventToWindows)
// Test On
eventName := "test"
counter := 0
var wg sync.WaitGroup
wg.Add(1)
unregisterFn := eventProcessor.Once(eventName, func(event *application.CustomEvent) {
// This is called in a goroutine
counter++
wg.Done()
})
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
wg.Wait()
i.Equal(1, counter)
// Unregister
notifier.Reset()
unregisterFn()
counter = 0
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
i.Equal(0, counter)
}
func Test_EventsOnMultiple(t *testing.T) {
i := is.New(t)
notifier := &mockNotifier{}
eventProcessor := application.NewCustomEventProcessor(notifier.dispatchEventToWindows)
// Test On
eventName := "test"
counter := 0
var wg sync.WaitGroup
wg.Add(2)
unregisterFn := eventProcessor.OnMultiple(eventName, func(event *application.CustomEvent) {
// This is called in a goroutine
counter++
wg.Done()
}, 2)
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
wg.Wait()
i.Equal(2, counter)
// Unregister
notifier.Reset()
unregisterFn()
counter = 0
eventProcessor.Emit(&application.CustomEvent{
Name: "test",
Data: "test payload",
})
i.Equal(0, counter)
}

View File

@ -29,7 +29,7 @@ func dispatchOnMainThreadCallback(callbackID C.uint) {
id := uint(callbackID) id := uint(callbackID)
fn := mainThreadFunctionStore[id] fn := mainThreadFunctionStore[id]
if fn == nil { if fn == nil {
Fatal("dispatchCallback called with invalid id: ", id) Fatal("dispatchCallback called with invalid id: %v", id)
} }
delete(mainThreadFunctionStore, id) delete(mainThreadFunctionStore, id)
mainThreadFunctionStoreLock.RUnlock() mainThreadFunctionStoreLock.RUnlock()

View File

@ -163,7 +163,7 @@ func newRole(role Role) *MenuItem {
case ToggleDevTools: case ToggleDevTools:
return newToggleDevToolsMenuItem() return newToggleDevToolsMenuItem()
case ResetZoom: case ResetZoom:
return newResetZoomMenuItem() return newZoomResetMenuItem()
case ZoomIn: case ZoomIn:
return newZoomInMenuItem() return newZoomInMenuItem()
case ZoomOut: case ZoomOut:

View File

@ -564,14 +564,14 @@ func newToggleDevToolsMenuItem() *MenuItem {
}) })
} }
func newResetZoomMenuItem() *MenuItem { func newZoomResetMenuItem() *MenuItem {
// reset zoom menu item // reset zoom menu item
return newMenuItem("Actual Size"). return newMenuItem("Actual Size").
SetAccelerator("CmdOrCtrl+0"). SetAccelerator("CmdOrCtrl+0").
OnClick(func(ctx *Context) { OnClick(func(ctx *Context) {
currentWindow := globalApplication.CurrentWindow() currentWindow := globalApplication.CurrentWindow()
if currentWindow != nil { if currentWindow != nil {
currentWindow.ResetZoom() currentWindow.ZoomReset()
} }
}) })
} }

View File

@ -2,7 +2,10 @@ package application
import ( import (
"fmt" "fmt"
"net/http"
"strings" "strings"
jsoniter "github.com/json-iterator/go"
) )
type MessageProcessor struct { type MessageProcessor struct {
@ -15,46 +18,63 @@ func NewMessageProcessor(w *WebviewWindow) *MessageProcessor {
} }
} }
func (m *MessageProcessor) ProcessMessage(message string) { func (m *MessageProcessor) httpError(rw http.ResponseWriter, message string, args ...any) {
m.Error(message, args...)
rw.WriteHeader(http.StatusBadRequest)
rw.Write([]byte(fmt.Sprintf(message, args...)))
}
// TODO: Implement calls to other windows func (m *MessageProcessor) HandleRuntimeCall(rw http.ResponseWriter, r *http.Request) {
// Check for prefix "WINDOWID" // Read "method" from query string
// If prefix exists, get window ID by parsing: "WINDOWID:12:MESSAGE" method := r.URL.Query().Get("method")
if method == "" {
if strings.HasPrefix(message, "WINDOWID") { m.httpError(rw, "No method specified")
m.Error("Window ID prefix not yet implemented")
return return
} }
splitMethod := strings.Split(method, ".")
window := m.window if len(splitMethod) != 2 {
m.httpError(rw, "Invalid method format")
if message == "" {
m.Error("Blank message received")
return return
} }
m.Info("Processing message: %s", message) // Get the object
switch message[0] { object := splitMethod[0]
//case 'L': // Get the method
// m.processLogMessage(message) method = splitMethod[1]
//case 'E':
// return m.processEventMessage(message) params := QueryParams(r.URL.Query())
//case 'C':
// return m.processCallMessage(message) var targetWindow = m.window
//case 'c': windowID := params.UInt("windowID")
// return m.processSecureCallMessage(message) if windowID != nil {
case 'W': // Get window for ID
m.processWindowMessage(message, window) targetWindow = globalApplication.getWindowForID(*windowID)
//case 'B': if targetWindow == nil {
// return m.processBrowserMessage(message) m.Error("Window ID %s not found", *windowID)
case 'Q': return
globalApplication.Quit() }
case 'S': }
//globalApplication.Show()
case 'H': switch object {
//globalApplication.Hide() case "window":
m.processWindowMethod(method, rw, r, targetWindow, params)
case "clipboard":
m.processClipboardMethod(method, rw, r, targetWindow, params)
case "dialog":
m.processDialogMethod(method, rw, r, targetWindow, params)
case "events":
m.processEventsMethod(method, rw, r, targetWindow, params)
case "application":
m.processApplicationMethod(method, rw, r, targetWindow, params)
case "log":
m.processLogMethod(method, rw, r, targetWindow, params)
default: default:
m.Error("Unknown message from front end:", message) m.httpError(rw, "Unknown runtime call: %s", object)
} }
}
func (m *MessageProcessor) ProcessMessage(message string) {
m.Info("ProcessMessage from front end:", message)
} }
func (m *MessageProcessor) Error(message string, args ...any) { func (m *MessageProcessor) Error(message string, args ...any) {
@ -64,3 +84,37 @@ func (m *MessageProcessor) Error(message string, args ...any) {
func (m *MessageProcessor) Info(message string, args ...any) { func (m *MessageProcessor) Info(message string, args ...any) {
fmt.Printf("[MessageProcessor] Info: "+message, args...) fmt.Printf("[MessageProcessor] Info: "+message, args...)
} }
func (m *MessageProcessor) json(rw http.ResponseWriter, data any) {
// convert data to json
var jsonPayload = []byte("{}")
var err error
if data != nil {
jsonPayload, err = jsoniter.Marshal(data)
if err != nil {
m.Error("Unable to convert data to JSON. Please report this to the Wails team! Error: %s", err)
return
}
}
_, err = rw.Write(jsonPayload)
if err != nil {
m.Error("Unable to write json payload. Please report this to the Wails team! Error: %s", err)
return
}
rw.Header().Set("Content-Type", "application/json")
m.ok(rw)
}
func (m *MessageProcessor) text(rw http.ResponseWriter, data string) {
_, err := rw.Write([]byte(data))
if err != nil {
m.Error("Unable to write json payload. Please report this to the Wails team! Error: %s", err)
return
}
rw.Header().Set("Content-Type", "text/plain")
m.ok(rw)
}
func (m *MessageProcessor) ok(rw http.ResponseWriter) {
rw.WriteHeader(http.StatusOK)
}

View File

@ -0,0 +1,23 @@
package application
import (
"net/http"
)
func (m *MessageProcessor) processApplicationMethod(method string, rw http.ResponseWriter, r *http.Request, window *WebviewWindow, params QueryParams) {
switch method {
case "Quit":
globalApplication.Quit()
m.ok(rw)
case "Hide":
globalApplication.Hide()
m.ok(rw)
case "Show":
globalApplication.Show()
m.ok(rw)
default:
m.httpError(rw, "Unknown event method: %s", method)
}
}

View File

@ -0,0 +1,25 @@
package application
import (
"net/http"
)
func (m *MessageProcessor) processClipboardMethod(method string, rw http.ResponseWriter, _ *http.Request, _ *WebviewWindow, params QueryParams) {
switch method {
case "SetText":
title := params.String("text")
if title == nil {
m.Error("SetText: text is required")
return
}
globalApplication.Clipboard().SetText(*title)
m.ok(rw)
case "Text":
text := globalApplication.Clipboard().Text()
m.text(rw, text)
default:
m.httpError(rw, "Unknown clipboard method: %s", method)
}
}

View File

@ -0,0 +1,128 @@
package application
import (
"encoding/json"
"fmt"
"net/http"
"runtime"
"strconv"
)
func (m *MessageProcessor) dialogErrorCallback(message string, dialogID *string, err error) {
errorMsg := fmt.Sprintf(message, err)
m.Error(errorMsg)
msg := "_wails.dialogErrorCallback('" + *dialogID + "', " + strconv.Quote(errorMsg) + ");"
m.window.ExecJS(msg)
}
func (m *MessageProcessor) dialogCallback(dialogID *string, result string, isJSON bool) {
msg := fmt.Sprintf("_wails.dialogCallback('%s', %s, %v);", *dialogID, strconv.Quote(result), isJSON)
m.window.ExecJS(msg)
}
func (m *MessageProcessor) processDialogMethod(method string, rw http.ResponseWriter, r *http.Request, window *WebviewWindow, params QueryParams) {
args, err := params.Args()
if err != nil {
m.httpError(rw, "Unable to parse arguments: %s", err)
return
}
dialogID := args.String("dialog-id")
if dialogID == nil {
m.Error("dialog-id is required")
return
}
switch method {
case "Info", "Warning", "Error", "Question":
var options MessageDialogOptions
err := params.ToStruct(&options)
if err != nil {
m.dialogErrorCallback("Error parsing dialog options: %s", dialogID, err)
return
}
if len(options.Buttons) == 0 {
switch runtime.GOOS {
case "darwin":
options.Buttons = []*Button{{Label: "OK", IsDefault: true}}
}
}
var dialog *MessageDialog
switch method {
case "Info":
dialog = globalApplication.InfoDialog()
case "Warning":
dialog = globalApplication.WarningDialog()
case "Error":
dialog = globalApplication.ErrorDialog()
case "Question":
dialog = globalApplication.QuestionDialog()
}
// TODO: Add support for attaching Message dialogs to windows
dialog.SetTitle(options.Title)
dialog.SetMessage(options.Message)
for _, button := range options.Buttons {
label := button.Label
button.OnClick(func() {
m.dialogCallback(dialogID, label, false)
})
}
dialog.AddButtons(options.Buttons)
dialog.Show()
m.ok(rw)
case "OpenFile":
var options OpenFileDialogOptions
err := params.ToStruct(&options)
if err != nil {
m.httpError(rw, "Error parsing dialog options: %s", err.Error())
return
}
dialog := globalApplication.OpenFileDialogWithOptions(&options)
go func() {
if options.AllowsMultipleSelection {
files, err := dialog.PromptForMultipleSelection()
if err != nil {
m.dialogErrorCallback("Error getting selection: %s", dialogID, err)
return
} else {
result, err := json.Marshal(files)
if err != nil {
m.dialogErrorCallback("Error marshalling files: %s", dialogID, err)
return
}
m.dialogCallback(dialogID, string(result), true)
}
} else {
file, err := dialog.PromptForSingleSelection()
if err != nil {
m.dialogErrorCallback("Error getting selection: %s", dialogID, err)
return
}
m.dialogCallback(dialogID, file, false)
}
}()
m.ok(rw)
case "SaveFile":
var options SaveFileDialogOptions
err := params.ToStruct(&options)
if err != nil {
m.httpError(rw, "Error parsing dialog options: %s", err.Error())
return
}
dialog := globalApplication.SaveFileDialogWithOptions(&options)
go func() {
file, err := dialog.PromptForSingleSelection()
if err != nil {
m.dialogErrorCallback("Error getting selection: %s", dialogID, err)
return
}
m.dialogCallback(dialogID, file, false)
}()
m.ok(rw)
default:
m.httpError(rw, "Unknown dialog method: %s", method)
}
}

View File

@ -0,0 +1,28 @@
package application
import (
"net/http"
)
func (m *MessageProcessor) processEventsMethod(method string, rw http.ResponseWriter, _ *http.Request, window *WebviewWindow, params QueryParams) {
switch method {
case "Emit":
var event CustomEvent
err := params.ToStruct(&event)
if err != nil {
m.httpError(rw, "Error parsing event: %s", err)
return
}
if event.Name == "" {
m.httpError(rw, "Event name must be specified")
return
}
event.Sender = window.Name()
globalApplication.Events.Emit(&event)
m.ok(rw)
default:
m.httpError(rw, "Unknown event method: %s", method)
}
}

View File

@ -1,47 +1,25 @@
package application package application
// import (
//import "errors" "net/http"
//
////var logLevelMap = map[byte]logger.LogLevel{ "github.com/wailsapp/wails/v3/pkg/logger"
//// '1': pkgLogger.TRACE, )
//// '2': pkgLogger.DEBUG,
//// '3': pkgLogger.INFO, func (m *MessageProcessor) processLogMethod(method string, rw http.ResponseWriter, _ *http.Request, window *WebviewWindow, params QueryParams) {
//// '4': pkgLogger.WARNING, switch method {
//// '5': pkgLogger.ERROR, case "Log":
////} var msg logger.Message
// err := params.ToStruct(&msg)
//func (m *MessageProcessor) processLogMessage(message string) { if err != nil {
// if len(message) < 3 { m.httpError(rw, "error parsing log message: %s", err.Error())
// m.Error("Invalid Log Message: " + message) return
// return }
// } msg.Sender = window.Name()
// globalApplication.Log(&msg)
// messageText := message[2:] m.ok(rw)
// default:
// switch message[1] { m.httpError(rw, "Unknown log method: %s", method)
// case 'T': }
// d.log.Trace(messageText)
// case 'P': }
// d.log.Print(messageText)
// case 'D':
// d.log.Debug(messageText)
// case 'I':
// d.log.Info(messageText)
// case 'W':
// d.log.Warning(messageText)
// case 'E':
// d.log.Error(messageText)
// case 'F':
// d.log.Fatal(messageText)
// case 'S':
// loglevel, exists := logLevelMap[message[2]]
// if !exists {
// return "", errors.New("Invalid Set Log Level Message: " + message)
// }
// d.log.SetLogLevel(loglevel)
// default:
// return "", errors.New("Invalid Log Message: " + message)
// }
// return "", nil
//}

View File

@ -0,0 +1,190 @@
package application
import (
"encoding/json"
"fmt"
"strconv"
)
type QueryParams map[string][]string
func (qp QueryParams) String(key string) *string {
if qp == nil {
return nil
}
values := qp[key]
if len(values) == 0 {
return nil
}
return &values[0]
}
func (qp QueryParams) Int(key string) *int {
val := qp.String(key)
if val == nil {
return nil
}
result, err := strconv.Atoi(*val)
if err != nil {
return nil
}
return &result
}
func (qp QueryParams) UInt8(key string) *uint8 {
val := qp.String(key)
if val == nil {
return nil
}
intResult, err := strconv.Atoi(*val)
if err != nil {
return nil
}
if intResult < 0 {
intResult = 0
}
if intResult > 255 {
intResult = 255
}
var result = uint8(intResult)
return &result
}
func (qp QueryParams) UInt(key string) *uint {
val := qp.String(key)
if val == nil {
return nil
}
intResult, err := strconv.Atoi(*val)
if err != nil {
return nil
}
if intResult < 0 {
intResult = 0
}
if intResult > 255 {
intResult = 255
}
var result = uint(intResult)
return &result
}
func (qp QueryParams) Bool(key string) *bool {
val := qp.String(key)
if val == nil {
return nil
}
result, err := strconv.ParseBool(*val)
if err != nil {
return nil
}
return &result
}
func (qp QueryParams) Float64(key string) *float64 {
val := qp.String(key)
if val == nil {
return nil
}
result, err := strconv.ParseFloat(*val, 64)
if err != nil {
return nil
}
return &result
}
func (qp QueryParams) ToStruct(str any) error {
args := qp["args"]
if len(args) == 1 {
return json.Unmarshal([]byte(args[0]), &str)
}
return nil
}
type Args struct {
data map[string]any
}
func (a *Args) String(key string) *string {
if a == nil {
return nil
}
if val := a.data[key]; val != nil {
result := fmt.Sprintf("%v", val)
return &result
}
return nil
}
func (a *Args) Int(s string) *int {
if a == nil {
return nil
}
if val := a.data[s]; val != nil {
result := val.(int)
return &result
}
return nil
}
func (a *Args) UInt8(s string) *uint8 {
if a == nil {
return nil
}
if val := a.data[s]; val != nil {
result := val.(uint8)
return &result
}
return nil
}
func (a *Args) UInt(s string) *uint {
if a == nil {
return nil
}
if val := a.data[s]; val != nil {
result := val.(uint)
return &result
}
return nil
}
func (a *Args) Float64(s string) *float64 {
if a == nil {
return nil
}
if val := a.data[s]; val != nil {
result := val.(float64)
return &result
}
return nil
}
func (a *Args) Bool(s string) *bool {
if a == nil {
return nil
}
if val := a.data[s]; val != nil {
result := val.(bool)
return &result
}
return nil
}
func (qp QueryParams) Args() (*Args, error) {
argData := qp["args"]
var result = &Args{
data: make(map[string]any),
}
if len(argData) == 1 {
err := json.Unmarshal([]byte(argData[0]), &result.data)
if err != nil {
return nil, err
}
}
return result, nil
}

View File

@ -1,95 +1,191 @@
package application package application
import ( import (
"encoding/json" "net/http"
"strconv"
"strings"
"github.com/wailsapp/wails/v3/pkg/options" "github.com/wailsapp/wails/v3/pkg/options"
) )
func (m *MessageProcessor) mustAtoI(input string) int { func (m *MessageProcessor) processWindowMethod(method string, rw http.ResponseWriter, r *http.Request, window *WebviewWindow, params QueryParams) {
result, err := strconv.Atoi(input)
args, err := params.Args()
if err != nil { if err != nil {
m.Error("cannot convert %s to integer!", input) m.httpError(rw, "Unable to parse arguments: %s", err)
} return
return result
}
func (m *MessageProcessor) processWindowMessage(message string, window *WebviewWindow) {
if len(message) < 2 {
m.Error("Invalid Window Message: " + message)
} }
switch message[1] { switch method {
case 'A': case "SetTitle":
switch message[2:] { title := args.String("title")
//case "SDT": if title == nil {
// go window.WindowSetSystemDefaultTheme() m.Error("SetTitle: title is required")
//case "LT": return
// go window.SetLightTheme()
//case "DT":
// go window.SetDarkTheme()
case "TP:0", "TP:1":
if message[2:] == "TP:0" {
go window.SetAlwaysOnTop(false)
} else if message[2:] == "TP:1" {
go window.SetAlwaysOnTop(true)
}
} }
case 'c': window.SetTitle(*title)
go window.Center() m.ok(rw)
case 'T': case "SetSize":
title := message[2:] width := args.Int("width")
go window.SetTitle(title) height := args.Int("height")
case 'F': if width == nil || height == nil {
go window.Fullscreen() m.Error("Invalid SetSize Message")
case 'f': return
go window.UnFullscreen() }
case 's': window.SetSize(*width, *height)
parts := strings.Split(message[3:], ":") m.ok(rw)
w := m.mustAtoI(parts[0]) case "SetPosition":
h := m.mustAtoI(parts[1]) x := args.Int("x")
go window.SetSize(w, h) y := args.Int("y")
case 'p': if x == nil || y == nil {
parts := strings.Split(message[3:], ":") m.Error("Invalid SetPosition Message")
x := m.mustAtoI(parts[0]) return
y := m.mustAtoI(parts[1]) }
go window.SetPosition(x, y) window.SetPosition(*x, *y)
case 'H': m.ok(rw)
go window.Hide() case "Fullscreen":
case 'S': window.Fullscreen()
go window.Show() m.ok(rw)
//case 'R': case "UnFullscreen":
// go window.ReloadApp() window.UnFullscreen()
case 'r': m.ok(rw)
var rgba options.RGBA case "Minimise":
err := json.Unmarshal([]byte(message[3:]), &rgba) window.Minimize()
m.ok(rw)
case "UnMinimise":
window.UnMinimise()
m.ok(rw)
case "Maximise":
window.Maximise()
m.ok(rw)
case "UnMaximise":
window.UnMaximise()
m.ok(rw)
case "Show":
window.Show()
m.ok(rw)
case "Hide":
window.Hide()
m.ok(rw)
case "Close":
window.Close()
m.ok(rw)
case "Center":
window.Center()
m.ok(rw)
case "Size":
width, height := window.Size()
m.json(rw, map[string]interface{}{
"width": width,
"height": height,
})
case "Position":
x, y := window.Position()
m.json(rw, map[string]interface{}{
"x": x,
"y": y,
})
case "SetBackgroundColour":
r := args.UInt8("r")
if r == nil {
m.Error("Invalid SetBackgroundColour Message: 'r' value required")
return
}
g := args.UInt8("g")
if g == nil {
m.Error("Invalid SetBackgroundColour Message: 'g' value required")
return
}
b := args.UInt8("b")
if b == nil {
m.Error("Invalid SetBackgroundColour Message: 'b' value required")
return
}
a := args.UInt8("a")
if a == nil {
m.Error("Invalid SetBackgroundColour Message: 'a' value required")
return
}
window.SetBackgroundColour(&options.RGBA{
Red: *r,
Green: *g,
Blue: *b,
Alpha: *a,
})
m.ok(rw)
case "SetAlwaysOnTop":
alwaysOnTop := args.Bool("alwaysOnTop")
if alwaysOnTop == nil {
m.Error("Invalid SetAlwaysOnTop Message: 'alwaysOnTop' value required")
return
}
window.SetAlwaysOnTop(*alwaysOnTop)
m.ok(rw)
case "SetResizable":
resizable := args.Bool("resizable")
if resizable == nil {
m.Error("Invalid SetResizable Message: 'resizable' value required")
return
}
window.SetResizable(*resizable)
m.ok(rw)
case "SetMinSize":
width := args.Int("width")
height := args.Int("height")
if width == nil || height == nil {
m.Error("Invalid SetMinSize Message")
return
}
window.SetMinSize(*width, *height)
m.ok(rw)
case "SetMaxSize":
width := args.Int("width")
height := args.Int("height")
if width == nil || height == nil {
m.Error("Invalid SetMaxSize Message")
return
}
window.SetMaxSize(*width, *height)
m.ok(rw)
case "Width":
width := window.Width()
m.json(rw, map[string]interface{}{
"width": width,
})
case "Height":
height := window.Height()
m.json(rw, map[string]interface{}{
"height": height,
})
case "ZoomIn":
window.ZoomIn()
m.ok(rw)
case "ZoomOut":
window.ZoomOut()
m.ok(rw)
case "ZoomReset":
window.ZoomReset()
m.ok(rw)
case "GetZoom":
zoomLevel := window.GetZoom()
m.json(rw, map[string]interface{}{
"zoomLevel": zoomLevel,
})
case "Screen":
screen, err := window.GetScreen()
if err != nil { if err != nil {
m.Error("Invalid RGBA Message: %s", err.Error()) m.httpError(rw, err.Error())
return
} }
go window.SetBackgroundColour(&rgba) m.json(rw, screen)
case 'M': case "SetZoom":
go window.Maximise() zoomLevel := args.Float64("zoomLevel")
//case 't': if zoomLevel == nil {
// go window.ToggleMaximise() m.Error("Invalid SetZoom Message: invalid 'zoomLevel' value")
case 'U': return
go window.UnMaximise() }
case 'm': window.SetZoom(*zoomLevel)
go window.Minimise() m.ok(rw)
case 'u':
go window.UnMinimise()
case 'Z':
parts := strings.Split(message[3:], ":")
w := m.mustAtoI(parts[0])
h := m.mustAtoI(parts[1])
go window.SetMaxSize(w, h)
case 'z':
parts := strings.Split(message[3:], ":")
w := m.mustAtoI(parts[0])
h := m.mustAtoI(parts[1])
go window.SetMinSize(w, h)
default: default:
m.Error("unknown Window message: %s", message) m.httpError(rw, "Unknown window method: %s", method)
} }
} }

View File

@ -3,6 +3,9 @@ package application
import ( import (
"fmt" "fmt"
"sync" "sync"
"time"
"github.com/wailsapp/wails/v3/pkg/logger"
"github.com/wailsapp/wails/v2/pkg/assetserver" "github.com/wailsapp/wails/v2/pkg/assetserver"
"github.com/wailsapp/wails/v2/pkg/assetserver/webview" "github.com/wailsapp/wails/v2/pkg/assetserver/webview"
@ -34,9 +37,11 @@ type (
reload() reload()
forceReload() forceReload()
toggleDevTools() toggleDevTools()
resetZoom() zoomReset()
zoomIn() zoomIn()
zoomOut() zoomOut()
getZoom() float64
setZoom(zoom float64)
close() close()
zoom() zoom()
minimize() minimize()
@ -96,11 +101,10 @@ func NewWindow(options *options.WebviewWindow) *WebviewWindow {
} }
opts := assetserveroptions.Options{Assets: options.Assets.FS, Handler: options.Assets.Handler, Middleware: options.Assets.Middleware} opts := assetserveroptions.Options{Assets: options.Assets.FS, Handler: options.Assets.Handler, Middleware: options.Assets.Middleware}
// TODO Bindings, Logger, ServingFrom disk? // TODO Bindings, ServingFrom disk?
srv, err := assetserver.NewAssetServer("", opts, false, nil, runtime.RuntimeAssetsBundle) srv, err := assetserver.NewAssetServer("", opts, false, nil, runtime.RuntimeAssetsBundle)
if err != nil { if err != nil {
// TODO handle errors globalApplication.fatal(err.Error())
panic(err)
} }
result := &WebviewWindow{ result := &WebviewWindow{
@ -112,6 +116,7 @@ func NewWindow(options *options.WebviewWindow) *WebviewWindow {
} }
result.messageProcessor = NewMessageProcessor(result) result.messageProcessor = NewMessageProcessor(result)
srv.UseRuntimeHandler(result.messageProcessor)
return result return result
} }
@ -126,6 +131,13 @@ func (w *WebviewWindow) SetTitle(title string) *WebviewWindow {
return w return w
} }
func (w *WebviewWindow) Name() string {
if w.options.Name != "" {
return w.options.Name
}
return fmt.Sprintf("Window %d", w.id)
}
func (w *WebviewWindow) SetSize(width, height int) *WebviewWindow { func (w *WebviewWindow) SetSize(width, height int) *WebviewWindow {
// Don't set size if fullscreen // Don't set size if fullscreen
if w.IsFullscreen() { if w.IsFullscreen() {
@ -211,6 +223,21 @@ func (w *WebviewWindow) SetURL(s string) *WebviewWindow {
return w return w
} }
func (w *WebviewWindow) SetZoom(magnification float64) *WebviewWindow {
w.options.Zoom = magnification
if w.impl != nil {
w.impl.setZoom(magnification)
}
return w
}
func (w *WebviewWindow) GetZoom() float64 {
if w.impl != nil {
return w.impl.getZoom()
}
return 1
}
func (w *WebviewWindow) SetResizable(b bool) *WebviewWindow { func (w *WebviewWindow) SetResizable(b bool) *WebviewWindow {
w.options.DisableResize = !b w.options.DisableResize = !b
if w.impl != nil { if w.impl != nil {
@ -347,7 +374,7 @@ func (w *WebviewWindow) SetBackgroundColour(colour *options.RGBA) *WebviewWindow
} }
func (w *WebviewWindow) handleMessage(message string) { func (w *WebviewWindow) handleMessage(message string) {
fmt.Printf("[window %d] %s\n", w.id, message) w.info(message)
// Check for special messages // Check for special messages
if message == "test" { if message == "test" {
w.SetTitle("Hello World") w.SetTitle("Hello World")
@ -358,7 +385,7 @@ func (w *WebviewWindow) handleMessage(message string) {
func (w *WebviewWindow) handleWebViewRequest(request webview.Request) { func (w *WebviewWindow) handleWebViewRequest(request webview.Request) {
url, _ := request.URL() url, _ := request.URL()
fmt.Printf("[window %d] Request %s\n", w.id, url) w.info("Request: %s", url)
w.assets.ServeWebViewRequest(request) w.assets.ServeWebViewRequest(request)
} }
@ -449,9 +476,9 @@ func (w *WebviewWindow) ToggleDevTools() {
w.impl.toggleDevTools() w.impl.toggleDevTools()
} }
func (w *WebviewWindow) ResetZoom() *WebviewWindow { func (w *WebviewWindow) ZoomReset() *WebviewWindow {
if w.impl != nil { if w.impl != nil {
w.impl.resetZoom() w.impl.zoomReset()
} }
return w return w
@ -598,3 +625,19 @@ func (w *WebviewWindow) SetFrameless(frameless bool) *WebviewWindow {
} }
return w return w
} }
func (w *WebviewWindow) dispatchCustomEvent(event *CustomEvent) {
msg := fmt.Sprintf("_wails.dispatchCustomEvent(%s);", event.ToJSON())
w.ExecJS(msg)
}
func (w *WebviewWindow) info(message string, args ...any) {
globalApplication.Log(&logger.Message{
Level: "INFO",
Message: message,
Data: args,
Sender: w.Name(),
Time: time.Now(),
})
}

View File

@ -251,8 +251,8 @@ void windowEnableDevTools(void* nsWindow) {
}); });
} }
// windowResetZoom // windowZoomReset
void windowResetZoom(void* nsWindow) { void windowZoomReset(void* nsWindow) {
// Reset zoom on main thread // Reset zoom on main thread
dispatch_async(dispatch_get_main_queue(), ^{ dispatch_async(dispatch_get_main_queue(), ^{
// Get window delegate // Get window delegate
@ -262,6 +262,24 @@ void windowResetZoom(void* nsWindow) {
}); });
} }
// windowZoomSet
void windowZoomSet(void* nsWindow, double zoom) {
// Reset zoom on main thread
dispatch_async(dispatch_get_main_queue(), ^{
// Get window delegate
WebviewWindowDelegate* delegate = (WebviewWindowDelegate*)[(WebviewWindow*)nsWindow delegate];
// Reset zoom
[delegate.webView setMagnification:zoom];
});
}
// windowZoomGet
float windowZoomGet(void* nsWindow) {
// Get zoom
WebviewWindowDelegate* delegate = (WebviewWindowDelegate*)[(WebviewWindow*)nsWindow delegate];
return [delegate.webView magnification];
}
// windowZoomIn // windowZoomIn
void windowZoomIn(void* nsWindow) { void windowZoomIn(void* nsWindow) {
// Zoom in on main thread // Zoom in on main thread
@ -762,6 +780,14 @@ type macosWebviewWindow struct {
parent *WebviewWindow parent *WebviewWindow
} }
func (w *macosWebviewWindow) getZoom() float64 {
return float64(C.windowZoomGet(w.nsWindow))
}
func (w *macosWebviewWindow) setZoom(zoom float64) {
C.windowZoomSet(w.nsWindow, C.double(zoom))
}
func (w *macosWebviewWindow) setFrameless(frameless bool) { func (w *macosWebviewWindow) setFrameless(frameless bool) {
C.windowSetFrameless(w.nsWindow, C.bool(frameless)) C.windowSetFrameless(w.nsWindow, C.bool(frameless))
if frameless { if frameless {
@ -848,8 +874,8 @@ func (w *macosWebviewWindow) zoomOut() {
C.windowZoomOut(w.nsWindow) C.windowZoomOut(w.nsWindow)
} }
func (w *macosWebviewWindow) resetZoom() { func (w *macosWebviewWindow) zoomReset() {
C.windowResetZoom(w.nsWindow) C.windowZoomReset(w.nsWindow)
} }
func (w *macosWebviewWindow) toggleDevTools() { func (w *macosWebviewWindow) toggleDevTools() {
@ -1017,6 +1043,7 @@ func (w *macosWebviewWindow) run() {
if w.parent.options.MaxWidth != 0 || w.parent.options.MaxHeight != 0 { if w.parent.options.MaxWidth != 0 || w.parent.options.MaxHeight != 0 {
w.setMaxSize(w.parent.options.MaxWidth, w.parent.options.MaxHeight) w.setMaxSize(w.parent.options.MaxWidth, w.parent.options.MaxHeight)
} }
//w.setZoom(w.parent.options.Zoom)
w.enableDevTools() w.enableDevTools()
w.setBackgroundColour(w.parent.options.BackgroundColour) w.setBackgroundColour(w.parent.options.BackgroundColour)
@ -1078,7 +1105,6 @@ func (w *macosWebviewWindow) run() {
if w.parent.options.Hidden == false { if w.parent.options.Hidden == false {
C.windowShow(w.nsWindow) C.windowShow(w.nsWindow)
} }
C.printWindowStyle(w.nsWindow)
}) })
} }

35
v3/pkg/logger/log.go Normal file
View File

@ -0,0 +1,35 @@
package logger
import (
"fmt"
)
type Logger struct {
output []Output
}
func New(outputs ...Output) *Logger {
result := &Logger{}
if outputs != nil {
result.output = outputs
}
return result
}
func (l *Logger) AddOutput(output Output) {
l.output = append(l.output, output)
}
func (l *Logger) Log(message *Message) {
for _, o := range l.output {
go o.Log(message)
}
}
func (l *Logger) Flush() {
for _, o := range l.output {
if err := o.Flush(); err != nil {
fmt.Printf("Error flushing '%s' Logger: %s\n", o.Name(), err.Error())
}
}
}

View File

@ -0,0 +1,27 @@
package logger
import "fmt"
type Console struct{}
func (l *Console) Name() string {
return "Console"
}
func (l *Console) Log(message *Message) {
msg := fmt.Sprintf(message.Message+"\n", message.Data...)
level := ""
if message.Level != "" {
level = fmt.Sprintf("[%s] ", message.Level)
}
sender := ""
if message.Sender != "" {
sender = fmt.Sprintf("%s: ", message.Sender)
}
fmt.Printf("%s%s%s", level, sender, msg)
}
func (l *Console) Flush() error {
return nil
}

11
v3/pkg/logger/message.go Normal file
View File

@ -0,0 +1,11 @@
package logger
import "time"
type Message struct {
Level string `json:"log"`
Message string `json:"message"`
Data []any `json:"data,omitempty"`
Sender string `json:"-"`
Time time.Time `json:"-"`
}

7
v3/pkg/logger/output.go Normal file
View File

@ -0,0 +1,7 @@
package logger
type Output interface {
Name() string
Log(message *Message)
Flush() error
}

View File

@ -1,9 +1,15 @@
package options package options
import "github.com/wailsapp/wails/v3/pkg/logger"
type Application struct { type Application struct {
Name string Name string
Description string Description string
Icon []byte Icon []byte
Mac Mac Mac Mac
Bind []interface{} Bind []interface{}
Logger struct {
Silent bool
CustomLoggers []logger.Output
}
} }

View File

@ -15,8 +15,7 @@ const (
) )
type WebviewWindow struct { type WebviewWindow struct {
// Alias is a human-readable name for the window. This can be used to reference the window in the frontend. Name string
Alias string
Title string Title string
Width, Height int Width, Height int
AlwaysOnTop bool AlwaysOnTop bool
@ -39,6 +38,7 @@ type WebviewWindow struct {
FullscreenButtonEnabled bool FullscreenButtonEnabled bool
Hidden bool Hidden bool
EnableFraudulentWebsiteWarnings bool EnableFraudulentWebsiteWarnings bool
Zoom float64
} }
var WindowDefaults = &WebviewWindow{ var WindowDefaults = &WebviewWindow{