Files
wails/v2/internal/webserver/websockets.go
Lea Anthony 71bfd29376 Feature/v2 mac (#555)
* 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>
2020-11-15 09:27:23 +11:00

127 lines
3.3 KiB
Go

package webserver
import (
"context"
"net/http"
"strings"
"github.com/wailsapp/wails/v2/internal/logger"
ws "nhooyr.io/websocket"
"nhooyr.io/websocket/wsjson"
)
// WebClient represents an individual web session
type WebClient struct {
conn *ws.Conn
identifier string
logger *logger.Logger
running bool
}
// Quit terminates the webclient session
func (wc *WebClient) Quit() {
wc.running = false
}
// NotifyEvent sends the event
func (wc *WebClient) NotifyEvent(message string) {
wc.SendMessage("E" + message)
}
// CallResult sends the result of the Go function call back to the
// originator in the frontend
func (wc *WebClient) CallResult(message string) {
wc.SendMessage("R" + message)
}
// OpenDialog is a noop in the webclient
func (wc *WebClient) OpenDialog(title string) string {
return ""
}
// WindowSetTitle is a noop in the webclient
func (wc *WebClient) WindowSetTitle(title string) {}
// WindowFullscreen is a noop in the webclient
func (wc *WebClient) WindowFullscreen() {}
// WindowUnFullscreen is a noop in the webclient
func (wc *WebClient) WindowUnFullscreen() {}
// WindowSetColour is a noop in the webclient
func (wc *WebClient) WindowSetColour(colour int) {
}
// Run processes messages from the remote webclient
func (wc *WebClient) Run(w *WebServer) {
dispatcher := w.dispatcher.RegisterClient(wc)
defer w.dispatcher.RemoveClient(dispatcher)
defer w.unregisterClient(wc.identifier)
for wc.running {
var v interface{}
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
if err := wsjson.Read(ctx, wc.conn, &v); err != nil {
if ws.CloseStatus(err) == ws.StatusNormalClosure || ws.CloseStatus(err) == ws.StatusGoingAway {
break
}
if ws.CloseStatus(err) != -1 {
w.logger.Debug("Connection error: %s - %s", wc.identifier, err)
break
}
if !strings.Contains(err.Error(), "status = Status") {
w.logger.Debug("Error encountered on socket: %v", err)
break
}
}
dispatcher.DispatchMessage(v.(string))
}
wc.conn.Close(ws.StatusNormalClosure, "Goodbye")
w.logger.Debug("Connection closed: %v", wc.identifier)
}
// SendMessage converts the string to a []byte and passes it to
// the connection's Writer to send to the remote client
// The Writer itself prevents multiple users at the same time.
func (wc *WebClient) SendMessage(message string) {
wc.logger.Debug("WebClient.SendMessage() - %s", message)
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
wc.conn.Write(ctx, ws.MessageText, []byte(message))
}
// unregisterClient is called automatically by a WebClient session during termination
// so that it's registration can be removed
func (w *WebServer) unregisterClient(identifier string) {
w.logger.Debug("Removing WebClient : %v", identifier)
w.lock.Lock()
delete(w.connections, identifier)
w.lock.Unlock()
}
func (w *WebServer) websocketConnection(resp http.ResponseWriter, req *http.Request) {
conn, err := ws.Accept(resp, req, nil)
if err != nil {
w.logger.Debug("Failed to upgrade websocket connection")
return
}
wc := &WebClient{
conn: conn,
identifier: req.RemoteAddr,
logger: w.logger,
running: true,
}
w.lock.Lock()
w.connections[wc.identifier] = wc
w.lock.Unlock()
w.logger.Debug("Connection from: %v", wc.identifier)
go wc.Run(w)
}