Remove window subclassing (#1933)

* Remove window subclassing

* Always call `DefWindowProcW` when we don't process a message

* Improve window initialization

Note that the error path in `init` is kind of cursed at the moment.

* Rename `ThreadMsgTargetCallbackData` to `ThreadMsgTargetData`

* Simplify window initialization

* Fix compilation on 32-bit targets

* Simplify the creation of the event target window

* Use `.clone()` rather than `Rc::clone()`

* Use concrete types for args to `SetWindowLongPtrW`

On 32-bit targets, `SetWindowLongPtrW` is an alias to `SetWindowLongW`,
which returns `LONG` (`i32`) rather than `LONG_PTR` (`isisze`).

* Minor comment adjustments
This commit is contained in:
Markus Røyset 2021-07-16 12:40:48 +02:00 committed by GitHub
parent 27e6548343
commit 63ad47a7bf
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 295 additions and 266 deletions

File diff suppressed because it is too large Load diff

View file

@ -23,7 +23,7 @@ use crate::{
pub(crate) type EventLoopRunnerShared<T> = Rc<EventLoopRunner<T>>; 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
thread_msg_target: HWND, pub(super) thread_msg_target: HWND,
wait_thread_id: DWORD, wait_thread_id: DWORD,
control_flow: Cell<ControlFlow>, control_flow: Cell<ControlFlow>,

View file

@ -7,7 +7,7 @@ use std::{
ffi::OsStr, ffi::OsStr,
io, mem, io, mem,
os::windows::ffi::OsStrExt, os::windows::ffi::OsStrExt,
ptr, panic, ptr,
sync::{mpsc::channel, Arc}, sync::{mpsc::channel, Arc},
}; };
@ -40,7 +40,7 @@ use crate::{
dark_mode::try_theme, dark_mode::try_theme,
dpi::{dpi_to_scale_factor, hwnd_dpi}, dpi::{dpi_to_scale_factor, hwnd_dpi},
drop_handler::FileDropHandler, drop_handler::FileDropHandler,
event_loop::{self, EventLoopWindowTarget, DESTROY_MSG_ID}, event_loop::{self, EventLoopWindowTarget, WindowData, 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},
@ -73,7 +73,7 @@ impl Window {
// done. you owe me -- ossi // done. you owe me -- ossi
unsafe { unsafe {
let drag_and_drop = pl_attr.drag_and_drop; let drag_and_drop = pl_attr.drag_and_drop;
init(w_attr, pl_attr, event_loop).map(|win| { init(w_attr, pl_attr, event_loop, |win| {
let file_drop_handler = if drag_and_drop { let file_drop_handler = if drag_and_drop {
use winapi::shared::winerror::{OLE_E_WRONGCOMPOBJ, RPC_E_CHANGED_MODE, S_OK}; use winapi::shared::winerror::{OLE_E_WRONGCOMPOBJ, RPC_E_CHANGED_MODE, S_OK};
@ -111,16 +111,15 @@ impl Window {
None None
}; };
let subclass_input = event_loop::SubclassInput { event_loop.runner_shared.register_window(win.window.0);
event_loop::WindowData {
window_state: win.window_state.clone(), window_state: win.window_state.clone(),
event_loop_runner: event_loop.runner_shared.clone(), event_loop_runner: event_loop.runner_shared.clone(),
file_drop_handler, file_drop_handler,
subclass_removed: Cell::new(false), userdata_removed: Cell::new(false),
recurse_depth: Cell::new(0), recurse_depth: Cell::new(0),
}; }
event_loop::subclass_window(win.window.0, subclass_input);
win
}) })
} }
} }
@ -724,18 +723,31 @@ pub struct WindowWrapper(HWND);
unsafe impl Sync for WindowWrapper {} unsafe impl Sync for WindowWrapper {}
unsafe impl Send for WindowWrapper {} unsafe impl Send for WindowWrapper {}
unsafe fn init<T: 'static>( pub(super) struct InitData<'a, T: 'static> {
// inputs
pub event_loop: &'a EventLoopWindowTarget<T>,
pub post_init: &'a dyn Fn(HWND) -> (Window, WindowData<T>),
// outputs
pub window: Option<Window>,
}
unsafe fn init<T, F>(
attributes: WindowAttributes, attributes: WindowAttributes,
pl_attribs: PlatformSpecificWindowBuilderAttributes, pl_attribs: PlatformSpecificWindowBuilderAttributes,
event_loop: &EventLoopWindowTarget<T>, event_loop: &EventLoopWindowTarget<T>,
) -> Result<Window, RootOsError> { create_window_data: F,
) -> Result<Window, RootOsError>
where
T: 'static,
F: Fn(&mut Window) -> WindowData<T>,
{
let title = OsStr::new(&attributes.title) let title = OsStr::new(&attributes.title)
.encode_wide() .encode_wide()
.chain(Some(0).into_iter()) .chain(Some(0).into_iter())
.collect::<Vec<_>>(); .collect::<Vec<_>>();
// registering the window class // registering the window class
let class_name = register_window_class(&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();
window_flags.set(WindowFlags::DECORATIONS, attributes.decorations); window_flags.set(WindowFlags::DECORATIONS, attributes.decorations);
@ -766,8 +778,22 @@ unsafe fn init<T: 'static>(
} }
}; };
// creating the real window this time, by using the functions in `extra_functions` let mut initdata = InitData {
let real_window = { event_loop,
post_init: &|hwnd| {
let mut window = post_init(
WindowWrapper(hwnd),
attributes.clone(),
pl_attribs.clone(),
window_flags,
event_loop,
);
let window_data = create_window_data(&mut window);
(window, window_data)
},
window: None,
};
let (style, ex_style) = window_flags.to_window_styles(); let (style, ex_style) = window_flags.to_window_styles();
let handle = winuser::CreateWindowExW( let handle = winuser::CreateWindowExW(
ex_style, ex_style,
@ -781,16 +807,30 @@ unsafe fn init<T: 'static>(
parent.unwrap_or(ptr::null_mut()), parent.unwrap_or(ptr::null_mut()),
pl_attribs.menu.unwrap_or(ptr::null_mut()), pl_attribs.menu.unwrap_or(ptr::null_mut()),
libloaderapi::GetModuleHandleW(ptr::null()), libloaderapi::GetModuleHandleW(ptr::null()),
ptr::null_mut(), &mut initdata as *mut _ as *mut _,
); );
// If the `post_init` callback in `InitData` panicked, then should resume panicking here
if let Err(panic_error) = event_loop.runner_shared.take_panic_error() {
panic::resume_unwind(panic_error)
}
if handle.is_null() { if handle.is_null() {
return Err(os_error!(io::Error::last_os_error())); return Err(os_error!(io::Error::last_os_error()));
} }
WindowWrapper(handle) // If the handle is non-null, then window creation must have succeeded, which means
}; // that we *must* have populated the `InitData.window` field.
Ok(initdata.window.unwrap())
}
unsafe fn post_init<T: 'static>(
real_window: WindowWrapper,
attributes: WindowAttributes,
pl_attribs: PlatformSpecificWindowBuilderAttributes,
window_flags: WindowFlags,
event_loop: &EventLoopWindowTarget<T>,
) -> Window {
// Register for touch events if applicable // Register for touch events if applicable
{ {
let digitizer = winuser::GetSystemMetrics(winuser::SM_DIGITIZER) as u32; let digitizer = winuser::GetSystemMetrics(winuser::SM_DIGITIZER) as u32;
@ -862,10 +902,10 @@ unsafe fn init<T: 'static>(
win.set_outer_position(position); win.set_outer_position(position);
} }
Ok(win) win
} }
unsafe fn register_window_class( 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> {
@ -886,7 +926,7 @@ unsafe fn register_window_class(
let class = winuser::WNDCLASSEXW { let class = winuser::WNDCLASSEXW {
cbSize: mem::size_of::<winuser::WNDCLASSEXW>() as UINT, cbSize: mem::size_of::<winuser::WNDCLASSEXW>() as UINT,
style: winuser::CS_HREDRAW | winuser::CS_VREDRAW | winuser::CS_OWNDC, style: winuser::CS_HREDRAW | winuser::CS_VREDRAW | winuser::CS_OWNDC,
lpfnWndProc: Some(winuser::DefWindowProcW), lpfnWndProc: Some(super::event_loop::public_window_callback::<T>),
cbClsExtra: 0, cbClsExtra: 0,
cbWndExtra: 0, cbWndExtra: 0,
hInstance: libloaderapi::GetModuleHandleW(ptr::null()), hInstance: libloaderapi::GetModuleHandleW(ptr::null()),