refactor: make the single-instance feature additive

This commit is contained in:
Ashley Wulber 2023-11-21 15:18:39 -05:00 committed by Michael Murphy
parent 001fd744c5
commit 57f4abb8a0
5 changed files with 53 additions and 77 deletions

View file

@ -8,7 +8,7 @@ publish = false
[dependencies] [dependencies]
apply = "0.3.0" apply = "0.3.0"
fraction = "0.13.0" fraction = "0.13.0"
libcosmic = { path = "../..", features = ["debug", "winit", "tokio"] } libcosmic = { path = "../..", features = ["debug", "winit", "tokio", "single-instance"] }
once_cell = "1.18" once_cell = "1.18"
slotmap = "1.0.6" slotmap = "1.0.6"
env_logger = "0.10" env_logger = "0.10"

View file

@ -63,7 +63,8 @@ pub struct Core {
#[cfg(feature = "applet")] #[cfg(feature = "applet")]
pub applet: crate::applet::Context, pub applet: crate::applet::Context,
pub single_instance: bool, #[cfg(feature = "single-instance")]
pub(crate) single_instance: bool,
} }
impl Default for Core { impl Default for Core {
@ -106,6 +107,7 @@ impl Default for Core {
}, },
#[cfg(feature = "applet")] #[cfg(feature = "applet")]
applet: crate::applet::Context::default(), applet: crate::applet::Context::default(),
#[cfg(feature = "single-instance")]
single_instance: false, single_instance: false,
} }
} }

View file

@ -97,6 +97,8 @@ where
super::Message::App(message) => self.app.update(message), super::Message::App(message) => self.app.update(message),
super::Message::Cosmic(message) => self.cosmic_update(message), super::Message::Cosmic(message) => self.cosmic_update(message),
super::Message::None => iced::Command::none(), super::Message::None => iced::Command::none(),
#[cfg(feature = "single-instance")]
super::Message::DbusActivation(message) => self.app.dbus_activation(message),
} }
} }
@ -184,7 +186,7 @@ where
.core() .core()
.single_instance .single_instance
.then(|| super::single_instance_subscription::<T>()) .then(|| super::single_instance_subscription::<T>())
.unwrap_or_else(Subscription::none), .unwrap_or_else(|| Subscription::none()),
]) ])
} }
@ -364,11 +366,12 @@ impl<T: Application> Cosmic<T> {
}); });
} }
} }
Message::Activate(token) => { Message::Activate(_token) => {
#[cfg(feature = "wayland")] #[cfg(feature = "wayland")]
return iced_sctk::commands::activation::activate( return iced_sctk::commands::activation::activate(
iced::window::Id::default(), iced::window::Id::default(),
token, #[allow(clippy::used_underscore_binding)]
_token,
); );
} }
} }

View file

@ -19,6 +19,9 @@ pub mod message {
App(M), App(M),
/// Internal messages to be handled by libcosmic. /// Internal messages to be handled by libcosmic.
Cosmic(super::cosmic::Message), Cosmic(super::cosmic::Message),
#[cfg(feature = "single-instance")]
/// Dbus activation messages
DbusActivation(super::DbusActivationMessage),
/// Do nothing /// Do nothing
None, None,
} }
@ -36,8 +39,6 @@ pub mod message {
} }
} }
use std::str::FromStr;
pub use self::command::Command; pub use self::command::Command;
pub use self::core::Core; pub use self::core::Core;
pub use self::settings::Settings; pub use self::settings::Settings;
@ -57,12 +58,11 @@ use {
std::collections::HashMap, std::collections::HashMap,
zbus::{dbus_interface, dbus_proxy, zvariant::Value}, zbus::{dbus_interface, dbus_proxy, zvariant::Value},
}; };
/// Launch a COSMIC application with the given [`Settings`].
/// pub(crate) fn iced_settings<App: Application>(
/// # Errors settings: Settings,
/// flags: App::Flags,
/// Returns error on application failure. ) -> iced::Settings<(Core, App::Flags)> {
pub fn run<App: Application>(settings: Settings, flags: App::Flags) -> iced::Result {
if let Some(icon_theme) = settings.default_icon_theme { if let Some(icon_theme) = settings.default_icon_theme {
crate::icon_theme::set_default(icon_theme); crate::icon_theme::set_default(icon_theme);
} }
@ -72,7 +72,6 @@ pub fn run<App: Application>(settings: Settings, flags: App::Flags) -> iced::Res
core.set_scale_factor(settings.scale_factor); core.set_scale_factor(settings.scale_factor);
core.set_window_width(settings.size.0); core.set_window_width(settings.size.0);
core.set_window_height(settings.size.1); core.set_window_height(settings.size.1);
core.single_instance = settings.single_instance;
THEME.with(move |t| { THEME.with(move |t| {
let mut cosmic_theme = t.borrow_mut(); let mut cosmic_theme = t.borrow_mut();
@ -120,8 +119,20 @@ pub fn run<App: Application>(settings: Settings, flags: App::Flags) -> iced::Res
iced.window.transparent = settings.transparent; iced.window.transparent = settings.transparent;
} }
cosmic::Cosmic::<App>::run(iced) iced
} }
/// Launch a COSMIC application with the given [`Settings`].
///
/// # Errors
///
/// Returns error on application failure.
pub fn run<App: Application>(settings: Settings, flags: App::Flags) -> iced::Result {
let settings = iced_settings::<App>(settings, flags);
cosmic::Cosmic::<App>::run(settings)
}
#[cfg(feature = "single-instance")] #[cfg(feature = "single-instance")]
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct DbusActivationMessage<Action = String, Args = Vec<String>> { pub struct DbusActivationMessage<Action = String, Args = Vec<String>> {
@ -259,16 +270,16 @@ impl DbusActivation {
} }
#[cfg(feature = "single-instance")] #[cfg(feature = "single-instance")]
/// Launch a COSMIC application with the given [`Settings`]. /// Launch a COSMIC application with the given [`Settings`].
/// If the application is already running, the arguments will be passed to the /// If the application is already running, the arguments will be passed to the
/// running instance. /// running instance.
/// # Errors /// # Errors
/// Returns error on application failure. /// Returns error on application failure.
pub fn run_single_instance<App: Application>( pub fn run_single_instance<App: Application>(settings: Settings, flags: App::Flags) -> iced::Result
mut settings: Settings, where
flags: App::Flags, App::Flags: CosmicFlags + Clone,
) -> iced::Result { App::Message: Clone + std::fmt::Debug + Send + 'static,
{
let activation_token = std::env::var("XDG_ACTIVATION_TOKEN").ok(); let activation_token = std::env::var("XDG_ACTIVATION_TOKEN").ok();
let override_single = std::env::var("COSMIC_SINGLE_INSTANCE") let override_single = std::env::var("COSMIC_SINGLE_INSTANCE")
@ -279,7 +290,6 @@ pub fn run_single_instance<App: Application>(
} }
let path: String = format!("/{}", App::APP_ID.replace('.', "/")); let path: String = format!("/{}", App::APP_ID.replace('.', "/"));
settings.single_instance = true;
let Ok(conn) = zbus::blocking::Connection::session() else { let Ok(conn) = zbus::blocking::Connection::session() else {
tracing::warn!("Failed to connect to dbus"); tracing::warn!("Failed to connect to dbus");
@ -322,13 +332,15 @@ pub fn run_single_instance<App: Application>(
tracing::info!("Another instance is running"); tracing::info!("Another instance is running");
Ok(()) Ok(())
} else { } else {
run::<App>(settings, flags) let mut settings = iced_settings::<App>(settings, flags);
settings.flags.0.single_instance = true;
cosmic::Cosmic::<App>::run(settings)
} }
} }
pub trait CosmicFlags { pub trait CosmicFlags {
type SubCommand: FromStr + ToString + std::fmt::Debug + Clone + Send + 'static; type SubCommand: ToString + std::fmt::Debug + Clone + Send + 'static;
type Args: TryFrom<Vec<String>> + Into<Vec<String>> + std::fmt::Debug + Clone + Send + 'static; type Args: Into<Vec<String>> + std::fmt::Debug + Clone + Send + 'static;
#[must_use] #[must_use]
fn action(&self) -> Option<&Self::SubCommand> { fn action(&self) -> Option<&Self::SubCommand> {
None None
@ -349,27 +361,9 @@ where
/// Default async executor to use with the app. /// Default async executor to use with the app.
type Executor: iced_futures::Executor; type Executor: iced_futures::Executor;
#[cfg(feature = "single-instance")]
/// Argument received [`Application::new`].
type Flags: Clone + CosmicFlags;
#[cfg(not(feature = "single-instance"))]
/// Argument received [`Application::new`]. /// Argument received [`Application::new`].
type Flags: Clone; type Flags: Clone;
#[cfg(feature = "single-instance")]
/// Message type specific to our app.
type Message: Clone
+ From<
DbusActivationDetails<
<Self::Flags as CosmicFlags>::SubCommand,
<Self::Flags as CosmicFlags>::Args,
>,
> + std::fmt::Debug
+ Send
+ 'static;
#[cfg(not(feature = "single-instance"))]
/// Message type specific to our app. /// Message type specific to our app.
type Message: Clone + std::fmt::Debug + Send + 'static; type Message: Clone + std::fmt::Debug + Send + 'static;
@ -465,6 +459,15 @@ where
fn style(&self) -> Option<<crate::Theme as iced_style::application::StyleSheet>::Style> { fn style(&self) -> Option<<crate::Theme as iced_style::application::StyleSheet>::Style> {
None None
} }
/// Handles dbus activation messages
#[cfg(feature = "single-instance")]
fn dbus_activation(
&mut self,
msg: DbusActivationMessage,
) -> iced::Command<Message<Self::Message>> {
iced::Command::none()
}
} }
/// Methods automatically derived for all types implementing [`Application`]. /// Methods automatically derived for all types implementing [`Application`].
@ -633,7 +636,7 @@ fn single_instance_subscription<App: ApplicationExt>() -> Subscription<Message<A
iced::subscription::channel( iced::subscription::channel(
TypeId::of::<DbusActivation>(), TypeId::of::<DbusActivation>(),
10, 10,
|mut output| async move { move |mut output| async move {
let mut single_instance: DbusActivation = DbusActivation::new(); let mut single_instance: DbusActivation = DbusActivation::new();
let mut rx = single_instance.rx(); let mut rx = single_instance.rx();
if let Ok(builder) = zbus::ConnectionBuilder::session() { if let Ok(builder) = zbus::ConnectionBuilder::session() {
@ -680,36 +683,8 @@ fn single_instance_subscription<App: ApplicationExt>() -> Subscription<Message<A
tracing::error!(?err, "Failed to send message"); tracing::error!(?err, "Failed to send message");
} }
} }
if let Some(msg) = match msg.msg { if let Err(err) = output.send(Message::DbusActivation(msg)).await {
DbusActivationDetails::Activate => { tracing::error!(?err, "Failed to send message");
Some(DbusActivationDetails::Activate)
}
DbusActivationDetails::Open { url } => {
Some(DbusActivationDetails::Open { url })
}
DbusActivationDetails::ActivateAction { action, args } => {
if let (Ok(action), Ok(args)) = (
<App::Flags as CosmicFlags>::SubCommand::from_str(&action),
<App::Flags as CosmicFlags>::Args::try_from(args),
) {
Some(DbusActivationDetails::ActivateAction::<
<App::Flags as CosmicFlags>::SubCommand,
<App::Flags as CosmicFlags>::Args,
> {
action,
args,
})
} else {
tracing::error!("Invalid action or args");
None
}
}
} {
if let Err(err) =
output.send(Message::App(App::Message::from(msg))).await
{
tracing::error!(?err, "Failed to send message");
}
} }
} }
} }

View file

@ -62,9 +62,6 @@ pub struct Settings {
/// Whether the application should exit when there are no open windows /// Whether the application should exit when there are no open windows
pub(crate) exit_on_close: bool, pub(crate) exit_on_close: bool,
/// Only allow a single instance of the application to run
pub single_instance: bool,
} }
impl Settings { impl Settings {
@ -100,7 +97,6 @@ impl Default for Settings {
theme: crate::theme::system_preference(), theme: crate::theme::system_preference(),
transparent: false, transparent: false,
exit_on_close: true, exit_on_close: true,
single_instance: false,
} }
} }
} }