2016-05-18 20:17:50 +02:00
|
|
|
// Copyright 2016 Hajime Hoshi
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2021-06-10 18:03:35 +02:00
|
|
|
//go:build android || ios
|
2017-01-25 17:32:33 +01:00
|
|
|
// +build android ios
|
2016-05-18 20:17:50 +02:00
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
package mobile
|
2016-05-18 20:17:50 +02:00
|
|
|
|
|
|
|
import (
|
2019-02-23 12:41:52 +01:00
|
|
|
"fmt"
|
|
|
|
"runtime/debug"
|
2017-12-31 13:01:48 +01:00
|
|
|
"sync"
|
2020-08-28 17:24:30 +02:00
|
|
|
"sync/atomic"
|
2020-02-22 16:26:48 +01:00
|
|
|
"unicode"
|
2016-05-19 16:37:58 +02:00
|
|
|
|
2018-04-06 19:14:23 +02:00
|
|
|
"golang.org/x/mobile/app"
|
2020-02-21 18:03:56 +01:00
|
|
|
"golang.org/x/mobile/event/key"
|
2018-04-06 19:14:23 +02:00
|
|
|
"golang.org/x/mobile/event/lifecycle"
|
|
|
|
"golang.org/x/mobile/event/paint"
|
|
|
|
"golang.org/x/mobile/event/size"
|
|
|
|
"golang.org/x/mobile/event/touch"
|
|
|
|
"golang.org/x/mobile/gl"
|
|
|
|
|
2020-10-03 19:35:13 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/devicescale"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/driver"
|
2022-02-04 07:47:22 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
2020-10-12 17:39:45 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/graphicscommand"
|
2020-10-03 19:35:13 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/opengl"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/hooks"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/restorable"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/thread"
|
2016-05-18 20:17:50 +02:00
|
|
|
)
|
|
|
|
|
2018-04-06 19:14:23 +02:00
|
|
|
var (
|
2019-12-16 16:18:16 +01:00
|
|
|
glContextCh = make(chan gl.Context, 1)
|
2019-05-30 18:44:01 +02:00
|
|
|
|
2019-07-31 18:07:19 +02:00
|
|
|
// renderCh receives when updating starts.
|
2019-05-30 18:44:01 +02:00
|
|
|
renderCh = make(chan struct{})
|
|
|
|
|
|
|
|
// renderEndCh receives when updating finishes.
|
|
|
|
renderEndCh = make(chan struct{})
|
|
|
|
|
2020-01-21 17:58:59 +01:00
|
|
|
theUI = &UserInterface{
|
2020-08-28 17:24:30 +02:00
|
|
|
foreground: 1,
|
2020-02-11 15:19:43 +01:00
|
|
|
errCh: make(chan error),
|
2020-02-11 14:58:59 +01:00
|
|
|
|
|
|
|
// Give a default outside size so that the game can start without initializing them.
|
|
|
|
outsideWidth: 640,
|
|
|
|
outsideHeight: 480,
|
|
|
|
sizeChanged: true,
|
2020-01-21 17:58:59 +01:00
|
|
|
}
|
2018-04-06 19:14:23 +02:00
|
|
|
)
|
|
|
|
|
2019-04-07 12:27:30 +02:00
|
|
|
func init() {
|
|
|
|
theUI.input.ui = theUI
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func Get() *UserInterface {
|
|
|
|
return theUI
|
|
|
|
}
|
|
|
|
|
2020-02-11 05:43:58 +01:00
|
|
|
// Update is called from mobile/ebitenmobileview.
|
2020-08-28 17:18:42 +02:00
|
|
|
//
|
|
|
|
// Update must be called on the rendering thread.
|
2020-02-11 15:19:43 +01:00
|
|
|
func (u *UserInterface) Update() error {
|
|
|
|
select {
|
|
|
|
case err := <-u.errCh:
|
|
|
|
return err
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
|
2020-03-20 16:08:40 +01:00
|
|
|
if !u.IsFocused() {
|
2020-02-11 15:19:43 +01:00
|
|
|
return nil
|
2020-01-21 17:58:59 +01:00
|
|
|
}
|
|
|
|
|
2022-02-04 07:47:22 +01:00
|
|
|
gamepad.Update()
|
2021-10-16 17:32:50 +02:00
|
|
|
|
2019-05-31 20:11:09 +02:00
|
|
|
renderCh <- struct{}{}
|
2020-08-29 19:10:49 +02:00
|
|
|
go func() {
|
|
|
|
<-renderEndCh
|
2022-01-02 19:30:29 +01:00
|
|
|
u.t.Stop()
|
2020-08-29 19:10:49 +02:00
|
|
|
}()
|
|
|
|
u.t.Loop()
|
2020-02-11 15:19:43 +01:00
|
|
|
return nil
|
2016-06-09 18:21:46 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
type UserInterface struct {
|
2020-02-11 09:53:50 +01:00
|
|
|
outsideWidth float64
|
|
|
|
outsideHeight float64
|
2019-12-16 02:57:57 +01:00
|
|
|
|
2016-06-18 17:55:24 +02:00
|
|
|
sizeChanged bool
|
2020-08-28 17:24:30 +02:00
|
|
|
foreground int32
|
2020-02-11 15:19:43 +01:00
|
|
|
errCh chan error
|
2017-12-31 13:01:48 +01:00
|
|
|
|
2018-03-23 17:07:36 +01:00
|
|
|
// Used for gomobile-build
|
2019-12-16 16:18:16 +01:00
|
|
|
gbuildWidthPx int
|
|
|
|
gbuildHeightPx int
|
|
|
|
setGBuildSizeCh chan struct{}
|
2020-01-08 04:50:57 +01:00
|
|
|
once sync.Once
|
2018-03-23 17:07:36 +01:00
|
|
|
|
2020-01-03 09:32:46 +01:00
|
|
|
context driver.UIContext
|
2019-06-26 17:07:39 +02:00
|
|
|
|
2019-04-07 11:55:56 +02:00
|
|
|
input Input
|
|
|
|
|
2021-07-22 18:07:28 +02:00
|
|
|
fpsMode driver.FPSMode
|
|
|
|
renderRequester RenderRequester
|
|
|
|
|
2020-10-20 19:55:17 +02:00
|
|
|
t *thread.OSThread
|
2019-06-26 17:07:39 +02:00
|
|
|
|
2017-12-31 13:01:48 +01:00
|
|
|
m sync.RWMutex
|
2016-05-18 20:17:50 +02:00
|
|
|
}
|
|
|
|
|
2019-08-18 19:32:35 +02:00
|
|
|
func deviceScale() float64 {
|
|
|
|
return devicescale.GetAt(0, 0)
|
2018-10-01 20:51:13 +02:00
|
|
|
}
|
|
|
|
|
2018-04-06 19:14:23 +02:00
|
|
|
// appMain is the main routine for gomobile-build mode.
|
2019-04-08 00:51:32 +02:00
|
|
|
func (u *UserInterface) appMain(a app.App) {
|
2018-04-06 19:14:23 +02:00
|
|
|
var glctx gl.Context
|
2019-12-16 16:18:16 +01:00
|
|
|
var sizeInited bool
|
2020-02-21 18:03:56 +01:00
|
|
|
|
2021-10-16 11:16:38 +02:00
|
|
|
touches := map[touch.Sequence]Touch{}
|
2020-02-21 18:03:56 +01:00
|
|
|
keys := map[driver.Key]struct{}{}
|
|
|
|
|
2018-04-06 19:14:23 +02:00
|
|
|
for e := range a.Events() {
|
2020-02-21 18:03:56 +01:00
|
|
|
var updateInput bool
|
2020-02-22 16:26:48 +01:00
|
|
|
var runes []rune
|
2020-02-21 18:03:56 +01:00
|
|
|
|
2018-04-06 19:14:23 +02:00
|
|
|
switch e := a.Filter(e).(type) {
|
|
|
|
case lifecycle.Event:
|
|
|
|
switch e.Crosses(lifecycle.StageVisible) {
|
|
|
|
case lifecycle.CrossOn:
|
2021-05-04 15:24:31 +02:00
|
|
|
if err := u.SetForeground(true); err != nil {
|
|
|
|
// There are no other ways than panicking here.
|
|
|
|
panic(err)
|
|
|
|
}
|
2020-08-19 16:59:23 +02:00
|
|
|
restorable.OnContextLost()
|
2018-04-06 19:14:23 +02:00
|
|
|
glctx, _ = e.DrawContext.(gl.Context)
|
|
|
|
// Assume that glctx is always a same instance.
|
|
|
|
// Then, only once initializing should be enough.
|
|
|
|
if glContextCh != nil {
|
|
|
|
glContextCh <- glctx
|
|
|
|
glContextCh = nil
|
|
|
|
}
|
|
|
|
a.Send(paint.Event{})
|
|
|
|
case lifecycle.CrossOff:
|
2021-05-04 15:24:31 +02:00
|
|
|
if err := u.SetForeground(false); err != nil {
|
|
|
|
// There are no other ways than panicking here.
|
|
|
|
panic(err)
|
|
|
|
}
|
2018-04-06 19:14:23 +02:00
|
|
|
glctx = nil
|
|
|
|
}
|
|
|
|
case size.Event:
|
2019-12-16 16:18:16 +01:00
|
|
|
u.setGBuildSize(e.WidthPx, e.HeightPx)
|
|
|
|
sizeInited = true
|
2018-04-06 19:14:23 +02:00
|
|
|
case paint.Event:
|
2019-12-16 16:18:16 +01:00
|
|
|
if !sizeInited {
|
|
|
|
a.Send(paint.Event{})
|
|
|
|
continue
|
|
|
|
}
|
2018-04-06 19:14:23 +02:00
|
|
|
if glctx == nil || e.External {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
renderCh <- struct{}{}
|
2019-05-30 18:44:01 +02:00
|
|
|
<-renderEndCh
|
2018-04-06 19:14:23 +02:00
|
|
|
a.Publish()
|
|
|
|
a.Send(paint.Event{})
|
|
|
|
case touch.Event:
|
2019-12-16 16:18:16 +01:00
|
|
|
if !sizeInited {
|
|
|
|
continue
|
|
|
|
}
|
2018-04-06 19:14:23 +02:00
|
|
|
switch e.Type {
|
|
|
|
case touch.TypeBegin, touch.TypeMove:
|
2019-08-18 19:32:35 +02:00
|
|
|
s := deviceScale()
|
2018-04-06 19:14:23 +02:00
|
|
|
x, y := float64(e.X)/s, float64(e.Y)/s
|
|
|
|
// TODO: Is it ok to cast from int64 to int here?
|
2021-10-16 11:16:38 +02:00
|
|
|
touches[e.Sequence] = Touch{
|
2020-10-09 20:36:14 +02:00
|
|
|
ID: driver.TouchID(e.Sequence),
|
2019-03-30 17:08:04 +01:00
|
|
|
X: int(x),
|
|
|
|
Y: int(y),
|
|
|
|
}
|
2018-04-06 19:14:23 +02:00
|
|
|
case touch.TypeEnd:
|
|
|
|
delete(touches, e.Sequence)
|
|
|
|
}
|
2020-02-21 18:03:56 +01:00
|
|
|
updateInput = true
|
|
|
|
case key.Event:
|
|
|
|
k, ok := gbuildKeyToDriverKey[e.Code]
|
2020-02-22 16:26:48 +01:00
|
|
|
if ok {
|
|
|
|
switch e.Direction {
|
|
|
|
case key.DirPress, key.DirNone:
|
|
|
|
keys[k] = struct{}{}
|
|
|
|
case key.DirRelease:
|
|
|
|
delete(keys, k)
|
|
|
|
}
|
2020-02-21 18:03:56 +01:00
|
|
|
}
|
2020-02-22 16:26:48 +01:00
|
|
|
|
2020-02-21 18:03:56 +01:00
|
|
|
switch e.Direction {
|
|
|
|
case key.DirPress, key.DirNone:
|
2020-02-22 16:26:48 +01:00
|
|
|
if e.Rune != -1 && unicode.IsPrint(e.Rune) {
|
|
|
|
runes = []rune{e.Rune}
|
|
|
|
}
|
2020-02-21 18:03:56 +01:00
|
|
|
}
|
|
|
|
updateInput = true
|
|
|
|
}
|
|
|
|
|
|
|
|
if updateInput {
|
2021-10-16 11:16:38 +02:00
|
|
|
var ts []Touch
|
2018-04-06 19:14:23 +02:00
|
|
|
for _, t := range touches {
|
|
|
|
ts = append(ts, t)
|
|
|
|
}
|
2021-10-16 13:37:01 +02:00
|
|
|
u.input.update(keys, runes, ts)
|
2018-04-06 19:14:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-05-19 16:37:58 +02:00
|
|
|
|
2021-05-04 15:24:31 +02:00
|
|
|
func (u *UserInterface) SetForeground(foreground bool) error {
|
2020-08-28 17:24:30 +02:00
|
|
|
var v int32
|
|
|
|
if foreground {
|
|
|
|
v = 1
|
|
|
|
}
|
|
|
|
atomic.StoreInt32(&u.foreground, v)
|
2020-01-21 17:58:59 +01:00
|
|
|
|
|
|
|
if foreground {
|
2021-05-04 15:24:31 +02:00
|
|
|
return hooks.ResumeAudio()
|
2020-01-21 17:58:59 +01:00
|
|
|
} else {
|
2021-05-04 15:24:31 +02:00
|
|
|
return hooks.SuspendAudio()
|
2020-01-21 17:58:59 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-03 09:32:46 +01:00
|
|
|
func (u *UserInterface) Run(context driver.UIContext) error {
|
2019-12-16 16:18:16 +01:00
|
|
|
u.setGBuildSizeCh = make(chan struct{})
|
2019-04-10 03:44:02 +02:00
|
|
|
go func() {
|
2020-02-11 09:53:50 +01:00
|
|
|
if err := u.run(context, true); err != nil {
|
2019-04-10 03:44:02 +02:00
|
|
|
// As mobile apps never ends, Loop can't return. Just panic here.
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
app.Main(u.appMain)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-02-11 15:19:43 +01:00
|
|
|
func (u *UserInterface) RunWithoutMainLoop(context driver.UIContext) {
|
2019-04-10 03:44:02 +02:00
|
|
|
go func() {
|
2020-02-11 09:53:50 +01:00
|
|
|
if err := u.run(context, false); err != nil {
|
2020-02-11 15:19:43 +01:00
|
|
|
u.errCh <- err
|
2019-04-10 03:44:02 +02:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-02-11 09:53:50 +01:00
|
|
|
func (u *UserInterface) run(context driver.UIContext, mainloop bool) (err error) {
|
2019-02-23 12:41:52 +01:00
|
|
|
// Convert the panic to a regular error so that Java/Objective-C layer can treat this easily e.g., for
|
|
|
|
// Crashlytics. A panic is treated as SIGABRT, and there is no way to handle this on Java/Objective-C layer
|
|
|
|
// unfortunately.
|
|
|
|
// TODO: Panic on other goroutines cannot be handled here.
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
2021-01-01 18:08:34 +01:00
|
|
|
err = fmt.Errorf("%v\n%s", r, string(debug.Stack()))
|
2019-02-23 12:41:52 +01:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2017-12-31 13:01:48 +01:00
|
|
|
u.m.Lock()
|
|
|
|
u.sizeChanged = true
|
|
|
|
u.m.Unlock()
|
2018-03-23 17:07:36 +01:00
|
|
|
|
2020-08-28 20:02:20 +02:00
|
|
|
u.context = context
|
|
|
|
|
2020-10-18 11:28:23 +02:00
|
|
|
if mainloop {
|
|
|
|
// When mainloop is true, gomobile-build is used. In this case, GL functions must be called via
|
|
|
|
// gl.Context so that they are called on the appropriate thread.
|
|
|
|
ctx := <-glContextCh
|
|
|
|
u.Graphics().(*opengl.Graphics).SetGomobileGLContext(ctx)
|
2019-09-02 21:08:58 +02:00
|
|
|
} else {
|
2020-10-20 19:55:17 +02:00
|
|
|
u.t = thread.NewOSThread()
|
2020-10-12 17:39:45 +02:00
|
|
|
graphicscommand.SetMainThread(u.t)
|
2018-04-06 19:14:23 +02:00
|
|
|
}
|
2018-03-23 17:07:36 +01:00
|
|
|
|
2019-12-16 16:18:16 +01:00
|
|
|
// If gomobile-build is used, wait for the outside size fixed.
|
|
|
|
if u.setGBuildSizeCh != nil {
|
|
|
|
<-u.setGBuildSizeCh
|
|
|
|
}
|
|
|
|
|
2018-05-26 15:50:58 +02:00
|
|
|
// Force to set the screen size
|
2020-08-28 19:43:04 +02:00
|
|
|
u.layoutIfNeeded()
|
2016-08-31 19:38:47 +02:00
|
|
|
for {
|
2020-04-02 17:06:42 +02:00
|
|
|
if err := u.update(); err != nil {
|
2016-08-31 19:38:47 +02:00
|
|
|
return err
|
|
|
|
}
|
2016-09-01 18:34:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-28 19:43:04 +02:00
|
|
|
// layoutIfNeeded must be called on the same goroutine as update().
|
|
|
|
func (u *UserInterface) layoutIfNeeded() {
|
2020-02-11 09:53:50 +01:00
|
|
|
var outsideWidth, outsideHeight float64
|
2017-12-31 13:01:48 +01:00
|
|
|
|
2020-08-28 20:02:20 +02:00
|
|
|
u.m.RLock()
|
2018-05-11 19:29:39 +02:00
|
|
|
sizeChanged := u.sizeChanged
|
2017-12-31 13:01:48 +01:00
|
|
|
if sizeChanged {
|
2019-12-09 18:37:10 +01:00
|
|
|
if u.gbuildWidthPx == 0 || u.gbuildHeightPx == 0 {
|
2020-02-11 09:53:50 +01:00
|
|
|
outsideWidth = u.outsideWidth
|
|
|
|
outsideHeight = u.outsideHeight
|
2019-12-09 18:37:10 +01:00
|
|
|
} else {
|
|
|
|
// gomobile build
|
|
|
|
d := deviceScale()
|
2020-02-11 09:53:50 +01:00
|
|
|
outsideWidth = float64(u.gbuildWidthPx) / d
|
|
|
|
outsideHeight = float64(u.gbuildHeightPx) / d
|
2019-12-09 18:37:10 +01:00
|
|
|
}
|
2017-12-31 13:01:48 +01:00
|
|
|
}
|
|
|
|
u.sizeChanged = false
|
2020-08-28 20:02:20 +02:00
|
|
|
u.m.RUnlock()
|
2017-12-31 13:01:48 +01:00
|
|
|
|
|
|
|
if sizeChanged {
|
2020-04-02 17:06:42 +02:00
|
|
|
u.context.Layout(outsideWidth, outsideHeight)
|
2016-08-31 19:38:47 +02:00
|
|
|
}
|
2018-02-01 18:08:03 +01:00
|
|
|
}
|
|
|
|
|
2020-04-02 17:06:42 +02:00
|
|
|
func (u *UserInterface) update() error {
|
2020-01-21 17:58:59 +01:00
|
|
|
<-renderCh
|
2018-02-01 18:08:03 +01:00
|
|
|
defer func() {
|
2019-05-30 18:44:01 +02:00
|
|
|
renderEndCh <- struct{}{}
|
2018-02-01 18:08:03 +01:00
|
|
|
}()
|
|
|
|
|
2021-08-04 18:29:45 +02:00
|
|
|
if err := u.context.UpdateFrame(); err != nil {
|
2016-09-01 18:34:51 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
2016-05-18 20:17:50 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) ScreenSizeInFullscreen() (int, int) {
|
2019-12-12 18:50:57 +01:00
|
|
|
// TODO: This function should return gbuildWidthPx, gbuildHeightPx,
|
2018-05-04 09:09:55 +02:00
|
|
|
// but these values are not initialized until the main loop starts.
|
|
|
|
return 0, 0
|
|
|
|
}
|
|
|
|
|
2020-02-11 09:53:50 +01:00
|
|
|
// SetOutsideSize is called from mobile/ebitenmobileview.
|
2020-08-28 17:18:42 +02:00
|
|
|
//
|
|
|
|
// SetOutsideSize is concurrent safe.
|
2020-02-11 09:53:50 +01:00
|
|
|
func (u *UserInterface) SetOutsideSize(outsideWidth, outsideHeight float64) {
|
2017-12-31 13:01:48 +01:00
|
|
|
u.m.Lock()
|
2020-02-11 09:53:50 +01:00
|
|
|
if u.outsideWidth != outsideWidth || u.outsideHeight != outsideHeight {
|
|
|
|
u.outsideWidth = outsideWidth
|
|
|
|
u.outsideHeight = outsideHeight
|
2017-12-31 13:01:48 +01:00
|
|
|
u.sizeChanged = true
|
|
|
|
}
|
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-12-16 16:18:16 +01:00
|
|
|
func (u *UserInterface) setGBuildSize(widthPx, heightPx int) {
|
2018-03-23 17:07:36 +01:00
|
|
|
u.m.Lock()
|
2019-12-12 18:50:57 +01:00
|
|
|
u.gbuildWidthPx = widthPx
|
|
|
|
u.gbuildHeightPx = heightPx
|
2018-03-23 17:07:36 +01:00
|
|
|
u.sizeChanged = true
|
2020-08-28 19:43:04 +02:00
|
|
|
u.m.Unlock()
|
|
|
|
|
2020-01-08 04:50:57 +01:00
|
|
|
u.once.Do(func() {
|
|
|
|
close(u.setGBuildSizeCh)
|
|
|
|
})
|
2018-03-23 17:07:36 +01:00
|
|
|
}
|
|
|
|
|
2019-12-16 16:18:16 +01:00
|
|
|
func (u *UserInterface) adjustPosition(x, y int) (int, int) {
|
2020-10-16 22:16:02 +02:00
|
|
|
xf, yf := u.context.AdjustPosition(float64(x), float64(y), deviceScale())
|
2019-12-16 16:18:16 +01:00
|
|
|
return int(xf), int(yf)
|
2017-06-30 21:27:38 +02:00
|
|
|
}
|
|
|
|
|
2019-12-14 04:30:03 +01:00
|
|
|
func (u *UserInterface) CursorMode() driver.CursorMode {
|
|
|
|
return driver.CursorModeHidden
|
2017-08-12 08:39:41 +02:00
|
|
|
}
|
|
|
|
|
2019-12-14 04:30:03 +01:00
|
|
|
func (u *UserInterface) SetCursorMode(mode driver.CursorMode) {
|
2016-09-03 10:17:54 +02:00
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2021-04-11 07:21:38 +02:00
|
|
|
func (u *UserInterface) CursorShape() driver.CursorShape {
|
|
|
|
return driver.CursorShapeDefault
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) SetCursorShape(shape driver.CursorShape) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) IsFullscreen() bool {
|
2017-09-13 20:37:38 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetFullscreen(fullscreen bool) {
|
2017-06-29 17:35:34 +02:00
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2020-03-20 16:08:40 +01:00
|
|
|
func (u *UserInterface) IsFocused() bool {
|
2020-08-28 17:24:30 +02:00
|
|
|
return atomic.LoadInt32(&u.foreground) != 0
|
2020-01-16 02:47:23 +01:00
|
|
|
}
|
|
|
|
|
2020-03-20 16:34:12 +01:00
|
|
|
func (u *UserInterface) IsRunnableOnUnfocused() bool {
|
2017-06-29 17:35:34 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-03-20 16:34:12 +01:00
|
|
|
func (u *UserInterface) SetRunnableOnUnfocused(runnableOnUnfocused bool) {
|
2017-08-02 16:37:50 +02:00
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2021-07-22 16:55:26 +02:00
|
|
|
func (u *UserInterface) FPSMode() driver.FPSMode {
|
2021-07-22 18:07:28 +02:00
|
|
|
return u.fpsMode
|
2018-07-14 13:50:09 +02:00
|
|
|
}
|
|
|
|
|
2021-07-22 16:55:26 +02:00
|
|
|
func (u *UserInterface) SetFPSMode(mode driver.FPSMode) {
|
2021-07-22 18:07:28 +02:00
|
|
|
u.fpsMode = mode
|
|
|
|
u.updateExplicitRenderingModeIfNeeded()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) updateExplicitRenderingModeIfNeeded() {
|
|
|
|
if u.renderRequester == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
u.renderRequester.SetExplicitRenderingMode(u.fpsMode == driver.FPSModeVsyncOffMinimum)
|
2018-07-14 13:50:09 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) DeviceScaleFactor() float64 {
|
2019-08-18 19:32:35 +02:00
|
|
|
return deviceScale()
|
2018-10-07 18:24:19 +02:00
|
|
|
}
|
2019-04-07 11:28:50 +02:00
|
|
|
|
2019-11-30 16:07:41 +01:00
|
|
|
func (u *UserInterface) SetScreenTransparent(transparent bool) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) IsScreenTransparent() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-04-02 17:06:42 +02:00
|
|
|
func (u *UserInterface) ResetForFrame() {
|
2020-08-28 19:43:04 +02:00
|
|
|
u.layoutIfNeeded()
|
2020-04-02 17:06:42 +02:00
|
|
|
u.input.resetForFrame()
|
|
|
|
}
|
|
|
|
|
2020-02-09 15:03:03 +01:00
|
|
|
func (u *UserInterface) SetInitFocused(focused bool) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2019-04-07 11:28:50 +02:00
|
|
|
func (u *UserInterface) Input() driver.Input {
|
2019-04-07 11:55:56 +02:00
|
|
|
return &u.input
|
|
|
|
}
|
|
|
|
|
2019-12-24 16:05:56 +01:00
|
|
|
func (u *UserInterface) Window() driver.Window {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-04-08 01:21:17 +02:00
|
|
|
type Touch struct {
|
2020-10-09 20:36:14 +02:00
|
|
|
ID driver.TouchID
|
2019-04-08 01:21:17 +02:00
|
|
|
X int
|
|
|
|
Y int
|
|
|
|
}
|
|
|
|
|
2021-10-16 13:37:01 +02:00
|
|
|
func (u *UserInterface) UpdateInput(keys map[driver.Key]struct{}, runes []rune, touches []Touch) {
|
|
|
|
u.input.update(keys, runes, touches)
|
|
|
|
if u.fpsMode == driver.FPSModeVsyncOffMinimum {
|
|
|
|
u.renderRequester.RequestRenderIfNeeded()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-22 18:07:28 +02:00
|
|
|
type RenderRequester interface {
|
|
|
|
SetExplicitRenderingMode(explicitRendering bool)
|
|
|
|
RequestRenderIfNeeded()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) SetRenderRequester(renderRequester RenderRequester) {
|
|
|
|
u.renderRequester = renderRequester
|
|
|
|
u.updateExplicitRenderingModeIfNeeded()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) ScheduleFrame() {
|
|
|
|
if u.renderRequester != nil && u.fpsMode == driver.FPSModeVsyncOffMinimum {
|
|
|
|
u.renderRequester.RequestRenderIfNeeded()
|
|
|
|
}
|
2019-04-07 11:28:50 +02:00
|
|
|
}
|