// Copyright 2018 The Ebiten Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // +build !js package devicescale import ( "fmt" "syscall" "unsafe" ) const ( logPixelsX = 88 monitorDefaultToNearest = 2 mdtEffectiveDpi = 0 ) var ( user32 = syscall.NewLazyDLL("user32") gdi32 = syscall.NewLazyDLL("gdi32") shcore = syscall.NewLazyDLL("shcore") ) var ( procSetProcessDPIAware = user32.NewProc("SetProcessDPIAware") procGetActiveWindow = user32.NewProc("GetActiveWindow") procGetWindowDC = user32.NewProc("GetWindowDC") procReleaseDC = user32.NewProc("ReleaseDC") procMonitorFromWindow = user32.NewProc("MonitorFromWindow") procGetMonitorInfo = user32.NewProc("GetMonitorInfoW") procGetDeviceCaps = gdi32.NewProc("GetDeviceCaps") // GetScaleFactorForMonitor function can return unrelaiavle value (e.g. returning 180 // for 200% scale). Use GetDpiForMonitor instead. procGetDpiForMonitor = shcore.NewProc("GetDpiForMonitor") ) var shcoreAvailable = false func init() { if shcore.Load() == nil { shcoreAvailable = true } } func setProcessDPIAware() error { r, _, e := syscall.Syscall(procSetProcessDPIAware.Addr(), 0, 0, 0, 0) if e != 0 { return fmt.Errorf("devicescale: SetProcessDPIAware failed: error code: %d", e) } if r == 0 { return fmt.Errorf("devicescale: SetProcessDPIAware failed: returned value: %d", r) } return nil } func getActiveWindow() (uintptr, error) { r, _, e := syscall.Syscall(procGetActiveWindow.Addr(), 0, 0, 0, 0) if e != 0 { return 0, fmt.Errorf("devicescale: GetActiveWindow failed: error code: %d", e) } return r, nil } func getWindowDC(hwnd uintptr) (uintptr, error) { r, _, e := syscall.Syscall(procGetWindowDC.Addr(), 1, hwnd, 0, 0) if e != 0 { return 0, fmt.Errorf("devicescale: GetWindowDC failed: error code: %d", e) } if r == 0 { return 0, fmt.Errorf("devicescale: GetWindowDC failed: returned value: %d", r) } return r, nil } func releaseDC(hwnd, hdc uintptr) error { r, _, e := syscall.Syscall(procReleaseDC.Addr(), 2, hwnd, hdc, 0) if e != 0 { return fmt.Errorf("devicescale: ReleaseDC failed: error code: %d", e) } if r == 0 { return fmt.Errorf("devicescale: ReleaseDC failed: returned value: %d", r) } return nil } func getDeviceCaps(hdc uintptr, nindex int) (int, error) { r, _, e := syscall.Syscall(procGetDeviceCaps.Addr(), 2, hdc, uintptr(nindex), 0) if e != 0 { return 0, fmt.Errorf("devicescale: GetDeviceCaps failed: error code: %d", e) } return int(r), nil } func monitorFromWindow(hwnd uintptr, dwFlags int) (uintptr, error) { r, _, e := syscall.Syscall(procMonitorFromWindow.Addr(), 2, hwnd, uintptr(dwFlags), 0) if e != 0 { return 0, fmt.Errorf("devicescale: MonitorFromWindow failed: error code: %d", e) } if r == 0 { return 0, fmt.Errorf("devicescale: MonitorFromWindow failed: returned value: %d", r) } return r, nil } func getMonitorInfo(hMonitor uintptr, lpMonitorInfo uintptr) error { r, _, e := syscall.Syscall(procGetMonitorInfo.Addr(), 2, hMonitor, lpMonitorInfo, 0) if e != 0 { return fmt.Errorf("devicescale: GetMonitorInfo failed: error code: %d", e) } if r == 0 { return fmt.Errorf("devicescale: GetMonitorInfo failed: returned value: %d", r) } return nil } func getDpiForMonitor(hMonitor uintptr, dpiType uintptr, dpiX, dpiY uintptr) error { r, _, e := syscall.Syscall6(procGetDpiForMonitor.Addr(), 4, hMonitor, dpiType, dpiX, dpiY, 0, 0) if e != 0 { return fmt.Errorf("devicescale: GetDpiForMonitor failed: error code: %d", e) } if r != 0 { return fmt.Errorf("devicescale: GetDpiForMonitor failed: returned value: %d", r) } return nil } func getFromLogPixelSx() float64 { dc, err := getWindowDC(0) if err != nil { panic(err) } // Note that GetDeviceCaps with LOGPIXELSX always returns a same value for any monitors // even if multiple monitors are used. dpi, err := getDeviceCaps(dc, logPixelsX) if err != nil { panic(err) } if err := releaseDC(0, dc); err != nil { panic(err) } return float64(dpi) / 96 } func impl() float64 { if err := setProcessDPIAware(); err != nil { panic(err) } // On Windows 7 or older, shcore.dll is not available. if !shcoreAvailable { return getFromLogPixelSx() } w, err := getActiveWindow() if err != nil { panic(err) } // The window is not initialized yet when w == 0. if w == 0 { // TODO: Use the primary monitor instead. return getFromLogPixelSx() } m, err := monitorFromWindow(w, monitorDefaultToNearest) if err != nil { panic(err) } dpiX := uint32(0) dpiY := uint32(0) // Passing dpiY is needed even though this is not used. if err := getDpiForMonitor(m, mdtEffectiveDpi, uintptr(unsafe.Pointer(&dpiX)), uintptr(unsafe.Pointer(&dpiY))); err != nil { panic(err) } return float64(dpiX) / 96 }