cosmic-applets/cosmic-applet-status-area/src/components/status_menu.rs
Ian Douglas Scott 468017654c status-area: Don't try to call activate() in status_menu
This was added in https://github.com/pop-os/cosmic-applets/pull/1143,
but I don't think this code will ever be reached? This code is called
when activating an item in a menu, so it should only apply when there is
a menu.
2026-02-02 10:34:25 -08:00

272 lines
9.4 KiB
Rust

// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: GPL-3.0-only
use cosmic::{
Application,
applet::{menu_button, token::subscription::TokenRequest},
cctk::sctk::reexports::calloop,
iced,
widget::icon::{self, IconFallback},
};
use std::path::{Path, PathBuf};
use crate::subscriptions::status_notifier_item::{IconUpdate, Layout, StatusNotifierItem};
#[derive(Clone, Debug)]
pub enum Msg {
Layout(Result<Layout, String>),
Icon(IconUpdate),
Click(i32, bool),
ClickToken(String),
}
pub struct State {
pub item: StatusNotifierItem,
layout: Option<Layout>,
expanded: Option<i32>,
// TODO handle icon with multiple sizes?
icon_handle: icon::Handle,
click_event: Option<(i32, bool)>,
}
impl State {
pub fn new(item: StatusNotifierItem) -> (Self, iced::Task<Msg>) {
(
Self {
item,
layout: None,
expanded: None,
icon_handle: icon::from_name("application-default")
.prefer_svg(true)
.handle(),
click_event: None,
},
iced::Task::none(),
)
}
pub fn update(
&mut self,
message: Msg,
menu_id: usize,
token_tx: Option<&calloop::channel::Sender<TokenRequest>>,
) -> iced::Task<Msg> {
match message {
Msg::Layout(layout) => {
match layout {
Ok(layout) => {
self.layout = Some(layout);
}
Err(err) => eprintln!("Error getting layout from icon: {err}"),
}
iced::Task::none()
}
Msg::Icon(update) => {
let icon_name = update.name.unwrap_or_default();
// Use the icon pixmap if an icon was not defined by name.
if icon_name.is_empty() {
let icon_pixmap = update.pixmap.and_then(|icons| icons
.into_iter()
.max_by_key(|i| (i.width, i.height))
.map(|mut i| {
if i.width <= 0 || i.height <= 0 || i.bytes.is_empty() {
// App sent invalid icon data during initialization - show placeholder until NewIcon signal
tracing::debug!("Skipping invalid icon: {}x{} with {} bytes, app may still be initializing",
i.width, i.height, i.bytes.len());
return icon::from_name("dialog-question").symbolic(true).handle();
}
// Convert ARGB to RGBA
for pixel in i.bytes.chunks_exact_mut(4) {
pixel.rotate_left(1);
}
icon::from_raster_pixels(i.width as u32, i.height as u32, i.bytes)
}));
if let Some(icon) = icon_pixmap {
self.icon_handle = icon.clone();
return iced::Task::none();
}
}
// Load icon by path if the name is a path.
self.icon_handle = if Path::new(&icon_name).exists() {
icon::from_path(Path::new(&icon_name).to_path_buf()).symbolic(true)
} else {
icon::from_name(icon_name)
.prefer_svg(true)
.fallback(Some(IconFallback::Names(vec![
"application-default".into(),
"application-x-executable".into(),
])))
.handle()
};
iced::Task::none()
}
Msg::Click(id, is_submenu) => {
if let Some(token_tx) = token_tx {
_ = token_tx.send(TokenRequest {
app_id: super::app::App::APP_ID.to_string(),
exec: menu_id.to_string(),
});
}
self.click_event = Some((id, is_submenu));
iced::Task::none()
}
Msg::ClickToken(token) => {
let Some((id, is_submenu)) = self.click_event else {
return iced::Task::none();
};
let item_proxy = self.item.item_proxy().clone();
let Some(menu_proxy) = self.item.menu_proxy().cloned() else {
return iced::Task::none();
};
tokio::spawn(async move {
let _ = item_proxy.provide_xdg_activation_token(token).await;
let _ = menu_proxy.event(id, "clicked", &0.into(), 0).await;
});
if is_submenu {
self.expanded = if self.expanded.is_none() {
Some(id)
} else {
None
};
} else {
// TODO: Close menu?
}
iced::Task::none()
}
}
}
pub fn name(&self) -> &str {
self.item.name()
}
pub fn icon_handle(&self) -> &icon::Handle {
&self.icon_handle
}
pub fn popup_view(&self) -> cosmic::Element<'_, Msg> {
if let Some(layout) = self.layout.as_ref() {
layout_view(layout, self.expanded)
} else {
iced::widget::text("").into()
}
}
pub fn subscription(&self) -> iced::Subscription<Msg> {
iced::Subscription::batch([
self.item.layout_subscription().map(Msg::Layout),
self.item.icon_subscription().map(Msg::Icon),
])
}
pub fn opened(&self) {
let Some(menu_proxy) = self.item.menu_proxy().cloned() else {
return;
};
tokio::spawn(async move {
let _ = menu_proxy.event(0, "opened", &0i32.into(), 0).await;
let _ = menu_proxy.about_to_show(0).await;
});
}
pub fn closed(&self) {
let Some(menu_proxy) = self.item.menu_proxy().cloned() else {
return;
};
tokio::spawn(async move {
let _ = menu_proxy.event(0, "closed", &0i32.into(), 0).await;
});
}
}
fn layout_view(layout: &Layout, expanded: Option<i32>) -> cosmic::Element<'_, Msg> {
iced::widget::column(layout.children().iter().filter_map(|i| {
if !i.visible() {
None
} else if i.type_() == Some("separator") {
Some(iced::widget::horizontal_rule(2).into())
} else if let Some(label) = i.label() {
// Strip _ when not doubled
// TODO: interpret as "access key"? And label with underline.
let mut is_underscore = false;
let label = label
.chars()
.filter(|c| {
let prev_is_underscore = is_underscore;
is_underscore = !is_underscore && *c == '_';
*c != '_' || prev_is_underscore
})
.collect::<String>();
let is_submenu = i.children_display() == Some("submenu");
let is_expanded = expanded == Some(i.id());
let text = iced::widget::text(label).width(iced::Length::Fill);
let mut children: Vec<cosmic::Element<_>> = vec![text.into()];
if is_submenu {
let icon = cosmic::widget::icon::from_name(if is_expanded {
"go-down-symbolic"
} else {
"go-next-symbolic"
})
.size(14)
.symbolic(true);
children.push(icon.into());
}
if let Some(icon_data) = i.icon_data() {
let handle = iced::widget::image::Handle::from_bytes(icon_data.to_vec());
children.insert(0, iced::widget::Image::new(handle).into());
} else if let Some(icon_name) = i.icon_name() {
let icon = cosmic::widget::icon::from_name(icon_name)
.size(14)
.symbolic(true);
children.insert(0, icon.into());
}
if i.toggle_state() == Some(1) {
let icon = cosmic::widget::icon::from_name("emblem-ok-symbolic")
.size(14)
.symbolic(true);
children.push(icon.into());
}
let mut button = row_button(children);
if i.enabled() {
button = button.on_press(Msg::Click(i.id(), is_submenu));
}
if is_submenu && is_expanded {
Some(
iced::widget::column![
button,
// XXX nested
iced::widget::container(layout_view(i, None)).padding(iced::Padding {
left: 12.,
..iced::Padding::ZERO
})
]
.into(),
)
} else {
Some(button.into())
}
} else {
None
}
}))
.into()
}
fn row_button(content: Vec<cosmic::Element<Msg>>) -> cosmic::widget::Button<Msg> {
menu_button(
iced::widget::Row::with_children(content)
.spacing(8)
.align_y(iced::Alignment::Center)
.width(iced::Length::Fill),
)
}