|
@@ -1,3 +1,5 @@
|
|
|
|
|
+#![allow(dead_code)]
|
|
|
|
|
+
|
|
|
use std::collections::HashMap;
|
|
use std::collections::HashMap;
|
|
|
|
|
|
|
|
use ansi_term::Style;
|
|
use ansi_term::Style;
|
|
@@ -5,7 +7,7 @@ use ansi_term::Colour::*;
|
|
|
|
|
|
|
|
|
|
|
|
|
pub struct LSColors<'var> {
|
|
pub struct LSColors<'var> {
|
|
|
- contents: HashMap<&'var str, &'var str>
|
|
|
|
|
|
|
+ contents: HashMap<&'var str, &'var str>
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
impl<'var> LSColors<'var> {
|
|
impl<'var> LSColors<'var> {
|
|
@@ -17,23 +19,83 @@ impl<'var> LSColors<'var> {
|
|
|
.take(3)
|
|
.take(3)
|
|
|
.collect::<Vec<_>>();
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
|
|
- if bits.len() == 2 { Some((bits[0], bits[1])) }
|
|
|
|
|
- else { None }
|
|
|
|
|
|
|
+ if bits.len() != 2 || bits[0].is_empty() || bits[1].is_empty() { None }
|
|
|
|
|
+ else { Some( (bits[0], bits[1]) ) }
|
|
|
}).collect();
|
|
}).collect();
|
|
|
LSColors { contents }
|
|
LSColors { contents }
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
fn get(&self, facet_name: &str) -> Option<Style> {
|
|
fn get(&self, facet_name: &str) -> Option<Style> {
|
|
|
- self.contents.get(facet_name).and_then(ansi_to_style)
|
|
|
|
|
|
|
+ self.contents.get(facet_name).map(ansi_to_style)
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+fn ansi_to_style(ansi: &&str) -> Style {
|
|
|
|
|
+ let mut style = Style::default();
|
|
|
|
|
+
|
|
|
|
|
+ for num in ansi.split(";") {
|
|
|
|
|
+ match num {
|
|
|
|
|
+
|
|
|
|
|
+ // Bold and italic
|
|
|
|
|
+ "1" => style = style.bold(),
|
|
|
|
|
+ "4" => style = style.underline(),
|
|
|
|
|
+
|
|
|
|
|
+ // Foreground colours
|
|
|
|
|
+ "30" => style = style.fg(Black),
|
|
|
|
|
+ "31" => style = style.fg(Red),
|
|
|
|
|
+ "32" => style = style.fg(Green),
|
|
|
|
|
+ "33" => style = style.fg(Yellow),
|
|
|
|
|
+ "34" => style = style.fg(Blue),
|
|
|
|
|
+ "35" => style = style.fg(Purple),
|
|
|
|
|
+ "36" => style = style.fg(Cyan),
|
|
|
|
|
+ "37" => style = style.fg(White),
|
|
|
|
|
+
|
|
|
|
|
+ // Background colours
|
|
|
|
|
+ "40" => style = style.on(Black),
|
|
|
|
|
+ "41" => style = style.on(Red),
|
|
|
|
|
+ "42" => style = style.on(Green),
|
|
|
|
|
+ "43" => style = style.on(Yellow),
|
|
|
|
|
+ "44" => style = style.on(Blue),
|
|
|
|
|
+ "45" => style = style.on(Purple),
|
|
|
|
|
+ "46" => style = style.on(Cyan),
|
|
|
|
|
+ "47" => style = style.on(White),
|
|
|
|
|
+ _ => {/* ignore the error and do nothing */},
|
|
|
|
|
+ }
|
|
|
}
|
|
}
|
|
|
|
|
+
|
|
|
|
|
+ style
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
-fn ansi_to_style(ansi: &&str) -> Option<Style> {
|
|
|
|
|
- match *ansi {
|
|
|
|
|
- "31" => Some(Red.normal()),
|
|
|
|
|
- "34" => Some(Blue.normal()),
|
|
|
|
|
- _ => None,
|
|
|
|
|
|
|
+
|
|
|
|
|
+#[cfg(test)]
|
|
|
|
|
+mod ansi_test {
|
|
|
|
|
+ use super::*;
|
|
|
|
|
+ use ansi_term::Style;
|
|
|
|
|
+
|
|
|
|
|
+ macro_rules! test {
|
|
|
|
|
+ ($name:ident: $input:expr => $result:expr) => {
|
|
|
|
|
+ #[test]
|
|
|
|
|
+ fn $name() {
|
|
|
|
|
+ assert_eq!(ansi_to_style(&$input), $result);
|
|
|
|
|
+ }
|
|
|
|
|
+ };
|
|
|
}
|
|
}
|
|
|
|
|
+
|
|
|
|
|
+ // Styles
|
|
|
|
|
+ test!(bold: "1" => Style::default().bold());
|
|
|
|
|
+ test!(under: "4" => Style::default().underline());
|
|
|
|
|
+ test!(both: "1;4" => Style::default().bold().underline());
|
|
|
|
|
+ test!(fg: "31" => Red.normal());
|
|
|
|
|
+ test!(bg: "43" => Style::default().on(Yellow));
|
|
|
|
|
+ test!(bfg: "31;43" => Red.on(Yellow));
|
|
|
|
|
+ test!(all: "43;31;1;4" => Red.on(Yellow).bold().underline());
|
|
|
|
|
+ test!(again: "1;1;1;1;1" => Style::default().bold());
|
|
|
|
|
+
|
|
|
|
|
+ // Failure cases
|
|
|
|
|
+ test!(empty: "" => Style::default());
|
|
|
|
|
+ test!(semis: ";;;;;;" => Style::default());
|
|
|
|
|
+ test!(nines: "99999999" => Style::default());
|
|
|
|
|
+ test!(word: "GREEN" => Style::default());
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@@ -42,59 +104,45 @@ fn ansi_to_style(ansi: &&str) -> Option<Style> {
|
|
|
mod test {
|
|
mod test {
|
|
|
use super::*;
|
|
use super::*;
|
|
|
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_empty() {
|
|
|
|
|
- let lsc = LSColors::parse("");
|
|
|
|
|
- assert_eq!(lsc.get("di"), None);
|
|
|
|
|
- assert_eq!(lsc.get(""), None);
|
|
|
|
|
|
|
+ macro_rules! test {
|
|
|
|
|
+ ($name:ident: $input:expr, $facet:expr => $result:expr) => {
|
|
|
|
|
+ #[test]
|
|
|
|
|
+ fn $name() {
|
|
|
|
|
+ let lsc = LSColors::parse($input);
|
|
|
|
|
+ assert_eq!(lsc.get($facet), $result.into());
|
|
|
|
|
+ assert_eq!(lsc.get(""), None);
|
|
|
|
|
+ }
|
|
|
|
|
+ };
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_gibberish() {
|
|
|
|
|
- let lsc = LSColors::parse("gibberish");
|
|
|
|
|
- assert_eq!(lsc.get("di"), None);
|
|
|
|
|
- assert_eq!(lsc.get("gibberish"), None);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ // Bad parses
|
|
|
|
|
+ test!(empty: "", "di" => None);
|
|
|
|
|
+ test!(jibber: "blah", "di" => None);
|
|
|
|
|
|
|
|
|
|
+ test!(equals: "=", "di" => None);
|
|
|
|
|
+ test!(starts: "=di", "di" => None);
|
|
|
|
|
+ test!(ends: "id=", "id" => None);
|
|
|
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_one() {
|
|
|
|
|
- let lsc = LSColors::parse("di=34");
|
|
|
|
|
- assert_eq!(lsc.get("di"), Some(Blue.normal()));
|
|
|
|
|
- assert_eq!(lsc.get("ln"), None);
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_and_ignore_one() {
|
|
|
|
|
- let lsc = LSColors::parse("di=34=56");
|
|
|
|
|
- assert_eq!(lsc.get("di"), None);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ // Foreground colours
|
|
|
|
|
+ test!(red: "di=31", "di" => Red.normal());
|
|
|
|
|
+ test!(green: "cb=32", "cb" => Green.normal());
|
|
|
|
|
+ test!(blue: "la=34", "la" => Blue.normal());
|
|
|
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_and_ignore_again() {
|
|
|
|
|
- let lsc = LSColors::parse("di=");
|
|
|
|
|
- assert_eq!(lsc.get("di"), None);
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_and_ignore_other() {
|
|
|
|
|
- let lsc = LSColors::parse("=id");
|
|
|
|
|
- assert_eq!(lsc.get("di"), None);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ // Background colours
|
|
|
|
|
+ test!(yellow: "do=43", "do" => Style::default().on(Yellow));
|
|
|
|
|
+ test!(purple: "re=45", "re" => Style::default().on(Purple));
|
|
|
|
|
+ test!(cyan: "mi=46", "mi" => Style::default().on(Cyan));
|
|
|
|
|
|
|
|
|
|
+ // Bold and underline
|
|
|
|
|
+ test!(bold: "fa=1", "fa" => Style::default().bold());
|
|
|
|
|
+ test!(under: "so=4", "so" => Style::default().underline());
|
|
|
|
|
+ test!(both: "la=1;4", "la" => Style::default().bold().underline());
|
|
|
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_and_ignore_equals() {
|
|
|
|
|
- let lsc = LSColors::parse("=");
|
|
|
|
|
- assert_eq!(lsc.get("di"), None);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ // More and many
|
|
|
|
|
+ test!(more_1: "me=43;21;55;34:yu=1;4;1", "me" => Blue.on(Yellow));
|
|
|
|
|
+ test!(more_2: "me=43;21;55;34:yu=1;4;1", "yu" => Style::default().bold().underline());
|
|
|
|
|
|
|
|
-
|
|
|
|
|
- #[test]
|
|
|
|
|
- fn parse_two() {
|
|
|
|
|
- let lsc = LSColors::parse("di=34:ln=31");
|
|
|
|
|
- assert_eq!(lsc.get("di"), Some(Blue.normal()));
|
|
|
|
|
- assert_eq!(lsc.get("ln"), Some(Red.normal()));
|
|
|
|
|
- assert_eq!(lsc.get("cd"), None);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ test!(many_1: "red=31:green=32:blue=34", "red" => Red.normal());
|
|
|
|
|
+ test!(many_2: "red=31:green=32:blue=34", "green" => Green.normal());
|
|
|
|
|
+ test!(many_3: "red=31:green=32:blue=34", "blue" => Blue.normal());
|
|
|
}
|
|
}
|