use crossbeam::channel;
use nih_plug::prelude::GuiContext;
use std::sync::Arc;
use crate::futures::FutureExt;
use crate::{
futures, subscription, Application, Color, Command, Element, IcedEditor, ParameterUpdate,
Subscription, WindowQueue, WindowScalePolicy, WindowSubs,
};
pub(crate) struct IcedEditorWrapperApplication<E: IcedEditor> {
editor: E,
parameter_updates_receiver: Arc<channel::Receiver<ParameterUpdate>>,
}
pub enum Message<E: IcedEditor> {
EditorMessage(E::Message),
ParameterUpdate,
}
impl<E: IcedEditor> std::fmt::Debug for Message<E> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Self::EditorMessage(arg0) => f.debug_tuple("EditorMessage").field(arg0).finish(),
Self::ParameterUpdate => write!(f, "ParameterUpdate"),
}
}
}
impl<E: IcedEditor> Clone for Message<E> {
fn clone(&self) -> Self {
match self {
Self::EditorMessage(arg0) => Self::EditorMessage(arg0.clone()),
Self::ParameterUpdate => Self::ParameterUpdate,
}
}
}
impl<E: IcedEditor> Application for IcedEditorWrapperApplication<E> {
type Executor = E::Executor;
type Message = Message<E>;
type Flags = (
Arc<dyn GuiContext>,
Arc<channel::Receiver<ParameterUpdate>>,
E::InitializationFlags,
);
fn new(
(context, parameter_updates_receiver, flags): Self::Flags,
) -> (Self, Command<Self::Message>) {
let (editor, command) = E::new(flags, context);
(
Self {
editor,
parameter_updates_receiver,
},
command.map(Message::EditorMessage),
)
}
#[inline]
fn update(
&mut self,
window: &mut WindowQueue,
message: Self::Message,
) -> Command<Self::Message> {
match message {
Message::EditorMessage(message) => self
.editor
.update(window, message)
.map(Message::EditorMessage),
Message::ParameterUpdate => Command::none(),
}
}
#[inline]
fn subscription(
&self,
window_subs: &mut WindowSubs<Self::Message>,
) -> Subscription<Self::Message> {
let mut editor_window_subs = WindowSubs {
on_frame: match &window_subs.on_frame {
Some(Message::EditorMessage(message)) => Some(message.clone()),
_ => None,
},
on_window_will_close: match &window_subs.on_window_will_close {
Some(Message::EditorMessage(message)) => Some(message.clone()),
_ => None,
},
};
let subscription = Subscription::batch([
subscription::unfold(
"parameter updates",
self.parameter_updates_receiver.clone(),
|parameter_updates_receiver| match parameter_updates_receiver.try_recv() {
Ok(_) => futures::future::ready((
Some(Message::ParameterUpdate),
parameter_updates_receiver,
))
.boxed(),
Err(_) => futures::future::pending().boxed(),
},
),
self.editor
.subscription(&mut editor_window_subs)
.map(Message::EditorMessage),
]);
if let Some(message) = editor_window_subs.on_frame {
window_subs.on_frame = Some(Message::EditorMessage(message));
}
if let Some(message) = editor_window_subs.on_window_will_close {
window_subs.on_window_will_close = Some(Message::EditorMessage(message));
}
subscription
}
#[inline]
fn view(&mut self) -> Element<'_, Self::Message> {
self.editor.view().map(Message::EditorMessage)
}
#[inline]
fn background_color(&self) -> Color {
self.editor.background_color()
}
#[inline]
fn scale_policy(&self) -> WindowScalePolicy {
self.editor.scale_policy()
}
#[inline]
fn renderer_settings() -> iced_baseview::backend::settings::Settings {
E::renderer_settings()
}
}