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.
|
|
|
|
|
2017-01-25 17:32:33 +01:00
|
|
|
// +build android ios
|
2016-05-18 20:17:50 +02:00
|
|
|
|
|
|
|
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"
|
2017-12-31 13:01:48 +01:00
|
|
|
"sync"
|
2016-06-09 18:21:46 +02:00
|
|
|
"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"
|
2016-11-03 15:31:25 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/opengl"
|
2016-05-18 20:17:50 +02:00
|
|
|
)
|
|
|
|
|
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
|
2016-06-09 18:21:46 +02:00
|
|
|
select {
|
|
|
|
case chRender <- struct{}{}:
|
2017-05-30 19:09:27 +02:00
|
|
|
return opengl.GetContext().DoWork(chError, chRenderEnd)
|
2016-06-09 18:21:46 +02:00
|
|
|
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.
|
2016-06-09 18:21:46 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-19 17:07:04 +02:00
|
|
|
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
|
2017-12-31 13:01:48 +01:00
|
|
|
|
2018-03-23 17:07:36 +01:00
|
|
|
// Used for gomobile-build
|
|
|
|
fullscreenScale float64
|
|
|
|
fullscreenWidthPx int
|
|
|
|
fullscreenHeightPx int
|
|
|
|
|
2017-12-31 13:01:48 +01:00
|
|
|
m sync.RWMutex
|
2016-05-18 20:17:50 +02:00
|
|
|
}
|
|
|
|
|
2016-05-19 16:37:58 +02:00
|
|
|
var (
|
2016-07-03 18:04:27 +02:00
|
|
|
chRender = make(chan struct{})
|
|
|
|
chRenderEnd = make(chan struct{})
|
2017-12-31 13:01:48 +01:00
|
|
|
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
|
2017-12-31 13:01:48 +01:00
|
|
|
|
|
|
|
u.m.Lock()
|
2016-05-19 16:37:58 +02:00
|
|
|
u.width = width
|
|
|
|
u.height = height
|
|
|
|
u.scale = scale
|
2017-12-31 13:01:48 +01:00
|
|
|
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()
|
|
|
|
|
2016-08-31 19:38:47 +02:00
|
|
|
for {
|
2016-09-01 18:34:51 +02:00
|
|
|
if err := u.update(g); err != nil {
|
2016-08-31 19:38:47 +02:00
|
|
|
return err
|
|
|
|
}
|
2016-09-01 18:34:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-01 18:08:03 +01:00
|
|
|
func (u *userInterface) updateGraphicsContext(g GraphicsContext) {
|
2017-12-31 13:01:48 +01:00
|
|
|
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()
|
2017-12-31 13:01:48 +01:00
|
|
|
}
|
|
|
|
u.sizeChanged = false
|
|
|
|
u.m.Unlock()
|
|
|
|
|
|
|
|
if sizeChanged {
|
2016-09-01 18:34:51 +02:00
|
|
|
// Sizing also calls GL functions
|
2017-12-31 13:01:48 +01:00
|
|
|
g.SetSize(width, height, actualScale)
|
2016-08-31 19:38:47 +02:00
|
|
|
}
|
2018-02-01 18:08:03 +01:00
|
|
|
}
|
|
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
2018-02-01 18:08:03 +01:00
|
|
|
func (u *userInterface) update(g GraphicsContext) error {
|
|
|
|
<-chRender
|
|
|
|
defer func() {
|
|
|
|
chRenderEnd <- struct{}{}
|
|
|
|
}()
|
|
|
|
|
|
|
|
u.updateGraphicsContext(g)
|
2017-12-31 13:01:48 +01:00
|
|
|
|
2018-02-01 18:08:03 +01:00
|
|
|
if err := g.Update(func() {
|
|
|
|
u.updateGraphicsContext(g)
|
|
|
|
}); err != nil {
|
2016-09-01 18:34:51 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
2016-05-18 20:17:50 +02:00
|
|
|
}
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2017-03-03 02:58:29 +01:00
|
|
|
func SetScreenSize(width, height int) bool {
|
2017-12-31 13:01:48 +01:00
|
|
|
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()
|
2017-12-31 13:01:48 +01:00
|
|
|
u.sizeChanged = true
|
|
|
|
}
|
|
|
|
u.m.Unlock()
|
2016-05-18 20:17:50 +02:00
|
|
|
}
|
|
|
|
|
2017-03-03 02:58:29 +01:00
|
|
|
func SetScreenScale(scale float64) bool {
|
2017-12-31 13:01:48 +01:00
|
|
|
currentUI.setScreenScale(scale)
|
2017-03-03 02:58:29 +01:00
|
|
|
return false
|
2016-05-18 20:17:50 +02:00
|
|
|
}
|
|
|
|
|
2017-12-31 13:01:48 +01:00
|
|
|
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 {
|
2017-12-31 13:01:48 +01:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
2018-02-24 13:16:30 +01:00
|
|
|
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
|
2017-06-30 21:12:09 +02:00
|
|
|
}
|
|
|
|
|
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 (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-06-30 21:27:38 +02:00
|
|
|
}
|
|
|
|
|
2017-08-12 08:39:41 +02:00
|
|
|
func IsCursorVisible() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2018-01-08 18:01:33 +01:00
|
|
|
func SetCursorVisible(visible bool) {
|
2016-09-03 10:17:54 +02:00
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2017-09-13 20:37:38 +02:00
|
|
|
func IsFullscreen() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-08-02 16:37:50 +02:00
|
|
|
func SetFullscreen(fullscreen bool) {
|
2017-06-29 17:35:34 +02:00
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2017-09-13 20:37:38 +02:00
|
|
|
func IsRunnableInBackground() bool {
|
2017-06-29 17:35:34 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-08-02 16:37:50 +02:00
|
|
|
func SetRunnableInBackground(runnableInBackground bool) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2017-09-13 20:37:38 +02:00
|
|
|
func SetWindowIcon(iconImages []image.Image) {
|
2017-08-02 16:37:50 +02:00
|
|
|
// Do nothing
|
2017-09-13 20:37:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func IsWindowDecorated() bool {
|
2017-08-02 16:37:50 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-09-13 20:37:38 +02:00
|
|
|
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
|
|
|
}
|