cosmic-applets/cosmic-panel-button/src/lib.rs

247 lines
8.2 KiB
Rust
Raw Normal View History

// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: GPL-3.0-only
use config::{CosmicPanelButtonConfig, IndividualConfig, Override};
2025-08-12 21:38:51 +02:00
use cosmic::desktop::fde::{self, DesktopEntry, get_languages_from_env};
2026-03-31 16:34:59 -04:00
use cosmic::widget::space;
use cosmic::{
2025-08-12 21:38:51 +02:00
Task, app,
applet::{
Size,
2025-08-12 21:38:51 +02:00
cosmic_panel_config::{PanelAnchor, PanelSize},
},
iced::widget::row,
2024-10-30 22:51:08 -04:00
iced::{self, Length},
2025-03-14 13:14:51 -04:00
surface,
2026-03-31 16:34:59 -04:00
widget::{Id, autosize},
};
use cosmic_config::{Config, CosmicConfigEntry};
use std::{env, fs, process::Command, sync::LazyLock};
mod config;
static AUTOSIZE_MAIN_ID: LazyLock<Id> = LazyLock::new(|| Id::new("autosize-main"));
2024-10-30 22:51:08 -04:00
#[derive(Debug, Clone, Default)]
struct Desktop {
name: String,
icon: Option<String>,
exec: String,
}
struct Button {
core: cosmic::app::Core,
desktop: Desktop,
config: IndividualConfig,
}
#[derive(Debug, Clone)]
enum Msg {
Press,
ConfigUpdated(CosmicPanelButtonConfig),
2025-03-14 13:14:51 -04:00
Surface(surface::Action),
}
impl Button {
pub fn icon_button_from_handle<'a, Message: Clone + 'static>(
&self,
icon: cosmic::widget::icon::Handle,
) -> cosmic::widget::Button<'a, Message> {
let theme = cosmic::theme::active();
2026-05-23 20:49:24 +02:00
let _theme = theme.cosmic();
let suggested = self.core.applet.suggested_size(icon.symbolic);
let (major_padding, applet_padding_minor_axis) =
self.core.applet.suggested_padding(icon.symbolic);
let (horizontal_padding, vertical_padding) = if self.core.applet.is_horizontal() {
(major_padding, applet_padding_minor_axis)
} else {
(applet_padding_minor_axis, major_padding)
};
let symbolic = icon.symbolic;
cosmic::widget::button::custom(
cosmic::widget::layer_container(
cosmic::widget::icon(icon)
.class(if symbolic {
cosmic::theme::Svg::Custom(std::rc::Rc::new(|theme| {
cosmic::iced::widget::svg::Style {
color: Some(theme.cosmic().background.on.into()),
}
}))
} else {
cosmic::theme::Svg::default()
})
.width(Length::Fixed(suggested.0 as f32))
.height(Length::Fixed(suggested.1 as f32)),
)
.center(Length::Fill),
)
.width(Length::Fixed((suggested.0 + 2 * horizontal_padding) as f32))
.height(Length::Fixed((suggested.1 + 2 * vertical_padding) as f32))
.class(cosmic::theme::Button::AppletIcon)
}
}
impl cosmic::Application for Button {
type Message = Msg;
type Executor = cosmic::SingleThreadExecutor;
type Flags = Desktop;
const APP_ID: &'static str = "com.system76.CosmicPanelButton";
2024-10-30 22:51:08 -04:00
fn init(core: cosmic::app::Core, desktop: Desktop) -> (Self, app::Task<Msg>) {
let config = Config::new(Self::APP_ID, CosmicPanelButtonConfig::VERSION)
.ok()
.and_then(|c| CosmicPanelButtonConfig::get_entry(&c).ok())
.unwrap_or_default()
.configs
.get(&core.applet.panel_type.to_string())
.cloned()
.unwrap_or_default();
(
Self {
core,
desktop,
config,
},
2024-10-30 22:51:08 -04:00
Task::none(),
)
}
fn core(&self) -> &cosmic::app::Core {
&self.core
}
fn core_mut(&mut self) -> &mut cosmic::app::Core {
&mut self.core
}
2026-03-31 16:34:59 -04:00
fn style(&self) -> Option<iced::theme::Style> {
Some(cosmic::applet::style())
}
2024-10-30 22:51:08 -04:00
fn update(&mut self, message: Msg) -> app::Task<Msg> {
match message {
Msg::Press => {
if let Ok(mut child) = Command::new("sh")
2024-10-30 22:51:08 -04:00
.arg("-c")
.arg(format!("exec {}", self.desktop.exec))
2024-10-30 22:51:08 -04:00
.spawn()
{
std::thread::spawn(move || {
let _ = child.wait();
});
}
}
Msg::ConfigUpdated(conf) => {
self.config = conf
.configs
.get(&self.core.applet.panel_type.to_string())
.cloned()
.unwrap_or_default();
}
2025-03-14 13:14:51 -04:00
Msg::Surface(a) => {
return cosmic::task::message(cosmic::Action::Cosmic(
cosmic::app::Action::Surface(a),
));
}
}
2024-10-30 22:51:08 -04:00
Task::none()
}
fn view(&self) -> cosmic::Element<'_, Msg> {
// currently, panel being anchored to the left or right is a hard
// override for icon, later if text is updated to wrap, we may
// use Override::Text to override this behavior
2024-10-30 22:51:08 -04:00
autosize::autosize(
if self.desktop.icon.is_some()
&& matches!(
self.core.applet.anchor,
PanelAnchor::Left | PanelAnchor::Right
)
2024-10-30 22:51:08 -04:00
|| matches!(self.config.force_presentation, Some(Override::Icon))
|| matches!(
(&self.core.applet.size, &self.config.force_presentation),
(
Size::PanelSize(PanelSize::S | PanelSize::M | PanelSize::L | PanelSize::XL),
2024-10-30 22:51:08 -04:00
None
)
)
{
cosmic::Element::from(
2025-03-14 13:14:51 -04:00
self.core.applet.applet_tooltip::<Msg>(
self.icon_button_from_handle(
cosmic::widget::icon::from_name(self.desktop.icon.clone().unwrap())
.handle(),
)
.on_press_down(Msg::Press),
self.desktop.name.clone(),
false,
2025-03-14 13:14:51 -04:00
Msg::Surface,
None,
),
2024-10-30 22:51:08 -04:00
)
} else {
let content = row!(
self.core.applet.text(&self.desktop.name),
2026-03-31 16:34:59 -04:00
space::vertical().height(Length::Fixed(
2024-10-30 22:51:08 -04:00
(self.core.applet.suggested_size(true).1
+ 2 * self.core.applet.suggested_padding(true).1)
2024-10-30 22:51:08 -04:00
as f32
))
)
.align_y(iced::Alignment::Center);
cosmic::widget::button::custom(content)
.padding([0, self.core.applet.suggested_padding(true).0])
2024-10-30 22:51:08 -04:00
.class(cosmic::theme::Button::AppletIcon)
.on_press_down(Msg::Press)
.into()
},
2024-10-30 22:51:08 -04:00
AUTOSIZE_MAIN_ID.clone(),
)
.into()
}
fn subscription(&self) -> iced::Subscription<Self::Message> {
self.core.watch_config(Self::APP_ID).map(|u| {
for why in u.errors {
tracing::error!(why = why.to_string(), "Error watching config");
}
Msg::ConfigUpdated(u.config)
})
}
}
pub fn run() -> iced::Result {
let id = env::args()
.nth(1)
.expect("Requires desktop file id as argument.");
let filename = format!("{id}.desktop");
let mut desktop = None;
let locales = get_languages_from_env();
for mut path in fde::default_paths() {
path.push(&filename);
if let Ok(bytes) = fs::read_to_string(&path) {
if let Ok(entry) = DesktopEntry::from_str(&path, &bytes, Some(&locales)) {
desktop = Some(Desktop {
name: entry.name(&locales).map_or_else(
|| panic!("Desktop file '{filename}' doesn't have `Name`"),
|x| x.to_string(),
),
icon: entry.icon().map(|x| x.to_string()),
exec: entry.exec().map_or_else(
|| panic!("Desktop file '{filename}' doesn't have `Exec`"),
|x| x.to_string(),
),
});
break;
}
}
}
let desktop = desktop.unwrap_or_else(|| {
panic!("Failed to find valid desktop file '{filename}' in search paths")
});
2024-10-30 22:51:08 -04:00
cosmic::applet::run::<Button>(desktop)
}