ebiten/internal/uidriver/mobile/ui.go

421 lines
9.1 KiB
Go
Raw Normal View History

// 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.
2017-01-25 17:32:33 +01:00
// +build android ios
package mobile
import (
"context"
"fmt"
"runtime/debug"
"sync"
"time"
2016-05-19 16:37:58 +02:00
"golang.org/x/mobile/app"
"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"
2018-01-02 21:22:56 +01:00
"github.com/hajimehoshi/ebiten/internal/devicescale"
"github.com/hajimehoshi/ebiten/internal/driver"
"github.com/hajimehoshi/ebiten/internal/graphicsdriver/opengl"
"github.com/hajimehoshi/ebiten/internal/hooks"
"github.com/hajimehoshi/ebiten/internal/thread"
)
var (
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{})
theUI = &UserInterface{}
)
2019-04-07 12:27:30 +02:00
func init() {
theUI.input.ui = theUI
}
func Get() *UserInterface {
return theUI
}
func (u *UserInterface) Update() {
renderCh <- struct{}{}
ctx, cancel := context.WithCancel(context.Background())
go func() {
<-renderEndCh
if u.t != nil {
u.t.Call(func() error {
cancel()
return nil
})
} else {
cancel()
}
}()
if u.Graphics().IsGL() {
if u.glWorker == nil {
panic("mobile: glWorker must be initialized but not")
}
workAvailable := u.glWorker.WorkAvailable()
loop:
for {
select {
case <-workAvailable:
u.glWorker.DoWork()
case <-ctx.Done():
break loop
}
}
return
} else {
u.t.Loop(ctx)
}
}
type UserInterface struct {
2019-12-16 02:57:57 +01:00
// TODO: Remove these members: the driver layer should not care about the game screen size.
width int
height int
scale float64
2016-06-18 17:55:24 +02:00
sizeChanged bool
2018-03-23 17:07:36 +01:00
// Used for gomobile-build
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
context driver.UIContext
input Input
t *thread.Thread
glWorker gl.Worker
m sync.RWMutex
}
func deviceScale() float64 {
return devicescale.GetAt(0, 0)
2018-10-01 20:51:13 +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) {
var glctx gl.Context
var sizeInited bool
2019-04-08 01:21:17 +02:00
touches := map[touch.Sequence]*Touch{}
for e := range a.Events() {
switch e := a.Filter(e).(type) {
case lifecycle.Event:
switch e.Crosses(lifecycle.StageVisible) {
case lifecycle.CrossOn:
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:
glctx = nil
}
case size.Event:
u.setGBuildSize(e.WidthPx, e.HeightPx)
sizeInited = true
case paint.Event:
if !sizeInited {
a.Send(paint.Event{})
continue
}
if glctx == nil || e.External {
continue
}
renderCh <- struct{}{}
2019-05-30 18:44:01 +02:00
<-renderEndCh
a.Publish()
a.Send(paint.Event{})
case touch.Event:
if !sizeInited {
continue
}
switch e.Type {
case touch.TypeBegin, touch.TypeMove:
s := deviceScale()
x, y := float64(e.X)/s, float64(e.Y)/s
// TODO: Is it ok to cast from int64 to int here?
2019-04-08 01:21:17 +02:00
touches[e.Sequence] = &Touch{
ID: int(e.Sequence),
X: int(x),
Y: int(y),
}
case touch.TypeEnd:
delete(touches, e.Sequence)
}
2019-04-08 01:21:17 +02:00
ts := []*Touch{}
for _, t := range touches {
ts = append(ts, t)
}
2019-04-08 00:51:32 +02:00
u.input.update(ts)
}
}
}
2016-05-19 16:37:58 +02:00
func (u *UserInterface) Run(context driver.UIContext) error {
// TODO: Remove width/height/scale arguments. They are not used from gomobile-build.
u.setGBuildSizeCh = make(chan struct{})
go func() {
if err := u.run(16, 16, 1, context, true); err != nil {
// As mobile apps never ends, Loop can't return. Just panic here.
panic(err)
}
}()
app.Main(u.appMain)
return nil
}
func (u *UserInterface) RunWithoutMainLoop(width, height int, scale float64, title string, context driver.UIContext) <-chan error {
ch := make(chan error)
go func() {
defer close(ch)
// title is ignored?
if err := u.run(width, height, scale, context, false); err != nil {
ch <- err
}
}()
return ch
}
func (u *UserInterface) run(width, height int, scale float64, context driver.UIContext, mainloop bool) (err error) {
// 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 {
err = fmt.Errorf("%v\n%q", r, string(debug.Stack()))
}
}()
u.m.Lock()
2016-05-19 16:37:58 +02:00
u.width = width
u.height = height
u.scale = scale
u.sizeChanged = true
u.context = context
u.m.Unlock()
2018-03-23 17:07:36 +01:00
if u.Graphics().IsGL() {
var ctx gl.Context
if mainloop {
ctx = <-glContextCh
} else {
ctx, u.glWorker = gl.NewContext()
}
u.Graphics().(*opengl.Driver).SetMobileGLContext(ctx)
} else {
u.t = thread.New()
u.Graphics().SetThread(u.t)
}
2018-03-23 17:07:36 +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
u.updateSize(context)
for {
if err := u.update(context); err != nil {
return err
}
2016-09-01 18:34:51 +02:00
}
}
func (u *UserInterface) updateSize(context driver.UIContext) {
var width, height float64
u.m.Lock()
2018-05-11 19:29:39 +02:00
sizeChanged := u.sizeChanged
if sizeChanged {
if u.gbuildWidthPx == 0 || u.gbuildHeightPx == 0 {
s := u.scale
width = float64(u.width) * s
height = float64(u.height) * s
} else {
// gomobile build
d := deviceScale()
width = float64(u.gbuildWidthPx) / d
height = float64(u.gbuildHeightPx) / d
}
}
u.sizeChanged = false
u.m.Unlock()
if sizeChanged {
// Dirty hack to set the offscreen size for gomobile-bind.
// TODO: Remove this. The layouting logic must be in the package ebiten, not here.
if u.gbuildWidthPx == 0 || u.gbuildHeightPx == 0 {
context.(interface {
SetScreenSize(width, height int)
}).SetScreenSize(u.width, u.height)
}
2016-09-01 18:34:51 +02:00
// Sizing also calls GL functions
context.Layout(width, height)
}
}
func (u *UserInterface) update(context driver.UIContext) error {
t := time.NewTimer(500 * time.Millisecond)
defer t.Stop()
select {
case <-renderCh:
case <-t.C:
hooks.SuspendAudio()
<-renderCh
}
hooks.ResumeAudio()
defer func() {
2019-05-30 18:44:01 +02:00
renderEndCh <- struct{}{}
}()
if err := context.Update(func() {
u.updateSize(context)
}); err != nil {
2016-09-01 18:34:51 +02:00
return err
}
return nil
}
func (u *UserInterface) ScreenSizeInFullscreen() (int, int) {
// 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
}
func (u *UserInterface) SetScreenSizeAndScale(width, height int, scale float64) {
// Called from ebitenmobileview.
u.m.Lock()
if u.width != width || u.height != height || u.scale != scale {
u.width = width
u.height = height
u.scale = scale
u.sizeChanged = true
}
u.m.Unlock()
}
func (u *UserInterface) setGBuildSize(widthPx, heightPx int) {
2018-03-23 17:07:36 +01:00
u.m.Lock()
u.gbuildWidthPx = widthPx
u.gbuildHeightPx = heightPx
2018-03-23 17:07:36 +01:00
u.sizeChanged = true
2020-01-08 04:50:57 +01:00
u.once.Do(func() {
close(u.setGBuildSizeCh)
})
2018-03-23 17:07:36 +01:00
u.m.Unlock()
}
func (u *UserInterface) adjustPosition(x, y int) (int, int) {
// This function's caller already protects this function by the mutex.
if u.gbuildWidthPx == 0 || u.gbuildHeightPx == 0 {
s := u.scale
return int(float64(x) / s), int(float64(y) / s)
2018-03-23 17:07:36 +01:00
}
xf, yf := u.context.AdjustPosition(float64(x), float64(y))
return int(xf), int(yf)
}
func (u *UserInterface) CursorMode() driver.CursorMode {
return driver.CursorModeHidden
2017-08-12 08:39:41 +02:00
}
func (u *UserInterface) SetCursorMode(mode driver.CursorMode) {
2016-09-03 10:17:54 +02:00
// Do nothing
}
func (u *UserInterface) IsFullscreen() bool {
return false
}
func (u *UserInterface) SetFullscreen(fullscreen bool) {
// Do nothing
}
func (u *UserInterface) IsRunnableInBackground() bool {
return false
}
func (u *UserInterface) SetRunnableInBackground(runnableInBackground bool) {
// Do nothing
}
func (u *UserInterface) IsVsyncEnabled() bool {
return true
}
func (u *UserInterface) SetVsyncEnabled(enabled bool) {
// Do nothing
}
func (u *UserInterface) DeviceScaleFactor() float64 {
return deviceScale()
}
2019-04-07 11:28:50 +02:00
func (u *UserInterface) SetScreenTransparent(transparent bool) {
// Do nothing
}
func (u *UserInterface) IsScreenTransparent() bool {
return false
}
func (u *UserInterface) MonitorPosition() (int, int) {
return 0, 0
}
2019-04-07 11:28:50 +02:00
func (u *UserInterface) Input() driver.Input {
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 {
ID int
X int
Y int
}
func (u *UserInterface) UpdateInput(touches []*Touch) {
u.input.update(touches)
2019-04-07 11:28:50 +02:00
}