forked from joejulian/gio
09b5752659
Signed-off-by: Elias Naur <mail@eliasnaur.com>
303 lines
6.2 KiB
Go
303 lines
6.2 KiB
Go
// SPDX-License-Identifier: Unlicense OR MIT
|
|
|
|
package app
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"image"
|
|
"time"
|
|
|
|
"gioui.org/ui"
|
|
"gioui.org/ui/app/internal/gpu"
|
|
iinput "gioui.org/ui/app/internal/input"
|
|
"gioui.org/ui/input"
|
|
"gioui.org/ui/key"
|
|
"gioui.org/ui/system"
|
|
)
|
|
|
|
type WindowOptions struct {
|
|
Width ui.Value
|
|
Height ui.Value
|
|
Title string
|
|
}
|
|
|
|
type Window struct {
|
|
driver *window
|
|
lastFrame time.Time
|
|
drawStart time.Time
|
|
gpu *gpu.GPU
|
|
inputState key.TextInputState
|
|
|
|
out chan Event
|
|
in chan Event
|
|
ack chan struct{}
|
|
redraws chan struct{}
|
|
frames chan *ui.Ops
|
|
|
|
stage Stage
|
|
animating bool
|
|
hasNextFrame bool
|
|
nextFrame time.Time
|
|
delayedDraw *time.Timer
|
|
|
|
router iinput.Router
|
|
}
|
|
|
|
// driverEvent is sent when a new native driver
|
|
// is available for the Window.
|
|
type driverEvent struct {
|
|
driver *window
|
|
}
|
|
|
|
// driver is the interface for the platform implementation
|
|
// of a Window.
|
|
var _ interface {
|
|
// setAnimating sets the animation flag. When the window is animating,
|
|
// DrawEvents are delivered as fast as the display can handle them.
|
|
setAnimating(anim bool)
|
|
// setTextInput updates the virtual keyboard state.
|
|
setTextInput(s key.TextInputState)
|
|
} = (*window)(nil)
|
|
|
|
// Pre-allocate the ack event to avoid garbage.
|
|
var ackEvent Event
|
|
|
|
// NewWindow creates a new window for a set of window
|
|
// options. The options are hints; the platform is free to
|
|
// ignore or adjust them.
|
|
// If the current program is running on iOS and Android,
|
|
// NewWindow returns the window previously by the platform.
|
|
func NewWindow(opts *WindowOptions) *Window {
|
|
if opts == nil {
|
|
opts = &WindowOptions{
|
|
Width: ui.Dp(800),
|
|
Height: ui.Dp(600),
|
|
Title: "Gio program",
|
|
}
|
|
}
|
|
if opts.Width.V <= 0 || opts.Height.V <= 0 {
|
|
panic("window width and height must be larger than 0")
|
|
}
|
|
|
|
w := &Window{
|
|
in: make(chan Event),
|
|
out: make(chan Event),
|
|
ack: make(chan struct{}),
|
|
redraws: make(chan struct{}, 1),
|
|
frames: make(chan *ui.Ops),
|
|
}
|
|
go w.run(opts)
|
|
return w
|
|
}
|
|
|
|
func (w *Window) Events() <-chan Event {
|
|
return w.out
|
|
}
|
|
|
|
func (w *Window) setTextInput(s key.TextInputState) {
|
|
if s != w.inputState && (s == key.TextInputClose || s == key.TextInputOpen) {
|
|
w.driver.setTextInput(s)
|
|
}
|
|
if s == key.TextInputFocus {
|
|
w.setNextFrame(time.Time{})
|
|
w.updateAnimation()
|
|
}
|
|
w.inputState = s
|
|
}
|
|
|
|
func (w *Window) Queue() input.Queue {
|
|
return &w.router
|
|
}
|
|
|
|
func (w *Window) Draw(frame *ui.Ops) {
|
|
w.frames <- frame
|
|
}
|
|
|
|
func (w *Window) draw(size image.Point, frame *ui.Ops) {
|
|
var drawDur time.Duration
|
|
if !w.drawStart.IsZero() {
|
|
drawDur = time.Since(w.drawStart)
|
|
w.drawStart = time.Time{}
|
|
}
|
|
w.gpu.Draw(w.router.Profiling(), size, frame)
|
|
w.router.Frame(frame)
|
|
now := time.Now()
|
|
w.setTextInput(w.router.InputState())
|
|
frameDur := now.Sub(w.lastFrame)
|
|
frameDur = frameDur.Truncate(100 * time.Microsecond)
|
|
w.lastFrame = now
|
|
if w.router.Profiling() {
|
|
q := 100 * time.Microsecond
|
|
timings := fmt.Sprintf("tot:%7s cpu:%7s %s", frameDur.Round(q), drawDur.Round(q), w.gpu.Timings())
|
|
w.router.AddProfile(system.ProfileEvent{Timings: timings})
|
|
w.setNextFrame(time.Time{})
|
|
}
|
|
if t, ok := w.router.RedrawTime(); ok {
|
|
w.setNextFrame(t)
|
|
}
|
|
w.updateAnimation()
|
|
}
|
|
|
|
func (w *Window) Redraw() {
|
|
select {
|
|
case w.redraws <- struct{}{}:
|
|
default:
|
|
}
|
|
}
|
|
|
|
func (w *Window) updateAnimation() {
|
|
animate := false
|
|
if w.delayedDraw != nil {
|
|
w.delayedDraw.Stop()
|
|
w.delayedDraw = nil
|
|
}
|
|
if w.stage >= StageRunning && w.hasNextFrame {
|
|
if dt := time.Until(w.nextFrame); dt <= 0 {
|
|
animate = true
|
|
} else {
|
|
w.delayedDraw = time.NewTimer(dt)
|
|
}
|
|
}
|
|
if animate != w.animating {
|
|
w.animating = animate
|
|
w.driver.setAnimating(animate)
|
|
}
|
|
}
|
|
|
|
func (w *Window) setNextFrame(at time.Time) {
|
|
if !w.hasNextFrame || at.Before(w.nextFrame) {
|
|
w.hasNextFrame = true
|
|
w.nextFrame = at
|
|
}
|
|
}
|
|
|
|
func (w *Window) setDriver(d *window) {
|
|
w.event(driverEvent{d})
|
|
}
|
|
|
|
func (w *Window) event(e Event) {
|
|
w.in <- e
|
|
<-w.ack
|
|
}
|
|
|
|
func (w *Window) waitAck() {
|
|
// Send a dummy event; when it gets through we
|
|
// know the application has processed the previous event.
|
|
w.out <- ackEvent
|
|
}
|
|
|
|
// Prematurely destroy the window and wait for the native window
|
|
// destroy event.
|
|
func (w *Window) destroy(err error) {
|
|
// Ack the current event.
|
|
w.ack <- struct{}{}
|
|
w.out <- DestroyEvent{err}
|
|
for e := range w.in {
|
|
w.ack <- struct{}{}
|
|
if _, ok := e.(DestroyEvent); ok {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (w *Window) run(opts *WindowOptions) {
|
|
defer close(w.in)
|
|
defer close(w.out)
|
|
if err := createWindow(w, opts); err != nil {
|
|
w.destroy(err)
|
|
return
|
|
}
|
|
for {
|
|
var timer <-chan time.Time
|
|
if w.delayedDraw != nil {
|
|
timer = w.delayedDraw.C
|
|
}
|
|
select {
|
|
case <-timer:
|
|
w.setNextFrame(time.Time{})
|
|
w.updateAnimation()
|
|
case <-w.redraws:
|
|
w.setNextFrame(time.Time{})
|
|
w.updateAnimation()
|
|
case e := <-w.in:
|
|
switch e2 := e.(type) {
|
|
case StageEvent:
|
|
if w.gpu != nil {
|
|
if e2.Stage < StageRunning {
|
|
w.gpu.Release()
|
|
w.gpu = nil
|
|
} else {
|
|
w.gpu.Refresh()
|
|
}
|
|
}
|
|
w.stage = e2.Stage
|
|
w.updateAnimation()
|
|
w.out <- e
|
|
w.waitAck()
|
|
case DrawEvent:
|
|
if e2.Size == (image.Point{}) {
|
|
panic(errors.New("internal error: zero-sized Draw"))
|
|
}
|
|
if w.stage < StageRunning {
|
|
// No drawing if not visible.
|
|
break
|
|
}
|
|
w.drawStart = time.Now()
|
|
w.hasNextFrame = false
|
|
w.out <- e
|
|
frame := <-w.frames
|
|
if w.gpu != nil {
|
|
if e2.sync {
|
|
w.gpu.Refresh()
|
|
}
|
|
if err := w.gpu.Flush(); err != nil {
|
|
w.gpu.Release()
|
|
w.gpu = nil
|
|
w.destroy(err)
|
|
return
|
|
}
|
|
} else {
|
|
ctx, err := newContext(w.driver)
|
|
if err != nil {
|
|
w.destroy(err)
|
|
return
|
|
}
|
|
w.gpu, err = gpu.NewGPU(ctx)
|
|
if err != nil {
|
|
w.destroy(err)
|
|
return
|
|
}
|
|
}
|
|
w.draw(e2.Size, frame)
|
|
if e2.sync {
|
|
if err := w.gpu.Flush(); err != nil {
|
|
w.gpu.Release()
|
|
w.gpu = nil
|
|
w.destroy(err)
|
|
return
|
|
}
|
|
}
|
|
case *CommandEvent:
|
|
w.out <- e
|
|
w.waitAck()
|
|
case input.Event:
|
|
if w.router.Add(e2) {
|
|
w.setNextFrame(time.Time{})
|
|
w.updateAnimation()
|
|
}
|
|
w.out <- e
|
|
case driverEvent:
|
|
w.driver = e2.driver
|
|
case DestroyEvent:
|
|
w.out <- e2
|
|
w.ack <- struct{}{}
|
|
return
|
|
}
|
|
w.ack <- struct{}{}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (_ driverEvent) ImplementsEvent() {}
|