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:
Philippe Renon 2020-03-07 20:04:24 +01:00 committed by GitHub
parent cbb60d29a2
commit 2f27f64cdb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 158 additions and 277 deletions

View file

@ -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.

View file

@ -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();
}
} }
} }
} }

View file

@ -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)
}
} }
} }

View file

@ -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);
} }
} }

View file

@ -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);
} }
} }
} }

View file

@ -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);
} }
} }