mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-12 20:18:59 +01:00
internal: add a new package glfwwin
internal/glfwwin is a pure Go implementation of GLFW for Windows. Updates #1764
This commit is contained in:
parent
58c871df38
commit
edd617f80e
1939
internal/glfwwin/api_windows.go
Normal file
1939
internal/glfwwin/api_windows.go
Normal file
File diff suppressed because it is too large
Load Diff
622
internal/glfwwin/context_windows.go
Normal file
622
internal/glfwwin/context_windows.go
Normal file
@ -0,0 +1,622 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"golang.org/x/sys/windows"
|
||||||
|
)
|
||||||
|
|
||||||
|
func checkValidContextConfig(ctxconfig *ctxconfig) error {
|
||||||
|
if ctxconfig.share != nil {
|
||||||
|
if ctxconfig.client == NoAPI || ctxconfig.share.context.client == NoAPI {
|
||||||
|
return NoWindowContext
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.source != NativeContextAPI &&
|
||||||
|
ctxconfig.source != EGLContextAPI &&
|
||||||
|
ctxconfig.source != OSMesaContextAPI {
|
||||||
|
return fmt.Errorf("glfwwin: invalid context creation API 0x%08X: %w", ctxconfig.source, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.client != NoAPI &&
|
||||||
|
ctxconfig.client != OpenGLAPI &&
|
||||||
|
ctxconfig.client != OpenGLESAPI {
|
||||||
|
return fmt.Errorf("glfwwin: invalid client API 0x%08X: %w", ctxconfig.client, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.client == OpenGLAPI {
|
||||||
|
if (ctxconfig.major < 1 || ctxconfig.minor < 0) ||
|
||||||
|
(ctxconfig.major == 1 && ctxconfig.minor > 5) ||
|
||||||
|
(ctxconfig.major == 2 && ctxconfig.minor > 1) ||
|
||||||
|
(ctxconfig.major == 3 && ctxconfig.minor > 3) {
|
||||||
|
// OpenGL 1.0 is the smallest valid version
|
||||||
|
// OpenGL 1.x series ended with version 1.5
|
||||||
|
// OpenGL 2.x series ended with version 2.1
|
||||||
|
// OpenGL 3.x series ended with version 3.3
|
||||||
|
// For now, let everything else through
|
||||||
|
|
||||||
|
return fmt.Errorf("glfwwin: invalid OpenGL version %d.%d: %w", ctxconfig.major, ctxconfig.minor, InvalidValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.profile != 0 {
|
||||||
|
if ctxconfig.profile != OpenGLCoreProfile && ctxconfig.profile != OpenGLCompatProfile {
|
||||||
|
return fmt.Errorf("glfwwin: invalid OpenGL profile 0x%08X: %w", ctxconfig.profile, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.major <= 2 || (ctxconfig.major == 3 && ctxconfig.minor < 2) {
|
||||||
|
// Desktop OpenGL context profiles are only defined for version 3.2
|
||||||
|
// and above
|
||||||
|
|
||||||
|
return fmt.Errorf("glfwwin: context profiles are only defined for OpenGL version 3.2 and above: %w", InvalidValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.forward && ctxconfig.major <= 2 {
|
||||||
|
// Forward-compatible contexts are only defined for OpenGL version 3.0 and above
|
||||||
|
return fmt.Errorf("glfwwin: forward-compatibility is only defined for OpenGL version 3.0 and above: %w", InvalidValue)
|
||||||
|
}
|
||||||
|
} else if ctxconfig.client == OpenGLESAPI {
|
||||||
|
if ctxconfig.major < 1 || ctxconfig.minor < 0 ||
|
||||||
|
(ctxconfig.major == 1 && ctxconfig.minor > 1) ||
|
||||||
|
(ctxconfig.major == 2 && ctxconfig.minor > 0) {
|
||||||
|
// OpenGL ES 1.0 is the smallest valid version
|
||||||
|
// OpenGL ES 1.x series ended with version 1.1
|
||||||
|
// OpenGL ES 2.x series ended with version 2.0
|
||||||
|
// For now, let everything else through
|
||||||
|
|
||||||
|
return fmt.Errorf("glfwwin: invalid OpenGL ES version %d.%d: %w", ctxconfig.major, ctxconfig.minor, InvalidValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.robustness != 0 {
|
||||||
|
if ctxconfig.robustness != NoResetNotification && ctxconfig.robustness != LoseContextOnReset {
|
||||||
|
return fmt.Errorf("glfwwin: invalid context robustness mode 0x%08X: %w", ctxconfig.robustness, InvalidEnum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.release != 0 {
|
||||||
|
if ctxconfig.release != ReleaseBehaviorNone && ctxconfig.release != ReleaseBehaviorFlush {
|
||||||
|
return fmt.Errorf("glfwwin: invalid context release behavior 0x%08X: %w", ctxconfig.release, InvalidEnum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func chooseFBConfig(desired *fbconfig, alternatives []*fbconfig) *fbconfig {
|
||||||
|
leastMissing := math.MaxInt32
|
||||||
|
leastColorDiff := math.MaxInt32
|
||||||
|
leastExtraDiff := math.MaxInt32
|
||||||
|
|
||||||
|
var closest *fbconfig
|
||||||
|
for _, current := range alternatives {
|
||||||
|
if desired.stereo && !current.stereo {
|
||||||
|
// Stereo is a hard constraint
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Count number of missing buffers
|
||||||
|
missing := 0
|
||||||
|
|
||||||
|
if desired.alphaBits > 0 && current.alphaBits == 0 {
|
||||||
|
missing++
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.depthBits > 0 && current.depthBits == 0 {
|
||||||
|
missing++
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.stencilBits > 0 && current.stencilBits == 0 {
|
||||||
|
missing++
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.auxBuffers > 0 &&
|
||||||
|
current.auxBuffers < desired.auxBuffers {
|
||||||
|
missing += desired.auxBuffers - current.auxBuffers
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.samples > 0 && current.samples == 0 {
|
||||||
|
// Technically, several multisampling buffers could be
|
||||||
|
// involved, but that's a lower level implementation detail and
|
||||||
|
// not important to us here, so we count them as one
|
||||||
|
missing++
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.transparent != current.transparent {
|
||||||
|
missing++
|
||||||
|
}
|
||||||
|
|
||||||
|
// These polynomials make many small channel size differences matter
|
||||||
|
// less than one large channel size difference
|
||||||
|
|
||||||
|
// Calculate color channel size difference value
|
||||||
|
colorDiff := 0
|
||||||
|
|
||||||
|
if desired.redBits != DontCare {
|
||||||
|
colorDiff += (desired.redBits - current.redBits) *
|
||||||
|
(desired.redBits - current.redBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.greenBits != DontCare {
|
||||||
|
colorDiff += (desired.greenBits - current.greenBits) *
|
||||||
|
(desired.greenBits - current.greenBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.blueBits != DontCare {
|
||||||
|
colorDiff += (desired.blueBits - current.blueBits) *
|
||||||
|
(desired.blueBits - current.blueBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate non-color channel size difference value
|
||||||
|
extraDiff := 0
|
||||||
|
|
||||||
|
if desired.alphaBits != DontCare {
|
||||||
|
extraDiff += (desired.alphaBits - current.alphaBits) *
|
||||||
|
(desired.alphaBits - current.alphaBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.depthBits != DontCare {
|
||||||
|
extraDiff += (desired.depthBits - current.depthBits) *
|
||||||
|
(desired.depthBits - current.depthBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.stencilBits != DontCare {
|
||||||
|
extraDiff += (desired.stencilBits - current.stencilBits) *
|
||||||
|
(desired.stencilBits - current.stencilBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.accumRedBits != DontCare {
|
||||||
|
extraDiff += (desired.accumRedBits - current.accumRedBits) *
|
||||||
|
(desired.accumRedBits - current.accumRedBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.accumGreenBits != DontCare {
|
||||||
|
extraDiff += (desired.accumGreenBits - current.accumGreenBits) *
|
||||||
|
(desired.accumGreenBits - current.accumGreenBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.accumBlueBits != DontCare {
|
||||||
|
extraDiff += (desired.accumBlueBits - current.accumBlueBits) *
|
||||||
|
(desired.accumBlueBits - current.accumBlueBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.accumAlphaBits != DontCare {
|
||||||
|
extraDiff += (desired.accumAlphaBits - current.accumAlphaBits) *
|
||||||
|
(desired.accumAlphaBits - current.accumAlphaBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.samples != DontCare {
|
||||||
|
extraDiff += (desired.samples - current.samples) *
|
||||||
|
(desired.samples - current.samples)
|
||||||
|
}
|
||||||
|
|
||||||
|
if desired.sRGB && !current.sRGB {
|
||||||
|
extraDiff++
|
||||||
|
}
|
||||||
|
|
||||||
|
// Figure out if the current one is better than the best one found so far
|
||||||
|
// Least number of missing buffers is the most important heuristic,
|
||||||
|
// then color buffer size match and lastly size match for other buffers
|
||||||
|
|
||||||
|
if missing < leastMissing {
|
||||||
|
closest = current
|
||||||
|
} else if missing == leastMissing {
|
||||||
|
if (colorDiff < leastColorDiff) || (colorDiff == leastColorDiff && extraDiff < leastExtraDiff) {
|
||||||
|
closest = current
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if current == closest {
|
||||||
|
leastMissing = missing
|
||||||
|
leastColorDiff = colorDiff
|
||||||
|
leastExtraDiff = extraDiff
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return closest
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) refreshContextAttribs(ctxconfig *ctxconfig) (ferr error) {
|
||||||
|
const (
|
||||||
|
GL_COLOR_BUFFER_BIT = 0x00004000
|
||||||
|
GL_CONTEXT_COMPATIBILITY_PROFILE_BIT = 0x00000002
|
||||||
|
GL_CONTEXT_CORE_PROFILE_BIT = 0x00000001
|
||||||
|
GL_CONTEXT_FLAG_DEBUG_BIT = 0x00000002
|
||||||
|
GL_CONTEXT_FLAG_FORWARD_COMPATIBLE_BIT = 0x00000001
|
||||||
|
GL_CONTEXT_FLAG_NO_ERROR_BIT_KHR = 0x00000008
|
||||||
|
GL_CONTEXT_FLAGS = 0x821E
|
||||||
|
GL_CONTEXT_PROFILE_MASK = 0x9126
|
||||||
|
GL_CONTEXT_RELEASE_BEHAVIOR = 0x82FB
|
||||||
|
GL_CONTEXT_RELEASE_BEHAVIOR_FLUSH = 0x82FC
|
||||||
|
GL_LOSE_CONTEXT_ON_RESET_ARB = 0x8252
|
||||||
|
GL_NO_RESET_NOTIFICATION_ARB = 0x8261
|
||||||
|
GL_NONE = 0
|
||||||
|
GL_RESET_NOTIFICATION_STRATEGY_ARB = 0x8256
|
||||||
|
GL_VERSION = 0x1F02
|
||||||
|
)
|
||||||
|
|
||||||
|
w.context.source = ctxconfig.source
|
||||||
|
w.context.client = OpenGLAPI
|
||||||
|
|
||||||
|
p, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
previous := (*Window)(unsafe.Pointer(p))
|
||||||
|
defer func() {
|
||||||
|
err := previous.MakeContextCurrent()
|
||||||
|
if ferr == nil {
|
||||||
|
ferr = err
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
if err := w.MakeContextCurrent(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
getIntegerv := w.context.getProcAddress("glGetIntegerv")
|
||||||
|
getString := w.context.getProcAddress("glGetString")
|
||||||
|
if getIntegerv == 0 || getString == 0 {
|
||||||
|
return fmt.Errorf("glfwwin: entry point retrieval is broken: %w", PlatformError)
|
||||||
|
}
|
||||||
|
|
||||||
|
r, _, _ := syscall.Syscall(getString, 1, GL_VERSION, 0, 0)
|
||||||
|
version := windows.BytePtrToString((*byte)(unsafe.Pointer(r)))
|
||||||
|
if version == "" {
|
||||||
|
if ctxconfig.client == OpenGLAPI {
|
||||||
|
return fmt.Errorf("glfwwin: OpenGL version string retrieval is broken: %w", PlatformError)
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("glfwwin: OpenGL ES version string retrieval is broken: %w", PlatformError)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, prefix := range []string{
|
||||||
|
"OpenGL ES-CM ",
|
||||||
|
"OpenGL ES-CL ",
|
||||||
|
"OpenGL ES "} {
|
||||||
|
if strings.HasPrefix(version, prefix) {
|
||||||
|
version = version[len(prefix):]
|
||||||
|
w.context.client = OpenGLESAPI
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m := regexp.MustCompile(`^(\d+)(\.(\d+)(\.(\d+))?)?`).FindStringSubmatch(version)
|
||||||
|
if m == nil {
|
||||||
|
if w.context.client == OpenGLAPI {
|
||||||
|
return fmt.Errorf("glfwwin: no version found in OpenGL version string: %w", PlatformError)
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("glfwwin: no version found in OpenGL ES version string: %w", PlatformError)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
w.context.major, _ = strconv.Atoi(m[1])
|
||||||
|
w.context.minor, _ = strconv.Atoi(m[3])
|
||||||
|
w.context.revision, _ = strconv.Atoi(m[5])
|
||||||
|
|
||||||
|
if w.context.major < ctxconfig.major || (w.context.major == ctxconfig.major && w.context.minor < ctxconfig.minor) {
|
||||||
|
// The desired OpenGL version is greater than the actual version
|
||||||
|
// This only happens if the machine lacks {GLX|WGL}_ARB_create_context
|
||||||
|
// /and/ the user has requested an OpenGL version greater than 1.0
|
||||||
|
|
||||||
|
// For API consistency, we emulate the behavior of the
|
||||||
|
// {GLX|WGL}_ARB_create_context extension and fail here
|
||||||
|
|
||||||
|
if w.context.client == OpenGLAPI {
|
||||||
|
return fmt.Errorf("glfwwin: requested OpenGL version %d.%d, got version %d.%d: %w", ctxconfig.major, ctxconfig.minor, w.context.major, w.context.minor, VersionUnavailable)
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("glfwwin: requested OpenGL ES version %d.%d, got version %d.%d: %w", ctxconfig.major, ctxconfig.minor, w.context.major, w.context.minor, VersionUnavailable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.context.major >= 3 {
|
||||||
|
// OpenGL 3.0+ uses a different function for extension string retrieval
|
||||||
|
// We cache it here instead of in glfwExtensionSupported mostly to alert
|
||||||
|
// users as early as possible that their build may be broken
|
||||||
|
|
||||||
|
glGetStringi := w.context.getProcAddress("glGetStringi")
|
||||||
|
if glGetStringi == 0 {
|
||||||
|
return fmt.Errorf("glfwwin: entry point retrieval is broken: %w", PlatformError)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.context.client == OpenGLAPI {
|
||||||
|
// Read back context flags (OpenGL 3.0 and above)
|
||||||
|
if w.context.major >= 3 {
|
||||||
|
var flags int32
|
||||||
|
syscall.Syscall(getIntegerv, 2, GL_CONTEXT_FLAGS, uintptr(unsafe.Pointer(&flags)), 0)
|
||||||
|
|
||||||
|
if flags&GL_CONTEXT_FLAG_FORWARD_COMPATIBLE_BIT != 0 {
|
||||||
|
w.context.forward = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if flags&GL_CONTEXT_FLAG_DEBUG_BIT != 0 {
|
||||||
|
w.context.debug = true
|
||||||
|
} else {
|
||||||
|
ok, err := ExtensionSupported("GL_ARB_debug_output")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ok && ctxconfig.debug {
|
||||||
|
// HACK: This is a workaround for older drivers (pre KHR_debug)
|
||||||
|
// not setting the debug bit in the context flags for
|
||||||
|
// debug contexts
|
||||||
|
w.context.debug = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if flags&GL_CONTEXT_FLAG_NO_ERROR_BIT_KHR != 0 {
|
||||||
|
w.context.noerror = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read back OpenGL context profile (OpenGL 3.2 and above)
|
||||||
|
if w.context.major >= 4 || (w.context.major == 3 && w.context.minor >= 2) {
|
||||||
|
var mask int32
|
||||||
|
syscall.Syscall(getIntegerv, 2, GL_CONTEXT_PROFILE_MASK, uintptr(unsafe.Pointer(&mask)), 0)
|
||||||
|
|
||||||
|
if mask&GL_CONTEXT_COMPATIBILITY_PROFILE_BIT != 0 {
|
||||||
|
w.context.profile = OpenGLCompatProfile
|
||||||
|
} else if mask&GL_CONTEXT_CORE_PROFILE_BIT != 0 {
|
||||||
|
w.context.profile = OpenGLCoreProfile
|
||||||
|
} else {
|
||||||
|
ok, err := ExtensionSupported("GL_ARB_compatibility")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
// HACK: This is a workaround for the compatibility profile bit
|
||||||
|
// not being set in the context flags if an OpenGL 3.2+
|
||||||
|
// context was created without having requested a specific
|
||||||
|
// version
|
||||||
|
w.context.profile = OpenGLCompatProfile
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read back robustness strategy
|
||||||
|
ok, err := ExtensionSupported("GL_ARB_robustness")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
// NOTE: We avoid using the context flags for detection, as they are
|
||||||
|
// only present from 3.0 while the extension applies from 1.1
|
||||||
|
|
||||||
|
var strategy int32
|
||||||
|
syscall.Syscall(getIntegerv, 2, GL_RESET_NOTIFICATION_STRATEGY_ARB, uintptr(unsafe.Pointer(&strategy)), 0)
|
||||||
|
|
||||||
|
if strategy == GL_LOSE_CONTEXT_ON_RESET_ARB {
|
||||||
|
w.context.robustness = LoseContextOnReset
|
||||||
|
} else if strategy == GL_NO_RESET_NOTIFICATION_ARB {
|
||||||
|
w.context.robustness = NoResetNotification
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Read back robustness strategy
|
||||||
|
ok, err := ExtensionSupported("GL_EXT_robustness")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
// NOTE: The values of these constants match those of the OpenGL ARB
|
||||||
|
// one, so we can reuse them here
|
||||||
|
|
||||||
|
var strategy int32
|
||||||
|
syscall.Syscall(getIntegerv, 2, GL_RESET_NOTIFICATION_STRATEGY_ARB, uintptr(unsafe.Pointer(&strategy)), 0)
|
||||||
|
|
||||||
|
if strategy == GL_LOSE_CONTEXT_ON_RESET_ARB {
|
||||||
|
w.context.robustness = LoseContextOnReset
|
||||||
|
} else if strategy == GL_NO_RESET_NOTIFICATION_ARB {
|
||||||
|
w.context.robustness = NoResetNotification
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ok, err := ExtensionSupported("GL_KHR_context_flush_control")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
var behavior int32
|
||||||
|
syscall.Syscall(getIntegerv, 2, GL_CONTEXT_RELEASE_BEHAVIOR, uintptr(unsafe.Pointer(&behavior)), 0)
|
||||||
|
|
||||||
|
if behavior == GL_NONE {
|
||||||
|
w.context.release = ReleaseBehaviorNone
|
||||||
|
} else if behavior == GL_CONTEXT_RELEASE_BEHAVIOR_FLUSH {
|
||||||
|
w.context.release = ReleaseBehaviorFlush
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clearing the front buffer to black to avoid garbage pixels left over from
|
||||||
|
// previous uses of our bit of VRAM
|
||||||
|
glClear := w.context.getProcAddress("glClear")
|
||||||
|
syscall.Syscall(glClear, 1, GL_COLOR_BUFFER_BIT, 0, 0)
|
||||||
|
|
||||||
|
if w.doublebuffer {
|
||||||
|
w.context.swapBuffers(w)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) MakeContextCurrent() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
previous := (*Window)(unsafe.Pointer(ptr))
|
||||||
|
|
||||||
|
if w != nil && w.context.client == NoAPI {
|
||||||
|
return fmt.Errorf("glfwwin: cannot make current with a window that has no OpenGL or OpenGL ES context: %w", NoWindowContext)
|
||||||
|
}
|
||||||
|
|
||||||
|
if previous != nil {
|
||||||
|
if w == nil || w.context.source != previous.context.source {
|
||||||
|
if err := previous.context.makeCurrent(nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if w != nil {
|
||||||
|
if err := w.context.makeCurrent(w); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetCurrentContext() (*Window, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
ptr, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return (*Window)(unsafe.Pointer(ptr)), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SwapBuffers() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.context.client == NoAPI {
|
||||||
|
return fmt.Errorf("glfwwin: cannot swap buffers of a window that has no OpenGL or OpenGL ES context: %w", NoWindowContext)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := w.context.swapBuffers(w); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func SwapInterval(interval int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
window := (*Window)(unsafe.Pointer(ptr))
|
||||||
|
if window == nil {
|
||||||
|
return fmt.Errorf("glfwwin: cannot set swap interval without a current OpenGL or OpenGL ES context %w", NoCurrentContext)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := window.context.swapInterval(interval); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func ExtensionSupported(extension string) (bool, error) {
|
||||||
|
const (
|
||||||
|
GL_EXTENSIONS = 0x1F03
|
||||||
|
GL_NUM_EXTENSIONS = 0x821D
|
||||||
|
)
|
||||||
|
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return false, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
window := (*Window)(unsafe.Pointer(ptr))
|
||||||
|
if window == nil {
|
||||||
|
return false, fmt.Errorf("glfwwin: cannot query extension without a current OpenGL or OpenGL ES context %w", NoCurrentContext)
|
||||||
|
}
|
||||||
|
|
||||||
|
if window.context.major >= 3 {
|
||||||
|
// Check if extension is in the modern OpenGL extensions string list
|
||||||
|
|
||||||
|
glGetIntegerv := window.context.getProcAddress("glGetIntegerv")
|
||||||
|
var count int32
|
||||||
|
syscall.Syscall(glGetIntegerv, 2, GL_NUM_EXTENSIONS, uintptr(unsafe.Pointer(&count)), 0)
|
||||||
|
|
||||||
|
glGetStringi := window.context.getProcAddress("glGetStringi")
|
||||||
|
for i := 0; i < int(count); i++ {
|
||||||
|
r, _, _ := syscall.Syscall(glGetStringi, 2, GL_EXTENSIONS, uintptr(i), 0)
|
||||||
|
if r == 0 {
|
||||||
|
return false, fmt.Errorf("glfwwin: extension string retrieval is broken: %w", PlatformError)
|
||||||
|
}
|
||||||
|
|
||||||
|
en := windows.BytePtrToString((*byte)(unsafe.Pointer(r)))
|
||||||
|
if en == extension {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Check if extension is in the old style OpenGL extensions string
|
||||||
|
|
||||||
|
glGetString := window.context.getProcAddress("glGetString")
|
||||||
|
r, _, _ := syscall.Syscall(glGetString, 1, GL_EXTENSIONS, 0, 0)
|
||||||
|
if r == 0 {
|
||||||
|
return false, fmt.Errorf("glfwwin: extension string retrieval is broken: %w", PlatformError)
|
||||||
|
}
|
||||||
|
|
||||||
|
extensions := windows.BytePtrToString((*byte)(unsafe.Pointer(r)))
|
||||||
|
for _, str := range strings.Split(extensions, " ") {
|
||||||
|
if str == extension {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if extension is in the platform-specific string
|
||||||
|
return window.context.extensionSupported(extension), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetProcAddress(procname string) (unsafe.Pointer, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
window := (*Window)(unsafe.Pointer(ptr))
|
||||||
|
if window == nil {
|
||||||
|
return nil, fmt.Errorf("glfwwin: cannot query entry point without a current OpenGL or OpenGL ES context %w", NoCurrentContext)
|
||||||
|
}
|
||||||
|
|
||||||
|
return unsafe.Pointer(window.context.getProcAddress(procname)), nil
|
||||||
|
}
|
358
internal/glfwwin/glfw3h_windows.go
Normal file
358
internal/glfwwin/glfw3h_windows.go
Normal file
@ -0,0 +1,358 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
const (
|
||||||
|
NoAPI = 0
|
||||||
|
OpenGLAPI = 0x00030001
|
||||||
|
OpenGLESAPI = 0x00030002
|
||||||
|
|
||||||
|
NoRobustness = 0
|
||||||
|
NoResetNotification = 0x00031001
|
||||||
|
LoseContextOnReset = 0x00031002
|
||||||
|
|
||||||
|
OpenGLAnyProfile = 0
|
||||||
|
OpenGLCoreProfile = 0x00032001
|
||||||
|
OpenGLCompatProfile = 0x00032002
|
||||||
|
|
||||||
|
CursorNormal = 0x00034001
|
||||||
|
CursorHidden = 0x00034002
|
||||||
|
CursorDisabled = 0x00034003
|
||||||
|
|
||||||
|
AnyReleaseBehavior = 0
|
||||||
|
ReleaseBehaviorFlush = 0x00035001
|
||||||
|
ReleaseBehaviorNone = 0x00035002
|
||||||
|
|
||||||
|
NativeContextAPI = 0x00036001
|
||||||
|
EGLContextAPI = 0x00036002
|
||||||
|
OSMesaContextAPI = 0x00036003
|
||||||
|
|
||||||
|
DontCare = -1
|
||||||
|
)
|
||||||
|
|
||||||
|
type Action int
|
||||||
|
|
||||||
|
const (
|
||||||
|
Release Action = 0
|
||||||
|
Press Action = 1
|
||||||
|
Repeat Action = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
type Hint int
|
||||||
|
|
||||||
|
const (
|
||||||
|
Focused Hint = 0x00020001
|
||||||
|
Iconified Hint = 0x00020002
|
||||||
|
Resizable Hint = 0x00020003
|
||||||
|
Visible Hint = 0x00020004
|
||||||
|
Decorated Hint = 0x00020005
|
||||||
|
AutoIconify Hint = 0x00020006
|
||||||
|
Floating Hint = 0x00020007
|
||||||
|
Maximized Hint = 0x00020008
|
||||||
|
CenterCursor Hint = 0x00020009
|
||||||
|
TransparentFramebuffer Hint = 0x0002000A
|
||||||
|
Hovered Hint = 0x0002000B
|
||||||
|
FocusOnShow Hint = 0x0002000C
|
||||||
|
|
||||||
|
RedBits Hint = 0x00021001
|
||||||
|
GreenBits Hint = 0x00021002
|
||||||
|
BlueBits Hint = 0x00021003
|
||||||
|
AlphaBits Hint = 0x00021004
|
||||||
|
DepthBits Hint = 0x00021005
|
||||||
|
StencilBits Hint = 0x00021006
|
||||||
|
AccumRedBits Hint = 0x00021007
|
||||||
|
AccumGreenBits Hint = 0x00021008
|
||||||
|
AccumBlueBits Hint = 0x00021009
|
||||||
|
AccumAlphaBits Hint = 0x0002100A
|
||||||
|
AuxBuffers Hint = 0x0002100B
|
||||||
|
Stereo Hint = 0x0002100C
|
||||||
|
Samples Hint = 0x0002100D
|
||||||
|
SRGBCapable Hint = 0x0002100E
|
||||||
|
RefreshRate Hint = 0x0002100F
|
||||||
|
DoubleBuffer Hint = 0x00021010
|
||||||
|
|
||||||
|
ClientAPI Hint = 0x00022001
|
||||||
|
ContextVersionMajor Hint = 0x00022002
|
||||||
|
ContextVersionMinor Hint = 0x00022003
|
||||||
|
ContextRevision Hint = 0x00022004
|
||||||
|
ContextRobustness Hint = 0x00022005
|
||||||
|
OpenGLForwardCompat Hint = 0x00022006
|
||||||
|
OpenGLDebugContext Hint = 0x00022007
|
||||||
|
OpenGLProfile Hint = 0x00022008
|
||||||
|
ContextReleaseBehavior Hint = 0x00022009
|
||||||
|
ContextNoError Hint = 0x0002200A
|
||||||
|
ContextCreationAPI Hint = 0x0002200B
|
||||||
|
ScaleToMonitor Hint = 0x0002200C
|
||||||
|
)
|
||||||
|
|
||||||
|
type InputMode int
|
||||||
|
|
||||||
|
const (
|
||||||
|
CursorMode InputMode = 0x00033001
|
||||||
|
StickyKeysMode InputMode = 0x00033002
|
||||||
|
StickyMouseButtonsMode InputMode = 0x00033003
|
||||||
|
LockKeyMods InputMode = 0x00033004
|
||||||
|
RawMouseMotion InputMode = 0x00033005
|
||||||
|
)
|
||||||
|
|
||||||
|
type Key int
|
||||||
|
|
||||||
|
const (
|
||||||
|
KeyUnknown Key = -1
|
||||||
|
|
||||||
|
// Printable keys
|
||||||
|
KeySpace Key = 32
|
||||||
|
KeyApostrophe Key = 39 // '
|
||||||
|
KeyComma Key = 44 // ,
|
||||||
|
KeyMinus Key = 45 // -
|
||||||
|
KeyPeriod Key = 46 // .
|
||||||
|
KeySlash Key = 47 // /
|
||||||
|
Key0 Key = 48
|
||||||
|
Key1 Key = 49
|
||||||
|
Key2 Key = 50
|
||||||
|
Key3 Key = 51
|
||||||
|
Key4 Key = 52
|
||||||
|
Key5 Key = 53
|
||||||
|
Key6 Key = 54
|
||||||
|
Key7 Key = 55
|
||||||
|
Key8 Key = 56
|
||||||
|
Key9 Key = 57
|
||||||
|
KeySemicolon Key = 59 // ;
|
||||||
|
KeyEqual Key = 61 // =
|
||||||
|
KeyA Key = 65
|
||||||
|
KeyB Key = 66
|
||||||
|
KeyC Key = 67
|
||||||
|
KeyD Key = 68
|
||||||
|
KeyE Key = 69
|
||||||
|
KeyF Key = 70
|
||||||
|
KeyG Key = 71
|
||||||
|
KeyH Key = 72
|
||||||
|
KeyI Key = 73
|
||||||
|
KeyJ Key = 74
|
||||||
|
KeyK Key = 75
|
||||||
|
KeyL Key = 76
|
||||||
|
KeyM Key = 77
|
||||||
|
KeyN Key = 78
|
||||||
|
KeyO Key = 79
|
||||||
|
KeyP Key = 80
|
||||||
|
KeyQ Key = 81
|
||||||
|
KeyR Key = 82
|
||||||
|
KeyS Key = 83
|
||||||
|
KeyT Key = 84
|
||||||
|
KeyU Key = 85
|
||||||
|
KeyV Key = 86
|
||||||
|
KeyW Key = 87
|
||||||
|
KeyX Key = 88
|
||||||
|
KeyY Key = 89
|
||||||
|
KeyZ Key = 90
|
||||||
|
KeyLeftBracket Key = 91 // [
|
||||||
|
KeyBackslash Key = 92 // \
|
||||||
|
KeyRightBracket Key = 93 // ]
|
||||||
|
KeyGraveAccent Key = 96 // `
|
||||||
|
KeyWorld1 Key = 161 // non-US #1
|
||||||
|
KeyWorld2 Key = 162 // non-US #2
|
||||||
|
|
||||||
|
// Function keys
|
||||||
|
KeyEscape Key = 256
|
||||||
|
KeyEnter Key = 257
|
||||||
|
KeyTab Key = 258
|
||||||
|
KeyBackspace Key = 259
|
||||||
|
KeyInsert Key = 260
|
||||||
|
KeyDelete Key = 261
|
||||||
|
KeyRight Key = 262
|
||||||
|
KeyLeft Key = 263
|
||||||
|
KeyDown Key = 264
|
||||||
|
KeyUp Key = 265
|
||||||
|
KeyPageUp Key = 266
|
||||||
|
KeyPageDown Key = 267
|
||||||
|
KeyHome Key = 268
|
||||||
|
KeyEnd Key = 269
|
||||||
|
KeyCapsLock Key = 280
|
||||||
|
KeyScrollLock Key = 281
|
||||||
|
KeyNumLock Key = 282
|
||||||
|
KeyPrintScreen Key = 283
|
||||||
|
KeyPause Key = 284
|
||||||
|
KeyF1 Key = 290
|
||||||
|
KeyF2 Key = 291
|
||||||
|
KeyF3 Key = 292
|
||||||
|
KeyF4 Key = 293
|
||||||
|
KeyF5 Key = 294
|
||||||
|
KeyF6 Key = 295
|
||||||
|
KeyF7 Key = 296
|
||||||
|
KeyF8 Key = 297
|
||||||
|
KeyF9 Key = 298
|
||||||
|
KeyF10 Key = 299
|
||||||
|
KeyF11 Key = 300
|
||||||
|
KeyF12 Key = 301
|
||||||
|
KeyF13 Key = 302
|
||||||
|
KeyF14 Key = 303
|
||||||
|
KeyF15 Key = 304
|
||||||
|
KeyF16 Key = 305
|
||||||
|
KeyF17 Key = 306
|
||||||
|
KeyF18 Key = 307
|
||||||
|
KeyF19 Key = 308
|
||||||
|
KeyF20 Key = 309
|
||||||
|
KeyF21 Key = 310
|
||||||
|
KeyF22 Key = 311
|
||||||
|
KeyF23 Key = 312
|
||||||
|
KeyF24 Key = 313
|
||||||
|
KeyF25 Key = 314
|
||||||
|
KeyKP0 Key = 320
|
||||||
|
KeyKP1 Key = 321
|
||||||
|
KeyKP2 Key = 322
|
||||||
|
KeyKP3 Key = 323
|
||||||
|
KeyKP4 Key = 324
|
||||||
|
KeyKP5 Key = 325
|
||||||
|
KeyKP6 Key = 326
|
||||||
|
KeyKP7 Key = 327
|
||||||
|
KeyKP8 Key = 328
|
||||||
|
KeyKP9 Key = 329
|
||||||
|
KeyKPDecimal Key = 330
|
||||||
|
KeyKPDivide Key = 331
|
||||||
|
KeyKPMultiply Key = 332
|
||||||
|
KeyKPSubtract Key = 333
|
||||||
|
KeyKPAdd Key = 334
|
||||||
|
KeyKPEnter Key = 335
|
||||||
|
KeyKPEqual Key = 336
|
||||||
|
KeyLeftShift Key = 340
|
||||||
|
KeyLeftControl Key = 341
|
||||||
|
KeyLeftAlt Key = 342
|
||||||
|
KeyLeftSuper Key = 343
|
||||||
|
KeyRightShift Key = 344
|
||||||
|
KeyRightControl Key = 345
|
||||||
|
KeyRightAlt Key = 346
|
||||||
|
KeyRightSuper Key = 347
|
||||||
|
KeyMenu Key = 348
|
||||||
|
|
||||||
|
KeyLast Key = KeyMenu
|
||||||
|
)
|
||||||
|
|
||||||
|
type ModifierKey int
|
||||||
|
|
||||||
|
const (
|
||||||
|
ModShift ModifierKey = 0x0001
|
||||||
|
ModControl ModifierKey = 0x0002
|
||||||
|
ModAlt ModifierKey = 0x0004
|
||||||
|
ModSuper ModifierKey = 0x0008
|
||||||
|
ModCapsLock ModifierKey = 0x0010
|
||||||
|
ModNumLock ModifierKey = 0x0020
|
||||||
|
)
|
||||||
|
|
||||||
|
type MouseButton int
|
||||||
|
|
||||||
|
const (
|
||||||
|
MouseButton1 MouseButton = 0
|
||||||
|
MouseButton2 MouseButton = 1
|
||||||
|
MouseButton3 MouseButton = 2
|
||||||
|
MouseButton4 MouseButton = 3
|
||||||
|
MouseButton5 MouseButton = 4
|
||||||
|
MouseButton6 MouseButton = 5
|
||||||
|
MouseButton7 MouseButton = 6
|
||||||
|
MouseButton8 MouseButton = 7
|
||||||
|
MouseButtonLast MouseButton = MouseButton8
|
||||||
|
MouseButtonLeft MouseButton = MouseButton1
|
||||||
|
MouseButtonRight MouseButton = MouseButton2
|
||||||
|
MouseButtonMiddle MouseButton = MouseButton3
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
NotInitialized Error = 0x00010001
|
||||||
|
NoCurrentContext Error = 0x00010002
|
||||||
|
InvalidEnum Error = 0x00010003
|
||||||
|
InvalidValue Error = 0x00010004
|
||||||
|
OutOfMemory Error = 0x00010005
|
||||||
|
ApiUnavailable Error = 0x00010006
|
||||||
|
VersionUnavailable Error = 0x00010007
|
||||||
|
PlatformError Error = 0x00010008
|
||||||
|
FormatUnavailable Error = 0x00010009
|
||||||
|
NoWindowContext Error = 0x0001000A
|
||||||
|
)
|
||||||
|
|
||||||
|
type PeripheralEvent int
|
||||||
|
|
||||||
|
const (
|
||||||
|
Connected PeripheralEvent = 0x00040001
|
||||||
|
Disconnected PeripheralEvent = 0x00040002
|
||||||
|
)
|
||||||
|
|
||||||
|
type StandardCursor int
|
||||||
|
|
||||||
|
const (
|
||||||
|
ArrowCursor StandardCursor = 0x00036001
|
||||||
|
IBeamCursor StandardCursor = 0x00036002
|
||||||
|
CrosshairCursor StandardCursor = 0x00036003
|
||||||
|
HandCursor StandardCursor = 0x00036004
|
||||||
|
HResizeCursor StandardCursor = 0x00036005
|
||||||
|
VResizeCursor StandardCursor = 0x00036006
|
||||||
|
)
|
||||||
|
|
||||||
|
type Error int
|
||||||
|
|
||||||
|
func (e Error) Error() string {
|
||||||
|
switch e {
|
||||||
|
case NotInitialized:
|
||||||
|
return "the GLFW library is not initialized"
|
||||||
|
case NoCurrentContext:
|
||||||
|
return "there is no current context"
|
||||||
|
case InvalidEnum:
|
||||||
|
return "invalid argument for enum parameter"
|
||||||
|
case InvalidValue:
|
||||||
|
return "invalid value for parameter"
|
||||||
|
case OutOfMemory:
|
||||||
|
return "out of memory"
|
||||||
|
case ApiUnavailable:
|
||||||
|
return "the requested API is unavailable"
|
||||||
|
case VersionUnavailable:
|
||||||
|
return "the requested API version is unavailable"
|
||||||
|
case PlatformError:
|
||||||
|
return "a platform-specific error occurred"
|
||||||
|
case FormatUnavailable:
|
||||||
|
return "the requested format is unavailable"
|
||||||
|
case NoWindowContext:
|
||||||
|
return "the specified window has no context"
|
||||||
|
default:
|
||||||
|
return "ERROR: UNKNOWN GLFW ERROR"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type VidMode struct {
|
||||||
|
Width int
|
||||||
|
Height int
|
||||||
|
RedBits int
|
||||||
|
GreenBits int
|
||||||
|
BlueBits int
|
||||||
|
RefreshRate int
|
||||||
|
}
|
||||||
|
|
||||||
|
type GammaRamp struct {
|
||||||
|
Red []int
|
||||||
|
Green []int
|
||||||
|
Blue []int
|
||||||
|
}
|
||||||
|
|
||||||
|
type Image struct {
|
||||||
|
Width int
|
||||||
|
Height int
|
||||||
|
Pixels []byte
|
||||||
|
}
|
97
internal/glfwwin/init_windows.go
Normal file
97
internal/glfwwin/init_windows.go
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
func terminate() error {
|
||||||
|
for _, w := range _glfw.windows {
|
||||||
|
if err := w.Destroy(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range _glfw.cursors {
|
||||||
|
if err := c.Destroy(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, monitor := range _glfw.monitors {
|
||||||
|
if len(monitor.originalRamp.Red) != 0 {
|
||||||
|
monitor.platformSetGammaRamp(&monitor.originalRamp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_glfw.monitors = nil
|
||||||
|
|
||||||
|
if err := platformTerminate(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_glfw.initialized = false
|
||||||
|
|
||||||
|
if err := _glfw.contextSlot.destroy(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func Init() (ferr error) {
|
||||||
|
defer func() {
|
||||||
|
if ferr != nil {
|
||||||
|
terminate()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
if _glfw.initialized {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
_glfw.hints.init.hatButtons = true
|
||||||
|
|
||||||
|
if err := platformInit(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := _glfw.contextSlot.create(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_glfw.initialized = true
|
||||||
|
|
||||||
|
if err := defaultWindowHints(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func Terminate() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if err := terminate(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
532
internal/glfwwin/input_windows.go
Normal file
532
internal/glfwwin/input_windows.go
Normal file
@ -0,0 +1,532 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
)
|
||||||
|
|
||||||
|
const stick = 3
|
||||||
|
|
||||||
|
func (w *Window) inputKey(key Key, scancode int, action Action, mods ModifierKey) {
|
||||||
|
if key >= 0 && key <= KeyLast {
|
||||||
|
var repeated bool
|
||||||
|
|
||||||
|
if action == Release && w.keys[key] == Release {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if action == Press && w.keys[key] == Press {
|
||||||
|
repeated = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if action == Release && w.stickyKeys {
|
||||||
|
w.keys[key] = stick
|
||||||
|
} else {
|
||||||
|
w.keys[key] = action
|
||||||
|
}
|
||||||
|
|
||||||
|
if repeated {
|
||||||
|
action = Repeat
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !w.lockKeyMods {
|
||||||
|
mods &^= ModCapsLock | ModNumLock
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.callbacks.key != nil {
|
||||||
|
w.callbacks.key(w, key, scancode, action, mods)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputChar(codepoint rune, mods ModifierKey, plain bool) {
|
||||||
|
if codepoint < 32 || (codepoint > 126 && codepoint < 160) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !w.lockKeyMods {
|
||||||
|
mods &^= ModCapsLock | ModNumLock
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.callbacks.charmods != nil {
|
||||||
|
w.callbacks.charmods(w, codepoint, mods)
|
||||||
|
}
|
||||||
|
|
||||||
|
if plain {
|
||||||
|
if w.callbacks.character != nil {
|
||||||
|
w.callbacks.character(w, codepoint)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputScroll(xoffset, yoffset float64) {
|
||||||
|
if w.callbacks.scroll != nil {
|
||||||
|
w.callbacks.scroll(w, xoffset, yoffset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputMouseClick(button MouseButton, action Action, mods ModifierKey) {
|
||||||
|
if button < 0 || button > MouseButtonLast {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !w.lockKeyMods {
|
||||||
|
mods &^= ModCapsLock | ModNumLock
|
||||||
|
}
|
||||||
|
|
||||||
|
if action == Release && w.stickyMouseButtons {
|
||||||
|
w.mouseButtons[button] = stick
|
||||||
|
} else {
|
||||||
|
w.mouseButtons[button] = action
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.callbacks.mouseButton != nil {
|
||||||
|
w.callbacks.mouseButton(w, button, action, mods)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputCursorPos(xpos float64, ypos float64) {
|
||||||
|
if w.virtualCursorPosX == xpos && w.virtualCursorPosY == ypos {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.virtualCursorPosX = xpos
|
||||||
|
w.virtualCursorPosY = ypos
|
||||||
|
|
||||||
|
if w.callbacks.cursorPos != nil {
|
||||||
|
w.callbacks.cursorPos(w, xpos, ypos)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputCursorEnter(entered bool) {
|
||||||
|
if w.callbacks.cursorEnter != nil {
|
||||||
|
w.callbacks.cursorEnter(w, entered)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputDrop(paths []string) {
|
||||||
|
if w.callbacks.drop != nil {
|
||||||
|
w.callbacks.drop(w, paths)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) centerCursorInContentArea() error {
|
||||||
|
width, height, err := w.platformGetWindowSize()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := w.platformSetCursorPos(float64(width/2), float64(height/2)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetInputMode(mode InputMode) (int, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, NotInitialized
|
||||||
|
}
|
||||||
|
switch mode {
|
||||||
|
case CursorMode:
|
||||||
|
return w.cursorMode, nil
|
||||||
|
case StickyKeysMode:
|
||||||
|
return boolToInt(w.stickyKeys), nil
|
||||||
|
case StickyMouseButtonsMode:
|
||||||
|
return boolToInt(w.stickyMouseButtons), nil
|
||||||
|
case LockKeyMods:
|
||||||
|
return boolToInt(w.lockKeyMods), nil
|
||||||
|
case RawMouseMotion:
|
||||||
|
return boolToInt(w.rawMouseMotion), nil
|
||||||
|
default:
|
||||||
|
return 0, fmt.Errorf("glfwwin: invalid input mode 0x%08X: %w", mode, InvalidEnum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetInputMode(mode InputMode, value int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
switch mode {
|
||||||
|
case CursorMode:
|
||||||
|
if value != CursorNormal && value != CursorHidden && value != CursorDisabled {
|
||||||
|
return fmt.Errorf("glfwwin: invalid cursor mode 0x%08X: %w", value, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.cursorMode == value {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
w.cursorMode = value
|
||||||
|
|
||||||
|
x, y, err := w.platformGetCursorPos()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
w.virtualCursorPosX = x
|
||||||
|
w.virtualCursorPosY = y
|
||||||
|
|
||||||
|
if err := w.platformSetCursorMode(value); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
|
||||||
|
case StickyKeysMode:
|
||||||
|
if w.stickyKeys == intToBool(value) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !intToBool(value) {
|
||||||
|
// Release all sticky keys
|
||||||
|
for i := Key(0); i <= KeyLast; i++ {
|
||||||
|
if w.keys[i] == stick {
|
||||||
|
w.keys[i] = Release
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w.stickyKeys = intToBool(value)
|
||||||
|
return nil
|
||||||
|
|
||||||
|
case StickyMouseButtonsMode:
|
||||||
|
if w.stickyMouseButtons == intToBool(value) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !intToBool(value) {
|
||||||
|
// Release all sticky mouse buttons
|
||||||
|
for i := MouseButton(0); i <= MouseButtonLast; i++ {
|
||||||
|
if w.mouseButtons[i] == stick {
|
||||||
|
w.mouseButtons[i] = Release
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w.stickyMouseButtons = intToBool(value)
|
||||||
|
return nil
|
||||||
|
|
||||||
|
case LockKeyMods:
|
||||||
|
w.lockKeyMods = intToBool(value)
|
||||||
|
return nil
|
||||||
|
|
||||||
|
case RawMouseMotion:
|
||||||
|
if !platformRawMouseMotionSupported() {
|
||||||
|
return fmt.Errorf("glfwwin: raw mouse motion is not supported on this system: %w", PlatformError)
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.rawMouseMotion == intToBool(value) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
w.rawMouseMotion = intToBool(value)
|
||||||
|
if err := w.platformSetRawMouseMotion(intToBool(value)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("glfwwin: invalid input mode 0x%08X: %w", mode, InvalidEnum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func RawMouseMotionSupported() (bool, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return false, NotInitialized
|
||||||
|
}
|
||||||
|
return platformRawMouseMotionSupported(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetKeyName(key Key, scancode int) (string, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return "", NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if key != KeyUnknown {
|
||||||
|
if key != KeyKPEqual && (key < KeyKP0 || key > KeyKPAdd) && (key < KeyApostrophe || key > KeyWorld2) {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
scancode = platformGetKeyScancode(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
return platformGetScancodeName(scancode)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetKeyScancode(key Key) (int, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if key < KeySpace || key > KeyLast {
|
||||||
|
return 0, fmt.Errorf("glfwwin: invalid key %d: %w", key, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
return platformGetKeyScancode(key), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetKey(key Key) (Action, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if key < KeySpace || key > KeyLast {
|
||||||
|
return 0, fmt.Errorf("glfwwin: invalid key %d: %w", key, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.keys[key] == stick {
|
||||||
|
// Sticky mode: release key now
|
||||||
|
w.keys[key] = Release
|
||||||
|
return Press, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return w.keys[key], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetMouseButton(button MouseButton) (Action, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if button < MouseButton1 || button > MouseButtonLast {
|
||||||
|
return 0, fmt.Errorf("glfwwin: invalid mouse button %d: %w", button, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.mouseButtons[button] == stick {
|
||||||
|
// Sticky mode: release mouse button now
|
||||||
|
w.mouseButtons[button] = Release
|
||||||
|
return Press, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return w.mouseButtons[button], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetCursorPos() (xpos, ypos float64, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.cursorMode == CursorDisabled {
|
||||||
|
return w.virtualCursorPosX, w.virtualCursorPosY, nil
|
||||||
|
} else {
|
||||||
|
return w.platformGetCursorPos()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCursorPos(xpos, ypos float64) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if xpos != xpos || xpos < -math.MaxFloat64 || xpos > math.MaxFloat64 || ypos != ypos || ypos < -math.MaxFloat64 || ypos > math.MaxFloat64 {
|
||||||
|
return fmt.Errorf("glfwwin: invalid cursor position %f %f: %w", xpos, ypos, InvalidValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !w.platformWindowFocused() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.cursorMode == CursorDisabled {
|
||||||
|
// Only update the accumulated position if the cursor is disabled
|
||||||
|
w.virtualCursorPosX = xpos
|
||||||
|
w.virtualCursorPosY = ypos
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
// Update system cursor position
|
||||||
|
return w.platformSetCursorPos(xpos, ypos)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateCursor(image *Image, xhot, yhot int) (*Cursor, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor := &Cursor{}
|
||||||
|
_glfw.cursors = append(_glfw.cursors, cursor)
|
||||||
|
|
||||||
|
if err := cursor.platformCreateCursor(image, xhot, yhot); err != nil {
|
||||||
|
cursor.Destroy()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return cursor, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateStandardCursor(shape StandardCursor) (*Cursor, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if shape != ArrowCursor &&
|
||||||
|
shape != IBeamCursor &&
|
||||||
|
shape != CrosshairCursor &&
|
||||||
|
shape != HandCursor &&
|
||||||
|
shape != HResizeCursor &&
|
||||||
|
shape != VResizeCursor {
|
||||||
|
return nil, fmt.Errorf("glfwwin: invalid standard cursor 0x%08X: %w", shape, InvalidEnum)
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor := &Cursor{}
|
||||||
|
_glfw.cursors = append(_glfw.cursors, cursor)
|
||||||
|
|
||||||
|
if err := cursor.platformCreateStandardCursor(shape); err != nil {
|
||||||
|
cursor.Destroy()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return cursor, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Cursor) Destroy() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if c == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure the cursor is not being used by any window
|
||||||
|
for _, window := range _glfw.windows {
|
||||||
|
if window.cursor == c {
|
||||||
|
if err := window.SetCursor(nil); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := c.platformDestroyCursor(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unlink cursor from global linked list
|
||||||
|
for i, cursor := range _glfw.cursors {
|
||||||
|
if cursor == c {
|
||||||
|
copy(_glfw.cursors[i:], _glfw.cursors[i+1:])
|
||||||
|
_glfw.cursors = _glfw.cursors[:len(_glfw.cursors)-1]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCursor(cursor *Cursor) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
w.cursor = cursor
|
||||||
|
|
||||||
|
if err := w.platformSetCursor(cursor); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetKeyCallback(cbfun KeyCallback) (KeyCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.key
|
||||||
|
w.callbacks.key = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCharCallback(cbfun CharCallback) (CharCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.character
|
||||||
|
w.callbacks.character = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCharModsCallback(cbfun CharModsCallback) (CharModsCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.charmods
|
||||||
|
w.callbacks.charmods = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetMouseButtonCallback(cbfun MouseButtonCallback) (MouseButtonCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.mouseButton
|
||||||
|
w.callbacks.mouseButton = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCursorPosCallback(cbfun CursorPosCallback) (CursorPosCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.cursorPos
|
||||||
|
w.callbacks.cursorPos = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCursorEnterCallback(cbfun CursorEnterCallback) (CursorEnterCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.cursorEnter
|
||||||
|
w.callbacks.cursorEnter = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetScrollCallback(cbfun ScrollCallback) (ScrollCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.scroll
|
||||||
|
w.callbacks.scroll = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetDropCallback(cbfun DropCallback) (DropCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.drop
|
||||||
|
w.callbacks.drop = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetClipboardString(str string) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
return platformSetClipboardString(str)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetClipboardString() (string, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return "", NotInitialized
|
||||||
|
}
|
||||||
|
return platformGetClipboardString()
|
||||||
|
}
|
328
internal/glfwwin/internal_windows.go
Normal file
328
internal/glfwwin/internal_windows.go
Normal file
@ -0,0 +1,328 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"golang.org/x/sys/windows"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
_GLFW_INSERT_FIRST = 0
|
||||||
|
_GLFW_INSERT_LAST = 1
|
||||||
|
)
|
||||||
|
|
||||||
|
var _glfw library
|
||||||
|
|
||||||
|
type initconfig struct {
|
||||||
|
hatButtons bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type wndconfig struct {
|
||||||
|
width int
|
||||||
|
height int
|
||||||
|
title string
|
||||||
|
resizable bool
|
||||||
|
visible bool
|
||||||
|
decorated bool
|
||||||
|
focused bool
|
||||||
|
autoIconify bool
|
||||||
|
floating bool
|
||||||
|
maximized bool
|
||||||
|
centerCursor bool
|
||||||
|
focusOnShow bool
|
||||||
|
scaleToMonitor bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type ctxconfig struct {
|
||||||
|
client int
|
||||||
|
source int
|
||||||
|
major int
|
||||||
|
minor int
|
||||||
|
forward bool
|
||||||
|
debug bool
|
||||||
|
noerror bool
|
||||||
|
profile int
|
||||||
|
robustness int
|
||||||
|
release int
|
||||||
|
share *Window
|
||||||
|
}
|
||||||
|
|
||||||
|
type fbconfig struct {
|
||||||
|
redBits int
|
||||||
|
greenBits int
|
||||||
|
blueBits int
|
||||||
|
alphaBits int
|
||||||
|
depthBits int
|
||||||
|
stencilBits int
|
||||||
|
accumRedBits int
|
||||||
|
accumGreenBits int
|
||||||
|
accumBlueBits int
|
||||||
|
accumAlphaBits int
|
||||||
|
auxBuffers int
|
||||||
|
stereo bool
|
||||||
|
samples int
|
||||||
|
sRGB bool
|
||||||
|
doublebuffer bool
|
||||||
|
transparent bool
|
||||||
|
handle uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
type context struct {
|
||||||
|
client int
|
||||||
|
source int
|
||||||
|
major int
|
||||||
|
minor int
|
||||||
|
revision int
|
||||||
|
forward bool
|
||||||
|
debug bool
|
||||||
|
noerror bool
|
||||||
|
profile int
|
||||||
|
robustness int
|
||||||
|
release int
|
||||||
|
|
||||||
|
// TODO: Put these functions in an interface type.
|
||||||
|
makeCurrent func(*Window) error
|
||||||
|
swapBuffers func(*Window) error
|
||||||
|
swapInterval func(int) error
|
||||||
|
extensionSupported func(string) bool
|
||||||
|
getProcAddress func(string) uintptr
|
||||||
|
destroy func(*Window) error
|
||||||
|
|
||||||
|
wgl struct {
|
||||||
|
dc _HDC
|
||||||
|
handle _HGLRC
|
||||||
|
interval int
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type (
|
||||||
|
PosCallback func(w *Window, xpos int, ypos int)
|
||||||
|
SizeCallback func(w *Window, width int, height int)
|
||||||
|
CloseCallback func(w *Window)
|
||||||
|
RefreshCallback func(w *Window)
|
||||||
|
FocusCallback func(w *Window, focused bool)
|
||||||
|
IconifyCallback func(w *Window, iconified bool)
|
||||||
|
MaximizeCallback func(w *Window, iconified bool)
|
||||||
|
FramebufferSizeCallback func(w *Window, width int, height int)
|
||||||
|
ContentScaleCallback func(w *Window, x float32, y float32)
|
||||||
|
MouseButtonCallback func(w *Window, button MouseButton, action Action, mods ModifierKey)
|
||||||
|
CursorPosCallback func(w *Window, xpos float64, ypos float64)
|
||||||
|
CursorEnterCallback func(w *Window, entered bool)
|
||||||
|
ScrollCallback func(w *Window, xoff float64, yoff float64)
|
||||||
|
KeyCallback func(w *Window, key Key, scancode int, action Action, mods ModifierKey)
|
||||||
|
CharCallback func(w *Window, char rune)
|
||||||
|
CharModsCallback func(w *Window, char rune, mods ModifierKey)
|
||||||
|
DropCallback func(w *Window, names []string)
|
||||||
|
MonitorCallback func(monitor *Monitor, event PeripheralEvent)
|
||||||
|
)
|
||||||
|
|
||||||
|
type Window struct {
|
||||||
|
resizable bool
|
||||||
|
decorated bool
|
||||||
|
autoIconify bool
|
||||||
|
floating bool
|
||||||
|
focusOnShow bool
|
||||||
|
shouldClose bool
|
||||||
|
userPointer unsafe.Pointer
|
||||||
|
doublebuffer bool
|
||||||
|
videoMode VidMode
|
||||||
|
monitor *Monitor
|
||||||
|
cursor *Cursor
|
||||||
|
|
||||||
|
minwidth int
|
||||||
|
minheight int
|
||||||
|
maxwidth int
|
||||||
|
maxheight int
|
||||||
|
numer int
|
||||||
|
denom int
|
||||||
|
|
||||||
|
stickyKeys bool
|
||||||
|
stickyMouseButtons bool
|
||||||
|
lockKeyMods bool
|
||||||
|
cursorMode int
|
||||||
|
mouseButtons [MouseButtonLast + 1]Action
|
||||||
|
keys [KeyLast + 1]Action
|
||||||
|
// Virtual cursor position when cursor is disabled
|
||||||
|
virtualCursorPosX float64
|
||||||
|
virtualCursorPosY float64
|
||||||
|
rawMouseMotion bool
|
||||||
|
|
||||||
|
context context
|
||||||
|
|
||||||
|
callbacks struct {
|
||||||
|
pos PosCallback
|
||||||
|
size SizeCallback
|
||||||
|
close CloseCallback
|
||||||
|
refresh RefreshCallback
|
||||||
|
focus FocusCallback
|
||||||
|
iconify IconifyCallback
|
||||||
|
maximize MaximizeCallback
|
||||||
|
fbsize FramebufferSizeCallback
|
||||||
|
scale ContentScaleCallback
|
||||||
|
mouseButton MouseButtonCallback
|
||||||
|
cursorPos CursorPosCallback
|
||||||
|
cursorEnter CursorEnterCallback
|
||||||
|
scroll ScrollCallback
|
||||||
|
key KeyCallback
|
||||||
|
character CharCallback
|
||||||
|
charmods CharModsCallback
|
||||||
|
drop DropCallback
|
||||||
|
}
|
||||||
|
|
||||||
|
win32 struct {
|
||||||
|
handle windows.HWND
|
||||||
|
bigIcon _HICON
|
||||||
|
smallIcon _HICON
|
||||||
|
|
||||||
|
cursorTracked bool
|
||||||
|
frameAction bool
|
||||||
|
iconified bool
|
||||||
|
maximized bool
|
||||||
|
transparent bool // Whether to enable framebuffer transparency on DWM
|
||||||
|
scaleToMonitor bool
|
||||||
|
|
||||||
|
// Cached size used to filter out duplicate events
|
||||||
|
width int
|
||||||
|
height int
|
||||||
|
|
||||||
|
// The last received cursor position, regardless of source
|
||||||
|
lastCursorPosX int
|
||||||
|
lastCursorPosY int
|
||||||
|
|
||||||
|
// The last recevied high surrogate when decoding pairs of UTF-16 messages
|
||||||
|
highSurrogate uint16
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type Monitor struct {
|
||||||
|
name string
|
||||||
|
|
||||||
|
widthMM int
|
||||||
|
heightMM int
|
||||||
|
|
||||||
|
window *Window
|
||||||
|
|
||||||
|
modes []*VidMode
|
||||||
|
currentMode *VidMode
|
||||||
|
|
||||||
|
originalRamp GammaRamp
|
||||||
|
currentRamp GammaRamp
|
||||||
|
|
||||||
|
win32 struct {
|
||||||
|
handle _HMONITOR
|
||||||
|
|
||||||
|
// This size matches the static size of DISPLAY_DEVICE.DeviceName
|
||||||
|
adapterName string
|
||||||
|
displayName string
|
||||||
|
modesPruned bool
|
||||||
|
modeChanged bool
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type Cursor struct {
|
||||||
|
win32 struct {
|
||||||
|
handle _HCURSOR
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type tls struct {
|
||||||
|
win32 struct {
|
||||||
|
allocated bool
|
||||||
|
index uint32
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type library struct {
|
||||||
|
initialized bool
|
||||||
|
|
||||||
|
hints struct {
|
||||||
|
init initconfig
|
||||||
|
framebuffer fbconfig
|
||||||
|
window wndconfig
|
||||||
|
context ctxconfig
|
||||||
|
refreshRate int
|
||||||
|
}
|
||||||
|
|
||||||
|
errors []error // TODO: Check the error at polling?
|
||||||
|
cursors []*Cursor
|
||||||
|
windows []*Window
|
||||||
|
|
||||||
|
monitors []*Monitor
|
||||||
|
|
||||||
|
contextSlot tls
|
||||||
|
|
||||||
|
callbacks struct {
|
||||||
|
monitor MonitorCallback
|
||||||
|
}
|
||||||
|
|
||||||
|
win32 struct {
|
||||||
|
helperWindowHandle windows.HWND
|
||||||
|
deviceNotificationHandle _HDEVNOTIFY
|
||||||
|
foregroundLockTimeout uint32
|
||||||
|
acquiredMonitorCount int
|
||||||
|
clipboardString string
|
||||||
|
keycodes [512]Key
|
||||||
|
scancodes [KeyLast + 1]int
|
||||||
|
keynames [KeyLast + 1]string
|
||||||
|
|
||||||
|
// Where to place the cursor when re-enabled
|
||||||
|
restoreCursorPosX float64
|
||||||
|
restoreCursorPosY float64
|
||||||
|
|
||||||
|
// The window whose disabled cursor mode is active
|
||||||
|
disabledCursorWindow *Window
|
||||||
|
rawInput []byte
|
||||||
|
mouseTrailSize uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
wgl struct {
|
||||||
|
inited bool
|
||||||
|
|
||||||
|
EXT_swap_control bool
|
||||||
|
EXT_colorspace bool
|
||||||
|
ARB_multisample bool
|
||||||
|
ARB_framebuffer_sRGB bool
|
||||||
|
EXT_framebuffer_sRGB bool
|
||||||
|
ARB_pixel_format bool
|
||||||
|
ARB_create_context bool
|
||||||
|
ARB_create_context_profile bool
|
||||||
|
EXT_create_context_es2_profile bool
|
||||||
|
ARB_create_context_robustness bool
|
||||||
|
ARB_create_context_no_error bool
|
||||||
|
ARB_context_flush_control bool
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func boolToInt(x bool) int {
|
||||||
|
if x {
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func intToBool(x int) bool {
|
||||||
|
return x != 0
|
||||||
|
}
|
313
internal/glfwwin/monitor_windows.go
Normal file
313
internal/glfwwin/monitor_windows.go
Normal file
@ -0,0 +1,313 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func abs(x int) uint {
|
||||||
|
if x < 0 {
|
||||||
|
return uint(-x)
|
||||||
|
}
|
||||||
|
return uint(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *VidMode) equals(other *VidMode) bool {
|
||||||
|
if v.RedBits+v.GreenBits+v.BlueBits != other.RedBits+other.GreenBits+other.BlueBits {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.Width != other.Width {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.Height != other.Height {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.RefreshRate != other.RefreshRate {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) refreshVideoModes() error {
|
||||||
|
m.modes = m.modes[:0]
|
||||||
|
modes, err := m.platformAppendVideoModes(m.modes)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
sort.Slice(modes, func(i, j int) bool {
|
||||||
|
a := modes[i]
|
||||||
|
b := modes[j]
|
||||||
|
abpp := a.RedBits + a.GreenBits + a.BlueBits
|
||||||
|
bbpp := b.RedBits + b.GreenBits + b.BlueBits
|
||||||
|
if abpp != bbpp {
|
||||||
|
return abpp < bbpp
|
||||||
|
}
|
||||||
|
aarea := a.Width * a.Height
|
||||||
|
barea := b.Width * b.Height
|
||||||
|
if aarea != barea {
|
||||||
|
return aarea < barea
|
||||||
|
}
|
||||||
|
if a.Width != b.Width {
|
||||||
|
return a.Width < b.Width
|
||||||
|
}
|
||||||
|
return a.RefreshRate < b.RefreshRate
|
||||||
|
})
|
||||||
|
m.modes = modes
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func inputMonitor(monitor *Monitor, action PeripheralEvent, placement int) error {
|
||||||
|
switch action {
|
||||||
|
case Connected:
|
||||||
|
switch placement {
|
||||||
|
case _GLFW_INSERT_FIRST:
|
||||||
|
_glfw.monitors = append(_glfw.monitors, nil)
|
||||||
|
copy(_glfw.monitors[1:], _glfw.monitors)
|
||||||
|
_glfw.monitors[0] = monitor
|
||||||
|
case _GLFW_INSERT_LAST:
|
||||||
|
_glfw.monitors = append(_glfw.monitors, monitor)
|
||||||
|
}
|
||||||
|
case Disconnected:
|
||||||
|
for _, window := range _glfw.windows {
|
||||||
|
if window.monitor == monitor {
|
||||||
|
width, height, err := window.platformGetWindowSize()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
window.platformSetWindowMonitor(nil, 0, 0, width, height, 0)
|
||||||
|
xoff, yoff, _, _, err := window.platformGetWindowFrameSize()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
window.platformSetWindowPos(xoff, yoff)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for i, m := range _glfw.monitors {
|
||||||
|
if m == monitor {
|
||||||
|
copy(_glfw.monitors[i:], _glfw.monitors[i+1:])
|
||||||
|
_glfw.monitors = _glfw.monitors[:len(_glfw.monitors)-1]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if _glfw.callbacks.monitor != nil {
|
||||||
|
_glfw.callbacks.monitor(monitor, action)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) inputMonitorWindow(window *Window) {
|
||||||
|
m.window = window
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) chooseVideoMode(desired *VidMode) (*VidMode, error) {
|
||||||
|
if err := m.refreshVideoModes(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// math.MaxUint was added at Go 1.17. See https://github.com/golang/go/issues/28538
|
||||||
|
const (
|
||||||
|
intSize = 32 << (^uint(0) >> 63)
|
||||||
|
maxUint = 1<<intSize - 1
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
leastColorDiff uint = maxUint
|
||||||
|
leastSizeDiff uint = maxUint
|
||||||
|
leastRateDiff uint = maxUint
|
||||||
|
)
|
||||||
|
|
||||||
|
var closest *VidMode
|
||||||
|
for _, v := range m.modes {
|
||||||
|
var colorDiff uint
|
||||||
|
if desired.RedBits != DontCare {
|
||||||
|
colorDiff += abs(v.RedBits - desired.RedBits)
|
||||||
|
}
|
||||||
|
if desired.GreenBits != DontCare {
|
||||||
|
colorDiff += abs(v.GreenBits - desired.GreenBits)
|
||||||
|
}
|
||||||
|
if desired.BlueBits != DontCare {
|
||||||
|
colorDiff += abs(v.BlueBits - desired.BlueBits)
|
||||||
|
}
|
||||||
|
|
||||||
|
sizeDiff := abs((v.Width-desired.Width)*(v.Width-desired.Width) +
|
||||||
|
(v.Height-desired.Height)*(v.Height-desired.Height))
|
||||||
|
|
||||||
|
var rateDiff uint
|
||||||
|
if desired.RefreshRate != DontCare {
|
||||||
|
rateDiff = abs(v.RefreshRate - desired.RefreshRate)
|
||||||
|
} else {
|
||||||
|
rateDiff = maxUint - uint(v.RefreshRate)
|
||||||
|
}
|
||||||
|
|
||||||
|
if colorDiff < leastColorDiff ||
|
||||||
|
colorDiff == leastColorDiff && sizeDiff < leastSizeDiff ||
|
||||||
|
colorDiff == leastColorDiff && sizeDiff == leastSizeDiff && rateDiff < leastRateDiff {
|
||||||
|
closest = v
|
||||||
|
leastColorDiff = colorDiff
|
||||||
|
leastSizeDiff = sizeDiff
|
||||||
|
leastRateDiff = rateDiff
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return closest, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func splitBPP(bpp int) (red, green, blue int) {
|
||||||
|
// We assume that by 32 the user really meant 24
|
||||||
|
if bpp == 32 {
|
||||||
|
bpp = 24
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert "bits per pixel" to red, green & blue sizes
|
||||||
|
red = bpp / 3
|
||||||
|
green = bpp / 3
|
||||||
|
blue = bpp / 3
|
||||||
|
delta := bpp - (red * 3)
|
||||||
|
if delta >= 1 {
|
||||||
|
green++
|
||||||
|
}
|
||||||
|
if delta == 2 {
|
||||||
|
red++
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GLFW public APIs
|
||||||
|
|
||||||
|
func GetMonitors() ([]*Monitor, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
return _glfw.monitors, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetPrimaryMonitor() (*Monitor, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
if len(_glfw.monitors) == 0 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return _glfw.monitors[0], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetPos() (xpos, ypos int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
xpos, ypos, ok := m.platformGetMonitorPos()
|
||||||
|
if !ok {
|
||||||
|
return 0, 0, nil
|
||||||
|
}
|
||||||
|
return xpos, ypos, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetWorkarea() (xpos, ypos, width, height int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
xpos, ypos, width, height = m.platformGetMonitorWorkarea()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetPhysicalSize() (widthMM, heightMM int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
return m.widthMM, m.heightMM, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetContentScale() (xscale, yscale float32, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
xscale, yscale, err = m.platformGetMonitorContentScale()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetName() (string, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return "", NotInitialized
|
||||||
|
}
|
||||||
|
return m.name, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) SetUserPointer(pointer unsafe.Pointer) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
panic("glfwwin: Monitor.SetUserPointer is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetUserPointer() (unsafe.Pointer, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
panic("glfwwin: Monitor.GetUserPointer is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetMonitorCallback(cbfun MonitorCallback) (MonitorCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := _glfw.callbacks.monitor
|
||||||
|
_glfw.callbacks.monitor = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetVideoModes() ([]*VidMode, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
return m.modes, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetVideoMode() (*VidMode, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
m.currentMode = m.platformGetVideoMode()
|
||||||
|
return m.currentMode, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) SetGamma(gamma float32) {
|
||||||
|
panic("glfwwin: Monitor.SetGamme is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) GetGammaRamp() *GammaRamp {
|
||||||
|
panic("glfwwin: Monitor.GetGammeRamp is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) SetGammaRamp(ramp *GammaRamp) {
|
||||||
|
panic("glfwwin: Monitor.SetGammeRamp is not implemented")
|
||||||
|
}
|
573
internal/glfwwin/wglcontext_windows.go
Normal file
573
internal/glfwwin/wglcontext_windows.go
Normal file
@ -0,0 +1,573 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func findPixelFormatAttribValue(attribs []int32, values []int32, attrib int32) int32 {
|
||||||
|
for i := range attribs {
|
||||||
|
if attribs[i] == attrib {
|
||||||
|
return values[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) choosePixelFormat(ctxconfig *ctxconfig, fbconfig_ *fbconfig) (int, error) {
|
||||||
|
var nativeCount int32
|
||||||
|
var attribs []int32
|
||||||
|
|
||||||
|
if _glfw.wgl.ARB_pixel_format {
|
||||||
|
var attrib int32 = _WGL_NUMBER_PIXEL_FORMATS_ARB
|
||||||
|
if err := wglGetPixelFormatAttribivARB(w.context.wgl.dc, 1, 0, 1, &attrib, &nativeCount); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
attribs = append(attribs,
|
||||||
|
_WGL_SUPPORT_OPENGL_ARB,
|
||||||
|
_WGL_DRAW_TO_WINDOW_ARB,
|
||||||
|
_WGL_PIXEL_TYPE_ARB,
|
||||||
|
_WGL_ACCELERATION_ARB,
|
||||||
|
_WGL_RED_BITS_ARB,
|
||||||
|
_WGL_RED_SHIFT_ARB,
|
||||||
|
_WGL_GREEN_BITS_ARB,
|
||||||
|
_WGL_GREEN_SHIFT_ARB,
|
||||||
|
_WGL_BLUE_BITS_ARB,
|
||||||
|
_WGL_BLUE_SHIFT_ARB,
|
||||||
|
_WGL_ALPHA_BITS_ARB,
|
||||||
|
_WGL_ALPHA_SHIFT_ARB,
|
||||||
|
_WGL_DEPTH_BITS_ARB,
|
||||||
|
_WGL_STENCIL_BITS_ARB,
|
||||||
|
_WGL_ACCUM_BITS_ARB,
|
||||||
|
_WGL_ACCUM_RED_BITS_ARB,
|
||||||
|
_WGL_ACCUM_GREEN_BITS_ARB,
|
||||||
|
_WGL_ACCUM_BLUE_BITS_ARB,
|
||||||
|
_WGL_ACCUM_ALPHA_BITS_ARB,
|
||||||
|
_WGL_AUX_BUFFERS_ARB,
|
||||||
|
_WGL_STEREO_ARB,
|
||||||
|
_WGL_DOUBLE_BUFFER_ARB)
|
||||||
|
|
||||||
|
if _glfw.wgl.ARB_multisample {
|
||||||
|
attribs = append(attribs, _WGL_SAMPLES_ARB)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.client == OpenGLAPI {
|
||||||
|
if _glfw.wgl.ARB_framebuffer_sRGB || _glfw.wgl.EXT_framebuffer_sRGB {
|
||||||
|
attribs = append(attribs, _WGL_FRAMEBUFFER_SRGB_CAPABLE_ARB)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if _glfw.wgl.EXT_colorspace {
|
||||||
|
attribs = append(attribs, _WGL_COLORSPACE_EXT)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
c, err := _DescribePixelFormat(w.context.wgl.dc, 1, uint32(unsafe.Sizeof(_PIXELFORMATDESCRIPTOR{})), nil)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
nativeCount = c
|
||||||
|
}
|
||||||
|
|
||||||
|
usableConfigs := make([]*fbconfig, 0, nativeCount)
|
||||||
|
for i := int32(0); i < nativeCount; i++ {
|
||||||
|
var u fbconfig
|
||||||
|
pixelFormat := uintptr(i) + 1
|
||||||
|
|
||||||
|
if _glfw.wgl.ARB_pixel_format {
|
||||||
|
// Get pixel format attributes through "modern" extension
|
||||||
|
values := make([]int32, len(attribs))
|
||||||
|
if err := wglGetPixelFormatAttribivARB(w.context.wgl.dc, int32(pixelFormat), 0, uint32(len(attribs)), &attribs[0], &values[0]); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
findAttribValue := func(attrib int32) int32 {
|
||||||
|
return findPixelFormatAttribValue(attribs, values, attrib)
|
||||||
|
}
|
||||||
|
|
||||||
|
if findAttribValue(_WGL_SUPPORT_OPENGL_ARB) == 0 || findAttribValue(_WGL_DRAW_TO_WINDOW_ARB) == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if findAttribValue(_WGL_PIXEL_TYPE_ARB) != _WGL_TYPE_RGBA_ARB {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if findAttribValue(_WGL_ACCELERATION_ARB) == _WGL_NO_ACCELERATION_ARB {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if (findAttribValue(_WGL_DOUBLE_BUFFER_ARB) != 0) != fbconfig_.doublebuffer {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
u.redBits = int(findAttribValue(_WGL_RED_BITS_ARB))
|
||||||
|
u.greenBits = int(findAttribValue(_WGL_GREEN_BITS_ARB))
|
||||||
|
u.blueBits = int(findAttribValue(_WGL_BLUE_BITS_ARB))
|
||||||
|
u.alphaBits = int(findAttribValue(_WGL_ALPHA_BITS_ARB))
|
||||||
|
|
||||||
|
u.depthBits = int(findAttribValue(_WGL_DEPTH_BITS_ARB))
|
||||||
|
u.stencilBits = int(findAttribValue(_WGL_STENCIL_BITS_ARB))
|
||||||
|
|
||||||
|
u.accumRedBits = int(findAttribValue(_WGL_ACCUM_RED_BITS_ARB))
|
||||||
|
u.accumGreenBits = int(findAttribValue(_WGL_ACCUM_GREEN_BITS_ARB))
|
||||||
|
u.accumBlueBits = int(findAttribValue(_WGL_ACCUM_BLUE_BITS_ARB))
|
||||||
|
u.accumAlphaBits = int(findAttribValue(_WGL_ACCUM_ALPHA_BITS_ARB))
|
||||||
|
|
||||||
|
u.auxBuffers = int(findAttribValue(_WGL_AUX_BUFFERS_ARB))
|
||||||
|
|
||||||
|
if findAttribValue(_WGL_STEREO_ARB) != 0 {
|
||||||
|
u.stereo = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if _glfw.wgl.ARB_multisample {
|
||||||
|
u.samples = int(findAttribValue(_WGL_SAMPLES_ARB))
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.client == OpenGLAPI {
|
||||||
|
if _glfw.wgl.ARB_framebuffer_sRGB || _glfw.wgl.EXT_framebuffer_sRGB {
|
||||||
|
if findAttribValue(_WGL_FRAMEBUFFER_SRGB_CAPABLE_ARB) != 0 {
|
||||||
|
u.sRGB = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if _glfw.wgl.EXT_colorspace {
|
||||||
|
if findAttribValue(_WGL_COLORSPACE_EXT) == _WGL_COLORSPACE_SRGB_EXT {
|
||||||
|
u.sRGB = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Get pixel format attributes through legacy PFDs
|
||||||
|
|
||||||
|
var pfd _PIXELFORMATDESCRIPTOR
|
||||||
|
if _, err := _DescribePixelFormat(w.context.wgl.dc, int32(pixelFormat), uint32(unsafe.Sizeof(pfd)), &pfd); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if pfd.dwFlags&_PFD_DRAW_TO_WINDOW == 0 || pfd.dwFlags&_PFD_SUPPORT_OPENGL == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if pfd.dwFlags&_PFD_GENERIC_ACCELERATED == 0 && pfd.dwFlags&_PFD_GENERIC_FORMAT != 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if pfd.iPixelType != _PFD_TYPE_RGBA {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pfd.dwFlags&_PFD_DOUBLEBUFFER != 0) != fbconfig_.doublebuffer {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
u.redBits = int(pfd.cRedBits)
|
||||||
|
u.greenBits = int(pfd.cGreenBits)
|
||||||
|
u.blueBits = int(pfd.cBlueBits)
|
||||||
|
u.alphaBits = int(pfd.cAlphaBits)
|
||||||
|
|
||||||
|
u.depthBits = int(pfd.cDepthBits)
|
||||||
|
u.stencilBits = int(pfd.cStencilBits)
|
||||||
|
|
||||||
|
u.accumRedBits = int(pfd.cAccumRedBits)
|
||||||
|
u.accumGreenBits = int(pfd.cAccumGreenBits)
|
||||||
|
u.accumBlueBits = int(pfd.cAccumBlueBits)
|
||||||
|
u.accumAlphaBits = int(pfd.cAccumAlphaBits)
|
||||||
|
|
||||||
|
u.auxBuffers = int(pfd.cAuxBuffers)
|
||||||
|
|
||||||
|
if pfd.dwFlags&_PFD_STEREO != 0 {
|
||||||
|
u.stereo = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
u.handle = pixelFormat
|
||||||
|
usableConfigs = append(usableConfigs, &u)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(usableConfigs) == 0 {
|
||||||
|
return 0, fmt.Errorf("glfwwin: the driver does not appear to support OpenGL")
|
||||||
|
}
|
||||||
|
|
||||||
|
closest := chooseFBConfig(fbconfig_, usableConfigs)
|
||||||
|
if closest == nil {
|
||||||
|
return 0, fmt.Errorf("glfwwin: failed to find a suitable pixel format")
|
||||||
|
}
|
||||||
|
|
||||||
|
return int(closest.handle), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeContextCurrentWGL(window *Window) error {
|
||||||
|
if window != nil {
|
||||||
|
if err := wglMakeCurrent(window.context.wgl.dc, window.context.wgl.handle); err != nil {
|
||||||
|
_glfw.contextSlot.set(0)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := _glfw.contextSlot.set(uintptr(unsafe.Pointer(window))); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := wglMakeCurrent(0, 0); err != nil {
|
||||||
|
_glfw.contextSlot.set(0)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := _glfw.contextSlot.set(0); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func swapBuffersWGL(window *Window) error {
|
||||||
|
if window.monitor == nil && _IsWindowsVistaOrGreater() {
|
||||||
|
// DWM Composition is always enabled on Win8+
|
||||||
|
enabled := _IsWindows8OrGreater()
|
||||||
|
|
||||||
|
if !enabled {
|
||||||
|
var err error
|
||||||
|
enabled, err = _DwmIsCompositionEnabled()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HACK: Use DwmFlush when desktop composition is enabled
|
||||||
|
if enabled {
|
||||||
|
for i := 0; i < window.context.wgl.interval; i++ {
|
||||||
|
if err := _DwmFlush(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := _SwapBuffers(window.context.wgl.dc); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func swapIntervalWGL(interval int) error {
|
||||||
|
ptr, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
window := (*Window)(unsafe.Pointer(ptr))
|
||||||
|
|
||||||
|
window.context.wgl.interval = interval
|
||||||
|
|
||||||
|
if window.monitor == nil && _IsWindowsVistaOrGreater() {
|
||||||
|
// DWM Composition is always enabled on Win8+
|
||||||
|
enabled := _IsWindows8OrGreater()
|
||||||
|
|
||||||
|
if !enabled {
|
||||||
|
var err error
|
||||||
|
enabled, err = _DwmIsCompositionEnabled()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HACK: Disable WGL swap interval when desktop composition is enabled to
|
||||||
|
// avoid interfering with DWM vsync
|
||||||
|
if enabled {
|
||||||
|
interval = 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if _glfw.wgl.EXT_swap_control {
|
||||||
|
if err := wglSwapIntervalEXT(int32(interval)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func extensionSupportedWGL(extension string) bool {
|
||||||
|
var extensions string
|
||||||
|
|
||||||
|
if wglGetExtensionsStringARB_Available() {
|
||||||
|
extensions = wglGetExtensionsStringARB(wglGetCurrentDC())
|
||||||
|
} else if wglGetExtensionsStringEXT_Available() {
|
||||||
|
extensions = wglGetExtensionsStringEXT()
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(extensions) == 0 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, str := range strings.Split(extensions, " ") {
|
||||||
|
if extension == str {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func getProcAddressWGL(procname string) uintptr {
|
||||||
|
proc := wglGetProcAddress(procname)
|
||||||
|
if proc != 0 {
|
||||||
|
return proc
|
||||||
|
}
|
||||||
|
return opengl32.NewProc(procname).Addr()
|
||||||
|
}
|
||||||
|
|
||||||
|
func destroyContextWGL(window *Window) error {
|
||||||
|
if window.context.wgl.handle != 0 {
|
||||||
|
if err := wglDeleteContext(window.context.wgl.handle); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
window.context.wgl.handle = 0
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func initWGL() error {
|
||||||
|
if _glfw.wgl.inited {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// opengl32.dll must be loaded first. The loading state might affect Windows APIs.
|
||||||
|
// This is needed at least before SetPixelFormat.
|
||||||
|
if err := opengl32.Load(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: A dummy context has to be created for opengl32.dll to load the
|
||||||
|
// OpenGL ICD, from which we can then query WGL extensions
|
||||||
|
// NOTE: This code will accept the Microsoft GDI ICD; accelerated context
|
||||||
|
// creation failure occurs during manual pixel format enumeration
|
||||||
|
|
||||||
|
dc, err := _GetDC(_glfw.win32.helperWindowHandle)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
pfd := _PIXELFORMATDESCRIPTOR{
|
||||||
|
nVersion: 1,
|
||||||
|
dwFlags: _PFD_DRAW_TO_WINDOW | _PFD_SUPPORT_OPENGL | _PFD_DOUBLEBUFFER,
|
||||||
|
iPixelType: _PFD_TYPE_RGBA,
|
||||||
|
cColorBits: 24,
|
||||||
|
}
|
||||||
|
pfd.nSize = uint16(unsafe.Sizeof(pfd))
|
||||||
|
|
||||||
|
format, err := _ChoosePixelFormat(dc, &pfd)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := _SetPixelFormat(dc, format, &pfd); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
rc, err := wglCreateContext(dc)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
pdc := wglGetCurrentDC()
|
||||||
|
prc := wglGetCurrentContext()
|
||||||
|
|
||||||
|
if err := wglMakeCurrent(dc, rc); err != nil {
|
||||||
|
wglMakeCurrent(pdc, prc)
|
||||||
|
wglDeleteContext(rc)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: WGL_ARB_extensions_string and WGL_EXT_extensions_string are not
|
||||||
|
// checked below as we are already using them
|
||||||
|
_glfw.wgl.ARB_multisample = extensionSupportedWGL("WGL_ARB_multisample")
|
||||||
|
_glfw.wgl.ARB_framebuffer_sRGB = extensionSupportedWGL("WGL_ARB_framebuffer_sRGB")
|
||||||
|
_glfw.wgl.EXT_framebuffer_sRGB = extensionSupportedWGL("WGL_EXT_framebuffer_sRGB")
|
||||||
|
_glfw.wgl.ARB_create_context = extensionSupportedWGL("WGL_ARB_create_context")
|
||||||
|
_glfw.wgl.ARB_create_context_profile = extensionSupportedWGL("WGL_ARB_create_context_profile")
|
||||||
|
_glfw.wgl.EXT_create_context_es2_profile = extensionSupportedWGL("WGL_EXT_create_context_es2_profile")
|
||||||
|
_glfw.wgl.ARB_create_context_robustness = extensionSupportedWGL("WGL_ARB_create_context_robustness")
|
||||||
|
_glfw.wgl.ARB_create_context_no_error = extensionSupportedWGL("WGL_ARB_create_context_no_error")
|
||||||
|
_glfw.wgl.EXT_swap_control = extensionSupportedWGL("WGL_EXT_swap_control")
|
||||||
|
_glfw.wgl.EXT_colorspace = extensionSupportedWGL("WGL_EXT_colorspace")
|
||||||
|
_glfw.wgl.ARB_pixel_format = extensionSupportedWGL("WGL_ARB_pixel_format")
|
||||||
|
_glfw.wgl.ARB_context_flush_control = extensionSupportedWGL("WGL_ARB_context_flush_control")
|
||||||
|
|
||||||
|
if err := wglMakeCurrent(pdc, prc); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := wglDeleteContext(rc); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_glfw.wgl.inited = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func terminateWGL() {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) createContextWGL(ctxconfig *ctxconfig, fbconfig *fbconfig) error {
|
||||||
|
var share _HGLRC
|
||||||
|
if ctxconfig.share != nil {
|
||||||
|
share = ctxconfig.share.context.wgl.handle
|
||||||
|
}
|
||||||
|
|
||||||
|
dc, err := _GetDC(w.win32.handle)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
w.context.wgl.dc = dc
|
||||||
|
|
||||||
|
pixelFormat, err := w.choosePixelFormat(ctxconfig, fbconfig)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
var pfd _PIXELFORMATDESCRIPTOR
|
||||||
|
if _, err := _DescribePixelFormat(w.context.wgl.dc, int32(pixelFormat), uint32(unsafe.Sizeof(pfd)), &pfd); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := _SetPixelFormat(w.context.wgl.dc, int32(pixelFormat), &pfd); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.client == OpenGLAPI {
|
||||||
|
if ctxconfig.forward && !_glfw.wgl.ARB_create_context {
|
||||||
|
return fmt.Errorf("glfwwin: a forward compatible OpenGL context requested but WGL_ARB_create_context is unavailable: %w", VersionUnavailable)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.profile != 0 && !_glfw.wgl.ARB_create_context_profile {
|
||||||
|
return fmt.Errorf("glfwwin: OpenGL profile requested but WGL_ARB_create_context_profile is unavailable: %w", VersionUnavailable)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !_glfw.wgl.ARB_create_context || !_glfw.wgl.ARB_create_context_profile || !_glfw.wgl.EXT_create_context_es2_profile {
|
||||||
|
return fmt.Errorf("glfwwin: OpenGL ES requested but WGL_ARB_create_context_es2_profile is unavailable: %w", ApiUnavailable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if _glfw.wgl.ARB_create_context {
|
||||||
|
var flags int32
|
||||||
|
var mask int32
|
||||||
|
if ctxconfig.client == OpenGLAPI {
|
||||||
|
if ctxconfig.forward {
|
||||||
|
flags |= _WGL_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.profile == OpenGLCoreProfile {
|
||||||
|
mask |= _WGL_CONTEXT_CORE_PROFILE_BIT_ARB
|
||||||
|
} else if ctxconfig.profile == OpenGLCompatProfile {
|
||||||
|
mask |= _WGL_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
mask |= _WGL_CONTEXT_ES2_PROFILE_BIT_EXT
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.debug {
|
||||||
|
flags |= _WGL_CONTEXT_DEBUG_BIT_ARB
|
||||||
|
}
|
||||||
|
|
||||||
|
var attribs []int32
|
||||||
|
if ctxconfig.robustness != 0 {
|
||||||
|
if _glfw.wgl.ARB_create_context_robustness {
|
||||||
|
if ctxconfig.robustness == NoResetNotification {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_RESET_NOTIFICATION_STRATEGY_ARB, _WGL_NO_RESET_NOTIFICATION_ARB)
|
||||||
|
} else if ctxconfig.robustness == LoseContextOnReset {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_RESET_NOTIFICATION_STRATEGY_ARB, _WGL_LOSE_CONTEXT_ON_RESET_ARB)
|
||||||
|
}
|
||||||
|
flags |= _WGL_CONTEXT_ROBUST_ACCESS_BIT_ARB
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.release != 0 {
|
||||||
|
if _glfw.wgl.ARB_context_flush_control {
|
||||||
|
if ctxconfig.release == ReleaseBehaviorNone {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_RELEASE_BEHAVIOR_ARB, _WGL_CONTEXT_RELEASE_BEHAVIOR_NONE_ARB)
|
||||||
|
} else if ctxconfig.release == ReleaseBehaviorFlush {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_RELEASE_BEHAVIOR_ARB, _WGL_CONTEXT_RELEASE_BEHAVIOR_FLUSH_ARB)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.noerror {
|
||||||
|
if _glfw.wgl.ARB_create_context_no_error {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_OPENGL_NO_ERROR_ARB, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: Only request an explicitly versioned context when necessary, as
|
||||||
|
// explicitly requesting version 1.0 does not always return the
|
||||||
|
// highest version supported by the driver
|
||||||
|
if ctxconfig.major != 1 || ctxconfig.minor != 0 {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_MAJOR_VERSION_ARB, int32(ctxconfig.major))
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_MINOR_VERSION_ARB, int32(ctxconfig.minor))
|
||||||
|
}
|
||||||
|
|
||||||
|
if flags != 0 {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_FLAGS_ARB, flags)
|
||||||
|
}
|
||||||
|
|
||||||
|
if mask != 0 {
|
||||||
|
attribs = append(attribs, _WGL_CONTEXT_PROFILE_MASK_ARB, mask)
|
||||||
|
}
|
||||||
|
|
||||||
|
attribs = append(attribs, 0, 0)
|
||||||
|
|
||||||
|
var err error
|
||||||
|
w.context.wgl.handle, err = wglCreateContextAttribsARB(w.context.wgl.dc, share, &attribs[0])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
var err error
|
||||||
|
w.context.wgl.handle, err = wglCreateContext(w.context.wgl.dc)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if share != 0 {
|
||||||
|
if err := wglShareLists(share, w.context.wgl.handle); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w.context.makeCurrent = makeContextCurrentWGL
|
||||||
|
w.context.swapBuffers = swapBuffersWGL
|
||||||
|
w.context.swapInterval = swapIntervalWGL
|
||||||
|
w.context.extensionSupported = extensionSupportedWGL
|
||||||
|
w.context.getProcAddress = getProcAddressWGL
|
||||||
|
w.context.destroy = destroyContextWGL
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getWGLContext(handle *Window) _HGLRC {
|
||||||
|
window := handle
|
||||||
|
if !_glfw.initialized {
|
||||||
|
panic(NotInitialized)
|
||||||
|
}
|
||||||
|
if window.context.source != NativeContextAPI {
|
||||||
|
// TODO: Should this return an error?
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return window.context.wgl.handle
|
||||||
|
}
|
344
internal/glfwwin/win32init_windows.go
Normal file
344
internal/glfwwin/win32init_windows.go
Normal file
@ -0,0 +1,344 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"golang.org/x/sys/windows"
|
||||||
|
)
|
||||||
|
|
||||||
|
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[0x146] = 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 {
|
||||||
|
m, err := _GetModuleHandleW("")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
h, err := _CreateWindowExW(_WS_EX_OVERLAPPEDWINDOW, _GLFW_WNDCLASSNAME, "GLFW message window", _WS_CLIPSIBLINGS|_WS_CLIPCHILDREN, 0, 0, 1, 1, 0, 0, _HINSTANCE(m), nil)
|
||||||
|
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)
|
||||||
|
|
||||||
|
_GUID_DEVINTERFACE_HID := windows.GUID{0x4d1e55b2, 0xf16f, 0x11cf, [...]byte{0x88, 0xcb, 0x00, 0x11, 0x11, 0x00, 0x00, 0x30}}
|
||||||
|
|
||||||
|
// Register for HID device notifications
|
||||||
|
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
|
||||||
|
|
||||||
|
var msg _MSG
|
||||||
|
for _PeekMessageW(&msg, _glfw.win32.helperWindowHandle, 0, 0, _PM_REMOVE) {
|
||||||
|
_TranslateMessage(&msg)
|
||||||
|
_DispatchMessageW(&msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func updateKeyNamesWin32() {
|
||||||
|
var state [256]byte
|
||||||
|
for key := KeySpace; key <= KeyLast; key++ {
|
||||||
|
scancode := _glfw.win32.scancodes[key]
|
||||||
|
if scancode == -1 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
var vk uint32
|
||||||
|
if key >= KeyKP0 && key <= KeyKPAdd {
|
||||||
|
vks := []uint32{
|
||||||
|
_VK_NUMPAD0, _VK_NUMPAD1, _VK_NUMPAD2, _VK_NUMPAD3,
|
||||||
|
_VK_NUMPAD4, _VK_NUMPAD5, _VK_NUMPAD6, _VK_NUMPAD7,
|
||||||
|
_VK_NUMPAD8, _VK_NUMPAD9, _VK_DECIMAL, _VK_DIVIDE,
|
||||||
|
_VK_MULTIPLY, _VK_SUBTRACT, _VK_ADD,
|
||||||
|
}
|
||||||
|
vk = vks[key-KeyKP0]
|
||||||
|
} else {
|
||||||
|
vk = _MapVirtualKeyW(uint32(scancode), _MAPVK_VSC_TO_VK)
|
||||||
|
}
|
||||||
|
|
||||||
|
var chars [16]uint16
|
||||||
|
length := _ToUnicode(vk, uint32(scancode), &state[0], &chars[0], int32(len(chars)), 0)
|
||||||
|
if length == -1 {
|
||||||
|
// TODO: Why is ToUnicode called twice?
|
||||||
|
length = _ToUnicode(vk, uint32(scancode), &state[0], &chars[0], int32(len(chars)), 0)
|
||||||
|
}
|
||||||
|
if length < 1 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
_glfw.win32.keynames[key] = windows.UTF16ToString(chars[:1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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 {
|
||||||
|
// To make SetForegroundWindow work as we want, we need to fiddle
|
||||||
|
// with the FOREGROUNDLOCKTIMEOUT system setting (we do this as early
|
||||||
|
// as possible in the hope of still being the foreground process)
|
||||||
|
if err := _SystemParametersInfoW(_SPI_GETFOREGROUNDLOCKTIMEOUT, 0, uintptr(unsafe.Pointer(&_glfw.win32.foregroundLockTimeout)), 0); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := _SystemParametersInfoW(_SPI_SETFOREGROUNDLOCKTIMEOUT, 0, 0, _SPIF_SENDCHANGE); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
createKeyTables()
|
||||||
|
updateKeyNamesWin32()
|
||||||
|
|
||||||
|
if isWindows10CreatorsUpdateOrGreaterWin32() {
|
||||||
|
if err := _SetProcessDpiAwarenessContext(_DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2); err != nil && !errors.Is(err, windows.ERROR_ACCESS_DENIED) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else if _IsWindows8Point1OrGreater() {
|
||||||
|
if err := _SetProcessDpiAwareness(_PROCESS_PER_MONITOR_DPI_AWARE); err != nil && !errors.Is(err, windows.ERROR_ACCESS_DENIED) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else if _IsWindowsVistaOrGreater() {
|
||||||
|
_SetProcessDPIAware()
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := registerWindowClassWin32(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := createHelperWindow(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := pollMonitorsWin32(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
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 {
|
||||||
|
if err := _DestroyWindow(_glfw.win32.helperWindowHandle); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := unregisterWindowClassWin32(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restore previous foreground lock timeout system setting
|
||||||
|
if err := _SystemParametersInfoW(_SPI_SETFOREGROUNDLOCKTIMEOUT, 0, uintptr(_glfw.win32.foregroundLockTimeout), _SPIF_SENDCHANGE); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
terminateWGL()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
375
internal/glfwwin/win32monitor_windows.go
Normal file
375
internal/glfwwin/win32monitor_windows.go
Normal file
@ -0,0 +1,375 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"golang.org/x/sys/windows"
|
||||||
|
)
|
||||||
|
|
||||||
|
func monitorCallback(handle _HMONITOR, dc _HDC, rect *_RECT, data _LPARAM) uintptr /* _BOOL */ {
|
||||||
|
if mi, ok := _GetMonitorInfoW_Ex(handle); ok {
|
||||||
|
monitor := (*Monitor)(unsafe.Pointer(data))
|
||||||
|
if windows.UTF16ToString(mi.szDevice[:]) == monitor.win32.adapterName {
|
||||||
|
monitor.win32.handle = handle
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
var monitorCallbackPtr = windows.NewCallbackCDecl(monitorCallback)
|
||||||
|
|
||||||
|
func createMonitor(adapter *_DISPLAY_DEVICEW, display *_DISPLAY_DEVICEW) (*Monitor, error) {
|
||||||
|
var name string
|
||||||
|
if display != nil {
|
||||||
|
name = windows.UTF16ToString(display.DeviceString[:])
|
||||||
|
} else {
|
||||||
|
name = windows.UTF16ToString(adapter.DeviceString[:])
|
||||||
|
}
|
||||||
|
if name == "" {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
adapterDeviceName := windows.UTF16ToString(adapter.DeviceName[:])
|
||||||
|
dm, ok := _EnumDisplaySettingsW(adapterDeviceName, _ENUM_CURRENT_SETTINGS)
|
||||||
|
if !ok {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var widthMM, heightMM int
|
||||||
|
dc, err := _CreateDCW("DISPLAY", adapterDeviceName, "", nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if _IsWindows8Point1OrGreater() {
|
||||||
|
widthMM = int(_GetDeviceCaps(dc, _HORZSIZE))
|
||||||
|
heightMM = int(_GetDeviceCaps(dc, _VERTSIZE))
|
||||||
|
} else {
|
||||||
|
widthMM = int(float64(dm.dmPelsWidth) * 25.4 / float64(_GetDeviceCaps(dc, _LOGPIXELSX)))
|
||||||
|
heightMM = int(float64(dm.dmPelsHeight) * 25.4 / float64(_GetDeviceCaps(dc, _LOGPIXELSY)))
|
||||||
|
}
|
||||||
|
if err := _DeleteDC(dc); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
monitor := &Monitor{
|
||||||
|
name: name,
|
||||||
|
widthMM: widthMM,
|
||||||
|
heightMM: heightMM,
|
||||||
|
}
|
||||||
|
|
||||||
|
if adapter.StateFlags&_DISPLAY_DEVICE_MODESPRUNED != 0 {
|
||||||
|
monitor.win32.modesPruned = true
|
||||||
|
}
|
||||||
|
|
||||||
|
monitor.win32.adapterName = adapterDeviceName
|
||||||
|
if display != nil {
|
||||||
|
monitor.win32.displayName = windows.UTF16ToString(display.DeviceName[:])
|
||||||
|
}
|
||||||
|
|
||||||
|
rect := _RECT{
|
||||||
|
left: dm.dmPosition.x,
|
||||||
|
top: dm.dmPosition.y,
|
||||||
|
right: dm.dmPosition.x + int32(dm.dmPelsWidth),
|
||||||
|
bottom: dm.dmPosition.y + int32(dm.dmPelsHeight),
|
||||||
|
}
|
||||||
|
if err := _EnumDisplayMonitors(0, &rect, monitorCallbackPtr, _LPARAM(unsafe.Pointer(monitor))); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return monitor, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func pollMonitorsWin32() error {
|
||||||
|
disconnected := make([]*Monitor, len(_glfw.monitors))
|
||||||
|
copy(disconnected, _glfw.monitors)
|
||||||
|
|
||||||
|
adapterLoop:
|
||||||
|
for adapterIndex := uint32(0); ; adapterIndex++ {
|
||||||
|
adapter, ok := _EnumDisplayDevicesW("", adapterIndex, 0)
|
||||||
|
if !ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if adapter.StateFlags&_DISPLAY_DEVICE_ACTIVE == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
typ := _GLFW_INSERT_LAST
|
||||||
|
if adapter.StateFlags&_DISPLAY_DEVICE_PRIMARY_DEVICE != 0 {
|
||||||
|
typ = _GLFW_INSERT_FIRST
|
||||||
|
}
|
||||||
|
|
||||||
|
var found bool
|
||||||
|
displayLoop:
|
||||||
|
for displayIndex := uint32(0); ; displayIndex++ {
|
||||||
|
display, ok := _EnumDisplayDevicesW(windows.UTF16ToString(adapter.DeviceName[:]), displayIndex, 0)
|
||||||
|
if !ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
found = true
|
||||||
|
if display.StateFlags&_DISPLAY_DEVICE_ACTIVE == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, monitor := range disconnected {
|
||||||
|
if monitor != nil && monitor.win32.displayName == windows.UTF16ToString(display.DeviceName[:]) {
|
||||||
|
disconnected[i] = nil
|
||||||
|
_EnumDisplayMonitors(0, nil, monitorCallbackPtr, _LPARAM(unsafe.Pointer(_glfw.monitors[i])))
|
||||||
|
continue displayLoop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
monitor, err := createMonitor(&adapter, &display)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if monitor == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := inputMonitor(monitor, Connected, typ); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
typ = _GLFW_INSERT_LAST
|
||||||
|
}
|
||||||
|
|
||||||
|
// HACK: If an active adapter does not have any display devices
|
||||||
|
// (as sometimes happens), add it directly as a monitor
|
||||||
|
if !found {
|
||||||
|
for i, monitor := range disconnected {
|
||||||
|
if monitor != nil && monitor.win32.displayName == windows.UTF16ToString(adapter.DeviceName[:]) {
|
||||||
|
disconnected[i] = nil
|
||||||
|
continue adapterLoop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
monitor, err := createMonitor(&adapter, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if monitor == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := inputMonitor(monitor, Connected, typ); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, monitor := range disconnected {
|
||||||
|
if monitor != nil {
|
||||||
|
if err := inputMonitor(monitor, Disconnected, 0); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) setVideoModeWin32(desired *VidMode) error {
|
||||||
|
best, err := m.chooseVideoMode(desired)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
current := m.platformGetVideoMode()
|
||||||
|
if best.equals(current) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
dm := _DEVMODEW{
|
||||||
|
dmFields: _DM_PELSWIDTH | _DM_PELSHEIGHT | _DM_BITSPERPEL | _DM_DISPLAYFREQUENCY,
|
||||||
|
dmPelsWidth: uint32(best.Width),
|
||||||
|
dmPelsHeight: uint32(best.Height),
|
||||||
|
dmBitsPerPel: uint32(best.RedBits + best.GreenBits + best.BlueBits),
|
||||||
|
dmDisplayFrequency: uint32(best.RefreshRate),
|
||||||
|
}
|
||||||
|
dm.dmSize = uint16(unsafe.Sizeof(dm))
|
||||||
|
if dm.dmBitsPerPel < 15 || dm.dmBitsPerPel >= 24 {
|
||||||
|
dm.dmBitsPerPel = 32
|
||||||
|
}
|
||||||
|
switch _ChangeDisplaySettingsExW(m.win32.adapterName, &dm, 0, _CDS_FULLSCREEN, nil) {
|
||||||
|
case _DISP_CHANGE_SUCCESSFUL:
|
||||||
|
m.win32.modeChanged = true
|
||||||
|
return nil
|
||||||
|
case _DISP_CHANGE_BADDUALVIEW:
|
||||||
|
return errors.New("glfwwin: the system uses DualView at Monitor.setVideoModeWin32")
|
||||||
|
case _DISP_CHANGE_BADFLAGS:
|
||||||
|
return errors.New("glfwwin: invalid flags at Monitor.setVideoModeWin32")
|
||||||
|
case _DISP_CHANGE_BADMODE:
|
||||||
|
return errors.New("glfwwin: graphics mode not supported at Monitor.setVideoModeWin32")
|
||||||
|
case _DISP_CHANGE_BADPARAM:
|
||||||
|
return errors.New("glfwwin: invalid parameter at Monitor.setVideoModeWin32")
|
||||||
|
case _DISP_CHANGE_FAILED:
|
||||||
|
return errors.New("glfwwin: graphics mode failed at Monitor.setVideoModeWin32")
|
||||||
|
case _DISP_CHANGE_NOTUPDATED:
|
||||||
|
return errors.New("glfwwin: failed to write to registry at Monitor.setVideoModeWin32")
|
||||||
|
case _DISP_CHANGE_RESTART:
|
||||||
|
return errors.New("glfwwin: computer restart required at Monitor.setVideoModeWin32")
|
||||||
|
default:
|
||||||
|
return errors.New("glfwwin: unknown error at Monitor.setVideoModeWin32")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) restoreVideoModeWin32() {
|
||||||
|
if m.win32.modeChanged {
|
||||||
|
_ChangeDisplaySettingsExW(m.win32.adapterName, nil, 0, _CDS_FULLSCREEN, nil)
|
||||||
|
m.win32.modeChanged = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getMonitorContentScaleWin32(handle _HMONITOR) (xscale, yscale float32, err error) {
|
||||||
|
var xdpi, ydpi uint32
|
||||||
|
|
||||||
|
if _IsWindows8Point1OrGreater() {
|
||||||
|
var err error
|
||||||
|
xdpi, ydpi, err = _GetDpiForMonitor(handle, _MDT_EFFECTIVE_DPI)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
dc, err := _GetDC(0)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
defer _ReleaseDC(0, dc)
|
||||||
|
|
||||||
|
xdpi = uint32(_GetDeviceCaps(dc, _LOGPIXELSX))
|
||||||
|
ydpi = uint32(_GetDeviceCaps(dc, _LOGPIXELSY))
|
||||||
|
}
|
||||||
|
|
||||||
|
xscale = float32(xdpi) / _USER_DEFAULT_SCREEN_DPI
|
||||||
|
yscale = float32(ydpi) / _USER_DEFAULT_SCREEN_DPI
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformGetMonitorPos() (xpos, ypos int, ok bool) {
|
||||||
|
dm, ok := _EnumDisplaySettingsExW(m.win32.adapterName, _ENUM_CURRENT_SETTINGS, _EDS_ROTATEDMODE)
|
||||||
|
if !ok {
|
||||||
|
return 0, 0, false
|
||||||
|
}
|
||||||
|
return int(dm.dmPosition.x), int(dm.dmPosition.y), true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformGetMonitorContentScale() (xscale, yscale float32, err error) {
|
||||||
|
return getMonitorContentScaleWin32(m.win32.handle)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformGetMonitorWorkarea() (xpos, ypos, width, height int) {
|
||||||
|
mi, ok := _GetMonitorInfoW(m.win32.handle)
|
||||||
|
if !ok {
|
||||||
|
return 0, 0, 0, 0
|
||||||
|
}
|
||||||
|
xpos = int(mi.rcWork.left)
|
||||||
|
ypos = int(mi.rcWork.top)
|
||||||
|
width = int(mi.rcWork.right - mi.rcWork.left)
|
||||||
|
height = int(mi.rcWork.bottom - mi.rcWork.top)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformAppendVideoModes(monitors []*VidMode) ([]*VidMode, error) {
|
||||||
|
origLen := len(monitors)
|
||||||
|
loop:
|
||||||
|
for modeIndex := uint32(0); ; modeIndex++ {
|
||||||
|
dm, ok := _EnumDisplaySettingsW(m.win32.adapterName, modeIndex)
|
||||||
|
if !ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip modes with less than 15 BPP
|
||||||
|
if dm.dmBitsPerPel < 15 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
r, g, b := splitBPP(int(dm.dmBitsPerPel))
|
||||||
|
mode := &VidMode{
|
||||||
|
Width: int(dm.dmPelsWidth),
|
||||||
|
Height: int(dm.dmPelsHeight),
|
||||||
|
RefreshRate: int(dm.dmDisplayFrequency),
|
||||||
|
RedBits: r,
|
||||||
|
GreenBits: g,
|
||||||
|
BlueBits: b,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip duplicate modes
|
||||||
|
for _, m := range monitors[origLen:] {
|
||||||
|
if m.equals(mode) {
|
||||||
|
continue loop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.win32.modesPruned {
|
||||||
|
// Skip modes not supported by the connected displays
|
||||||
|
if _ChangeDisplaySettingsExW(m.win32.adapterName, &dm, 0, _CDS_TEST, nil) != _DISP_CHANGE_SUCCESSFUL {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
monitors = append(monitors, mode)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(monitors) == origLen {
|
||||||
|
// HACK: Report the current mode if no valid modes were found
|
||||||
|
monitors = append(monitors, m.platformGetVideoMode())
|
||||||
|
}
|
||||||
|
|
||||||
|
return monitors, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformGetVideoMode() *VidMode {
|
||||||
|
dm, _ := _EnumDisplaySettingsW(m.win32.adapterName, _ENUM_CURRENT_SETTINGS)
|
||||||
|
r, g, b := splitBPP(int(dm.dmBitsPerPel))
|
||||||
|
return &VidMode{
|
||||||
|
Width: int(dm.dmPelsWidth),
|
||||||
|
Height: int(dm.dmPelsHeight),
|
||||||
|
RefreshRate: int(dm.dmDisplayFrequency),
|
||||||
|
RedBits: r,
|
||||||
|
GreenBits: g,
|
||||||
|
BlueBits: b,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformGetGammaRamp(ramp *GammaRamp) {
|
||||||
|
panic("glfwwin: platformGetGammaRamp is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) platformSetGammaRamp(ramp *GammaRamp) {
|
||||||
|
panic("glfwwin: platformSetGammaRamp is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) in32Adapter() (string, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return "", NotInitialized
|
||||||
|
}
|
||||||
|
return m.win32.adapterName, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Monitor) win32Monitor() (string, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return "", NotInitialized
|
||||||
|
}
|
||||||
|
return m.win32.displayName, nil
|
||||||
|
}
|
56
internal/glfwwin/win32platform_windows.go
Normal file
56
internal/glfwwin/win32platform_windows.go
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
const (
|
||||||
|
_GLFW_WNDCLASSNAME = "GLFW30"
|
||||||
|
)
|
||||||
|
|
||||||
|
func _IsWindowsXPOrGreater() bool {
|
||||||
|
return isWindowsVersionOrGreaterWin32(uint16(_HIBYTE(_WIN32_WINNT_WINXP)), uint16(_LOBYTE(_WIN32_WINNT_WINXP)), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _IsWindowsVistaOrGreater() bool {
|
||||||
|
return isWindowsVersionOrGreaterWin32(uint16(_HIBYTE(_WIN32_WINNT_VISTA)), uint16(_LOBYTE(_WIN32_WINNT_VISTA)), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _IsWindows7OrGreater() bool {
|
||||||
|
return isWindowsVersionOrGreaterWin32(uint16(_HIBYTE(_WIN32_WINNT_WIN7)), uint16(_LOBYTE(_WIN32_WINNT_WIN7)), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _IsWindows8OrGreater() bool {
|
||||||
|
return isWindowsVersionOrGreaterWin32(uint16(_HIBYTE(_WIN32_WINNT_WIN8)), uint16(_LOBYTE(_WIN32_WINNT_WIN8)), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _IsWindows8Point1OrGreater() bool {
|
||||||
|
return isWindowsVersionOrGreaterWin32(uint16(_HIBYTE(_WIN32_WINNT_WINBLUE)), uint16(_LOBYTE(_WIN32_WINNT_WINBLUE)), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func isWindows10AnniversaryUpdateOrGreaterWin32() bool {
|
||||||
|
return isWindows10BuildOrGreaterWin32(14393)
|
||||||
|
}
|
||||||
|
|
||||||
|
func isWindows10CreatorsUpdateOrGreaterWin32() bool {
|
||||||
|
return isWindows10BuildOrGreaterWin32(15063)
|
||||||
|
}
|
65
internal/glfwwin/win32thread_windows.go
Normal file
65
internal/glfwwin/win32thread_windows.go
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
func (t *tls) create() error {
|
||||||
|
if t.win32.allocated {
|
||||||
|
panic("glfwwin: TLS must not be allocated")
|
||||||
|
}
|
||||||
|
|
||||||
|
i, err := _TlsAlloc()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
t.win32.index = i
|
||||||
|
t.win32.allocated = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *tls) destroy() error {
|
||||||
|
if t.win32.allocated {
|
||||||
|
if err := _TlsFree(t.win32.index); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
t.win32.allocated = false
|
||||||
|
t.win32.index = 0
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *tls) get() (uintptr, error) {
|
||||||
|
if !t.win32.allocated {
|
||||||
|
panic("glfwwin: TLS must be allocated")
|
||||||
|
}
|
||||||
|
|
||||||
|
return _TlsGetValue(t.win32.index)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *tls) set(value uintptr) error {
|
||||||
|
if !t.win32.allocated {
|
||||||
|
panic("glfwwin: TLS must be allocated")
|
||||||
|
}
|
||||||
|
|
||||||
|
return _TlsSetValue(t.win32.index, value)
|
||||||
|
}
|
2115
internal/glfwwin/win32window_windows.go
Normal file
2115
internal/glfwwin/win32window_windows.go
Normal file
File diff suppressed because it is too large
Load Diff
880
internal/glfwwin/window_windows.go
Normal file
880
internal/glfwwin/window_windows.go
Normal file
@ -0,0 +1,880 @@
|
|||||||
|
// Copyright 2002-2006 Marcus Geelnard
|
||||||
|
// Copyright 2006-2019 Camilla Löwy
|
||||||
|
// Copyright 2022 The Ebiten Authors
|
||||||
|
//
|
||||||
|
// This software is provided 'as-is', without any express or implied
|
||||||
|
// warranty. In no event will the authors be held liable for any damages
|
||||||
|
// arising from the use of this software.
|
||||||
|
//
|
||||||
|
// Permission is granted to anyone to use this software for any purpose,
|
||||||
|
// including commercial applications, and to alter it and redistribute it
|
||||||
|
// freely, subject to the following restrictions:
|
||||||
|
//
|
||||||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||||||
|
// claim that you wrote the original software. If you use this software
|
||||||
|
// in a product, an acknowledgment in the product documentation would
|
||||||
|
// be appreciated but is not required.
|
||||||
|
//
|
||||||
|
// 2. Altered source versions must be plainly marked as such, and must not
|
||||||
|
// be misrepresented as being the original software.
|
||||||
|
//
|
||||||
|
// 3. This notice may not be removed or altered from any source
|
||||||
|
// distribution.
|
||||||
|
|
||||||
|
package glfwwin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (w *Window) inputWindowFocus(focused bool) {
|
||||||
|
if w.callbacks.focus != nil {
|
||||||
|
w.callbacks.focus(w, focused)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !focused {
|
||||||
|
for key := Key(0); key <= KeyLast; key++ {
|
||||||
|
if w.keys[key] == Press {
|
||||||
|
scancode := platformGetKeyScancode(key)
|
||||||
|
w.inputKey(key, scancode, Release, 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for button := MouseButton(0); button <= MouseButtonLast; button++ {
|
||||||
|
if w.mouseButtons[button] == Press {
|
||||||
|
w.inputMouseClick(button, Release, 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowPos(x, y int) {
|
||||||
|
if w.callbacks.pos != nil {
|
||||||
|
w.callbacks.pos(w, x, y)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowSize(width, height int) {
|
||||||
|
if w.callbacks.size != nil {
|
||||||
|
w.callbacks.size(w, width, height)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowIconify(iconified bool) {
|
||||||
|
if w.callbacks.iconify != nil {
|
||||||
|
w.callbacks.iconify(w, iconified)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowMaximize(maximized bool) {
|
||||||
|
if w.callbacks.maximize != nil {
|
||||||
|
w.callbacks.maximize(w, maximized)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputFramebufferSize(width, height int) {
|
||||||
|
if w.callbacks.fbsize != nil {
|
||||||
|
w.callbacks.fbsize(w, width, height)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowContentScale(xscale, yscale float32) {
|
||||||
|
if w.callbacks.scale != nil {
|
||||||
|
w.callbacks.scale(w, xscale, yscale)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowDamage() {
|
||||||
|
if w.callbacks.refresh != nil {
|
||||||
|
w.callbacks.refresh(w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowCloseRequest() {
|
||||||
|
w.shouldClose = true
|
||||||
|
|
||||||
|
if w.callbacks.close != nil {
|
||||||
|
w.callbacks.close(w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) inputWindowMonitor(monitor *Monitor) {
|
||||||
|
w.monitor = monitor
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateWindow(width, height int, title string, monitor *Monitor, share *Window) (window *Window, ferr error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if width <= 0 || height <= 0 {
|
||||||
|
return nil, fmt.Errorf("glfwwin: invalid window size %dx%d: %w", width, height, InvalidValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
fbconfig := _glfw.hints.framebuffer
|
||||||
|
ctxconfig := _glfw.hints.context
|
||||||
|
wndconfig := _glfw.hints.window
|
||||||
|
|
||||||
|
wndconfig.width = width
|
||||||
|
wndconfig.height = height
|
||||||
|
wndconfig.title = title
|
||||||
|
ctxconfig.share = share
|
||||||
|
|
||||||
|
if err := checkValidContextConfig(&ctxconfig); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
window = &Window{
|
||||||
|
videoMode: VidMode{
|
||||||
|
Width: width,
|
||||||
|
Height: height,
|
||||||
|
RedBits: fbconfig.redBits,
|
||||||
|
GreenBits: fbconfig.greenBits,
|
||||||
|
BlueBits: fbconfig.blueBits,
|
||||||
|
RefreshRate: _glfw.hints.refreshRate,
|
||||||
|
},
|
||||||
|
|
||||||
|
monitor: monitor,
|
||||||
|
resizable: wndconfig.resizable,
|
||||||
|
decorated: wndconfig.decorated,
|
||||||
|
autoIconify: wndconfig.autoIconify,
|
||||||
|
floating: wndconfig.floating,
|
||||||
|
focusOnShow: wndconfig.focusOnShow,
|
||||||
|
cursorMode: CursorNormal,
|
||||||
|
|
||||||
|
doublebuffer: fbconfig.doublebuffer,
|
||||||
|
|
||||||
|
minwidth: DontCare,
|
||||||
|
minheight: DontCare,
|
||||||
|
maxwidth: DontCare,
|
||||||
|
maxheight: DontCare,
|
||||||
|
numer: DontCare,
|
||||||
|
denom: DontCare,
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
if ferr != nil {
|
||||||
|
window.Destroy()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Open the actual window and create its context
|
||||||
|
if err := window.platformCreateWindow(&wndconfig, &ctxconfig, &fbconfig); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctxconfig.client != NoAPI {
|
||||||
|
if err := window.refreshContextAttribs(&ctxconfig); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if window.monitor != nil {
|
||||||
|
if wndconfig.centerCursor {
|
||||||
|
if err := window.centerCursorInContentArea(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if wndconfig.visible {
|
||||||
|
window.platformShowWindow()
|
||||||
|
if wndconfig.focused {
|
||||||
|
if err := window.platformFocusWindow(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return window, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func defaultWindowHints() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
// The default is OpenGL with minimum version 1.0
|
||||||
|
_glfw.hints.context = ctxconfig{
|
||||||
|
client: OpenGLAPI,
|
||||||
|
source: NativeContextAPI,
|
||||||
|
major: 1,
|
||||||
|
minor: 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
// The default is a focused, visible, resizable window with decorations
|
||||||
|
_glfw.hints.window = wndconfig{
|
||||||
|
resizable: true,
|
||||||
|
visible: true,
|
||||||
|
decorated: true,
|
||||||
|
focused: true,
|
||||||
|
autoIconify: true,
|
||||||
|
centerCursor: true,
|
||||||
|
focusOnShow: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
// The default is 24 bits of color, 24 bits of depth and 8 bits of stencil,
|
||||||
|
// double buffered
|
||||||
|
_glfw.hints.framebuffer = fbconfig{
|
||||||
|
redBits: 8,
|
||||||
|
greenBits: 8,
|
||||||
|
blueBits: 8,
|
||||||
|
alphaBits: 8,
|
||||||
|
depthBits: 24,
|
||||||
|
stencilBits: 8,
|
||||||
|
doublebuffer: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
// The default is to select the highest available refresh rate
|
||||||
|
_glfw.hints.refreshRate = DontCare
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func WindowHint(hint Hint, value int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
switch hint {
|
||||||
|
case RedBits:
|
||||||
|
_glfw.hints.framebuffer.redBits = value
|
||||||
|
case GreenBits:
|
||||||
|
_glfw.hints.framebuffer.greenBits = value
|
||||||
|
case BlueBits:
|
||||||
|
_glfw.hints.framebuffer.blueBits = value
|
||||||
|
case AlphaBits:
|
||||||
|
_glfw.hints.framebuffer.alphaBits = value
|
||||||
|
case DepthBits:
|
||||||
|
_glfw.hints.framebuffer.depthBits = value
|
||||||
|
case StencilBits:
|
||||||
|
_glfw.hints.framebuffer.stencilBits = value
|
||||||
|
case AccumRedBits:
|
||||||
|
_glfw.hints.framebuffer.accumRedBits = value
|
||||||
|
case AccumGreenBits:
|
||||||
|
_glfw.hints.framebuffer.accumGreenBits = value
|
||||||
|
case AccumBlueBits:
|
||||||
|
_glfw.hints.framebuffer.accumBlueBits = value
|
||||||
|
case AccumAlphaBits:
|
||||||
|
_glfw.hints.framebuffer.accumAlphaBits = value
|
||||||
|
case AuxBuffers:
|
||||||
|
_glfw.hints.framebuffer.auxBuffers = value
|
||||||
|
case Stereo:
|
||||||
|
_glfw.hints.framebuffer.stereo = intToBool(value)
|
||||||
|
case DoubleBuffer:
|
||||||
|
_glfw.hints.framebuffer.doublebuffer = intToBool(value)
|
||||||
|
case TransparentFramebuffer:
|
||||||
|
_glfw.hints.framebuffer.transparent = intToBool(value)
|
||||||
|
case Samples:
|
||||||
|
_glfw.hints.framebuffer.samples = value
|
||||||
|
case SRGBCapable:
|
||||||
|
_glfw.hints.framebuffer.sRGB = intToBool(value)
|
||||||
|
case Resizable:
|
||||||
|
_glfw.hints.window.resizable = intToBool(value)
|
||||||
|
case Decorated:
|
||||||
|
_glfw.hints.window.decorated = intToBool(value)
|
||||||
|
case Focused:
|
||||||
|
_glfw.hints.window.focused = intToBool(value)
|
||||||
|
case AutoIconify:
|
||||||
|
_glfw.hints.window.autoIconify = intToBool(value)
|
||||||
|
case Floating:
|
||||||
|
_glfw.hints.window.floating = intToBool(value)
|
||||||
|
case Maximized:
|
||||||
|
_glfw.hints.window.maximized = intToBool(value)
|
||||||
|
case Visible:
|
||||||
|
_glfw.hints.window.visible = intToBool(value)
|
||||||
|
case ScaleToMonitor:
|
||||||
|
_glfw.hints.window.scaleToMonitor = intToBool(value)
|
||||||
|
case CenterCursor:
|
||||||
|
_glfw.hints.window.centerCursor = intToBool(value)
|
||||||
|
case FocusOnShow:
|
||||||
|
_glfw.hints.window.focusOnShow = intToBool(value)
|
||||||
|
case ClientAPI:
|
||||||
|
_glfw.hints.context.client = value
|
||||||
|
case ContextCreationAPI:
|
||||||
|
_glfw.hints.context.source = value
|
||||||
|
case ContextVersionMajor:
|
||||||
|
_glfw.hints.context.major = value
|
||||||
|
case ContextVersionMinor:
|
||||||
|
_glfw.hints.context.minor = value
|
||||||
|
case ContextRobustness:
|
||||||
|
_glfw.hints.context.robustness = value
|
||||||
|
case OpenGLForwardCompat:
|
||||||
|
_glfw.hints.context.forward = intToBool(value)
|
||||||
|
case OpenGLDebugContext:
|
||||||
|
_glfw.hints.context.debug = intToBool(value)
|
||||||
|
case ContextNoError:
|
||||||
|
_glfw.hints.context.noerror = intToBool(value)
|
||||||
|
case OpenGLProfile:
|
||||||
|
_glfw.hints.context.profile = value
|
||||||
|
case ContextReleaseBehavior:
|
||||||
|
_glfw.hints.context.release = value
|
||||||
|
case RefreshRate:
|
||||||
|
_glfw.hints.refreshRate = value
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("glfwwin: invalid window hint 0x%08X: %w", hint, InvalidEnum)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func WindowHintString(hint Hint, value string) {
|
||||||
|
panic("glfwwin: WindowHintString is not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Destroy() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allow closing of NULL (to match the behavior of free)
|
||||||
|
if w == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clear all callbacks to avoid exposing a half torn-down w object
|
||||||
|
//memset(&w.callbacks, 0, sizeof(w.callbacks))
|
||||||
|
|
||||||
|
// The w's context must not be current on another thread when the
|
||||||
|
// w is destroyed
|
||||||
|
current, err := _glfw.contextSlot.get()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if uintptr(unsafe.Pointer(w)) == current {
|
||||||
|
if err := (*Window)(nil).MakeContextCurrent(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w.platformDestroyWindow()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) ShouldClose() (bool, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return false, NotInitialized
|
||||||
|
}
|
||||||
|
return w.shouldClose, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetShouldClose(value bool) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
w.shouldClose = value
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetTitle(title string) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if err := w.platformSetWindowTitle(title); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetIcon(images []*Image) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if err := w.platformSetWindowIcon(images); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetPos() (xpos, ypos int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
return w.platformGetWindowPos()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetPos(xpos, ypos int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if w.monitor != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if err := w.platformSetWindowPos(xpos, ypos); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetSize() (width, height int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
return w.platformGetWindowSize()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetSize(width, height int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
w.videoMode.Width = width
|
||||||
|
w.videoMode.Height = height
|
||||||
|
if err := w.platformSetWindowSize(width, height); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetSizeLimits(minwidth, minheight, maxwidth, maxheight int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if minwidth != DontCare && minheight != DontCare {
|
||||||
|
if minwidth < 0 || minheight < 0 {
|
||||||
|
return fmt.Errorf("glfwwin: invalid window minimum size %dx%d: %w", minwidth, minheight, InvalidValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if maxwidth != DontCare && maxheight != DontCare {
|
||||||
|
if maxwidth < 0 || maxheight < 0 || maxwidth < minwidth || maxheight < minheight {
|
||||||
|
return fmt.Errorf("glfwwin: invalid window maximum size %dx%d: %w", maxwidth, maxheight, InvalidValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w.minwidth = minwidth
|
||||||
|
w.minheight = minheight
|
||||||
|
w.maxwidth = maxwidth
|
||||||
|
w.maxheight = maxheight
|
||||||
|
|
||||||
|
if w.monitor != nil || !w.resizable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := w.platformSetWindowSizeLimits(minwidth, minheight, maxwidth, maxheight); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetAspectRatio(numer, denom int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if numer != DontCare && denom != DontCare {
|
||||||
|
if numer <= 0 || denom <= 0 {
|
||||||
|
return fmt.Errorf("glfwwin: invalid window aspect ratio %d:%d: %w", numer, denom, InvalidValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w.numer = numer
|
||||||
|
w.denom = denom
|
||||||
|
|
||||||
|
if w.monitor != nil || !w.resizable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := w.platformSetWindowAspectRatio(numer, denom); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetFramebufferSize() (width, height int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
return w.platformGetFramebufferSize()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetFrameSize() (left, top, right, bottom int, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
return w.platformGetWindowFrameSize()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetContentScale() (xscale, yscale float32, err error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, 0, NotInitialized
|
||||||
|
}
|
||||||
|
return w.platformGetWindowContentScale()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetOpacity() (float32, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, NotInitialized
|
||||||
|
}
|
||||||
|
return w.platformGetWindowOpacity()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetOpacity(opacity float32) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if opacity != opacity || opacity < 0 || opacity > 1 {
|
||||||
|
return fmt.Errorf("glfwwin: invalid window opacity %f: %w", opacity, InvalidValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := w.platformSetWindowOpacity(opacity); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Iconify() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
w.platformIconifyWindow()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Restore() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
w.platformRestoreWindow()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Maximize() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if w.monitor != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
w.platformMaximizeWindow()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Show() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if w.monitor != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
w.platformShowWindow()
|
||||||
|
|
||||||
|
if w.focusOnShow {
|
||||||
|
if err := w.platformFocusWindow(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) RequestAttention() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
w.platformRequestWindowAttention()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Hide() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if w.monitor != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
w.platformHideWindow()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) Focus() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if err := w.platformFocusWindow(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetAttrib(attrib Hint) (int, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return 0, NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
switch attrib {
|
||||||
|
case Focused:
|
||||||
|
return boolToInt(w.platformWindowFocused()), nil
|
||||||
|
case Iconified:
|
||||||
|
return boolToInt(w.platformWindowIconified()), nil
|
||||||
|
case Visible:
|
||||||
|
return boolToInt(w.platformWindowVisible()), nil
|
||||||
|
case Maximized:
|
||||||
|
return boolToInt(w.platformWindowMaximized()), nil
|
||||||
|
case Hovered:
|
||||||
|
b, err := w.platformWindowHovered()
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return boolToInt(b), nil
|
||||||
|
case FocusOnShow:
|
||||||
|
return boolToInt(w.focusOnShow), nil
|
||||||
|
case TransparentFramebuffer:
|
||||||
|
return boolToInt(w.platformFramebufferTransparent()), nil
|
||||||
|
case Resizable:
|
||||||
|
return boolToInt(w.resizable), nil
|
||||||
|
case Decorated:
|
||||||
|
return boolToInt(w.decorated), nil
|
||||||
|
case Floating:
|
||||||
|
return boolToInt(w.floating), nil
|
||||||
|
case AutoIconify:
|
||||||
|
return boolToInt(w.autoIconify), nil
|
||||||
|
case ClientAPI:
|
||||||
|
return w.context.client, nil
|
||||||
|
case ContextCreationAPI:
|
||||||
|
return w.context.source, nil
|
||||||
|
case ContextVersionMajor:
|
||||||
|
return w.context.major, nil
|
||||||
|
case ContextVersionMinor:
|
||||||
|
return w.context.minor, nil
|
||||||
|
case ContextRevision:
|
||||||
|
return w.context.revision, nil
|
||||||
|
case ContextRobustness:
|
||||||
|
return w.context.robustness, nil
|
||||||
|
case OpenGLForwardCompat:
|
||||||
|
return boolToInt(w.context.forward), nil
|
||||||
|
case OpenGLDebugContext:
|
||||||
|
return boolToInt(w.context.debug), nil
|
||||||
|
case OpenGLProfile:
|
||||||
|
return w.context.profile, nil
|
||||||
|
case ContextReleaseBehavior:
|
||||||
|
return w.context.release, nil
|
||||||
|
case ContextNoError:
|
||||||
|
return boolToInt(w.context.noerror), nil
|
||||||
|
default:
|
||||||
|
return 0, fmt.Errorf("glfwwin: invalid window attribute 0x%08X: %w", attrib, InvalidEnum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetAttrib(attrib Hint, value int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
bValue := intToBool(value)
|
||||||
|
|
||||||
|
switch attrib {
|
||||||
|
case AutoIconify:
|
||||||
|
w.autoIconify = bValue
|
||||||
|
return nil
|
||||||
|
case Resizable:
|
||||||
|
if w.resizable == bValue {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
w.resizable = bValue
|
||||||
|
if w.monitor == nil {
|
||||||
|
if err := w.platformSetWindowResizable(bValue); err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
case Decorated:
|
||||||
|
if w.decorated == bValue {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
w.decorated = bValue
|
||||||
|
if w.monitor == nil {
|
||||||
|
if err := w.platformSetWindowDecorated(bValue); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
case Floating:
|
||||||
|
if w.floating == bValue {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
w.floating = bValue
|
||||||
|
if w.monitor == nil {
|
||||||
|
if err := w.platformSetWindowFloating(bValue); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
case FocusOnShow:
|
||||||
|
w.focusOnShow = bValue
|
||||||
|
return nil
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("glfwwin: invalid window attribute 0x%08X: %w", attrib, InvalidEnum)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetMonitor() (*Monitor, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
return w.monitor, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetMonitor(monitor *Monitor, xpos, ypos, width, height, refreshRate int) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
|
||||||
|
if width <= 0 || height <= 0 {
|
||||||
|
return fmt.Errorf("glfwwin: invalid window size %dx%d: %w", width, height, InvalidValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
if refreshRate < 0 && refreshRate != DontCare {
|
||||||
|
return fmt.Errorf("glfwwin: invalid refresh rate %d: %w", refreshRate, InvalidValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
w.videoMode.Width = width
|
||||||
|
w.videoMode.Height = height
|
||||||
|
w.videoMode.RefreshRate = refreshRate
|
||||||
|
|
||||||
|
if err := w.platformSetWindowMonitor(monitor, xpos, ypos, width, height, refreshRate); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetUserPointer(pointer unsafe.Pointer) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
w.userPointer = pointer
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) GetUserPointer() (unsafe.Pointer, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
return w.userPointer, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetPosCallback(cbfun PosCallback) (PosCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.pos
|
||||||
|
w.callbacks.pos = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetSizeCallback(cbfun SizeCallback) (SizeCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.size
|
||||||
|
w.callbacks.size = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetCloseCallback(cbfun CloseCallback) (CloseCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.close
|
||||||
|
w.callbacks.close = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetRefreshCallback(cbfun RefreshCallback) (RefreshCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.refresh
|
||||||
|
w.callbacks.refresh = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetFocusCallback(cbfun FocusCallback) (FocusCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.focus
|
||||||
|
w.callbacks.focus = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetIconifyCallback(cbfun IconifyCallback) (IconifyCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.iconify
|
||||||
|
w.callbacks.iconify = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetMaximizeCallback(cbfun MaximizeCallback) (MaximizeCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.maximize
|
||||||
|
w.callbacks.maximize = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetFramebufferSizeCallback(cbfun FramebufferSizeCallback) (FramebufferSizeCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.fbsize
|
||||||
|
w.callbacks.fbsize = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Window) SetContentScaleCallback(cbfun ContentScaleCallback) (ContentScaleCallback, error) {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return nil, NotInitialized
|
||||||
|
}
|
||||||
|
old := w.callbacks.scale
|
||||||
|
w.callbacks.scale = cbfun
|
||||||
|
return old, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func PollEvents() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if err := platformPollEvents(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func WaitEvents() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if err := platformWaitEvents(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func WaitEventsTimeout(timeout float64) error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if timeout != timeout || timeout < 0.0 || timeout > math.MaxFloat64 {
|
||||||
|
return fmt.Errorf("glfwwin: invalid time %f: %w", timeout, InvalidValue)
|
||||||
|
}
|
||||||
|
if err := platformWaitEventsTimeout(timeout); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func PostEmptyEvent() error {
|
||||||
|
if !_glfw.initialized {
|
||||||
|
return NotInitialized
|
||||||
|
}
|
||||||
|
if err := platformPostEmptyEvent(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user