use std::{ collections::HashMap, path::{Path, PathBuf}, }; use anyhow::Context; use helix_core::hashmap; use log::warn; use once_cell::sync::Lazy; use serde::{Deserialize, Deserializer}; use toml::Value; pub use crate::graphics::{Color, Modifier, Style}; pub static DEFAULT_THEME: Lazy = Lazy::new(|| { toml::from_slice(include_bytes!("../../theme.toml")).expect("Failed to parse default theme") }); #[derive(Clone, Debug)] pub struct Loader { user_dir: PathBuf, default_dir: PathBuf, } impl Loader { /// Creates a new loader that can load themes from two directories. pub fn new>(user_dir: P, default_dir: P) -> Self { Self { user_dir: user_dir.as_ref().join("themes"), default_dir: default_dir.as_ref().join("themes"), } } /// Loads a theme first looking in the `user_dir` then in `default_dir` pub fn load(&self, name: &str) -> Result { if name == "default" { return Ok(self.default()); } let filename = format!("{}.toml", name); let user_path = self.user_dir.join(&filename); let path = if user_path.exists() { user_path } else { self.default_dir.join(filename) }; let data = std::fs::read(&path)?; toml::from_slice(data.as_slice()).context("Failed to deserialize theme") } pub fn read_names(path: &Path) -> Vec { std::fs::read_dir(path) .map(|entries| { entries .filter_map(|entry| { let entry = entry.ok()?; let path = entry.path(); (path.extension()? == "toml") .then(|| path.file_stem().unwrap().to_string_lossy().into_owned()) }) .collect() }) .unwrap_or_default() } /// Lists all theme names available in default and user directory pub fn names(&self) -> Vec { let mut names = Self::read_names(&self.user_dir); names.extend(Self::read_names(&self.default_dir)); names } /// Returns the default theme pub fn default(&self) -> Theme { DEFAULT_THEME.clone() } } #[derive(Clone, Debug)] pub struct Theme { scopes: Vec, styles: HashMap, } impl<'de> Deserialize<'de> for Theme { fn deserialize(deserializer: D) -> Result where D: Deserializer<'de>, { let mut styles = HashMap::new(); if let Ok(mut colors) = HashMap::::deserialize(deserializer) { // TODO: alert user of parsing failures in editor let palette = colors .remove("palette") .map(|value| { ThemePalette::try_from(value).unwrap_or_else(|err| { warn!("{}", err); ThemePalette::default() }) }) .unwrap_or_default(); styles.reserve(colors.len()); for (name, style_value) in colors { let mut style = Style::default(); if let Err(err) = palette.parse_style(&mut style, style_value) { warn!("{}", err); } styles.insert(name, style); } } let scopes = styles.keys().map(ToString::to_string).collect(); Ok(Self { scopes, styles }) } } impl Theme { pub fn get(&self, scope: &str) -> Style { self.try_get(scope) .unwrap_or_else(|| Style::default().fg(Color::Rgb(0, 0, 255))) } pub fn try_get(&self, scope: &str) -> Option