mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-11 13:31:29 +11:00
Adopt windows-sys (#2057)
This commit is contained in:
parent
78e5a395da
commit
b222dde835
|
@ -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)
|
||||||
|
|
54
Cargo.toml
54
Cargo.toml
|
@ -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]
|
||||||
|
|
|
@ -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 })
|
||||||
}
|
}
|
||||||
|
|
|
@ -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()
|
|
||||||
}
|
}
|
||||||
|
|
140
src/platform_impl/windows/definitions.rs
Normal file
140
src/platform_impl/windows/definitions.rs
Normal 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],
|
||||||
|
};
|
|
@ -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
|
||||||
|
|
|
@ -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::{
|
},
|
||||||
objidl::IDataObject,
|
UI::Shell::{DragFinish, DragQueryFileW, HDROP},
|
||||||
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,
|
||||||
|
|
|
@ -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
|
@ -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,18 +208,16 @@ 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 if self.should_buffer() {
|
||||||
|
// 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.
|
||||||
|
self.event_buffer
|
||||||
|
.borrow_mut()
|
||||||
|
.push_back(BufferedEvent::from_event(event))
|
||||||
} else {
|
} else {
|
||||||
if self.should_buffer() {
|
self.move_state_to(RunnerState::HandlingMainEvents);
|
||||||
// If the runner is already borrowed, we're in the middle of an event loop invocation. Add
|
self.call_event_handler(event);
|
||||||
// the event to a buffer to be processed later.
|
self.dispatch_buffered_events();
|
||||||
self.event_buffer
|
|
||||||
.borrow_mut()
|
|
||||||
.push_back(BufferedEvent::from_event(event))
|
|
||||||
} else {
|
|
||||||
self.move_state_to(RunnerState::HandlingMainEvents);
|
|
||||||
self.call_event_handler(event);
|
|
||||||
self.dispatch_buffered_events();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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) {
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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,
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -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::{
|
||||||
um::{
|
HiDpi::{DPI_AWARENESS_CONTEXT, MONITOR_DPI_TYPE, PROCESS_DPI_AWARENESS},
|
||||||
libloaderapi::{GetProcAddress, LoadLibraryA},
|
Input::KeyboardAndMouse::GetActiveWindow,
|
||||||
shellscalingapi::{MONITOR_DPI_TYPE, PROCESS_DPI_AWARENESS},
|
WindowsAndMessaging::{
|
||||||
winbase::lstrlenW,
|
AdjustWindowRectEx, ClipCursor, GetClientRect, GetClipCursor, GetMenu,
|
||||||
winnt::{HRESULT, LONG, LPCSTR},
|
GetSystemMetrics, GetWindowLongW, GetWindowRect, SetWindowPos, ShowCursor,
|
||||||
winuser,
|
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,
|
||||||
|
},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
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! {
|
||||||
|
|
|
@ -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,
|
|
||||||
},
|
},
|
||||||
um::{
|
Graphics::{
|
||||||
combaseapi, dwmapi,
|
Dwm::{DwmEnableBlurBehindWindow, DWM_BB_BLURREGION, DWM_BB_ENABLE, DWM_BLURBEHIND},
|
||||||
imm::{CFS_POINT, COMPOSITIONFORM},
|
Gdi::{
|
||||||
libloaderapi,
|
ChangeDisplaySettingsExW, ClientToScreen, CreateRectRgn, DeleteObject, InvalidateRgn,
|
||||||
objbase::COINIT_APARTMENTTHREADED,
|
RedrawWindow, CDS_FULLSCREEN, DISP_CHANGE_BADFLAGS, DISP_CHANGE_BADMODE,
|
||||||
ole2,
|
DISP_CHANGE_BADPARAM, DISP_CHANGE_FAILED, DISP_CHANGE_SUCCESSFUL, RDW_INTERNALPAINT,
|
||||||
oleidl::LPDROPTARGET,
|
},
|
||||||
shobjidl_core::{CLSID_TaskbarList, ITaskbarList2},
|
},
|
||||||
wingdi::{CreateRectRgn, DeleteObject},
|
System::{
|
||||||
winnt::{LPCWSTR, SHORT},
|
Com::{
|
||||||
winuser,
|
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,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue