use std::sync::Arc;
use dioxus_core::prelude::{
consume_context,
try_consume_context,
use_hook,
};
use dioxus_signals::{
Readable,
Signal,
};
use freya_core::{
prelude::EventMessage,
types::AccessibilityId,
};
use tokio::sync::{
broadcast,
mpsc::UnboundedSender,
};
use torin::prelude::Area;
use winit::{
event_loop::EventLoopProxy,
window::{
CursorIcon,
Fullscreen,
Window,
},
};
#[derive(Clone, Copy, PartialEq)]
pub struct UsePlatform {
ticker: Signal<Arc<broadcast::Receiver<()>>>,
event_loop_proxy: Signal<Option<EventLoopProxy<EventMessage>>>,
platform_emitter: Signal<Option<UnboundedSender<EventMessage>>>,
}
#[derive(PartialEq, Eq, Debug)]
pub enum UsePlatformError {
EventLoopProxyFailed,
PlatformEmitterFailed,
}
impl UsePlatform {
#[allow(clippy::new_without_default)]
pub fn new() -> Self {
UsePlatform {
event_loop_proxy: Signal::new(try_consume_context::<EventLoopProxy<EventMessage>>()),
platform_emitter: Signal::new(try_consume_context::<UnboundedSender<EventMessage>>()),
ticker: Signal::new(consume_context::<Arc<broadcast::Receiver<()>>>()),
}
}
pub fn send(&self, event: EventMessage) -> Result<(), UsePlatformError> {
if let Some(event_loop_proxy) = &*self.event_loop_proxy.peek() {
event_loop_proxy
.send_event(event)
.map_err(|_| UsePlatformError::EventLoopProxyFailed)?;
} else if let Some(platform_emitter) = &*self.platform_emitter.peek() {
platform_emitter
.send(event)
.map_err(|_| UsePlatformError::PlatformEmitterFailed)?;
}
Ok(())
}
pub fn set_cursor(&self, cursor_icon: CursorIcon) {
self.send(EventMessage::SetCursorIcon(cursor_icon)).ok();
}
pub fn set_title(&self, title: impl Into<String>) {
let title = title.into();
self.with_window(move |window| {
window.set_title(&title);
});
}
pub fn with_window(&self, cb: impl FnOnce(&Window) + 'static + Send + Sync) {
self.send(EventMessage::WithWindow(Box::new(cb))).ok();
}
pub fn drag_window(&self) {
self.with_window(|window| {
window.drag_window().ok();
});
}
pub fn set_maximize_window(&self, maximize: bool) {
self.with_window(move |window| {
window.set_maximized(maximize);
});
}
pub fn toggle_maximize_window(&self) {
self.with_window(|window| {
window.set_maximized(!window.is_maximized());
});
}
pub fn set_minimize_window(&self, minimize: bool) {
self.with_window(move |window| {
window.set_minimized(minimize);
});
}
pub fn toggle_minimize_window(&self) {
self.with_window(|window| {
window.set_minimized(window.is_minimized().map(|v| !v).unwrap_or_default());
});
}
pub fn toggle_fullscreen_window(&self) {
self.with_window(|window| match window.fullscreen() {
Some(_) => window.set_fullscreen(None),
None => window.set_fullscreen(Some(Fullscreen::Borderless(None))),
});
}
pub fn set_fullscreen_window(&self, fullscreen: bool) {
self.with_window(move |window| {
if fullscreen {
window.set_fullscreen(Some(Fullscreen::Borderless(None)))
} else {
window.set_fullscreen(None)
}
});
}
pub fn invalidate_drawing_area(&self, area: Area) {
self.send(EventMessage::InvalidateArea(area)).ok();
}
pub fn request_animation_frame(&self) {
self.send(EventMessage::RequestRerender).ok();
}
pub fn focus(&self, id: AccessibilityId) {
self.send(EventMessage::FocusAccessibilityNode(id)).ok();
}
pub fn new_ticker(&self) -> Ticker {
Ticker {
inner: self.ticker.peek().resubscribe(),
}
}
pub fn exit(&self) {
self.send(EventMessage::ExitApp).ok();
}
}
pub fn use_platform() -> UsePlatform {
use_hook(UsePlatform::new)
}
pub struct Ticker {
inner: broadcast::Receiver<()>,
}
impl Ticker {
pub async fn tick(&mut self) {
self.inner.recv().await.ok();
}
}