1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
pub mod executor;

use std::{pin::Pin, task::Poll};

use async_component::AsyncComponent;
use executor::WinitExecutor;
use winit::{
    event::Event,
    event_loop::{ControlFlow, EventLoop},
};

pub trait WinitComponent {
    fn on_event(&mut self, event: Event<()>, control_flow: &mut ControlFlow);
}

#[derive(Debug, Clone, Copy)]
#[non_exhaustive]
pub struct ExecutorPollEvent;

pub fn run(
    event_loop: EventLoop<ExecutorPollEvent>,
    component: impl AsyncComponent + WinitComponent + 'static,
) -> ! {
    let mut component = component;

    let executor = WinitExecutor::new(event_loop.create_proxy());

    let mut should_repoll = false;

    event_loop.run(move |event, _, control_flow| {
        control_flow.set_wait();

        match event {
            Event::RedrawEventsCleared => {
                component.on_event(Event::RedrawEventsCleared, control_flow);

                if let ControlFlow::ExitWithCode(_) = control_flow {
                    return;
                }

                if should_repoll {
                    should_repoll = false;

                    control_flow.set_poll();
                }
            }

            Event::MainEventsCleared => {
                component.on_event(Event::MainEventsCleared, control_flow);

                if let Poll::Ready(_) = executor.poll_component(Pin::new(&mut component)) {
                    should_repoll = true;
                }
            }

            // Event::RedrawEventsCleared
            Event::UserEvent(_) => {
                control_flow.set_poll();
            }

            _ => {
                component.on_event(event.map_nonuser_event().unwrap(), control_flow);
            }
        }
    });
}