2016-06-18 22:04:38 +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.
|
|
|
|
|
2022-02-06 08:08:22 +01:00
|
|
|
//go:build !ios && !ebitencbackend
|
|
|
|
// +build !ios,!ebitencbackend
|
2016-06-18 22:04:38 +02:00
|
|
|
|
2022-02-06 08:08:22 +01:00
|
|
|
package ui
|
2016-06-18 22:04:38 +02:00
|
|
|
|
2018-10-06 15:42:28 +02:00
|
|
|
// #cgo CFLAGS: -x objective-c
|
|
|
|
// #cgo LDFLAGS: -framework AppKit
|
|
|
|
//
|
|
|
|
// #import <AppKit/AppKit.h>
|
|
|
|
//
|
2021-12-29 14:08:59 +01:00
|
|
|
// @interface EbitenWindowDelegate : NSObject <NSWindowDelegate>
|
|
|
|
// @end
|
|
|
|
//
|
|
|
|
// @implementation EbitenWindowDelegate {
|
|
|
|
// id<NSWindowDelegate> origDelegate_;
|
|
|
|
// bool origResizable_;
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (instancetype)initWithOrigDelegate:(id<NSWindowDelegate>)origDelegate {
|
|
|
|
// self = [super init];
|
|
|
|
// if (self != nil) {
|
|
|
|
// origDelegate_ = origDelegate;
|
|
|
|
// }
|
|
|
|
// return self;
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// // The method set of origDelegate_ must sync with GLFWWindowDelegate's implementation.
|
|
|
|
// // See cocoa_window.m in GLFW.
|
|
|
|
// - (BOOL)windowShouldClose:(id)sender {
|
|
|
|
// return [origDelegate_ windowShouldClose:sender];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidResize:(NSNotification *)notification {
|
|
|
|
// [origDelegate_ windowDidResize:notification];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidMove:(NSNotification *)notification {
|
|
|
|
// [origDelegate_ windowDidMove:notification];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidMiniaturize:(NSNotification *)notification {
|
|
|
|
// [origDelegate_ windowDidMiniaturize:notification];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidDeminiaturize:(NSNotification *)notification {
|
|
|
|
// [origDelegate_ windowDidDeminiaturize:notification];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidBecomeKey:(NSNotification *)notification {
|
|
|
|
// [origDelegate_ windowDidBecomeKey:notification];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidResignKey:(NSNotification *)notification {
|
|
|
|
// [origDelegate_ windowDidResignKey:notification];
|
|
|
|
// }
|
|
|
|
// - (void)windowDidChangeOcclusionState:(NSNotification* )notification {
|
|
|
|
// [origDelegate_ windowDidChangeOcclusionState:notification];
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (void)pushResizableState:(NSWindow*)window {
|
|
|
|
// origResizable_ = window.styleMask & NSWindowStyleMaskResizable;
|
|
|
|
// if (!origResizable_) {
|
|
|
|
// window.styleMask |= NSWindowStyleMaskResizable;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (void)popResizableState:(NSWindow*)window {
|
|
|
|
// if (!origResizable_) {
|
|
|
|
// window.styleMask &= ~NSWindowStyleMaskResizable;
|
|
|
|
// }
|
|
|
|
// origResizable_ = false;
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (void)windowWillEnterFullScreen:(NSNotification *)notification {
|
|
|
|
// NSWindow* window = (NSWindow*)[notification object];
|
|
|
|
// [self pushResizableState:window];
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (void)windowDidEnterFullScreen:(NSNotification *)notification {
|
|
|
|
// NSWindow* window = (NSWindow*)[notification object];
|
|
|
|
// [self popResizableState:window];
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (void)windowWillExitFullScreen:(NSNotification *)notification {
|
|
|
|
// NSWindow* window = (NSWindow*)[notification object];
|
|
|
|
// [self pushResizableState:window];
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// - (void)windowDidExitFullScreen:(NSNotification *)notification {
|
|
|
|
// NSWindow* window = (NSWindow*)[notification object];
|
|
|
|
// [self popResizableState:window];
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// @end
|
|
|
|
//
|
|
|
|
// static void initializeWindow(uintptr_t windowPtr) {
|
|
|
|
// NSWindow* window = (NSWindow*)windowPtr;
|
|
|
|
// // This delegate is never released. This assumes that the window lives until the process lives.
|
|
|
|
// window.delegate = [[EbitenWindowDelegate alloc] initWithOrigDelegate:window.delegate];
|
|
|
|
// }
|
|
|
|
//
|
2020-09-03 17:43:51 +02:00
|
|
|
// static void currentMonitorPos(uintptr_t windowPtr, int* x, int* y) {
|
2021-07-06 21:34:19 +02:00
|
|
|
// @autoreleasepool {
|
|
|
|
// NSScreen* screen = [NSScreen mainScreen];
|
|
|
|
// if (windowPtr) {
|
|
|
|
// NSWindow* window = (NSWindow*)windowPtr;
|
|
|
|
// if ([window isVisible]) {
|
|
|
|
// // When the window is visible, the window is already initialized.
|
|
|
|
// // [NSScreen mainScreen] sometimes tells a lie when the window is put across monitors (#703).
|
|
|
|
// screen = [window screen];
|
|
|
|
// }
|
2018-12-18 18:07:45 +01:00
|
|
|
// }
|
2021-07-06 21:34:19 +02:00
|
|
|
// NSDictionary* screenDictionary = [screen deviceDescription];
|
|
|
|
// NSNumber* screenID = [screenDictionary objectForKey:@"NSScreenNumber"];
|
|
|
|
// CGDirectDisplayID aID = [screenID unsignedIntValue];
|
|
|
|
// const CGRect bounds = CGDisplayBounds(aID);
|
|
|
|
// *x = bounds.origin.x;
|
|
|
|
// *y = bounds.origin.y;
|
2018-10-07 22:38:46 +02:00
|
|
|
// }
|
2018-10-06 15:42:28 +02:00
|
|
|
// }
|
2021-04-18 10:35:46 +02:00
|
|
|
//
|
2021-08-08 10:45:44 +02:00
|
|
|
// static bool isNativeFullscreen(uintptr_t windowPtr) {
|
|
|
|
// if (!windowPtr) {
|
|
|
|
// return false;
|
|
|
|
// }
|
|
|
|
// NSWindow* window = (NSWindow*)windowPtr;
|
|
|
|
// return (window.styleMask & NSWindowStyleMaskFullScreen) != 0;
|
2021-04-18 10:35:46 +02:00
|
|
|
// }
|
2021-05-02 07:50:50 +02:00
|
|
|
//
|
2021-09-17 19:21:24 +02:00
|
|
|
// static void setNativeFullscreen(uintptr_t windowPtr, bool fullscreen) {
|
|
|
|
// NSWindow* window = (NSWindow*)windowPtr;
|
|
|
|
// if (((window.styleMask & NSWindowStyleMaskFullScreen) != 0) == fullscreen) {
|
|
|
|
// return;
|
|
|
|
// }
|
2021-12-29 14:08:59 +01:00
|
|
|
//
|
|
|
|
// // Even though EbitenWindowDelegate is used, this hack is still required.
|
|
|
|
// // toggleFullscreen doesn't work when the window is not resizable.
|
2021-09-17 19:21:24 +02:00
|
|
|
// bool origResizable = window.styleMask & NSWindowStyleMaskResizable;
|
|
|
|
// if (!origResizable) {
|
|
|
|
// window.styleMask |= NSWindowStyleMaskResizable;
|
|
|
|
// }
|
|
|
|
// [window toggleFullScreen:nil];
|
|
|
|
// if (!origResizable) {
|
|
|
|
// window.styleMask &= ~NSWindowStyleMaskResizable;
|
|
|
|
// }
|
2021-09-18 15:11:26 +02:00
|
|
|
// }
|
2021-09-18 12:20:45 +02:00
|
|
|
//
|
2021-09-18 15:11:26 +02:00
|
|
|
// static void adjustViewSize(uintptr_t windowPtr) {
|
|
|
|
// NSWindow* window = (NSWindow*)windowPtr;
|
2021-09-18 12:20:45 +02:00
|
|
|
// if ((window.styleMask & NSWindowStyleMaskFullScreen) == 0) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
//
|
2021-11-27 10:47:01 +01:00
|
|
|
// // Apparently, adjusting the view size is not needed as of macOS 12 (#1745).
|
|
|
|
// static int majorVersion = 0;
|
|
|
|
// if (majorVersion == 0) {
|
|
|
|
// majorVersion = [[NSProcessInfo processInfo] operatingSystemVersion].majorVersion;
|
|
|
|
// }
|
|
|
|
// if (majorVersion >= 12) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
//
|
2021-09-18 12:20:45 +02:00
|
|
|
// // Reduce the view height (#1745).
|
|
|
|
// // https://stackoverflow.com/questions/27758027/sprite-kit-serious-fps-issue-in-full-screen-mode-on-os-x
|
2021-09-18 15:11:26 +02:00
|
|
|
// CGSize windowSize = [window frame].size;
|
2021-09-18 12:20:45 +02:00
|
|
|
// NSView* view = [window contentView];
|
2021-09-18 15:11:26 +02:00
|
|
|
// CGSize viewSize = [view frame].size;
|
|
|
|
// if (windowSize.width != viewSize.width || windowSize.height != viewSize.height) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
// viewSize.width--;
|
|
|
|
// [view setFrameSize:viewSize];
|
2021-09-18 12:20:45 +02:00
|
|
|
//
|
|
|
|
// // NSColor.blackColor (0, 0, 0, 1) didn't work.
|
|
|
|
// // Use the transparent color instead.
|
|
|
|
// [window setBackgroundColor: [NSColor colorWithSRGBRed:0 green:0 blue:0 alpha:0]];
|
2021-09-17 19:21:24 +02:00
|
|
|
// }
|
|
|
|
//
|
2021-05-02 07:50:50 +02:00
|
|
|
// static void setNativeCursor(int cursorID) {
|
|
|
|
// id cursor = [[NSCursor class] performSelector:@selector(arrowCursor)];
|
|
|
|
// switch (cursorID) {
|
|
|
|
// case 0:
|
|
|
|
// cursor = [[NSCursor class] performSelector:@selector(arrowCursor)];
|
|
|
|
// break;
|
|
|
|
// case 1:
|
|
|
|
// cursor = [[NSCursor class] performSelector:@selector(IBeamCursor)];
|
|
|
|
// break;
|
|
|
|
// case 2:
|
|
|
|
// cursor = [[NSCursor class] performSelector:@selector(crosshairCursor)];
|
|
|
|
// break;
|
|
|
|
// case 3:
|
|
|
|
// cursor = [[NSCursor class] performSelector:@selector(pointingHandCursor)];
|
|
|
|
// break;
|
|
|
|
// case 4:
|
|
|
|
// cursor = [[NSCursor class] performSelector:@selector(_windowResizeEastWestCursor)];
|
|
|
|
// break;
|
|
|
|
// case 5:
|
|
|
|
// cursor = [[NSCursor class] performSelector:@selector(_windowResizeNorthSouthCursor)];
|
|
|
|
// break;
|
|
|
|
// }
|
|
|
|
// [cursor push];
|
|
|
|
// }
|
2022-02-07 13:56:16 +01:00
|
|
|
//
|
|
|
|
// static void currentMouseLocation(int* x, int* y) {
|
|
|
|
// NSPoint location = [NSEvent mouseLocation];
|
|
|
|
// *x = (int)(location.x);
|
|
|
|
// *y = (int)(location.y);
|
|
|
|
// }
|
2021-12-29 14:08:59 +01:00
|
|
|
//
|
|
|
|
// static void setAllowFullscreen(uintptr_t windowPtr, bool allowFullscreen) {
|
|
|
|
// NSWindow* window = (NSWindow*)windowPtr;
|
|
|
|
// if (allowFullscreen) {
|
|
|
|
// window.collectionBehavior |= NSWindowCollectionBehaviorFullScreenPrimary;
|
|
|
|
// } else {
|
|
|
|
// window.collectionBehavior &= ~NSWindowCollectionBehaviorFullScreenPrimary;
|
|
|
|
// }
|
|
|
|
// }
|
2018-10-06 15:42:28 +02:00
|
|
|
import "C"
|
|
|
|
|
|
|
|
import (
|
2020-10-03 19:35:13 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/glfw"
|
2018-10-06 15:42:28 +02:00
|
|
|
)
|
|
|
|
|
2021-09-15 05:47:04 +02:00
|
|
|
// clearVideoModeScaleCache must be called from the main thread.
|
|
|
|
func clearVideoModeScaleCache() {}
|
|
|
|
|
2021-10-10 08:38:13 +02:00
|
|
|
// dipFromGLFWMonitorPixel must be called from the main thread.
|
|
|
|
func (u *UserInterface) dipFromGLFWMonitorPixel(x float64, monitor *glfw.Monitor) float64 {
|
2021-10-09 09:49:47 +02:00
|
|
|
return x
|
2020-09-18 17:21:08 +02:00
|
|
|
}
|
|
|
|
|
2021-10-10 08:38:13 +02:00
|
|
|
// dipFromGLFWPixel must be called from the main thread.
|
|
|
|
func (u *UserInterface) dipFromGLFWPixel(x float64, monitor *glfw.Monitor) float64 {
|
2021-09-14 18:03:04 +02:00
|
|
|
// NOTE: On macOS, GLFW exposes the device independent coordinate system.
|
|
|
|
// Thus, the conversion functions are unnecessary,
|
|
|
|
// however we still need the deviceScaleFactor internally
|
|
|
|
// so we can create and maintain a HiDPI frame buffer.
|
2020-09-18 17:31:34 +02:00
|
|
|
return x
|
|
|
|
}
|
|
|
|
|
2021-10-10 08:38:13 +02:00
|
|
|
// dipToGLFWPixel must be called from the main thread.
|
|
|
|
func (u *UserInterface) dipToGLFWPixel(x float64, monitor *glfw.Monitor) float64 {
|
2020-09-18 17:21:08 +02:00
|
|
|
return x
|
2016-06-18 22:04:38 +02:00
|
|
|
}
|
2017-04-18 17:51:15 +02:00
|
|
|
|
2022-02-07 15:51:40 +01:00
|
|
|
func (u *UserInterface) adjustWindowPosition(x, y int, monitor *glfw.Monitor) (int, int) {
|
2017-04-18 17:51:15 +02:00
|
|
|
return x, y
|
|
|
|
}
|
2018-10-06 15:42:28 +02:00
|
|
|
|
2022-02-07 15:31:08 +01:00
|
|
|
func initialMonitorByOS() (*glfw.Monitor, error) {
|
2022-02-07 13:56:16 +01:00
|
|
|
var cx, cy C.int
|
|
|
|
C.currentMouseLocation(&cx, &cy)
|
|
|
|
x, y := int(cx), int(cy)
|
|
|
|
|
|
|
|
// Flip Y.
|
|
|
|
for _, m := range ensureMonitors() {
|
|
|
|
if m.x == 0 && m.y == 0 {
|
|
|
|
y = -y
|
|
|
|
y += m.vm.Height
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the monitor including the cursor.
|
|
|
|
for _, m := range ensureMonitors() {
|
|
|
|
w, h := m.vm.Width, m.vm.Height
|
|
|
|
if x >= m.x && x < m.x+w && y >= m.y && y < m.y+h {
|
2022-02-07 15:31:08 +01:00
|
|
|
return m.m, nil
|
2022-02-07 13:56:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-07 15:31:08 +01:00
|
|
|
return nil, nil
|
2021-10-09 09:49:47 +02:00
|
|
|
}
|
|
|
|
|
2022-02-08 16:26:49 +01:00
|
|
|
func monitorFromWindowByOS(w *glfw.Window) *glfw.Monitor {
|
2022-02-07 13:56:16 +01:00
|
|
|
var x, y C.int
|
2022-02-08 16:26:49 +01:00
|
|
|
C.currentMonitorPos(C.uintptr_t(w.GetCocoaWindow()), &x, &y)
|
2021-09-09 04:11:56 +02:00
|
|
|
for _, m := range ensureMonitors() {
|
|
|
|
if int(x) == m.x && int(y) == m.y {
|
|
|
|
return m.m
|
2018-10-06 15:42:28 +02:00
|
|
|
}
|
|
|
|
}
|
2020-08-23 20:27:38 +02:00
|
|
|
return nil
|
2018-10-06 15:42:28 +02:00
|
|
|
}
|
2018-11-12 16:00:10 +01:00
|
|
|
|
2020-09-03 17:43:51 +02:00
|
|
|
func (u *UserInterface) nativeWindow() uintptr {
|
2018-12-28 06:08:44 +01:00
|
|
|
return u.window.GetCocoaWindow()
|
2018-11-12 16:00:10 +01:00
|
|
|
}
|
2021-04-18 10:35:46 +02:00
|
|
|
|
|
|
|
func (u *UserInterface) isNativeFullscreen() bool {
|
2021-08-08 10:45:44 +02:00
|
|
|
return bool(C.isNativeFullscreen(C.uintptr_t(u.window.GetCocoaWindow())))
|
2021-04-18 10:35:46 +02:00
|
|
|
}
|
2021-05-02 07:50:50 +02:00
|
|
|
|
2022-02-06 09:43:52 +01:00
|
|
|
func (u *UserInterface) setNativeCursor(shape CursorShape) {
|
2021-05-02 07:50:50 +02:00
|
|
|
C.setNativeCursor(C.int(shape))
|
|
|
|
}
|
2021-09-17 19:21:24 +02:00
|
|
|
|
|
|
|
func (u *UserInterface) isNativeFullscreenAvailable() bool {
|
2021-10-31 10:21:00 +01:00
|
|
|
// TODO: If the window is transparent, we should use GLFW's windowed fullscreen (#1822, #1857).
|
|
|
|
// However, if the user clicks the green button, should this window be in native fullscreen mode?
|
|
|
|
return true
|
2021-09-17 19:21:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *UserInterface) setNativeFullscreen(fullscreen bool) {
|
|
|
|
// Toggling fullscreen might ignore events like keyUp. Ensure that events are fired.
|
2021-09-20 14:01:17 +02:00
|
|
|
glfw.WaitEventsTimeout(0.1)
|
2021-09-17 19:21:24 +02:00
|
|
|
C.setNativeFullscreen(C.uintptr_t(u.window.GetCocoaWindow()), C.bool(fullscreen))
|
|
|
|
}
|
2021-09-18 15:11:26 +02:00
|
|
|
|
|
|
|
func (u *UserInterface) adjustViewSize() {
|
2022-03-19 15:55:14 +01:00
|
|
|
if graphicsDriver().IsGL() {
|
2021-09-18 16:59:37 +02:00
|
|
|
return
|
|
|
|
}
|
2021-09-18 15:11:26 +02:00
|
|
|
C.adjustViewSize(C.uintptr_t(u.window.GetCocoaWindow()))
|
|
|
|
}
|
2021-09-24 19:46:18 +02:00
|
|
|
|
2021-12-29 14:08:59 +01:00
|
|
|
func (u *UserInterface) setWindowResizingModeForOS(mode WindowResizingMode) {
|
|
|
|
allowFullscreen := mode == WindowResizingModeOnlyFullscreenEnabled ||
|
|
|
|
mode == WindowResizingModeEnabled
|
|
|
|
C.setAllowFullscreen(C.uintptr_t(u.window.GetCocoaWindow()), C.bool(allowFullscreen))
|
|
|
|
}
|
|
|
|
|
2021-09-24 19:46:18 +02:00
|
|
|
func initializeWindowAfterCreation(w *glfw.Window) {
|
2021-12-29 14:08:59 +01:00
|
|
|
// TODO: Register NSWindowWillEnterFullScreenNotification and so on.
|
|
|
|
// Enable resizing temporary before making the window fullscreen.
|
|
|
|
C.initializeWindow(C.uintptr_t(w.GetCocoaWindow()))
|
2021-09-24 19:46:18 +02:00
|
|
|
}
|