mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-11 21:31:29 +11:00
On Windows, improve handling of window destruction (#1798)
This commit is contained in:
parent
932cbe40bf
commit
c05952b813
|
@ -4,6 +4,7 @@ mod runner;
|
||||||
|
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
use std::{
|
use std::{
|
||||||
|
cell::Cell,
|
||||||
collections::VecDeque,
|
collections::VecDeque,
|
||||||
marker::PhantomData,
|
marker::PhantomData,
|
||||||
mem, panic, ptr,
|
mem, panic, ptr,
|
||||||
|
@ -86,6 +87,8 @@ pub(crate) struct SubclassInput<T: 'static> {
|
||||||
pub window_state: Arc<Mutex<WindowState>>,
|
pub window_state: Arc<Mutex<WindowState>>,
|
||||||
pub event_loop_runner: EventLoopRunnerShared<T>,
|
pub event_loop_runner: EventLoopRunnerShared<T>,
|
||||||
pub file_drop_handler: Option<FileDropHandler>,
|
pub file_drop_handler: Option<FileDropHandler>,
|
||||||
|
pub subclass_removed: Cell<bool>,
|
||||||
|
pub recurse_depth: Cell<u32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> SubclassInput<T> {
|
impl<T> SubclassInput<T> {
|
||||||
|
@ -616,6 +619,17 @@ fn subclass_event_target_window<T>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn remove_event_target_window_subclass<T: 'static>(window: HWND) {
|
||||||
|
let removal_result = unsafe {
|
||||||
|
commctrl::RemoveWindowSubclass(
|
||||||
|
window,
|
||||||
|
Some(thread_event_target_callback::<T>),
|
||||||
|
THREAD_EVENT_TARGET_SUBCLASS_ID,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
assert_eq!(removal_result, 1);
|
||||||
|
}
|
||||||
|
|
||||||
/// Capture mouse input, allowing `window` to receive mouse events when the cursor is outside of
|
/// Capture mouse input, allowing `window` to receive mouse events when the cursor is outside of
|
||||||
/// the window.
|
/// the window.
|
||||||
unsafe fn capture_mouse(window: HWND, window_state: &mut WindowState) {
|
unsafe fn capture_mouse(window: HWND, window_state: &mut WindowState) {
|
||||||
|
@ -650,6 +664,17 @@ pub(crate) fn subclass_window<T>(window: HWND, subclass_input: SubclassInput<T>)
|
||||||
assert_eq!(subclass_result, 1);
|
assert_eq!(subclass_result, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn remove_window_subclass<T: 'static>(window: HWND) {
|
||||||
|
let removal_result = unsafe {
|
||||||
|
commctrl::RemoveWindowSubclass(
|
||||||
|
window,
|
||||||
|
Some(public_window_callback::<T>),
|
||||||
|
WINDOW_SUBCLASS_ID,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
assert_eq!(removal_result, 1);
|
||||||
|
}
|
||||||
|
|
||||||
fn normalize_pointer_pressure(pressure: u32) -> Option<Force> {
|
fn normalize_pointer_pressure(pressure: u32) -> Option<Force> {
|
||||||
match pressure {
|
match pressure {
|
||||||
1..=1024 => Some(Force::Normalized(pressure as f64 / 1024.0)),
|
1..=1024 => Some(Force::Normalized(pressure as f64 / 1024.0)),
|
||||||
|
@ -752,11 +777,41 @@ unsafe extern "system" fn public_window_callback<T: 'static>(
|
||||||
msg: UINT,
|
msg: UINT,
|
||||||
wparam: WPARAM,
|
wparam: WPARAM,
|
||||||
lparam: LPARAM,
|
lparam: LPARAM,
|
||||||
_: UINT_PTR,
|
uidsubclass: UINT_PTR,
|
||||||
subclass_input_ptr: DWORD_PTR,
|
subclass_input_ptr: DWORD_PTR,
|
||||||
) -> LRESULT {
|
) -> LRESULT {
|
||||||
let subclass_input = &*(subclass_input_ptr as *const SubclassInput<T>);
|
let subclass_input_ptr = subclass_input_ptr as *mut SubclassInput<T>;
|
||||||
|
let (result, subclass_removed, recurse_depth) = {
|
||||||
|
let subclass_input = &*subclass_input_ptr;
|
||||||
|
subclass_input
|
||||||
|
.recurse_depth
|
||||||
|
.set(subclass_input.recurse_depth.get() + 1);
|
||||||
|
|
||||||
|
let result =
|
||||||
|
public_window_callback_inner(window, msg, wparam, lparam, uidsubclass, subclass_input);
|
||||||
|
|
||||||
|
let subclass_removed = subclass_input.subclass_removed.get();
|
||||||
|
let recurse_depth = subclass_input.recurse_depth.get() - 1;
|
||||||
|
subclass_input.recurse_depth.set(recurse_depth);
|
||||||
|
|
||||||
|
(result, subclass_removed, recurse_depth)
|
||||||
|
};
|
||||||
|
|
||||||
|
if subclass_removed && recurse_depth == 0 {
|
||||||
|
Box::from_raw(subclass_input_ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn public_window_callback_inner<T: 'static>(
|
||||||
|
window: HWND,
|
||||||
|
msg: UINT,
|
||||||
|
wparam: WPARAM,
|
||||||
|
lparam: LPARAM,
|
||||||
|
_: UINT_PTR,
|
||||||
|
subclass_input: &SubclassInput<T>,
|
||||||
|
) -> LRESULT {
|
||||||
winuser::RedrawWindow(
|
winuser::RedrawWindow(
|
||||||
subclass_input.event_loop_runner.thread_msg_target(),
|
subclass_input.event_loop_runner.thread_msg_target(),
|
||||||
ptr::null(),
|
ptr::null(),
|
||||||
|
@ -816,8 +871,8 @@ unsafe extern "system" fn public_window_callback<T: 'static>(
|
||||||
}
|
}
|
||||||
|
|
||||||
winuser::WM_NCDESTROY => {
|
winuser::WM_NCDESTROY => {
|
||||||
drop(subclass_input);
|
remove_window_subclass::<T>(window);
|
||||||
Box::from_raw(subclass_input_ptr as *mut SubclassInput<T>);
|
subclass_input.subclass_removed.set(true);
|
||||||
0
|
0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1931,8 +1986,7 @@ unsafe extern "system" fn thread_event_target_callback<T: 'static>(
|
||||||
_: UINT_PTR,
|
_: UINT_PTR,
|
||||||
subclass_input_ptr: DWORD_PTR,
|
subclass_input_ptr: DWORD_PTR,
|
||||||
) -> LRESULT {
|
) -> LRESULT {
|
||||||
let subclass_input = &mut *(subclass_input_ptr as *mut ThreadMsgTargetSubclassInput<T>);
|
let subclass_input = Box::from_raw(subclass_input_ptr as *mut ThreadMsgTargetSubclassInput<T>);
|
||||||
let runner = subclass_input.event_loop_runner.clone();
|
|
||||||
|
|
||||||
if msg != winuser::WM_PAINT {
|
if msg != winuser::WM_PAINT {
|
||||||
winuser::RedrawWindow(
|
winuser::RedrawWindow(
|
||||||
|
@ -1943,13 +1997,15 @@ unsafe extern "system" fn thread_event_target_callback<T: 'static>(
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mut subclass_removed = false;
|
||||||
|
|
||||||
// I decided to bind the closure to `callback` and pass it to catch_unwind rather than passing
|
// I decided to bind the closure to `callback` and pass it to catch_unwind rather than passing
|
||||||
// the closure to catch_unwind directly so that the match body indendation wouldn't change and
|
// the closure to catch_unwind directly so that the match body indendation wouldn't change and
|
||||||
// the git blame and history would be preserved.
|
// the git blame and history would be preserved.
|
||||||
let callback = || match msg {
|
let callback = || match msg {
|
||||||
winuser::WM_NCDESTROY => {
|
winuser::WM_NCDESTROY => {
|
||||||
Box::from_raw(subclass_input);
|
remove_event_target_window_subclass::<T>(window);
|
||||||
drop(subclass_input);
|
subclass_removed = true;
|
||||||
0
|
0
|
||||||
}
|
}
|
||||||
// Because WM_PAINT comes after all other messages, we use it during modal loops to detect
|
// Because WM_PAINT comes after all other messages, we use it during modal loops to detect
|
||||||
|
@ -2156,5 +2212,14 @@ unsafe extern "system" fn thread_event_target_callback<T: 'static>(
|
||||||
_ => commctrl::DefSubclassProc(window, msg, wparam, lparam),
|
_ => commctrl::DefSubclassProc(window, msg, wparam, lparam),
|
||||||
};
|
};
|
||||||
|
|
||||||
runner.catch_unwind(callback).unwrap_or(-1)
|
let result = subclass_input
|
||||||
|
.event_loop_runner
|
||||||
|
.catch_unwind(callback)
|
||||||
|
.unwrap_or(-1);
|
||||||
|
if subclass_removed {
|
||||||
|
mem::drop(subclass_input);
|
||||||
|
} else {
|
||||||
|
Box::into_raw(subclass_input);
|
||||||
|
}
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
|
@ -114,6 +114,8 @@ impl Window {
|
||||||
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),
|
||||||
|
recurse_depth: Cell::new(0),
|
||||||
};
|
};
|
||||||
|
|
||||||
event_loop::subclass_window(win.window.0, subclass_input);
|
event_loop::subclass_window(win.window.0, subclass_input);
|
||||||
|
|
Loading…
Reference in a new issue