use nu_ansi_term::Style;
use std::collections::HashMap;
use crate::fs::File;
use crate::info::filetype::FileType;
use crate::options::config::ThemeConfig;
use crate::output::color_scale::ColorScaleOptions;
use crate::output::file_name::Colours as FileNameColours;
use crate::output::render;
mod ui_styles;
pub(crate) use self::ui_styles::FileType as ThemeFileType;
pub use self::ui_styles::UiStyles;
pub(crate) use self::ui_styles::*;
mod lsc;
pub use self::lsc::LSColors;
mod default_theme;
#[derive(PartialEq, Eq, Debug)]
pub struct Options {
pub use_colours: UseColours,
pub colour_scale: ColorScaleOptions,
pub definitions: Definitions,
pub theme_config: Option<ThemeConfig>,
}
#[derive(PartialEq, Eq, Debug, Copy, Clone)]
pub enum UseColours {
Always,
Automatic,
Never,
}
#[derive(PartialEq, Eq, Debug, Default)]
pub struct Definitions {
pub ls: Option<String>,
pub exa: Option<String>,
}
pub struct Theme {
pub ui: UiStyles,
pub exts: Box<dyn FileStyle>,
}
impl Options {
#[must_use]
pub fn to_theme(&self, isatty: bool) -> Theme {
if self.use_colours == UseColours::Never
|| (self.use_colours == UseColours::Automatic && !isatty)
{
let ui = UiStyles::plain();
let exts = Box::new(NoFileStyle);
return Theme { ui, exts };
}
#[cfg(windows)]
if nu_ansi_term::enable_ansi_support().is_err() {
if self.use_colours == UseColours::Always {
eprintln!("eza: Ignoring option color=always in legacy console.");
}
let ui = UiStyles::plain();
let exts = Box::new(NoFileStyle);
return Theme { ui, exts };
}
match self.theme_config {
Some(ref theme) => {
if let Some(mut ui) = theme.to_theme() {
let (exts, use_default_filetypes) = self.definitions.parse_color_vars(&mut ui);
let exts: Box<dyn FileStyle> =
match (exts.is_non_empty(), use_default_filetypes) {
(false, false) => Box::new(NoFileStyle),
(false, true) => Box::new(FileTypes),
(true, false) => Box::new(exts),
(true, true) => Box::new((exts, FileTypes)),
};
return Theme { ui, exts };
}
self.default_theme()
}
None => self.default_theme(),
}
}
fn default_theme(&self) -> Theme {
let mut ui = UiStyles::default_theme(self.colour_scale);
let (exts, use_default_filetypes) = self.definitions.parse_color_vars(&mut ui);
let exts: Box<dyn FileStyle> = match (exts.is_non_empty(), use_default_filetypes) {
(false, false) => Box::new(NoFileStyle),
(false, true) => Box::new(FileTypes),
(true, false) => Box::new(exts),
(true, true) => Box::new((exts, FileTypes)),
};
Theme { ui, exts }
}
}
impl Definitions {
fn parse_color_vars(&self, colours: &mut UiStyles) -> (ExtensionMappings, bool) {
use log::warn;
let mut exts = ExtensionMappings::default();
if let Some(lsc) = &self.ls {
LSColors(lsc).each_pair(|pair| {
if !colours.set_ls(&pair) {
match glob::Pattern::new(pair.key) {
Ok(pat) => {
exts.add(pat, pair.to_style());
}
Err(e) => {
warn!("Couldn't parse glob pattern {:?}: {}", pair.key, e);
}
}
}
});
}
let mut use_default_filetypes = true;
if let Some(exa) = &self.exa {
if exa == "reset" || exa.starts_with("reset:") {
use_default_filetypes = false;
}
LSColors(exa).each_pair(|pair| {
if !colours.set_ls(&pair) && !colours.set_exa(&pair) {
match glob::Pattern::new(pair.key) {
Ok(pat) => {
exts.add(pat, pair.to_style());
}
Err(e) => {
warn!("Couldn't parse glob pattern {:?}: {}", pair.key, e);
}
}
}
});
}
(exts, use_default_filetypes)
}
}
pub trait FileStyle: Sync {
fn get_style(&self, file: &File<'_>, theme: &Theme) -> Option<Style>;
}
#[derive(PartialEq, Debug)]
struct NoFileStyle;
impl FileStyle for NoFileStyle {
fn get_style(&self, _file: &File<'_>, _theme: &Theme) -> Option<Style> {
None
}
}
impl<A, B> FileStyle for (A, B)
where
A: FileStyle,
B: FileStyle,
{
fn get_style(&self, file: &File<'_>, theme: &Theme) -> Option<Style> {
self.0
.get_style(file, theme)
.or_else(|| self.1.get_style(file, theme))
}
}
#[derive(PartialEq, Debug, Default)]
struct ExtensionMappings {
mappings: Vec<GlobPattern>,
}
#[derive(PartialEq, Debug)]
enum GlobPattern {
Complex(glob::Pattern, Style),
Simple(HashMap<String, Style>),
}
impl ExtensionMappings {
fn is_non_empty(&self) -> bool {
!self.mappings.is_empty()
}
fn add(&mut self, pattern: glob::Pattern, style: Style) {
match (self.mappings.last_mut(), is_simple_pattern(pattern)) {
(Some(GlobPattern::Simple(h)), Ok(s)) => {
h.insert(s, style);
}
(_, Ok(s)) => {
self.mappings
.push(GlobPattern::Simple(HashMap::from([(s, style)])));
}
(_, Err(p)) => {
self.mappings.push(GlobPattern::Complex(p, style));
}
}
}
}
fn is_simple_pattern(pattern: glob::Pattern) -> Result<String, glob::Pattern> {
match pattern.as_str().strip_prefix("*.") {
None => Err(pattern),
Some(ext) if ext.contains(['?', '*', '[', ']', '.']) => Err(pattern),
Some(ext) => Ok(ext.to_string()),
}
}
impl FileStyle for ExtensionMappings {
fn get_style(&self, file: &File<'_>, _theme: &Theme) -> Option<Style> {
let maybe_ext = file.name.rsplit_once('.').map(|x| x.1);
for mapping in self.mappings.iter().rev() {
match mapping {
GlobPattern::Complex(pat, style) => {
if pat.matches(&file.name) {
return Some(*style);
}
}
GlobPattern::Simple(map) => {
if let Some(ext) = maybe_ext {
if let Some(style) = map.get(ext) {
return Some(*style);
}
}
}
}
}
None
}
}
#[derive(Debug)]
struct FileTypes;
impl FileStyle for FileTypes {
fn get_style(&self, file: &File<'_>, theme: &Theme) -> Option<Style> {
#[rustfmt::skip]
return match FileType::get_file_type(file) {
Some(FileType::Image) => theme.ui.file_type.unwrap_or_default().image,
Some(FileType::Video) => theme.ui.file_type.unwrap_or_default().video,
Some(FileType::Music) => theme.ui.file_type.unwrap_or_default().music,
Some(FileType::Lossless) => theme.ui.file_type.unwrap_or_default().lossless,
Some(FileType::Crypto) => theme.ui.file_type.unwrap_or_default().crypto,
Some(FileType::Document) => theme.ui.file_type.unwrap_or_default().document,
Some(FileType::Compressed) => theme.ui.file_type.unwrap_or_default().compressed,
Some(FileType::Temp) => theme.ui.file_type.unwrap_or_default().temp,
Some(FileType::Compiled) => theme.ui.file_type.unwrap_or_default().compiled,
Some(FileType::Build) => theme.ui.file_type.unwrap_or_default().build,
Some(FileType::Source) => theme.ui.file_type.unwrap_or_default().source,
None => None
};
}
}
#[cfg(unix)]
impl render::BlocksColours for Theme {
fn blocksize(&self, prefix: Option<number_prefix::Prefix>) -> Style {
use number_prefix::Prefix::{Gibi, Giga, Kibi, Kilo, Mebi, Mega};
#[rustfmt::skip]
let style = match prefix {
Some(Kilo | Kibi) => self.ui.size.unwrap_or_default().number_kilo,
Some(Mega | Mebi) => self.ui.size.unwrap_or_default().number_mega,
Some(Giga | Gibi) => self.ui.size.unwrap_or_default().number_giga,
Some(_) => self.ui.size.unwrap_or_default().number_huge,
None => self.ui.size.unwrap_or_default().number_byte,
};
style.unwrap_or_default()
}
fn unit(&self, prefix: Option<number_prefix::Prefix>) -> Style {
use number_prefix::Prefix::{Gibi, Giga, Kibi, Kilo, Mebi, Mega};
#[rustfmt::skip]
let style = match prefix {
Some(Kilo | Kibi) => self.ui.size.unwrap_or_default().unit_kilo,
Some(Mega | Mebi) => self.ui.size.unwrap_or_default().unit_mega,
Some(Giga | Gibi) => self.ui.size.unwrap_or_default().unit_giga,
Some(_) => self.ui.size.unwrap_or_default().unit_huge,
None => self.ui.size.unwrap_or_default().unit_byte,
};
style.unwrap_or_default()
}
fn no_blocksize(&self) -> Style {
self.ui.punctuation.unwrap_or_default()
}
}
#[rustfmt::skip]
impl render::FiletypeColours for Theme {
fn normal(&self) -> Style { self.ui.filekinds.unwrap_or_default().normal() }
fn directory(&self) -> Style { self.ui.filekinds.unwrap_or_default().directory() }
fn pipe(&self) -> Style { self.ui.filekinds.unwrap_or_default().pipe() }
fn symlink(&self) -> Style { self.ui.filekinds.unwrap_or_default().symlink() }
fn block_device(&self) -> Style { self.ui.filekinds.unwrap_or_default().block_device() }
fn char_device(&self) -> Style { self.ui.filekinds.unwrap_or_default().char_device() }
fn socket(&self) -> Style { self.ui.filekinds.unwrap_or_default().socket() }
fn special(&self) -> Style { self.ui.filekinds.unwrap_or_default().special() }
}
#[rustfmt::skip]
impl render::GitColours for Theme {
fn not_modified(&self) -> Style { self.ui.punctuation() }
#[allow(clippy::new_ret_no_self)]
fn new(&self) -> Style { self.ui.git.unwrap_or_default().new() }
fn modified(&self) -> Style { self.ui.git.unwrap_or_default().modified() }
fn deleted(&self) -> Style { self.ui.git.unwrap_or_default().deleted() }
fn renamed(&self) -> Style { self.ui.git.unwrap_or_default().renamed() }
fn type_change(&self) -> Style { self.ui.git.unwrap_or_default().typechange() }
fn ignored(&self) -> Style { self.ui.git.unwrap_or_default().ignored() }
fn conflicted(&self) -> Style { self.ui.git.unwrap_or_default().conflicted() }
}
#[rustfmt::skip]
impl render::GitRepoColours for Theme {
fn branch_main(&self) -> Style { self.ui.git_repo.unwrap_or_default().branch_main() }
fn branch_other(&self) -> Style { self.ui.git_repo.unwrap_or_default().branch_other() }
fn no_repo(&self) -> Style { self.ui.punctuation() }
fn git_clean(&self) -> Style { self.ui.git_repo.unwrap_or_default().git_clean() }
fn git_dirty(&self) -> Style { self.ui.git_repo.unwrap_or_default().git_dirty() }
}
#[rustfmt::skip]
#[cfg(unix)]
impl render::GroupColours for Theme {
fn yours(&self) -> Style { self.ui.users.unwrap_or_default().group_yours() }
fn not_yours(&self) -> Style { self.ui.users.unwrap_or_default().group_other() }
fn root_group(&self) -> Style { self.ui.users.unwrap_or_default().group_root() }
fn no_group(&self) -> Style { self.ui.punctuation() }
}
#[rustfmt::skip]
impl render::LinksColours for Theme {
fn normal(&self) -> Style { self.ui.links.unwrap_or_default().normal() }
fn multi_link_file(&self) -> Style { self.ui.links.unwrap_or_default().multi_link_file() }
}
#[rustfmt::skip]
impl render::PermissionsColours for Theme {
fn dash(&self) -> Style { self.ui.punctuation() }
fn user_read(&self) -> Style { self.ui.perms.unwrap_or_default().user_read() }
fn user_write(&self) -> Style { self.ui.perms.unwrap_or_default().user_write() }
fn user_execute_file(&self) -> Style { self.ui.perms.unwrap_or_default().user_execute_file() }
fn user_execute_other(&self) -> Style { self.ui.perms.unwrap_or_default().user_execute_other() }
fn group_read(&self) -> Style { self.ui.perms.unwrap_or_default().group_read() }
fn group_write(&self) -> Style { self.ui.perms.unwrap_or_default().group_write() }
fn group_execute(&self) -> Style { self.ui.perms.unwrap_or_default().group_execute() }
fn other_read(&self) -> Style { self.ui.perms.unwrap_or_default().other_read() }
fn other_write(&self) -> Style { self.ui.perms.unwrap_or_default().other_write() }
fn other_execute(&self) -> Style { self.ui.perms.unwrap_or_default().other_execute() }
fn special_user_file(&self) -> Style { self.ui.perms.unwrap_or_default().special_user_file() }
fn special_other(&self) -> Style { self.ui.perms.unwrap_or_default().special_other() }
fn attribute(&self) -> Style { self.ui.perms.unwrap_or_default().attribute() }
}
impl render::SizeColours for Theme {
fn size(&self, prefix: Option<number_prefix::Prefix>) -> Style {
use number_prefix::Prefix::{Gibi, Giga, Kibi, Kilo, Mebi, Mega};
#[rustfmt::skip]
return match prefix {
Some(Kilo | Kibi) => self.ui.size.unwrap_or_default().number_kilo(),
Some(Mega | Mebi) => self.ui.size.unwrap_or_default().number_mega(),
Some(Giga | Gibi) => self.ui.size.unwrap_or_default().number_giga(),
Some(_) => self.ui.size.unwrap_or_default().number_huge(),
None => self.ui.size.unwrap_or_default().number_byte(),
};
}
fn unit(&self, prefix: Option<number_prefix::Prefix>) -> Style {
use number_prefix::Prefix::{Gibi, Giga, Kibi, Kilo, Mebi, Mega};
#[rustfmt::skip]
return match prefix {
Some(Kilo | Kibi) => self.ui.size.unwrap_or_default().unit_kilo(),
Some(Mega | Mebi) => self.ui.size.unwrap_or_default().unit_mega(),
Some(Giga | Gibi) => self.ui.size.unwrap_or_default().unit_giga(),
Some(_) => self.ui.size.unwrap_or_default().unit_huge(),
None => self.ui.size.unwrap_or_default().unit_byte(),
};
}
#[rustfmt::skip]
fn no_size(&self) -> Style { self.ui.punctuation() }
#[rustfmt::skip]
fn major(&self) -> Style { self.ui.size.unwrap_or_default().major() }
#[rustfmt::skip]
fn comma(&self) -> Style { self.ui.punctuation() }
#[rustfmt::skip]
fn minor(&self) -> Style { self.ui.size.unwrap_or_default().minor() }
}
#[rustfmt::skip]
#[cfg(unix)]
impl render::UserColours for Theme {
fn you(&self) -> Style { self.ui.users.unwrap_or_default().user_you() }
fn other(&self) -> Style { self.ui.users.unwrap_or_default().user_other() }
fn root(&self) -> Style { self.ui.users.unwrap_or_default().user_root() }
fn no_user(&self) -> Style { self.ui.punctuation() }
}
#[rustfmt::skip]
impl FileNameColours for Theme {
fn symlink_path(&self) -> Style { self.ui.symlink_path() }
fn normal_arrow(&self) -> Style { self.ui.punctuation() }
fn broken_symlink(&self) -> Style { self.ui.broken_symlink() }
fn broken_filename(&self) -> Style { apply_overlay(self.ui.broken_symlink(), self.ui.broken_path_overlay()) }
fn control_char(&self) -> Style { self.ui.control_char() }
fn broken_control_char(&self) -> Style { apply_overlay(self.ui.control_char(), self.ui.broken_path_overlay()) }
fn executable_file(&self) -> Style { self.ui.filekinds.unwrap_or_default().executable() }
fn mount_point(&self) -> Style { self.ui.filekinds.unwrap_or_default().mount_point() }
fn colour_file(&self, file: &File<'_>) -> Style {
self.exts
.get_style(file, self)
.unwrap_or(self.ui.filekinds.unwrap_or_default().normal())
}
fn style_override(&self, file: &File<'_>) -> Option<FileNameStyle> {
if let Some(ref name_overrides) = self.ui.filenames {
if let Some(file_override) = name_overrides.get(&file.name) {
return Some(*file_override);
}
}
if let Some(ref ext_overrides) = self.ui.extensions {
if let Some(ext) = file.ext.clone() {
if let Some(file_override) = ext_overrides.get(&ext) {
return Some(*file_override);
}
}
}
None
}
}
#[rustfmt::skip]
impl render::SecurityCtxColours for Theme {
fn none(&self) -> Style { self.ui.security_context.unwrap_or_default().none() }
fn selinux_colon(&self) -> Style { self.ui.security_context.unwrap_or_default().selinux().colon() }
fn selinux_user(&self) -> Style { self.ui.security_context.unwrap_or_default().selinux().user() }
fn selinux_role(&self) -> Style { self.ui.security_context.unwrap_or_default().selinux().role() }
fn selinux_type(&self) -> Style { self.ui.security_context.unwrap_or_default().selinux().typ() }
fn selinux_range(&self) -> Style { self.ui.security_context.unwrap_or_default().selinux().range() }
}
#[rustfmt::skip]
fn apply_overlay(mut base: Style, overlay: Style) -> Style {
if let Some(fg) = overlay.foreground { base.foreground = Some(fg); }
if let Some(bg) = overlay.background { base.background = Some(bg); }
if overlay.is_bold { base.is_bold = true; }
if overlay.is_dimmed { base.is_dimmed = true; }
if overlay.is_italic { base.is_italic = true; }
if overlay.is_underline { base.is_underline = true; }
if overlay.is_blink { base.is_blink = true; }
if overlay.is_reverse { base.is_reverse = true; }
if overlay.is_hidden { base.is_hidden = true; }
if overlay.is_strikethrough { base.is_strikethrough = true; }
base
}
#[cfg(test)]
#[cfg(unix)]
mod customs_test {
use super::*;
use crate::theme::ui_styles::UiStyles;
use nu_ansi_term::Color::*;
impl ExtensionMappings {
fn to_vec_pat_style(&self) -> Vec<(glob::Pattern, Style)> {
let mut out = Vec::new();
for map in &self.mappings {
match map {
GlobPattern::Complex(p, s) => {
out.push((p.clone(), *s));
}
GlobPattern::Simple(h) => {
let mut simple_pats = h
.iter()
.map(|(k, v)| (glob::Pattern::new(&format!("*.{k}")).unwrap(), *v))
.collect::<Vec<(glob::Pattern, Style)>>();
simple_pats.sort_by_key(|x| x.0.clone());
out.extend(simple_pats);
}
}
}
out
}
}
macro_rules! test {
($name:ident: ls $ls:expr, exa $exa:expr => colours $expected:ident -> $process_expected:expr) => {
#[allow(non_snake_case)]
#[test]
fn $name() {
let mut $expected = UiStyles::default();
$process_expected();
let definitions = Definitions {
ls: Some($ls.into()),
exa: Some($exa.into()),
};
let mut result = UiStyles::default();
let (_, _) = definitions.parse_color_vars(&mut result);
assert_eq!($expected, result);
}
};
($name:ident: ls $ls:expr, exa $exa:expr => exts $mappings:expr) => {
#[test]
fn $name() {
let mappings: Vec<(glob::Pattern, Style)> = $mappings
.iter()
.map(|t| (glob::Pattern::new(t.0).unwrap(), t.1))
.collect();
let definitions = Definitions {
ls: Some($ls.into()),
exa: Some($exa.into()),
};
let (result, _) = definitions.parse_color_vars(&mut UiStyles::default());
assert_eq!(mappings, result.to_vec_pat_style());
}
};
($name:ident: ls $ls:expr, exa $exa:expr => colours $expected:ident -> $process_expected:expr, exts $mappings:expr) => {
#[test]
fn $name() {
let mut $expected = UiStyles::default();
$process_expected();
let mappings: Vec<(glob::Pattern, Style)> = $mappings
.iter()
.map(|t| (glob::Pattern::new(t.0).unwrap(), t.1))
.collect();
let definitions = Definitions {
ls: Some($ls.into()),
exa: Some($exa.into()),
};
let mut result = UiStyles::default();
let (exts, _) = definitions.parse_color_vars(&mut result);
assert_eq!(mappings, exts.to_vec_pat_style());
assert_eq!($expected, result);
}
};
}
test!(ls_di: ls "di=31", exa "" => colours c -> { c.filekinds().directory = Some(Red.normal()); });
test!(ls_ex: ls "ex=32", exa "" => colours c -> { c.filekinds().executable = Some(Green.normal()); });
test!(ls_fi: ls "fi=33", exa "" => colours c -> { c.filekinds().normal = Some(Yellow.normal()); });
test!(ls_pi: ls "pi=34", exa "" => colours c -> { c.filekinds().pipe = Some(Blue.normal()); });
test!(ls_so: ls "so=35", exa "" => colours c -> { c.filekinds().socket = Some(Purple.normal()); });
test!(ls_bd: ls "bd=36", exa "" => colours c -> { c.filekinds().block_device = Some(Cyan.normal()); });
test!(ls_cd: ls "cd=35", exa "" => colours c -> { c.filekinds().char_device = Some(Purple.normal()); });
test!(ls_ln: ls "ln=34", exa "" => colours c -> { c.filekinds().symlink = Some(Blue.normal()); });
test!(ls_or: ls "or=33", exa "" => colours c -> { c.broken_symlink = Some(Yellow.normal()); });
test!(exa_di: ls "", exa "di=32" => colours c -> { c.filekinds().directory = Some(Green.normal()); });
test!(exa_ex: ls "", exa "ex=33" => colours c -> { c.filekinds().executable = Some(Yellow.normal()); });
test!(exa_fi: ls "", exa "fi=34" => colours c -> { c.filekinds().normal = Some(Blue.normal()); });
test!(exa_pi: ls "", exa "pi=35" => colours c -> { c.filekinds().pipe = Some(Purple.normal()); });
test!(exa_so: ls "", exa "so=36" => colours c -> { c.filekinds().socket = Some(Cyan.normal()); });
test!(exa_bd: ls "", exa "bd=35" => colours c -> { c.filekinds().block_device = Some(Purple.normal()); });
test!(exa_cd: ls "", exa "cd=34" => colours c -> { c.filekinds().char_device = Some(Blue.normal()); });
test!(exa_ln: ls "", exa "ln=33" => colours c -> { c.filekinds().symlink = Some(Yellow.normal()); });
test!(exa_or: ls "", exa "or=32" => colours c -> { c.broken_symlink = Some(Green.normal()); });
test!(ls_exa_di: ls "di=31", exa "di=32" => colours c -> { c.filekinds().directory = Some(Green.normal()); });
test!(ls_exa_ex: ls "ex=32", exa "ex=33" => colours c -> { c.filekinds().executable = Some(Yellow.normal()); });
test!(ls_exa_fi: ls "fi=33", exa "fi=34" => colours c -> { c.filekinds().normal = Some(Blue.normal()); });
test!(exa_ur: ls "", exa "ur=38;5;100" => colours c -> { c.perms().user_read = Some(Fixed(100).normal()); });
test!(exa_uw: ls "", exa "uw=38;5;101" => colours c -> { c.perms().user_write = Some(Fixed(101).normal()); });
test!(exa_ux: ls "", exa "ux=38;5;102" => colours c -> { c.perms().user_execute_file = Some(Fixed(102).normal()); });
test!(exa_ue: ls "", exa "ue=38;5;103" => colours c -> { c.perms().user_execute_other = Some(Fixed(103).normal()); });
test!(exa_gr: ls "", exa "gr=38;5;104" => colours c -> { c.perms().group_read = Some(Fixed(104).normal()); });
test!(exa_gw: ls "", exa "gw=38;5;105" => colours c -> { c.perms().group_write = Some(Fixed(105).normal()); });
test!(exa_gx: ls "", exa "gx=38;5;106" => colours c -> { c.perms().group_execute = Some(Fixed(106).normal()); });
test!(exa_tr: ls "", exa "tr=38;5;107" => colours c -> { c.perms().other_read = Some(Fixed(107).normal()); });
test!(exa_tw: ls "", exa "tw=38;5;108" => colours c -> { c.perms().other_write = Some(Fixed(108).normal()); });
test!(exa_tx: ls "", exa "tx=38;5;109" => colours c -> { c.perms().other_execute = Some(Fixed(109).normal()); });
test!(exa_su: ls "", exa "su=38;5;110" => colours c -> { c.perms().special_user_file = Some(Fixed(110).normal()); });
test!(exa_sf: ls "", exa "sf=38;5;111" => colours c -> { c.perms().special_other = Some(Fixed(111).normal()); });
test!(exa_xa: ls "", exa "xa=38;5;112" => colours c -> { c.perms().attribute = Some(Fixed(112).normal()); });
test!(exa_sn: ls "", exa "sn=38;5;113" => colours c -> {
c.size().number_byte = Some(Fixed(113).normal());
c.size().number_kilo = Some(Fixed(113).normal());
c.size().number_mega = Some(Fixed(113).normal());
c.size().number_giga = Some(Fixed(113).normal());
c.size().number_huge = Some(Fixed(113).normal());
});
test!(exa_sb: ls "", exa "sb=38;5;114" => colours c -> {
c.size().unit_byte = Some(Fixed(114).normal());
c.size().unit_kilo = Some(Fixed(114).normal());
c.size().unit_mega = Some(Fixed(114).normal());
c.size().unit_giga = Some(Fixed(114).normal());
c.size().unit_huge = Some(Fixed(114).normal());
});
test!(exa_nb: ls "", exa "nb=38;5;115" => colours c -> { c.size().number_byte = Some(Fixed(115).normal()); });
test!(exa_nk: ls "", exa "nk=38;5;116" => colours c -> { c.size().number_kilo = Some(Fixed(116).normal()); });
test!(exa_nm: ls "", exa "nm=38;5;117" => colours c -> { c.size().number_mega = Some(Fixed(117).normal()); });
test!(exa_ng: ls "", exa "ng=38;5;118" => colours c -> { c.size().number_giga = Some(Fixed(118).normal()); });
test!(exa_nt: ls "", exa "nt=38;5;119" => colours c -> { c.size().number_huge = Some(Fixed(119).normal()); });
test!(exa_ub: ls "", exa "ub=38;5;115" => colours c -> { c.size().unit_byte = Some(Fixed(115).normal()); });
test!(exa_uk: ls "", exa "uk=38;5;116" => colours c -> { c.size().unit_kilo = Some(Fixed(116).normal()); });
test!(exa_um: ls "", exa "um=38;5;117" => colours c -> { c.size().unit_mega = Some(Fixed(117).normal()); });
test!(exa_ug: ls "", exa "ug=38;5;118" => colours c -> { c.size().unit_giga = Some(Fixed(118).normal()); });
test!(exa_ut: ls "", exa "ut=38;5;119" => colours c -> { c.size().unit_huge = Some(Fixed(119).normal()); });
test!(exa_df: ls "", exa "df=38;5;115" => colours c -> { c.size().major = Some(Fixed(115).normal()); });
test!(exa_ds: ls "", exa "ds=38;5;116" => colours c -> { c.size().minor = Some(Fixed(116).normal()); });
test!(exa_uu: ls "", exa "uu=38;5;117" => colours c -> { c.users().user_you = Some(Fixed(117).normal()); });
test!(exa_un: ls "", exa "un=38;5;118" => colours c -> { c.users().user_other = Some(Fixed(118).normal()); });
test!(exa_gu: ls "", exa "gu=38;5;119" => colours c -> { c.users().group_yours = Some(Fixed(119).normal()); });
test!(exa_gn: ls "", exa "gn=38;5;120" => colours c -> { c.users().group_other = Some(Fixed(120).normal()); });
test!(exa_lc: ls "", exa "lc=38;5;121" => colours c -> { c.links().normal = Some(Fixed(121).normal()); });
test!(exa_lm: ls "", exa "lm=38;5;122" => colours c -> { c.links().multi_link_file = Some(Fixed(122).normal()); });
test!(exa_ga: ls "", exa "ga=38;5;123" => colours c -> { c.git().new = Some(Fixed(123).normal()); });
test!(exa_gm: ls "", exa "gm=38;5;124" => colours c -> { c.git().modified = Some(Fixed(124).normal()); });
test!(exa_gd: ls "", exa "gd=38;5;125" => colours c -> { c.git().deleted = Some(Fixed(125).normal()); });
test!(exa_gv: ls "", exa "gv=38;5;126" => colours c -> { c.git().renamed = Some(Fixed(126).normal()); });
test!(exa_gt: ls "", exa "gt=38;5;127" => colours c -> { c.git().typechange = Some(Fixed(127).normal()); });
test!(exa_gi: ls "", exa "gi=38;5;128" => colours c -> { c.git().ignored = Some(Fixed(128).normal()); });
test!(exa_gc: ls "", exa "gc=38;5;129" => colours c -> { c.git().conflicted = Some(Fixed(129).normal()); });
test!(exa_xx: ls "", exa "xx=38;5;128" => colours c -> { c.punctuation = Some(Fixed(128).normal()); });
test!(exa_da: ls "", exa "da=38;5;129" => colours c -> { c.date = Some(Fixed(129).normal()); });
test!(exa_in: ls "", exa "in=38;5;130" => colours c -> { c.inode = Some(Fixed(130).normal()); });
test!(exa_bl: ls "", exa "bl=38;5;131" => colours c -> { c.blocks = Some(Fixed(131).normal()); });
test!(exa_hd: ls "", exa "hd=38;5;132" => colours c -> { c.header = Some(Fixed(132).normal()); });
test!(exa_lp: ls "", exa "lp=38;5;133" => colours c -> { c.symlink_path = Some(Fixed(133).normal()); });
test!(exa_cc: ls "", exa "cc=38;5;134" => colours c -> { c.control_char = Some(Fixed(134).normal()); });
test!(exa_oc: ls "", exa "oc=38;5;135" => colours c -> { c.octal = Some(Fixed(135).normal()); });
test!(exa_ff: ls "", exa "ff=38;5;136" => colours c -> { c.flags = Some(Fixed(136).normal()); });
test!(exa_bo: ls "", exa "bO=4" => colours c -> { c.broken_path_overlay = Some(Style::default().underline()); });
test!(exa_mp: ls "", exa "mp=1;34;4" => colours c -> { c.filekinds().mount_point = Some(Blue.bold().underline()); });
test!(exa_sp: ls "", exa "sp=1;35;4" => colours c -> { c.filekinds().special = Some(Purple.bold().underline()); });
test!(exa_im: ls "", exa "im=38;5;128" => colours c -> { c.file_type().image = Some(Fixed(128).normal()); });
test!(exa_vi: ls "", exa "vi=38;5;129" => colours c -> { c.file_type().video = Some(Fixed(129).normal()); });
test!(exa_mu: ls "", exa "mu=38;5;130" => colours c -> { c.file_type().music = Some(Fixed(130).normal()); });
test!(exa_lo: ls "", exa "lo=38;5;131" => colours c -> { c.file_type().lossless = Some(Fixed(131).normal()); });
test!(exa_cr: ls "", exa "cr=38;5;132" => colours c -> { c.file_type().crypto = Some(Fixed(132).normal()); });
test!(exa_do: ls "", exa "do=38;5;133" => colours c -> { c.file_type().document = Some(Fixed(133).normal()); });
test!(exa_co: ls "", exa "co=38;5;134" => colours c -> { c.file_type().compressed = Some(Fixed(134).normal()); });
test!(exa_tm: ls "", exa "tm=38;5;135" => colours c -> { c.file_type().temp = Some(Fixed(135).normal()); });
test!(exa_cm: ls "", exa "cm=38;5;136" => colours c -> { c.file_type().compiled = Some(Fixed(136).normal()); });
test!(exa_ie: ls "", exa "bu=38;5;137" => colours c -> { c.file_type().build = Some(Fixed(137).normal()); });
test!(exa_bu: ls "", exa "bu=38;5;137" => colours c -> { c.file_type().build = Some(Fixed(137).normal()); });
test!(exa_sc: ls "", exa "sc=38;5;138" => colours c -> { c.file_type().source = Some(Fixed(138).normal()); });
test!(exa_Sn: ls "", exa "Sn=38;5;128" => colours c -> { c.security_context().none = Some(Fixed(128).normal()); });
test!(exa_Su: ls "", exa "Su=38;5;129" => colours c -> { c.security_context().selinux().user = Some(Fixed(129).normal()); });
test!(exa_Sr: ls "", exa "Sr=38;5;130" => colours c -> { c.security_context().selinux().role = Some(Fixed(130).normal()); });
test!(exa_St: ls "", exa "St=38;5;131" => colours c -> { c.security_context().selinux().typ = Some(Fixed(131).normal()); });
test!(exa_Sl: ls "", exa "Sl=38;5;132" => colours c -> { c.security_context().selinux().range = Some(Fixed(132).normal()); });
test!(ls_uu: ls "uu=38;5;117", exa "" => exts [ ("uu", Fixed(117).normal()) ]);
test!(ls_un: ls "un=38;5;118", exa "" => exts [ ("un", Fixed(118).normal()) ]);
test!(ls_gu: ls "gu=38;5;119", exa "" => exts [ ("gu", Fixed(119).normal()) ]);
test!(ls_gn: ls "gn=38;5;120", exa "" => exts [ ("gn", Fixed(120).normal()) ]);
test!(ls_txt: ls "*.txt=31", exa "" => exts [ ("*.txt", Red.normal()) ]);
test!(ls_mp3: ls "*.mp3=38;5;135", exa "" => exts [ ("*.mp3", Fixed(135).normal()) ]);
test!(ls_mak: ls "Makefile=1;32;4", exa "" => exts [ ("Makefile", Green.bold().underline()) ]);
test!(exa_txt: ls "", exa "*.zip=31" => exts [ ("*.zip", Red.normal()) ]);
test!(exa_mp3: ls "", exa "lev.*=38;5;153" => exts [ ("lev.*", Fixed(153).normal()) ]);
test!(exa_mak: ls "", exa "Cargo.toml=4;32;1" => exts [ ("Cargo.toml", Green.bold().underline()) ]);
test!(ls_multi: ls "*.txt=31:*.rtf=32", exa "" => exts [ ("*.rtf", Green.normal()), ("*.txt", Red.normal()) ]);
test!(exa_multi: ls "", exa "*.tmp=37:*.log=37" => exts [ ("*.log", White.normal()), ("*.tmp", White.normal()) ]);
test!(ls_exa_multi: ls "*.txt=31", exa "*.rtf=32" => exts [ ("*.rtf", Green.normal()), ("*.txt", Red.normal())]);
test!(ls_five: ls "1*1=31:2*2=32:3*3=1;33:4*4=34;1:5*5=35;4", exa "" => exts [
("1*1", Red.normal()), ("2*2", Green.normal()), ("3*3", Yellow.bold()), ("4*4", Blue.bold()), ("5*5", Purple.underline())
]);
test!(ls_overwrite: ls "pi=31:pi=32:pi=33", exa "" => colours c -> { c.filekinds().pipe = Some(Yellow.normal()); });
test!(exa_overwrite: ls "", exa "da=36:da=35:da=34" => colours c -> { c.date = Some(Blue.normal()); });
test!(ls_fi_ls_txt: ls "fi=33:*.txt=31", exa "" => colours c -> { c.filekinds().normal = Some(Yellow.normal()); }, exts [ ("*.txt", Red.normal()) ]);
test!(ls_fi_exa_txt: ls "fi=33", exa "*.txt=31" => colours c -> { c.filekinds().normal = Some(Yellow.normal()); }, exts [ ("*.txt", Red.normal()) ]);
test!(ls_txt_exa_fi: ls "*.txt=31", exa "fi=33" => colours c -> { c.filekinds().normal = Some(Yellow.normal()); }, exts [ ("*.txt", Red.normal()) ]);
test!(eza_fi_exa_txt: ls "", exa "fi=33:*.txt=31" => colours c -> { c.filekinds().normal = Some(Yellow.normal()); }, exts [ ("*.txt", Red.normal()) ]);
}