2022-05-27 13:38:45 +02:00
|
|
|
// SPDX-License-Identifier: Zlib
|
|
|
|
// SPDX-FileCopyrightText: 2002-2006 Marcus Geelnard
|
|
|
|
// SPDX-FileCopyrightText: 2006-2019 Camilla Löwy
|
|
|
|
// SPDX-FileCopyrightText: 2022 The Ebitengine Authors
|
2022-04-19 17:49:43 +02:00
|
|
|
|
|
|
|
package glfwwin
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"unsafe"
|
|
|
|
|
|
|
|
"golang.org/x/sys/windows"
|
2022-05-29 12:13:31 +02:00
|
|
|
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/microsoftgdk"
|
2022-04-19 17:49:43 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func createKeyTables() {
|
|
|
|
for i := range _glfw.win32.keycodes {
|
|
|
|
_glfw.win32.keycodes[i] = -1
|
|
|
|
}
|
|
|
|
for i := range _glfw.win32.scancodes {
|
|
|
|
_glfw.win32.keycodes[i] = -1
|
|
|
|
}
|
|
|
|
|
|
|
|
_glfw.win32.keycodes[0x00B] = Key0
|
|
|
|
_glfw.win32.keycodes[0x002] = Key1
|
|
|
|
_glfw.win32.keycodes[0x003] = Key2
|
|
|
|
_glfw.win32.keycodes[0x004] = Key3
|
|
|
|
_glfw.win32.keycodes[0x005] = Key4
|
|
|
|
_glfw.win32.keycodes[0x006] = Key5
|
|
|
|
_glfw.win32.keycodes[0x007] = Key6
|
|
|
|
_glfw.win32.keycodes[0x008] = Key7
|
|
|
|
_glfw.win32.keycodes[0x009] = Key8
|
|
|
|
_glfw.win32.keycodes[0x00A] = Key9
|
|
|
|
_glfw.win32.keycodes[0x01E] = KeyA
|
|
|
|
_glfw.win32.keycodes[0x030] = KeyB
|
|
|
|
_glfw.win32.keycodes[0x02E] = KeyC
|
|
|
|
_glfw.win32.keycodes[0x020] = KeyD
|
|
|
|
_glfw.win32.keycodes[0x012] = KeyE
|
|
|
|
_glfw.win32.keycodes[0x021] = KeyF
|
|
|
|
_glfw.win32.keycodes[0x022] = KeyG
|
|
|
|
_glfw.win32.keycodes[0x023] = KeyH
|
|
|
|
_glfw.win32.keycodes[0x017] = KeyI
|
|
|
|
_glfw.win32.keycodes[0x024] = KeyJ
|
|
|
|
_glfw.win32.keycodes[0x025] = KeyK
|
|
|
|
_glfw.win32.keycodes[0x026] = KeyL
|
|
|
|
_glfw.win32.keycodes[0x032] = KeyM
|
|
|
|
_glfw.win32.keycodes[0x031] = KeyN
|
|
|
|
_glfw.win32.keycodes[0x018] = KeyO
|
|
|
|
_glfw.win32.keycodes[0x019] = KeyP
|
|
|
|
_glfw.win32.keycodes[0x010] = KeyQ
|
|
|
|
_glfw.win32.keycodes[0x013] = KeyR
|
|
|
|
_glfw.win32.keycodes[0x01F] = KeyS
|
|
|
|
_glfw.win32.keycodes[0x014] = KeyT
|
|
|
|
_glfw.win32.keycodes[0x016] = KeyU
|
|
|
|
_glfw.win32.keycodes[0x02F] = KeyV
|
|
|
|
_glfw.win32.keycodes[0x011] = KeyW
|
|
|
|
_glfw.win32.keycodes[0x02D] = KeyX
|
|
|
|
_glfw.win32.keycodes[0x015] = KeyY
|
|
|
|
_glfw.win32.keycodes[0x02C] = KeyZ
|
|
|
|
|
|
|
|
_glfw.win32.keycodes[0x028] = KeyApostrophe
|
|
|
|
_glfw.win32.keycodes[0x02B] = KeyBackslash
|
|
|
|
_glfw.win32.keycodes[0x033] = KeyComma
|
|
|
|
_glfw.win32.keycodes[0x00D] = KeyEqual
|
|
|
|
_glfw.win32.keycodes[0x029] = KeyGraveAccent
|
|
|
|
_glfw.win32.keycodes[0x01A] = KeyLeftBracket
|
|
|
|
_glfw.win32.keycodes[0x00C] = KeyMinus
|
|
|
|
_glfw.win32.keycodes[0x034] = KeyPeriod
|
|
|
|
_glfw.win32.keycodes[0x01B] = KeyRightBracket
|
|
|
|
_glfw.win32.keycodes[0x027] = KeySemicolon
|
|
|
|
_glfw.win32.keycodes[0x035] = KeySlash
|
|
|
|
_glfw.win32.keycodes[0x056] = KeyWorld2
|
|
|
|
|
|
|
|
_glfw.win32.keycodes[0x00E] = KeyBackspace
|
|
|
|
_glfw.win32.keycodes[0x153] = KeyDelete
|
|
|
|
_glfw.win32.keycodes[0x14F] = KeyEnd
|
|
|
|
_glfw.win32.keycodes[0x01C] = KeyEnter
|
|
|
|
_glfw.win32.keycodes[0x001] = KeyEscape
|
|
|
|
_glfw.win32.keycodes[0x147] = KeyHome
|
|
|
|
_glfw.win32.keycodes[0x152] = KeyInsert
|
|
|
|
_glfw.win32.keycodes[0x15D] = KeyMenu
|
|
|
|
_glfw.win32.keycodes[0x151] = KeyPageDown
|
|
|
|
_glfw.win32.keycodes[0x149] = KeyPageUp
|
|
|
|
_glfw.win32.keycodes[0x045] = KeyPause
|
|
|
|
_glfw.win32.keycodes[0x039] = KeySpace
|
|
|
|
_glfw.win32.keycodes[0x00F] = KeyTab
|
|
|
|
_glfw.win32.keycodes[0x03A] = KeyCapsLock
|
|
|
|
_glfw.win32.keycodes[0x145] = KeyNumLock
|
|
|
|
_glfw.win32.keycodes[0x046] = KeyScrollLock
|
|
|
|
_glfw.win32.keycodes[0x03B] = KeyF1
|
|
|
|
_glfw.win32.keycodes[0x03C] = KeyF2
|
|
|
|
_glfw.win32.keycodes[0x03D] = KeyF3
|
|
|
|
_glfw.win32.keycodes[0x03E] = KeyF4
|
|
|
|
_glfw.win32.keycodes[0x03F] = KeyF5
|
|
|
|
_glfw.win32.keycodes[0x040] = KeyF6
|
|
|
|
_glfw.win32.keycodes[0x041] = KeyF7
|
|
|
|
_glfw.win32.keycodes[0x042] = KeyF8
|
|
|
|
_glfw.win32.keycodes[0x043] = KeyF9
|
|
|
|
_glfw.win32.keycodes[0x044] = KeyF10
|
|
|
|
_glfw.win32.keycodes[0x057] = KeyF11
|
|
|
|
_glfw.win32.keycodes[0x058] = KeyF12
|
|
|
|
_glfw.win32.keycodes[0x064] = KeyF13
|
|
|
|
_glfw.win32.keycodes[0x065] = KeyF14
|
|
|
|
_glfw.win32.keycodes[0x066] = KeyF15
|
|
|
|
_glfw.win32.keycodes[0x067] = KeyF16
|
|
|
|
_glfw.win32.keycodes[0x068] = KeyF17
|
|
|
|
_glfw.win32.keycodes[0x069] = KeyF18
|
|
|
|
_glfw.win32.keycodes[0x06A] = KeyF19
|
|
|
|
_glfw.win32.keycodes[0x06B] = KeyF20
|
|
|
|
_glfw.win32.keycodes[0x06C] = KeyF21
|
|
|
|
_glfw.win32.keycodes[0x06D] = KeyF22
|
|
|
|
_glfw.win32.keycodes[0x06E] = KeyF23
|
|
|
|
_glfw.win32.keycodes[0x076] = KeyF24
|
|
|
|
_glfw.win32.keycodes[0x038] = KeyLeftAlt
|
|
|
|
_glfw.win32.keycodes[0x01D] = KeyLeftControl
|
|
|
|
_glfw.win32.keycodes[0x02A] = KeyLeftShift
|
|
|
|
_glfw.win32.keycodes[0x15B] = KeyLeftSuper
|
|
|
|
_glfw.win32.keycodes[0x137] = KeyPrintScreen
|
|
|
|
_glfw.win32.keycodes[0x138] = KeyRightAlt
|
|
|
|
_glfw.win32.keycodes[0x11D] = KeyRightControl
|
|
|
|
_glfw.win32.keycodes[0x036] = KeyRightShift
|
|
|
|
_glfw.win32.keycodes[0x15C] = KeyRightSuper
|
|
|
|
_glfw.win32.keycodes[0x150] = KeyDown
|
|
|
|
_glfw.win32.keycodes[0x14B] = KeyLeft
|
|
|
|
_glfw.win32.keycodes[0x14D] = KeyRight
|
|
|
|
_glfw.win32.keycodes[0x148] = KeyUp
|
|
|
|
|
|
|
|
_glfw.win32.keycodes[0x052] = KeyKP0
|
|
|
|
_glfw.win32.keycodes[0x04F] = KeyKP1
|
|
|
|
_glfw.win32.keycodes[0x050] = KeyKP2
|
|
|
|
_glfw.win32.keycodes[0x051] = KeyKP3
|
|
|
|
_glfw.win32.keycodes[0x04B] = KeyKP4
|
|
|
|
_glfw.win32.keycodes[0x04C] = KeyKP5
|
|
|
|
_glfw.win32.keycodes[0x04D] = KeyKP6
|
|
|
|
_glfw.win32.keycodes[0x047] = KeyKP7
|
|
|
|
_glfw.win32.keycodes[0x048] = KeyKP8
|
|
|
|
_glfw.win32.keycodes[0x049] = KeyKP9
|
|
|
|
_glfw.win32.keycodes[0x04E] = KeyKPAdd
|
|
|
|
_glfw.win32.keycodes[0x053] = KeyKPDecimal
|
|
|
|
_glfw.win32.keycodes[0x135] = KeyKPDivide
|
|
|
|
_glfw.win32.keycodes[0x11C] = KeyKPEnter
|
|
|
|
_glfw.win32.keycodes[0x059] = KeyKPEqual
|
|
|
|
_glfw.win32.keycodes[0x037] = KeyKPMultiply
|
|
|
|
_glfw.win32.keycodes[0x04A] = KeyKPSubtract
|
|
|
|
|
|
|
|
for scancode := 0; scancode < 512; scancode++ {
|
|
|
|
if _glfw.win32.keycodes[scancode] > 0 {
|
|
|
|
_glfw.win32.scancodes[_glfw.win32.keycodes[scancode]] = scancode
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func createHelperWindow() error {
|
2022-08-07 07:32:00 +02:00
|
|
|
h, err := _CreateWindowExW(_WS_EX_OVERLAPPEDWINDOW, _GLFW_WNDCLASSNAME, "GLFW message window", _WS_CLIPSIBLINGS|_WS_CLIPCHILDREN, 0, 0, 1, 1, 0, 0, _glfw.win32.instance, nil)
|
2022-04-19 17:49:43 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_glfw.win32.helperWindowHandle = h
|
|
|
|
|
|
|
|
// HACK: The command to the first ShowWindow call is ignored if the parent
|
|
|
|
// process passed along a STARTUPINFO, so clear that with a no-op call
|
|
|
|
_ShowWindow(_glfw.win32.helperWindowHandle, _SW_HIDE)
|
|
|
|
|
|
|
|
// Register for HID device notifications
|
2022-05-29 12:13:31 +02:00
|
|
|
if !microsoftgdk.IsXbox() {
|
2022-05-27 08:55:21 +02:00
|
|
|
_GUID_DEVINTERFACE_HID := windows.GUID{
|
|
|
|
Data1: 0x4d1e55b2,
|
|
|
|
Data2: 0xf16f,
|
|
|
|
Data3: 0x11cf,
|
|
|
|
Data4: [...]byte{0x88, 0xcb, 0x00, 0x11, 0x11, 0x00, 0x00, 0x30},
|
|
|
|
}
|
|
|
|
|
|
|
|
var dbi _DEV_BROADCAST_DEVICEINTERFACE_W
|
|
|
|
dbi.dbcc_size = uint32(unsafe.Sizeof(dbi))
|
|
|
|
dbi.dbcc_devicetype = _DBT_DEVTYP_DEVICEINTERFACE
|
|
|
|
dbi.dbcc_classguid = _GUID_DEVINTERFACE_HID
|
|
|
|
notify, err := _RegisterDeviceNotificationW(windows.Handle(_glfw.win32.helperWindowHandle), unsafe.Pointer(&dbi), _DEVICE_NOTIFY_WINDOW_HANDLE)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_glfw.win32.deviceNotificationHandle = notify
|
2022-04-19 17:49:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var msg _MSG
|
|
|
|
for _PeekMessageW(&msg, _glfw.win32.helperWindowHandle, 0, 0, _PM_REMOVE) {
|
|
|
|
_TranslateMessage(&msg)
|
|
|
|
_DispatchMessageW(&msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func isWindowsVersionOrGreaterWin32(major, minor, sp uint16) bool {
|
|
|
|
osvi := _OSVERSIONINFOEXW{
|
|
|
|
dwMajorVersion: uint32(major),
|
|
|
|
dwMinorVersion: uint32(minor),
|
|
|
|
wServicePackMajor: sp,
|
|
|
|
}
|
|
|
|
osvi.dwOSVersionInfoSize = uint32(unsafe.Sizeof(osvi))
|
|
|
|
var mask uint32 = _VER_MAJORVERSION | _VER_MINORVERSION | _VER_SERVICEPACKMAJOR
|
|
|
|
cond := _VerSetConditionMask(0, _VER_MAJORVERSION, _VER_GREATER_EQUAL)
|
|
|
|
cond = _VerSetConditionMask(cond, _VER_MINORVERSION, _VER_GREATER_EQUAL)
|
|
|
|
cond = _VerSetConditionMask(cond, _VER_SERVICEPACKMAJOR, _VER_GREATER_EQUAL)
|
|
|
|
// HACK: Use RtlVerifyVersionInfo instead of VerifyVersionInfoW as the
|
|
|
|
// latter lies unless the user knew to embed a non-default manifest
|
|
|
|
// announcing support for Windows 10 via supportedOS GUID
|
|
|
|
return _RtlVerifyVersionInfo(&osvi, mask, cond) == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func isWindows10BuildOrGreaterWin32(build uint16) bool {
|
|
|
|
osvi := _OSVERSIONINFOEXW{
|
|
|
|
dwMajorVersion: 10,
|
|
|
|
dwMinorVersion: 0,
|
|
|
|
dwBuildNumber: uint32(build),
|
|
|
|
}
|
|
|
|
osvi.dwOSVersionInfoSize = uint32(unsafe.Sizeof(osvi))
|
|
|
|
var mask uint32 = _VER_MAJORVERSION | _VER_MINORVERSION | _VER_BUILDNUMBER
|
|
|
|
cond := _VerSetConditionMask(0, _VER_MAJORVERSION, _VER_GREATER_EQUAL)
|
|
|
|
cond = _VerSetConditionMask(cond, _VER_MINORVERSION, _VER_GREATER_EQUAL)
|
|
|
|
cond = _VerSetConditionMask(cond, _VER_BUILDNUMBER, _VER_GREATER_EQUAL)
|
|
|
|
|
|
|
|
// HACK: Use RtlVerifyVersionInfo instead of VerifyVersionInfoW as the
|
|
|
|
// latter lies unless the user knew to embed a non-default manifest
|
|
|
|
// announcing support for Windows 10 via supportedOS GUID
|
|
|
|
return _RtlVerifyVersionInfo(&osvi, mask, cond) == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func platformInit() error {
|
2022-05-24 17:34:46 +02:00
|
|
|
// Changing the foreground lock timeout was removed from the original code.
|
|
|
|
// See https://github.com/glfw/glfw/commit/58b48a3a00d9c2a5ca10cc23069a71d8773cc7a4
|
2022-04-19 17:49:43 +02:00
|
|
|
|
2022-08-07 07:32:00 +02:00
|
|
|
m, err := _GetModuleHandleExW(_GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS|_GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, unsafe.Pointer(&_glfw))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_glfw.win32.instance = _HINSTANCE(m)
|
|
|
|
|
2022-04-19 17:49:43 +02:00
|
|
|
createKeyTables()
|
|
|
|
|
|
|
|
if isWindows10CreatorsUpdateOrGreaterWin32() {
|
2022-05-29 12:13:31 +02:00
|
|
|
if !microsoftgdk.IsXbox() {
|
2022-06-01 16:11:04 +02:00
|
|
|
// Ignore the error as SetProcessDpiAwarenessContext returns an error on Steam Deck (#2113).
|
|
|
|
// This seems an issue in Wine and/or Proton, but there is nothing we can do.
|
|
|
|
_ = _SetProcessDpiAwarenessContext(_DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2)
|
2022-04-19 17:49:43 +02:00
|
|
|
}
|
|
|
|
} else if _IsWindows8Point1OrGreater() {
|
2022-10-01 18:52:21 +02:00
|
|
|
if err := _SetProcessDpiAwareness(_PROCESS_PER_MONITOR_DPI_AWARE); err != nil && !errors.Is(err, handleError(windows.E_ACCESSDENIED)) {
|
2022-04-19 17:49:43 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else if _IsWindowsVistaOrGreater() {
|
|
|
|
_SetProcessDPIAware()
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := registerWindowClassWin32(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := createHelperWindow(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-05-29 12:13:31 +02:00
|
|
|
if microsoftgdk.IsXbox() {
|
2022-05-29 08:38:13 +02:00
|
|
|
// On Xbox, APIs to get monitors are not available.
|
|
|
|
// Create a pseudo monitor instance instead.
|
2022-05-29 12:13:31 +02:00
|
|
|
w, h := microsoftgdk.MonitorResolution()
|
2022-05-29 08:38:13 +02:00
|
|
|
mode := &VidMode{
|
|
|
|
Width: w,
|
|
|
|
Height: h,
|
|
|
|
RedBits: 8,
|
|
|
|
GreenBits: 8,
|
|
|
|
BlueBits: 8,
|
|
|
|
RefreshRate: 0, // TODO: Is it possible to get an appropriate refresh rate?
|
|
|
|
}
|
|
|
|
m := &Monitor{
|
2022-05-29 08:53:43 +02:00
|
|
|
name: "Xbox Monitor",
|
|
|
|
modes: []*VidMode{mode},
|
2022-05-29 08:38:13 +02:00
|
|
|
}
|
|
|
|
if err := inputMonitor(m, Connected, _GLFW_INSERT_LAST); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-05-27 15:11:05 +02:00
|
|
|
} else {
|
|
|
|
if err := pollMonitorsWin32(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-04-19 17:49:43 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func platformTerminate() error {
|
|
|
|
if _glfw.win32.deviceNotificationHandle != 0 {
|
|
|
|
if err := _UnregisterDeviceNotification(_glfw.win32.deviceNotificationHandle); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if _glfw.win32.helperWindowHandle != 0 {
|
2022-07-30 11:30:46 +02:00
|
|
|
if !microsoftgdk.IsXbox() {
|
|
|
|
if err := _DestroyWindow(_glfw.win32.helperWindowHandle); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-04-19 17:49:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := unregisterWindowClassWin32(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
terminateWGL()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|