mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-11 21:31:29 +11:00
x11: Destroy dropped windows; handle WM_DELETE_WINDOW (#416)
Fixes #79 #414 This changes the implementation of Drop for Window to send a WM_DELETE_WINDOW ClientMessage, offloading all the cleanup and window destruction to the event loop. Unsurprisingly, this entails that the event loop now handles WM_DELETE_WINDOW using the behavior that was previously contained in Window's Drop implementation, along with destroying the Window. Not only does this mean that dropped windows are closed, but also that clicking the × button on the window actually closes it now. The previous implemention of Drop was also broken, as the event loop would be (seemingly permenanently) frozen after its invocation. That was caused specifically by the mutex locking, and is no longer an issue now that the locking is done in the event loop. While I don't have full confidence that it makes sense for the Drop implementation to behave this way, this is nonetheless a significant improvement. The previous behavior led to inconsistent state, panics, and event loop breakage, along with not actually destroying the window. This additionally makes the assumption that users don't need Focused or CursorLeft events for the destroyed window, as Closed is adequate to indicate unfocus, and users may not expect to receive events for closed/dropped windows. In my testing, those specific events were sent immediately after the window was destroyed, though this sort of behavior could be WM-specific. I've opted to explicitly suppress those events in the case of the window no longer existing.
This commit is contained in:
parent
76118af341
commit
d667a395b6
|
@ -1,5 +1,6 @@
|
||||||
# Unreleased
|
# Unreleased
|
||||||
|
|
||||||
|
- On X11, dropping a `Window` actually closes it now, and clicking the window's × button (or otherwise having the WM signal to close it) will result in the window closing.
|
||||||
- Added `WindowBuilderExt` methods for macos: `with_titlebar_transparent`,
|
- Added `WindowBuilderExt` methods for macos: `with_titlebar_transparent`,
|
||||||
`with_title_hidden`, `with_titlebar_buttons_hidden`,
|
`with_title_hidden`, `with_titlebar_buttons_hidden`,
|
||||||
`with_fullsize_content_view`.
|
`with_fullsize_content_view`.
|
||||||
|
|
|
@ -217,7 +217,22 @@ impl EventsLoop {
|
||||||
let window_id = mkwid(window);
|
let window_id = mkwid(window);
|
||||||
|
|
||||||
if client_msg.data.get_long(0) as ffi::Atom == self.wm_delete_window {
|
if client_msg.data.get_long(0) as ffi::Atom == self.wm_delete_window {
|
||||||
callback(Event::WindowEvent { window_id, event: WindowEvent::Closed })
|
callback(Event::WindowEvent { window_id, event: WindowEvent::Closed });
|
||||||
|
|
||||||
|
let mut windows = self.windows.lock().unwrap();
|
||||||
|
let window_data = windows.remove(&WindowId(window));
|
||||||
|
let _lock = GLOBAL_XOPENIM_LOCK.lock().unwrap();
|
||||||
|
unsafe {
|
||||||
|
if let Some(window_data) = window_data {
|
||||||
|
(self.display.xlib.XDestroyIC)(window_data.ic);
|
||||||
|
(self.display.xlib.XCloseIM)(window_data.im);
|
||||||
|
self.display.check_errors()
|
||||||
|
.expect("Failed to close XIM");
|
||||||
|
}
|
||||||
|
(self.display.xlib.XDestroyWindow)(self.display.display, window);
|
||||||
|
self.display.check_errors()
|
||||||
|
.expect("Failed to destroy window");
|
||||||
|
}
|
||||||
} else if client_msg.message_type == self.dnd.atoms.enter {
|
} else if client_msg.message_type == self.dnd.atoms.enter {
|
||||||
let source_window = client_msg.data.get_long(0) as c_ulong;
|
let source_window = client_msg.data.get_long(0) as c_ulong;
|
||||||
let flags = client_msg.data.get_long(1);
|
let flags = client_msg.data.get_long(1);
|
||||||
|
@ -357,20 +372,23 @@ impl EventsLoop {
|
||||||
// Gymnastics to ensure self.windows isn't locked when we invoke callback
|
// Gymnastics to ensure self.windows isn't locked when we invoke callback
|
||||||
let (resized, moved) = {
|
let (resized, moved) = {
|
||||||
let mut windows = self.windows.lock().unwrap();
|
let mut windows = self.windows.lock().unwrap();
|
||||||
let window_data = windows.get_mut(&WindowId(window)).unwrap();
|
if let Some(window_data) = windows.get_mut(&WindowId(window)) {
|
||||||
if window_data.config.is_none() {
|
if window_data.config.is_none() {
|
||||||
window_data.config = Some(WindowConfig::new(xev));
|
window_data.config = Some(WindowConfig::new(xev));
|
||||||
(true, true)
|
(true, true)
|
||||||
|
} else {
|
||||||
|
let window_state = window_data.config.as_mut().unwrap();
|
||||||
|
(if window_state.size != new_size {
|
||||||
|
window_state.size = new_size;
|
||||||
|
true
|
||||||
|
} else { false },
|
||||||
|
if window_state.position != new_position {
|
||||||
|
window_state.position = new_position;
|
||||||
|
true
|
||||||
|
} else { false })
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
let window_state = window_data.config.as_mut().unwrap();
|
return;
|
||||||
(if window_state.size != new_size {
|
|
||||||
window_state.size = new_size;
|
|
||||||
true
|
|
||||||
} else { false },
|
|
||||||
if window_state.position != new_position {
|
|
||||||
window_state.position = new_position;
|
|
||||||
true
|
|
||||||
} else { false })
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
if resized {
|
if resized {
|
||||||
|
@ -444,7 +462,13 @@ impl EventsLoop {
|
||||||
|
|
||||||
const INIT_BUFF_SIZE: usize = 16;
|
const INIT_BUFF_SIZE: usize = 16;
|
||||||
let mut windows = self.windows.lock().unwrap();
|
let mut windows = self.windows.lock().unwrap();
|
||||||
let window_data = windows.get_mut(&WindowId(window)).unwrap();
|
let window_data = {
|
||||||
|
if let Some(window_data) = windows.get_mut(&WindowId(window)) {
|
||||||
|
window_data
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
/* buffer allocated on heap instead of stack, due to the possible
|
/* buffer allocated on heap instead of stack, due to the possible
|
||||||
* reallocation */
|
* reallocation */
|
||||||
let mut buffer: Vec<u8> = vec![mem::uninitialized(); INIT_BUFF_SIZE];
|
let mut buffer: Vec<u8> = vec![mem::uninitialized(); INIT_BUFF_SIZE];
|
||||||
|
@ -494,9 +518,16 @@ impl EventsLoop {
|
||||||
let xev: &ffi::XIDeviceEvent = unsafe { &*(xev.data as *const _) };
|
let xev: &ffi::XIDeviceEvent = unsafe { &*(xev.data as *const _) };
|
||||||
let window_id = mkwid(xev.event);
|
let window_id = mkwid(xev.event);
|
||||||
let device_id = mkdid(xev.deviceid);
|
let device_id = mkdid(xev.deviceid);
|
||||||
if (xev.flags & ffi::XIPointerEmulated) != 0 && self.windows.lock().unwrap().get(&WindowId(xev.event)).unwrap().multitouch {
|
if (xev.flags & ffi::XIPointerEmulated) != 0 {
|
||||||
// Deliver multi-touch events instead of emulated mouse events.
|
let windows = self.windows.lock().unwrap();
|
||||||
return;
|
if let Some(window_data) = windows.get(&WindowId(xev.event)) {
|
||||||
|
if window_data.multitouch {
|
||||||
|
// Deliver multi-touch events instead of emulated mouse events.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let modifiers = ModifiersState::from(xev.mods);
|
let modifiers = ModifiersState::from(xev.mods);
|
||||||
|
@ -578,7 +609,13 @@ impl EventsLoop {
|
||||||
// Gymnastics to ensure self.windows isn't locked when we invoke callback
|
// Gymnastics to ensure self.windows isn't locked when we invoke callback
|
||||||
if {
|
if {
|
||||||
let mut windows = self.windows.lock().unwrap();
|
let mut windows = self.windows.lock().unwrap();
|
||||||
let window_data = windows.get_mut(&WindowId(xev.event)).unwrap();
|
let window_data = {
|
||||||
|
if let Some(window_data) = windows.get_mut(&WindowId(xev.event)) {
|
||||||
|
window_data
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
if Some(new_cursor_pos) != window_data.cursor_pos {
|
if Some(new_cursor_pos) != window_data.cursor_pos {
|
||||||
window_data.cursor_pos = Some(new_cursor_pos);
|
window_data.cursor_pos = Some(new_cursor_pos);
|
||||||
true
|
true
|
||||||
|
@ -678,29 +715,43 @@ impl EventsLoop {
|
||||||
ffi::XI_Leave => {
|
ffi::XI_Leave => {
|
||||||
let xev: &ffi::XILeaveEvent = unsafe { &*(xev.data as *const _) };
|
let xev: &ffi::XILeaveEvent = unsafe { &*(xev.data as *const _) };
|
||||||
|
|
||||||
callback(Event::WindowEvent {
|
// Leave, FocusIn, and FocusOut can be received by a window that's already
|
||||||
window_id: mkwid(xev.event),
|
// been destroyed, which the user presumably doesn't want to deal with.
|
||||||
event: CursorLeft { device_id: mkdid(xev.deviceid) },
|
let window_closed = self.windows
|
||||||
});
|
.lock()
|
||||||
|
.unwrap()
|
||||||
|
.get(&WindowId(xev.event))
|
||||||
|
.is_none();
|
||||||
|
|
||||||
|
if !window_closed {
|
||||||
|
callback(Event::WindowEvent {
|
||||||
|
window_id: mkwid(xev.event),
|
||||||
|
event: CursorLeft { device_id: mkdid(xev.deviceid) },
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ffi::XI_FocusIn => {
|
ffi::XI_FocusIn => {
|
||||||
let xev: &ffi::XIFocusInEvent = unsafe { &*(xev.data as *const _) };
|
let xev: &ffi::XIFocusInEvent = unsafe { &*(xev.data as *const _) };
|
||||||
|
|
||||||
let window_id = mkwid(xev.event);
|
let window_id = mkwid(xev.event);
|
||||||
|
|
||||||
unsafe {
|
let mut windows = self.windows.lock().unwrap();
|
||||||
let mut windows = self.windows.lock().unwrap();
|
if let Some(window_data) = windows.get_mut(&WindowId(xev.event)) {
|
||||||
let window_data = windows.get_mut(&WindowId(xev.event)).unwrap();
|
unsafe {
|
||||||
(self.display.xlib.XSetICFocus)(window_data.ic);
|
(self.display.xlib.XSetICFocus)(window_data.ic);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(Event::WindowEvent { window_id, event: Focused(true) });
|
callback(Event::WindowEvent { window_id, event: Focused(true) });
|
||||||
|
|
||||||
// The deviceid for this event is for a keyboard instead of a pointer, so
|
// The deviceid for this event is for a keyboard instead of a pointer,
|
||||||
// we have to do a little extra work.
|
// so we have to do a little extra work.
|
||||||
let device_info = DeviceInfo::get(&self.display, xev.deviceid);
|
let device_info = DeviceInfo::get(&self.display, xev.deviceid);
|
||||||
// For master devices, the attachment field contains the ID of the paired
|
// For master devices, the attachment field contains the ID of the
|
||||||
// master device; for the master keyboard, the attachment is the master
|
// paired master device; for the master keyboard, the attachment is
|
||||||
// pointer, and vice versa.
|
// the master pointer, and vice versa.
|
||||||
let pointer_id = unsafe { (*device_info.info) }.attachment;
|
let pointer_id = unsafe { (*device_info.info) }.attachment;
|
||||||
|
|
||||||
callback(Event::WindowEvent {
|
callback(Event::WindowEvent {
|
||||||
|
@ -714,11 +765,16 @@ impl EventsLoop {
|
||||||
}
|
}
|
||||||
ffi::XI_FocusOut => {
|
ffi::XI_FocusOut => {
|
||||||
let xev: &ffi::XIFocusOutEvent = unsafe { &*(xev.data as *const _) };
|
let xev: &ffi::XIFocusOutEvent = unsafe { &*(xev.data as *const _) };
|
||||||
unsafe {
|
|
||||||
let mut windows = self.windows.lock().unwrap();
|
let mut windows = self.windows.lock().unwrap();
|
||||||
let window_data = windows.get_mut(&WindowId(xev.event)).unwrap();
|
if let Some(window_data) = windows.get_mut(&WindowId(xev.event)) {
|
||||||
(self.display.xlib.XUnsetICFocus)(window_data.ic);
|
unsafe {
|
||||||
|
(self.display.xlib.XUnsetICFocus)(window_data.ic);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(Event::WindowEvent {
|
callback(Event::WindowEvent {
|
||||||
window_id: mkwid(xev.event),
|
window_id: mkwid(xev.event),
|
||||||
event: Focused(false),
|
event: Focused(false),
|
||||||
|
@ -1020,13 +1076,27 @@ impl Window {
|
||||||
impl Drop for Window {
|
impl Drop for Window {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
if let (Some(windows), Some(display)) = (self.windows.upgrade(), self.display.upgrade()) {
|
if let (Some(windows), Some(display)) = (self.windows.upgrade(), self.display.upgrade()) {
|
||||||
let mut windows = windows.lock().unwrap();
|
// It's possible for the Window object to outlive the actual window, so we need to
|
||||||
let w = windows.remove(&self.window.id()).unwrap();
|
// check for that, lest the program explode with BadWindow errors soon after this.
|
||||||
let _lock = GLOBAL_XOPENIM_LOCK.lock().unwrap();
|
let window_closed = windows
|
||||||
unsafe {
|
.lock()
|
||||||
(display.xlib.XDestroyIC)(w.ic);
|
.unwrap()
|
||||||
(display.xlib.XCloseIM)(w.im);
|
.get(&self.window.id())
|
||||||
}
|
.is_none();
|
||||||
|
if !window_closed { unsafe {
|
||||||
|
let wm_protocols_atom = util::get_atom(&display, b"WM_PROTOCOLS\0")
|
||||||
|
.expect("Failed to call XInternAtom (WM_PROTOCOLS)");
|
||||||
|
let wm_delete_atom = util::get_atom(&display, b"WM_DELETE_WINDOW\0")
|
||||||
|
.expect("Failed to call XInternAtom (WM_DELETE_WINDOW)");
|
||||||
|
util::send_client_msg(
|
||||||
|
&display,
|
||||||
|
self.window.id().0,
|
||||||
|
self.window.id().0,
|
||||||
|
wm_protocols_atom,
|
||||||
|
None,
|
||||||
|
(wm_delete_atom as _, ffi::CurrentTime as _, 0, 0, 0),
|
||||||
|
).expect("Failed to send window deletion message");
|
||||||
|
} }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue