mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-26 03:36:32 +11:00
On Windows, fix request_redraw() related panics (#1461)
* On Windows, fix request_redraw() related panics
These panics were introduced by 6a330a2894
Fixes https://github.com/rust-windowing/winit/issues/1391
Fixes https://github.com/rust-windowing/winit/issues/1400
Fixes https://github.com/rust-windowing/winit/issues/1466
Probably fixes other related issues
See https://github.com/rust-windowing/winit/issues/1429
* On Windows, replace all calls to UpdateWindow by calls to InvalidateRgn
This avoids directly sending a WM_PAINT message,
which might cause buffering of RedrawRequested events.
We don't want to buffer RedrawRequested events because:
- we wan't to handle RedrawRequested during processing of WM_PAINT messages
- state transitionning is broken when handling buffered RedrawRequested events
Fixes https://github.com/rust-windowing/winit/issues/1469
* On Windows, panic if we are trying to buffer a RedrawRequested event
* On Windows, move modal loop jumpstart to set_modal_loop() method
This fixes a panic.
Note that the WM_PAINT event is now sent to the modal_redraw_method
which is more correct and avoids an unecessary redraw of the window.
Relates to but does does not fix https://github.com/rust-windowing/winit/issues/1484
* On Window, filter by paint messages when draining paint messages
This seems to prevent PeekMessage from dispatching unrelated sent messages
* Change recently added panic/assert calls with warn calls
This makes the code less panicky...
And actually, winit's Windoww callbacks should not panic
because the panic will unwind into Windows code.
It is currently undefined behavior to unwind from Rust code into foreign code.
See https://doc.rust-lang.org/std/panic/fn.catch_unwind.html
* add comments to clarify WM_PAINT handling in non modal loop
* made redraw_events_cleared more explicit and more comments
This commit is contained in:
parent
cbb60d29a2
commit
2f27f64cdb
6 changed files with 158 additions and 277 deletions
|
@ -1,6 +1,7 @@
|
||||||
# Unreleased
|
# Unreleased
|
||||||
|
|
||||||
- On Windows, fix minor timing issue in wait_until_time_or_msg
|
- On Windows, fix minor timing issue in wait_until_time_or_msg
|
||||||
|
- On Windows, rework handling of request_redraw() to address panics.
|
||||||
- On macOS, fix `set_simple_screen` to remember frame excluding title bar.
|
- On macOS, fix `set_simple_screen` to remember frame excluding title bar.
|
||||||
- On Wayland, fix coordinates in touch events when scale factor isn't 1.
|
- On Wayland, fix coordinates in touch events when scale factor isn't 1.
|
||||||
- On Wayland, fix color from `close_button_icon_color` not applying.
|
- On Wayland, fix color from `close_button_icon_color` not applying.
|
||||||
|
|
|
@ -218,69 +218,68 @@ impl<T: 'static> EventLoop<T> {
|
||||||
}
|
}
|
||||||
|
|
||||||
runner.new_events();
|
runner.new_events();
|
||||||
loop {
|
|
||||||
if !unread_message_exists {
|
|
||||||
if 0 == winuser::PeekMessageW(&mut msg, ptr::null_mut(), 0, 0, 1) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if msg.message == winuser::WM_PAINT {
|
|
||||||
unread_message_exists = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
winuser::TranslateMessage(&mut msg);
|
|
||||||
winuser::DispatchMessageW(&mut msg);
|
|
||||||
|
|
||||||
unread_message_exists = false;
|
|
||||||
}
|
|
||||||
runner.main_events_cleared();
|
|
||||||
loop {
|
loop {
|
||||||
if !unread_message_exists {
|
if !unread_message_exists {
|
||||||
if 0 == winuser::PeekMessageW(
|
if 0 == winuser::PeekMessageW(
|
||||||
&mut msg,
|
&mut msg,
|
||||||
ptr::null_mut(),
|
ptr::null_mut(),
|
||||||
winuser::WM_PAINT,
|
0,
|
||||||
winuser::WM_PAINT,
|
0,
|
||||||
1,
|
winuser::PM_REMOVE,
|
||||||
) {
|
) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
winuser::TranslateMessage(&mut msg);
|
winuser::TranslateMessage(&mut msg);
|
||||||
winuser::DispatchMessageW(&mut msg);
|
winuser::DispatchMessageW(&mut msg);
|
||||||
|
|
||||||
unread_message_exists = false;
|
unread_message_exists = false;
|
||||||
}
|
|
||||||
if runner.redraw_events_cleared().events_buffered() {
|
|
||||||
if runner.control_flow() == ControlFlow::Exit {
|
|
||||||
break 'main;
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if !unread_message_exists {
|
if msg.message == winuser::WM_PAINT {
|
||||||
let control_flow = runner.control_flow();
|
// An "external" redraw was requested.
|
||||||
match control_flow {
|
// Note that the WM_PAINT has been dispatched and
|
||||||
ControlFlow::Exit => break 'main,
|
// has caused the event loop to emit the MainEventsCleared event.
|
||||||
ControlFlow::Wait => {
|
// See EventLoopRunner::process_event().
|
||||||
if 0 == winuser::GetMessageW(&mut msg, ptr::null_mut(), 0, 0) {
|
// The call to main_events_cleared() below will do nothing.
|
||||||
break 'main;
|
break;
|
||||||
}
|
|
||||||
unread_message_exists = true;
|
|
||||||
}
|
|
||||||
ControlFlow::WaitUntil(resume_time) => {
|
|
||||||
wait_until_time_or_msg(resume_time);
|
|
||||||
}
|
|
||||||
ControlFlow::Poll => (),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Make sure we emit the MainEventsCleared event if no WM_PAINT message was received.
|
||||||
|
runner.main_events_cleared();
|
||||||
|
// Drain eventual WM_PAINT messages sent if user called request_redraw()
|
||||||
|
// during handling of MainEventsCleared.
|
||||||
|
loop {
|
||||||
|
if 0 == winuser::PeekMessageW(
|
||||||
|
&mut msg,
|
||||||
|
ptr::null_mut(),
|
||||||
|
winuser::WM_PAINT,
|
||||||
|
winuser::WM_PAINT,
|
||||||
|
winuser::PM_QS_PAINT | winuser::PM_REMOVE,
|
||||||
|
) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
winuser::TranslateMessage(&mut msg);
|
||||||
|
winuser::DispatchMessageW(&mut msg);
|
||||||
|
}
|
||||||
|
runner.redraw_events_cleared();
|
||||||
|
match runner.control_flow() {
|
||||||
|
ControlFlow::Exit => break 'main,
|
||||||
|
ControlFlow::Wait => {
|
||||||
|
if 0 == winuser::GetMessageW(&mut msg, ptr::null_mut(), 0, 0) {
|
||||||
|
break 'main;
|
||||||
|
}
|
||||||
|
unread_message_exists = true;
|
||||||
|
}
|
||||||
|
ControlFlow::WaitUntil(resume_time) => {
|
||||||
|
wait_until_time_or_msg(resume_time);
|
||||||
|
}
|
||||||
|
ControlFlow::Poll => (),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe {
|
runner.destroy_loop();
|
||||||
runner.call_event_handler(Event::LoopDestroyed);
|
|
||||||
}
|
|
||||||
runner.destroy_runner();
|
runner.destroy_runner();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -652,13 +651,6 @@ unsafe extern "system" fn public_window_callback<T: 'static>(
|
||||||
}
|
}
|
||||||
|
|
||||||
winuser::WM_NCLBUTTONDOWN => {
|
winuser::WM_NCLBUTTONDOWN => {
|
||||||
// jumpstart the modal loop
|
|
||||||
winuser::RedrawWindow(
|
|
||||||
window,
|
|
||||||
ptr::null(),
|
|
||||||
ptr::null_mut(),
|
|
||||||
winuser::RDW_INTERNALPAINT,
|
|
||||||
);
|
|
||||||
if wparam == winuser::HTCAPTION as _ {
|
if wparam == winuser::HTCAPTION as _ {
|
||||||
winuser::PostMessageW(window, winuser::WM_MOUSEMOVE, 0, 0);
|
winuser::PostMessageW(window, winuser::WM_MOUSEMOVE, 0, 0);
|
||||||
}
|
}
|
||||||
|
@ -688,7 +680,6 @@ unsafe extern "system" fn public_window_callback<T: 'static>(
|
||||||
}
|
}
|
||||||
|
|
||||||
winuser::WM_PAINT => {
|
winuser::WM_PAINT => {
|
||||||
subclass_input.event_loop_runner.main_events_cleared();
|
|
||||||
subclass_input.send_event(Event::RedrawRequested(RootWindowId(WindowId(window))));
|
subclass_input.send_event(Event::RedrawRequested(RootWindowId(WindowId(window))));
|
||||||
commctrl::DefSubclassProc(window, msg, wparam, lparam)
|
commctrl::DefSubclassProc(window, msg, wparam, lparam)
|
||||||
}
|
}
|
||||||
|
@ -1780,50 +1771,39 @@ unsafe extern "system" fn thread_event_target_callback<T: 'static>(
|
||||||
};
|
};
|
||||||
let in_modal_loop = subclass_input.event_loop_runner.in_modal_loop();
|
let in_modal_loop = subclass_input.event_loop_runner.in_modal_loop();
|
||||||
if in_modal_loop {
|
if in_modal_loop {
|
||||||
|
let runner = &subclass_input.event_loop_runner;
|
||||||
|
runner.main_events_cleared();
|
||||||
|
// Drain eventual WM_PAINT messages sent if user called request_redraw()
|
||||||
|
// during handling of MainEventsCleared.
|
||||||
let mut msg = mem::zeroed();
|
let mut msg = mem::zeroed();
|
||||||
if 0 == winuser::PeekMessageW(&mut msg, ptr::null_mut(), 0, 0, 0) {
|
loop {
|
||||||
if msg.message != 0 && msg.message != winuser::WM_PAINT {
|
if 0 == winuser::PeekMessageW(
|
||||||
queue_call_again();
|
&mut msg,
|
||||||
return 0;
|
ptr::null_mut(),
|
||||||
|
winuser::WM_PAINT,
|
||||||
|
winuser::WM_PAINT,
|
||||||
|
winuser::PM_QS_PAINT | winuser::PM_REMOVE,
|
||||||
|
) {
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
subclass_input.event_loop_runner.main_events_cleared();
|
if msg.hwnd != window {
|
||||||
loop {
|
winuser::TranslateMessage(&mut msg);
|
||||||
if 0 == winuser::PeekMessageW(
|
winuser::DispatchMessageW(&mut msg);
|
||||||
&mut msg,
|
|
||||||
ptr::null_mut(),
|
|
||||||
winuser::WM_PAINT,
|
|
||||||
winuser::WM_PAINT,
|
|
||||||
1,
|
|
||||||
) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if msg.hwnd != window {
|
|
||||||
winuser::TranslateMessage(&mut msg);
|
|
||||||
winuser::DispatchMessageW(&mut msg);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
runner.redraw_events_cleared();
|
||||||
// we don't borrow until here because TODO SAFETY
|
match runner.control_flow() {
|
||||||
let runner = &subclass_input.event_loop_runner;
|
// Waiting is handled by the modal loop.
|
||||||
if runner.redraw_events_cleared().events_buffered() {
|
ControlFlow::Exit | ControlFlow::Wait => runner.new_events(),
|
||||||
queue_call_again();
|
ControlFlow::WaitUntil(resume_time) => {
|
||||||
runner.new_events();
|
wait_until_time_or_msg(resume_time);
|
||||||
} else {
|
runner.new_events();
|
||||||
match runner.control_flow() {
|
queue_call_again();
|
||||||
// Waiting is handled by the modal loop.
|
}
|
||||||
ControlFlow::Exit | ControlFlow::Wait => runner.new_events(),
|
ControlFlow::Poll => {
|
||||||
ControlFlow::WaitUntil(resume_time) => {
|
runner.new_events();
|
||||||
wait_until_time_or_msg(resume_time);
|
queue_call_again();
|
||||||
runner.new_events();
|
|
||||||
queue_call_again();
|
|
||||||
}
|
|
||||||
ControlFlow::Poll => {
|
|
||||||
runner.new_events();
|
|
||||||
queue_call_again();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ use crate::{
|
||||||
dpi::PhysicalSize,
|
dpi::PhysicalSize,
|
||||||
event::{Event, StartCause, WindowEvent},
|
event::{Event, StartCause, WindowEvent},
|
||||||
event_loop::ControlFlow,
|
event_loop::ControlFlow,
|
||||||
platform_impl::platform::{event_loop::EventLoop, util},
|
platform_impl::platform::event_loop::{util, EventLoop},
|
||||||
window::WindowId,
|
window::WindowId,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -14,8 +14,8 @@ pub(crate) type EventLoopRunnerShared<T> = Rc<ELRShared<T>>;
|
||||||
pub(crate) struct ELRShared<T: 'static> {
|
pub(crate) struct ELRShared<T: 'static> {
|
||||||
runner: RefCell<Option<EventLoopRunner<T>>>,
|
runner: RefCell<Option<EventLoopRunner<T>>>,
|
||||||
buffer: RefCell<VecDeque<BufferedEvent<T>>>,
|
buffer: RefCell<VecDeque<BufferedEvent<T>>>,
|
||||||
redraw_buffer: Rc<RefCell<VecDeque<WindowId>>>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct EventLoopRunner<T: 'static> {
|
struct EventLoopRunner<T: 'static> {
|
||||||
control_flow: ControlFlow,
|
control_flow: ControlFlow,
|
||||||
runner_state: RunnerState,
|
runner_state: RunnerState,
|
||||||
|
@ -23,8 +23,8 @@ struct EventLoopRunner<T: 'static> {
|
||||||
in_modal_loop: bool,
|
in_modal_loop: bool,
|
||||||
event_handler: Box<dyn FnMut(Event<'_, T>, &mut ControlFlow)>,
|
event_handler: Box<dyn FnMut(Event<'_, T>, &mut ControlFlow)>,
|
||||||
panic_error: Option<PanicError>,
|
panic_error: Option<PanicError>,
|
||||||
redraw_buffer: Rc<RefCell<VecDeque<WindowId>>>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type PanicError = Box<dyn Any + Send + 'static>;
|
pub type PanicError = Box<dyn Any + Send + 'static>;
|
||||||
|
|
||||||
pub enum BufferedEvent<T: 'static> {
|
pub enum BufferedEvent<T: 'static> {
|
||||||
|
@ -32,22 +32,6 @@ pub enum BufferedEvent<T: 'static> {
|
||||||
ScaleFactorChanged(WindowId, f64, PhysicalSize<u32>),
|
ScaleFactorChanged(WindowId, f64, PhysicalSize<u32>),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
|
||||||
pub enum AreEventsBuffered {
|
|
||||||
EventsBuffered,
|
|
||||||
ReadyToSleep,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AreEventsBuffered {
|
|
||||||
pub fn events_buffered(&self) -> bool {
|
|
||||||
match self {
|
|
||||||
Self::EventsBuffered => true,
|
|
||||||
Self::ReadyToSleep => false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T> BufferedEvent<T> {
|
impl<T> BufferedEvent<T> {
|
||||||
pub fn from_event(event: Event<'_, T>) -> BufferedEvent<T> {
|
pub fn from_event(event: Event<'_, T>) -> BufferedEvent<T> {
|
||||||
match event {
|
match event {
|
||||||
|
@ -89,7 +73,6 @@ impl<T> ELRShared<T> {
|
||||||
ELRShared {
|
ELRShared {
|
||||||
runner: RefCell::new(None),
|
runner: RefCell::new(None),
|
||||||
buffer: RefCell::new(VecDeque::new()),
|
buffer: RefCell::new(VecDeque::new()),
|
||||||
redraw_buffer: Default::default(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,16 +80,11 @@ impl<T> ELRShared<T> {
|
||||||
where
|
where
|
||||||
F: FnMut(Event<'_, T>, &mut ControlFlow),
|
F: FnMut(Event<'_, T>, &mut ControlFlow),
|
||||||
{
|
{
|
||||||
let mut runner = EventLoopRunner::new(event_loop, self.redraw_buffer.clone(), f);
|
let mut runner = EventLoopRunner::new(event_loop, f);
|
||||||
{
|
{
|
||||||
let mut runner_ref = self.runner.borrow_mut();
|
let mut runner_ref = self.runner.borrow_mut();
|
||||||
loop {
|
// Dispatch any events that were buffered during the creation of the window
|
||||||
let event = self.buffer.borrow_mut().pop_front();
|
self.dispatch_buffered_events(&mut runner);
|
||||||
match event {
|
|
||||||
Some(e) => e.dispatch_event(|e| runner.process_event(e)),
|
|
||||||
None => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*runner_ref = Some(runner);
|
*runner_ref = Some(runner);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -119,80 +97,46 @@ impl<T> ELRShared<T> {
|
||||||
let mut runner_ref = self.runner.borrow_mut();
|
let mut runner_ref = self.runner.borrow_mut();
|
||||||
if let Some(ref mut runner) = *runner_ref {
|
if let Some(ref mut runner) = *runner_ref {
|
||||||
runner.new_events();
|
runner.new_events();
|
||||||
loop {
|
// Dispatch any events that were buffered during the call `new_events`
|
||||||
let buffered_event_opt = self.buffer.borrow_mut().pop_front();
|
self.dispatch_buffered_events(runner);
|
||||||
match buffered_event_opt {
|
|
||||||
Some(e) => e.dispatch_event(|e| runner.process_event(e)),
|
|
||||||
None => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) unsafe fn send_event(&self, event: Event<'_, T>) {
|
pub(crate) fn send_event(&self, event: Event<'_, T>) {
|
||||||
let handling_redraw = self
|
if let Err(event) = self.send_event_unbuffered(event) {
|
||||||
.runner
|
// If the runner is already borrowed, we're in the middle of an event loop invocation.
|
||||||
.borrow()
|
// Add the event to a buffer to be processed later.
|
||||||
.as_ref()
|
|
||||||
.map(|r| RunnerState::HandlingRedraw == r.runner_state)
|
|
||||||
.unwrap_or(false);
|
|
||||||
let mut send = None;
|
|
||||||
if handling_redraw {
|
|
||||||
if let Event::RedrawRequested(_) = event {
|
if let Event::RedrawRequested(_) = event {
|
||||||
send = Some(event);
|
panic!("buffering RedrawRequested event");
|
||||||
} else {
|
|
||||||
self.buffer_event(event);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
send = Some(event);
|
|
||||||
}
|
|
||||||
if let Some(event) = send {
|
|
||||||
if let Err(event) = self.send_event_unbuffered(event) {
|
|
||||||
// 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.buffer_event(event);
|
|
||||||
}
|
}
|
||||||
|
self.buffer
|
||||||
|
.borrow_mut()
|
||||||
|
.push_back(BufferedEvent::from_event(event));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe fn send_event_unbuffered<'e>(&self, event: Event<'e, T>) -> Result<(), Event<'e, T>> {
|
fn send_event_unbuffered<'e>(&self, event: Event<'e, T>) -> Result<(), Event<'e, T>> {
|
||||||
if let Ok(mut runner_ref) = self.runner.try_borrow_mut() {
|
if let Ok(mut runner_ref) = self.runner.try_borrow_mut() {
|
||||||
if let Some(ref mut runner) = *runner_ref {
|
if let Some(ref mut runner) = *runner_ref {
|
||||||
runner.process_event(event);
|
runner.process_event(event);
|
||||||
|
// Dispatch any events that were buffered during the call to `process_event`.
|
||||||
let handling_redraw = if let RunnerState::HandlingRedraw = runner.runner_state {
|
self.dispatch_buffered_events(runner);
|
||||||
true
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
};
|
|
||||||
|
|
||||||
if !handling_redraw {
|
|
||||||
// Dispatch any events that were buffered during the call to `process_event`.
|
|
||||||
loop {
|
|
||||||
// We do this instead of using a `while let` loop because if we use a `while let`
|
|
||||||
// loop the reference returned `borrow_mut()` doesn't get dropped until the end
|
|
||||||
// of the loop's body and attempts to add events to the event buffer while in
|
|
||||||
// `process_event` will fail.
|
|
||||||
let buffered_event_opt = self.buffer.borrow_mut().pop_front();
|
|
||||||
match buffered_event_opt {
|
|
||||||
Some(e) => e.dispatch_event(|e| runner.process_event(e)),
|
|
||||||
None => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Err(event)
|
Err(event)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) unsafe fn call_event_handler(&self, event: Event<'static, T>) {
|
fn dispatch_buffered_events(&self, runner: &mut EventLoopRunner<T>) {
|
||||||
if let Ok(mut runner_ref) = self.runner.try_borrow_mut() {
|
// We do this instead of using a `while let` loop because if we use a `while let`
|
||||||
if let Some(ref mut runner) = *runner_ref {
|
// loop the reference returned `borrow_mut()` doesn't get dropped until the end
|
||||||
runner.call_event_handler(event);
|
// of the loop's body and attempts to add events to the event buffer while in
|
||||||
return;
|
// `process_event` will fail.
|
||||||
|
loop {
|
||||||
|
let buffered_event_opt = self.buffer.borrow_mut().pop_front();
|
||||||
|
match buffered_event_opt {
|
||||||
|
Some(e) => e.dispatch_event(|e| runner.process_event(e)),
|
||||||
|
None => break,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -201,17 +145,27 @@ impl<T> ELRShared<T> {
|
||||||
let mut runner_ref = self.runner.borrow_mut();
|
let mut runner_ref = self.runner.borrow_mut();
|
||||||
if let Some(ref mut runner) = *runner_ref {
|
if let Some(ref mut runner) = *runner_ref {
|
||||||
runner.main_events_cleared();
|
runner.main_events_cleared();
|
||||||
|
if !self.buffer.borrow().is_empty() {
|
||||||
|
warn!("Buffered events while dispatching MainEventsCleared");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn redraw_events_cleared(&self) -> AreEventsBuffered {
|
pub(crate) fn redraw_events_cleared(&self) {
|
||||||
let mut runner_ref = self.runner.borrow_mut();
|
let mut runner_ref = self.runner.borrow_mut();
|
||||||
if let Some(ref mut runner) = *runner_ref {
|
if let Some(ref mut runner) = *runner_ref {
|
||||||
runner.redraw_events_cleared();
|
runner.redraw_events_cleared();
|
||||||
|
if !self.buffer.borrow().is_empty() {
|
||||||
|
warn!("Buffered events while dispatching RedrawEventsCleared");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
match self.buffer.borrow().len() {
|
}
|
||||||
0 => AreEventsBuffered::ReadyToSleep,
|
|
||||||
_ => AreEventsBuffered::EventsBuffered,
|
pub(crate) fn destroy_loop(&self) {
|
||||||
|
if let Ok(mut runner_ref) = self.runner.try_borrow_mut() {
|
||||||
|
if let Some(ref mut runner) = *runner_ref {
|
||||||
|
runner.call_event_handler(Event::LoopDestroyed);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -228,6 +182,17 @@ impl<T> ELRShared<T> {
|
||||||
let mut runner_ref = self.runner.borrow_mut();
|
let mut runner_ref = self.runner.borrow_mut();
|
||||||
if let Some(ref mut runner) = *runner_ref {
|
if let Some(ref mut runner) = *runner_ref {
|
||||||
runner.in_modal_loop = in_modal_loop;
|
runner.in_modal_loop = in_modal_loop;
|
||||||
|
if in_modal_loop {
|
||||||
|
// jumpstart the modal loop
|
||||||
|
unsafe {
|
||||||
|
winuser::RedrawWindow(
|
||||||
|
runner.modal_redraw_window,
|
||||||
|
ptr::null(),
|
||||||
|
ptr::null_mut(),
|
||||||
|
winuser::RDW_INTERNALPAINT,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -248,18 +213,6 @@ impl<T> ELRShared<T> {
|
||||||
ControlFlow::Exit
|
ControlFlow::Exit
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn buffer_event(&self, event: Event<'_, T>) {
|
|
||||||
match event {
|
|
||||||
Event::RedrawRequested(window_id) => {
|
|
||||||
self.redraw_buffer.borrow_mut().push_back(window_id)
|
|
||||||
}
|
|
||||||
_ => self
|
|
||||||
.buffer
|
|
||||||
.borrow_mut()
|
|
||||||
.push_back(BufferedEvent::from_event(event)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
|
@ -275,15 +228,13 @@ enum RunnerState {
|
||||||
/// The event loop is handling the OS's events and sending them to the user's callback.
|
/// The event loop is handling the OS's events and sending them to the user's callback.
|
||||||
/// `NewEvents` has been sent, and `MainEventsCleared` hasn't.
|
/// `NewEvents` has been sent, and `MainEventsCleared` hasn't.
|
||||||
HandlingEvents,
|
HandlingEvents,
|
||||||
|
/// The event loop is handling the redraw events and sending them to the user's callback.
|
||||||
|
/// `MainEventsCleared` has been sent, and `RedrawEventsCleared` hasn't.
|
||||||
HandlingRedraw,
|
HandlingRedraw,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> EventLoopRunner<T> {
|
impl<T> EventLoopRunner<T> {
|
||||||
unsafe fn new<F>(
|
unsafe fn new<F>(event_loop: &EventLoop<T>, f: F) -> EventLoopRunner<T>
|
||||||
event_loop: &EventLoop<T>,
|
|
||||||
redraw_buffer: Rc<RefCell<VecDeque<WindowId>>>,
|
|
||||||
f: F,
|
|
||||||
) -> EventLoopRunner<T>
|
|
||||||
where
|
where
|
||||||
F: FnMut(Event<'_, T>, &mut ControlFlow),
|
F: FnMut(Event<'_, T>, &mut ControlFlow),
|
||||||
{
|
{
|
||||||
|
@ -297,7 +248,6 @@ impl<T> EventLoopRunner<T> {
|
||||||
Box<dyn FnMut(Event<'_, T>, &mut ControlFlow)>,
|
Box<dyn FnMut(Event<'_, T>, &mut ControlFlow)>,
|
||||||
>(Box::new(f)),
|
>(Box::new(f)),
|
||||||
panic_error: None,
|
panic_error: None,
|
||||||
redraw_buffer,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -323,6 +273,8 @@ impl<T> EventLoopRunner<T> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// When `NewEvents` gets sent after an idle depends on the control flow...
|
// When `NewEvents` gets sent after an idle depends on the control flow...
|
||||||
|
// Some `NewEvents` are deferred because not all Windows messages trigger an event_loop event.
|
||||||
|
// So we defer the `NewEvents` to when we actually process an event.
|
||||||
RunnerState::Idle(wait_start) => {
|
RunnerState::Idle(wait_start) => {
|
||||||
match self.control_flow {
|
match self.control_flow {
|
||||||
// If we're polling, send `NewEvents` and immediately move into event processing.
|
// If we're polling, send `NewEvents` and immediately move into event processing.
|
||||||
|
@ -411,24 +363,21 @@ impl<T> EventLoopRunner<T> {
|
||||||
// that was sent after `MainEventsCleared`.
|
// that was sent after `MainEventsCleared`.
|
||||||
ControlFlow::Poll => self.call_event_handler(Event::NewEvents(StartCause::Poll)),
|
ControlFlow::Poll => self.call_event_handler(Event::NewEvents(StartCause::Poll)),
|
||||||
}
|
}
|
||||||
|
self.runner_state = RunnerState::HandlingEvents;
|
||||||
}
|
}
|
||||||
|
|
||||||
match (self.runner_state, &event) {
|
match (self.runner_state, &event) {
|
||||||
(RunnerState::HandlingRedraw, Event::RedrawRequested(_)) => {
|
(RunnerState::HandlingEvents, Event::RedrawRequested(window_id)) => {
|
||||||
self.call_event_handler(event)
|
self.call_event_handler(Event::MainEventsCleared);
|
||||||
|
self.runner_state = RunnerState::HandlingRedraw;
|
||||||
|
self.call_event_handler(Event::RedrawRequested(*window_id));
|
||||||
}
|
}
|
||||||
(RunnerState::New, Event::RedrawRequested(_))
|
(RunnerState::HandlingRedraw, Event::RedrawRequested(window_id)) => {
|
||||||
| (RunnerState::Idle(..), Event::RedrawRequested(_)) => {
|
self.call_event_handler(Event::RedrawRequested(*window_id));
|
||||||
self.new_events();
|
|
||||||
self.main_events_cleared();
|
|
||||||
self.call_event_handler(event);
|
|
||||||
}
|
|
||||||
(_, Event::RedrawRequested(_)) => {
|
|
||||||
panic!("redraw event in non-redraw phase");
|
|
||||||
}
|
}
|
||||||
(RunnerState::HandlingRedraw, _) => {
|
(RunnerState::HandlingRedraw, _) => {
|
||||||
panic!(
|
warn!(
|
||||||
"Non-redraw event dispatched durning redraw phase: {:?}",
|
"non-redraw event in redraw phase: {:?}",
|
||||||
event.map_nonuser_event::<()>().ok()
|
event.map_nonuser_event::<()>().ok()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -439,16 +388,6 @@ impl<T> EventLoopRunner<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn flush_redraws(&mut self) {
|
|
||||||
loop {
|
|
||||||
let redraw_window_opt = self.redraw_buffer.borrow_mut().pop_front();
|
|
||||||
match redraw_window_opt {
|
|
||||||
Some(window_id) => self.process_event(Event::RedrawRequested(window_id)),
|
|
||||||
None => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main_events_cleared(&mut self) {
|
fn main_events_cleared(&mut self) {
|
||||||
match self.runner_state {
|
match self.runner_state {
|
||||||
// If we were handling events, send the MainEventsCleared message.
|
// If we were handling events, send the MainEventsCleared message.
|
||||||
|
@ -457,7 +396,9 @@ impl<T> EventLoopRunner<T> {
|
||||||
self.runner_state = RunnerState::HandlingRedraw;
|
self.runner_state = RunnerState::HandlingRedraw;
|
||||||
}
|
}
|
||||||
|
|
||||||
RunnerState::HandlingRedraw => (),
|
// We already cleared the main events, we don't have to do anything.
|
||||||
|
// This happens when process_events() processed a RedrawRequested event.
|
||||||
|
RunnerState::HandlingRedraw => {}
|
||||||
|
|
||||||
// If we *weren't* handling events, we don't have to do anything.
|
// If we *weren't* handling events, we don't have to do anything.
|
||||||
RunnerState::New | RunnerState::Idle(..) => (),
|
RunnerState::New | RunnerState::Idle(..) => (),
|
||||||
|
@ -469,6 +410,7 @@ impl<T> EventLoopRunner<T> {
|
||||||
// If we had deferred a Poll, send the Poll NewEvents and MainEventsCleared.
|
// If we had deferred a Poll, send the Poll NewEvents and MainEventsCleared.
|
||||||
ControlFlow::Poll => {
|
ControlFlow::Poll => {
|
||||||
self.call_event_handler(Event::NewEvents(StartCause::Poll));
|
self.call_event_handler(Event::NewEvents(StartCause::Poll));
|
||||||
|
self.runner_state = RunnerState::HandlingEvents;
|
||||||
self.call_event_handler(Event::MainEventsCleared);
|
self.call_event_handler(Event::MainEventsCleared);
|
||||||
self.runner_state = RunnerState::HandlingRedraw;
|
self.runner_state = RunnerState::HandlingRedraw;
|
||||||
}
|
}
|
||||||
|
@ -482,6 +424,7 @@ impl<T> EventLoopRunner<T> {
|
||||||
requested_resume: resume_time,
|
requested_resume: resume_time,
|
||||||
},
|
},
|
||||||
));
|
));
|
||||||
|
self.runner_state = RunnerState::HandlingEvents;
|
||||||
self.call_event_handler(Event::MainEventsCleared);
|
self.call_event_handler(Event::MainEventsCleared);
|
||||||
self.runner_state = RunnerState::HandlingRedraw;
|
self.runner_state = RunnerState::HandlingRedraw;
|
||||||
}
|
}
|
||||||
|
@ -496,57 +439,18 @@ impl<T> EventLoopRunner<T> {
|
||||||
|
|
||||||
fn redraw_events_cleared(&mut self) {
|
fn redraw_events_cleared(&mut self) {
|
||||||
match self.runner_state {
|
match self.runner_state {
|
||||||
// If we were handling events, send the MainEventsCleared message.
|
// If we were handling redraws, send the RedrawEventsCleared message.
|
||||||
RunnerState::HandlingEvents => {
|
|
||||||
self.call_event_handler(Event::MainEventsCleared);
|
|
||||||
self.runner_state = RunnerState::HandlingRedraw;
|
|
||||||
self.flush_redraws();
|
|
||||||
self.call_event_handler(Event::RedrawEventsCleared);
|
|
||||||
self.runner_state = RunnerState::Idle(Instant::now());
|
|
||||||
}
|
|
||||||
|
|
||||||
RunnerState::HandlingRedraw => {
|
RunnerState::HandlingRedraw => {
|
||||||
self.flush_redraws();
|
|
||||||
self.call_event_handler(Event::RedrawEventsCleared);
|
self.call_event_handler(Event::RedrawEventsCleared);
|
||||||
self.runner_state = RunnerState::Idle(Instant::now());
|
self.runner_state = RunnerState::Idle(Instant::now());
|
||||||
}
|
}
|
||||||
|
// No event was processed, we don't have to do anything.
|
||||||
// If we *weren't* handling events, we don't have to do anything.
|
RunnerState::DeferredNewEvents(_) => (),
|
||||||
RunnerState::New | RunnerState::Idle(..) => (),
|
// Should not happen.
|
||||||
|
_ => warn!(
|
||||||
// Some control flows require a NewEvents call even if no events were received. This
|
"unexpected state in redraw_events_cleared: {:?}",
|
||||||
// branch handles those.
|
self.runner_state
|
||||||
RunnerState::DeferredNewEvents(wait_start) => {
|
),
|
||||||
match self.control_flow {
|
|
||||||
// If we had deferred a Poll, send the Poll NewEvents and MainEventsCleared.
|
|
||||||
ControlFlow::Poll => {
|
|
||||||
self.call_event_handler(Event::NewEvents(StartCause::Poll));
|
|
||||||
self.call_event_handler(Event::MainEventsCleared);
|
|
||||||
self.flush_redraws();
|
|
||||||
self.call_event_handler(Event::RedrawEventsCleared);
|
|
||||||
}
|
|
||||||
// If we had deferred a WaitUntil and the resume time has since been reached,
|
|
||||||
// send the resume notification and MainEventsCleared event.
|
|
||||||
ControlFlow::WaitUntil(resume_time) => {
|
|
||||||
if Instant::now() >= resume_time {
|
|
||||||
self.call_event_handler(Event::NewEvents(
|
|
||||||
StartCause::ResumeTimeReached {
|
|
||||||
start: wait_start,
|
|
||||||
requested_resume: resume_time,
|
|
||||||
},
|
|
||||||
));
|
|
||||||
self.call_event_handler(Event::MainEventsCleared);
|
|
||||||
self.flush_redraws();
|
|
||||||
self.call_event_handler(Event::RedrawEventsCleared);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// If we deferred a wait and no events were received, the user doesn't have to
|
|
||||||
// get an event.
|
|
||||||
ControlFlow::Wait | ControlFlow::Exit => (),
|
|
||||||
}
|
|
||||||
// Mark that we've entered an idle state.
|
|
||||||
self.runner_state = RunnerState::Idle(wait_start)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -116,7 +116,7 @@ pub(crate) fn set_inner_size_physical(window: HWND, x: u32, y: u32) {
|
||||||
| winuser::SWP_NOMOVE
|
| winuser::SWP_NOMOVE
|
||||||
| winuser::SWP_NOACTIVATE,
|
| winuser::SWP_NOACTIVATE,
|
||||||
);
|
);
|
||||||
winuser::UpdateWindow(window);
|
winuser::InvalidateRgn(window, ptr::null_mut(), 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -187,7 +187,7 @@ impl Window {
|
||||||
| winuser::SWP_NOSIZE
|
| winuser::SWP_NOSIZE
|
||||||
| winuser::SWP_NOACTIVATE,
|
| winuser::SWP_NOACTIVATE,
|
||||||
);
|
);
|
||||||
winuser::UpdateWindow(self.window.0);
|
winuser::InvalidateRgn(self.window.0, ptr::null_mut(), 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -506,7 +506,7 @@ impl Window {
|
||||||
size.1 as i32,
|
size.1 as i32,
|
||||||
winuser::SWP_ASYNCWINDOWPOS | winuser::SWP_NOZORDER,
|
winuser::SWP_ASYNCWINDOWPOS | winuser::SWP_NOZORDER,
|
||||||
);
|
);
|
||||||
winuser::UpdateWindow(window.0);
|
winuser::InvalidateRgn(window.0, ptr::null_mut(), 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
|
@ -532,7 +532,7 @@ impl Window {
|
||||||
| winuser::SWP_NOZORDER
|
| winuser::SWP_NOZORDER
|
||||||
| winuser::SWP_NOACTIVATE,
|
| winuser::SWP_NOACTIVATE,
|
||||||
);
|
);
|
||||||
winuser::UpdateWindow(window.0);
|
winuser::InvalidateRgn(window.0, ptr::null_mut(), 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,9 +31,6 @@ pub struct WindowState {
|
||||||
|
|
||||||
pub modifiers_state: ModifiersState,
|
pub modifiers_state: ModifiersState,
|
||||||
pub fullscreen: Option<Fullscreen>,
|
pub fullscreen: Option<Fullscreen>,
|
||||||
/// Used to supress duplicate redraw attempts when calling `request_redraw` multiple
|
|
||||||
/// times in `MainEventsCleared`.
|
|
||||||
pub queued_out_of_band_redraw: bool,
|
|
||||||
pub is_dark_mode: bool,
|
pub is_dark_mode: bool,
|
||||||
pub high_surrogate: Option<u16>,
|
pub high_surrogate: Option<u16>,
|
||||||
window_flags: WindowFlags,
|
window_flags: WindowFlags,
|
||||||
|
@ -123,7 +120,6 @@ impl WindowState {
|
||||||
|
|
||||||
modifiers_state: ModifiersState::default(),
|
modifiers_state: ModifiersState::default(),
|
||||||
fullscreen: None,
|
fullscreen: None,
|
||||||
queued_out_of_band_redraw: false,
|
|
||||||
is_dark_mode,
|
is_dark_mode,
|
||||||
high_surrogate: None,
|
high_surrogate: None,
|
||||||
window_flags: WindowFlags::empty(),
|
window_flags: WindowFlags::empty(),
|
||||||
|
@ -273,7 +269,7 @@ impl WindowFlags {
|
||||||
| winuser::SWP_NOSIZE
|
| winuser::SWP_NOSIZE
|
||||||
| winuser::SWP_NOACTIVATE,
|
| winuser::SWP_NOACTIVATE,
|
||||||
);
|
);
|
||||||
winuser::UpdateWindow(window);
|
winuser::InvalidateRgn(window, ptr::null_mut(), 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue