mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-26 03:36:32 +11:00
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:
parent
27e6548343
commit
63ad47a7bf
3 changed files with 295 additions and 266 deletions
File diff suppressed because it is too large
Load diff
|
@ -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>,
|
||||||
|
|
|
@ -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()),
|
||||||
|
|
Loading…
Add table
Reference in a new issue