use std::fs::read_to_string;
use std::path::Path;
use super::ThemeType;
use crate::color;
use crate::keybindings::Keybindings;
use anyhow::Context;
use serde::Deserialize;
use syntect::highlighting::Theme as SyntectTheme;
#[derive(Deserialize, Debug, PartialEq, Eq, Default, Clone)]
#[serde(rename_all = "kebab-case")]
pub struct FontOptions {
#[serde(default)]
pub regular_font: Option<String>,
#[serde(default)]
pub monospace_font: Option<String>,
}
#[derive(Deserialize, Debug, Default, PartialEq)]
#[serde(default, rename_all = "kebab-case")]
pub struct OptionalTheme {
pub text_color: Option<u32>,
pub background_color: Option<u32>,
pub code_color: Option<u32>,
pub quote_block_color: Option<u32>,
pub link_color: Option<u32>,
pub select_color: Option<u32>,
pub checkbox_color: Option<u32>,
pub code_highlighter: Option<color::SyntaxTheme>,
}
impl OptionalTheme {
pub fn merge(self, other: color::Theme) -> anyhow::Result<color::Theme> {
let code_highlighter = match self.code_highlighter {
Some(theme) => SyntectTheme::try_from(theme)?,
None => other.code_highlighter,
};
Ok(color::Theme {
text_color: self.text_color.unwrap_or(other.text_color),
background_color: self.background_color.unwrap_or(other.background_color),
code_color: self.code_color.unwrap_or(other.code_color),
quote_block_color: self.quote_block_color.unwrap_or(other.quote_block_color),
link_color: self.link_color.unwrap_or(other.link_color),
select_color: self.select_color.unwrap_or(other.select_color),
checkbox_color: self.checkbox_color.unwrap_or(other.checkbox_color),
code_highlighter,
})
}
}
#[derive(Deserialize, Debug, PartialEq)]
pub struct LinesToScroll(pub f32);
impl From<LinesToScroll> for f32 {
fn from(value: LinesToScroll) -> Self {
value.0
}
}
impl Default for LinesToScroll {
fn default() -> Self {
Self(3.0)
}
}
#[derive(Deserialize, Clone, Debug, Default, PartialEq)]
pub struct KeybindingsSection {
#[serde(default)]
pub base: Keybindings,
pub extra: Option<Keybindings>,
}
#[derive(Deserialize, Debug, Default, PartialEq)]
#[serde(default, rename_all = "kebab-case")]
pub struct Config {
pub theme: Option<ThemeType>,
pub scale: Option<f32>,
pub page_width: Option<f32>,
pub lines_to_scroll: LinesToScroll,
pub light_theme: Option<OptionalTheme>,
pub dark_theme: Option<OptionalTheme>,
pub font_options: Option<FontOptions>,
pub keybindings: KeybindingsSection,
}
impl Config {
pub fn load_from_str(s: &str) -> anyhow::Result<Self> {
let config = toml::from_str(s)?;
Ok(config)
}
pub fn load_from_file(path: &Path) -> anyhow::Result<Self> {
let config_content = read_to_string(path).context(format!(
"Failed to read configuration file at '{}'",
path.display()
))?;
Self::load_from_str(&config_content)
}
pub fn load_from_system() -> anyhow::Result<Self> {
let config_dir =
dirs::config_dir().context("Failed to find the configuration directory")?;
let config_path = config_dir.join("inlyne").join("inlyne.toml");
if !config_path.is_file() {
return Ok(Self::default());
}
Self::load_from_file(&config_path)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn defaults_file_is_in_sync() {
let mut config = Config::load_from_file(Path::new("inlyne.default.toml")).unwrap();
let theme = config.theme.take().unwrap();
let _ = config.font_options.take();
let dark_theme = config.dark_theme.take().unwrap();
let light_theme = config.light_theme.take().unwrap();
assert_eq!(config, Config::default());
assert_eq!(theme, ThemeType::Auto);
assert_eq!(
dark_theme.merge(color::Theme::dark_default()).unwrap(),
color::Theme::dark_default()
);
assert_eq!(
light_theme.merge(color::Theme::light_default()).unwrap(),
color::Theme::light_default()
);
}
}