use crate::theme::error::ThemeError; use crate::theme::paths::ThemePath; use ini::Ini; use once_cell::sync::Lazy; pub(crate) use paths::BASE_PATHS; use std::collections::BTreeMap; use std::fmt::{Debug, Formatter}; use std::path::{Path, PathBuf}; mod directories; pub mod error; mod parse; mod paths; type Result = std::result::Result; pub static THEMES: Lazy> = Lazy::new(|| get_all_themes().expect("Failed to get theme paths")); pub struct Theme { pub path: ThemePath, pub index: Ini, } impl Theme { pub fn try_get_icon( &self, name: &str, size: u16, scale: u16, force_svg: bool, ) -> Option { self.try_get_icon_exact_size(name, size, scale, force_svg) .or_else(|| self.try_get_icon_closest_size(name, size, scale, force_svg)) } fn try_get_icon_exact_size( &self, name: &str, size: u16, scale: u16, force_svg: bool, ) -> Option { self.match_size(size, scale) .find_map(|path| try_build_icon_path(name, path, force_svg)) } fn match_size(&self, size: u16, scale: u16) -> impl Iterator + '_ { let dirs = self.get_all_directories(); dirs.filter(move |directory| directory.match_size(size, scale)) .map(|dir| dir.name) .map(|dir| self.path().join(dir)) } fn try_get_icon_closest_size( &self, name: &str, size: u16, scale: u16, force_svg: bool, ) -> Option { self.closest_match_size(size, scale) .iter() .find_map(|path| try_build_icon_path(name, path, force_svg)) } fn closest_match_size(&self, size: u16, scale: u16) -> Vec { let dirs = self.get_all_directories(); dirs.filter(|directory| directory.directory_size_distance(size, scale) < i16::MAX) .map(|dir| dir.name) .map(|dir| self.path().join(dir)) .collect() } fn path(&self) -> &PathBuf { &self.path.0 } } pub(super) fn try_build_icon_path>( name: &str, path: P, force_svg: bool, ) -> Option { if force_svg { try_build_svg(name, path.as_ref()) } else { try_build_png(name, path.as_ref()) .or_else(|| try_build_svg(name, path.as_ref())) .or_else(|| try_build_xmp(name, path.as_ref())) } } fn try_build_svg>(name: &str, path: P) -> Option { let path = path.as_ref(); let svg = path.join(format!("{name}.svg")); if svg.exists() { Some(svg) } else { None } } fn try_build_png>(name: &str, path: P) -> Option { let path = path.as_ref(); let png = path.join(format!("{name}.png")); if png.exists() { Some(png) } else { None } } fn try_build_xmp>(name: &str, path: P) -> Option { let path = path.as_ref(); let xmp = path.join(format!("{name}.xmp")); if xmp.exists() { Some(xmp) } else { None } } // Iter through the base paths and get all theme directories pub(super) fn get_all_themes() -> Result> { let mut icon_themes = BTreeMap::new(); for theme_base_dir in BASE_PATHS.iter() { for entry in theme_base_dir.read_dir()? { let entry = entry?; if let Some(theme) = Theme::from_path(entry.path()) { let name = entry.file_name().to_string_lossy().to_string(); icon_themes.insert(name, theme); } } } Ok(icon_themes) } impl Theme { pub(crate) fn from_path>(path: P) -> Option { let path = path.as_ref(); let has_index = path.join("index.theme").exists(); if !has_index || !path.is_dir() { return None; } let path = ThemePath(path.into()); match path.index() { Ok(index) => Some(Theme { path, index }), Err(_) => None, } } } impl Debug for Theme { fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { let mut content = vec![]; self.index.write_to(&mut content).expect("Write error"); let content = String::from_utf8_lossy(&content); writeln!(f, "ThemeIndex{{path: {:?}, index: {content:?}}}", self.path) } } #[cfg(test)] mod test { use crate::THEMES; use speculoos::prelude::*; use std::path::PathBuf; #[test] fn get_one_icon() { let theme = THEMES.get("Adwaita").unwrap(); println!( "{:?}", theme.try_get_icon_exact_size("edit-delete-symbolic", 24, 1, false) ); } #[test] fn should_get_png_first() { let theme = THEMES.get("hicolor").unwrap(); let icon = theme.try_get_icon_exact_size("blueman", 24, 1, true); assert_that!(icon).is_some().is_equal_to(PathBuf::from( "/usr/share/icons/hicolor/scalable/apps/blueman.svg", )); } #[test] fn should_get_svg_first() { let theme = THEMES.get("hicolor").unwrap(); let icon = theme.try_get_icon_exact_size("blueman", 24, 1, false); assert_that!(icon).is_some().is_equal_to(PathBuf::from( "/usr/share/icons/hicolor/22x22/apps/blueman.png", )); } }