ebiten/internal/ui/window_glfw.go

528 lines
9.7 KiB
Go
Raw Normal View History

2019-12-24 16:05:56 +01:00
// Copyright 2019 The Ebiten Authors
//
// 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.
//go:build !android && !ios && !js && !nintendosdk && !playstation5
2019-12-24 16:05:56 +01:00
package ui
2019-12-24 16:05:56 +01:00
import (
"image"
"runtime"
2019-12-24 16:05:56 +01:00
2020-10-03 19:35:13 +02:00
"github.com/hajimehoshi/ebiten/v2/internal/glfw"
2019-12-24 16:05:56 +01:00
)
type glfwWindow struct {
ui *UserInterface
2019-12-24 16:05:56 +01:00
}
func (w *glfwWindow) IsDecorated() bool {
if w.ui.isTerminated() {
return false
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
return w.ui.isInitWindowDecorated()
}
var v bool
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
a, err := w.ui.window.GetAttrib(glfw.Decorated)
if err != nil {
w.ui.setError(err)
return
}
v = a == glfw.True
2019-12-24 16:05:56 +01:00
})
return v
}
func (w *glfwWindow) SetDecorated(decorated bool) {
if w.ui.isTerminated() {
return
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
w.ui.setInitWindowDecorated(decorated)
return
}
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.setWindowDecorated(decorated); err != nil {
w.ui.setError(err)
return
}
2019-12-24 16:05:56 +01:00
})
}
func (w *glfwWindow) ResizingMode() WindowResizingMode {
if w.ui.isTerminated() {
return 0
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
w.ui.m.Lock()
mode := w.ui.windowResizingMode
w.ui.m.Unlock()
return mode
2019-12-24 16:05:56 +01:00
}
var mode WindowResizingMode
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
mode = w.ui.windowResizingMode
2019-12-24 16:05:56 +01:00
})
return mode
2019-12-24 16:05:56 +01:00
}
func (w *glfwWindow) SetResizingMode(mode WindowResizingMode) {
if w.ui.isTerminated() {
return
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
w.ui.m.Lock()
w.ui.windowResizingMode = mode
w.ui.m.Unlock()
2019-12-24 16:05:56 +01:00
return
}
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.setWindowResizingMode(mode); err != nil {
w.ui.setError(err)
return
}
2019-12-24 16:05:56 +01:00
})
}
func (w *glfwWindow) IsFloating() bool {
if w.ui.isTerminated() {
return false
}
if !w.ui.isRunning() {
return w.ui.isInitWindowFloating()
}
var v bool
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
a, err := w.ui.window.GetAttrib(glfw.Floating)
if err != nil {
w.ui.setError(err)
return
}
v = a == glfw.True
})
return v
}
func (w *glfwWindow) SetFloating(floating bool) {
if w.ui.isTerminated() {
return
}
if !w.ui.isRunning() {
w.ui.setInitWindowFloating(floating)
return
}
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.setWindowFloating(floating); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) IsMaximized() bool {
if w.ui.isTerminated() {
return false
}
if !w.ui.isRunning() {
return w.ui.isInitWindowMaximized()
}
if w.ResizingMode() != WindowResizingModeEnabled {
2022-02-10 11:31:38 +01:00
return false
}
var v bool
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
m, err := w.ui.isWindowMaximized()
if err != nil {
w.ui.setError(err)
return
}
v = m
})
return v
}
func (w *glfwWindow) Maximize() {
if w.ui.isTerminated() {
return
}
2022-02-10 11:31:38 +01:00
// Do not allow maximizing the window when the window is not resizable.
// On Windows, it is possible to restore the window from being maximized by mouse-dragging,
// and this can be an unexpected behavior (#1990).
if w.ResizingMode() != WindowResizingModeEnabled {
return
}
if !w.ui.isWindowMaximizable() {
return
}
if !w.ui.isRunning() {
w.ui.setInitWindowMaximized(true)
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.maximizeWindow(); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) IsMinimized() bool {
if !w.ui.isRunning() {
return false
}
var v bool
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
a, err := w.ui.window.GetAttrib(glfw.Iconified)
if err != nil {
w.ui.setError(err)
return
}
v = a == glfw.True
})
return v
}
func (w *glfwWindow) Minimize() {
if !w.ui.isRunning() {
// Do nothing
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.iconifyWindow(); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) Restore() {
if w.ui.isTerminated() {
return
}
if !w.ui.isWindowMaximizable() {
return
}
if !w.ui.isRunning() {
// Do nothing
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.restoreWindow(); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) SetMonitor(monitor *Monitor) {
if monitor == nil {
panic("ui: monitor cannot be nil at SetMonitor")
}
if w.ui.isTerminated() {
return
}
if !w.ui.isRunning() {
w.ui.setInitMonitor(monitor)
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.setWindowMonitor(monitor); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) Position() (int, int) {
if w.ui.isTerminated() {
return 0, 0
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
panic("ui: WindowPosition can't be called before the main loop starts")
2019-12-24 16:05:56 +01:00
}
var x, y int
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
f, err := w.ui.isFullscreen()
if err != nil {
w.ui.setError(err)
return
}
2019-12-24 16:05:56 +01:00
var wx, wy int
if f {
wx, wy = w.ui.origWindowPos()
2019-12-24 16:05:56 +01:00
} else {
x, y, err := w.ui.window.GetPos()
if err != nil {
w.ui.setError(err)
return
}
wx, wy = x, y
}
m, err := w.ui.currentMonitor()
if err != nil {
w.ui.setError(err)
return
2019-12-24 16:05:56 +01:00
}
wx -= m.boundsInGLFWPixels.Min.X
wy -= m.boundsInGLFWPixels.Min.Y
2024-03-03 15:27:02 +01:00
s := m.DeviceScaleFactor()
xf := dipFromGLFWPixel(float64(wx), s)
yf := dipFromGLFWPixel(float64(wy), s)
2019-12-24 16:05:56 +01:00
x, y = int(xf), int(yf)
})
return x, y
}
func (w *glfwWindow) SetPosition(x, y int) {
if w.ui.isTerminated() {
return
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
w.ui.setInitWindowPositionInDIP(x, y)
2019-12-24 16:05:56 +01:00
return
}
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
m, err := w.ui.currentMonitor()
if err != nil {
w.ui.setError(err)
return
}
if err := w.ui.setWindowPositionInDIP(x, y, m); err != nil {
w.ui.setError(err)
return
}
2019-12-24 16:05:56 +01:00
})
}
func (w *glfwWindow) Size() (int, int) {
if w.ui.isTerminated() {
return 0, 0
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
ww, wh := w.ui.getInitWindowSizeInDIP()
return w.ui.adjustWindowSizeBasedOnSizeLimitsInDIP(ww, wh)
2019-12-24 16:05:56 +01:00
}
var ww, wh int
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
2022-09-24 19:03:27 +02:00
// Unlike origWindowPos, origWindow{Width,Height}InDPI are always updated via the callback.
ww = w.ui.origWindowWidthInDIP
wh = w.ui.origWindowHeightInDIP
})
2019-12-24 16:05:56 +01:00
return ww, wh
}
func (w *glfwWindow) SetSize(width, height int) {
if w.ui.isTerminated() {
return
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
// If the window is initially maximized, the set size is ignored anyway.
w.ui.setInitWindowSizeInDIP(width, height)
2019-12-24 16:05:56 +01:00
return
}
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
m, err := w.ui.isWindowMaximized()
if err != nil {
w.ui.setError(err)
return
}
if m && runtime.GOOS != "darwin" {
return
}
if err := w.ui.setWindowSizeInDIP(width, height, true); err != nil {
w.ui.setError(err)
return
}
2020-10-16 21:21:07 +02:00
})
2019-12-24 16:05:56 +01:00
}
func (w *glfwWindow) SizeLimits() (minw, minh, maxw, maxh int) {
return w.ui.getWindowSizeLimitsInDIP()
}
func (w *glfwWindow) SetSizeLimits(minw, minh, maxw, maxh int) {
if w.ui.isTerminated() {
return
}
if !w.ui.setWindowSizeLimitsInDIP(minw, minh, maxw, maxh) {
return
}
if !w.ui.isRunning() {
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.updateWindowSizeLimits(); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) SetIcon(iconImages []image.Image) {
if w.ui.isTerminated() {
return
}
// The icons are actually set at (*UserInterface).loop.
w.ui.setIconImages(iconImages)
2019-12-24 16:05:56 +01:00
}
func (w *glfwWindow) SetTitle(title string) {
if w.ui.isTerminated() {
return
}
2019-12-24 16:05:56 +01:00
if !w.ui.isRunning() {
w.ui.m.Lock()
w.ui.title = title
w.ui.m.Unlock()
2019-12-24 16:05:56 +01:00
return
}
w.ui.title = title
2022-12-28 07:59:53 +01:00
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.setWindowTitle(title); err != nil {
w.ui.setError(err)
return
}
2019-12-24 16:05:56 +01:00
})
}
func (w *glfwWindow) SetClosingHandled(handled bool) {
w.ui.setWindowClosingHandled(handled)
}
func (w *glfwWindow) IsClosingHandled() bool {
return w.ui.isWindowClosingHandled()
}
func (w *glfwWindow) SetMousePassthrough(enabled bool) {
if w.ui.isTerminated() {
return
}
if !w.ui.isRunning() {
w.ui.setInitWindowMousePassthrough(enabled)
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.setWindowMousePassthrough(enabled); err != nil {
w.ui.setError(err)
return
}
})
}
func (w *glfwWindow) IsMousePassthrough() bool {
if w.ui.isTerminated() {
return false
}
if !w.ui.isRunning() {
return w.ui.isInitWindowMousePassthrough()
}
var v bool
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
a, err := w.ui.window.GetAttrib(glfw.MousePassthrough)
if err != nil {
w.ui.setError(err)
return
}
v = a == glfw.True
})
return v
}
func (w *glfwWindow) RequestAttention() {
if w.ui.isTerminated() {
return
}
if !w.ui.isRunning() {
// Do nothing
return
}
w.ui.mainThread.Call(func() {
if w.ui.isTerminated() {
return
}
if err := w.ui.window.RequestAttention(); err != nil {
w.ui.setError(err)
return
}
})
}