Adopt windows-sys (#2057)

This commit is contained in:
Clemens Wasser 2022-03-07 22:58:12 +01:00 committed by GitHub
parent 78e5a395da
commit b222dde835
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
17 changed files with 1424 additions and 1239 deletions

View file

@ -21,6 +21,8 @@ And please only add new entries to the top of this list, right below the `# Unre
- **Breaking:** Replaced `EventLoopExtMacOS` with `EventLoopBuilderExtMacOS` (which also has renamed methods). - **Breaking:** Replaced `EventLoopExtMacOS` with `EventLoopBuilderExtMacOS` (which also has renamed methods).
- **Breaking:** Replaced `EventLoopExtWindows` with `EventLoopBuilderExtWindows` (which also has renamed methods). - **Breaking:** Replaced `EventLoopExtWindows` with `EventLoopBuilderExtWindows` (which also has renamed methods).
- **Breaking:** Replaced `EventLoopExtUnix` with `EventLoopBuilderExtUnix` (which also has renamed methods). - **Breaking:** Replaced `EventLoopExtUnix` with `EventLoopBuilderExtUnix` (which also has renamed methods).
- **Breaking:** The platform specific extensions for Windows `winit::platform::windows` have changed. All `HANDLE`-like types e.g. `HWND` and `HMENU` were converted from winapi types or `*mut c_void` to `isize`. This was done to be consistent with the type definitions in windows-sys and to not expose internal dependencies.
- The internal bindings to the [Windows API](https://docs.microsoft.com/en-us/windows/) were changed from the unofficial [winapi](https://github.com/retep998/winapi-rs) bindings to the official Microsoft [windows-sys](https://github.com/microsoft/windows-rs) bindings.
- On Wayland, fix resize and scale factor changes not being propagated properly. - On Wayland, fix resize and scale factor changes not being propagated properly.
# 0.26.1 (2022-01-05) # 0.26.1 (2022-01-05)

View file

@ -55,33 +55,35 @@ default_features = false
features = ["display_link"] features = ["display_link"]
[target.'cfg(target_os = "windows")'.dependencies] [target.'cfg(target_os = "windows")'.dependencies]
parking_lot = "0.11" parking_lot = "0.12"
[target.'cfg(target_os = "windows")'.dependencies.winapi] [target.'cfg(target_os = "windows")'.dependencies.windows-sys]
version = "0.3.9" version = "0.33"
features = [ features = [
"combaseapi", "Win32_Devices_HumanInterfaceDevice",
"commctrl", "Win32_Foundation",
"dwmapi", "Win32_Globalization",
"errhandlingapi", "Win32_Graphics_Dwm",
"imm", "Win32_Graphics_Gdi",
"hidusage", "Win32_Media",
"libloaderapi", "Win32_System_Com_StructuredStorage",
"objbase", "Win32_System_Com",
"ole2", "Win32_System_LibraryLoader",
"processthreadsapi", "Win32_System_Ole",
"shellapi", "Win32_System_SystemInformation",
"shellscalingapi", "Win32_System_SystemServices",
"shobjidl_core", "Win32_System_Threading",
"unknwnbase", "Win32_System_WindowsProgramming",
"winbase", "Win32_UI_Accessibility",
"windowsx", "Win32_UI_Controls",
"winerror", "Win32_UI_HiDpi",
"wingdi", "Win32_UI_Input_Ime",
"winnt", "Win32_UI_Input_KeyboardAndMouse",
"winuser", "Win32_UI_Input_Pointer",
"mmsystem", "Win32_UI_Input_Touch",
"timeapi" "Win32_UI_Shell",
"Win32_UI_TextServices",
"Win32_UI_WindowsAndMessaging",
] ]
[target.'cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "openbsd", target_os = "netbsd"))'.dependencies] [target.'cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "openbsd", target_os = "netbsd"))'.dependencies]
@ -91,7 +93,7 @@ sctk = { package = "smithay-client-toolkit", version = "0.15.1", default_feature
mio = { version = "0.8", features = ["os-ext"], optional = true } mio = { version = "0.8", features = ["os-ext"], optional = true }
x11-dl = { version = "2.18.5", optional = true } x11-dl = { version = "2.18.5", optional = true }
percent-encoding = { version = "2.0", optional = true } percent-encoding = { version = "2.0", optional = true }
parking_lot = { version = "0.11.0", optional = true } parking_lot = { version = "0.12.0", optional = true }
libc = "0.2.64" libc = "0.2.64"
[target.'cfg(target_arch = "wasm32")'.dependencies.web_sys] [target.'cfg(target_arch = "wasm32")'.dependencies.web_sys]

View file

@ -1,11 +1,7 @@
#![cfg(target_os = "windows")] #![cfg(target_os = "windows")]
use std::os::raw::c_void;
use std::path::Path; use std::path::Path;
use winapi::shared::minwindef::WORD;
use winapi::shared::windef::{HMENU, HWND};
use crate::{ use crate::{
dpi::PhysicalSize, dpi::PhysicalSize,
event::DeviceId, event::DeviceId,
@ -15,6 +11,15 @@ use crate::{
window::{BadIcon, Icon, Theme, Window, WindowBuilder}, window::{BadIcon, Icon, Theme, Window, WindowBuilder},
}; };
/// Window Handle type used by Win32 API
pub type HWND = isize;
/// Menu Handle type used by Win32 API
pub type HMENU = isize;
/// Monitor Handle type used by Win32 API
pub type HMONITOR = isize;
/// Instance Handle type used by Win32 API
pub type HINSTANCE = isize;
/// Additional methods on `EventLoop` that are specific to Windows. /// Additional methods on `EventLoop` that are specific to Windows.
pub trait EventLoopBuilderExtWindows { pub trait EventLoopBuilderExtWindows {
/// Whether to allow the event loop to be created off of the main thread. /// Whether to allow the event loop to be created off of the main thread.
@ -70,11 +75,11 @@ impl<T> EventLoopBuilderExtWindows for EventLoopBuilder<T> {
/// Additional methods on `Window` that are specific to Windows. /// Additional methods on `Window` that are specific to Windows.
pub trait WindowExtWindows { pub trait WindowExtWindows {
/// Returns the HINSTANCE of the window /// Returns the HINSTANCE of the window
fn hinstance(&self) -> *mut c_void; fn hinstance(&self) -> HINSTANCE;
/// Returns the native handle that is used by this window. /// Returns the native handle that is used by this window.
/// ///
/// The pointer will become invalid when the native window was destroyed. /// The pointer will become invalid when the native window was destroyed.
fn hwnd(&self) -> *mut c_void; fn hwnd(&self) -> HWND;
/// Enables or disables mouse and keyboard input to the specified window. /// Enables or disables mouse and keyboard input to the specified window.
/// ///
@ -100,13 +105,13 @@ pub trait WindowExtWindows {
impl WindowExtWindows for Window { impl WindowExtWindows for Window {
#[inline] #[inline]
fn hinstance(&self) -> *mut c_void { fn hinstance(&self) -> HINSTANCE {
self.window.hinstance() as *mut _ self.window.hinstance()
} }
#[inline] #[inline]
fn hwnd(&self) -> *mut c_void { fn hwnd(&self) -> HWND {
self.window.hwnd() as *mut _ self.window.hwnd()
} }
#[inline] #[inline]
@ -150,10 +155,12 @@ pub trait WindowBuilderExtWindows {
/// ///
/// Parent and menu are mutually exclusive; a child window cannot have a menu! /// Parent and menu are mutually exclusive; a child window cannot have a menu!
/// ///
/// The menu must have been manually created beforehand with [`winapi::um::winuser::CreateMenu`] or similar. /// The menu must have been manually created beforehand with [`CreateMenu`] or similar.
/// ///
/// Note: Dark mode cannot be supported for win32 menus, it's simply not possible to change how the menus look. /// Note: Dark mode cannot be supported for win32 menus, it's simply not possible to change how the menus look.
/// If you use this, it is recommended that you combine it with `with_theme(Some(Theme::Light))` to avoid a jarring effect. /// If you use this, it is recommended that you combine it with `with_theme(Some(Theme::Light))` to avoid a jarring effect.
///
/// [`CreateMenu`]: windows_sys::Win32::UI::WindowsAndMessaging::CreateMenu
fn with_menu(self, menu: HMENU) -> WindowBuilder; fn with_menu(self, menu: HMENU) -> WindowBuilder;
/// This sets `ICON_BIG`. A good ceiling here is 256x256. /// This sets `ICON_BIG`. A good ceiling here is 256x256.
@ -224,7 +231,7 @@ pub trait MonitorHandleExtWindows {
fn native_id(&self) -> String; fn native_id(&self) -> String;
/// Returns the handle of the monitor - `HMONITOR`. /// Returns the handle of the monitor - `HMONITOR`.
fn hmonitor(&self) -> *mut c_void; fn hmonitor(&self) -> HMONITOR;
} }
impl MonitorHandleExtWindows for MonitorHandle { impl MonitorHandleExtWindows for MonitorHandle {
@ -234,8 +241,8 @@ impl MonitorHandleExtWindows for MonitorHandle {
} }
#[inline] #[inline]
fn hmonitor(&self) -> *mut c_void { fn hmonitor(&self) -> HMONITOR {
self.inner.hmonitor() as *mut _ self.inner.hmonitor()
} }
} }
@ -273,7 +280,7 @@ pub trait IconExtWindows: Sized {
/// ///
/// In cases where the specified size does not exist in the file, Windows may perform scaling /// In cases where the specified size does not exist in the file, Windows may perform scaling
/// to get an icon of the desired size. /// to get an icon of the desired size.
fn from_resource(ordinal: WORD, size: Option<PhysicalSize<u32>>) -> Result<Self, BadIcon>; fn from_resource(ordinal: u16, size: Option<PhysicalSize<u32>>) -> Result<Self, BadIcon>;
} }
impl IconExtWindows for Icon { impl IconExtWindows for Icon {
@ -285,7 +292,7 @@ impl IconExtWindows for Icon {
Ok(Icon { inner: win_icon }) Ok(Icon { inner: win_icon })
} }
fn from_resource(ordinal: WORD, size: Option<PhysicalSize<u32>>) -> Result<Self, BadIcon> { fn from_resource(ordinal: u16, size: Option<PhysicalSize<u32>>) -> Result<Self, BadIcon> {
let win_icon = WinIcon::from_resource(ordinal, size)?; let win_icon = WinIcon::from_resource(ordinal, size)?;
Ok(Icon { inner: win_icon }) Ok(Icon { inner: win_icon })
} }

View file

@ -1,29 +1,35 @@
/// This is a simple implementation of support for Windows Dark Mode, /// This is a simple implementation of support for Windows Dark Mode,
/// which is inspired by the solution in https://github.com/ysc3839/win32-darkmode /// which is inspired by the solution in https://github.com/ysc3839/win32-darkmode
use std::ffi::OsStr; use std::{ffi::c_void, ptr};
use std::os::windows::ffi::OsStrExt;
use winapi::{ use windows_sys::{
shared::{ core::PCSTR,
basetsd::SIZE_T, Win32::{
minwindef::{BOOL, DWORD, FALSE, WORD}, Foundation::{BOOL, HWND, NTSTATUS, S_OK},
ntdef::{NTSTATUS, NT_SUCCESS, PVOID}, System::{
windef::HWND, LibraryLoader::{GetProcAddress, LoadLibraryA},
winerror::S_OK, SystemInformation::OSVERSIONINFOW,
},
UI::{
Accessibility::{HCF_HIGHCONTRASTON, HIGHCONTRASTA},
Controls::SetWindowTheme,
WindowsAndMessaging::{SystemParametersInfoA, SPI_GETHIGHCONTRAST},
},
}, },
um::{libloaderapi, uxtheme, winnt, winuser},
}; };
use crate::window::Theme; use crate::window::Theme;
use super::util;
lazy_static! { lazy_static! {
static ref WIN10_BUILD_VERSION: Option<DWORD> = { static ref WIN10_BUILD_VERSION: Option<u32> = {
type RtlGetVersion = unsafe extern "system" fn (*mut winnt::OSVERSIONINFOW) -> NTSTATUS; type RtlGetVersion = unsafe extern "system" fn (*mut OSVERSIONINFOW) -> NTSTATUS;
let handle = get_function!("ntdll.dll", RtlGetVersion); let handle = get_function!("ntdll.dll", RtlGetVersion);
if let Some(rtl_get_version) = handle { if let Some(rtl_get_version) = handle {
unsafe { unsafe {
let mut vi = winnt::OSVERSIONINFOW { let mut vi = OSVERSIONINFOW {
dwOSVersionInfoSize: 0, dwOSVersionInfoSize: 0,
dwMajorVersion: 0, dwMajorVersion: 0,
dwMinorVersion: 0, dwMinorVersion: 0,
@ -32,9 +38,9 @@ lazy_static! {
szCSDVersion: [0; 128], szCSDVersion: [0; 128],
}; };
let status = (rtl_get_version)(&mut vi as _); let status = (rtl_get_version)(&mut vi);
if NT_SUCCESS(status) && vi.dwMajorVersion == 10 && vi.dwMinorVersion == 0 { if status >= 0 && vi.dwMajorVersion == 10 && vi.dwMinorVersion == 0 {
Some(vi.dwBuildNumber) Some(vi.dwBuildNumber)
} else { } else {
None None
@ -54,8 +60,8 @@ lazy_static! {
} }
}; };
static ref DARK_THEME_NAME: Vec<u16> = widestring("DarkMode_Explorer"); static ref DARK_THEME_NAME: Vec<u16> = util::encode_wide("DarkMode_Explorer");
static ref LIGHT_THEME_NAME: Vec<u16> = widestring(""); static ref LIGHT_THEME_NAME: Vec<u16> = util::encode_wide("");
} }
/// Attempt to set a theme on a window, if necessary. /// Attempt to set a theme on a window, if necessary.
@ -77,7 +83,7 @@ pub fn try_theme(hwnd: HWND, preferred_theme: Option<Theme>) -> Theme {
Theme::Light => LIGHT_THEME_NAME.as_ptr(), Theme::Light => LIGHT_THEME_NAME.as_ptr(),
}; };
let status = unsafe { uxtheme::SetWindowTheme(hwnd, theme_name as _, std::ptr::null()) }; let status = unsafe { SetWindowTheme(hwnd, theme_name, ptr::null()) };
if status == S_OK && set_dark_mode_for_window(hwnd, is_dark_mode) { if status == S_OK && set_dark_mode_for_window(hwnd, is_dark_mode) {
return theme; return theme;
@ -103,8 +109,8 @@ fn set_dark_mode_for_window(hwnd: HWND, is_dark_mode: bool) -> bool {
#[repr(C)] #[repr(C)]
struct WINDOWCOMPOSITIONATTRIBDATA { struct WINDOWCOMPOSITIONATTRIBDATA {
Attrib: WINDOWCOMPOSITIONATTRIB, Attrib: WINDOWCOMPOSITIONATTRIB,
pvData: PVOID, pvData: *mut c_void,
cbData: SIZE_T, cbData: usize,
} }
lazy_static! { lazy_static! {
@ -115,7 +121,7 @@ fn set_dark_mode_for_window(hwnd: HWND, is_dark_mode: bool) -> bool {
if let Some(set_window_composition_attribute) = *SET_WINDOW_COMPOSITION_ATTRIBUTE { if let Some(set_window_composition_attribute) = *SET_WINDOW_COMPOSITION_ATTRIBUTE {
unsafe { unsafe {
// SetWindowCompositionAttribute needs a bigbool (i32), not bool. // SetWindowCompositionAttribute needs a bigbool (i32), not bool.
let mut is_dark_mode_bigbool = is_dark_mode as BOOL; let mut is_dark_mode_bigbool = BOOL::from(is_dark_mode);
let mut data = WINDOWCOMPOSITIONATTRIBDATA { let mut data = WINDOWCOMPOSITIONATTRIBDATA {
Attrib: WCA_USEDARKMODECOLORS, Attrib: WCA_USEDARKMODECOLORS,
@ -123,9 +129,9 @@ fn set_dark_mode_for_window(hwnd: HWND, is_dark_mode: bool) -> bool {
cbData: std::mem::size_of_val(&is_dark_mode_bigbool) as _, cbData: std::mem::size_of_val(&is_dark_mode_bigbool) as _,
}; };
let status = set_window_composition_attribute(hwnd, &mut data as *mut _); let status = set_window_composition_attribute(hwnd, &mut data);
status != FALSE status != false.into()
} }
} else { } else {
false false
@ -141,24 +147,17 @@ fn should_apps_use_dark_mode() -> bool {
lazy_static! { lazy_static! {
static ref SHOULD_APPS_USE_DARK_MODE: Option<ShouldAppsUseDarkMode> = { static ref SHOULD_APPS_USE_DARK_MODE: Option<ShouldAppsUseDarkMode> = {
unsafe { unsafe {
const UXTHEME_SHOULDAPPSUSEDARKMODE_ORDINAL: WORD = 132; const UXTHEME_SHOULDAPPSUSEDARKMODE_ORDINAL: PCSTR = 132 as PCSTR;
let module = libloaderapi::LoadLibraryA("uxtheme.dll\0".as_ptr() as _); let module = LoadLibraryA("uxtheme.dll\0".as_ptr());
if module.is_null() { if module == 0 {
return None; return None;
} }
let handle = libloaderapi::GetProcAddress( let handle = GetProcAddress(module, UXTHEME_SHOULDAPPSUSEDARKMODE_ORDINAL);
module,
winuser::MAKEINTRESOURCEA(UXTHEME_SHOULDAPPSUSEDARKMODE_ORDINAL),
);
if handle.is_null() { handle.map(|handle| std::mem::transmute(handle))
None
} else {
Some(std::mem::transmute(handle))
}
} }
}; };
} }
@ -169,27 +168,20 @@ fn should_apps_use_dark_mode() -> bool {
} }
fn is_high_contrast() -> bool { fn is_high_contrast() -> bool {
let mut hc = winuser::HIGHCONTRASTA { let mut hc = HIGHCONTRASTA {
cbSize: 0, cbSize: 0,
dwFlags: 0, dwFlags: 0,
lpszDefaultScheme: std::ptr::null_mut(), lpszDefaultScheme: ptr::null_mut(),
}; };
let ok = unsafe { let ok = unsafe {
winuser::SystemParametersInfoA( SystemParametersInfoA(
winuser::SPI_GETHIGHCONTRAST, SPI_GETHIGHCONTRAST,
std::mem::size_of_val(&hc) as _, std::mem::size_of_val(&hc) as _,
&mut hc as *mut _ as _, &mut hc as *mut _ as _,
0, 0,
) )
}; };
ok != FALSE && (winuser::HCF_HIGHCONTRASTON & hc.dwFlags) == 1 ok != false.into() && util::has_flag(hc.dwFlags, HCF_HIGHCONTRASTON)
}
fn widestring(src: &'static str) -> Vec<u16> {
OsStr::new(src)
.encode_wide()
.chain(Some(0).into_iter())
.collect()
} }

View file

@ -0,0 +1,140 @@
#![allow(non_snake_case)]
#![allow(non_upper_case_globals)]
use std::ffi::c_void;
use windows_sys::{
core::{IUnknown, GUID, HRESULT},
Win32::{
Foundation::{BOOL, HWND, POINTL},
System::Com::{
IAdviseSink, IDataObject, IEnumFORMATETC, IEnumSTATDATA, FORMATETC, STGMEDIUM,
},
UI::Shell::ITaskbarList,
},
};
#[repr(C)]
pub struct IUnknownVtbl {
pub QueryInterface: unsafe extern "system" fn(
This: *mut IUnknown,
riid: *const GUID,
ppvObject: *mut *mut c_void,
) -> HRESULT,
pub AddRef: unsafe extern "system" fn(This: *mut IUnknown) -> u32,
pub Release: unsafe extern "system" fn(This: *mut IUnknown) -> u32,
}
#[repr(C)]
pub struct IDataObjectVtbl {
pub parent: IUnknownVtbl,
pub GetData: unsafe extern "system" fn(
This: *mut IDataObject,
pformatetcIn: *const FORMATETC,
pmedium: *mut STGMEDIUM,
) -> HRESULT,
pub GetDataHere: unsafe extern "system" fn(
This: *mut IDataObject,
pformatetc: *const FORMATETC,
pmedium: *mut STGMEDIUM,
) -> HRESULT,
QueryGetData:
unsafe extern "system" fn(This: *mut IDataObject, pformatetc: *const FORMATETC) -> HRESULT,
pub GetCanonicalFormatEtc: unsafe extern "system" fn(
This: *mut IDataObject,
pformatetcIn: *const FORMATETC,
pformatetcOut: *mut FORMATETC,
) -> HRESULT,
pub SetData: unsafe extern "system" fn(
This: *mut IDataObject,
pformatetc: *const FORMATETC,
pformatetcOut: *const FORMATETC,
fRelease: BOOL,
) -> HRESULT,
pub EnumFormatEtc: unsafe extern "system" fn(
This: *mut IDataObject,
dwDirection: u32,
ppenumFormatEtc: *mut *mut IEnumFORMATETC,
) -> HRESULT,
pub DAdvise: unsafe extern "system" fn(
This: *mut IDataObject,
pformatetc: *const FORMATETC,
advf: u32,
pAdvSInk: *const IAdviseSink,
pdwConnection: *mut u32,
) -> HRESULT,
pub DUnadvise: unsafe extern "system" fn(This: *mut IDataObject, dwConnection: u32) -> HRESULT,
pub EnumDAdvise: unsafe extern "system" fn(
This: *mut IDataObject,
ppenumAdvise: *const *const IEnumSTATDATA,
) -> HRESULT,
}
#[repr(C)]
pub struct IDropTargetVtbl {
pub parent: IUnknownVtbl,
pub DragEnter: unsafe extern "system" fn(
This: *mut IDropTarget,
pDataObj: *const IDataObject,
grfKeyState: u32,
pt: *const POINTL,
pdwEffect: *mut u32,
) -> HRESULT,
pub DragOver: unsafe extern "system" fn(
This: *mut IDropTarget,
grfKeyState: u32,
pt: *const POINTL,
pdwEffect: *mut u32,
) -> HRESULT,
pub DragLeave: unsafe extern "system" fn(This: *mut IDropTarget) -> HRESULT,
pub Drop: unsafe extern "system" fn(
This: *mut IDropTarget,
pDataObj: *const IDataObject,
grfKeyState: u32,
pt: *const POINTL,
pdwEffect: *mut u32,
) -> HRESULT,
}
#[repr(C)]
pub struct IDropTarget {
pub lpVtbl: *const IDropTargetVtbl,
}
#[repr(C)]
pub struct ITaskbarListVtbl {
pub parent: IUnknownVtbl,
pub HrInit: unsafe extern "system" fn(This: *mut ITaskbarList) -> HRESULT,
pub AddTab: unsafe extern "system" fn(This: *mut ITaskbarList, hwnd: HWND) -> HRESULT,
pub DeleteTab: unsafe extern "system" fn(This: *mut ITaskbarList, hwnd: HWND) -> HRESULT,
pub ActivateTab: unsafe extern "system" fn(This: *mut ITaskbarList, hwnd: HWND) -> HRESULT,
pub SetActiveAlt: unsafe extern "system" fn(This: *mut ITaskbarList, hwnd: HWND) -> HRESULT,
}
#[repr(C)]
pub struct ITaskbarList2Vtbl {
pub parent: ITaskbarListVtbl,
pub MarkFullscreenWindow: unsafe extern "system" fn(
This: *mut ITaskbarList2,
hwnd: HWND,
fFullscreen: BOOL,
) -> HRESULT,
}
#[repr(C)]
pub struct ITaskbarList2 {
pub lpVtbl: *const ITaskbarList2Vtbl,
}
pub const CLSID_TaskbarList: GUID = GUID {
data1: 0x56fdf344,
data2: 0xfd6d,
data3: 0x11d0,
data4: [0x95, 0x8a, 0x00, 0x60, 0x97, 0xc9, 0xa0, 0x90],
};
pub const IID_ITaskbarList2: GUID = GUID {
data1: 0x602d4995,
data2: 0xb13a,
data3: 0x429b,
data4: [0xa6, 0x6e, 0x19, 0x35, 0xe4, 0x4f, 0x43, 0x17],
};

View file

@ -2,24 +2,26 @@
use std::sync::Once; use std::sync::Once;
use windows_sys::Win32::{
Foundation::{HWND, S_OK},
Graphics::Gdi::{
GetDC, GetDeviceCaps, MonitorFromWindow, HMONITOR, LOGPIXELSX, MONITOR_DEFAULTTONEAREST,
},
UI::{
HiDpi::{
DPI_AWARENESS_CONTEXT, DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE, MDT_EFFECTIVE_DPI,
PROCESS_PER_MONITOR_DPI_AWARE,
},
WindowsAndMessaging::IsProcessDPIAware,
},
};
use crate::platform_impl::platform::util::{ use crate::platform_impl::platform::util::{
ENABLE_NON_CLIENT_DPI_SCALING, GET_DPI_FOR_MONITOR, GET_DPI_FOR_WINDOW, SET_PROCESS_DPI_AWARE, ENABLE_NON_CLIENT_DPI_SCALING, GET_DPI_FOR_MONITOR, GET_DPI_FOR_WINDOW, SET_PROCESS_DPI_AWARE,
SET_PROCESS_DPI_AWARENESS, SET_PROCESS_DPI_AWARENESS_CONTEXT, SET_PROCESS_DPI_AWARENESS, SET_PROCESS_DPI_AWARENESS_CONTEXT,
}; };
use winapi::{
shared::{
minwindef::FALSE,
windef::{DPI_AWARENESS_CONTEXT, DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE, HMONITOR, HWND},
winerror::S_OK,
},
um::{
shellscalingapi::{MDT_EFFECTIVE_DPI, PROCESS_PER_MONITOR_DPI_AWARE},
wingdi::{GetDeviceCaps, LOGPIXELSX},
winuser::{self, MONITOR_DEFAULTTONEAREST},
},
};
const DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2: DPI_AWARENESS_CONTEXT = -4isize as _; const DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2: DPI_AWARENESS_CONTEXT = -4;
pub fn become_dpi_aware() { pub fn become_dpi_aware() {
static ENABLE_DPI_AWARENESS: Once = Once::new(); static ENABLE_DPI_AWARENESS: Once = Once::new();
@ -28,7 +30,7 @@ pub fn become_dpi_aware() {
if let Some(SetProcessDpiAwarenessContext) = *SET_PROCESS_DPI_AWARENESS_CONTEXT { if let Some(SetProcessDpiAwarenessContext) = *SET_PROCESS_DPI_AWARENESS_CONTEXT {
// We are on Windows 10 Anniversary Update (1607) or later. // We are on Windows 10 Anniversary Update (1607) or later.
if SetProcessDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2) if SetProcessDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2)
== FALSE == false.into()
{ {
// V2 only works with Windows 10 Creators Update (1703). Try using the older // V2 only works with Windows 10 Creators Update (1703). Try using the older
// V1 if we can't set V2. // V1 if we can't set V2.
@ -76,8 +78,8 @@ pub fn dpi_to_scale_factor(dpi: u32) -> f64 {
} }
pub unsafe fn hwnd_dpi(hwnd: HWND) -> u32 { pub unsafe fn hwnd_dpi(hwnd: HWND) -> u32 {
let hdc = winuser::GetDC(hwnd); let hdc = GetDC(hwnd);
if hdc.is_null() { if hdc == 0 {
panic!("[winit] `GetDC` returned null!"); panic!("[winit] `GetDC` returned null!");
} }
if let Some(GetDpiForWindow) = *GET_DPI_FOR_WINDOW { if let Some(GetDpiForWindow) = *GET_DPI_FOR_WINDOW {
@ -88,8 +90,8 @@ pub unsafe fn hwnd_dpi(hwnd: HWND) -> u32 {
} }
} else if let Some(GetDpiForMonitor) = *GET_DPI_FOR_MONITOR { } else if let Some(GetDpiForMonitor) = *GET_DPI_FOR_MONITOR {
// We are on Windows 8.1 or later. // We are on Windows 8.1 or later.
let monitor = winuser::MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST); let monitor = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);
if monitor.is_null() { if monitor == 0 {
return BASE_DPI; return BASE_DPI;
} }
@ -102,7 +104,7 @@ pub unsafe fn hwnd_dpi(hwnd: HWND) -> u32 {
} }
} else { } else {
// We are on Vista or later. // We are on Vista or later.
if winuser::IsProcessDPIAware() != FALSE { if IsProcessDPIAware() != false.into() {
// If the process is DPI aware, then scaling must be handled by the application using // If the process is DPI aware, then scaling must be handled by the application using
// this DPI value. // this DPI value.
GetDeviceCaps(hdc, LOGPIXELSX) as u32 GetDeviceCaps(hdc, LOGPIXELSX) as u32

View file

@ -1,29 +1,28 @@
use std::{ use std::{
ffi::OsString, ffi::{c_void, OsString},
os::windows::ffi::OsStringExt, os::windows::ffi::OsStringExt,
path::PathBuf, path::PathBuf,
ptr, ptr,
sync::atomic::{AtomicUsize, Ordering}, sync::atomic::{AtomicUsize, Ordering},
}; };
use winapi::{ use windows_sys::{
ctypes::c_void, core::{IUnknown, GUID, HRESULT},
shared::{ Win32::{
guiddef::REFIID, Foundation::{DV_E_FORMATETC, HWND, POINTL, S_OK},
minwindef::{DWORD, UINT, ULONG}, System::{
windef::{HWND, POINTL}, Com::{IDataObject, DVASPECT_CONTENT, FORMATETC, TYMED_HGLOBAL},
winerror::S_OK, Ole::{DROPEFFECT_COPY, DROPEFFECT_NONE},
SystemServices::CF_HDROP,
}, },
um::{ UI::Shell::{DragFinish, DragQueryFileW, HDROP},
objidl::IDataObject,
oleidl::{IDropTarget, IDropTargetVtbl, DROPEFFECT_COPY, DROPEFFECT_NONE},
shellapi, unknwnbase,
winnt::HRESULT,
}, },
}; };
use crate::platform_impl::platform::WindowId; use crate::platform_impl::platform::{
definitions::{IDataObjectVtbl, IDropTarget, IDropTargetVtbl, IUnknownVtbl},
WindowId,
};
use crate::{event::Event, window::WindowId as SuperWindowId}; use crate::{event::Event, window::WindowId as SuperWindowId};
#[repr(C)] #[repr(C)]
@ -32,7 +31,7 @@ pub struct FileDropHandlerData {
refcount: AtomicUsize, refcount: AtomicUsize,
window: HWND, window: HWND,
send_event: Box<dyn Fn(Event<'static, ()>)>, send_event: Box<dyn Fn(Event<'static, ()>)>,
cursor_effect: DWORD, cursor_effect: u32,
hovered_is_valid: bool, /* If the currently hovered item is not valid there must not be any `HoveredFileCancelled` emitted */ hovered_is_valid: bool, /* If the currently hovered item is not valid there must not be any `HoveredFileCancelled` emitted */
} }
@ -60,8 +59,8 @@ impl FileDropHandler {
// Implement IUnknown // Implement IUnknown
pub unsafe extern "system" fn QueryInterface( pub unsafe extern "system" fn QueryInterface(
_this: *mut unknwnbase::IUnknown, _this: *mut IUnknown,
_riid: REFIID, _riid: *const GUID,
_ppvObject: *mut *mut c_void, _ppvObject: *mut *mut c_void,
) -> HRESULT { ) -> HRESULT {
// This function doesn't appear to be required for an `IDropTarget`. // This function doesn't appear to be required for an `IDropTarget`.
@ -69,28 +68,28 @@ impl FileDropHandler {
unimplemented!(); unimplemented!();
} }
pub unsafe extern "system" fn AddRef(this: *mut unknwnbase::IUnknown) -> ULONG { pub unsafe extern "system" fn AddRef(this: *mut IUnknown) -> u32 {
let drop_handler_data = Self::from_interface(this); let drop_handler_data = Self::from_interface(this);
let count = drop_handler_data.refcount.fetch_add(1, Ordering::Release) + 1; let count = drop_handler_data.refcount.fetch_add(1, Ordering::Release) + 1;
count as ULONG count as u32
} }
pub unsafe extern "system" fn Release(this: *mut unknwnbase::IUnknown) -> ULONG { pub unsafe extern "system" fn Release(this: *mut IUnknown) -> u32 {
let drop_handler = Self::from_interface(this); let drop_handler = Self::from_interface(this);
let count = drop_handler.refcount.fetch_sub(1, Ordering::Release) - 1; let count = drop_handler.refcount.fetch_sub(1, Ordering::Release) - 1;
if count == 0 { if count == 0 {
// Destroy the underlying data // Destroy the underlying data
Box::from_raw(drop_handler as *mut FileDropHandlerData); Box::from_raw(drop_handler as *mut FileDropHandlerData);
} }
count as ULONG count as u32
} }
pub unsafe extern "system" fn DragEnter( pub unsafe extern "system" fn DragEnter(
this: *mut IDropTarget, this: *mut IDropTarget,
pDataObj: *const IDataObject, pDataObj: *const IDataObject,
_grfKeyState: DWORD, _grfKeyState: u32,
_pt: *const POINTL, _pt: *const POINTL,
pdwEffect: *mut DWORD, pdwEffect: *mut u32,
) -> HRESULT { ) -> HRESULT {
use crate::event::WindowEvent::HoveredFile; use crate::event::WindowEvent::HoveredFile;
let drop_handler = Self::from_interface(this); let drop_handler = Self::from_interface(this);
@ -113,9 +112,9 @@ impl FileDropHandler {
pub unsafe extern "system" fn DragOver( pub unsafe extern "system" fn DragOver(
this: *mut IDropTarget, this: *mut IDropTarget,
_grfKeyState: DWORD, _grfKeyState: u32,
_pt: *const POINTL, _pt: *const POINTL,
pdwEffect: *mut DWORD, pdwEffect: *mut u32,
) -> HRESULT { ) -> HRESULT {
let drop_handler = Self::from_interface(this); let drop_handler = Self::from_interface(this);
*pdwEffect = drop_handler.cursor_effect; *pdwEffect = drop_handler.cursor_effect;
@ -139,9 +138,9 @@ impl FileDropHandler {
pub unsafe extern "system" fn Drop( pub unsafe extern "system" fn Drop(
this: *mut IDropTarget, this: *mut IDropTarget,
pDataObj: *const IDataObject, pDataObj: *const IDataObject,
_grfKeyState: DWORD, _grfKeyState: u32,
_pt: *const POINTL, _pt: *const POINTL,
_pdwEffect: *mut DWORD, _pdwEffect: *mut u32,
) -> HRESULT { ) -> HRESULT {
use crate::event::WindowEvent::DroppedFile; use crate::event::WindowEvent::DroppedFile;
let drop_handler = Self::from_interface(this); let drop_handler = Self::from_interface(this);
@ -152,7 +151,7 @@ impl FileDropHandler {
}); });
}); });
if let Some(hdrop) = hdrop { if let Some(hdrop) = hdrop {
shellapi::DragFinish(hdrop); DragFinish(hdrop);
} }
S_OK S_OK
@ -162,38 +161,23 @@ impl FileDropHandler {
&mut *(this as *mut _) &mut *(this as *mut _)
} }
unsafe fn iterate_filenames<F>( unsafe fn iterate_filenames<F>(data_obj: *const IDataObject, callback: F) -> Option<HDROP>
data_obj: *const IDataObject,
callback: F,
) -> Option<shellapi::HDROP>
where where
F: Fn(PathBuf), F: Fn(PathBuf),
{ {
use winapi::{
shared::{
winerror::{DV_E_FORMATETC, SUCCEEDED},
wtypes::{CLIPFORMAT, DVASPECT_CONTENT},
},
um::{
objidl::{FORMATETC, TYMED_HGLOBAL},
shellapi::DragQueryFileW,
winuser::CF_HDROP,
},
};
let drop_format = FORMATETC { let drop_format = FORMATETC {
cfFormat: CF_HDROP as CLIPFORMAT, cfFormat: CF_HDROP as u16,
ptd: ptr::null(), ptd: ptr::null_mut(),
dwAspect: DVASPECT_CONTENT, dwAspect: DVASPECT_CONTENT as u32,
lindex: -1, lindex: -1,
tymed: TYMED_HGLOBAL, tymed: TYMED_HGLOBAL as u32,
}; };
let mut medium = std::mem::zeroed(); let mut medium = std::mem::zeroed();
let get_data_result = (*data_obj).GetData(&drop_format, &mut medium); let get_data_fn = (*(*data_obj).cast::<IDataObjectVtbl>()).GetData;
if SUCCEEDED(get_data_result) { let get_data_result = get_data_fn(data_obj as *mut _, &drop_format, &mut medium);
let hglobal = (*medium.u).hGlobal(); if get_data_result >= 0 {
let hdrop = (*hglobal) as shellapi::HDROP; let hdrop = medium.Anonymous.hGlobal;
// The second parameter (0xFFFFFFFF) instructs the function to return the item count // The second parameter (0xFFFFFFFF) instructs the function to return the item count
let item_count = DragQueryFileW(hdrop, 0xFFFFFFFF, ptr::null_mut(), 0); let item_count = DragQueryFileW(hdrop, 0xFFFFFFFF, ptr::null_mut(), 0);
@ -207,7 +191,7 @@ impl FileDropHandler {
// Fill path_buf with the null-terminated file name // Fill path_buf with the null-terminated file name
let mut path_buf = Vec::with_capacity(str_len); let mut path_buf = Vec::with_capacity(str_len);
DragQueryFileW(hdrop, i, path_buf.as_mut_ptr(), str_len as UINT); DragQueryFileW(hdrop, i, path_buf.as_mut_ptr(), str_len as u32);
path_buf.set_len(str_len); path_buf.set_len(str_len);
callback(OsString::from_wide(&path_buf[0..character_count]).into()); callback(OsString::from_wide(&path_buf[0..character_count]).into());
@ -235,13 +219,13 @@ impl FileDropHandlerData {
impl Drop for FileDropHandler { impl Drop for FileDropHandler {
fn drop(&mut self) { fn drop(&mut self) {
unsafe { unsafe {
FileDropHandler::Release(self.data as *mut unknwnbase::IUnknown); FileDropHandler::Release(self.data as *mut IUnknown);
} }
} }
} }
static DROP_TARGET_VTBL: IDropTargetVtbl = IDropTargetVtbl { static DROP_TARGET_VTBL: IDropTargetVtbl = IDropTargetVtbl {
parent: unknwnbase::IUnknownVtbl { parent: IUnknownVtbl {
QueryInterface: FileDropHandler::QueryInterface, QueryInterface: FileDropHandler::QueryInterface,
AddRef: FileDropHandler::AddRef, AddRef: FileDropHandler::AddRef,
Release: FileDropHandler::Release, Release: FileDropHandler::Release,

View file

@ -1,37 +1,60 @@
use std::{ use std::{
char, char,
os::raw::c_int, sync::atomic::{AtomicBool, AtomicIsize, Ordering},
ptr, };
sync::atomic::{AtomicBool, AtomicPtr, Ordering},
use windows_sys::Win32::{
Foundation::{LPARAM, WPARAM},
UI::{
Input::KeyboardAndMouse::{
GetKeyState, GetKeyboardLayout, GetKeyboardState, MapVirtualKeyA, ToUnicodeEx,
VIRTUAL_KEY, VK_0, VK_1, VK_2, VK_3, VK_4, VK_5, VK_6, VK_7, VK_8, VK_9, VK_A, VK_ADD,
VK_APPS, VK_B, VK_BACK, VK_BROWSER_BACK, VK_BROWSER_FAVORITES, VK_BROWSER_FORWARD,
VK_BROWSER_HOME, VK_BROWSER_REFRESH, VK_BROWSER_SEARCH, VK_BROWSER_STOP, VK_C,
VK_CAPITAL, VK_CONTROL, VK_CONVERT, VK_D, VK_DECIMAL, VK_DELETE, VK_DIVIDE, VK_DOWN,
VK_E, VK_END, VK_ESCAPE, VK_F, VK_F1, VK_F10, VK_F11, VK_F12, VK_F13, VK_F14, VK_F15,
VK_F16, VK_F17, VK_F18, VK_F19, VK_F2, VK_F20, VK_F21, VK_F22, VK_F23, VK_F24, VK_F3,
VK_F4, VK_F5, VK_F6, VK_F7, VK_F8, VK_F9, VK_G, VK_H, VK_HOME, VK_I, VK_INSERT, VK_J,
VK_K, VK_KANA, VK_KANJI, VK_L, VK_LAUNCH_MAIL, VK_LAUNCH_MEDIA_SELECT, VK_LCONTROL,
VK_LEFT, VK_LMENU, VK_LSHIFT, VK_LWIN, VK_M, VK_MEDIA_NEXT_TRACK, VK_MEDIA_PLAY_PAUSE,
VK_MEDIA_PREV_TRACK, VK_MEDIA_STOP, VK_MENU, VK_MULTIPLY, VK_N, VK_NEXT, VK_NONCONVERT,
VK_NUMLOCK, VK_NUMPAD0, VK_NUMPAD1, VK_NUMPAD2, VK_NUMPAD3, VK_NUMPAD4, VK_NUMPAD5,
VK_NUMPAD6, VK_NUMPAD7, VK_NUMPAD8, VK_NUMPAD9, VK_O, VK_OEM_1, VK_OEM_102, VK_OEM_2,
VK_OEM_3, VK_OEM_4, VK_OEM_5, VK_OEM_6, VK_OEM_7, VK_OEM_COMMA, VK_OEM_MINUS,
VK_OEM_PERIOD, VK_OEM_PLUS, VK_P, VK_PAUSE, VK_PRIOR, VK_Q, VK_R, VK_RCONTROL,
VK_RETURN, VK_RIGHT, VK_RMENU, VK_RSHIFT, VK_RWIN, VK_S, VK_SCROLL, VK_SHIFT, VK_SLEEP,
VK_SNAPSHOT, VK_SPACE, VK_SUBTRACT, VK_T, VK_TAB, VK_U, VK_UP, VK_V, VK_VOLUME_DOWN,
VK_VOLUME_MUTE, VK_VOLUME_UP, VK_W, VK_X, VK_Y, VK_Z,
},
TextServices::HKL,
WindowsAndMessaging::{MAPVK_VK_TO_CHAR, MAPVK_VSC_TO_VK_EX},
},
}; };
use crate::event::{ModifiersState, ScanCode, VirtualKeyCode}; use crate::event::{ModifiersState, ScanCode, VirtualKeyCode};
use winapi::{ use super::util::has_flag;
shared::minwindef::{HKL, HKL__, LPARAM, UINT, WPARAM},
um::winuser,
};
fn key_pressed(vkey: c_int) -> bool { fn key_pressed(vkey: VIRTUAL_KEY) -> bool {
unsafe { (winuser::GetKeyState(vkey) & (1 << 15)) == (1 << 15) } unsafe { has_flag(GetKeyState(vkey as i32), 1 << 15) }
} }
pub fn get_key_mods() -> ModifiersState { pub fn get_key_mods() -> ModifiersState {
let filter_out_altgr = layout_uses_altgr() && key_pressed(winuser::VK_RMENU); let filter_out_altgr = layout_uses_altgr() && key_pressed(VK_RMENU);
let mut mods = ModifiersState::empty(); let mut mods = ModifiersState::empty();
mods.set(ModifiersState::SHIFT, key_pressed(winuser::VK_SHIFT)); mods.set(ModifiersState::SHIFT, key_pressed(VK_SHIFT));
mods.set( mods.set(
ModifiersState::CTRL, ModifiersState::CTRL,
key_pressed(winuser::VK_CONTROL) && !filter_out_altgr, key_pressed(VK_CONTROL) && !filter_out_altgr,
); );
mods.set( mods.set(
ModifiersState::ALT, ModifiersState::ALT,
key_pressed(winuser::VK_MENU) && !filter_out_altgr, key_pressed(VK_MENU) && !filter_out_altgr,
); );
mods.set( mods.set(
ModifiersState::LOGO, ModifiersState::LOGO,
key_pressed(winuser::VK_LWIN) || key_pressed(winuser::VK_RWIN), key_pressed(VK_LWIN) || key_pressed(VK_RWIN),
); );
mods mods
} }
@ -85,19 +108,19 @@ impl From<ModifiersStateSide> for ModifiersState {
} }
} }
pub fn get_pressed_keys() -> impl Iterator<Item = c_int> { pub fn get_pressed_keys() -> impl Iterator<Item = VIRTUAL_KEY> {
let mut keyboard_state = vec![0u8; 256]; let mut keyboard_state = vec![0u8; 256];
unsafe { winuser::GetKeyboardState(keyboard_state.as_mut_ptr()) }; unsafe { GetKeyboardState(keyboard_state.as_mut_ptr()) };
keyboard_state keyboard_state
.into_iter() .into_iter()
.enumerate() .enumerate()
.filter(|(_, p)| (*p & (1 << 7)) != 0) // whether or not a key is pressed is communicated via the high-order bit .filter(|(_, p)| (*p & (1 << 7)) != 0) // whether or not a key is pressed is communicated via the high-order bit
.map(|(i, _)| i as c_int) .map(|(i, _)| i as u16)
} }
unsafe fn get_char(keyboard_state: &[u8; 256], v_key: u32, hkl: HKL) -> Option<char> { unsafe fn get_char(keyboard_state: &[u8; 256], v_key: u32, hkl: HKL) -> Option<char> {
let mut unicode_bytes = [0u16; 5]; let mut unicode_bytes = [0u16; 5];
let len = winuser::ToUnicodeEx( let len = ToUnicodeEx(
v_key, v_key,
0, 0,
keyboard_state.as_ptr(), keyboard_state.as_ptr(),
@ -126,10 +149,10 @@ unsafe fn get_char(keyboard_state: &[u8; 256], v_key: u32, hkl: HKL) -> Option<c
/// [source]: https://github.com/mozilla/gecko-dev/blob/265e6721798a455604328ed5262f430cfcc37c2f/widget/windows/KeyboardLayout.cpp#L4356-L4416 /// [source]: https://github.com/mozilla/gecko-dev/blob/265e6721798a455604328ed5262f430cfcc37c2f/widget/windows/KeyboardLayout.cpp#L4356-L4416
fn layout_uses_altgr() -> bool { fn layout_uses_altgr() -> bool {
unsafe { unsafe {
static ACTIVE_LAYOUT: AtomicPtr<HKL__> = AtomicPtr::new(ptr::null_mut()); static ACTIVE_LAYOUT: AtomicIsize = AtomicIsize::new(0);
static USES_ALTGR: AtomicBool = AtomicBool::new(false); static USES_ALTGR: AtomicBool = AtomicBool::new(false);
let hkl = winuser::GetKeyboardLayout(0); let hkl = GetKeyboardLayout(0);
let old_hkl = ACTIVE_LAYOUT.swap(hkl, Ordering::SeqCst); let old_hkl = ACTIVE_LAYOUT.swap(hkl, Ordering::SeqCst);
if hkl == old_hkl { if hkl == old_hkl {
@ -139,8 +162,8 @@ fn layout_uses_altgr() -> bool {
let mut keyboard_state_altgr = [0u8; 256]; let mut keyboard_state_altgr = [0u8; 256];
// AltGr is an alias for Ctrl+Alt for... some reason. Whatever it is, those are the keypresses // AltGr is an alias for Ctrl+Alt for... some reason. Whatever it is, those are the keypresses
// we have to emulate to do an AltGr test. // we have to emulate to do an AltGr test.
keyboard_state_altgr[winuser::VK_MENU as usize] = 0x80; keyboard_state_altgr[VK_MENU as usize] = 0x80;
keyboard_state_altgr[winuser::VK_CONTROL as usize] = 0x80; keyboard_state_altgr[VK_CONTROL as usize] = 0x80;
let keyboard_state_empty = [0u8; 256]; let keyboard_state_empty = [0u8; 256];
@ -160,206 +183,204 @@ fn layout_uses_altgr() -> bool {
} }
} }
pub fn vkey_to_winit_vkey(vkey: c_int) -> Option<VirtualKeyCode> { pub fn vkey_to_winit_vkey(vkey: VIRTUAL_KEY) -> Option<VirtualKeyCode> {
// VK_* codes are documented here https://msdn.microsoft.com/en-us/library/windows/desktop/dd375731(v=vs.85).aspx // VK_* codes are documented here https://msdn.microsoft.com/en-us/library/windows/desktop/dd375731(v=vs.85).aspx
match vkey { match vkey {
//winuser::VK_LBUTTON => Some(VirtualKeyCode::Lbutton), //VK_LBUTTON => Some(VirtualKeyCode::Lbutton),
//winuser::VK_RBUTTON => Some(VirtualKeyCode::Rbutton), //VK_RBUTTON => Some(VirtualKeyCode::Rbutton),
//winuser::VK_CANCEL => Some(VirtualKeyCode::Cancel), //VK_CANCEL => Some(VirtualKeyCode::Cancel),
//winuser::VK_MBUTTON => Some(VirtualKeyCode::Mbutton), //VK_MBUTTON => Some(VirtualKeyCode::Mbutton),
//winuser::VK_XBUTTON1 => Some(VirtualKeyCode::Xbutton1), //VK_XBUTTON1 => Some(VirtualKeyCode::Xbutton1),
//winuser::VK_XBUTTON2 => Some(VirtualKeyCode::Xbutton2), //VK_XBUTTON2 => Some(VirtualKeyCode::Xbutton2),
winuser::VK_BACK => Some(VirtualKeyCode::Back), VK_BACK => Some(VirtualKeyCode::Back),
winuser::VK_TAB => Some(VirtualKeyCode::Tab), VK_TAB => Some(VirtualKeyCode::Tab),
//winuser::VK_CLEAR => Some(VirtualKeyCode::Clear), //VK_CLEAR => Some(VirtualKeyCode::Clear),
winuser::VK_RETURN => Some(VirtualKeyCode::Return), VK_RETURN => Some(VirtualKeyCode::Return),
winuser::VK_LSHIFT => Some(VirtualKeyCode::LShift), VK_LSHIFT => Some(VirtualKeyCode::LShift),
winuser::VK_RSHIFT => Some(VirtualKeyCode::RShift), VK_RSHIFT => Some(VirtualKeyCode::RShift),
winuser::VK_LCONTROL => Some(VirtualKeyCode::LControl), VK_LCONTROL => Some(VirtualKeyCode::LControl),
winuser::VK_RCONTROL => Some(VirtualKeyCode::RControl), VK_RCONTROL => Some(VirtualKeyCode::RControl),
winuser::VK_LMENU => Some(VirtualKeyCode::LAlt), VK_LMENU => Some(VirtualKeyCode::LAlt),
winuser::VK_RMENU => Some(VirtualKeyCode::RAlt), VK_RMENU => Some(VirtualKeyCode::RAlt),
winuser::VK_PAUSE => Some(VirtualKeyCode::Pause), VK_PAUSE => Some(VirtualKeyCode::Pause),
winuser::VK_CAPITAL => Some(VirtualKeyCode::Capital), VK_CAPITAL => Some(VirtualKeyCode::Capital),
winuser::VK_KANA => Some(VirtualKeyCode::Kana), VK_KANA => Some(VirtualKeyCode::Kana),
//winuser::VK_HANGUEL => Some(VirtualKeyCode::Hanguel), //VK_HANGUEL => Some(VirtualKeyCode::Hanguel),
//winuser::VK_HANGUL => Some(VirtualKeyCode::Hangul), //VK_HANGUL => Some(VirtualKeyCode::Hangul),
//winuser::VK_JUNJA => Some(VirtualKeyCode::Junja), //VK_JUNJA => Some(VirtualKeyCode::Junja),
//winuser::VK_FINAL => Some(VirtualKeyCode::Final), //VK_FINAL => Some(VirtualKeyCode::Final),
//winuser::VK_HANJA => Some(VirtualKeyCode::Hanja), //VK_HANJA => Some(VirtualKeyCode::Hanja),
winuser::VK_KANJI => Some(VirtualKeyCode::Kanji), VK_KANJI => Some(VirtualKeyCode::Kanji),
winuser::VK_ESCAPE => Some(VirtualKeyCode::Escape), VK_ESCAPE => Some(VirtualKeyCode::Escape),
winuser::VK_CONVERT => Some(VirtualKeyCode::Convert), VK_CONVERT => Some(VirtualKeyCode::Convert),
winuser::VK_NONCONVERT => Some(VirtualKeyCode::NoConvert), VK_NONCONVERT => Some(VirtualKeyCode::NoConvert),
//winuser::VK_ACCEPT => Some(VirtualKeyCode::Accept), //VK_ACCEPT => Some(VirtualKeyCode::Accept),
//winuser::VK_MODECHANGE => Some(VirtualKeyCode::Modechange), //VK_MODECHANGE => Some(VirtualKeyCode::Modechange),
winuser::VK_SPACE => Some(VirtualKeyCode::Space), VK_SPACE => Some(VirtualKeyCode::Space),
winuser::VK_PRIOR => Some(VirtualKeyCode::PageUp), VK_PRIOR => Some(VirtualKeyCode::PageUp),
winuser::VK_NEXT => Some(VirtualKeyCode::PageDown), VK_NEXT => Some(VirtualKeyCode::PageDown),
winuser::VK_END => Some(VirtualKeyCode::End), VK_END => Some(VirtualKeyCode::End),
winuser::VK_HOME => Some(VirtualKeyCode::Home), VK_HOME => Some(VirtualKeyCode::Home),
winuser::VK_LEFT => Some(VirtualKeyCode::Left), VK_LEFT => Some(VirtualKeyCode::Left),
winuser::VK_UP => Some(VirtualKeyCode::Up), VK_UP => Some(VirtualKeyCode::Up),
winuser::VK_RIGHT => Some(VirtualKeyCode::Right), VK_RIGHT => Some(VirtualKeyCode::Right),
winuser::VK_DOWN => Some(VirtualKeyCode::Down), VK_DOWN => Some(VirtualKeyCode::Down),
//winuser::VK_SELECT => Some(VirtualKeyCode::Select), //VK_SELECT => Some(VirtualKeyCode::Select),
//winuser::VK_PRINT => Some(VirtualKeyCode::Print), //VK_PRINT => Some(VirtualKeyCode::Print),
//winuser::VK_EXECUTE => Some(VirtualKeyCode::Execute), //VK_EXECUTE => Some(VirtualKeyCode::Execute),
winuser::VK_SNAPSHOT => Some(VirtualKeyCode::Snapshot), VK_SNAPSHOT => Some(VirtualKeyCode::Snapshot),
winuser::VK_INSERT => Some(VirtualKeyCode::Insert), VK_INSERT => Some(VirtualKeyCode::Insert),
winuser::VK_DELETE => Some(VirtualKeyCode::Delete), VK_DELETE => Some(VirtualKeyCode::Delete),
//winuser::VK_HELP => Some(VirtualKeyCode::Help), //VK_HELP => Some(VirtualKeyCode::Help),
0x30 => Some(VirtualKeyCode::Key0), VK_0 => Some(VirtualKeyCode::Key0),
0x31 => Some(VirtualKeyCode::Key1), VK_1 => Some(VirtualKeyCode::Key1),
0x32 => Some(VirtualKeyCode::Key2), VK_2 => Some(VirtualKeyCode::Key2),
0x33 => Some(VirtualKeyCode::Key3), VK_3 => Some(VirtualKeyCode::Key3),
0x34 => Some(VirtualKeyCode::Key4), VK_4 => Some(VirtualKeyCode::Key4),
0x35 => Some(VirtualKeyCode::Key5), VK_5 => Some(VirtualKeyCode::Key5),
0x36 => Some(VirtualKeyCode::Key6), VK_6 => Some(VirtualKeyCode::Key6),
0x37 => Some(VirtualKeyCode::Key7), VK_7 => Some(VirtualKeyCode::Key7),
0x38 => Some(VirtualKeyCode::Key8), VK_8 => Some(VirtualKeyCode::Key8),
0x39 => Some(VirtualKeyCode::Key9), VK_9 => Some(VirtualKeyCode::Key9),
0x41 => Some(VirtualKeyCode::A), VK_A => Some(VirtualKeyCode::A),
0x42 => Some(VirtualKeyCode::B), VK_B => Some(VirtualKeyCode::B),
0x43 => Some(VirtualKeyCode::C), VK_C => Some(VirtualKeyCode::C),
0x44 => Some(VirtualKeyCode::D), VK_D => Some(VirtualKeyCode::D),
0x45 => Some(VirtualKeyCode::E), VK_E => Some(VirtualKeyCode::E),
0x46 => Some(VirtualKeyCode::F), VK_F => Some(VirtualKeyCode::F),
0x47 => Some(VirtualKeyCode::G), VK_G => Some(VirtualKeyCode::G),
0x48 => Some(VirtualKeyCode::H), VK_H => Some(VirtualKeyCode::H),
0x49 => Some(VirtualKeyCode::I), VK_I => Some(VirtualKeyCode::I),
0x4A => Some(VirtualKeyCode::J), VK_J => Some(VirtualKeyCode::J),
0x4B => Some(VirtualKeyCode::K), VK_K => Some(VirtualKeyCode::K),
0x4C => Some(VirtualKeyCode::L), VK_L => Some(VirtualKeyCode::L),
0x4D => Some(VirtualKeyCode::M), VK_M => Some(VirtualKeyCode::M),
0x4E => Some(VirtualKeyCode::N), VK_N => Some(VirtualKeyCode::N),
0x4F => Some(VirtualKeyCode::O), VK_O => Some(VirtualKeyCode::O),
0x50 => Some(VirtualKeyCode::P), VK_P => Some(VirtualKeyCode::P),
0x51 => Some(VirtualKeyCode::Q), VK_Q => Some(VirtualKeyCode::Q),
0x52 => Some(VirtualKeyCode::R), VK_R => Some(VirtualKeyCode::R),
0x53 => Some(VirtualKeyCode::S), VK_S => Some(VirtualKeyCode::S),
0x54 => Some(VirtualKeyCode::T), VK_T => Some(VirtualKeyCode::T),
0x55 => Some(VirtualKeyCode::U), VK_U => Some(VirtualKeyCode::U),
0x56 => Some(VirtualKeyCode::V), VK_V => Some(VirtualKeyCode::V),
0x57 => Some(VirtualKeyCode::W), VK_W => Some(VirtualKeyCode::W),
0x58 => Some(VirtualKeyCode::X), VK_X => Some(VirtualKeyCode::X),
0x59 => Some(VirtualKeyCode::Y), VK_Y => Some(VirtualKeyCode::Y),
0x5A => Some(VirtualKeyCode::Z), VK_Z => Some(VirtualKeyCode::Z),
winuser::VK_LWIN => Some(VirtualKeyCode::LWin), VK_LWIN => Some(VirtualKeyCode::LWin),
winuser::VK_RWIN => Some(VirtualKeyCode::RWin), VK_RWIN => Some(VirtualKeyCode::RWin),
winuser::VK_APPS => Some(VirtualKeyCode::Apps), VK_APPS => Some(VirtualKeyCode::Apps),
winuser::VK_SLEEP => Some(VirtualKeyCode::Sleep), VK_SLEEP => Some(VirtualKeyCode::Sleep),
winuser::VK_NUMPAD0 => Some(VirtualKeyCode::Numpad0), VK_NUMPAD0 => Some(VirtualKeyCode::Numpad0),
winuser::VK_NUMPAD1 => Some(VirtualKeyCode::Numpad1), VK_NUMPAD1 => Some(VirtualKeyCode::Numpad1),
winuser::VK_NUMPAD2 => Some(VirtualKeyCode::Numpad2), VK_NUMPAD2 => Some(VirtualKeyCode::Numpad2),
winuser::VK_NUMPAD3 => Some(VirtualKeyCode::Numpad3), VK_NUMPAD3 => Some(VirtualKeyCode::Numpad3),
winuser::VK_NUMPAD4 => Some(VirtualKeyCode::Numpad4), VK_NUMPAD4 => Some(VirtualKeyCode::Numpad4),
winuser::VK_NUMPAD5 => Some(VirtualKeyCode::Numpad5), VK_NUMPAD5 => Some(VirtualKeyCode::Numpad5),
winuser::VK_NUMPAD6 => Some(VirtualKeyCode::Numpad6), VK_NUMPAD6 => Some(VirtualKeyCode::Numpad6),
winuser::VK_NUMPAD7 => Some(VirtualKeyCode::Numpad7), VK_NUMPAD7 => Some(VirtualKeyCode::Numpad7),
winuser::VK_NUMPAD8 => Some(VirtualKeyCode::Numpad8), VK_NUMPAD8 => Some(VirtualKeyCode::Numpad8),
winuser::VK_NUMPAD9 => Some(VirtualKeyCode::Numpad9), VK_NUMPAD9 => Some(VirtualKeyCode::Numpad9),
winuser::VK_MULTIPLY => Some(VirtualKeyCode::NumpadMultiply), VK_MULTIPLY => Some(VirtualKeyCode::NumpadMultiply),
winuser::VK_ADD => Some(VirtualKeyCode::NumpadAdd), VK_ADD => Some(VirtualKeyCode::NumpadAdd),
//winuser::VK_SEPARATOR => Some(VirtualKeyCode::Separator), //VK_SEPARATOR => Some(VirtualKeyCode::Separator),
winuser::VK_SUBTRACT => Some(VirtualKeyCode::NumpadSubtract), VK_SUBTRACT => Some(VirtualKeyCode::NumpadSubtract),
winuser::VK_DECIMAL => Some(VirtualKeyCode::NumpadDecimal), VK_DECIMAL => Some(VirtualKeyCode::NumpadDecimal),
winuser::VK_DIVIDE => Some(VirtualKeyCode::NumpadDivide), VK_DIVIDE => Some(VirtualKeyCode::NumpadDivide),
winuser::VK_F1 => Some(VirtualKeyCode::F1), VK_F1 => Some(VirtualKeyCode::F1),
winuser::VK_F2 => Some(VirtualKeyCode::F2), VK_F2 => Some(VirtualKeyCode::F2),
winuser::VK_F3 => Some(VirtualKeyCode::F3), VK_F3 => Some(VirtualKeyCode::F3),
winuser::VK_F4 => Some(VirtualKeyCode::F4), VK_F4 => Some(VirtualKeyCode::F4),
winuser::VK_F5 => Some(VirtualKeyCode::F5), VK_F5 => Some(VirtualKeyCode::F5),
winuser::VK_F6 => Some(VirtualKeyCode::F6), VK_F6 => Some(VirtualKeyCode::F6),
winuser::VK_F7 => Some(VirtualKeyCode::F7), VK_F7 => Some(VirtualKeyCode::F7),
winuser::VK_F8 => Some(VirtualKeyCode::F8), VK_F8 => Some(VirtualKeyCode::F8),
winuser::VK_F9 => Some(VirtualKeyCode::F9), VK_F9 => Some(VirtualKeyCode::F9),
winuser::VK_F10 => Some(VirtualKeyCode::F10), VK_F10 => Some(VirtualKeyCode::F10),
winuser::VK_F11 => Some(VirtualKeyCode::F11), VK_F11 => Some(VirtualKeyCode::F11),
winuser::VK_F12 => Some(VirtualKeyCode::F12), VK_F12 => Some(VirtualKeyCode::F12),
winuser::VK_F13 => Some(VirtualKeyCode::F13), VK_F13 => Some(VirtualKeyCode::F13),
winuser::VK_F14 => Some(VirtualKeyCode::F14), VK_F14 => Some(VirtualKeyCode::F14),
winuser::VK_F15 => Some(VirtualKeyCode::F15), VK_F15 => Some(VirtualKeyCode::F15),
winuser::VK_F16 => Some(VirtualKeyCode::F16), VK_F16 => Some(VirtualKeyCode::F16),
winuser::VK_F17 => Some(VirtualKeyCode::F17), VK_F17 => Some(VirtualKeyCode::F17),
winuser::VK_F18 => Some(VirtualKeyCode::F18), VK_F18 => Some(VirtualKeyCode::F18),
winuser::VK_F19 => Some(VirtualKeyCode::F19), VK_F19 => Some(VirtualKeyCode::F19),
winuser::VK_F20 => Some(VirtualKeyCode::F20), VK_F20 => Some(VirtualKeyCode::F20),
winuser::VK_F21 => Some(VirtualKeyCode::F21), VK_F21 => Some(VirtualKeyCode::F21),
winuser::VK_F22 => Some(VirtualKeyCode::F22), VK_F22 => Some(VirtualKeyCode::F22),
winuser::VK_F23 => Some(VirtualKeyCode::F23), VK_F23 => Some(VirtualKeyCode::F23),
winuser::VK_F24 => Some(VirtualKeyCode::F24), VK_F24 => Some(VirtualKeyCode::F24),
winuser::VK_NUMLOCK => Some(VirtualKeyCode::Numlock), VK_NUMLOCK => Some(VirtualKeyCode::Numlock),
winuser::VK_SCROLL => Some(VirtualKeyCode::Scroll), VK_SCROLL => Some(VirtualKeyCode::Scroll),
winuser::VK_BROWSER_BACK => Some(VirtualKeyCode::NavigateBackward), VK_BROWSER_BACK => Some(VirtualKeyCode::NavigateBackward),
winuser::VK_BROWSER_FORWARD => Some(VirtualKeyCode::NavigateForward), VK_BROWSER_FORWARD => Some(VirtualKeyCode::NavigateForward),
winuser::VK_BROWSER_REFRESH => Some(VirtualKeyCode::WebRefresh), VK_BROWSER_REFRESH => Some(VirtualKeyCode::WebRefresh),
winuser::VK_BROWSER_STOP => Some(VirtualKeyCode::WebStop), VK_BROWSER_STOP => Some(VirtualKeyCode::WebStop),
winuser::VK_BROWSER_SEARCH => Some(VirtualKeyCode::WebSearch), VK_BROWSER_SEARCH => Some(VirtualKeyCode::WebSearch),
winuser::VK_BROWSER_FAVORITES => Some(VirtualKeyCode::WebFavorites), VK_BROWSER_FAVORITES => Some(VirtualKeyCode::WebFavorites),
winuser::VK_BROWSER_HOME => Some(VirtualKeyCode::WebHome), VK_BROWSER_HOME => Some(VirtualKeyCode::WebHome),
winuser::VK_VOLUME_MUTE => Some(VirtualKeyCode::Mute), VK_VOLUME_MUTE => Some(VirtualKeyCode::Mute),
winuser::VK_VOLUME_DOWN => Some(VirtualKeyCode::VolumeDown), VK_VOLUME_DOWN => Some(VirtualKeyCode::VolumeDown),
winuser::VK_VOLUME_UP => Some(VirtualKeyCode::VolumeUp), VK_VOLUME_UP => Some(VirtualKeyCode::VolumeUp),
winuser::VK_MEDIA_NEXT_TRACK => Some(VirtualKeyCode::NextTrack), VK_MEDIA_NEXT_TRACK => Some(VirtualKeyCode::NextTrack),
winuser::VK_MEDIA_PREV_TRACK => Some(VirtualKeyCode::PrevTrack), VK_MEDIA_PREV_TRACK => Some(VirtualKeyCode::PrevTrack),
winuser::VK_MEDIA_STOP => Some(VirtualKeyCode::MediaStop), VK_MEDIA_STOP => Some(VirtualKeyCode::MediaStop),
winuser::VK_MEDIA_PLAY_PAUSE => Some(VirtualKeyCode::PlayPause), VK_MEDIA_PLAY_PAUSE => Some(VirtualKeyCode::PlayPause),
winuser::VK_LAUNCH_MAIL => Some(VirtualKeyCode::Mail), VK_LAUNCH_MAIL => Some(VirtualKeyCode::Mail),
winuser::VK_LAUNCH_MEDIA_SELECT => Some(VirtualKeyCode::MediaSelect), VK_LAUNCH_MEDIA_SELECT => Some(VirtualKeyCode::MediaSelect),
/*winuser::VK_LAUNCH_APP1 => Some(VirtualKeyCode::Launch_app1), /*VK_LAUNCH_APP1 => Some(VirtualKeyCode::Launch_app1),
winuser::VK_LAUNCH_APP2 => Some(VirtualKeyCode::Launch_app2),*/ VK_LAUNCH_APP2 => Some(VirtualKeyCode::Launch_app2),*/
winuser::VK_OEM_PLUS => Some(VirtualKeyCode::Equals), VK_OEM_PLUS => Some(VirtualKeyCode::Equals),
winuser::VK_OEM_COMMA => Some(VirtualKeyCode::Comma), VK_OEM_COMMA => Some(VirtualKeyCode::Comma),
winuser::VK_OEM_MINUS => Some(VirtualKeyCode::Minus), VK_OEM_MINUS => Some(VirtualKeyCode::Minus),
winuser::VK_OEM_PERIOD => Some(VirtualKeyCode::Period), VK_OEM_PERIOD => Some(VirtualKeyCode::Period),
winuser::VK_OEM_1 => map_text_keys(vkey), VK_OEM_1 => map_text_keys(vkey),
winuser::VK_OEM_2 => map_text_keys(vkey), VK_OEM_2 => map_text_keys(vkey),
winuser::VK_OEM_3 => map_text_keys(vkey), VK_OEM_3 => map_text_keys(vkey),
winuser::VK_OEM_4 => map_text_keys(vkey), VK_OEM_4 => map_text_keys(vkey),
winuser::VK_OEM_5 => map_text_keys(vkey), VK_OEM_5 => map_text_keys(vkey),
winuser::VK_OEM_6 => map_text_keys(vkey), VK_OEM_6 => map_text_keys(vkey),
winuser::VK_OEM_7 => map_text_keys(vkey), VK_OEM_7 => map_text_keys(vkey),
/* winuser::VK_OEM_8 => Some(VirtualKeyCode::Oem_8), */ /* VK_OEM_8 => Some(VirtualKeyCode::Oem_8), */
winuser::VK_OEM_102 => Some(VirtualKeyCode::OEM102), VK_OEM_102 => Some(VirtualKeyCode::OEM102),
/*winuser::VK_PROCESSKEY => Some(VirtualKeyCode::Processkey), /*VK_PROCESSKEY => Some(VirtualKeyCode::Processkey),
winuser::VK_PACKET => Some(VirtualKeyCode::Packet), VK_PACKET => Some(VirtualKeyCode::Packet),
winuser::VK_ATTN => Some(VirtualKeyCode::Attn), VK_ATTN => Some(VirtualKeyCode::Attn),
winuser::VK_CRSEL => Some(VirtualKeyCode::Crsel), VK_CRSEL => Some(VirtualKeyCode::Crsel),
winuser::VK_EXSEL => Some(VirtualKeyCode::Exsel), VK_EXSEL => Some(VirtualKeyCode::Exsel),
winuser::VK_EREOF => Some(VirtualKeyCode::Ereof), VK_EREOF => Some(VirtualKeyCode::Ereof),
winuser::VK_PLAY => Some(VirtualKeyCode::Play), VK_PLAY => Some(VirtualKeyCode::Play),
winuser::VK_ZOOM => Some(VirtualKeyCode::Zoom), VK_ZOOM => Some(VirtualKeyCode::Zoom),
winuser::VK_NONAME => Some(VirtualKeyCode::Noname), VK_NONAME => Some(VirtualKeyCode::Noname),
winuser::VK_PA1 => Some(VirtualKeyCode::Pa1), VK_PA1 => Some(VirtualKeyCode::Pa1),
winuser::VK_OEM_CLEAR => Some(VirtualKeyCode::Oem_clear),*/ VK_OEM_CLEAR => Some(VirtualKeyCode::Oem_clear),*/
_ => None, _ => None,
} }
} }
pub fn handle_extended_keys( pub fn handle_extended_keys(
vkey: c_int, vkey: VIRTUAL_KEY,
mut scancode: UINT, mut scancode: u32,
extended: bool, extended: bool,
) -> Option<(c_int, UINT)> { ) -> Option<(VIRTUAL_KEY, u32)> {
// Welcome to hell https://blog.molecular-matters.com/2011/09/05/properly-handling-keyboard-input/ // Welcome to hell https://blog.molecular-matters.com/2011/09/05/properly-handling-keyboard-input/
scancode |= if extended { 0xE000 } else { 0x0000 }; scancode |= if extended { 0xE000 } else { 0x0000 };
let vkey = match vkey { let vkey = match vkey {
winuser::VK_SHIFT => unsafe { VK_SHIFT => (unsafe { MapVirtualKeyA(scancode, MAPVK_VSC_TO_VK_EX) } as u16),
winuser::MapVirtualKeyA(scancode, winuser::MAPVK_VSC_TO_VK_EX) as _ VK_CONTROL => {
},
winuser::VK_CONTROL => {
if extended { if extended {
winuser::VK_RCONTROL VK_RCONTROL
} else { } else {
winuser::VK_LCONTROL VK_LCONTROL
} }
} }
winuser::VK_MENU => { VK_MENU => {
if extended { if extended {
winuser::VK_RMENU VK_RMENU
} else { } else {
winuser::VK_LMENU VK_LMENU
} }
} }
_ => { _ => {
@ -367,20 +388,20 @@ pub fn handle_extended_keys(
// When VK_PAUSE is pressed it emits a LeftControl + NumLock scancode event sequence, but reports VK_PAUSE // When VK_PAUSE is pressed it emits a LeftControl + NumLock scancode event sequence, but reports VK_PAUSE
// as the virtual key on both events, or VK_PAUSE on the first event or 0xFF when using raw input. // as the virtual key on both events, or VK_PAUSE on the first event or 0xFF when using raw input.
// Don't emit anything for the LeftControl event in the pair... // Don't emit anything for the LeftControl event in the pair...
0xE01D if vkey == winuser::VK_PAUSE => return None, 0xE01D if vkey == VK_PAUSE => return None,
// ...and emit the Pause event for the second event in the pair. // ...and emit the Pause event for the second event in the pair.
0x45 if vkey == winuser::VK_PAUSE || vkey == 0xFF => { 0x45 if vkey == VK_PAUSE || vkey == 0xFF => {
scancode = 0xE059; scancode = 0xE059;
winuser::VK_PAUSE VK_PAUSE
} }
// VK_PAUSE has an incorrect vkey value when used with modifiers. VK_PAUSE also reports a different // VK_PAUSE has an incorrect vkey value when used with modifiers. VK_PAUSE also reports a different
// scancode when used with modifiers than when used without // scancode when used with modifiers than when used without
0xE046 => { 0xE046 => {
scancode = 0xE059; scancode = 0xE059;
winuser::VK_PAUSE VK_PAUSE
} }
// VK_SCROLL has an incorrect vkey value when used with modifiers. // VK_SCROLL has an incorrect vkey value when used with modifiers.
0x46 => winuser::VK_SCROLL, 0x46 => VK_SCROLL,
_ => vkey, _ => vkey,
} }
} }
@ -392,18 +413,16 @@ pub fn process_key_params(
wparam: WPARAM, wparam: WPARAM,
lparam: LPARAM, lparam: LPARAM,
) -> Option<(ScanCode, Option<VirtualKeyCode>)> { ) -> Option<(ScanCode, Option<VirtualKeyCode>)> {
let scancode = ((lparam >> 16) & 0xff) as UINT; let scancode = ((lparam >> 16) & 0xff) as u32;
let extended = (lparam & 0x01000000) != 0; let extended = (lparam & 0x01000000) != 0;
handle_extended_keys(wparam as _, scancode, extended) handle_extended_keys(wparam as u16, scancode, extended)
.map(|(vkey, scancode)| (scancode, vkey_to_winit_vkey(vkey))) .map(|(vkey, scancode)| (scancode, vkey_to_winit_vkey(vkey)))
} }
// This is needed as windows doesn't properly distinguish // This is needed as windows doesn't properly distinguish
// some virtual key codes for different keyboard layouts // some virtual key codes for different keyboard layouts
fn map_text_keys(win_virtual_key: i32) -> Option<VirtualKeyCode> { fn map_text_keys(win_virtual_key: VIRTUAL_KEY) -> Option<VirtualKeyCode> {
let char_key = let char_key = unsafe { MapVirtualKeyA(win_virtual_key as u32, MAPVK_VK_TO_CHAR) } & 0x7FFF;
unsafe { winuser::MapVirtualKeyA(win_virtual_key as u32, winuser::MAPVK_VK_TO_CHAR) }
& 0x7FFF;
match char::from_u32(char_key) { match char::from_u32(char_key) {
Some(';') => Some(VirtualKeyCode::Semicolon), Some(';') => Some(VirtualKeyCode::Semicolon),
Some('/') => Some(VirtualKeyCode::Slash), Some('/') => Some(VirtualKeyCode::Slash),

File diff suppressed because it is too large Load diff

View file

@ -7,9 +7,9 @@ use std::{
time::Instant, time::Instant,
}; };
use winapi::{ use windows_sys::Win32::{
shared::{minwindef::DWORD, windef::HWND}, Foundation::HWND,
um::winuser, Graphics::Gdi::{RedrawWindow, RDW_INTERNALPAINT},
}; };
use crate::{ use crate::{
@ -24,7 +24,7 @@ pub(crate) type EventLoopRunnerShared<T> = Rc<EventLoopRunner<T>>;
pub(crate) struct EventLoopRunner<T: 'static> { pub(crate) struct EventLoopRunner<T: 'static> {
// The event loop's win32 handles // The event loop's win32 handles
pub(super) thread_msg_target: HWND, pub(super) thread_msg_target: HWND,
wait_thread_id: DWORD, wait_thread_id: u32,
control_flow: Cell<ControlFlow>, control_flow: Cell<ControlFlow>,
runner_state: Cell<RunnerState>, runner_state: Cell<RunnerState>,
@ -63,7 +63,7 @@ enum BufferedEvent<T: 'static> {
} }
impl<T> EventLoopRunner<T> { impl<T> EventLoopRunner<T> {
pub(crate) fn new(thread_msg_target: HWND, wait_thread_id: DWORD) -> EventLoopRunner<T> { pub(crate) fn new(thread_msg_target: HWND, wait_thread_id: u32) -> EventLoopRunner<T> {
EventLoopRunner { EventLoopRunner {
thread_msg_target, thread_msg_target,
wait_thread_id, wait_thread_id,
@ -113,7 +113,7 @@ impl<T> EventLoopRunner<T> {
self.thread_msg_target self.thread_msg_target
} }
pub fn wait_thread_id(&self) -> DWORD { pub fn wait_thread_id(&self) -> u32 {
self.wait_thread_id self.wait_thread_id
} }
@ -208,8 +208,7 @@ impl<T> EventLoopRunner<T> {
self.move_state_to(RunnerState::HandlingRedrawEvents); self.move_state_to(RunnerState::HandlingRedrawEvents);
} }
self.call_event_handler(event); self.call_event_handler(event);
} else { } else if self.should_buffer() {
if self.should_buffer() {
// If the runner is already borrowed, we're in the middle of an event loop invocation. Add // If the runner is already borrowed, we're in the middle of an event loop invocation. Add
// the event to a buffer to be processed later. // the event to a buffer to be processed later.
self.event_buffer self.event_buffer
@ -221,7 +220,6 @@ impl<T> EventLoopRunner<T> {
self.dispatch_buffered_events(); self.dispatch_buffered_events();
} }
} }
}
pub(crate) unsafe fn main_events_cleared(&self) { pub(crate) unsafe fn main_events_cleared(&self) {
self.move_state_to(RunnerState::HandlingRedrawEvents); self.move_state_to(RunnerState::HandlingRedrawEvents);
@ -394,12 +392,7 @@ impl<T> EventLoopRunner<T> {
}; };
self.call_event_handler(Event::NewEvents(start_cause)); self.call_event_handler(Event::NewEvents(start_cause));
self.dispatch_buffered_events(); self.dispatch_buffered_events();
winuser::RedrawWindow( RedrawWindow(self.thread_msg_target, ptr::null(), 0, RDW_INTERNALPAINT);
self.thread_msg_target,
ptr::null(),
ptr::null_mut(),
winuser::RDW_INTERNALPAINT,
);
} }
unsafe fn call_redraw_events_cleared(&self) { unsafe fn call_redraw_events_cleared(&self) {

View file

@ -1,18 +1,22 @@
use std::{fmt, io, iter::once, mem, os::windows::ffi::OsStrExt, path::Path, ptr, sync::Arc}; use std::{fmt, io, mem, path::Path, ptr, sync::Arc};
use winapi::{ use windows_sys::{
ctypes::{c_int, wchar_t}, core::PCWSTR,
shared::{ Win32::{
minwindef::{BYTE, LPARAM, WORD, WPARAM}, Foundation::HWND,
windef::{HICON, HWND}, System::LibraryLoader::GetModuleHandleW,
UI::WindowsAndMessaging::{
CreateIcon, DestroyIcon, LoadImageW, SendMessageW, HICON, ICON_BIG, ICON_SMALL,
IMAGE_ICON, LR_DEFAULTSIZE, LR_LOADFROMFILE, WM_SETICON,
},
}, },
um::libloaderapi,
um::winuser,
}; };
use crate::dpi::PhysicalSize; use crate::dpi::PhysicalSize;
use crate::icon::*; use crate::icon::*;
use super::util;
impl Pixel { impl Pixel {
fn to_bgra(&mut self) { fn to_bgra(&mut self) {
mem::swap(&mut self.r, &mut self.b); mem::swap(&mut self.r, &mut self.b);
@ -21,28 +25,28 @@ impl Pixel {
impl RgbaIcon { impl RgbaIcon {
fn into_windows_icon(self) -> Result<WinIcon, BadIcon> { fn into_windows_icon(self) -> Result<WinIcon, BadIcon> {
let mut rgba = self.rgba; let rgba = self.rgba;
let pixel_count = rgba.len() / PIXEL_SIZE; let pixel_count = rgba.len() / PIXEL_SIZE;
let mut and_mask = Vec::with_capacity(pixel_count); let mut and_mask = Vec::with_capacity(pixel_count);
let pixels = let pixels =
unsafe { std::slice::from_raw_parts_mut(rgba.as_mut_ptr() as *mut Pixel, pixel_count) }; unsafe { std::slice::from_raw_parts_mut(rgba.as_ptr() as *mut Pixel, pixel_count) };
for pixel in pixels { for pixel in pixels {
and_mask.push(pixel.a.wrapping_sub(std::u8::MAX)); // invert alpha channel and_mask.push(pixel.a.wrapping_sub(std::u8::MAX)); // invert alpha channel
pixel.to_bgra(); pixel.to_bgra();
} }
assert_eq!(and_mask.len(), pixel_count); assert_eq!(and_mask.len(), pixel_count);
let handle = unsafe { let handle = unsafe {
winuser::CreateIcon( CreateIcon(
ptr::null_mut(), 0,
self.width as c_int, self.width as i32,
self.height as c_int, self.height as i32,
1, 1,
(PIXEL_SIZE * 8) as BYTE, (PIXEL_SIZE * 8) as u8,
and_mask.as_ptr() as *const BYTE, and_mask.as_ptr(),
rgba.as_ptr() as *const BYTE, rgba.as_ptr(),
) as HICON )
}; };
if !handle.is_null() { if handle != 0 {
Ok(WinIcon::from_handle(handle)) Ok(WinIcon::from_handle(handle))
} else { } else {
Err(BadIcon::OsError(io::Error::last_os_error())) Err(BadIcon::OsError(io::Error::last_os_error()))
@ -52,8 +56,8 @@ impl RgbaIcon {
#[derive(Debug)] #[derive(Debug)]
pub enum IconType { pub enum IconType {
Small = winuser::ICON_SMALL as isize, Small = ICON_SMALL as isize,
Big = winuser::ICON_BIG as isize, Big = ICON_BIG as isize,
} }
#[derive(Debug)] #[derive(Debug)]
@ -77,51 +81,46 @@ impl WinIcon {
path: P, path: P,
size: Option<PhysicalSize<u32>>, size: Option<PhysicalSize<u32>>,
) -> Result<Self, BadIcon> { ) -> Result<Self, BadIcon> {
let wide_path: Vec<u16> = path
.as_ref()
.as_os_str()
.encode_wide()
.chain(once(0))
.collect();
// width / height of 0 along with LR_DEFAULTSIZE tells windows to load the default icon size // width / height of 0 along with LR_DEFAULTSIZE tells windows to load the default icon size
let (width, height) = size.map(Into::into).unwrap_or((0, 0)); let (width, height) = size.map(Into::into).unwrap_or((0, 0));
let wide_path = util::encode_wide(path.as_ref());
let handle = unsafe { let handle = unsafe {
winuser::LoadImageW( LoadImageW(
ptr::null_mut(), 0,
wide_path.as_ptr() as *const wchar_t, wide_path.as_ptr(),
winuser::IMAGE_ICON, IMAGE_ICON,
width as c_int, width as i32,
height as c_int, height as i32,
winuser::LR_DEFAULTSIZE | winuser::LR_LOADFROMFILE, LR_DEFAULTSIZE | LR_LOADFROMFILE,
) as HICON )
}; };
if !handle.is_null() { if handle != 0 {
Ok(WinIcon::from_handle(handle)) Ok(WinIcon::from_handle(handle as HICON))
} else { } else {
Err(BadIcon::OsError(io::Error::last_os_error())) Err(BadIcon::OsError(io::Error::last_os_error()))
} }
} }
pub fn from_resource( pub fn from_resource(
resource_id: WORD, resource_id: u16,
size: Option<PhysicalSize<u32>>, size: Option<PhysicalSize<u32>>,
) -> Result<Self, BadIcon> { ) -> Result<Self, BadIcon> {
// width / height of 0 along with LR_DEFAULTSIZE tells windows to load the default icon size // width / height of 0 along with LR_DEFAULTSIZE tells windows to load the default icon size
let (width, height) = size.map(Into::into).unwrap_or((0, 0)); let (width, height) = size.map(Into::into).unwrap_or((0, 0));
let handle = unsafe { let handle = unsafe {
winuser::LoadImageW( LoadImageW(
libloaderapi::GetModuleHandleW(ptr::null_mut()), GetModuleHandleW(ptr::null()),
winuser::MAKEINTRESOURCEW(resource_id), resource_id as PCWSTR,
winuser::IMAGE_ICON, IMAGE_ICON,
width as c_int, width as i32,
height as c_int, height as i32,
winuser::LR_DEFAULTSIZE, LR_DEFAULTSIZE,
) as HICON )
}; };
if !handle.is_null() { if handle != 0 {
Ok(WinIcon::from_handle(handle)) Ok(WinIcon::from_handle(handle as HICON))
} else { } else {
Err(BadIcon::OsError(io::Error::last_os_error())) Err(BadIcon::OsError(io::Error::last_os_error()))
} }
@ -134,12 +133,7 @@ impl WinIcon {
pub fn set_for_window(&self, hwnd: HWND, icon_type: IconType) { pub fn set_for_window(&self, hwnd: HWND, icon_type: IconType) {
unsafe { unsafe {
winuser::SendMessageW( SendMessageW(hwnd, WM_SETICON, icon_type as usize, self.as_raw_handle());
hwnd,
winuser::WM_SETICON,
icon_type as WPARAM,
self.as_raw_handle() as LPARAM,
);
} }
} }
@ -152,7 +146,7 @@ impl WinIcon {
impl Drop for RaiiIcon { impl Drop for RaiiIcon {
fn drop(&mut self) { fn drop(&mut self) {
unsafe { winuser::DestroyIcon(self.handle) }; unsafe { DestroyIcon(self.handle) };
} }
} }
@ -164,6 +158,6 @@ impl fmt::Debug for WinIcon {
pub fn unset_for_window(hwnd: HWND, icon_type: IconType) { pub fn unset_for_window(hwnd: HWND, icon_type: IconType) {
unsafe { unsafe {
winuser::SendMessageW(hwnd, winuser::WM_SETICON, icon_type as WPARAM, 0 as LPARAM); SendMessageW(hwnd, WM_SETICON, icon_type as usize, 0);
} }
} }

View file

@ -1,6 +1,9 @@
#![cfg(target_os = "windows")] #![cfg(target_os = "windows")]
use winapi::{self, shared::windef::HMENU, shared::windef::HWND}; use windows_sys::Win32::{
Foundation::{HANDLE, HWND},
UI::WindowsAndMessaging::{HMENU, WINDOW_LONG_PTR_INDEX},
};
pub(crate) use self::{ pub(crate) use self::{
event_loop::{ event_loop::{
@ -52,7 +55,7 @@ unsafe impl Sync for PlatformSpecificWindowBuilderAttributes {}
// Cursor name in UTF-16. Used to set cursor in `WM_SETCURSOR`. // Cursor name in UTF-16. Used to set cursor in `WM_SETCURSOR`.
#[derive(Debug, Clone, Copy)] #[derive(Debug, Clone, Copy)]
pub struct Cursor(pub *const winapi::ctypes::wchar_t); pub struct Cursor(pub *const u16);
unsafe impl Send for Cursor {} unsafe impl Send for Cursor {}
unsafe impl Sync for Cursor {} unsafe impl Sync for Cursor {}
@ -68,7 +71,7 @@ impl DeviceId {
impl DeviceId { impl DeviceId {
pub fn persistent_identifier(&self) -> Option<String> { pub fn persistent_identifier(&self) -> Option<String> {
if self.0 != 0 { if self.0 != 0 {
raw_input::get_raw_input_device_name(self.0 as _) raw_input::get_raw_input_device_name(self.0 as HANDLE)
} else { } else {
None None
} }
@ -91,15 +94,59 @@ unsafe impl Sync for WindowId {}
impl WindowId { impl WindowId {
pub const unsafe fn dummy() -> Self { pub const unsafe fn dummy() -> Self {
use std::ptr::null_mut; WindowId(0)
WindowId(null_mut())
} }
} }
#[inline(always)]
const fn get_xbutton_wparam(x: u32) -> u16 {
loword(x)
}
#[inline(always)]
const fn get_x_lparam(x: u32) -> u16 {
loword(x)
}
#[inline(always)]
const fn get_y_lparam(x: u32) -> u16 {
hiword(x)
}
#[inline(always)]
const fn loword(x: u32) -> u16 {
(x & 0xFFFF) as u16
}
#[inline(always)]
const fn hiword(x: u32) -> u16 {
((x >> 16) & 0xFFFF) as u16
}
#[inline(always)]
unsafe fn get_window_long(hwnd: HWND, nindex: WINDOW_LONG_PTR_INDEX) -> isize {
#[cfg(target_pointer_width = "64")]
return windows_sys::Win32::UI::WindowsAndMessaging::GetWindowLongPtrW(hwnd, nindex);
#[cfg(target_pointer_width = "32")]
return windows_sys::Win32::UI::WindowsAndMessaging::GetWindowLongW(hwnd, nindex) as isize;
}
#[inline(always)]
unsafe fn set_window_long(hwnd: HWND, nindex: WINDOW_LONG_PTR_INDEX, dwnewlong: isize) -> isize {
#[cfg(target_pointer_width = "64")]
return windows_sys::Win32::UI::WindowsAndMessaging::SetWindowLongPtrW(hwnd, nindex, dwnewlong);
#[cfg(target_pointer_width = "32")]
return windows_sys::Win32::UI::WindowsAndMessaging::SetWindowLongW(
hwnd,
nindex,
dwnewlong as i32,
) as isize;
}
#[macro_use] #[macro_use]
mod util; mod util;
mod dark_mode; mod dark_mode;
mod definitions;
mod dpi; mod dpi;
mod drop_handler; mod drop_handler;
mod event; mod event;

View file

@ -1,22 +1,28 @@
use winapi::{
shared::{
minwindef::{BOOL, DWORD, LPARAM, TRUE, WORD},
windef::{HDC, HMONITOR, HWND, LPRECT, POINT},
},
um::{wingdi, winuser},
};
use std::{ use std::{
collections::{BTreeSet, VecDeque}, collections::{BTreeSet, VecDeque},
io, mem, ptr, ffi::OsString,
hash::Hash,
io, mem,
os::windows::prelude::OsStringExt,
ptr,
};
use windows_sys::Win32::{
Foundation::{BOOL, HWND, LPARAM, POINT, RECT},
Graphics::Gdi::{
EnumDisplayMonitors, EnumDisplaySettingsExW, GetMonitorInfoW, MonitorFromPoint,
MonitorFromWindow, DEVMODEW, DM_BITSPERPEL, DM_DISPLAYFREQUENCY, DM_PELSHEIGHT,
DM_PELSWIDTH, HDC, HMONITOR, MONITORINFO, MONITORINFOEXW, MONITOR_DEFAULTTONEAREST,
MONITOR_DEFAULTTOPRIMARY,
},
}; };
use super::util;
use crate::{ use crate::{
dpi::{PhysicalPosition, PhysicalSize}, dpi::{PhysicalPosition, PhysicalSize},
monitor::{MonitorHandle as RootMonitorHandle, VideoMode as RootVideoMode}, monitor::{MonitorHandle as RootMonitorHandle, VideoMode as RootVideoMode},
platform_impl::platform::{ platform_impl::platform::{
dpi::{dpi_to_scale_factor, get_monitor_dpi}, dpi::{dpi_to_scale_factor, get_monitor_dpi},
util::has_flag,
window::Window, window::Window,
}, },
}; };
@ -27,7 +33,7 @@ pub struct VideoMode {
pub(crate) bit_depth: u16, pub(crate) bit_depth: u16,
pub(crate) refresh_rate: u16, pub(crate) refresh_rate: u16,
pub(crate) monitor: MonitorHandle, pub(crate) monitor: MonitorHandle,
pub(crate) native_video_mode: wingdi::DEVMODEW, pub(crate) native_video_mode: DEVMODEW,
} }
impl PartialEq for VideoMode { impl PartialEq for VideoMode {
@ -94,20 +100,20 @@ unsafe impl Send for MonitorHandle {}
unsafe extern "system" fn monitor_enum_proc( unsafe extern "system" fn monitor_enum_proc(
hmonitor: HMONITOR, hmonitor: HMONITOR,
_hdc: HDC, _hdc: HDC,
_place: LPRECT, _place: *mut RECT,
data: LPARAM, data: LPARAM,
) -> BOOL { ) -> BOOL {
let monitors = data as *mut VecDeque<MonitorHandle>; let monitors = data as *mut VecDeque<MonitorHandle>;
(*monitors).push_back(MonitorHandle::new(hmonitor)); (*monitors).push_back(MonitorHandle::new(hmonitor));
TRUE // continue enumeration true.into() // continue enumeration
} }
pub fn available_monitors() -> VecDeque<MonitorHandle> { pub fn available_monitors() -> VecDeque<MonitorHandle> {
let mut monitors: VecDeque<MonitorHandle> = VecDeque::new(); let mut monitors: VecDeque<MonitorHandle> = VecDeque::new();
unsafe { unsafe {
winuser::EnumDisplayMonitors( EnumDisplayMonitors(
ptr::null_mut(), 0,
ptr::null_mut(), ptr::null(),
Some(monitor_enum_proc), Some(monitor_enum_proc),
&mut monitors as *mut _ as LPARAM, &mut monitors as *mut _ as LPARAM,
); );
@ -117,12 +123,12 @@ pub fn available_monitors() -> VecDeque<MonitorHandle> {
pub fn primary_monitor() -> MonitorHandle { pub fn primary_monitor() -> MonitorHandle {
const ORIGIN: POINT = POINT { x: 0, y: 0 }; const ORIGIN: POINT = POINT { x: 0, y: 0 };
let hmonitor = unsafe { winuser::MonitorFromPoint(ORIGIN, winuser::MONITOR_DEFAULTTOPRIMARY) }; let hmonitor = unsafe { MonitorFromPoint(ORIGIN, MONITOR_DEFAULTTOPRIMARY) };
MonitorHandle::new(hmonitor) MonitorHandle::new(hmonitor)
} }
pub fn current_monitor(hwnd: HWND) -> MonitorHandle { pub fn current_monitor(hwnd: HWND) -> MonitorHandle {
let hmonitor = unsafe { winuser::MonitorFromWindow(hwnd, winuser::MONITOR_DEFAULTTONEAREST) }; let hmonitor = unsafe { MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST) };
MonitorHandle::new(hmonitor) MonitorHandle::new(hmonitor)
} }
@ -137,16 +143,16 @@ impl Window {
} }
} }
pub(crate) fn get_monitor_info(hmonitor: HMONITOR) -> Result<winuser::MONITORINFOEXW, io::Error> { pub(crate) fn get_monitor_info(hmonitor: HMONITOR) -> Result<MONITORINFOEXW, io::Error> {
let mut monitor_info: winuser::MONITORINFOEXW = unsafe { mem::zeroed() }; let mut monitor_info: MONITORINFOEXW = unsafe { mem::zeroed() };
monitor_info.cbSize = mem::size_of::<winuser::MONITORINFOEXW>() as DWORD; monitor_info.monitorInfo.cbSize = mem::size_of::<MONITORINFOEXW>() as u32;
let status = unsafe { let status = unsafe {
winuser::GetMonitorInfoW( GetMonitorInfoW(
hmonitor, hmonitor,
&mut monitor_info as *mut winuser::MONITORINFOEXW as *mut winuser::MONITORINFO, &mut monitor_info as *mut MONITORINFOEXW as *mut MONITORINFO,
) )
}; };
if status == 0 { if status == false.into() {
Err(io::Error::last_os_error()) Err(io::Error::last_os_error())
} else { } else {
Ok(monitor_info) Ok(monitor_info)
@ -161,7 +167,11 @@ impl MonitorHandle {
#[inline] #[inline]
pub fn name(&self) -> Option<String> { pub fn name(&self) -> Option<String> {
let monitor_info = get_monitor_info(self.0).unwrap(); let monitor_info = get_monitor_info(self.0).unwrap();
Some(util::wchar_ptr_to_string(monitor_info.szDevice.as_ptr())) Some(
OsString::from_wide(&monitor_info.szDevice)
.to_string_lossy()
.to_string(),
)
} }
#[inline] #[inline]
@ -176,19 +186,19 @@ impl MonitorHandle {
#[inline] #[inline]
pub fn size(&self) -> PhysicalSize<u32> { pub fn size(&self) -> PhysicalSize<u32> {
let monitor_info = get_monitor_info(self.0).unwrap(); let rc_monitor = get_monitor_info(self.0).unwrap().monitorInfo.rcMonitor;
PhysicalSize { PhysicalSize {
width: (monitor_info.rcMonitor.right - monitor_info.rcMonitor.left) as u32, width: (rc_monitor.right - rc_monitor.left) as u32,
height: (monitor_info.rcMonitor.bottom - monitor_info.rcMonitor.top) as u32, height: (rc_monitor.bottom - rc_monitor.top) as u32,
} }
} }
#[inline] #[inline]
pub fn position(&self) -> PhysicalPosition<i32> { pub fn position(&self) -> PhysicalPosition<i32> {
let monitor_info = get_monitor_info(self.0).unwrap(); let rc_monitor = get_monitor_info(self.0).unwrap().monitorInfo.rcMonitor;
PhysicalPosition { PhysicalPosition {
x: monitor_info.rcMonitor.left, x: rc_monitor.left,
y: monitor_info.rcMonitor.top, y: rc_monitor.top,
} }
} }
@ -209,18 +219,16 @@ impl MonitorHandle {
unsafe { unsafe {
let monitor_info = get_monitor_info(self.0).unwrap(); let monitor_info = get_monitor_info(self.0).unwrap();
let device_name = monitor_info.szDevice.as_ptr(); let device_name = monitor_info.szDevice.as_ptr();
let mut mode: wingdi::DEVMODEW = mem::zeroed(); let mut mode: DEVMODEW = mem::zeroed();
mode.dmSize = mem::size_of_val(&mode) as WORD; mode.dmSize = mem::size_of_val(&mode) as u16;
if winuser::EnumDisplaySettingsExW(device_name, i, &mut mode, 0) == 0 { if EnumDisplaySettingsExW(device_name, i, &mut mode, 0) == false.into() {
break; break;
} }
i += 1; i += 1;
const REQUIRED_FIELDS: DWORD = wingdi::DM_BITSPERPEL const REQUIRED_FIELDS: u32 =
| wingdi::DM_PELSWIDTH (DM_BITSPERPEL | DM_PELSWIDTH | DM_PELSHEIGHT | DM_DISPLAYFREQUENCY) as u32;
| wingdi::DM_PELSHEIGHT assert!(has_flag(mode.dmFields, REQUIRED_FIELDS));
| wingdi::DM_DISPLAYFREQUENCY;
assert!(mode.dmFields & REQUIRED_FIELDS == REQUIRED_FIELDS);
modes.insert(RootVideoMode { modes.insert(RootVideoMode {
video_mode: VideoMode { video_mode: VideoMode {

View file

@ -1,22 +1,26 @@
use std::{ use std::{
ffi::OsString,
mem::{self, size_of}, mem::{self, size_of},
os::windows::prelude::OsStringExt,
ptr, ptr,
}; };
use winapi::{ use windows_sys::Win32::{
ctypes::wchar_t, Devices::HumanInterfaceDevice::{
shared::{ HID_USAGE_GENERIC_KEYBOARD, HID_USAGE_GENERIC_MOUSE, HID_USAGE_PAGE_GENERIC,
hidusage::{HID_USAGE_GENERIC_KEYBOARD, HID_USAGE_GENERIC_MOUSE, HID_USAGE_PAGE_GENERIC},
minwindef::{TRUE, UINT, USHORT},
windef::HWND,
}, },
um::{ Foundation::{HANDLE, HWND},
winnt::HANDLE, UI::{
winuser::{ Input::{
self, HRAWINPUT, RAWINPUT, RAWINPUTDEVICE, RAWINPUTDEVICELIST, RAWINPUTHEADER, GetRawInputData, GetRawInputDeviceInfoW, GetRawInputDeviceList,
RIDEV_DEVNOTIFY, RIDEV_INPUTSINK, RIDI_DEVICEINFO, RIDI_DEVICENAME, RID_DEVICE_INFO, RegisterRawInputDevices, HRAWINPUT, RAWINPUT, RAWINPUTDEVICE, RAWINPUTDEVICELIST,
RID_DEVICE_INFO_HID, RID_DEVICE_INFO_KEYBOARD, RID_DEVICE_INFO_MOUSE, RID_INPUT, RAWINPUTHEADER, RIDEV_DEVNOTIFY, RIDEV_INPUTSINK, RIDI_DEVICEINFO, RIDI_DEVICENAME,
RIM_TYPEHID, RIM_TYPEKEYBOARD, RIM_TYPEMOUSE, RID_DEVICE_INFO, RID_DEVICE_INFO_HID, RID_DEVICE_INFO_KEYBOARD, RID_DEVICE_INFO_MOUSE,
RID_INPUT, RIM_TYPEHID, RIM_TYPEKEYBOARD, RIM_TYPEMOUSE,
},
WindowsAndMessaging::{
RI_MOUSE_LEFT_BUTTON_DOWN, RI_MOUSE_LEFT_BUTTON_UP, RI_MOUSE_MIDDLE_BUTTON_DOWN,
RI_MOUSE_MIDDLE_BUTTON_UP, RI_MOUSE_RIGHT_BUTTON_DOWN, RI_MOUSE_RIGHT_BUTTON_UP,
}, },
}, },
}; };
@ -25,23 +29,21 @@ use crate::{event::ElementState, platform_impl::platform::util};
#[allow(dead_code)] #[allow(dead_code)]
pub fn get_raw_input_device_list() -> Option<Vec<RAWINPUTDEVICELIST>> { pub fn get_raw_input_device_list() -> Option<Vec<RAWINPUTDEVICELIST>> {
let list_size = size_of::<RAWINPUTDEVICELIST>() as UINT; let list_size = size_of::<RAWINPUTDEVICELIST>() as u32;
let mut num_devices = 0; let mut num_devices = 0;
let status = let status = unsafe { GetRawInputDeviceList(ptr::null_mut(), &mut num_devices, list_size) };
unsafe { winuser::GetRawInputDeviceList(ptr::null_mut(), &mut num_devices, list_size) };
if status == UINT::max_value() { if status == u32::MAX {
return None; return None;
} }
let mut buffer = Vec::with_capacity(num_devices as _); let mut buffer = Vec::with_capacity(num_devices as _);
let num_stored = unsafe { let num_stored =
winuser::GetRawInputDeviceList(buffer.as_ptr() as _, &mut num_devices, list_size) unsafe { GetRawInputDeviceList(buffer.as_mut_ptr(), &mut num_devices, list_size) };
};
if num_stored == UINT::max_value() { if num_stored == u32::MAX {
return None; return None;
} }
@ -63,9 +65,9 @@ impl From<RID_DEVICE_INFO> for RawDeviceInfo {
fn from(info: RID_DEVICE_INFO) -> Self { fn from(info: RID_DEVICE_INFO) -> Self {
unsafe { unsafe {
match info.dwType { match info.dwType {
RIM_TYPEMOUSE => RawDeviceInfo::Mouse(*info.u.mouse()), RIM_TYPEMOUSE => RawDeviceInfo::Mouse(info.Anonymous.mouse),
RIM_TYPEKEYBOARD => RawDeviceInfo::Keyboard(*info.u.keyboard()), RIM_TYPEKEYBOARD => RawDeviceInfo::Keyboard(info.Anonymous.keyboard),
RIM_TYPEHID => RawDeviceInfo::Hid(*info.u.hid()), RIM_TYPEHID => RawDeviceInfo::Hid(info.Anonymous.hid),
_ => unreachable!(), _ => unreachable!(),
} }
} }
@ -75,13 +77,13 @@ impl From<RID_DEVICE_INFO> for RawDeviceInfo {
#[allow(dead_code)] #[allow(dead_code)]
pub fn get_raw_input_device_info(handle: HANDLE) -> Option<RawDeviceInfo> { pub fn get_raw_input_device_info(handle: HANDLE) -> Option<RawDeviceInfo> {
let mut info: RID_DEVICE_INFO = unsafe { mem::zeroed() }; let mut info: RID_DEVICE_INFO = unsafe { mem::zeroed() };
let info_size = size_of::<RID_DEVICE_INFO>() as UINT; let info_size = size_of::<RID_DEVICE_INFO>() as u32;
info.cbSize = info_size; info.cbSize = info_size;
let mut minimum_size = 0; let mut minimum_size = 0;
let status = unsafe { let status = unsafe {
winuser::GetRawInputDeviceInfoW( GetRawInputDeviceInfoW(
handle, handle,
RIDI_DEVICEINFO, RIDI_DEVICEINFO,
&mut info as *mut _ as _, &mut info as *mut _ as _,
@ -89,7 +91,7 @@ pub fn get_raw_input_device_info(handle: HANDLE) -> Option<RawDeviceInfo> {
) )
}; };
if status == UINT::max_value() || status == 0 { if status == u32::MAX || status == 0 {
return None; return None;
} }
@ -101,17 +103,17 @@ pub fn get_raw_input_device_info(handle: HANDLE) -> Option<RawDeviceInfo> {
pub fn get_raw_input_device_name(handle: HANDLE) -> Option<String> { pub fn get_raw_input_device_name(handle: HANDLE) -> Option<String> {
let mut minimum_size = 0; let mut minimum_size = 0;
let status = unsafe { let status = unsafe {
winuser::GetRawInputDeviceInfoW(handle, RIDI_DEVICENAME, ptr::null_mut(), &mut minimum_size) GetRawInputDeviceInfoW(handle, RIDI_DEVICENAME, ptr::null_mut(), &mut minimum_size)
}; };
if status != 0 { if status != 0 {
return None; return None;
} }
let mut name: Vec<wchar_t> = Vec::with_capacity(minimum_size as _); let mut name: Vec<u16> = Vec::with_capacity(minimum_size as _);
let status = unsafe { let status = unsafe {
winuser::GetRawInputDeviceInfoW( GetRawInputDeviceInfoW(
handle, handle,
RIDI_DEVICENAME, RIDI_DEVICENAME,
name.as_ptr() as _, name.as_ptr() as _,
@ -119,7 +121,7 @@ pub fn get_raw_input_device_name(handle: HANDLE) -> Option<String> {
) )
}; };
if status == UINT::max_value() || status == 0 { if status == u32::MAX || status == 0 {
return None; return None;
} }
@ -127,17 +129,15 @@ pub fn get_raw_input_device_name(handle: HANDLE) -> Option<String> {
unsafe { name.set_len(minimum_size as _) }; unsafe { name.set_len(minimum_size as _) };
Some(util::wchar_to_string(&name)) OsString::from_wide(&name).into_string().ok()
} }
pub fn register_raw_input_devices(devices: &[RAWINPUTDEVICE]) -> bool { pub fn register_raw_input_devices(devices: &[RAWINPUTDEVICE]) -> bool {
let device_size = size_of::<RAWINPUTDEVICE>() as UINT; let device_size = size_of::<RAWINPUTDEVICE>() as u32;
let success = unsafe { unsafe {
winuser::RegisterRawInputDevices(devices.as_ptr() as _, devices.len() as _, device_size) RegisterRawInputDevices(devices.as_ptr(), devices.len() as u32, device_size) == true.into()
}; }
success == TRUE
} }
pub fn register_all_mice_and_keyboards_for_raw_input(window_handle: HWND) -> bool { pub fn register_all_mice_and_keyboards_for_raw_input(window_handle: HWND) -> bool {
@ -165,11 +165,11 @@ pub fn register_all_mice_and_keyboards_for_raw_input(window_handle: HWND) -> boo
pub fn get_raw_input_data(handle: HRAWINPUT) -> Option<RAWINPUT> { pub fn get_raw_input_data(handle: HRAWINPUT) -> Option<RAWINPUT> {
let mut data: RAWINPUT = unsafe { mem::zeroed() }; let mut data: RAWINPUT = unsafe { mem::zeroed() };
let mut data_size = size_of::<RAWINPUT>() as UINT; let mut data_size = size_of::<RAWINPUT>() as u32;
let header_size = size_of::<RAWINPUTHEADER>() as UINT; let header_size = size_of::<RAWINPUTHEADER>() as u32;
let status = unsafe { let status = unsafe {
winuser::GetRawInputData( GetRawInputData(
handle, handle,
RID_INPUT, RID_INPUT,
&mut data as *mut _ as _, &mut data as *mut _ as _,
@ -178,7 +178,7 @@ pub fn get_raw_input_data(handle: HRAWINPUT) -> Option<RAWINPUT> {
) )
}; };
if status == UINT::max_value() || status == 0 { if status == u32::MAX || status == 0 {
return None; return None;
} }
@ -186,9 +186,9 @@ pub fn get_raw_input_data(handle: HRAWINPUT) -> Option<RAWINPUT> {
} }
fn button_flags_to_element_state( fn button_flags_to_element_state(
button_flags: USHORT, button_flags: u32,
down_flag: USHORT, down_flag: u32,
up_flag: USHORT, up_flag: u32,
) -> Option<ElementState> { ) -> Option<ElementState> {
// We assume the same button won't be simultaneously pressed and released. // We assume the same button won't be simultaneously pressed and released.
if util::has_flag(button_flags, down_flag) { if util::has_flag(button_flags, down_flag) {
@ -200,22 +200,22 @@ fn button_flags_to_element_state(
} }
} }
pub fn get_raw_mouse_button_state(button_flags: USHORT) -> [Option<ElementState>; 3] { pub fn get_raw_mouse_button_state(button_flags: u32) -> [Option<ElementState>; 3] {
[ [
button_flags_to_element_state( button_flags_to_element_state(
button_flags, button_flags,
winuser::RI_MOUSE_LEFT_BUTTON_DOWN, RI_MOUSE_LEFT_BUTTON_DOWN,
winuser::RI_MOUSE_LEFT_BUTTON_UP, RI_MOUSE_LEFT_BUTTON_UP,
), ),
button_flags_to_element_state( button_flags_to_element_state(
button_flags, button_flags,
winuser::RI_MOUSE_MIDDLE_BUTTON_DOWN, RI_MOUSE_MIDDLE_BUTTON_DOWN,
winuser::RI_MOUSE_MIDDLE_BUTTON_UP, RI_MOUSE_MIDDLE_BUTTON_UP,
), ),
button_flags_to_element_state( button_flags_to_element_state(
button_flags, button_flags,
winuser::RI_MOUSE_RIGHT_BUTTON_DOWN, RI_MOUSE_RIGHT_BUTTON_DOWN,
winuser::RI_MOUSE_RIGHT_BUTTON_UP, RI_MOUSE_RIGHT_BUTTON_UP,
), ),
] ]
} }

View file

@ -1,27 +1,42 @@
use std::{ use std::{
io, mem, ffi::{c_void, OsStr},
io,
iter::once,
mem,
ops::BitAnd, ops::BitAnd,
os::raw::c_void, os::windows::prelude::OsStrExt,
ptr, slice, ptr,
sync::atomic::{AtomicBool, Ordering}, sync::atomic::{AtomicBool, Ordering},
}; };
use crate::{dpi::PhysicalSize, window::CursorIcon}; use windows_sys::{
use winapi::{ core::{HRESULT, PCWSTR},
ctypes::wchar_t, Win32::{
shared::{ Foundation::{BOOL, HWND, RECT},
minwindef::{BOOL, DWORD, UINT}, Graphics::Gdi::{ClientToScreen, InvalidateRgn, HMONITOR},
windef::{DPI_AWARENESS_CONTEXT, HMONITOR, HWND, LPRECT, RECT}, System::LibraryLoader::{GetProcAddress, LoadLibraryA},
UI::{
HiDpi::{DPI_AWARENESS_CONTEXT, MONITOR_DPI_TYPE, PROCESS_DPI_AWARENESS},
Input::KeyboardAndMouse::GetActiveWindow,
WindowsAndMessaging::{
AdjustWindowRectEx, ClipCursor, GetClientRect, GetClipCursor, GetMenu,
GetSystemMetrics, GetWindowLongW, GetWindowRect, SetWindowPos, ShowCursor,
GWL_EXSTYLE, GWL_STYLE, IDC_APPSTARTING, IDC_ARROW, IDC_CROSS, IDC_HAND, IDC_HELP,
IDC_IBEAM, IDC_NO, IDC_SIZEALL, IDC_SIZENESW, IDC_SIZENS, IDC_SIZENWSE, IDC_SIZEWE,
IDC_WAIT, SM_CXVIRTUALSCREEN, SM_CYVIRTUALSCREEN, SM_XVIRTUALSCREEN,
SM_YVIRTUALSCREEN, SWP_ASYNCWINDOWPOS, SWP_NOACTIVATE, SWP_NOMOVE,
SWP_NOREPOSITION, SWP_NOZORDER, WINDOW_EX_STYLE, WINDOW_STYLE,
},
}, },
um::{
libloaderapi::{GetProcAddress, LoadLibraryA},
shellscalingapi::{MONITOR_DPI_TYPE, PROCESS_DPI_AWARENESS},
winbase::lstrlenW,
winnt::{HRESULT, LONG, LPCSTR},
winuser,
}, },
}; };
use crate::{dpi::PhysicalSize, window::CursorIcon};
pub fn encode_wide(string: impl AsRef<OsStr>) -> Vec<u16> {
string.as_ref().encode_wide().chain(once(0)).collect()
}
pub fn has_flag<T>(bitset: T, flag: T) -> bool pub fn has_flag<T>(bitset: T, flag: T) -> bool
where where
T: Copy + PartialEq + BitAnd<T, Output = T>, T: Copy + PartialEq + BitAnd<T, Output = T>,
@ -29,19 +44,9 @@ where
bitset & flag == flag bitset & flag == flag
} }
pub fn wchar_to_string(wchar: &[wchar_t]) -> String {
String::from_utf16_lossy(wchar)
}
pub fn wchar_ptr_to_string(wchar: *const wchar_t) -> String {
let len = unsafe { lstrlenW(wchar) } as usize;
let wchar_slice = unsafe { slice::from_raw_parts(wchar, len) };
wchar_to_string(wchar_slice)
}
pub unsafe fn status_map<T, F: FnMut(&mut T) -> BOOL>(mut fun: F) -> Option<T> { pub unsafe fn status_map<T, F: FnMut(&mut T) -> BOOL>(mut fun: F) -> Option<T> {
let mut data: T = mem::zeroed(); let mut data: T = mem::zeroed();
if fun(&mut data) != 0 { if fun(&mut data) != false.into() {
Some(data) Some(data)
} else { } else {
None None
@ -49,7 +54,7 @@ pub unsafe fn status_map<T, F: FnMut(&mut T) -> BOOL>(mut fun: F) -> Option<T> {
} }
fn win_to_err<F: FnOnce() -> BOOL>(f: F) -> Result<(), io::Error> { fn win_to_err<F: FnOnce() -> BOOL>(f: F) -> Result<(), io::Error> {
if f() != 0 { if f() != false.into() {
Ok(()) Ok(())
} else { } else {
Err(io::Error::last_os_error()) Err(io::Error::last_os_error())
@ -57,7 +62,7 @@ fn win_to_err<F: FnOnce() -> BOOL>(f: F) -> Result<(), io::Error> {
} }
pub fn get_window_rect(hwnd: HWND) -> Option<RECT> { pub fn get_window_rect(hwnd: HWND) -> Option<RECT> {
unsafe { status_map(|rect| winuser::GetWindowRect(hwnd, rect)) } unsafe { status_map(|rect| GetWindowRect(hwnd, rect)) }
} }
pub fn get_client_rect(hwnd: HWND) -> Result<RECT, io::Error> { pub fn get_client_rect(hwnd: HWND) -> Result<RECT, io::Error> {
@ -65,8 +70,8 @@ pub fn get_client_rect(hwnd: HWND) -> Result<RECT, io::Error> {
let mut rect = mem::zeroed(); let mut rect = mem::zeroed();
let mut top_left = mem::zeroed(); let mut top_left = mem::zeroed();
win_to_err(|| winuser::ClientToScreen(hwnd, &mut top_left))?; win_to_err(|| ClientToScreen(hwnd, &mut top_left))?;
win_to_err(|| winuser::GetClientRect(hwnd, &mut rect))?; win_to_err(|| GetClientRect(hwnd, &mut rect))?;
rect.left += top_left.x; rect.left += top_left.x;
rect.top += top_left.y; rect.top += top_left.y;
rect.right += top_left.x; rect.right += top_left.x;
@ -80,9 +85,9 @@ pub fn adjust_size(hwnd: HWND, size: PhysicalSize<u32>) -> PhysicalSize<u32> {
let (width, height): (u32, u32) = size.into(); let (width, height): (u32, u32) = size.into();
let rect = RECT { let rect = RECT {
left: 0, left: 0,
right: width as LONG, right: width as i32,
top: 0, top: 0,
bottom: height as LONG, bottom: height as i32,
}; };
let rect = adjust_window_rect(hwnd, rect).unwrap_or(rect); let rect = adjust_window_rect(hwnd, rect).unwrap_or(rect);
PhysicalSize::new((rect.right - rect.left) as _, (rect.bottom - rect.top) as _) PhysicalSize::new((rect.right - rect.left) as _, (rect.bottom - rect.top) as _)
@ -95,57 +100,53 @@ pub(crate) fn set_inner_size_physical(window: HWND, x: u32, y: u32) {
RECT { RECT {
top: 0, top: 0,
left: 0, left: 0,
bottom: y as LONG, bottom: y as i32,
right: x as LONG, right: x as i32,
}, },
) )
.expect("adjust_window_rect failed"); .expect("adjust_window_rect failed");
let outer_x = (rect.right - rect.left).abs() as _; let outer_x = (rect.right - rect.left).abs() as _;
let outer_y = (rect.top - rect.bottom).abs() as _; let outer_y = (rect.top - rect.bottom).abs() as _;
winuser::SetWindowPos( SetWindowPos(
window, window,
ptr::null_mut(), 0,
0, 0,
0, 0,
outer_x, outer_x,
outer_y, outer_y,
winuser::SWP_ASYNCWINDOWPOS SWP_ASYNCWINDOWPOS | SWP_NOZORDER | SWP_NOREPOSITION | SWP_NOMOVE | SWP_NOACTIVATE,
| winuser::SWP_NOZORDER
| winuser::SWP_NOREPOSITION
| winuser::SWP_NOMOVE
| winuser::SWP_NOACTIVATE,
); );
winuser::InvalidateRgn(window, ptr::null_mut(), 0); InvalidateRgn(window, 0, false.into());
} }
} }
pub fn adjust_window_rect(hwnd: HWND, rect: RECT) -> Option<RECT> { pub fn adjust_window_rect(hwnd: HWND, rect: RECT) -> Option<RECT> {
unsafe { unsafe {
let style = winuser::GetWindowLongW(hwnd, winuser::GWL_STYLE); let style = GetWindowLongW(hwnd, GWL_STYLE) as u32;
let style_ex = winuser::GetWindowLongW(hwnd, winuser::GWL_EXSTYLE); let style_ex = GetWindowLongW(hwnd, GWL_EXSTYLE) as u32;
adjust_window_rect_with_styles(hwnd, style as _, style_ex as _, rect) adjust_window_rect_with_styles(hwnd, style, style_ex, rect)
} }
} }
pub fn adjust_window_rect_with_styles( pub fn adjust_window_rect_with_styles(
hwnd: HWND, hwnd: HWND,
style: DWORD, style: WINDOW_STYLE,
style_ex: DWORD, style_ex: WINDOW_EX_STYLE,
rect: RECT, rect: RECT,
) -> Option<RECT> { ) -> Option<RECT> {
unsafe { unsafe {
status_map(|r| { status_map(|r| {
*r = rect; *r = rect;
let b_menu = !winuser::GetMenu(hwnd).is_null() as BOOL; let b_menu = GetMenu(hwnd) != 0;
if let (Some(get_dpi_for_window), Some(adjust_window_rect_ex_for_dpi)) = if let (Some(get_dpi_for_window), Some(adjust_window_rect_ex_for_dpi)) =
(*GET_DPI_FOR_WINDOW, *ADJUST_WINDOW_RECT_EX_FOR_DPI) (*GET_DPI_FOR_WINDOW, *ADJUST_WINDOW_RECT_EX_FOR_DPI)
{ {
let dpi = get_dpi_for_window(hwnd); let dpi = get_dpi_for_window(hwnd);
adjust_window_rect_ex_for_dpi(r, style as _, b_menu, style_ex as _, dpi) adjust_window_rect_ex_for_dpi(r, style, b_menu.into(), style_ex, dpi)
} else { } else {
winuser::AdjustWindowRectEx(r, style as _, b_menu, style_ex as _) AdjustWindowRectEx(r, style, b_menu.into(), style_ex)
} }
}) })
} }
@ -155,14 +156,14 @@ pub fn set_cursor_hidden(hidden: bool) {
static HIDDEN: AtomicBool = AtomicBool::new(false); static HIDDEN: AtomicBool = AtomicBool::new(false);
let changed = HIDDEN.swap(hidden, Ordering::SeqCst) ^ hidden; let changed = HIDDEN.swap(hidden, Ordering::SeqCst) ^ hidden;
if changed { if changed {
unsafe { winuser::ShowCursor(!hidden as BOOL) }; unsafe { ShowCursor(BOOL::from(!hidden)) };
} }
} }
pub fn get_cursor_clip() -> Result<RECT, io::Error> { pub fn get_cursor_clip() -> Result<RECT, io::Error> {
unsafe { unsafe {
let mut rect: RECT = mem::zeroed(); let mut rect: RECT = mem::zeroed();
win_to_err(|| winuser::GetClipCursor(&mut rect)).map(|_| rect) win_to_err(|| GetClipCursor(&mut rect)).map(|_| rect)
} }
} }
@ -175,56 +176,52 @@ pub fn set_cursor_clip(rect: Option<RECT>) -> Result<(), io::Error> {
.as_ref() .as_ref()
.map(|r| r as *const RECT) .map(|r| r as *const RECT)
.unwrap_or(ptr::null()); .unwrap_or(ptr::null());
win_to_err(|| winuser::ClipCursor(rect_ptr)) win_to_err(|| ClipCursor(rect_ptr))
} }
} }
pub fn get_desktop_rect() -> RECT { pub fn get_desktop_rect() -> RECT {
unsafe { unsafe {
let left = winuser::GetSystemMetrics(winuser::SM_XVIRTUALSCREEN); let left = GetSystemMetrics(SM_XVIRTUALSCREEN);
let top = winuser::GetSystemMetrics(winuser::SM_YVIRTUALSCREEN); let top = GetSystemMetrics(SM_YVIRTUALSCREEN);
RECT { RECT {
left, left,
top, top,
right: left + winuser::GetSystemMetrics(winuser::SM_CXVIRTUALSCREEN), right: left + GetSystemMetrics(SM_CXVIRTUALSCREEN),
bottom: top + winuser::GetSystemMetrics(winuser::SM_CYVIRTUALSCREEN), bottom: top + GetSystemMetrics(SM_CYVIRTUALSCREEN),
} }
} }
} }
pub fn is_focused(window: HWND) -> bool { pub fn is_focused(window: HWND) -> bool {
window == unsafe { winuser::GetActiveWindow() } window == unsafe { GetActiveWindow() }
} }
impl CursorIcon { impl CursorIcon {
pub(crate) fn to_windows_cursor(self) -> *const wchar_t { pub(crate) fn to_windows_cursor(self) -> PCWSTR {
match self { match self {
CursorIcon::Arrow | CursorIcon::Default => winuser::IDC_ARROW, CursorIcon::Arrow | CursorIcon::Default => IDC_ARROW,
CursorIcon::Hand => winuser::IDC_HAND, CursorIcon::Hand => IDC_HAND,
CursorIcon::Crosshair => winuser::IDC_CROSS, CursorIcon::Crosshair => IDC_CROSS,
CursorIcon::Text | CursorIcon::VerticalText => winuser::IDC_IBEAM, CursorIcon::Text | CursorIcon::VerticalText => IDC_IBEAM,
CursorIcon::NotAllowed | CursorIcon::NoDrop => winuser::IDC_NO, CursorIcon::NotAllowed | CursorIcon::NoDrop => IDC_NO,
CursorIcon::Grab | CursorIcon::Grabbing | CursorIcon::Move | CursorIcon::AllScroll => { CursorIcon::Grab | CursorIcon::Grabbing | CursorIcon::Move | CursorIcon::AllScroll => {
winuser::IDC_SIZEALL IDC_SIZEALL
} }
CursorIcon::EResize CursorIcon::EResize
| CursorIcon::WResize | CursorIcon::WResize
| CursorIcon::EwResize | CursorIcon::EwResize
| CursorIcon::ColResize => winuser::IDC_SIZEWE, | CursorIcon::ColResize => IDC_SIZEWE,
CursorIcon::NResize CursorIcon::NResize
| CursorIcon::SResize | CursorIcon::SResize
| CursorIcon::NsResize | CursorIcon::NsResize
| CursorIcon::RowResize => winuser::IDC_SIZENS, | CursorIcon::RowResize => IDC_SIZENS,
CursorIcon::NeResize | CursorIcon::SwResize | CursorIcon::NeswResize => { CursorIcon::NeResize | CursorIcon::SwResize | CursorIcon::NeswResize => IDC_SIZENESW,
winuser::IDC_SIZENESW CursorIcon::NwResize | CursorIcon::SeResize | CursorIcon::NwseResize => IDC_SIZENWSE,
} CursorIcon::Wait => IDC_WAIT,
CursorIcon::NwResize | CursorIcon::SeResize | CursorIcon::NwseResize => { CursorIcon::Progress => IDC_APPSTARTING,
winuser::IDC_SIZENWSE CursorIcon::Help => IDC_HELP,
} _ => IDC_ARROW, // use arrow for the missing cases.
CursorIcon::Wait => winuser::IDC_WAIT,
CursorIcon::Progress => winuser::IDC_APPSTARTING,
CursorIcon::Help => winuser::IDC_HELP,
_ => winuser::IDC_ARROW, // use arrow for the missing cases.
} }
} }
} }
@ -236,17 +233,12 @@ pub(super) fn get_function_impl(library: &str, function: &str) -> Option<*const
assert_eq!(function.chars().last(), Some('\0')); assert_eq!(function.chars().last(), Some('\0'));
// Library names we will use are ASCII so we can use the A version to avoid string conversion. // Library names we will use are ASCII so we can use the A version to avoid string conversion.
let module = unsafe { LoadLibraryA(library.as_ptr() as LPCSTR) }; let module = unsafe { LoadLibraryA(library.as_ptr()) };
if module.is_null() { if module == 0 {
return None; return None;
} }
let function_ptr = unsafe { GetProcAddress(module, function.as_ptr() as LPCSTR) }; unsafe { GetProcAddress(module, function.as_ptr()) }.map(|function_ptr| function_ptr as _)
if function_ptr.is_null() {
return None;
}
Some(function_ptr as _)
} }
macro_rules! get_function { macro_rules! get_function {
@ -264,20 +256,20 @@ pub type SetProcessDpiAwareness =
unsafe extern "system" fn(value: PROCESS_DPI_AWARENESS) -> HRESULT; unsafe extern "system" fn(value: PROCESS_DPI_AWARENESS) -> HRESULT;
pub type SetProcessDpiAwarenessContext = pub type SetProcessDpiAwarenessContext =
unsafe extern "system" fn(value: DPI_AWARENESS_CONTEXT) -> BOOL; unsafe extern "system" fn(value: DPI_AWARENESS_CONTEXT) -> BOOL;
pub type GetDpiForWindow = unsafe extern "system" fn(hwnd: HWND) -> UINT; pub type GetDpiForWindow = unsafe extern "system" fn(hwnd: HWND) -> u32;
pub type GetDpiForMonitor = unsafe extern "system" fn( pub type GetDpiForMonitor = unsafe extern "system" fn(
hmonitor: HMONITOR, hmonitor: HMONITOR,
dpi_type: MONITOR_DPI_TYPE, dpi_type: MONITOR_DPI_TYPE,
dpi_x: *mut UINT, dpi_x: *mut u32,
dpi_y: *mut UINT, dpi_y: *mut u32,
) -> HRESULT; ) -> HRESULT;
pub type EnableNonClientDpiScaling = unsafe extern "system" fn(hwnd: HWND) -> BOOL; pub type EnableNonClientDpiScaling = unsafe extern "system" fn(hwnd: HWND) -> BOOL;
pub type AdjustWindowRectExForDpi = unsafe extern "system" fn( pub type AdjustWindowRectExForDpi = unsafe extern "system" fn(
rect: LPRECT, rect: *mut RECT,
dwStyle: DWORD, dwStyle: u32,
bMenu: BOOL, bMenu: BOOL,
dwExStyle: DWORD, dwExStyle: u32,
dpi: UINT, dpi: u32,
) -> BOOL; ) -> BOOL;
lazy_static! { lazy_static! {

View file

@ -4,31 +4,54 @@ use parking_lot::Mutex;
use raw_window_handle::{RawWindowHandle, Win32Handle}; use raw_window_handle::{RawWindowHandle, Win32Handle};
use std::{ use std::{
cell::Cell, cell::Cell,
ffi::OsStr, ffi::c_void,
io, mem, io, mem, panic, ptr,
os::windows::ffi::OsStrExt,
panic, ptr,
sync::{mpsc::channel, Arc}, sync::{mpsc::channel, Arc},
}; };
use winapi::{ use windows_sys::Win32::{
ctypes::c_int, Foundation::{
shared::{ HINSTANCE, HWND, LPARAM, OLE_E_WRONGCOMPOBJ, POINT, POINTS, RECT, RPC_E_CHANGED_MODE, S_OK,
minwindef::{HINSTANCE, LPARAM, UINT, WPARAM}, WPARAM,
windef::{HWND, POINT, POINTS, RECT}, },
winerror::SUCCEEDED, Graphics::{
Dwm::{DwmEnableBlurBehindWindow, DWM_BB_BLURREGION, DWM_BB_ENABLE, DWM_BLURBEHIND},
Gdi::{
ChangeDisplaySettingsExW, ClientToScreen, CreateRectRgn, DeleteObject, InvalidateRgn,
RedrawWindow, CDS_FULLSCREEN, DISP_CHANGE_BADFLAGS, DISP_CHANGE_BADMODE,
DISP_CHANGE_BADPARAM, DISP_CHANGE_FAILED, DISP_CHANGE_SUCCESSFUL, RDW_INTERNALPAINT,
},
},
System::{
Com::{
CoCreateInstance, CoInitializeEx, CoUninitialize, CLSCTX_ALL, COINIT_APARTMENTTHREADED,
},
LibraryLoader::GetModuleHandleW,
Ole::{OleInitialize, RegisterDragDrop},
},
UI::{
Input::{
Ime::{
ImmGetContext, ImmReleaseContext, ImmSetCompositionWindow, CFS_POINT,
COMPOSITIONFORM,
},
KeyboardAndMouse::{
EnableWindow, GetActiveWindow, MapVirtualKeyW, ReleaseCapture, SendInput, INPUT,
INPUT_0, INPUT_KEYBOARD, KEYBDINPUT, KEYEVENTF_EXTENDEDKEY, KEYEVENTF_KEYUP,
VK_LMENU, VK_MENU,
},
Touch::{RegisterTouchWindow, TWF_WANTPALM},
},
WindowsAndMessaging::{
CreateWindowExW, FlashWindowEx, GetClientRect, GetCursorPos, GetForegroundWindow,
GetSystemMetrics, GetWindowPlacement, IsWindowVisible, LoadCursorW, PeekMessageW,
PostMessageW, RegisterClassExW, SetCursor, SetCursorPos, SetForegroundWindow,
SetWindowPlacement, SetWindowPos, SetWindowTextW, CS_HREDRAW, CS_OWNDC, CS_VREDRAW,
CW_USEDEFAULT, FLASHWINFO, FLASHW_ALL, FLASHW_STOP, FLASHW_TIMERNOFG, FLASHW_TRAY,
GWLP_HINSTANCE, HTCAPTION, MAPVK_VK_TO_VSC, NID_READY, PM_NOREMOVE, SM_DIGITIZER,
SM_IMMENABLED, SWP_ASYNCWINDOWPOS, SWP_NOACTIVATE, SWP_NOSIZE, SWP_NOZORDER,
WM_NCLBUTTONDOWN, WNDCLASSEXW,
}, },
um::{
combaseapi, dwmapi,
imm::{CFS_POINT, COMPOSITIONFORM},
libloaderapi,
objbase::COINIT_APARTMENTTHREADED,
ole2,
oleidl::LPDROPTARGET,
shobjidl_core::{CLSID_TaskbarList, ITaskbarList2},
wingdi::{CreateRectRgn, DeleteObject},
winnt::{LPCWSTR, SHORT},
winuser,
}, },
}; };
@ -39,9 +62,11 @@ use crate::{
monitor::MonitorHandle as RootMonitorHandle, monitor::MonitorHandle as RootMonitorHandle,
platform_impl::platform::{ platform_impl::platform::{
dark_mode::try_theme, dark_mode::try_theme,
definitions::ITaskbarList2,
definitions::{CLSID_TaskbarList, IID_ITaskbarList2},
dpi::{dpi_to_scale_factor, enable_non_client_dpi_scaling, hwnd_dpi}, dpi::{dpi_to_scale_factor, enable_non_client_dpi_scaling, hwnd_dpi},
drop_handler::FileDropHandler, drop_handler::FileDropHandler,
event_loop::{self, EventLoopWindowTarget, WindowLongPtr, DESTROY_MSG_ID}, event_loop::{self, EventLoopWindowTarget, DESTROY_MSG_ID},
icon::{self, IconType}, icon::{self, IconType},
monitor, util, monitor, util,
window_state::{CursorFlags, SavedWindow, WindowFlags, WindowState}, window_state::{CursorFlags, SavedWindow, WindowFlags, WindowState},
@ -76,12 +101,9 @@ impl Window {
} }
pub fn set_title(&self, text: &str) { pub fn set_title(&self, text: &str) {
let text = OsStr::new(text) let wide_text = util::encode_wide(text);
.encode_wide()
.chain(Some(0).into_iter())
.collect::<Vec<_>>();
unsafe { unsafe {
winuser::SetWindowTextW(self.window.0, text.as_ptr() as LPCWSTR); SetWindowTextW(self.hwnd(), wide_text.as_ptr());
} }
} }
@ -99,24 +121,19 @@ impl Window {
#[inline] #[inline]
pub fn is_visible(&self) -> Option<bool> { pub fn is_visible(&self) -> Option<bool> {
Some(unsafe { winuser::IsWindowVisible(self.window.0) == 1 }) Some(unsafe { IsWindowVisible(self.window.0) == 1 })
} }
#[inline] #[inline]
pub fn request_redraw(&self) { pub fn request_redraw(&self) {
unsafe { unsafe {
winuser::RedrawWindow( RedrawWindow(self.hwnd(), ptr::null(), 0, RDW_INTERNALPAINT);
self.window.0,
ptr::null(),
ptr::null_mut(),
winuser::RDW_INTERNALPAINT,
);
} }
} }
#[inline] #[inline]
pub fn outer_position(&self) -> Result<PhysicalPosition<i32>, NotSupportedError> { pub fn outer_position(&self) -> Result<PhysicalPosition<i32>, NotSupportedError> {
util::get_window_rect(self.window.0) util::get_window_rect(self.hwnd())
.map(|rect| Ok(PhysicalPosition::new(rect.left as i32, rect.top as i32))) .map(|rect| Ok(PhysicalPosition::new(rect.left as i32, rect.top as i32)))
.expect("Unexpected GetWindowRect failure; please report this error to https://github.com/rust-windowing/winit") .expect("Unexpected GetWindowRect failure; please report this error to https://github.com/rust-windowing/winit")
} }
@ -124,7 +141,7 @@ impl Window {
#[inline] #[inline]
pub fn inner_position(&self) -> Result<PhysicalPosition<i32>, NotSupportedError> { pub fn inner_position(&self) -> Result<PhysicalPosition<i32>, NotSupportedError> {
let mut position: POINT = unsafe { mem::zeroed() }; let mut position: POINT = unsafe { mem::zeroed() };
if unsafe { winuser::ClientToScreen(self.window.0, &mut position) } == 0 { if unsafe { ClientToScreen(self.hwnd(), &mut position) } == false.into() {
panic!("Unexpected ClientToScreen failure: please report this error to https://github.com/rust-windowing/winit") panic!("Unexpected ClientToScreen failure: please report this error to https://github.com/rust-windowing/winit")
} }
Ok(PhysicalPosition::new(position.x as i32, position.y as i32)) Ok(PhysicalPosition::new(position.x as i32, position.y as i32))
@ -144,26 +161,23 @@ impl Window {
}); });
unsafe { unsafe {
winuser::SetWindowPos( SetWindowPos(
self.window.0, self.hwnd(),
ptr::null_mut(), 0,
x as c_int, x,
y as c_int, y,
0, 0,
0, 0,
winuser::SWP_ASYNCWINDOWPOS SWP_ASYNCWINDOWPOS | SWP_NOZORDER | SWP_NOSIZE | SWP_NOACTIVATE,
| winuser::SWP_NOZORDER
| winuser::SWP_NOSIZE
| winuser::SWP_NOACTIVATE,
); );
winuser::InvalidateRgn(self.window.0, ptr::null_mut(), 0); InvalidateRgn(self.hwnd(), 0, false.into());
} }
} }
#[inline] #[inline]
pub fn inner_size(&self) -> PhysicalSize<u32> { pub fn inner_size(&self) -> PhysicalSize<u32> {
let mut rect: RECT = unsafe { mem::zeroed() }; let mut rect: RECT = unsafe { mem::zeroed() };
if unsafe { winuser::GetClientRect(self.window.0, &mut rect) } == 0 { if unsafe { GetClientRect(self.hwnd(), &mut rect) } == false.into() {
panic!("Unexpected GetClientRect failure: please report this error to https://github.com/rust-windowing/winit") panic!("Unexpected GetClientRect failure: please report this error to https://github.com/rust-windowing/winit")
} }
PhysicalSize::new( PhysicalSize::new(
@ -174,7 +188,7 @@ impl Window {
#[inline] #[inline]
pub fn outer_size(&self) -> PhysicalSize<u32> { pub fn outer_size(&self) -> PhysicalSize<u32> {
util::get_window_rect(self.window.0) util::get_window_rect(self.hwnd())
.map(|rect| { .map(|rect| {
PhysicalSize::new( PhysicalSize::new(
(rect.right - rect.left) as u32, (rect.right - rect.left) as u32,
@ -198,7 +212,7 @@ impl Window {
}); });
}); });
util::set_inner_size_physical(self.window.0, width, height); util::set_inner_size_physical(self.hwnd(), width, height);
} }
#[inline] #[inline]
@ -244,7 +258,7 @@ impl Window {
#[inline] #[inline]
pub fn hinstance(&self) -> HINSTANCE { pub fn hinstance(&self) -> HINSTANCE {
unsafe { winuser::GetWindowLongPtrW(self.hwnd(), winuser::GWLP_HINSTANCE) as *mut _ } unsafe { super::get_window_long(self.hwnd(), GWLP_HINSTANCE) }
} }
#[inline] #[inline]
@ -259,8 +273,8 @@ impl Window {
pub fn set_cursor_icon(&self, cursor: CursorIcon) { pub fn set_cursor_icon(&self, cursor: CursorIcon) {
self.window_state.lock().mouse.cursor = cursor; self.window_state.lock().mouse.cursor = cursor;
self.thread_executor.execute_in_thread(move || unsafe { self.thread_executor.execute_in_thread(move || unsafe {
let cursor = winuser::LoadCursorW(ptr::null_mut(), cursor.to_windows_cursor()); let cursor = LoadCursorW(0, cursor.to_windows_cursor());
winuser::SetCursor(cursor); SetCursor(cursor);
}); });
} }
@ -312,10 +326,10 @@ impl Window {
let mut point = POINT { x, y }; let mut point = POINT { x, y };
unsafe { unsafe {
if winuser::ClientToScreen(self.window.0, &mut point) == 0 { if ClientToScreen(self.hwnd(), &mut point) == false.into() {
return Err(ExternalError::Os(os_error!(io::Error::last_os_error()))); return Err(ExternalError::Os(os_error!(io::Error::last_os_error())));
} }
if winuser::SetCursorPos(point.x, point.y) == 0 { if SetCursorPos(point.x, point.y) == false.into() {
return Err(ExternalError::Os(os_error!(io::Error::last_os_error()))); return Err(ExternalError::Os(os_error!(io::Error::last_os_error())));
} }
} }
@ -327,18 +341,18 @@ impl Window {
unsafe { unsafe {
let points = { let points = {
let mut pos = mem::zeroed(); let mut pos = mem::zeroed();
winuser::GetCursorPos(&mut pos); GetCursorPos(&mut pos);
pos pos
}; };
let points = POINTS { let points = POINTS {
x: points.x as SHORT, x: points.x as i16,
y: points.y as SHORT, y: points.y as i16,
}; };
winuser::ReleaseCapture(); ReleaseCapture();
winuser::PostMessageW( PostMessageW(
self.window.0, self.hwnd(),
winuser::WM_NCLBUTTONDOWN, WM_NCLBUTTONDOWN,
winuser::HTCAPTION as WPARAM, HTCAPTION as WPARAM,
&points as *const _ as LPARAM, &points as *const _ as LPARAM,
); );
} }
@ -348,7 +362,7 @@ impl Window {
#[inline] #[inline]
pub fn id(&self) -> WindowId { pub fn id(&self) -> WindowId {
WindowId(self.window.0) WindowId(self.hwnd())
} }
#[inline] #[inline]
@ -415,50 +429,43 @@ impl Window {
| (&Some(Fullscreen::Exclusive(_)), &Some(Fullscreen::Exclusive(ref video_mode))) => | (&Some(Fullscreen::Exclusive(_)), &Some(Fullscreen::Exclusive(ref video_mode))) =>
{ {
let monitor = video_mode.monitor(); let monitor = video_mode.monitor();
let monitor_info = monitor::get_monitor_info(monitor.inner.hmonitor()).unwrap();
let mut display_name = OsStr::new(&monitor.inner.native_identifier()) let native_video_mode = video_mode.video_mode.native_video_mode;
.encode_wide()
.collect::<Vec<_>>();
// `encode_wide` does not add a null-terminator but
// `ChangeDisplaySettingsExW` requires a null-terminated
// string, so add it
display_name.push(0);
let mut native_video_mode = video_mode.video_mode.native_video_mode;
let res = unsafe { let res = unsafe {
winuser::ChangeDisplaySettingsExW( ChangeDisplaySettingsExW(
display_name.as_ptr(), monitor_info.szDevice.as_ptr(),
&mut native_video_mode, &native_video_mode,
std::ptr::null_mut(), 0,
winuser::CDS_FULLSCREEN, CDS_FULLSCREEN,
std::ptr::null_mut(), ptr::null(),
) )
}; };
debug_assert!(res != winuser::DISP_CHANGE_BADFLAGS); debug_assert!(res != DISP_CHANGE_BADFLAGS);
debug_assert!(res != winuser::DISP_CHANGE_BADMODE); debug_assert!(res != DISP_CHANGE_BADMODE);
debug_assert!(res != winuser::DISP_CHANGE_BADPARAM); debug_assert!(res != DISP_CHANGE_BADPARAM);
debug_assert!(res != winuser::DISP_CHANGE_FAILED); debug_assert!(res != DISP_CHANGE_FAILED);
assert_eq!(res, winuser::DISP_CHANGE_SUCCESSFUL); assert_eq!(res, DISP_CHANGE_SUCCESSFUL);
} }
(&Some(Fullscreen::Exclusive(_)), &None) (&Some(Fullscreen::Exclusive(_)), &None)
| (&Some(Fullscreen::Exclusive(_)), &Some(Fullscreen::Borderless(_))) => { | (&Some(Fullscreen::Exclusive(_)), &Some(Fullscreen::Borderless(_))) => {
let res = unsafe { let res = unsafe {
winuser::ChangeDisplaySettingsExW( ChangeDisplaySettingsExW(
std::ptr::null_mut(), ptr::null(),
std::ptr::null_mut(), ptr::null(),
std::ptr::null_mut(), 0,
winuser::CDS_FULLSCREEN, CDS_FULLSCREEN,
std::ptr::null_mut(), ptr::null(),
) )
}; };
debug_assert!(res != winuser::DISP_CHANGE_BADFLAGS); debug_assert!(res != DISP_CHANGE_BADFLAGS);
debug_assert!(res != winuser::DISP_CHANGE_BADMODE); debug_assert!(res != DISP_CHANGE_BADMODE);
debug_assert!(res != winuser::DISP_CHANGE_BADPARAM); debug_assert!(res != DISP_CHANGE_BADPARAM);
debug_assert!(res != winuser::DISP_CHANGE_FAILED); debug_assert!(res != DISP_CHANGE_FAILED);
assert_eq!(res, winuser::DISP_CHANGE_SUCCESSFUL); assert_eq!(res, DISP_CHANGE_SUCCESSFUL);
} }
_ => (), _ => (),
} }
@ -473,7 +480,7 @@ impl Window {
// fine, taking control back from the DWM and ensuring that the `SetWindowPos` call // fine, taking control back from the DWM and ensuring that the `SetWindowPos` call
// below goes through. // below goes through.
let mut msg = mem::zeroed(); let mut msg = mem::zeroed();
winuser::PeekMessageW(&mut msg, ptr::null_mut(), 0, 0, 0); PeekMessageW(&mut msg, 0, 0, 0, PM_NOREMOVE);
} }
// Update window style // Update window style
@ -494,7 +501,7 @@ impl Window {
// Save window bounds before entering fullscreen // Save window bounds before entering fullscreen
let placement = unsafe { let placement = unsafe {
let mut placement = mem::zeroed(); let mut placement = mem::zeroed();
winuser::GetWindowPlacement(window.0, &mut placement); GetWindowPlacement(window.0, &mut placement);
placement placement
}; };
@ -512,16 +519,16 @@ impl Window {
let size: (u32, u32) = monitor.size().into(); let size: (u32, u32) = monitor.size().into();
unsafe { unsafe {
winuser::SetWindowPos( SetWindowPos(
window.0, window.0,
ptr::null_mut(), 0,
position.0, position.0,
position.1, position.1,
size.0 as i32, size.0 as i32,
size.1 as i32, size.1 as i32,
winuser::SWP_ASYNCWINDOWPOS | winuser::SWP_NOZORDER, SWP_ASYNCWINDOWPOS | SWP_NOZORDER,
); );
winuser::InvalidateRgn(window.0, ptr::null_mut(), 0); InvalidateRgn(window.0, 0, false.into());
} }
} }
None => { None => {
@ -529,8 +536,8 @@ impl Window {
if let Some(SavedWindow { placement }) = window_state_lock.saved_window.take() { if let Some(SavedWindow { placement }) = window_state_lock.saved_window.take() {
drop(window_state_lock); drop(window_state_lock);
unsafe { unsafe {
winuser::SetWindowPlacement(window.0, &placement); SetWindowPlacement(window.0, &placement);
winuser::InvalidateRgn(window.0, ptr::null_mut(), 0); InvalidateRgn(window.0, 0, false.into());
} }
} }
} }
@ -577,7 +584,7 @@ impl Window {
#[inline] #[inline]
pub fn current_monitor(&self) -> Option<RootMonitorHandle> { pub fn current_monitor(&self) -> Option<RootMonitorHandle> {
Some(RootMonitorHandle { Some(RootMonitorHandle {
inner: monitor::current_monitor(self.window.0), inner: monitor::current_monitor(self.hwnd()),
}) })
} }
@ -586,16 +593,16 @@ impl Window {
if let Some(ref window_icon) = window_icon { if let Some(ref window_icon) = window_icon {
window_icon window_icon
.inner .inner
.set_for_window(self.window.0, IconType::Small); .set_for_window(self.hwnd(), IconType::Small);
} else { } else {
icon::unset_for_window(self.window.0, IconType::Small); icon::unset_for_window(self.hwnd(), IconType::Small);
} }
self.window_state.lock().window_icon = window_icon; self.window_state.lock().window_icon = window_icon;
} }
#[inline] #[inline]
pub fn set_enable(&self, enabled: bool) { pub fn set_enable(&self, enabled: bool) {
unsafe { winuser::EnableWindow(self.hwnd() as _, enabled as _) }; unsafe { EnableWindow(self.hwnd(), enabled.into()) };
} }
#[inline] #[inline]
@ -603,24 +610,24 @@ impl Window {
if let Some(ref taskbar_icon) = taskbar_icon { if let Some(ref taskbar_icon) = taskbar_icon {
taskbar_icon taskbar_icon
.inner .inner
.set_for_window(self.window.0, IconType::Big); .set_for_window(self.hwnd(), IconType::Big);
} else { } else {
icon::unset_for_window(self.window.0, IconType::Big); icon::unset_for_window(self.hwnd(), IconType::Big);
} }
self.window_state.lock().taskbar_icon = taskbar_icon; self.window_state.lock().taskbar_icon = taskbar_icon;
} }
pub(crate) fn set_ime_position_physical(&self, x: i32, y: i32) { pub(crate) fn set_ime_position_physical(&self, x: i32, y: i32) {
if unsafe { winuser::GetSystemMetrics(winuser::SM_IMMENABLED) } != 0 { if unsafe { GetSystemMetrics(SM_IMMENABLED) } != 0 {
let mut composition_form = COMPOSITIONFORM { let composition_form = COMPOSITIONFORM {
dwStyle: CFS_POINT, dwStyle: CFS_POINT,
ptCurrentPos: POINT { x, y }, ptCurrentPos: POINT { x, y },
rcArea: unsafe { mem::zeroed() }, rcArea: unsafe { mem::zeroed() },
}; };
unsafe { unsafe {
let himc = winapi::um::imm::ImmGetContext(self.window.0); let himc = ImmGetContext(self.hwnd());
winapi::um::imm::ImmSetCompositionWindow(himc, &mut composition_form); ImmSetCompositionWindow(himc, &composition_form);
winapi::um::imm::ImmReleaseContext(self.window.0, himc); ImmReleaseContext(self.hwnd(), himc);
} }
} }
} }
@ -634,7 +641,7 @@ impl Window {
#[inline] #[inline]
pub fn request_user_attention(&self, request_type: Option<UserAttentionType>) { pub fn request_user_attention(&self, request_type: Option<UserAttentionType>) {
let window = self.window.clone(); let window = self.window.clone();
let active_window_handle = unsafe { winuser::GetActiveWindow() }; let active_window_handle = unsafe { GetActiveWindow() };
if window.0 == active_window_handle { if window.0 == active_window_handle {
return; return;
} }
@ -643,23 +650,19 @@ impl Window {
let _ = &window; let _ = &window;
let (flags, count) = request_type let (flags, count) = request_type
.map(|ty| match ty { .map(|ty| match ty {
UserAttentionType::Critical => { UserAttentionType::Critical => (FLASHW_ALL | FLASHW_TIMERNOFG, u32::MAX),
(winuser::FLASHW_ALL | winuser::FLASHW_TIMERNOFG, u32::MAX) UserAttentionType::Informational => (FLASHW_TRAY | FLASHW_TIMERNOFG, 0),
}
UserAttentionType::Informational => {
(winuser::FLASHW_TRAY | winuser::FLASHW_TIMERNOFG, 0)
}
}) })
.unwrap_or((winuser::FLASHW_STOP, 0)); .unwrap_or((FLASHW_STOP, 0));
let mut flash_info = winuser::FLASHWINFO { let flash_info = FLASHWINFO {
cbSize: mem::size_of::<winuser::FLASHWINFO>() as UINT, cbSize: mem::size_of::<FLASHWINFO>() as u32,
hwnd: window.0, hwnd: window.0,
dwFlags: flags, dwFlags: flags,
uCount: count, uCount: count,
dwTimeout: 0, dwTimeout: 0,
}; };
winuser::FlashWindowEx(&mut flash_info); FlashWindowEx(&flash_info);
}); });
} }
@ -675,7 +678,7 @@ impl Window {
let is_visible = window_flags.contains(WindowFlags::VISIBLE); let is_visible = window_flags.contains(WindowFlags::VISIBLE);
let is_minimized = window_flags.contains(WindowFlags::MINIMIZED); let is_minimized = window_flags.contains(WindowFlags::MINIMIZED);
let is_foreground = window.0 == unsafe { winuser::GetForegroundWindow() }; let is_foreground = window.0 == unsafe { GetForegroundWindow() };
if is_visible && !is_minimized && !is_foreground { if is_visible && !is_minimized && !is_foreground {
unsafe { force_window_active(window.0) }; unsafe { force_window_active(window.0) };
@ -689,7 +692,7 @@ impl Drop for Window {
unsafe { unsafe {
// The window must be destroyed from the same thread that created it, so we send a // The window must be destroyed from the same thread that created it, so we send a
// custom message to be handled by our callback to do the actual work. // custom message to be handled by our callback to do the actual work.
winuser::PostMessageW(self.window.0, *DESTROY_MSG_ID, 0, 0); PostMessageW(self.hwnd(), *DESTROY_MSG_ID, 0, 0);
} }
} }
} }
@ -720,9 +723,9 @@ impl<'a, T: 'static> InitData<'a, T> {
unsafe fn create_window(&self, window: HWND) -> Window { unsafe fn create_window(&self, window: HWND) -> Window {
// Register for touch events if applicable // Register for touch events if applicable
{ {
let digitizer = winuser::GetSystemMetrics(winuser::SM_DIGITIZER) as u32; let digitizer = GetSystemMetrics(SM_DIGITIZER) as u32;
if digitizer & winuser::NID_READY != 0 { if digitizer & NID_READY != 0 {
winuser::RegisterTouchWindow(window, winuser::TWF_WANTPALM); RegisterTouchWindow(window, TWF_WANTPALM);
} }
} }
@ -758,9 +761,7 @@ impl<'a, T: 'static> InitData<'a, T> {
unsafe fn create_window_data(&self, win: &Window) -> event_loop::WindowData<T> { unsafe fn create_window_data(&self, win: &Window) -> event_loop::WindowData<T> {
let file_drop_handler = if self.pl_attribs.drag_and_drop { let file_drop_handler = if self.pl_attribs.drag_and_drop {
use winapi::shared::winerror::{OLE_E_WRONGCOMPOBJ, RPC_E_CHANGED_MODE, S_OK}; let ole_init_result = OleInitialize(ptr::null_mut());
let ole_init_result = ole2::OleInitialize(ptr::null_mut());
// It is ok if the initialize result is `S_FALSE` because it might happen that // It is ok if the initialize result is `S_FALSE` because it might happen that
// multiple windows are created on the same thread. // multiple windows are created on the same thread.
if ole_init_result == OLE_E_WRONGCOMPOBJ { if ole_init_result == OLE_E_WRONGCOMPOBJ {
@ -782,12 +783,11 @@ impl<'a, T: 'static> InitData<'a, T> {
} }
}), }),
); );
let handler_interface_ptr = &mut (*file_drop_handler.data).interface as LPDROPTARGET;
assert_eq!( let handler_interface_ptr =
ole2::RegisterDragDrop(win.window.0, handler_interface_ptr), &mut (*file_drop_handler.data).interface as *mut _ as *mut c_void;
S_OK
); assert_eq!(RegisterDragDrop(win.window.0, handler_interface_ptr), S_OK);
Some(file_drop_handler) Some(file_drop_handler)
} else { } else {
None None
@ -806,7 +806,7 @@ impl<'a, T: 'static> InitData<'a, T> {
// Returns a pointer to window user data on success. // Returns a pointer to window user data on success.
// The user data will be registered for the window and can be accessed within the window event callback. // The user data will be registered for the window and can be accessed within the window event callback.
pub unsafe fn on_nccreate(&mut self, window: HWND) -> Option<WindowLongPtr> { pub unsafe fn on_nccreate(&mut self, window: HWND) -> Option<isize> {
let runner = self.event_loop.runner_shared.clone(); let runner = self.event_loop.runner_shared.clone();
let result = runner.catch_unwind(|| { let result = runner.catch_unwind(|| {
let mut window = self.create_window(window); let mut window = self.create_window(window);
@ -829,20 +829,20 @@ impl<'a, T: 'static> InitData<'a, T> {
// Empty region for the blur effect, so the window is fully transparent // Empty region for the blur effect, so the window is fully transparent
let region = CreateRectRgn(0, 0, -1, -1); let region = CreateRectRgn(0, 0, -1, -1);
let bb = dwmapi::DWM_BLURBEHIND { let bb = DWM_BLURBEHIND {
dwFlags: dwmapi::DWM_BB_ENABLE | dwmapi::DWM_BB_BLURREGION, dwFlags: DWM_BB_ENABLE | DWM_BB_BLURREGION,
fEnable: 1, fEnable: true.into(),
hRgnBlur: region, hRgnBlur: region,
fTransitionOnMaximized: 0, fTransitionOnMaximized: false.into(),
}; };
let hr = dwmapi::DwmEnableBlurBehindWindow(win.hwnd(), &bb); let hr = DwmEnableBlurBehindWindow(win.hwnd(), &bb);
if !SUCCEEDED(hr) { if hr < 0 {
warn!( warn!(
"Setting transparent window is failed. HRESULT Code: 0x{:X}", "Setting transparent window is failed. HRESULT Code: 0x{:X}",
hr hr
); );
} }
DeleteObject(region as _); DeleteObject(region);
} }
let attributes = self.attributes.clone(); let attributes = self.attributes.clone();
@ -879,12 +879,8 @@ unsafe fn init<T>(
where where
T: 'static, T: 'static,
{ {
let title = OsStr::new(&attributes.title) let title = util::encode_wide(&attributes.title);
.encode_wide()
.chain(Some(0).into_iter())
.collect::<Vec<_>>();
// registering the window class
let class_name = register_window_class::<T>(&attributes.window_icon, &pl_attribs.taskbar_icon); let class_name = register_window_class::<T>(&attributes.window_icon, &pl_attribs.taskbar_icon);
let mut window_flags = WindowFlags::empty(); let mut window_flags = WindowFlags::empty();
@ -925,18 +921,18 @@ where
}; };
let (style, ex_style) = window_flags.to_window_styles(); let (style, ex_style) = window_flags.to_window_styles();
let handle = winuser::CreateWindowExW( let handle = CreateWindowExW(
ex_style, ex_style,
class_name.as_ptr(), class_name.as_ptr(),
title.as_ptr() as LPCWSTR, title.as_ptr(),
style, style,
winuser::CW_USEDEFAULT, CW_USEDEFAULT,
winuser::CW_USEDEFAULT, CW_USEDEFAULT,
winuser::CW_USEDEFAULT, CW_USEDEFAULT,
winuser::CW_USEDEFAULT, CW_USEDEFAULT,
parent.unwrap_or(ptr::null_mut()), parent.unwrap_or(0),
pl_attribs.menu.unwrap_or(ptr::null_mut()), pl_attribs.menu.unwrap_or(0),
libloaderapi::GetModuleHandleW(ptr::null()), GetModuleHandleW(ptr::null()),
&mut initdata as *mut _ as *mut _, &mut initdata as *mut _ as *mut _,
); );
@ -945,7 +941,7 @@ where
panic::resume_unwind(panic_error) panic::resume_unwind(panic_error)
} }
if handle.is_null() { if handle == 0 {
return Err(os_error!(io::Error::last_os_error())); return Err(os_error!(io::Error::last_os_error()));
} }
@ -958,30 +954,27 @@ unsafe fn register_window_class<T: 'static>(
window_icon: &Option<Icon>, window_icon: &Option<Icon>,
taskbar_icon: &Option<Icon>, taskbar_icon: &Option<Icon>,
) -> Vec<u16> { ) -> Vec<u16> {
let class_name: Vec<_> = OsStr::new("Window Class") let class_name = util::encode_wide("Window Class");
.encode_wide()
.chain(Some(0).into_iter())
.collect();
let h_icon = taskbar_icon let h_icon = taskbar_icon
.as_ref() .as_ref()
.map(|icon| icon.inner.as_raw_handle()) .map(|icon| icon.inner.as_raw_handle())
.unwrap_or(ptr::null_mut()); .unwrap_or(0);
let h_icon_small = window_icon let h_icon_small = window_icon
.as_ref() .as_ref()
.map(|icon| icon.inner.as_raw_handle()) .map(|icon| icon.inner.as_raw_handle())
.unwrap_or(ptr::null_mut()); .unwrap_or(0);
let class = winuser::WNDCLASSEXW { let class = WNDCLASSEXW {
cbSize: mem::size_of::<winuser::WNDCLASSEXW>() as UINT, cbSize: mem::size_of::<WNDCLASSEXW>() as u32,
style: winuser::CS_HREDRAW | winuser::CS_VREDRAW | winuser::CS_OWNDC, style: CS_HREDRAW | CS_VREDRAW | CS_OWNDC,
lpfnWndProc: Some(super::event_loop::public_window_callback::<T>), lpfnWndProc: Some(super::event_loop::public_window_callback::<T>),
cbClsExtra: 0, cbClsExtra: 0,
cbWndExtra: 0, cbWndExtra: 0,
hInstance: libloaderapi::GetModuleHandleW(ptr::null()), hInstance: GetModuleHandleW(ptr::null()),
hIcon: h_icon, hIcon: h_icon,
hCursor: ptr::null_mut(), // must be null in order for cursor state to work properly hCursor: 0, // must be null in order for cursor state to work properly
hbrBackground: ptr::null_mut(), hbrBackground: 0,
lpszMenuName: ptr::null(), lpszMenuName: ptr::null(),
lpszClassName: class_name.as_ptr(), lpszClassName: class_name.as_ptr(),
hIconSm: h_icon_small, hIconSm: h_icon_small,
@ -991,7 +984,7 @@ unsafe fn register_window_class<T: 'static>(
// an error, and because errors here are detected during CreateWindowEx anyway. // an error, and because errors here are detected during CreateWindowEx anyway.
// Also since there is no weird element in the struct, there is no reason for this // Also since there is no weird element in the struct, there is no reason for this
// call to fail. // call to fail.
winuser::RegisterClassExW(&class); RegisterClassExW(&class);
class_name class_name
} }
@ -999,14 +992,14 @@ unsafe fn register_window_class<T: 'static>(
struct ComInitialized(*mut ()); struct ComInitialized(*mut ());
impl Drop for ComInitialized { impl Drop for ComInitialized {
fn drop(&mut self) { fn drop(&mut self) {
unsafe { combaseapi::CoUninitialize() }; unsafe { CoUninitialize() };
} }
} }
thread_local! { thread_local! {
static COM_INITIALIZED: ComInitialized = { static COM_INITIALIZED: ComInitialized = {
unsafe { unsafe {
combaseapi::CoInitializeEx(ptr::null_mut(), COINIT_APARTMENTTHREADED); CoInitializeEx(ptr::null(), COINIT_APARTMENTTHREADED);
ComInitialized(ptr::null_mut()) ComInitialized(ptr::null_mut())
} }
}; };
@ -1033,17 +1026,17 @@ unsafe fn taskbar_mark_fullscreen(handle: HWND, fullscreen: bool) {
let mut task_bar_list = task_bar_list_ptr.get(); let mut task_bar_list = task_bar_list_ptr.get();
if task_bar_list.is_null() { if task_bar_list.is_null() {
use winapi::{shared::winerror::S_OK, Interface}; let hr = CoCreateInstance(
let hr = combaseapi::CoCreateInstance(
&CLSID_TaskbarList, &CLSID_TaskbarList,
ptr::null_mut(), ptr::null_mut(),
combaseapi::CLSCTX_ALL, CLSCTX_ALL,
&ITaskbarList2::uuidof(), &IID_ITaskbarList2,
&mut task_bar_list as *mut _ as *mut _, &mut task_bar_list as *mut _ as *mut _,
); );
if hr != S_OK || (*task_bar_list).HrInit() != S_OK { let hr_init = (*(*task_bar_list).lpVtbl).parent.HrInit;
if hr != S_OK || hr_init(task_bar_list.cast()) != S_OK {
// In some old windows, the taskbar object could not be created, we just ignore it // In some old windows, the taskbar object could not be created, we just ignore it
return; return;
} }
@ -1051,7 +1044,8 @@ unsafe fn taskbar_mark_fullscreen(handle: HWND, fullscreen: bool) {
} }
task_bar_list = task_bar_list_ptr.get(); task_bar_list = task_bar_list_ptr.get();
(*task_bar_list).MarkFullscreenWindow(handle, if fullscreen { 1 } else { 0 }); let mark_fullscreen_window = (*(*task_bar_list).lpVtbl).MarkFullscreenWindow;
mark_fullscreen_window(task_bar_list, handle, if fullscreen { 1 } else { 0 });
}) })
} }
@ -1060,25 +1054,41 @@ unsafe fn force_window_active(handle: HWND) {
// This is a little hack which can "steal" the foreground window permission // This is a little hack which can "steal" the foreground window permission
// We only call this function in the window creation, so it should be fine. // We only call this function in the window creation, so it should be fine.
// See : https://stackoverflow.com/questions/10740346/setforegroundwindow-only-working-while-visual-studio-is-open // See : https://stackoverflow.com/questions/10740346/setforegroundwindow-only-working-while-visual-studio-is-open
let alt_sc = winuser::MapVirtualKeyW(winuser::VK_MENU as _, winuser::MAPVK_VK_TO_VSC); let alt_sc = MapVirtualKeyW(VK_MENU as u32, MAPVK_VK_TO_VSC);
let mut inputs: [winuser::INPUT; 2] = mem::zeroed(); let inputs = [
inputs[0].type_ = winuser::INPUT_KEYBOARD; INPUT {
inputs[0].u.ki_mut().wVk = winuser::VK_LMENU as _; r#type: INPUT_KEYBOARD,
inputs[0].u.ki_mut().wScan = alt_sc as _; Anonymous: INPUT_0 {
inputs[0].u.ki_mut().dwFlags = winuser::KEYEVENTF_EXTENDEDKEY; ki: KEYBDINPUT {
wVk: VK_LMENU,
inputs[1].type_ = winuser::INPUT_KEYBOARD; wScan: alt_sc as u16,
inputs[1].u.ki_mut().wVk = winuser::VK_LMENU as _; dwFlags: KEYEVENTF_EXTENDEDKEY,
inputs[1].u.ki_mut().wScan = alt_sc as _; dwExtraInfo: 0,
inputs[1].u.ki_mut().dwFlags = winuser::KEYEVENTF_EXTENDEDKEY | winuser::KEYEVENTF_KEYUP; time: 0,
},
},
},
INPUT {
r#type: INPUT_KEYBOARD,
Anonymous: INPUT_0 {
ki: KEYBDINPUT {
wVk: VK_LMENU,
wScan: alt_sc as u16,
dwFlags: KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP,
dwExtraInfo: 0,
time: 0,
},
},
},
];
// Simulate a key press and release // Simulate a key press and release
winuser::SendInput( SendInput(
inputs.len() as _, inputs.len() as u32,
inputs.as_mut_ptr(), inputs.as_ptr(),
mem::size_of::<winuser::INPUT>() as _, mem::size_of::<INPUT>() as i32,
); );
winuser::SetForegroundWindow(handle); SetForegroundWindow(handle);
} }

View file

@ -6,13 +6,20 @@ use crate::{
window::{CursorIcon, Fullscreen, Theme, WindowAttributes}, window::{CursorIcon, Fullscreen, Theme, WindowAttributes},
}; };
use parking_lot::MutexGuard; use parking_lot::MutexGuard;
use std::{io, ptr}; use std::io;
use winapi::{ use windows_sys::Win32::{
shared::{ Foundation::{HWND, RECT},
minwindef::DWORD, Graphics::Gdi::InvalidateRgn,
windef::{HWND, RECT}, UI::WindowsAndMessaging::{
SendMessageW, SetWindowLongW, SetWindowPos, ShowWindow, GWL_EXSTYLE, GWL_STYLE,
HWND_NOTOPMOST, HWND_TOPMOST, SWP_ASYNCWINDOWPOS, SWP_FRAMECHANGED, SWP_NOACTIVATE,
SWP_NOMOVE, SWP_NOSIZE, SWP_NOZORDER, SW_HIDE, SW_MAXIMIZE, SW_MINIMIZE, SW_RESTORE,
SW_SHOW, WINDOWPLACEMENT, WINDOW_EX_STYLE, WINDOW_STYLE, WS_BORDER, WS_CAPTION, WS_CHILD,
WS_CLIPCHILDREN, WS_CLIPSIBLINGS, WS_EX_ACCEPTFILES, WS_EX_APPWINDOW, WS_EX_LEFT,
WS_EX_NOREDIRECTIONBITMAP, WS_EX_TOPMOST, WS_EX_WINDOWEDGE, WS_MAXIMIZE, WS_MAXIMIZEBOX,
WS_MINIMIZE, WS_MINIMIZEBOX, WS_OVERLAPPED, WS_OVERLAPPEDWINDOW, WS_POPUP, WS_SIZEBOX,
WS_SYSMENU, WS_VISIBLE,
}, },
um::winuser,
}; };
/// Contains information about states and the window that the callback is going to use. /// Contains information about states and the window that the callback is going to use.
@ -39,7 +46,7 @@ pub struct WindowState {
#[derive(Clone)] #[derive(Clone)]
pub struct SavedWindow { pub struct SavedWindow {
pub placement: winuser::WINDOWPLACEMENT, pub placement: WINDOWPLACEMENT,
} }
#[derive(Clone)] #[derive(Clone)]
@ -187,10 +194,8 @@ impl WindowFlags {
self self
} }
pub fn to_window_styles(self) -> (DWORD, DWORD) { pub fn to_window_styles(self) -> (WINDOW_STYLE, WINDOW_EX_STYLE) {
use winapi::um::winuser::*; let (mut style, mut style_ex) = (WS_OVERLAPPED, WS_EX_LEFT);
let (mut style, mut style_ex) = (0, 0);
if self.contains(WindowFlags::RESIZABLE) { if self.contains(WindowFlags::RESIZABLE) {
style |= WS_SIZEBOX | WS_MAXIMIZEBOX; style |= WS_SIZEBOX | WS_MAXIMIZEBOX;
@ -248,43 +253,40 @@ impl WindowFlags {
if diff.contains(WindowFlags::VISIBLE) { if diff.contains(WindowFlags::VISIBLE) {
unsafe { unsafe {
winuser::ShowWindow( ShowWindow(
window, window,
match new.contains(WindowFlags::VISIBLE) { match new.contains(WindowFlags::VISIBLE) {
true => winuser::SW_SHOW, true => SW_SHOW,
false => winuser::SW_HIDE, false => SW_HIDE,
}, },
); );
} }
} }
if diff.contains(WindowFlags::ALWAYS_ON_TOP) { if diff.contains(WindowFlags::ALWAYS_ON_TOP) {
unsafe { unsafe {
winuser::SetWindowPos( SetWindowPos(
window, window,
match new.contains(WindowFlags::ALWAYS_ON_TOP) { match new.contains(WindowFlags::ALWAYS_ON_TOP) {
true => winuser::HWND_TOPMOST, true => HWND_TOPMOST,
false => winuser::HWND_NOTOPMOST, false => HWND_NOTOPMOST,
}, },
0, 0,
0, 0,
0, 0,
0, 0,
winuser::SWP_ASYNCWINDOWPOS SWP_ASYNCWINDOWPOS | SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE,
| winuser::SWP_NOMOVE
| winuser::SWP_NOSIZE
| winuser::SWP_NOACTIVATE,
); );
winuser::InvalidateRgn(window, ptr::null_mut(), 0); InvalidateRgn(window, 0, false.into());
} }
} }
if diff.contains(WindowFlags::MAXIMIZED) || new.contains(WindowFlags::MAXIMIZED) { if diff.contains(WindowFlags::MAXIMIZED) || new.contains(WindowFlags::MAXIMIZED) {
unsafe { unsafe {
winuser::ShowWindow( ShowWindow(
window, window,
match new.contains(WindowFlags::MAXIMIZED) { match new.contains(WindowFlags::MAXIMIZED) {
true => winuser::SW_MAXIMIZE, true => SW_MAXIMIZE,
false => winuser::SW_RESTORE, false => SW_RESTORE,
}, },
); );
} }
@ -293,11 +295,11 @@ impl WindowFlags {
// Minimize operations should execute after maximize for proper window animations // Minimize operations should execute after maximize for proper window animations
if diff.contains(WindowFlags::MINIMIZED) { if diff.contains(WindowFlags::MINIMIZED) {
unsafe { unsafe {
winuser::ShowWindow( ShowWindow(
window, window,
match new.contains(WindowFlags::MINIMIZED) { match new.contains(WindowFlags::MINIMIZED) {
true => winuser::SW_MINIMIZE, true => SW_MINIMIZE,
false => winuser::SW_RESTORE, false => SW_RESTORE,
}, },
); );
} }
@ -307,18 +309,15 @@ impl WindowFlags {
let (style, style_ex) = new.to_window_styles(); let (style, style_ex) = new.to_window_styles();
unsafe { unsafe {
winuser::SendMessageW(window, *event_loop::SET_RETAIN_STATE_ON_SIZE_MSG_ID, 1, 0); SendMessageW(window, *event_loop::SET_RETAIN_STATE_ON_SIZE_MSG_ID, 1, 0);
// This condition is necessary to avoid having an unrestorable window // This condition is necessary to avoid having an unrestorable window
if !new.contains(WindowFlags::MINIMIZED) { if !new.contains(WindowFlags::MINIMIZED) {
winuser::SetWindowLongW(window, winuser::GWL_STYLE, style as _); SetWindowLongW(window, GWL_STYLE, style as i32);
winuser::SetWindowLongW(window, winuser::GWL_EXSTYLE, style_ex as _); SetWindowLongW(window, GWL_EXSTYLE, style_ex as i32);
} }
let mut flags = winuser::SWP_NOZORDER let mut flags = SWP_NOZORDER | SWP_NOMOVE | SWP_NOSIZE | SWP_FRAMECHANGED;
| winuser::SWP_NOMOVE
| winuser::SWP_NOSIZE
| winuser::SWP_FRAMECHANGED;
// We generally don't want style changes here to affect window // We generally don't want style changes here to affect window
// focus, but for fullscreen windows they must be activated // focus, but for fullscreen windows they must be activated
@ -326,12 +325,12 @@ impl WindowFlags {
if !new.contains(WindowFlags::MARKER_EXCLUSIVE_FULLSCREEN) if !new.contains(WindowFlags::MARKER_EXCLUSIVE_FULLSCREEN)
&& !new.contains(WindowFlags::MARKER_BORDERLESS_FULLSCREEN) && !new.contains(WindowFlags::MARKER_BORDERLESS_FULLSCREEN)
{ {
flags |= winuser::SWP_NOACTIVATE; flags |= SWP_NOACTIVATE;
} }
// Refresh the window frame // Refresh the window frame
winuser::SetWindowPos(window, ptr::null_mut(), 0, 0, 0, 0, flags); SetWindowPos(window, 0, 0, 0, 0, 0, flags);
winuser::SendMessageW(window, *event_loop::SET_RETAIN_STATE_ON_SIZE_MSG_ID, 0, 0); SendMessageW(window, *event_loop::SET_RETAIN_STATE_ON_SIZE_MSG_ID, 0, 0);
} }
} }
} }