diff options
author | Michael Davis | 2024-05-01 22:52:02 +0000 |
---|---|---|
committer | JJ | 2024-05-01 23:54:41 +0000 |
commit | 9a3f23b0661f7a37a0dab885fe5eb844b615a22b (patch) | |
tree | 48a816a487a38bf3145263a61d96c4a06d28765f /helix-view/src/theme.rs | |
parent | 214f7ba218223cd292fb1788ad203e5168657975 (diff) |
Add rainbow tree-sitter highlights
ref: https://github.com/helix-editor/helix/issues/695
ref: https://github.com/helix-editor/helix/pull/2857
Diffstat (limited to 'helix-view/src/theme.rs')
-rw-r--r-- | helix-view/src/theme.rs | 113 |
1 files changed, 109 insertions, 4 deletions
diff --git a/helix-view/src/theme.rs b/helix-view/src/theme.rs index 4acc5664..16da247f 100644 --- a/helix-view/src/theme.rs +++ b/helix-view/src/theme.rs @@ -216,17 +216,19 @@ pub struct Theme { // tree-sitter highlight styles are stored in a Vec to optimize lookups scopes: Vec<String>, highlights: Vec<Style>, + rainbow_length: usize, } impl From<Value> for Theme { fn from(value: Value) -> Self { if let Value::Table(table) = value { - let (styles, scopes, highlights) = build_theme_values(table); + let (styles, scopes, highlights, rainbow_length) = build_theme_values(table); Self { styles, scopes, highlights, + rainbow_length, ..Default::default() } } else { @@ -243,12 +245,13 @@ impl<'de> Deserialize<'de> for Theme { { let values = Map::<String, Value>::deserialize(deserializer)?; - let (styles, scopes, highlights) = build_theme_values(values); + let (styles, scopes, highlights, rainbow_length) = build_theme_values(values); Ok(Self { styles, scopes, highlights, + rainbow_length, ..Default::default() }) } @@ -256,10 +259,11 @@ impl<'de> Deserialize<'de> for Theme { fn build_theme_values( mut values: Map<String, Value>, -) -> (HashMap<String, Style>, Vec<String>, Vec<Style>) { +) -> (HashMap<String, Style>, Vec<String>, Vec<Style>, usize) { let mut styles = HashMap::new(); let mut scopes = Vec::new(); let mut highlights = Vec::new(); + let mut rainbow_length = 0; // TODO: alert user of parsing failures in editor let palette = values @@ -276,6 +280,27 @@ fn build_theme_values( styles.reserve(values.len()); scopes.reserve(values.len()); highlights.reserve(values.len()); + + for (i, style) in values + .remove("rainbow") + .and_then(|value| match palette.parse_style_array(value) { + Ok(styles) => Some(styles), + Err(err) => { + warn!("{}", err); + None + } + }) + .unwrap_or_else(default_rainbow) + .iter() + .enumerate() + { + let name = format!("rainbow.{}", i); + styles.insert(name.clone(), *style); + scopes.push(name); + highlights.push(*style); + rainbow_length += 1; + } + for (name, style_value) in values { let mut style = Style::default(); if let Err(err) = palette.parse_style(&mut style, style_value) { @@ -288,7 +313,7 @@ fn build_theme_values( highlights.push(style); } - (styles, scopes, highlights) + (styles, scopes, highlights, rainbow_length) } impl Theme { @@ -354,6 +379,21 @@ impl Theme { .all(|color| !matches!(color, Some(Color::Rgb(..)))) }) } + + pub fn rainbow_length(&self) -> usize { + self.rainbow_length + } +} + +fn default_rainbow() -> Vec<Style> { + vec![ + Style::default().fg(Color::Red), + Style::default().fg(Color::Yellow), + Style::default().fg(Color::Green), + Style::default().fg(Color::Blue), + Style::default().fg(Color::Cyan), + Style::default().fg(Color::Magenta), + ] } struct ThemePalette { @@ -500,6 +540,24 @@ impl ThemePalette { } Ok(()) } + + /// Parses a TOML array into a [`Vec`] of [`Style`]. If the value cannot be + /// parsed as an array or if any style in the array cannot be parsed then an + /// error is returned. + pub fn parse_style_array(&self, value: Value) -> Result<Vec<Style>, String> { + let mut styles = Vec::new(); + + for v in value + .as_array() + .ok_or_else(|| format!("Theme: could not parse value as an array: '{}'", value))? + { + let mut style = Style::default(); + self.parse_style(&mut style, v.clone())?; + styles.push(style); + } + + Ok(styles) + } } impl TryFrom<Value> for ThemePalette { @@ -574,4 +632,51 @@ mod tests { .add_modifier(Modifier::BOLD) ); } + + #[test] + fn test_parse_valid_style_array() { + let theme = toml::toml! { + rainbow = ["#ff0000", "#ffa500", "#fff000", { fg = "#00ff00", modifiers = ["bold"] }] + }; + + let palette = ThemePalette::default(); + + let rainbow = theme.get("rainbow").unwrap(); + let parse_result = palette.parse_style_array(rainbow.clone()); + + assert_eq!( + Ok(vec![ + Style::default().fg(Color::Rgb(255, 0, 0)), + Style::default().fg(Color::Rgb(255, 165, 0)), + Style::default().fg(Color::Rgb(255, 240, 0)), + Style::default() + .fg(Color::Rgb(0, 255, 0)) + .add_modifier(Modifier::BOLD), + ]), + parse_result + ) + } + + #[test] + fn test_parse_invalid_style_array() { + let palette = ThemePalette::default(); + + let theme = toml::toml! { invalid_hex_code = ["#f00"] }; + let invalid_hex_code = theme.get("invalid_hex_code").unwrap(); + let parse_result = palette.parse_style_array(invalid_hex_code.clone()); + + assert_eq!( + Err("Theme: malformed hexcode: #f00".to_string()), + parse_result + ); + + let theme = toml::toml! { not_an_array = { red = "#ff0000" } }; + let not_an_array = theme.get("not_an_array").unwrap(); + let parse_result = palette.parse_style_array(not_an_array.clone()); + + assert_eq!( + Err("Theme: could not parse value as an array: '{ red = \"#ff0000\" }'".to_string()), + parse_result + ) + } } |