winit/src/platform_impl/web/event_loop/mod.rs

111 lines
3.7 KiB
Rust
Raw Normal View History

use std::marker::PhantomData;
use std::sync::mpsc::{self, Receiver, Sender};
use crate::error::EventLoopError;
use crate::event::Event;
use crate::event_loop::EventLoopWindowTarget as RootEventLoopWindowTarget;
use super::{backend, device, window};
2019-06-25 03:15:34 +02:00
mod proxy;
pub(crate) mod runner;
2019-06-25 03:15:34 +02:00
mod state;
mod window_target;
pub use proxy::EventLoopProxy;
pub use window_target::EventLoopWindowTarget;
2019-06-25 03:15:34 +02:00
pub struct EventLoop<T: 'static> {
elw: RootEventLoopWindowTarget<T>,
user_event_sender: Sender<T>,
user_event_receiver: Receiver<T>,
2019-06-25 03:15:34 +02:00
}
2022-06-10 13:43:33 +03:00
#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub(crate) struct PlatformSpecificEventLoopAttributes {}
2019-06-25 03:15:34 +02:00
impl<T> EventLoop<T> {
pub(crate) fn new(_: &PlatformSpecificEventLoopAttributes) -> Result<Self, EventLoopError> {
let (user_event_sender, user_event_receiver) = mpsc::channel();
Ok(EventLoop {
elw: RootEventLoopWindowTarget {
p: EventLoopWindowTarget::new(),
2019-06-25 03:15:34 +02:00
_marker: PhantomData,
},
user_event_sender,
user_event_receiver,
})
2019-06-25 03:15:34 +02:00
}
pub fn run<F>(self, mut event_handler: F) -> !
where
F: FnMut(Event<T>, &RootEventLoopWindowTarget<T>),
{
let target = RootEventLoopWindowTarget {
p: self.elw.p.clone(),
_marker: PhantomData,
};
// SAFETY: Don't use `move` to make sure we leak the `event_handler` and `target`.
let handler: Box<dyn FnMut(Event<()>)> = Box::new(|event| {
let event = match event.map_nonuser_event() {
Ok(event) => event,
Err(Event::UserEvent(())) => Event::UserEvent(
self.user_event_receiver
.try_recv()
.expect("handler woken up without user event"),
),
Err(_) => unreachable!(),
};
event_handler(event, &target)
});
// SAFETY: The `transmute` is necessary because `run()` requires `'static`. This is safe
// because this function will never return and all resources not cleaned up by the point we
// `throw` will leak, making this actually `'static`.
let handler = unsafe { std::mem::transmute(handler) };
self.elw.p.run(handler, false);
// Throw an exception to break out of Rust execution and use unreachable to tell the
// compiler this function won't return, giving it a return type of '!'
backend::throw(
"Using exceptions for control flow, don't mind me. This isn't actually an error!",
);
unreachable!();
}
pub fn spawn<F>(self, mut event_handler: F)
2019-06-25 03:15:34 +02:00
where
F: 'static + FnMut(Event<T>, &RootEventLoopWindowTarget<T>),
2019-06-25 03:15:34 +02:00
{
let target = RootEventLoopWindowTarget {
2019-06-25 03:15:34 +02:00
p: self.elw.p.clone(),
_marker: PhantomData,
};
self.elw.p.run(
Box::new(move |event| {
let event = match event.map_nonuser_event() {
Ok(event) => event,
Err(Event::UserEvent(())) => Event::UserEvent(
self.user_event_receiver
.try_recv()
.expect("handler woken up without user event"),
),
Err(_) => unreachable!(),
};
event_handler(event, &target)
}),
true,
);
2019-06-25 03:15:34 +02:00
}
pub fn create_proxy(&self) -> EventLoopProxy<T> {
EventLoopProxy::new(self.elw.p.runner.clone(), self.user_event_sender.clone())
2019-06-25 03:15:34 +02:00
}
pub fn window_target(&self) -> &RootEventLoopWindowTarget<T> {
2019-06-25 03:15:34 +02:00
&self.elw
}
}