feat(app-list): close toplevel button

This commit is contained in:
Vukašin Vojinović 2025-12-19 11:56:11 +01:00 committed by Michael Murphy
parent 90a46e915a
commit 7a2bad8f34
3 changed files with 278 additions and 238 deletions

View file

@ -9,7 +9,7 @@ use std::fmt::Debug;
pub const APP_ID: &str = "com.system76.CosmicAppList"; pub const APP_ID: &str = "com.system76.CosmicAppList";
#[derive(Debug, Clone, Deserialize, Serialize, Default, PartialEq, Eq)] #[derive(Debug, Clone, Deserialize, Serialize, Default, PartialEq, Eq)]
pub enum TopLevelFilter { pub enum ToplevelFilter {
#[default] #[default]
ActiveWorkspace, ActiveWorkspace,
ConfiguredOutput, ConfiguredOutput,
@ -18,7 +18,7 @@ pub enum TopLevelFilter {
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq, CosmicConfigEntry)] #[derive(Debug, Clone, Deserialize, Serialize, PartialEq, Eq, CosmicConfigEntry)]
#[version = 1] #[version = 1]
pub struct AppListConfig { pub struct AppListConfig {
pub filter_top_levels: Option<TopLevelFilter>, pub filter_top_levels: Option<ToplevelFilter>,
pub favorites: Vec<String>, pub favorites: Vec<String>,
pub enable_drag_source: bool, pub enable_drag_source: bool,
} }

View file

@ -19,8 +19,7 @@ use cctk::{
workspace::v1::client::ext_workspace_handle_v1::ExtWorkspaceHandleV1, workspace::v1::client::ext_workspace_handle_v1::ExtWorkspaceHandleV1,
}, },
}; };
use cosmic::desktop::fde::unicase::Ascii; use cosmic::desktop::fde::{self, DesktopEntry, get_languages_from_env, unicase::Ascii};
use cosmic::desktop::fde::{self, DesktopEntry, get_languages_from_env};
use cosmic::{ use cosmic::{
Apply, Element, Task, app, Apply, Element, Task, app,
applet::{ applet::{
@ -30,14 +29,16 @@ use cosmic::{
cosmic_config::{Config, CosmicConfigEntry}, cosmic_config::{Config, CosmicConfigEntry},
desktop::IconSourceExt, desktop::IconSourceExt,
iced::{ iced::{
self, Limits, Subscription, self, Alignment, Background, Border, Length, Limits, Padding, Subscription,
clipboard::mime::{AllowedMimeTypes, AsMimeTypes}, clipboard::mime::{AllowedMimeTypes, AsMimeTypes},
event::listen_with, event::listen_with,
platform_specific::shell::commands::popup::{destroy_popup, get_popup}, platform_specific::shell::commands::popup::{destroy_popup, get_popup},
widget::{Column, Row, column, mouse_area, row, vertical_rule, vertical_space}, widget::{
Column, Row, column, mouse_area, row, stack, text::Wrapping, vertical_rule,
vertical_space,
},
window, window,
}, },
iced_core::{Border, Padding},
iced_runtime::{core::event, dnd::peek_dnd}, iced_runtime::{core::event, dnd::peek_dnd},
surface, surface,
theme::{self, Button, Container}, theme::{self, Button, Container},
@ -52,7 +53,6 @@ use cosmic::{
use cosmic_app_list_config::{APP_ID, AppListConfig}; use cosmic_app_list_config::{APP_ID, AppListConfig};
use cosmic_protocols::toplevel_info::v1::client::zcosmic_toplevel_handle_v1::State; use cosmic_protocols::toplevel_info::v1::client::zcosmic_toplevel_handle_v1::State;
use futures::future::pending; use futures::future::pending;
use iced::{Alignment, Background, Length};
use rustc_hash::FxHashMap; use rustc_hash::FxHashMap;
use std::{borrow::Cow, path::PathBuf, rc::Rc, str::FromStr, time::Duration}; use std::{borrow::Cow, path::PathBuf, rc::Rc, str::FromStr, time::Duration};
use switcheroo_control::Gpu; use switcheroo_control::Gpu;
@ -279,7 +279,7 @@ impl DockItem {
.first() .first()
.map(|t| Message::Toggle(t.0.foreign_toplevel.clone())) .map(|t| Message::Toggle(t.0.foreign_toplevel.clone()))
} else { } else {
Some(Message::TopLevelListPopup(*id, window_id)) Some(Message::ToplevelListPopup(*id, window_id))
}) })
.width(Length::Shrink) .width(Length::Shrink)
.height(Length::Shrink), .height(Length::Shrink),
@ -357,6 +357,7 @@ struct CosmicAppList {
active_workspaces: Vec<ExtWorkspaceHandleV1>, active_workspaces: Vec<ExtWorkspaceHandleV1>,
output_list: FxHashMap<WlOutput, OutputInfo>, output_list: FxHashMap<WlOutput, OutputInfo>,
locales: Vec<String>, locales: Vec<String>,
hovered_toplevel: Option<ExtForeignToplevelHandleV1>,
overflow_favorites_popup: Option<window::Id>, overflow_favorites_popup: Option<window::Id>,
overflow_active_popup: Option<window::Id>, overflow_active_popup: Option<window::Id>,
} }
@ -364,7 +365,7 @@ struct CosmicAppList {
#[derive(Debug, Clone, PartialEq)] #[derive(Debug, Clone, PartialEq)]
pub enum PopupType { pub enum PopupType {
RightClickMenu, RightClickMenu,
TopLevelList, ToplevelList,
} }
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -374,13 +375,15 @@ enum Message {
UnpinApp(u32), UnpinApp(u32),
Popup(u32, window::Id), Popup(u32, window::Id),
Pressed(window::Id), Pressed(window::Id),
TopLevelListPopup(u32, window::Id), ToplevelListPopup(u32, window::Id),
ToplevelHoverChanged(ExtForeignToplevelHandleV1, bool),
GpuRequest(Option<Vec<Gpu>>), GpuRequest(Option<Vec<Gpu>>),
CloseRequested(window::Id), CloseRequested(window::Id),
ClosePopup, ClosePopup,
Activate(ExtForeignToplevelHandleV1), Activate(ExtForeignToplevelHandleV1),
Toggle(ExtForeignToplevelHandleV1), Toggle(ExtForeignToplevelHandleV1),
Exec(String, Option<usize>, bool), Exec(String, Option<usize>, bool),
CloseToplevel(ExtForeignToplevelHandleV1),
Quit(String), Quit(String),
NewSeat(WlSeat), NewSeat(WlSeat),
RemovedSeat, RemovedSeat,
@ -457,58 +460,76 @@ async fn try_get_gpus() -> Option<Vec<Gpu>> {
Some(gpus) Some(gpus)
} }
const TOPLEVEL_BUTTON_WIDTH: f32 = 160.0; const TOPLEVEL_BUTTON_WIDTH: f32 = 192.0;
const TOPLEVEL_BUTTON_HEIGHT: f32 = 130.0; const TOPLEVEL_BUTTON_HEIGHT: f32 = 156.0;
pub fn toplevel_button<'a, Msg>( fn toplevel_button<'a>(
img: Option<WaylandImage>, img: Option<WaylandImage>,
on_press: Msg,
title: String, title: String,
handle: ExtForeignToplevelHandleV1,
is_focused: bool, is_focused: bool,
) -> cosmic::widget::Button<'a, Msg> is_hovered: bool,
where ) -> Element<'a, Message> {
Msg: 'static + Clone, let title = if title.len() > 22 {
{ format!("{:.20}...", title)
} else {
title
};
let border = 1.0; let border = 1.0;
button::custom( let preview = column![
container( container(if let Some(img) = img {
column![ Element::from(Image::new(Handle::from_rgba(
container(if let Some(img) = img { img.width,
Element::from(Image::new(Handle::from_rgba( img.height,
img.width, img.img.clone(),
img.height, )))
img.img.clone(), } else {
))) Image::new(Handle::from_rgba(1, 1, [0u8, 0u8, 0u8, 255u8].as_slice())).into()
} else { })
Image::new(Handle::from_rgba(1, 1, [0u8, 0u8, 0u8, 255u8].as_slice())).into() .class(Container::custom(move |theme| container::Style {
}) border: Border {
.class(Container::Custom(Box::new(move |theme| { color: theme.cosmic().bg_divider().into(),
container::Style { width: border,
border: Border { radius: 1.0.into(),
color: theme.cosmic().bg_divider().into(), },
width: border, ..Default::default()
radius: 0.0.into(), }))
}, .padding(border as u16)
..Default::default() .apply(container)
}
})))
.padding(border as u16)
.height(Length::Shrink)
.width(Length::Shrink)
.apply(container)
.center_y(Length::Fixed(90.0)),
text::body(title),
]
.spacing(4)
.align_x(Alignment::Center),
)
.center(Length::Fill), .center(Length::Fill),
) text::body(title)
.on_press(on_press) .wrapping(Wrapping::None)
.class(window_menu_style(is_focused)) .width(Length::Fill)
.width(Length::Fixed(TOPLEVEL_BUTTON_WIDTH)) .center()
.height(Length::Fixed(TOPLEVEL_BUTTON_HEIGHT)) ]
.selected(is_focused) .spacing(4)
.padding([4, 4, 0, 4]);
let close_button_overlay = if is_hovered {
row![
horizontal_space(),
button::custom(icon::from_name("window-close-symbolic").size(16))
.class(Button::Destructive)
.on_press(Message::CloseToplevel(handle.clone()))
.padding(4)
]
} else {
row![]
}
.width(Length::Fill)
.height(Length::Fill);
stack![preview, close_button_overlay]
.apply(button::custom)
.on_press(Message::Toggle(handle.clone()))
.class(window_menu_style(is_focused))
.width(Length::Fixed(TOPLEVEL_BUTTON_WIDTH))
.height(Length::Fixed(TOPLEVEL_BUTTON_HEIGHT))
.padding(4)
.selected(is_focused)
.apply(mouse_area)
.on_enter(Message::ToplevelHoverChanged(handle.clone(), true))
.on_exit(Message::ToplevelHoverChanged(handle, false))
.apply(Element::from)
} }
fn window_menu_style(selected: bool) -> cosmic::theme::Button { fn window_menu_style(selected: bool) -> cosmic::theme::Button {
@ -589,7 +610,7 @@ fn app_list_icon_style(selected: bool) -> cosmic::theme::Button {
#[inline] #[inline]
pub fn menu_control_padding() -> Padding { pub fn menu_control_padding() -> Padding {
let spacing = cosmic::theme::spacing(); let spacing = theme::spacing();
[spacing.space_xxs, spacing.space_s].into() [spacing.space_xxs, spacing.space_s].into()
} }
@ -627,6 +648,158 @@ impl CosmicAppList {
}) })
.collect(); .collect();
} }
/// Close any open popups.
fn close_popups(&mut self) -> Task<cosmic::Action<Message>> {
let mut commands = Vec::new();
if let Some(popup) = self.popup.take() {
commands.push(destroy_popup(popup.id));
}
if let Some(popup) = self.overflow_active_popup.take() {
commands.push(destroy_popup(popup));
}
if let Some(popup) = self.overflow_favorites_popup.take() {
commands.push(destroy_popup(popup));
}
Task::batch(commands)
}
/// Returns the length of the group in the favorite list after which items are displayed in a popup.
/// Shrink the favorite list until it only has active windows, or until it fits in the length provided.
fn panel_overflow_lengths(&self) -> (Option<usize>, Option<usize>) {
let mut favorite_index;
let mut active_index = None;
let Some(mut max_major_axis_len) = self.core.applet.suggested_bounds.as_ref().map(|c| {
// if we have a configure for width and height, we're in a overflow popup
match self.core.applet.anchor {
PanelAnchor::Top | PanelAnchor::Bottom => c.width as u32,
PanelAnchor::Left | PanelAnchor::Right => c.height as u32,
}
}) else {
return (None, active_index);
};
// tracing::error!("{} {}", max_major_axis_len, self.pinned_list.len());
// subtract the divider width
max_major_axis_len -= 1;
let applet_icon = AppletIconData::new(&self.core.applet);
let button_total_size = self.core.applet.suggested_size(true).0
+ self.core.applet.suggested_padding(true).0 * 2
+ applet_icon.icon_spacing as u16;
let favorite_active_cnt = self
.pinned_list
.iter()
.filter(|t| !t.toplevels.is_empty())
.count();
// initial calculation of favorite_index
let btn_count = max_major_axis_len / button_total_size as u32;
if btn_count >= self.pinned_list.len() as u32 + self.active_list.len() as u32 {
return (None, active_index);
} else {
favorite_index = (btn_count as usize).min(favorite_active_cnt).max(2);
}
// calculation of active_index based on favorite_index if there is still not enough space
let active_index_max = (btn_count as i32)
- (self.pinned_list.len() as i32).saturating_sub(favorite_index as i32);
if active_index_max >= self.active_list.len() as i32 {
active_index = Some(self.active_list.len());
} else {
active_index = Some((active_index_max.max(2) as usize).min(self.active_list.len()));
}
// final calculation of favorite_index if there is still not enough space
if let Some(active_index) = active_index {
let favorite_index_max = (btn_count as i32) - active_index as i32;
favorite_index = favorite_index_max.max(2) as usize;
} else {
favorite_index = (btn_count as usize).min(self.pinned_list.len());
}
// tracing::error!("{} {} {:?}", btn_count, favorite_index, active_index);
(Some(favorite_index), active_index)
}
fn currently_active_toplevel(&self) -> Vec<ExtForeignToplevelHandleV1> {
if self.active_workspaces.is_empty() {
return Vec::new();
}
let current_output = &self.core.applet.output_name;
let mut focused_toplevels: Vec<ExtForeignToplevelHandleV1> = Vec::new();
let active_workspaces = &self.active_workspaces;
for toplevel_list in self.active_list.iter().chain(self.pinned_list.iter()) {
for (t_info, _) in &toplevel_list.toplevels {
if t_info.state.contains(&State::Activated)
&& active_workspaces
.iter()
.any(|workspace| t_info.workspace.contains(workspace))
&& t_info.output.iter().any(|x| {
self.output_list.get(x).is_some_and(|val| {
val.name.as_ref().is_some_and(|n| n == current_output)
})
})
{
focused_toplevels.push(t_info.foreign_toplevel.clone());
}
}
}
focused_toplevels
}
fn find_desktop_entry_for_toplevel(
&mut self,
info: &ToplevelInfo,
unicase_appid: Ascii<&str>,
) -> DesktopEntry {
if let Some(appid) = fde::find_app_by_id(&self.desktop_entries, unicase_appid) {
appid.clone()
} else {
// Update desktop entries in case it was not found.
self.update_desktop_entries();
if let Some(appid) = fde::find_app_by_id(&self.desktop_entries, unicase_appid) {
appid.clone()
} else {
tracing::error!(id = info.app_id, "could not find desktop entry for app");
let mut fallback_entry = fde::DesktopEntry::from_appid(info.app_id.clone());
// proton opens games as steam_app_X, where X is either
// the steam appid or "default". games with a steam appid
// can have a desktop entry generated elsewhere; this
// specifically handles non-steam games opened
// under proton
// in addition, try to match WINE entries who have its
// appid = the full name of the executable (incl. .exe)
let is_proton_game = info.app_id == "steam_app_default";
if is_proton_game || info.app_id.ends_with(".exe") {
for entry in &self.desktop_entries {
let localised_name = entry.name(&self.locales).unwrap_or_default();
if localised_name == info.title {
// if this is a proton game, we only want
// to look for game entries
if is_proton_game
&& !entry.categories().unwrap_or_default().contains(&"Game")
{
continue;
}
fallback_entry = entry.clone();
break;
}
}
}
fallback_entry
}
}
}
// Check if a specific toplevel is focused
fn is_focused(&self, handle: &ExtForeignToplevelHandleV1) -> bool {
self.currently_active_toplevel().contains(handle)
}
// Check if a specific toplevel button is currently hovered
fn is_hovered(&self, handle: &ExtForeignToplevelHandleV1) -> bool {
self.hovered_toplevel.as_ref() == Some(handle)
}
} }
impl cosmic::Application for CosmicAppList { impl cosmic::Application for CosmicAppList {
@ -731,7 +904,7 @@ impl cosmic::Application for CosmicAppList {
return Task::batch([gpu_update, get_popup(popup_settings)]); return Task::batch([gpu_update, get_popup(popup_settings)]);
} }
} }
Message::TopLevelListPopup(id, parent_window_id) => { Message::ToplevelListPopup(id, parent_window_id) => {
if let Some(Popup { if let Some(Popup {
parent, parent,
id: popup_id, id: popup_id,
@ -768,7 +941,7 @@ impl cosmic::Application for CosmicAppList {
parent: parent_window_id, parent: parent_window_id,
id: new_id, id: new_id,
dock_item: toplevel_group.clone(), dock_item: toplevel_group.clone(),
popup_type: PopupType::TopLevelList, popup_type: PopupType::ToplevelList,
}); });
let mut popup_settings = self.core.applet.get_popup_settings( let mut popup_settings = self.core.applet.get_popup_settings(
@ -814,6 +987,14 @@ impl cosmic::Application for CosmicAppList {
return get_popup(popup_settings); return get_popup(popup_settings);
} }
} }
Message::ToplevelHoverChanged(handle, entering) => {
match (entering, &self.hovered_toplevel) {
(true, _) => self.hovered_toplevel = Some(handle),
// prevents race condition
(false, Some(h)) if h == &handle => self.hovered_toplevel = None,
_ => {}
}
}
Message::PinApp(id) => { Message::PinApp(id) => {
if let Some(i) = self.active_list.iter().position(|t| t.id == id) { if let Some(i) = self.active_list.iter().position(|t| t.id == id) {
let entry = self.active_list.remove(i); let entry = self.active_list.remove(i);
@ -855,18 +1036,21 @@ impl cosmic::Application for CosmicAppList {
} }
Message::Toggle(handle) => { Message::Toggle(handle) => {
if let Some(tx) = self.wayland_sender.as_ref() { if let Some(tx) = self.wayland_sender.as_ref() {
let _ = tx.send(WaylandRequest::Toplevel( let _ = tx.send(WaylandRequest::Toplevel(if self.is_focused(&handle) {
if self.currently_active_toplevel().contains(&handle) { ToplevelRequest::Minimize(handle)
ToplevelRequest::Minimize(handle) } else {
} else { ToplevelRequest::Activate(handle)
ToplevelRequest::Activate(handle) }));
},
));
} }
if let Some(p) = self.popup.take() { if let Some(p) = self.popup.take() {
return destroy_popup(p.id); return destroy_popup(p.id);
} }
} }
Message::CloseToplevel(handle) => {
if let Some(tx) = self.wayland_sender.as_ref() {
let _ = tx.send(WaylandRequest::Toplevel(ToplevelRequest::Quit(handle)));
}
}
Message::Quit(id) => { Message::Quit(id) => {
if let Some(toplevel_group) = self if let Some(toplevel_group) = self
.active_list .active_list
@ -1144,6 +1328,21 @@ impl cosmic::Application for CosmicAppList {
.retain(|(info, _)| info.foreign_toplevel != handle); .retain(|(info, _)| info.foreign_toplevel != handle);
} }
self.active_list.retain(|t| !t.toplevels.is_empty()); self.active_list.retain(|t| !t.toplevels.is_empty());
if let Some(popup) = &mut self.popup
&& popup.popup_type == PopupType::ToplevelList
{
popup
.dock_item
.toplevels
.retain(|(info, _)| info.foreign_toplevel != handle);
if popup.dock_item.toplevels.is_empty() {
let id = popup.id;
self.popup = None;
return destroy_popup(id);
}
}
} }
ToplevelUpdate::Update(info) => { ToplevelUpdate::Update(info) => {
// TODO probably want to make sure it is removed // TODO probably want to make sure it is removed
@ -1809,7 +2008,7 @@ impl cosmic::Application for CosmicAppList {
content: impl Into<Element<'a, Message>>, content: impl Into<Element<'a, Message>>,
) -> cosmic::widget::Button<'a, Message> { ) -> cosmic::widget::Button<'a, Message> {
button::custom(content) button::custom(content)
.height(20 + 2 * theme::active().cosmic().space_xxs()) .height(20 + 2 * theme::spacing().space_xxs)
.class(Button::MenuItem) .class(Button::MenuItem)
.padding(menu_control_padding()) .padding(menu_control_padding())
.width(Length::Fill) .width(Length::Fill)
@ -1957,22 +2156,17 @@ impl cosmic::Application for CosmicAppList {
) )
.into() .into()
} }
PopupType::TopLevelList => match self.core.applet.anchor { PopupType::ToplevelList => match self.core.applet.anchor {
PanelAnchor::Left | PanelAnchor::Right => { PanelAnchor::Left | PanelAnchor::Right => {
let mut content = let mut content =
column![].padding(8).align_x(Alignment::Center).spacing(8); column![].padding(8).align_x(Alignment::Center).spacing(8);
for (info, img) in toplevels { for (info, img) in toplevels {
let title = if info.title.len() > 18 {
format!("{:.16}...", &info.title)
} else {
info.title.clone()
};
content = content.push(toplevel_button( content = content.push(toplevel_button(
img.clone(), img.clone(),
Message::Toggle(info.foreign_toplevel.clone()), info.title.clone(),
title, info.foreign_toplevel.clone(),
self.currently_active_toplevel() self.is_focused(&info.foreign_toplevel),
.contains(&info.foreign_toplevel), self.is_hovered(&info.foreign_toplevel),
)); ));
} }
self.core self.core
@ -1984,17 +2178,12 @@ impl cosmic::Application for CosmicAppList {
PanelAnchor::Bottom | PanelAnchor::Top => { PanelAnchor::Bottom | PanelAnchor::Top => {
let mut content = row![].padding(8).align_y(Alignment::Center).spacing(8); let mut content = row![].padding(8).align_y(Alignment::Center).spacing(8);
for (info, img) in toplevels { for (info, img) in toplevels {
let title = if info.title.len() > 18 {
format!("{:.16}...", &info.title)
} else {
info.title.clone()
};
content = content.push(toplevel_button( content = content.push(toplevel_button(
img.clone(), img.clone(),
Message::Toggle(info.foreign_toplevel.clone()), info.title.clone(),
title, info.foreign_toplevel.clone(),
self.currently_active_toplevel() self.is_focused(&info.foreign_toplevel),
.contains(&info.foreign_toplevel), self.is_hovered(&info.foreign_toplevel),
)); ));
} }
self.core self.core
@ -2236,155 +2425,6 @@ impl cosmic::Application for CosmicAppList {
} }
} }
impl CosmicAppList {
/// Close any open popups.
fn close_popups(&mut self) -> Task<cosmic::Action<Message>> {
let mut commands = Vec::new();
if let Some(popup) = self.popup.take() {
commands.push(destroy_popup(popup.id));
}
if let Some(popup) = self.overflow_active_popup.take() {
commands.push(destroy_popup(popup));
}
if let Some(popup) = self.overflow_favorites_popup.take() {
commands.push(destroy_popup(popup));
}
Task::batch(commands)
}
/// Returns the length of the group in the favorite list after which items are displayed in a popup.
/// Shrink the favorite list until it only has active windows, or until it fits in the length provided.
fn panel_overflow_lengths(&self) -> (Option<usize>, Option<usize>) {
let mut favorite_index;
let mut active_index = None;
let Some(mut max_major_axis_len) = self.core.applet.suggested_bounds.as_ref().map(|c| {
// if we have a configure for width and height, we're in a overflow popup
match self.core.applet.anchor {
PanelAnchor::Top | PanelAnchor::Bottom => c.width as u32,
PanelAnchor::Left | PanelAnchor::Right => c.height as u32,
}
}) else {
return (None, active_index);
};
// tracing::error!("{} {}", max_major_axis_len, self.pinned_list.len());
// subtract the divider width
max_major_axis_len -= 1;
let applet_icon = AppletIconData::new(&self.core.applet);
let button_total_size = self.core.applet.suggested_size(true).0
+ self.core.applet.suggested_padding(true).0 * 2
+ applet_icon.icon_spacing as u16;
let favorite_active_cnt = self
.pinned_list
.iter()
.filter(|t| !t.toplevels.is_empty())
.count();
// initial calculation of favorite_index
let btn_count = max_major_axis_len / button_total_size as u32;
if btn_count >= self.pinned_list.len() as u32 + self.active_list.len() as u32 {
return (None, active_index);
} else {
favorite_index = (btn_count as usize).min(favorite_active_cnt).max(2);
}
// calculation of active_index based on favorite_index if there is still not enough space
let active_index_max = (btn_count as i32)
- (self.pinned_list.len() as i32).saturating_sub(favorite_index as i32);
if active_index_max >= self.active_list.len() as i32 {
active_index = Some(self.active_list.len());
} else {
active_index = Some((active_index_max.max(2) as usize).min(self.active_list.len()));
}
// final calculation of favorite_index if there is still not enough space
if let Some(active_index) = active_index {
let favorite_index_max = (btn_count as i32) - active_index as i32;
favorite_index = favorite_index_max.max(2) as usize;
} else {
favorite_index = (btn_count as usize).min(self.pinned_list.len());
}
// tracing::error!("{} {} {:?}", btn_count, favorite_index, active_index);
(Some(favorite_index), active_index)
}
fn currently_active_toplevel(&self) -> Vec<ExtForeignToplevelHandleV1> {
if self.active_workspaces.is_empty() {
return Vec::new();
}
let current_output = &self.core.applet.output_name;
let mut focused_toplevels: Vec<ExtForeignToplevelHandleV1> = Vec::new();
let active_workspaces = &self.active_workspaces;
for toplevel_list in self.active_list.iter().chain(self.pinned_list.iter()) {
for (t_info, _) in &toplevel_list.toplevels {
if t_info.state.contains(&State::Activated)
&& active_workspaces
.iter()
.any(|workspace| t_info.workspace.contains(workspace))
&& t_info.output.iter().any(|x| {
self.output_list.get(x).is_some_and(|val| {
val.name.as_ref().is_some_and(|n| n == current_output)
})
})
{
focused_toplevels.push(t_info.foreign_toplevel.clone());
}
}
}
focused_toplevels
}
fn find_desktop_entry_for_toplevel(
&mut self,
info: &ToplevelInfo,
unicase_appid: Ascii<&str>,
) -> DesktopEntry {
if let Some(appid) = fde::find_app_by_id(&self.desktop_entries, unicase_appid) {
appid.clone()
} else {
// Update desktop entries in case it was not found.
self.update_desktop_entries();
if let Some(appid) = fde::find_app_by_id(&self.desktop_entries, unicase_appid) {
appid.clone()
} else {
tracing::error!(id = info.app_id, "could not find desktop entry for app");
let mut fallback_entry = fde::DesktopEntry::from_appid(info.app_id.clone());
// proton opens games as steam_app_X, where X is either
// the steam appid or "default". games with a steam appid
// can have a desktop entry generated elsewhere; this
// specifically handles non-steam games opened
// under proton
// in addition, try to match WINE entries who have its
// appid = the full name of the executable (incl. .exe)
let is_proton_game = info.app_id == "steam_app_default";
if is_proton_game || info.app_id.ends_with(".exe") {
for entry in &self.desktop_entries {
let localised_name = entry.name(&self.locales).unwrap_or_default();
if localised_name == info.title {
// if this is a proton game, we only want
// to look for game entries
if is_proton_game
&& !entry.categories().unwrap_or_default().contains(&"Game")
{
continue;
}
fallback_entry = entry.clone();
break;
}
}
}
fallback_entry
}
}
}
}
fn launch_on_preferred_gpu(desktop_info: &DesktopEntry, gpus: Option<&[Gpu]>) -> Option<Message> { fn launch_on_preferred_gpu(desktop_info: &DesktopEntry, gpus: Option<&[Gpu]>) -> Option<Message> {
let exec = desktop_info.exec()?; let exec = desktop_info.exec()?;

View file

@ -506,9 +506,9 @@ impl AppData {
return; return;
}; };
// resize to 128x128 // resize to 256x256
let max = img.width().max(img.height()); let max = img.width().max(img.height());
let ratio = max as f32 / 128.0; let ratio = max as f32 / 256.0;
let img = if ratio > 1.0 { let img = if ratio > 1.0 {
let new_width = (img.width() as f32 / ratio).round(); let new_width = (img.width() as f32 / ratio).round();