ebiten/internal/ui/ui_mobile.go

301 lines
6.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 ui
import (
2016-05-19 16:37:58 +02:00
"errors"
2017-09-22 21:12:02 +02:00
"image"
2016-06-30 17:44:15 +02:00
"runtime"
"sync"
"time"
2016-05-19 16:37:58 +02:00
2018-01-02 21:22:56 +01:00
"github.com/hajimehoshi/ebiten/internal/devicescale"
2018-04-01 16:20:45 +02:00
"github.com/hajimehoshi/ebiten/internal/input"
"github.com/hajimehoshi/ebiten/internal/opengl"
)
2016-05-19 16:37:58 +02:00
func Render(chError <-chan error) error {
2016-06-30 17:44:15 +02:00
runtime.LockOSThread()
defer runtime.UnlockOSThread()
2016-05-19 16:37:58 +02:00
if chError == nil {
return errors.New("ui: chError must not be nil")
}
// TODO: Check this is called on the rendering thread
select {
case chRender <- struct{}{}:
return opengl.GetContext().DoWork(chError, chRenderEnd)
case <-time.After(500 * time.Millisecond):
2016-06-10 17:56:29 +02:00
// This function must not be blocked. We need to break for timeout.
return nil
}
}
type userInterface struct {
2016-06-18 17:55:24 +02:00
width int
height int
2016-06-18 19:59:17 +02:00
scale float64
2016-06-18 17:55:24 +02:00
sizeChanged bool
2018-03-23 17:07:36 +01:00
// Used for gomobile-build
fullscreenScale float64
fullscreenWidthPx int
fullscreenHeightPx int
m sync.RWMutex
}
2016-05-19 16:37:58 +02:00
var (
chRender = make(chan struct{})
chRenderEnd = make(chan struct{})
currentUI = &userInterface{}
2016-05-19 16:37:58 +02:00
)
2016-09-02 17:20:05 +02:00
func Run(width, height int, scale float64, title string, g GraphicsContext) error {
u := currentUI
u.m.Lock()
2016-05-19 16:37:58 +02:00
u.width = width
u.height = height
u.scale = scale
u.sizeChanged = true
u.m.Unlock()
2016-05-19 16:37:58 +02:00
// title is ignored?
2018-03-23 17:07:36 +01:00
initOpenGL()
for {
2016-09-01 18:34:51 +02:00
if err := u.update(g); err != nil {
return err
}
2016-09-01 18:34:51 +02:00
}
}
func (u *userInterface) updateGraphicsContext(g GraphicsContext) {
sizeChanged := false
width, height := 0, 0
actualScale := 0.0
u.m.Lock()
sizeChanged = u.sizeChanged
if sizeChanged {
width = u.width
height = u.height
2018-03-23 17:07:36 +01:00
actualScale = u.actualScaleImpl()
}
u.sizeChanged = false
u.m.Unlock()
if sizeChanged {
2016-09-01 18:34:51 +02:00
// Sizing also calls GL functions
g.SetSize(width, height, actualScale)
}
}
2018-03-23 17:07:36 +01:00
func actualScale() float64 {
return currentUI.actualScale()
}
func (u *userInterface) actualScale() float64 {
u.m.Lock()
s := u.actualScaleImpl()
u.m.Unlock()
return s
}
func (u *userInterface) actualScaleImpl() float64 {
scale := u.scale
if u.fullscreenScale != 0 {
scale = u.fullscreenScale
}
return scale * devicescale.DeviceScale()
}
func (u *userInterface) update(g GraphicsContext) error {
<-chRender
defer func() {
chRenderEnd <- struct{}{}
}()
u.updateGraphicsContext(g)
if err := g.Update(func() {
u.updateGraphicsContext(g)
}); err != nil {
2016-09-01 18:34:51 +02:00
return err
}
return nil
}
2018-03-23 17:07:36 +01:00
func screenSize() (int, int) {
return currentUI.screenSize()
}
func (u *userInterface) screenSize() (int, int) {
u.m.Lock()
w, h := u.width, u.height
u.m.Unlock()
return w, h
}
func SetScreenSize(width, height int) bool {
currentUI.setScreenSize(width, height)
return true
}
func (u *userInterface) setScreenSize(width, height int) {
u.m.Lock()
if u.width != width || u.height != height {
u.width = width
u.height = height
2018-03-23 17:07:36 +01:00
u.updateFullscreenScaleIfNeeded()
u.sizeChanged = true
}
u.m.Unlock()
}
func SetScreenScale(scale float64) bool {
currentUI.setScreenScale(scale)
return false
}
func (u *userInterface) setScreenScale(scale float64) {
u.m.Lock()
if u.scale != scale {
u.scale = scale
u.sizeChanged = true
}
u.m.Unlock()
}
2016-09-02 16:38:02 +02:00
func ScreenScale() float64 {
u := currentUI
u.m.RLock()
s := u.scale
u.m.RUnlock()
return s
2016-05-19 16:37:58 +02:00
}
2018-03-23 17:07:36 +01:00
func setFullscreen(widthPx, heightPx int) {
currentUI.setFullscreen(widthPx, heightPx)
}
func (u *userInterface) setFullscreen(widthPx, heightPx int) {
u.m.Lock()
u.fullscreenWidthPx = widthPx
u.fullscreenHeightPx = heightPx
u.updateFullscreenScaleIfNeeded()
u.sizeChanged = true
u.m.Unlock()
}
func (u *userInterface) updateFullscreenScaleIfNeeded() {
if u.fullscreenWidthPx == 0 || u.fullscreenHeightPx == 0 {
return
}
w, h := u.width, u.height
scaleX := float64(u.fullscreenWidthPx) / float64(w)
scaleY := float64(u.fullscreenHeightPx) / float64(h)
scale := scaleX
if scale > scaleY {
scale = scaleY
}
u.fullscreenScale = scale / devicescale.DeviceScale()
}
func ScreenPadding() (x0, y0, x1, y1 float64) {
2018-03-23 17:07:36 +01:00
return currentUI.screenPadding()
}
func (u *userInterface) screenPadding() (x0, y0, x1, y1 float64) {
u.m.Lock()
x0, y0, x1, y1 = u.screenPaddingImpl()
u.m.Unlock()
return
}
func (u *userInterface) screenPaddingImpl() (x0, y0, x1, y1 float64) {
if u.fullscreenScale == 0 {
return 0, 0, 0, 0
}
s := u.fullscreenScale * devicescale.DeviceScale()
ox := (float64(u.fullscreenWidthPx) - float64(u.width)*s) / 2
oy := (float64(u.fullscreenHeightPx) - float64(u.height)*s) / 2
return ox, oy, ox, oy
}
2018-04-01 16:20:45 +02:00
func AdjustedCursorPosition() (x, y int) {
return currentUI.adjustCursorPosition(input.Get().CursorPosition())
2018-03-23 17:07:36 +01:00
}
func AdjustedTouches() []*input.Touch {
// TODO: Apply adjustment here
return input.Get().Touches()
}
2018-03-23 17:07:36 +01:00
func (u *userInterface) adjustCursorPosition(x, y int) (int, int) {
u.m.Lock()
ox, oy, _, _ := u.screenPaddingImpl()
s := u.actualScaleImpl()
u.m.Unlock()
return x - int(ox/s), y - int(oy/s)
}
2017-08-12 08:39:41 +02:00
func IsCursorVisible() bool {
return false
}
func SetCursorVisible(visible bool) {
2016-09-03 10:17:54 +02:00
// Do nothing
}
func IsFullscreen() bool {
return false
}
func SetFullscreen(fullscreen bool) {
// Do nothing
}
func IsRunnableInBackground() bool {
return false
}
func SetRunnableInBackground(runnableInBackground bool) {
// Do nothing
}
func SetWindowIcon(iconImages []image.Image) {
// Do nothing
}
func IsWindowDecorated() bool {
return false
}
func SetWindowDecorated(decorated bool) {
2017-09-22 21:12:02 +02:00
// Do nothing
}
2018-04-01 16:20:45 +02:00
func UpdateTouches(touches []*input.Touch) {
2018-03-23 17:07:36 +01:00
currentUI.m.Lock()
ox, oy, _, _ := currentUI.screenPaddingImpl()
s := currentUI.actualScaleImpl()
currentUI.m.Unlock()
2018-04-01 16:20:45 +02:00
input.Get().UpdateTouches(touches, -int(ox/s), -int(oy/s))
2016-05-22 19:06:01 +02:00
}