2015-01-01 17:20:20 +01:00
|
|
|
// Copyright 2015 Hajime Hoshi
|
2014-12-24 03:04:10 +01:00
|
|
|
//
|
|
|
|
// 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.
|
2014-12-09 15:16:04 +01:00
|
|
|
|
2017-05-11 12:09:13 +02:00
|
|
|
// +build darwin freebsd linux windows
|
2015-01-02 07:20:05 +01:00
|
|
|
// +build !js
|
2016-05-18 20:17:50 +02:00
|
|
|
// +build !android
|
2016-06-15 17:49:22 +02:00
|
|
|
// +build !ios
|
2015-01-02 07:20:05 +01:00
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
package glfw
|
2014-12-05 18:26:02 +01:00
|
|
|
|
|
|
|
import (
|
2019-06-07 18:43:07 +02:00
|
|
|
"context"
|
2019-12-07 11:28:15 +01:00
|
|
|
"fmt"
|
2017-09-22 21:12:02 +02:00
|
|
|
"image"
|
2017-10-19 20:25:21 +02:00
|
|
|
"math"
|
2019-11-17 11:58:41 +01:00
|
|
|
"os"
|
2014-12-14 10:57:29 +01:00
|
|
|
"runtime"
|
2016-09-03 14:14:06 +02:00
|
|
|
"sync"
|
2015-01-07 17:32:37 +01:00
|
|
|
"time"
|
2019-11-25 15:13:44 +01:00
|
|
|
"unsafe"
|
2015-06-20 18:33:28 +02:00
|
|
|
|
2018-10-06 10:29:40 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/devicescale"
|
2019-03-30 14:26:27 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/driver"
|
2018-12-29 11:20:41 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/glfw"
|
2019-12-08 08:10:43 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/hooks"
|
2019-06-05 17:05:53 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/thread"
|
2014-12-05 18:26:02 +01:00
|
|
|
)
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
type UserInterface struct {
|
2019-12-08 16:11:18 +01:00
|
|
|
title string
|
|
|
|
window *glfw.Window
|
|
|
|
screenWidthInDP int
|
|
|
|
screenHeightInDP int
|
|
|
|
scale float64
|
|
|
|
fullscreenScale float64
|
2018-01-02 16:23:18 +01:00
|
|
|
|
2017-08-02 16:37:50 +02:00
|
|
|
running bool
|
2018-02-24 16:42:37 +01:00
|
|
|
toChangeSize bool
|
2017-08-02 16:37:50 +02:00
|
|
|
origPosX int
|
|
|
|
origPosY int
|
|
|
|
runnableInBackground bool
|
2018-07-14 13:50:09 +02:00
|
|
|
vsync bool
|
2018-01-02 16:23:18 +01:00
|
|
|
|
2019-12-08 12:24:20 +01:00
|
|
|
lastDeviceScaleFactor float64
|
2018-08-05 14:19:22 +02:00
|
|
|
|
2019-12-08 16:11:18 +01:00
|
|
|
initMonitor *glfw.Monitor
|
|
|
|
initFullscreenWidthInDP int
|
|
|
|
initFullscreenHeightInDP int
|
|
|
|
initFullscreen bool
|
2019-12-14 04:30:03 +01:00
|
|
|
initCursorMode driver.CursorMode
|
2019-12-08 16:11:18 +01:00
|
|
|
initWindowDecorated bool
|
|
|
|
initWindowResizable bool
|
|
|
|
initWindowPositionX int
|
|
|
|
initWindowPositionY int
|
|
|
|
initScreenTransparent bool
|
|
|
|
initIconImages []image.Image
|
2018-01-02 16:23:18 +01:00
|
|
|
|
2019-01-06 16:21:59 +01:00
|
|
|
reqWidth int
|
|
|
|
reqHeight int
|
|
|
|
|
2019-04-06 16:03:21 +02:00
|
|
|
graphics driver.Graphics
|
2019-04-07 11:55:56 +02:00
|
|
|
input Input
|
2019-03-30 13:50:32 +01:00
|
|
|
|
2019-06-05 17:05:53 +02:00
|
|
|
t *thread.Thread
|
2019-08-18 18:01:43 +02:00
|
|
|
m sync.RWMutex
|
2016-03-24 16:38:30 +01:00
|
|
|
}
|
|
|
|
|
2019-02-24 15:31:26 +01:00
|
|
|
const (
|
|
|
|
maxInt = int(^uint(0) >> 1)
|
|
|
|
minInt = -maxInt - 1
|
|
|
|
invalidPos = minInt
|
|
|
|
)
|
|
|
|
|
2017-07-30 13:26:40 +02:00
|
|
|
var (
|
2019-04-07 03:42:55 +02:00
|
|
|
theUI = &UserInterface{
|
2019-02-24 15:31:26 +01:00
|
|
|
origPosX: invalidPos,
|
|
|
|
origPosY: invalidPos,
|
2019-12-14 04:30:03 +01:00
|
|
|
initCursorMode: driver.CursorModeVisible,
|
2017-09-13 20:37:38 +02:00
|
|
|
initWindowDecorated: true,
|
2019-11-30 16:21:11 +01:00
|
|
|
initWindowPositionX: invalidPos,
|
|
|
|
initWindowPositionY: invalidPos,
|
2018-07-14 13:50:09 +02:00
|
|
|
vsync: true,
|
2017-07-30 13:26:40 +02:00
|
|
|
}
|
|
|
|
)
|
2016-03-24 16:38:30 +01: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
|
|
|
|
}
|
|
|
|
|
2016-07-23 13:25:52 +02:00
|
|
|
func init() {
|
2018-12-18 17:01:16 +01:00
|
|
|
hideConsoleWindowOnWindows()
|
2018-05-04 16:55:23 +02:00
|
|
|
if err := initialize(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-11-25 15:13:44 +01:00
|
|
|
glfw.SetMonitorCallback(func(monitor *glfw.Monitor, event glfw.PeripheralEvent) {
|
2018-11-07 12:54:37 +01:00
|
|
|
cacheMonitors()
|
|
|
|
})
|
|
|
|
cacheMonitors()
|
2016-07-23 13:25:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func initialize() error {
|
2016-05-07 15:27:10 +02:00
|
|
|
if err := glfw.Init(); err != nil {
|
2016-07-23 13:25:52 +02:00
|
|
|
return err
|
2014-12-05 18:26:02 +01:00
|
|
|
}
|
2014-12-17 12:03:26 +01:00
|
|
|
glfw.WindowHint(glfw.Visible, glfw.False)
|
2017-09-13 20:37:38 +02:00
|
|
|
|
2018-12-18 17:01:16 +01:00
|
|
|
// Create a window to set the initial monitor.
|
|
|
|
w, err := glfw.CreateWindow(16, 16, "", nil, nil)
|
2014-12-07 20:22:50 +01:00
|
|
|
if err != nil {
|
2016-07-23 13:25:52 +02:00
|
|
|
return err
|
2014-12-07 20:22:50 +01:00
|
|
|
}
|
2019-07-21 16:41:51 +02:00
|
|
|
if w == nil {
|
|
|
|
// This can happen on Windows Remote Desktop (#903).
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: glfw.CreateWindow must not return nil")
|
2019-07-21 16:41:51 +02:00
|
|
|
}
|
|
|
|
|
2018-12-18 17:01:16 +01:00
|
|
|
// TODO: Fix this hack. currentMonitorImpl now requires u.window on POSIX.
|
2019-04-07 03:58:11 +02:00
|
|
|
theUI.window = w
|
|
|
|
theUI.initMonitor = theUI.currentMonitorFromPosition()
|
|
|
|
v := theUI.initMonitor.GetVideoMode()
|
2019-12-08 16:11:18 +01:00
|
|
|
theUI.initFullscreenWidthInDP = int(theUI.toDeviceIndependentPixel(float64(v.Width)))
|
|
|
|
theUI.initFullscreenHeightInDP = int(theUI.toDeviceIndependentPixel(float64(v.Height)))
|
2019-04-07 03:58:11 +02:00
|
|
|
theUI.window.Destroy()
|
|
|
|
theUI.window = nil
|
2017-08-12 08:39:41 +02:00
|
|
|
|
2016-07-23 13:25:52 +02:00
|
|
|
return nil
|
2014-12-17 12:03:26 +01:00
|
|
|
}
|
2014-12-10 15:52:37 +01:00
|
|
|
|
2018-11-07 12:54:37 +01:00
|
|
|
type cachedMonitor struct {
|
|
|
|
m *glfw.Monitor
|
|
|
|
vm *glfw.VidMode
|
|
|
|
// Pos of monitor in virtual coords
|
|
|
|
x int
|
|
|
|
y int
|
|
|
|
}
|
|
|
|
|
|
|
|
// monitors is the monitor list cache for desktop glfw compile targets.
|
|
|
|
// populated by 'cacheMonitors' which is called on init and every
|
|
|
|
// monitor config change event.
|
2018-11-14 17:08:36 +01:00
|
|
|
//
|
|
|
|
// monitors must be manipulated on the main thread.
|
2018-11-07 12:54:37 +01:00
|
|
|
var monitors []*cachedMonitor
|
|
|
|
|
|
|
|
func cacheMonitors() {
|
2018-11-14 17:08:36 +01:00
|
|
|
monitors = nil
|
2018-11-07 12:54:37 +01:00
|
|
|
ms := glfw.GetMonitors()
|
|
|
|
for _, m := range ms {
|
|
|
|
x, y := m.GetPos()
|
|
|
|
monitors = append(monitors, &cachedMonitor{
|
|
|
|
m: m,
|
|
|
|
vm: m.GetVideoMode(),
|
|
|
|
x: x,
|
|
|
|
y: y,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// getCachedMonitor returns a monitor for the given window x/y
|
|
|
|
// returns false if monitor is not found.
|
2018-11-14 17:08:36 +01:00
|
|
|
//
|
|
|
|
// getCachedMonitor must be called on the main thread.
|
2018-11-07 12:54:37 +01:00
|
|
|
func getCachedMonitor(wx, wy int) (*cachedMonitor, bool) {
|
|
|
|
for _, m := range monitors {
|
|
|
|
if m.x <= wx && wx < m.x+m.vm.Width && m.y <= wy && wy < m.y+m.vm.Height {
|
|
|
|
return m, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) isRunning() bool {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2017-08-02 17:19:04 +02:00
|
|
|
v := u.running
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2017-08-02 17:19:04 +02:00
|
|
|
return v
|
2016-09-03 14:14:06 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) setRunning(running bool) {
|
2016-09-03 14:14:06 +02:00
|
|
|
u.m.Lock()
|
|
|
|
u.running = running
|
2017-08-02 17:19:04 +02:00
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) isInitFullscreen() bool {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2017-08-02 17:19:04 +02:00
|
|
|
v := u.initFullscreen
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2017-08-02 17:19:04 +02:00
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) setInitFullscreen(initFullscreen bool) {
|
2017-08-02 17:19:04 +02:00
|
|
|
u.m.Lock()
|
|
|
|
u.initFullscreen = initFullscreen
|
|
|
|
u.m.Unlock()
|
2016-09-03 14:14:06 +02:00
|
|
|
}
|
|
|
|
|
2019-12-14 04:30:03 +01:00
|
|
|
func (u *UserInterface) getInitCursorMode() driver.CursorMode {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2019-12-14 04:30:03 +01:00
|
|
|
v := u.initCursorMode
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2017-08-12 08:39:41 +02:00
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-12-14 04:30:03 +01:00
|
|
|
func (u *UserInterface) setInitCursorMode(mode driver.CursorMode) {
|
2017-08-12 08:39:41 +02:00
|
|
|
u.m.Lock()
|
2019-12-14 04:30:03 +01:00
|
|
|
u.initCursorMode = mode
|
2017-08-12 08:39:41 +02:00
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) isInitWindowDecorated() bool {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2017-09-13 20:37:38 +02:00
|
|
|
v := u.initWindowDecorated
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2017-09-13 20:37:38 +02:00
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) setInitWindowDecorated(decorated bool) {
|
2017-09-13 20:37:38 +02:00
|
|
|
u.m.Lock()
|
|
|
|
u.initWindowDecorated = decorated
|
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) isRunnableInBackground() bool {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2017-08-02 16:37:50 +02:00
|
|
|
v := u.runnableInBackground
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2017-08-02 16:37:50 +02:00
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) setRunnableInBackground(runnableInBackground bool) {
|
2017-08-02 16:37:50 +02:00
|
|
|
u.m.Lock()
|
|
|
|
u.runnableInBackground = runnableInBackground
|
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) isInitWindowResizable() bool {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2019-01-06 16:21:59 +01:00
|
|
|
v := u.initWindowResizable
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2019-01-06 16:21:59 +01:00
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) setInitWindowResizable(resizable bool) {
|
2019-01-06 16:21:59 +01:00
|
|
|
u.m.Lock()
|
|
|
|
u.initWindowResizable = resizable
|
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-11-30 16:07:41 +01:00
|
|
|
func (u *UserInterface) isInitScreenTransparent() bool {
|
|
|
|
u.m.RLock()
|
|
|
|
v := u.initScreenTransparent
|
|
|
|
u.m.RUnlock()
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) setInitScreenTransparent(transparent bool) {
|
|
|
|
u.m.RLock()
|
|
|
|
u.initScreenTransparent = transparent
|
|
|
|
u.m.RUnlock()
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) getInitIconImages() []image.Image {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2017-09-22 21:12:02 +02:00
|
|
|
i := u.initIconImages
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2017-09-22 21:12:02 +02:00
|
|
|
return i
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) setInitIconImages(iconImages []image.Image) {
|
2017-09-22 21:12:02 +02:00
|
|
|
u.m.Lock()
|
|
|
|
u.initIconImages = iconImages
|
|
|
|
u.m.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-11-30 16:21:11 +01:00
|
|
|
func (u *UserInterface) getInitWindowPosition() (int, int) {
|
2019-11-30 16:11:46 +01:00
|
|
|
u.m.RLock()
|
|
|
|
defer u.m.RUnlock()
|
2019-11-30 16:21:11 +01:00
|
|
|
if u.initWindowPositionX != invalidPos && u.initWindowPositionY != invalidPos {
|
|
|
|
return u.initWindowPositionX, u.initWindowPositionY
|
2019-11-30 16:11:46 +01:00
|
|
|
}
|
2019-11-30 16:21:11 +01:00
|
|
|
return invalidPos, invalidPos
|
2019-11-30 16:11:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) setInitWindowPosition(x, y int) {
|
|
|
|
u.m.Lock()
|
|
|
|
defer u.m.Unlock()
|
|
|
|
|
2019-11-30 16:21:11 +01:00
|
|
|
u.initWindowPositionX = x
|
|
|
|
u.initWindowPositionY = y
|
2019-11-30 16:11:46 +01:00
|
|
|
}
|
|
|
|
|
2019-12-08 13:49:42 +01:00
|
|
|
// toDeviceIndependentPixel must be called from the main thread.
|
|
|
|
func (u *UserInterface) toDeviceIndependentPixel(x float64) float64 {
|
|
|
|
return x / u.glfwScale()
|
|
|
|
}
|
|
|
|
|
|
|
|
// toDeviceDependentPixel must be called from the main thread.
|
|
|
|
func (u *UserInterface) toDeviceDependentPixel(x float64) float64 {
|
|
|
|
return x * u.glfwScale()
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) ScreenSizeInFullscreen() (int, int) {
|
2018-12-18 17:17:59 +01:00
|
|
|
if !u.isRunning() {
|
2019-12-08 16:11:18 +01:00
|
|
|
return u.initFullscreenWidthInDP, u.initFullscreenHeightInDP
|
2018-12-18 17:17:59 +01:00
|
|
|
}
|
|
|
|
|
2019-12-08 13:49:42 +01:00
|
|
|
var w, h int
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 13:49:42 +01:00
|
|
|
v := u.currentMonitor().GetVideoMode()
|
|
|
|
w = int(u.toDeviceIndependentPixel(float64(v.Width)))
|
|
|
|
h = int(u.toDeviceIndependentPixel(float64(v.Height)))
|
2018-12-18 17:17:59 +01:00
|
|
|
return nil
|
|
|
|
})
|
2019-12-08 13:49:42 +01:00
|
|
|
return w, h
|
2018-05-04 09:09:55 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetScreenSize(width, height int) {
|
2016-09-03 14:14:06 +02:00
|
|
|
if !u.isRunning() {
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: SetScreenSize can't be called before the main loop starts")
|
2016-09-03 14:14:06 +02:00
|
|
|
}
|
2019-12-07 11:28:15 +01:00
|
|
|
u.setScreenSize(width, height, u.scale, u.isFullscreen(), u.vsync)
|
2015-01-27 14:02:23 +01:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetScreenScale(scale float64) {
|
2016-09-03 14:14:06 +02:00
|
|
|
if !u.isRunning() {
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: SetScreenScale can't be called before the main loop starts")
|
2016-09-03 14:14:06 +02:00
|
|
|
}
|
2019-12-08 16:11:18 +01:00
|
|
|
u.setScreenSize(u.screenWidthInDP, u.screenHeightInDP, scale, u.isFullscreen(), u.vsync)
|
2017-06-29 17:35:34 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) ScreenScale() float64 {
|
2016-09-03 14:14:06 +02:00
|
|
|
if !u.isRunning() {
|
|
|
|
return 0
|
|
|
|
}
|
2016-06-18 19:59:17 +02:00
|
|
|
s := 0.0
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2016-05-06 05:23:48 +02:00
|
|
|
s = u.scale
|
2016-07-23 14:01:30 +02:00
|
|
|
return nil
|
2016-05-06 05:23:48 +02:00
|
|
|
})
|
|
|
|
return s
|
2015-01-27 14:02:23 +01:00
|
|
|
}
|
|
|
|
|
2019-02-07 09:19:24 +01:00
|
|
|
// isFullscreen must be called from the main thread.
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) isFullscreen() bool {
|
2017-08-12 11:31:16 +02:00
|
|
|
if !u.isRunning() {
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: isFullscreen can't be called before the main loop starts")
|
2017-08-12 11:31:16 +02:00
|
|
|
}
|
|
|
|
return u.window.GetMonitor() != nil
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) IsFullscreen() bool {
|
2017-06-29 17:35:34 +02:00
|
|
|
if !u.isRunning() {
|
2017-08-02 17:19:04 +02:00
|
|
|
return u.isInitFullscreen()
|
2017-06-29 17:35:34 +02:00
|
|
|
}
|
2018-02-24 10:10:39 +01:00
|
|
|
b := false
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-02-07 09:19:24 +01:00
|
|
|
b = u.isFullscreen()
|
2018-02-24 10:10:39 +01:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return b
|
2017-06-29 17:35:34 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetFullscreen(fullscreen bool) {
|
2017-08-02 18:07:04 +02:00
|
|
|
if !u.isRunning() {
|
|
|
|
u.setInitFullscreen(fullscreen)
|
|
|
|
return
|
|
|
|
}
|
2019-12-08 16:11:18 +01:00
|
|
|
u.setScreenSize(u.screenWidthInDP, u.screenHeightInDP, u.scale, fullscreen, u.vsync)
|
2017-08-02 18:07:04 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetRunnableInBackground(runnableInBackground bool) {
|
|
|
|
u.setRunnableInBackground(runnableInBackground)
|
2017-08-02 16:37:50 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) IsRunnableInBackground() bool {
|
|
|
|
return u.isRunnableInBackground()
|
2017-08-02 16:37:50 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetVsyncEnabled(enabled bool) {
|
2018-07-14 13:50:09 +02:00
|
|
|
if !u.isRunning() {
|
2018-07-16 19:37:22 +02:00
|
|
|
// In general, m is used for locking init* values.
|
|
|
|
// m is not used for updating vsync in setScreenSize so far, but
|
|
|
|
// it should be OK since any goroutines can't reach here when
|
|
|
|
// the game already starts and setScreenSize can be called.
|
2019-08-18 18:47:55 +02:00
|
|
|
u.m.Lock()
|
2018-07-16 19:37:22 +02:00
|
|
|
u.vsync = enabled
|
2019-08-18 18:47:55 +02:00
|
|
|
u.m.Unlock()
|
2018-07-14 13:50:09 +02:00
|
|
|
return
|
|
|
|
}
|
2019-12-08 16:11:18 +01:00
|
|
|
u.setScreenSize(u.screenWidthInDP, u.screenHeightInDP, u.scale, u.isFullscreen(), enabled)
|
2018-07-14 13:50:09 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) IsVsyncEnabled() bool {
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RLock()
|
2018-10-10 16:15:38 +02:00
|
|
|
r := u.vsync
|
2019-08-18 18:01:43 +02:00
|
|
|
u.m.RUnlock()
|
2018-07-14 13:50:09 +02:00
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetWindowTitle(title string) {
|
|
|
|
if !u.isRunning() {
|
2018-05-02 12:09:03 +02:00
|
|
|
return
|
|
|
|
}
|
2019-11-26 03:45:52 +01:00
|
|
|
u.title = title
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-04-07 03:42:55 +02:00
|
|
|
u.window.SetTitle(title)
|
2018-05-02 12:09:03 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetWindowIcon(iconImages []image.Image) {
|
|
|
|
if !u.isRunning() {
|
|
|
|
u.setInitIconImages(iconImages)
|
2017-09-22 21:12:02 +02:00
|
|
|
return
|
|
|
|
}
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-04-07 03:42:55 +02:00
|
|
|
u.window.SetIcon(iconImages)
|
2017-09-22 21:12:02 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) ScreenPadding() (x0, y0, x1, y1 float64) {
|
2017-06-30 21:12:09 +02:00
|
|
|
if !u.isRunning() {
|
2018-02-24 13:16:30 +01:00
|
|
|
return 0, 0, 0, 0
|
2017-06-30 21:12:09 +02:00
|
|
|
}
|
2019-04-07 03:42:55 +02:00
|
|
|
if !u.IsFullscreen() {
|
2019-12-08 17:06:25 +01:00
|
|
|
w, _ := u.window.GetSize()
|
|
|
|
wf := u.toDeviceIndependentPixel(float64(w)) / u.getScale()
|
|
|
|
if u.screenWidthInDP == int(wf) {
|
2018-02-24 13:16:30 +01:00
|
|
|
return 0, 0, 0, 0
|
2017-10-19 20:25:21 +02:00
|
|
|
}
|
2018-02-24 11:26:14 +01:00
|
|
|
// The window width can be bigger than the game screen width (#444).
|
2019-01-16 15:32:47 +01:00
|
|
|
ox := 0.0
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 17:06:25 +01:00
|
|
|
ox = (wf*u.actualScreenScale() - float64(u.screenWidthInDP)*u.actualScreenScale()) / 2
|
2019-01-16 15:32:47 +01:00
|
|
|
return nil
|
|
|
|
})
|
2018-02-24 13:16:30 +01:00
|
|
|
return ox, 0, ox, 0
|
2017-06-30 21:12:09 +02:00
|
|
|
}
|
2018-09-29 14:47:39 +02:00
|
|
|
|
2019-03-11 16:28:37 +01:00
|
|
|
d := 0.0
|
2018-09-29 14:47:39 +02:00
|
|
|
sx := 0.0
|
|
|
|
sy := 0.0
|
2019-12-08 13:49:42 +01:00
|
|
|
mx := 0.0
|
|
|
|
my := 0.0
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 16:11:18 +01:00
|
|
|
sx = float64(u.screenWidthInDP) * u.actualScreenScale()
|
|
|
|
sy = float64(u.screenHeightInDP) * u.actualScreenScale()
|
2019-03-11 16:28:37 +01:00
|
|
|
|
2019-12-08 13:49:42 +01:00
|
|
|
v := u.currentMonitor().GetVideoMode()
|
|
|
|
d = u.deviceScaleFactor()
|
|
|
|
mx = u.toDeviceIndependentPixel(float64(v.Width)) * d
|
|
|
|
my = u.toDeviceIndependentPixel(float64(v.Height)) * d
|
2017-06-30 21:12:09 +02:00
|
|
|
return nil
|
|
|
|
})
|
2018-10-09 18:07:47 +02:00
|
|
|
|
2018-09-29 14:47:39 +02:00
|
|
|
ox := (mx - sx) / 2
|
|
|
|
oy := (my - sy) / 2
|
|
|
|
return ox, oy, (mx - sx) - ox, (my - sy) - oy
|
2017-06-30 21:12:09 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 12:27:30 +02:00
|
|
|
func (u *UserInterface) adjustPosition(x, y int) (int, int) {
|
2017-06-30 21:27:38 +02:00
|
|
|
if !u.isRunning() {
|
|
|
|
return x, y
|
|
|
|
}
|
2019-04-07 03:42:55 +02:00
|
|
|
ox, oy, _, _ := u.ScreenPadding()
|
2017-06-30 21:27:38 +02:00
|
|
|
s := 0.0
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-04-07 03:42:55 +02:00
|
|
|
s = u.actualScreenScale()
|
2017-06-30 21:27:38 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return x - int(ox/s), y - int(oy/s)
|
|
|
|
}
|
|
|
|
|
2019-12-14 04:30:03 +01:00
|
|
|
func (u *UserInterface) CursorMode() driver.CursorMode {
|
2017-08-12 08:39:41 +02:00
|
|
|
if !u.isRunning() {
|
2019-12-14 04:30:03 +01:00
|
|
|
return u.getInitCursorMode()
|
2017-08-12 08:39:41 +02:00
|
|
|
}
|
2019-12-14 04:30:03 +01:00
|
|
|
var v driver.CursorMode
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-14 04:30:03 +01:00
|
|
|
mode := u.window.GetInputMode(glfw.CursorMode)
|
|
|
|
switch mode {
|
|
|
|
case glfw.CursorNormal:
|
|
|
|
v = driver.CursorModeVisible
|
|
|
|
case glfw.CursorHidden:
|
|
|
|
v = driver.CursorModeHidden
|
|
|
|
case glfw.CursorDisabled:
|
|
|
|
v = driver.CursorModeCaptured
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("invalid cursor mode: %d", mode))
|
|
|
|
}
|
2017-08-12 08:39:41 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-12-14 04:30:03 +01:00
|
|
|
func (u *UserInterface) SetCursorMode(mode driver.CursorMode) {
|
2017-08-12 08:39:41 +02:00
|
|
|
if !u.isRunning() {
|
2019-12-14 04:30:03 +01:00
|
|
|
u.setInitCursorMode(mode)
|
2017-08-12 08:39:41 +02:00
|
|
|
return
|
|
|
|
}
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-14 04:30:03 +01:00
|
|
|
var c int
|
|
|
|
switch mode {
|
|
|
|
case driver.CursorModeVisible:
|
|
|
|
c = glfw.CursorNormal
|
|
|
|
case driver.CursorModeHidden:
|
2017-08-12 08:39:41 +02:00
|
|
|
c = glfw.CursorHidden
|
2019-12-14 04:30:03 +01:00
|
|
|
case driver.CursorModeCaptured:
|
|
|
|
c = glfw.CursorDisabled
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("invalid cursor mode: %d", mode))
|
2017-08-12 08:39:41 +02:00
|
|
|
}
|
2019-04-07 03:42:55 +02:00
|
|
|
u.window.SetInputMode(glfw.CursorMode, c)
|
2017-08-12 08:39:41 +02:00
|
|
|
return nil
|
|
|
|
})
|
2016-09-03 10:17:54 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) IsWindowDecorated() bool {
|
2017-09-13 20:37:38 +02:00
|
|
|
if !u.isRunning() {
|
|
|
|
return u.isInitWindowDecorated()
|
|
|
|
}
|
|
|
|
v := false
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-04-07 03:42:55 +02:00
|
|
|
v = u.window.GetAttrib(glfw.Decorated) == glfw.True
|
2017-09-13 20:37:38 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetWindowDecorated(decorated bool) {
|
2017-09-13 20:37:38 +02:00
|
|
|
if !u.isRunning() {
|
|
|
|
u.setInitWindowDecorated(decorated)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-26 03:45:52 +01:00
|
|
|
_ = u.t.Call(func() error {
|
|
|
|
v := glfw.False
|
|
|
|
if decorated {
|
|
|
|
v = glfw.True
|
|
|
|
}
|
|
|
|
u.window.SetAttrib(glfw.Decorated, v)
|
2017-09-13 20:37:38 +02:00
|
|
|
|
2019-11-26 03:45:52 +01:00
|
|
|
// The title can be lost when the decoration is gone. Recover this.
|
|
|
|
if v == glfw.True {
|
|
|
|
u.window.SetTitle(u.title)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2017-09-13 20:37:38 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) IsWindowResizable() bool {
|
2019-01-06 16:21:59 +01:00
|
|
|
if !u.isRunning() {
|
|
|
|
return u.isInitWindowResizable()
|
|
|
|
}
|
|
|
|
v := false
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-04-07 03:42:55 +02:00
|
|
|
v = u.window.GetAttrib(glfw.Resizable) == glfw.True
|
2019-01-06 16:21:59 +01:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) SetWindowResizable(resizable bool) {
|
|
|
|
if !u.isRunning() {
|
|
|
|
u.setInitWindowResizable(resizable)
|
2019-01-06 16:21:59 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: SetWindowResizable can't be called after the main loop so far.")
|
2019-01-06 16:21:59 +01:00
|
|
|
|
|
|
|
// TODO: Now SetAttrib doesn't exist on GLFW 3.2. Revisit later (#556).
|
|
|
|
}
|
|
|
|
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) DeviceScaleFactor() float64 {
|
2018-10-07 18:11:13 +02:00
|
|
|
if !u.isRunning() {
|
2018-12-18 17:01:16 +01:00
|
|
|
return devicescale.GetAt(u.initMonitor.GetPos())
|
2018-10-07 18:11:13 +02:00
|
|
|
}
|
|
|
|
|
2019-12-08 12:24:20 +01:00
|
|
|
f := 0.0
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 12:24:20 +01:00
|
|
|
f = u.deviceScaleFactor()
|
2018-10-07 18:11:13 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
2019-12-08 12:24:20 +01:00
|
|
|
// deviceScaleFactor must be called from the main thread.
|
|
|
|
func (u *UserInterface) deviceScaleFactor() float64 {
|
|
|
|
// Avoid calling monitor.GetPos if we have the monitor position cached already.
|
|
|
|
if cm, ok := getCachedMonitor(u.window.GetPos()); ok {
|
|
|
|
return devicescale.GetAt(cm.x, cm.y)
|
|
|
|
}
|
|
|
|
// TODO: When is this reached?
|
|
|
|
return devicescale.GetAt(u.currentMonitor().GetPos())
|
|
|
|
}
|
|
|
|
|
2019-06-05 17:05:53 +02:00
|
|
|
func init() {
|
|
|
|
// Lock the main thread.
|
|
|
|
runtime.LockOSThread()
|
|
|
|
}
|
|
|
|
|
2019-06-07 18:43:07 +02:00
|
|
|
func (u *UserInterface) Run(width, height int, scale float64, title string, uicontext driver.UIContext, graphics driver.Graphics) error {
|
2019-06-05 17:05:53 +02:00
|
|
|
// Initialize the main thread first so the thread is available at u.run (#809).
|
|
|
|
u.t = thread.New()
|
2019-06-23 19:03:12 +02:00
|
|
|
u.graphics = graphics
|
|
|
|
u.graphics.SetThread(u.t)
|
2019-06-05 17:05:53 +02:00
|
|
|
|
2019-06-07 18:43:07 +02:00
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
|
|
|
|
ch := make(chan error, 1)
|
2019-04-10 03:44:02 +02:00
|
|
|
go func() {
|
2019-06-07 18:43:07 +02:00
|
|
|
defer cancel()
|
2019-04-10 03:44:02 +02:00
|
|
|
defer close(ch)
|
2019-06-23 19:03:12 +02:00
|
|
|
if err := u.run(width, height, scale, title, uicontext); err != nil {
|
2019-04-10 03:44:02 +02:00
|
|
|
ch <- err
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2019-06-21 04:09:48 +02:00
|
|
|
u.setRunning(true)
|
|
|
|
u.t.Loop(ctx)
|
|
|
|
u.setRunning(false)
|
2019-06-07 18:43:07 +02:00
|
|
|
return <-ch
|
2019-04-10 03:44:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) RunWithoutMainLoop(width, height int, scale float64, title string, context driver.UIContext, graphics driver.Graphics) <-chan error {
|
|
|
|
panic("glfw: RunWithoutMainLoop is not implemented")
|
|
|
|
}
|
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
// createWindow creates a GLFW window.
|
|
|
|
//
|
|
|
|
// createWindow must be called from the main thread.
|
|
|
|
//
|
|
|
|
// createWindow does not set the position or size so far.
|
|
|
|
func (u *UserInterface) createWindow() error {
|
|
|
|
if u.window != nil {
|
|
|
|
panic("glfw: u.window must not exist at createWindow")
|
|
|
|
}
|
|
|
|
|
|
|
|
// As a start, create a window with temporary size to create OpenGL context thread.
|
|
|
|
window, err := glfw.CreateWindow(16, 16, "", nil, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
u.window = window
|
|
|
|
|
|
|
|
if u.graphics.IsGL() {
|
|
|
|
u.window.MakeContextCurrent()
|
|
|
|
}
|
|
|
|
|
|
|
|
u.window.SetInputMode(glfw.StickyMouseButtonsMode, glfw.True)
|
|
|
|
u.window.SetInputMode(glfw.StickyKeysMode, glfw.True)
|
|
|
|
|
|
|
|
mode := glfw.CursorNormal
|
2019-12-14 04:30:03 +01:00
|
|
|
switch u.getInitCursorMode() {
|
|
|
|
case driver.CursorModeHidden:
|
2019-12-07 11:28:15 +01:00
|
|
|
mode = glfw.CursorHidden
|
2019-12-14 04:30:03 +01:00
|
|
|
case driver.CursorModeCaptured:
|
|
|
|
mode = glfw.CursorDisabled
|
2019-12-07 11:28:15 +01:00
|
|
|
}
|
|
|
|
u.window.SetInputMode(glfw.CursorMode, mode)
|
|
|
|
u.window.SetTitle(u.title)
|
|
|
|
// TODO: Set icons
|
|
|
|
|
|
|
|
u.window.SetSizeCallback(func(_ *glfw.Window, width, height int) {
|
|
|
|
if u.window.GetAttrib(glfw.Resizable) == glfw.False {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if u.isFullscreen() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-12-08 13:49:42 +01:00
|
|
|
w := int(u.toDeviceIndependentPixel(float64(width)) / u.scale)
|
|
|
|
h := int(u.toDeviceIndependentPixel(float64(height)) / u.scale)
|
2019-12-07 11:28:15 +01:00
|
|
|
u.reqWidth = w
|
|
|
|
u.reqHeight = h
|
|
|
|
})
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-06-23 19:03:12 +02:00
|
|
|
func (u *UserInterface) run(width, height int, scale float64, title string, context driver.UIContext) error {
|
2019-12-07 11:28:15 +01:00
|
|
|
var (
|
|
|
|
m *glfw.Monitor
|
|
|
|
mx, my int
|
|
|
|
v *glfw.VidMode
|
|
|
|
)
|
|
|
|
|
2019-11-10 11:09:38 +01:00
|
|
|
if err := u.t.Call(func() error {
|
2019-06-23 19:03:12 +02:00
|
|
|
if u.graphics.IsGL() {
|
2019-03-27 19:36:31 +01:00
|
|
|
glfw.WindowHint(glfw.ContextVersionMajor, 2)
|
|
|
|
glfw.WindowHint(glfw.ContextVersionMinor, 1)
|
|
|
|
} else {
|
|
|
|
glfw.WindowHint(glfw.ClientAPI, glfw.NoAPI)
|
|
|
|
}
|
2018-12-18 17:01:16 +01:00
|
|
|
|
|
|
|
decorated := glfw.False
|
|
|
|
if u.isInitWindowDecorated() {
|
|
|
|
decorated = glfw.True
|
|
|
|
}
|
|
|
|
glfw.WindowHint(glfw.Decorated, decorated)
|
|
|
|
|
2019-11-30 16:07:41 +01:00
|
|
|
transparent := glfw.False
|
|
|
|
if u.isInitScreenTransparent() {
|
|
|
|
transparent = glfw.True
|
|
|
|
}
|
|
|
|
glfw.WindowHint(glfw.TransparentFramebuffer, transparent)
|
|
|
|
u.graphics.SetTransparent(u.isInitScreenTransparent())
|
|
|
|
|
2019-01-06 16:21:59 +01:00
|
|
|
resizable := glfw.False
|
|
|
|
if u.isInitWindowResizable() {
|
|
|
|
resizable = glfw.True
|
|
|
|
}
|
|
|
|
glfw.WindowHint(glfw.Resizable, resizable)
|
|
|
|
|
2019-11-10 11:26:46 +01:00
|
|
|
// Set the window visible explicitly or the application freezes on Wayland (#974).
|
2019-11-17 11:58:41 +01:00
|
|
|
if os.Getenv("WAYLAND_DISPLAY") != "" {
|
|
|
|
glfw.WindowHint(glfw.Visible, glfw.True)
|
|
|
|
}
|
2019-11-10 11:26:46 +01:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
if err := u.createWindow(); err != nil {
|
2018-12-18 17:01:16 +01:00
|
|
|
return err
|
|
|
|
}
|
2018-12-17 19:07:13 +01:00
|
|
|
|
2018-12-18 17:01:16 +01:00
|
|
|
if i := u.getInitIconImages(); i != nil {
|
|
|
|
u.window.SetIcon(i)
|
2018-12-18 15:16:06 +01:00
|
|
|
}
|
|
|
|
|
2018-10-07 20:58:15 +02:00
|
|
|
// Get the monitor before showing the window.
|
|
|
|
//
|
|
|
|
// On Windows, there are two types of windows:
|
|
|
|
//
|
|
|
|
// active window: The window that has input-focus and attached to the calling thread.
|
|
|
|
// foreground window: The window that has input-focus: this can be in another process
|
|
|
|
//
|
|
|
|
// currentMonitor returns the monitor for the active window when possible and then the monitor for
|
|
|
|
// the foreground window as fallback. In the current situation, the current window is hidden and
|
|
|
|
// there is not the active window but the foreground window. After showing the current window, the
|
2019-11-26 03:45:52 +01:00
|
|
|
// current window will be the active window. Thus, currentMonitor result varies before and after
|
2018-10-07 20:58:15 +02:00
|
|
|
// showing the window.
|
2019-12-07 11:28:15 +01:00
|
|
|
m = u.currentMonitor()
|
|
|
|
mx, my = m.GetPos()
|
|
|
|
v = m.GetVideoMode()
|
2018-10-07 20:24:12 +02:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// The game is in window mode (not fullscreen mode) at the first state.
|
|
|
|
// Don't refer u.initFullscreen here to avoid some GLFW problems.
|
|
|
|
u.setScreenSize(width, height, scale, false, u.vsync)
|
|
|
|
|
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 15:03:13 +01:00
|
|
|
// Get the window size before showing it. Showing the window might change the current monitor which
|
|
|
|
// affects deviceDependentWindowSize result.
|
2019-12-08 17:06:25 +01:00
|
|
|
w, h := u.window.GetSize()
|
2018-10-14 09:59:14 +02:00
|
|
|
|
2017-06-29 19:14:14 +02:00
|
|
|
u.title = title
|
2016-05-06 05:23:48 +02:00
|
|
|
u.window.SetTitle(title)
|
2014-12-14 10:57:29 +01:00
|
|
|
|
2019-11-30 16:21:11 +01:00
|
|
|
x, y := u.getInitWindowPosition()
|
|
|
|
if x == invalidPos || y == invalidPos {
|
2019-11-30 15:22:23 +01:00
|
|
|
x = mx + (v.Width-w)/2
|
|
|
|
y = my + (v.Height-h)/3
|
|
|
|
}
|
2019-02-24 15:26:07 +01:00
|
|
|
// Adjusting the position is needed only when the monitor is primary. (#829)
|
|
|
|
if mx == 0 && my == 0 {
|
|
|
|
x, y = adjustWindowPosition(x, y)
|
|
|
|
}
|
2016-05-06 05:23:48 +02:00
|
|
|
u.window.SetPos(x, y)
|
2019-12-08 15:00:10 +01:00
|
|
|
u.window.Show()
|
2019-01-06 16:21:59 +01:00
|
|
|
|
2016-07-23 14:01:30 +02:00
|
|
|
return nil
|
2019-12-07 11:28:15 +01:00
|
|
|
})
|
2018-12-28 06:08:44 +01:00
|
|
|
|
2019-11-25 15:13:44 +01:00
|
|
|
var w unsafe.Pointer
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2018-12-28 06:08:44 +01:00
|
|
|
w = u.nativeWindow()
|
|
|
|
return nil
|
|
|
|
})
|
2019-06-23 19:03:12 +02:00
|
|
|
u.graphics.SetWindow(w)
|
2019-04-09 05:21:37 +02:00
|
|
|
return u.loop(context)
|
2016-02-25 19:09:23 +01:00
|
|
|
}
|
|
|
|
|
2018-02-24 16:00:59 +01:00
|
|
|
// getScale must be called from the main thread.
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) getScale() float64 {
|
2019-02-07 09:19:24 +01:00
|
|
|
if !u.isFullscreen() {
|
2017-06-29 18:39:15 +02:00
|
|
|
return u.scale
|
|
|
|
}
|
|
|
|
if u.fullscreenScale == 0 {
|
2019-12-08 13:49:42 +01:00
|
|
|
v := u.currentMonitor().GetVideoMode()
|
2019-12-08 16:11:18 +01:00
|
|
|
sw := u.toDeviceIndependentPixel(float64(v.Width)) / float64(u.screenWidthInDP)
|
|
|
|
sh := u.toDeviceIndependentPixel(float64(v.Height)) / float64(u.screenHeightInDP)
|
2017-06-29 18:39:15 +02:00
|
|
|
s := sw
|
|
|
|
if s > sh {
|
|
|
|
s = sh
|
|
|
|
}
|
|
|
|
u.fullscreenScale = s
|
|
|
|
}
|
|
|
|
return u.fullscreenScale
|
|
|
|
}
|
|
|
|
|
2018-02-24 16:00:59 +01:00
|
|
|
// actualScreenScale must be called from the main thread.
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) actualScreenScale() float64 {
|
2019-12-08 12:24:20 +01:00
|
|
|
return u.getScale() * u.deviceScaleFactor()
|
2015-01-01 17:20:20 +01:00
|
|
|
}
|
|
|
|
|
2019-04-09 05:49:31 +02:00
|
|
|
func (u *UserInterface) updateSize(context driver.UIContext) {
|
2019-12-08 16:11:18 +01:00
|
|
|
u.setScreenSize(u.screenWidthInDP, u.screenHeightInDP, u.scale, u.isFullscreen(), u.vsync)
|
2019-12-07 11:28:15 +01:00
|
|
|
|
2018-02-01 18:08:03 +01:00
|
|
|
sizeChanged := false
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2018-10-02 20:26:08 +02:00
|
|
|
if !u.toChangeSize {
|
2018-02-01 18:08:03 +01:00
|
|
|
return nil
|
|
|
|
}
|
2018-08-05 14:19:22 +02:00
|
|
|
|
2018-02-24 16:42:37 +01:00
|
|
|
u.toChangeSize = false
|
2018-02-01 18:08:03 +01:00
|
|
|
sizeChanged = true
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if sizeChanged {
|
2019-12-07 09:32:25 +01:00
|
|
|
actualScale := 0.0
|
|
|
|
_ = u.t.Call(func() error {
|
|
|
|
actualScale = u.actualScreenScale()
|
|
|
|
return nil
|
|
|
|
})
|
2019-12-08 16:11:18 +01:00
|
|
|
context.SetSize(u.screenWidthInDP, u.screenHeightInDP, actualScale)
|
2018-02-01 18:08:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-09 05:21:37 +02:00
|
|
|
func (u *UserInterface) update(context driver.UIContext) error {
|
2016-05-17 18:30:27 +02:00
|
|
|
shouldClose := false
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2016-05-17 18:30:27 +02:00
|
|
|
shouldClose = u.window.ShouldClose()
|
2016-07-23 14:01:30 +02:00
|
|
|
return nil
|
2016-05-17 18:30:27 +02:00
|
|
|
})
|
|
|
|
if shouldClose {
|
2019-04-07 03:54:05 +02:00
|
|
|
return driver.RegularTermination
|
2016-05-17 18:30:27 +02:00
|
|
|
}
|
2016-05-18 04:56:43 +02:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
if u.isInitFullscreen() {
|
2019-12-08 16:11:18 +01:00
|
|
|
u.setScreenSize(u.screenWidthInDP, u.screenHeightInDP, u.scale, true, u.vsync)
|
2019-12-07 11:28:15 +01:00
|
|
|
u.setInitFullscreen(false)
|
|
|
|
}
|
2017-07-15 14:43:33 +02:00
|
|
|
|
2018-02-01 18:08:03 +01:00
|
|
|
// This call is needed for initialization.
|
2019-04-09 05:49:31 +02:00
|
|
|
u.updateSize(context)
|
2016-05-18 04:56:43 +02:00
|
|
|
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2019-03-31 12:51:53 +02:00
|
|
|
glfw.PollEvents()
|
2019-12-11 19:20:20 +01:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
u.input.update(u.window)
|
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 08:10:43 +01:00
|
|
|
defer hooks.ResumeAudio()
|
2019-03-31 12:51:53 +02:00
|
|
|
|
2017-08-02 16:37:50 +02:00
|
|
|
for !u.isRunnableInBackground() && u.window.GetAttrib(glfw.Focused) == 0 {
|
2019-12-08 08:10:43 +01:00
|
|
|
hooks.SuspendAudio()
|
2016-05-06 05:23:48 +02:00
|
|
|
// Wait for an arbitrary period to avoid busy loop.
|
|
|
|
time.Sleep(time.Second / 60)
|
2019-03-31 12:51:53 +02:00
|
|
|
glfw.PollEvents()
|
2016-05-06 05:23:48 +02:00
|
|
|
if u.window.ShouldClose() {
|
2016-07-23 14:01:30 +02:00
|
|
|
return nil
|
2016-05-06 05:23:48 +02:00
|
|
|
}
|
2016-03-14 16:31:46 +01:00
|
|
|
}
|
2016-07-23 14:01:30 +02:00
|
|
|
return nil
|
2017-03-03 03:15:07 +01:00
|
|
|
})
|
2019-04-09 05:21:37 +02:00
|
|
|
if err := context.Update(func() {
|
2018-02-01 18:08:03 +01:00
|
|
|
// The offscreens must be updated every frame (#490).
|
2019-04-09 05:49:31 +02:00
|
|
|
u.updateSize(context)
|
2017-11-09 17:18:41 +01:00
|
|
|
}); err != nil {
|
2016-09-01 17:53:05 +02:00
|
|
|
return err
|
|
|
|
}
|
2019-01-06 16:21:59 +01:00
|
|
|
|
|
|
|
// Update the screen size when the window is resizable.
|
2019-12-07 18:44:14 +01:00
|
|
|
var w, h int
|
|
|
|
_ = u.t.Call(func() error {
|
|
|
|
w, h = u.reqWidth, u.reqHeight
|
|
|
|
return nil
|
|
|
|
})
|
2019-12-07 11:28:15 +01:00
|
|
|
if w != 0 || h != 0 {
|
|
|
|
u.setScreenSize(w, h, u.scale, u.isFullscreen(), u.vsync)
|
|
|
|
}
|
2019-12-07 18:44:14 +01:00
|
|
|
_ = u.t.Call(func() error {
|
|
|
|
u.reqWidth = 0
|
|
|
|
u.reqHeight = 0
|
|
|
|
return nil
|
|
|
|
})
|
2016-09-01 17:53:05 +02:00
|
|
|
return nil
|
2015-01-07 17:32:37 +01:00
|
|
|
}
|
|
|
|
|
2019-04-09 05:21:37 +02:00
|
|
|
func (u *UserInterface) loop(context driver.UIContext) error {
|
2016-09-01 18:07:41 +02:00
|
|
|
defer func() {
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2016-09-01 18:07:41 +02:00
|
|
|
glfw.Terminate()
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}()
|
2016-08-31 19:38:47 +02:00
|
|
|
for {
|
2019-11-17 06:10:35 +01:00
|
|
|
var unfocused bool
|
|
|
|
|
|
|
|
// On Windows, the focusing state might be always false (#987).
|
|
|
|
// On Windows, even if a window is in another workspace, vsync seems to work.
|
|
|
|
// Then let's assume the window is alwasy 'focused' as a workaround.
|
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
unfocused = u.window.GetAttrib(glfw.Focused) == glfw.False
|
|
|
|
}
|
2019-11-14 05:07:10 +01:00
|
|
|
|
|
|
|
var t1, t2 time.Time
|
|
|
|
|
|
|
|
if unfocused {
|
|
|
|
t1 = time.Now()
|
|
|
|
}
|
2019-04-09 05:21:37 +02:00
|
|
|
if err := u.update(context); err != nil {
|
2016-08-31 19:38:47 +02:00
|
|
|
return err
|
|
|
|
}
|
2018-09-30 07:43:53 +02:00
|
|
|
|
2019-06-05 17:19:12 +02:00
|
|
|
_ = u.t.Call(func() error {
|
2017-03-03 02:50:47 +01:00
|
|
|
u.swapBuffers()
|
|
|
|
return nil
|
|
|
|
})
|
2019-11-14 05:07:10 +01:00
|
|
|
if unfocused {
|
|
|
|
t2 = time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
// When a window is not focused, SwapBuffers might return immediately and CPU might be busy.
|
|
|
|
// Mitigate this by sleeping (#982).
|
|
|
|
if unfocused {
|
|
|
|
d := t2.Sub(t1)
|
|
|
|
const wait = time.Second / 60
|
|
|
|
if d < wait {
|
|
|
|
time.Sleep(wait - d)
|
|
|
|
}
|
|
|
|
}
|
2016-07-23 14:01:30 +02:00
|
|
|
}
|
2016-03-26 09:24:40 +01:00
|
|
|
}
|
|
|
|
|
2018-02-24 16:00:59 +01:00
|
|
|
// swapBuffers must be called from the main thread.
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) swapBuffers() {
|
2019-04-06 16:03:21 +02:00
|
|
|
if u.graphics.IsGL() {
|
2018-11-12 16:00:10 +01:00
|
|
|
u.window.SwapBuffers()
|
|
|
|
}
|
2014-12-14 10:57:29 +01:00
|
|
|
}
|
2015-02-09 02:25:00 +01:00
|
|
|
|
2019-12-07 09:32:25 +01:00
|
|
|
func (u *UserInterface) setScreenSize(width, height int, scale float64, fullscreen bool, vsync bool) {
|
2019-12-07 11:28:15 +01:00
|
|
|
windowRecreated := false
|
2016-03-22 03:56:40 +01:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
_ = u.t.Call(func() error {
|
2019-12-08 16:11:18 +01:00
|
|
|
if u.screenWidthInDP == width && u.screenHeightInDP == height && u.scale == scale && u.isFullscreen() == fullscreen && u.vsync == vsync && u.lastDeviceScaleFactor == u.deviceScaleFactor() {
|
2019-12-07 11:28:15 +01:00
|
|
|
return nil
|
|
|
|
}
|
2017-10-19 20:25:21 +02:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
if width < 1 {
|
|
|
|
width = 1
|
2017-06-30 03:48:18 +02:00
|
|
|
}
|
2019-12-07 11:28:15 +01:00
|
|
|
if height < 1 {
|
|
|
|
height = 1
|
2017-06-29 19:11:46 +02:00
|
|
|
}
|
2017-10-19 20:25:21 +02:00
|
|
|
|
2019-12-08 16:11:18 +01:00
|
|
|
u.screenWidthInDP = width
|
|
|
|
u.screenHeightInDP = height
|
2019-12-07 11:28:15 +01:00
|
|
|
u.scale = scale
|
|
|
|
u.fullscreenScale = 0
|
|
|
|
u.vsync = vsync
|
2019-12-08 12:24:20 +01:00
|
|
|
u.lastDeviceScaleFactor = u.deviceScaleFactor()
|
2019-12-07 11:28:15 +01:00
|
|
|
|
|
|
|
// To make sure the current existing framebuffers are rendered,
|
|
|
|
// swap buffers here before SetSize is called.
|
|
|
|
u.swapBuffers()
|
|
|
|
|
|
|
|
if fullscreen {
|
|
|
|
if u.origPosX == invalidPos || u.origPosY == invalidPos {
|
|
|
|
u.origPosX, u.origPosY = u.window.GetPos()
|
|
|
|
}
|
|
|
|
m := u.currentMonitor()
|
|
|
|
v := m.GetVideoMode()
|
|
|
|
u.window.SetMonitor(m, 0, 0, v.Width, v.Height, v.RefreshRate)
|
|
|
|
|
|
|
|
// Swapping buffer is necesary to prevent the image lag (#1004).
|
2019-12-07 18:04:00 +01:00
|
|
|
// TODO: This might not work when vsync is disabled.
|
2019-12-07 11:28:15 +01:00
|
|
|
if u.graphics.IsGL() {
|
2017-10-19 20:25:21 +02:00
|
|
|
glfw.PollEvents()
|
2019-12-07 11:28:15 +01:00
|
|
|
u.swapBuffers()
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if u.window.GetMonitor() != nil {
|
|
|
|
if u.graphics.IsGL() {
|
|
|
|
// When OpenGL is used, swapping buffer is enough to solve the image-lag
|
|
|
|
// issue (#1004). Rather, recreating window destroys GPU resources.
|
2019-12-07 18:04:00 +01:00
|
|
|
// TODO: This might not work when vsync is disabled.
|
2019-12-07 11:28:15 +01:00
|
|
|
u.window.SetMonitor(nil, 0, 0, 16, 16, 0)
|
|
|
|
glfw.PollEvents()
|
|
|
|
u.swapBuffers()
|
|
|
|
} else {
|
|
|
|
// Recreate the window since an image lag remains after coming back from
|
|
|
|
// fullscreen (#1004).
|
|
|
|
if u.window != nil {
|
|
|
|
u.window.Destroy()
|
|
|
|
u.window = nil
|
|
|
|
}
|
|
|
|
if err := u.createWindow(); err != nil {
|
|
|
|
// TODO: This should return an error.
|
|
|
|
panic(fmt.Sprintf("glfw: failed to recreate window: %v", err))
|
|
|
|
}
|
2019-12-08 15:00:10 +01:00
|
|
|
u.window.Show()
|
2019-12-07 11:28:15 +01:00
|
|
|
windowRecreated = true
|
2017-10-19 20:25:21 +02:00
|
|
|
}
|
2017-06-29 17:35:34 +02:00
|
|
|
}
|
2018-10-05 20:53:34 +02:00
|
|
|
|
2019-12-08 17:06:25 +01:00
|
|
|
// On Windows, giving a too small width doesn't call a callback (#165).
|
|
|
|
// To prevent hanging up, return asap if the width is too small.
|
|
|
|
// 252 is an arbitrary number and I guess this is small enough.
|
|
|
|
minWindowWidth := 252
|
|
|
|
if u.window.GetAttrib(glfw.Decorated) == glfw.False {
|
|
|
|
minWindowWidth = 1
|
|
|
|
}
|
|
|
|
windowWidthInDP := width
|
|
|
|
s := scale * u.deviceScaleFactor()
|
|
|
|
if int(float64(width)*s) < minWindowWidth {
|
|
|
|
windowWidthInDP = int(math.Ceil(float64(minWindowWidth) / s))
|
|
|
|
}
|
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
oldW, oldH := u.window.GetSize()
|
2019-12-08 17:06:25 +01:00
|
|
|
newW := int(u.toDeviceDependentPixel(float64(windowWidthInDP) * u.getScale()))
|
|
|
|
newH := int(u.toDeviceDependentPixel(float64(u.screenHeightInDP) * u.getScale()))
|
|
|
|
if oldW != newW || oldH != newH {
|
2019-12-07 11:28:15 +01:00
|
|
|
ch := make(chan struct{})
|
|
|
|
u.window.SetFramebufferSizeCallback(func(_ *glfw.Window, _, _ int) {
|
|
|
|
u.window.SetFramebufferSizeCallback(nil)
|
|
|
|
close(ch)
|
|
|
|
})
|
2019-12-08 13:49:42 +01:00
|
|
|
u.window.SetSize(newW, newH)
|
2019-12-07 11:28:15 +01:00
|
|
|
event:
|
|
|
|
for {
|
|
|
|
glfw.PollEvents()
|
|
|
|
select {
|
|
|
|
case <-ch:
|
|
|
|
break event
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if u.origPosX != invalidPos && u.origPosY != invalidPos {
|
|
|
|
x := u.origPosX
|
|
|
|
y := u.origPosY
|
2019-12-01 17:06:55 +01:00
|
|
|
u.window.SetPos(x, y)
|
2019-12-07 11:28:15 +01:00
|
|
|
// Dirty hack for macOS (#703). Rendering doesn't work correctly with one SetPos, but
|
|
|
|
// work with two or more SetPos.
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
u.window.SetPos(x+1, y)
|
|
|
|
u.window.SetPos(x, y)
|
|
|
|
}
|
|
|
|
u.origPosX = invalidPos
|
|
|
|
u.origPosY = invalidPos
|
2019-12-01 17:06:55 +01:00
|
|
|
}
|
2018-10-05 20:53:34 +02:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
// Window title might be lost on macOS after coming back from fullscreen.
|
|
|
|
u.window.SetTitle(u.title)
|
|
|
|
}
|
2018-07-14 13:50:09 +02:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
if u.graphics.IsGL() {
|
|
|
|
// SwapInterval is affected by the current monitor of the window.
|
|
|
|
// This needs to be called at least after SetMonitor.
|
|
|
|
// Without SwapInterval after SetMonitor, vsynch doesn't work (#375).
|
|
|
|
//
|
|
|
|
// TODO: (#405) If triple buffering is needed, SwapInterval(0) should be called,
|
|
|
|
// but is this correct? If glfw.SwapInterval(0) and the driver doesn't support triple
|
|
|
|
// buffering, what will happen?
|
|
|
|
if u.vsync {
|
|
|
|
glfw.SwapInterval(1)
|
|
|
|
} else {
|
|
|
|
glfw.SwapInterval(0)
|
|
|
|
}
|
2018-11-12 16:00:10 +01:00
|
|
|
}
|
2019-12-07 11:28:15 +01:00
|
|
|
u.graphics.SetVsyncEnabled(vsync)
|
2017-08-12 08:39:41 +02:00
|
|
|
|
2019-12-07 11:28:15 +01:00
|
|
|
u.toChangeSize = true
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
if windowRecreated {
|
|
|
|
u.graphics.SetWindow(u.nativeWindow())
|
|
|
|
}
|
2015-02-09 02:25:00 +01:00
|
|
|
}
|
2018-10-05 16:59:30 +02:00
|
|
|
|
2018-10-06 13:04:48 +02:00
|
|
|
// currentMonitor returns the monitor most suitable with the current window.
|
2018-10-05 16:59:30 +02:00
|
|
|
//
|
|
|
|
// currentMonitor must be called on the main thread.
|
2019-04-07 03:42:55 +02:00
|
|
|
func (u *UserInterface) currentMonitor() *glfw.Monitor {
|
2018-10-05 16:59:30 +02:00
|
|
|
w := u.window
|
|
|
|
if m := w.GetMonitor(); m != nil {
|
|
|
|
return m
|
|
|
|
}
|
2018-10-06 15:42:28 +02:00
|
|
|
// Get the monitor which the current window belongs to. This requires OS API.
|
2018-12-21 18:32:02 +01:00
|
|
|
return u.currentMonitorFromPosition()
|
2018-10-05 16:59:30 +02:00
|
|
|
}
|
2019-04-07 11:28:50 +02:00
|
|
|
|
2019-11-30 15:22:23 +01:00
|
|
|
func (u *UserInterface) SetWindowPosition(x, y int) {
|
|
|
|
if !u.isRunning() {
|
2019-11-30 16:11:46 +01:00
|
|
|
u.setInitWindowPosition(x, y)
|
2019-11-30 15:22:23 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
_ = u.t.Call(func() error {
|
2019-11-30 16:39:04 +01:00
|
|
|
if u.isFullscreen() {
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-30 15:22:23 +01:00
|
|
|
u.window.SetPos(x, y)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-11-30 14:37:53 +01:00
|
|
|
func (u *UserInterface) WindowPosition() (int, int) {
|
|
|
|
if !u.isRunning() {
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: WindowPosition can't be called before the main loop starts")
|
2019-11-30 14:37:53 +01:00
|
|
|
}
|
|
|
|
x, y := 0, 0
|
|
|
|
_ = u.t.Call(func() error {
|
2019-11-30 16:39:04 +01:00
|
|
|
if u.isFullscreen() {
|
|
|
|
x, y = u.origPosX, u.origPosY
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-30 14:37:53 +01:00
|
|
|
x, y = u.window.GetPos()
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return x, y
|
|
|
|
}
|
|
|
|
|
2019-11-30 16:07:41 +01:00
|
|
|
func (u *UserInterface) SetScreenTransparent(transparent bool) {
|
|
|
|
if !u.isRunning() {
|
|
|
|
u.setInitScreenTransparent(transparent)
|
|
|
|
return
|
|
|
|
}
|
2019-11-30 19:34:09 +01:00
|
|
|
panic("glfw: SetScreenTransparent can't be called after the main loop starts")
|
2019-11-30 16:07:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) IsScreenTransparent() bool {
|
|
|
|
if !u.isRunning() {
|
|
|
|
return u.isInitScreenTransparent()
|
|
|
|
}
|
|
|
|
val := false
|
|
|
|
_ = u.t.Call(func() error {
|
|
|
|
val = u.window.GetAttrib(glfw.TransparentFramebuffer) == glfw.True
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return val
|
|
|
|
}
|
|
|
|
|
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-04-07 11:28:50 +02:00
|
|
|
}
|