libcosmic/src/widget/icon/named.rs

150 lines
3.8 KiB
Rust
Raw Normal View History

// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: MPL-2.0
use super::{Handle, Icon};
2023-12-29 12:09:32 -05:00
use std::{borrow::Cow, path::PathBuf, sync::Arc};
#[derive(Debug, Clone, Default, Hash)]
/// Fallback icon to use if the icon was not found.
pub enum IconFallback {
#[default]
/// Default fallback using the icon name.
Default,
/// Fallback to specific icon names.
Names(Vec<Cow<'static, str>>),
}
#[must_use]
#[derive(derive_setters::Setters, Clone, Debug, Hash)]
pub struct Named {
/// Name of icon to locate in an XDG icon path.
pub(super) name: Arc<str>,
/// Checks for a fallback if the icon was not found.
2023-12-29 12:09:32 -05:00
pub fallback: Option<IconFallback>,
/// Restrict the lookup to a given scale.
#[setters(strip_option)]
pub scale: Option<u16>,
/// Restrict the lookup to a given size.
#[setters(strip_option)]
pub size: Option<u16>,
/// Whether the icon is symbolic or not.
pub symbolic: bool,
/// Prioritizes SVG over PNG
pub prefer_svg: bool,
}
impl Named {
pub fn new(name: impl Into<Arc<str>>) -> Self {
let name = name.into();
Self {
symbolic: name.ends_with("-symbolic"),
name,
2023-12-29 12:09:32 -05:00
fallback: Some(IconFallback::Default),
size: None,
scale: None,
prefer_svg: false,
}
}
2023-11-16 16:36:29 -08:00
#[cfg(not(windows))]
#[must_use]
pub fn path(self) -> Option<PathBuf> {
2024-01-02 14:18:27 -05:00
let name = &*self.name;
2023-12-29 12:09:32 -05:00
let fallback = &self.fallback;
crate::icon_theme::DEFAULT.with(|theme| {
let theme = theme.borrow();
2024-01-02 14:18:27 -05:00
let locate = |name| {
let mut lookup = freedesktop_icons::lookup(name)
.with_theme(theme.as_ref())
.with_cache();
if let Some(scale) = self.scale {
lookup = lookup.with_scale(scale);
}
if let Some(size) = self.size {
lookup = lookup.with_size(size);
}
if self.prefer_svg {
lookup = lookup.force_svg();
}
lookup.find()
};
2024-01-02 14:18:27 -05:00
let mut result = locate(name);
// On failure, attempt to locate fallback icon.
2023-12-29 12:09:32 -05:00
if result.is_none() {
if matches!(fallback, Some(IconFallback::Default)) {
for new_name in name.rmatch_indices('-').map(|(pos, _)| &name[..pos]) {
2024-01-02 14:18:27 -05:00
result = locate(new_name);
2023-12-29 12:09:32 -05:00
if result.is_some() {
break;
}
}
} else if let Some(IconFallback::Names(fallbacks)) = fallback {
for fallback in fallbacks {
2024-01-02 14:18:27 -05:00
result = locate(fallback);
2023-12-29 12:09:32 -05:00
if result.is_some() {
break;
}
}
}
}
result
})
}
2023-11-16 16:36:29 -08:00
#[cfg(windows)]
#[must_use]
pub fn path(self) -> Option<PathBuf> {
//TODO: implement icon lookup for Windows
None
}
pub fn handle(self) -> Handle {
Handle {
symbolic: self.symbolic,
data: super::Data::Name(self),
}
}
pub fn icon(self) -> Icon {
let size = self.size;
let icon = super::icon(self.handle());
match size {
Some(size) => icon.size(size),
None => icon,
}
}
}
impl From<Named> for Handle {
fn from(builder: Named) -> Self {
builder.handle()
}
}
impl From<Named> for Icon {
fn from(builder: Named) -> Self {
builder.icon()
}
}
impl<'a, Message: 'static> From<Named> for crate::Element<'a, Message> {
fn from(builder: Named) -> Self {
builder.icon().into()
}
}