mirror of
https://github.com/taigrr/wails.git
synced 2026-04-10 09:01:32 -07:00
* Get it compiling
* Stubs in place to compile
* Semi runs
* add darwin platform for server
* Evaluation working correctly. Still WIP
* Ignore favicon for desktop
* lint
* Remove feature flag code
* More feature flag removal
* Support sending messages to the backend
* Callbacks working
* Add Center + refactor prefs
* Fix logger
* Callback hooks for MOAE
* Update packages
* ignore test builds
* Support Un/Fullscreen
* vscode stuff
* Only show window when rendered
* Get it compiling again!
* Support tons of stuff!
* Tidy up
* More refactoring
* WIP [bugged]
* Got get frame working
* fix setsize and setposition
* Add Mac Application Options
* Add HideTitleBar
* Support more mac window options
* Add Toolbar support for Mac
* Support colour
* Support runtime colour change
* Moved options to own package
* Refactored mac titlebar options
* Support hidden titlebar
* Support HiddenInset Titlebar
* Support TitleBar Default
Fixed merging defaults
* Sample titlebars
* Fix minmax app
* Min/Max size supported
* WIP: Support multiple value return
* Support OpenDialog
* Remove old dialog code
* change service bus topics for dialogs
* Revert changes to v1
* Use options struct for dialogs
* Initial support for OpenDialog
* Support selecting files+dirs
* Support multiple files in dialog
* Support all dialog properties
* Add comments
* Filter support
* Support default directory
* Support SaveDialog
* Tidy Up
* WIP: Basics of window drag
* Support window dragging
* Update tests
* Frameless is calculated for Mac
* Tidy up
* Support vibrancy and transparency for webview
Options Colour -> RGBA
* Rename vibrancy to appearance
* Add default appearance
* Refactor part 1
* Refactor Part 2
* Support Translucent Window Background
* Update runtime test
* Add IsDarkMode
Updated runtime test
* Support theme mode change event
* Misc fixes for events
* Support OnMultiple
* Small fixes to frontend events
* Add System calls to runtime
* Add system calls to js runtime
* Support System calls in Go Runtime
* Port Sync Store
* Refactor store. Add get().
* Refactor system. Add IsDarkMode state store
* Use IsDarkMode state store
* Remove generated files
* Support setting app state at startup
* Add Store to go runtime
* Update runtime to v1.0.3
* Remove unused event messages
* Debugging
* initial kitchen sink
* Fix right click crash
* Better drag support
* WIP
* Remove log package
* Add Log to Go runtime
* Add logging to kitchen sink
* Improved CodeBlock. Dark mode to store.
* Start Events. List styling moved to global scope.
* Make logger a public package
* Revert logger package
* Major logging refactor
* Make Ffenestri use logging subsystem.
* Debug refactor
* Add trace to JS runtime
* Migrate runtime to @wails
* Support Trace in kitchensink
* Support trace in go runtime
* Support log level
* Support Print in JS runtime
* Runtime v1.0.1
* Move Info message to Trace
* Support Print logging
* Updated Logger interface
* Fix number of methods in Log
* Support SetLogLevel() at runtime
Refactor of loglevel
* Made go runtime package public.
Using loglevel store to keep loglevel in sync
* Support dynamic loglevel
* Runtime refactor
* Fully refactored logging
* Better looking scrollbar
* Terminal output component
* Link component
* SetLogLevel fully supported
* Runtime defs update.
Slight System refactor
* More Logging updates
* Move preview for SetLogLevel
* Fix log level reactivity.
Misc fixes and tweaks
* logging: slight refactor
* Update logger constants to fix default values
* @wails/runtime v1.0.4
* Fix change in logging levels
* hook in windowWillClose
* refactor clilogger
* WIP Events.On
* Add Events.On
* Improved error handling?
* Disable annoying smart quotes
* update runtime definitions
* Support Emit & Once. Improved On.
* Remove old event methods
* Remove old Event methods
* Update runtime in kitchensink
* Revert Fatal on JS Error
* Tidy up events runtime
* Finish events page
* Update Browser runtime API
* Unify Browser runtime
* JS Runtime v1.0.8
* Fix browser runtime export
* Remove debug line
* Add Browser examples
* Update title
* Improved runtime.System
* Update runtime.System to make all methods
* Expose System methods in Go runtime
* Add System to kitchensink
* Huge improvement to calls: Now handles objects
* Add JS runtime Dialog
* Dialog WIP
* Js package generation (#554)
* WIP
* Generation of index.js
* Add RelativeToCwd
* Add JSDoc comments
* Convert to ES6 syntax
* Fix typo
* Initial generation of typescript declarations
* Typescript improvements
* Improved @returns jsdoc
* Improved declaration files
* Simplified output
* Rename file
* Tidy up
* Revert "Simplified output"
This reverts commit 15cdf7382b.
* Now parsing actual code
* Support Array types
* Reimagined parser
* Wrap parsing in Parser
* Rewritten module generator (TS Only)
* Final touches
* Slight refactor to improve output
* Struct comments. External struct literal binding
* Reworked project parser *working*
* remove debug info
* Refactor of parser
* remove the spew
* Better Ts support
* Better project generation logic
* Support local functions in bind()
* JS Object generation. Linting.
* Support json tags in module generation
* Updated mod files
* Support vscode file generation
* Better global.d.ts
* add ts-check to templates
* Support TS declaration files
* improved 'generate' command for module
Co-authored-by: Travis McLane <tmclane@gmail.com>
296 lines
6.8 KiB
Go
296 lines
6.8 KiB
Go
// package runtime contains all the methods and data structures related to the
|
|
// runtime library of Wails. This includes both Go and JS runtimes.
|
|
package runtime
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
"reflect"
|
|
"sync"
|
|
)
|
|
|
|
// Options defines the optional data that may be used
|
|
// when creating a Store
|
|
type Options struct {
|
|
|
|
// The name of the store
|
|
Name string
|
|
|
|
// The runtime to attach the store to
|
|
Runtime *Runtime
|
|
|
|
// Indicates if notifying Go listeners should be notified of updates
|
|
// synchronously (on the current thread) or asynchronously using
|
|
// goroutines
|
|
NotifySynchronously bool
|
|
}
|
|
|
|
// StoreProvider is a struct that creates Stores
|
|
type StoreProvider struct {
|
|
runtime *Runtime
|
|
}
|
|
|
|
// newStore creates new stores using the provided Runtime reference.
|
|
func newStore(runtime *Runtime) *StoreProvider {
|
|
return &StoreProvider{
|
|
runtime: runtime,
|
|
}
|
|
}
|
|
|
|
// Store is where we keep named data
|
|
type Store struct {
|
|
name string
|
|
data reflect.Value
|
|
dataType reflect.Type
|
|
eventPrefix string
|
|
callbacks []reflect.Value
|
|
runtime *Runtime
|
|
notifySynchronously bool
|
|
|
|
// Lock
|
|
mux sync.Mutex
|
|
|
|
// Error handler
|
|
errorHandler func(error)
|
|
}
|
|
|
|
func fatal(err error) {
|
|
println(err.Error())
|
|
os.Exit(1)
|
|
}
|
|
|
|
// New creates a new store
|
|
func (p *StoreProvider) New(name string, defaultValue interface{}) *Store {
|
|
|
|
dataType := reflect.TypeOf(defaultValue)
|
|
|
|
result := Store{
|
|
name: name,
|
|
runtime: p.runtime,
|
|
data: reflect.ValueOf(defaultValue),
|
|
dataType: dataType,
|
|
}
|
|
|
|
// Setup the sync listener
|
|
result.setupListener()
|
|
|
|
return &result
|
|
}
|
|
|
|
// OnError takes a function that will be called
|
|
// whenever an error occurs
|
|
func (s *Store) OnError(callback func(error)) {
|
|
s.errorHandler = callback
|
|
}
|
|
|
|
// Processes the updates sent by the front end
|
|
func (s *Store) processUpdatedData(data string) error {
|
|
|
|
// Decode incoming data
|
|
var rawdata json.RawMessage
|
|
d := json.NewDecoder(bytes.NewBufferString(data))
|
|
err := d.Decode(&rawdata)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Create a new instance of our data and unmarshal
|
|
// the received value into it
|
|
newData := reflect.New(s.dataType).Interface()
|
|
err = json.Unmarshal(rawdata, &newData)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Lock mutex for writing
|
|
s.mux.Lock()
|
|
|
|
// Handle nulls
|
|
if newData == nil {
|
|
s.data = reflect.Zero(s.dataType)
|
|
} else {
|
|
// Store the resultant value in the data store
|
|
s.data = reflect.ValueOf(newData).Elem()
|
|
}
|
|
|
|
// Unlock mutex
|
|
s.mux.Unlock()
|
|
|
|
return nil
|
|
}
|
|
|
|
// Setup listener for front end changes
|
|
func (s *Store) setupListener() {
|
|
|
|
// Listen for updates from the front end
|
|
s.runtime.Events.On("wails:sync:store:updatedbyfrontend:"+s.name, func(data ...interface{}) {
|
|
|
|
// Process the incoming data
|
|
err := s.processUpdatedData(data[0].(string))
|
|
|
|
if err != nil {
|
|
if s.errorHandler != nil {
|
|
s.errorHandler(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
// Notify listeners
|
|
s.notify()
|
|
})
|
|
}
|
|
|
|
// notify the listeners of the current data state
|
|
func (s *Store) notify() {
|
|
|
|
// Execute callbacks
|
|
for _, callback := range s.callbacks {
|
|
|
|
// Build args
|
|
args := []reflect.Value{s.data}
|
|
|
|
if s.notifySynchronously {
|
|
callback.Call(args)
|
|
} else {
|
|
go callback.Call(args)
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// Set will update the data held by the store
|
|
// and notify listeners of the change
|
|
func (s *Store) Set(data interface{}) error {
|
|
|
|
inType := reflect.TypeOf(data)
|
|
|
|
if inType != s.dataType {
|
|
return fmt.Errorf("invalid data given in Store.Set(). Expected %s, got %s", s.dataType.String(), inType.String())
|
|
}
|
|
|
|
// Save data
|
|
s.mux.Lock()
|
|
s.data = reflect.ValueOf(data)
|
|
s.mux.Unlock()
|
|
|
|
// Stringify data
|
|
newdata, err := json.Marshal(data)
|
|
if err != nil {
|
|
if s.errorHandler != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Emit event to front end
|
|
s.runtime.Events.Emit("wails:sync:store:updatedbybackend:"+s.name, string(newdata))
|
|
|
|
// Notify subscribers
|
|
s.notify()
|
|
|
|
return nil
|
|
}
|
|
|
|
// callbackCheck ensures the given function to Subscribe() is
|
|
// of the correct signature. Absolutely cannot wait for
|
|
// generics to land rather than writing this nonsense.
|
|
func (s *Store) callbackCheck(callback interface{}) error {
|
|
|
|
// Get type
|
|
callbackType := reflect.TypeOf(callback)
|
|
|
|
// Check callback is a function
|
|
if callbackType.Kind() != reflect.Func {
|
|
return fmt.Errorf("invalid value given to store.Subscribe(). Expected 'func(%s)'", s.dataType.String())
|
|
}
|
|
|
|
// Check input param
|
|
if callbackType.NumIn() != 1 {
|
|
return fmt.Errorf("invalid number of parameters given in callback function. Expected 1")
|
|
}
|
|
|
|
// Check input data type
|
|
if callbackType.In(0) != s.dataType {
|
|
return fmt.Errorf("invalid type for input parameter given in callback function. Expected %s, got %s", s.dataType.String(), callbackType.In(0))
|
|
}
|
|
|
|
// Check output param
|
|
if callbackType.NumOut() != 0 {
|
|
return fmt.Errorf("invalid number of return parameters given in callback function. Expected 0")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Subscribe will subscribe to updates to the store by
|
|
// providing a callback. Any updates to the store are sent
|
|
// to the callback
|
|
func (s *Store) Subscribe(callback interface{}) {
|
|
|
|
err := s.callbackCheck(callback)
|
|
if err != nil {
|
|
fatal(err)
|
|
}
|
|
|
|
callbackFunc := reflect.ValueOf(callback)
|
|
|
|
s.callbacks = append(s.callbacks, callbackFunc)
|
|
}
|
|
|
|
// updaterCheck ensures the given function to Update() is
|
|
// of the correct signature. Absolutely cannot wait for
|
|
// generics to land rather than writing this nonsense.
|
|
func (s *Store) updaterCheck(updater interface{}) error {
|
|
|
|
// Get type
|
|
updaterType := reflect.TypeOf(updater)
|
|
|
|
// Check updater is a function
|
|
if updaterType.Kind() != reflect.Func {
|
|
return fmt.Errorf("invalid value given to store.Update(). Expected 'func(%s) %s'", s.dataType.String(), s.dataType.String())
|
|
}
|
|
|
|
// Check input param
|
|
if updaterType.NumIn() != 1 {
|
|
return fmt.Errorf("invalid number of parameters given in updater function. Expected 1")
|
|
}
|
|
|
|
// Check input data type
|
|
if updaterType.In(0) != s.dataType {
|
|
return fmt.Errorf("invalid type for input parameter given in updater function. Expected %s, got %s", s.dataType.String(), updaterType.In(0))
|
|
}
|
|
|
|
// Check output param
|
|
if updaterType.NumOut() != 1 {
|
|
return fmt.Errorf("invalid number of return parameters given in updater function. Expected 1")
|
|
}
|
|
|
|
// Check output data type
|
|
if updaterType.Out(0) != s.dataType {
|
|
return fmt.Errorf("invalid type for return parameter given in updater function. Expected %s, got %s", s.dataType.String(), updaterType.Out(0))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Update takes a function that is passed the current state.
|
|
// The result of that function is then set as the new state
|
|
// of the store. This will notify listeners of the change
|
|
func (s *Store) Update(updater interface{}) {
|
|
|
|
err := s.updaterCheck(updater)
|
|
if err != nil {
|
|
fatal(err)
|
|
}
|
|
|
|
// Build args
|
|
args := []reflect.Value{s.data}
|
|
|
|
// Make call
|
|
results := reflect.ValueOf(updater).Call(args)
|
|
|
|
// We will only have 1 result. Set the store to it
|
|
s.Set(results[0].Interface())
|
|
}
|