exa/src/output/render/size.rs

183 lines
5.5 KiB
Rust
Raw Normal View History

use ansi_term::Style;
use locale::Numeric as NumericLocale;
use number_prefix::Prefix;
2018-12-07 23:43:31 +00:00
use crate::fs::fields as f;
use crate::output::cell::{TextCell, DisplayWidth};
use crate::output::table::SizeFormat;
2017-05-20 20:43:04 +00:00
impl f::Size {
pub fn render<C: Colours>(self, colours: &C, size_format: SizeFormat, numerics: &NumericLocale) -> TextCell {
use number_prefix::NumberPrefix;
2017-05-20 20:43:04 +00:00
let size = match self {
Self::Some(s) => s,
Self::None => return TextCell::blank(colours.no_size()),
Self::DeviceIDs(ref ids) => return ids.render(colours),
2017-05-20 20:43:04 +00:00
};
let result = match size_format {
2019-04-17 05:54:06 +00:00
SizeFormat::DecimalBytes => NumberPrefix::decimal(size as f64),
SizeFormat::BinaryBytes => NumberPrefix::binary(size as f64),
2017-05-20 20:43:04 +00:00
SizeFormat::JustBytes => {
// Use the binary prefix to select a style.
let prefix = match NumberPrefix::binary(size as f64) {
NumberPrefix::Standalone(_) => None,
NumberPrefix::Prefixed(p, _) => Some(p),
};
// But format the number directly using the locale.
2017-05-20 20:43:04 +00:00
let string = numerics.format_int(size);
return TextCell::paint(colours.size(prefix), string);
}
2017-05-20 20:43:04 +00:00
};
let (prefix, n) = match result {
NumberPrefix::Standalone(b) => return TextCell::paint(colours.size(None), b.to_string()),
NumberPrefix::Prefixed(p, n) => (p, n),
2017-05-20 20:43:04 +00:00
};
let symbol = prefix.symbol();
2020-10-10 14:57:40 +00:00
let number = if n < 10_f64 { numerics.format_float(n, 1) }
else { numerics.format_int(n as isize) };
2017-05-20 20:43:04 +00:00
// The numbers and symbols are guaranteed to be written in ASCII, so
// we can skip the display width calculation.
let width = DisplayWidth::from(number.len() + symbol.len());
TextCell {
2018-06-19 12:58:03 +00:00
width,
2017-05-20 20:43:04 +00:00
contents: vec![
colours.size(Some(prefix)).paint(number),
colours.unit(Some(prefix)).paint(symbol),
2017-05-20 20:43:04 +00:00
].into(),
}
}
}
impl f::DeviceIDs {
fn render<C: Colours>(self, colours: &C) -> TextCell {
let major = self.major.to_string();
let minor = self.minor.to_string();
TextCell {
width: DisplayWidth::from(major.len() + 1 + minor.len()),
contents: vec![
colours.major().paint(major),
colours.comma().paint(","),
colours.minor().paint(minor),
].into(),
}
2017-05-20 20:43:04 +00:00
}
}
2017-05-21 08:49:17 +00:00
pub trait Colours {
fn size(&self, prefix: Option<Prefix>) -> Style;
fn unit(&self, prefix: Option<Prefix>) -> Style;
2017-08-26 22:53:47 +00:00
fn no_size(&self) -> Style;
2017-08-26 22:53:47 +00:00
fn major(&self) -> Style;
fn comma(&self) -> Style;
fn minor(&self) -> Style;
}
2017-05-21 08:49:17 +00:00
#[cfg(test)]
pub mod test {
2017-08-26 22:53:47 +00:00
use super::Colours;
2018-12-07 23:43:31 +00:00
use crate::output::cell::{TextCell, DisplayWidth};
use crate::output::table::SizeFormat;
use crate::fs::fields as f;
2017-05-21 08:49:17 +00:00
use locale::Numeric as NumericLocale;
2017-05-21 08:49:17 +00:00
use ansi_term::Colour::*;
use ansi_term::Style;
use number_prefix::Prefix;
2017-08-26 22:53:47 +00:00
struct TestColours;
2017-08-26 22:53:47 +00:00
impl Colours for TestColours {
fn size(&self, _prefix: Option<Prefix>) -> Style { Fixed(66).normal() }
fn unit(&self, _prefix: Option<Prefix>) -> Style { Fixed(77).bold() }
2017-08-26 22:53:47 +00:00
fn no_size(&self) -> Style { Black.italic() }
fn major(&self) -> Style { Blue.on(Red) }
fn comma(&self) -> Style { Green.italic() }
fn minor(&self) -> Style { Cyan.on(Yellow) }
}
2017-05-21 08:49:17 +00:00
#[test]
fn directory() {
let directory = f::Size::None;
let expected = TextCell::blank(Black.italic());
assert_eq!(expected, directory.render(&TestColours, SizeFormat::JustBytes, &NumericLocale::english()))
2017-05-21 08:49:17 +00:00
}
#[test]
fn file_decimal() {
let directory = f::Size::Some(2_100_000);
let expected = TextCell {
width: DisplayWidth::from(4),
contents: vec![
Fixed(66).paint("2.1"),
Fixed(77).bold().paint("M"),
2017-05-21 08:49:17 +00:00
].into(),
};
assert_eq!(expected, directory.render(&TestColours, SizeFormat::DecimalBytes, &NumericLocale::english()))
2017-05-21 08:49:17 +00:00
}
#[test]
fn file_binary() {
let directory = f::Size::Some(1_048_576);
let expected = TextCell {
width: DisplayWidth::from(5),
contents: vec![
Fixed(66).paint("1.0"),
Fixed(77).bold().paint("Mi"),
2017-05-21 08:49:17 +00:00
].into(),
};
assert_eq!(expected, directory.render(&TestColours, SizeFormat::BinaryBytes, &NumericLocale::english()))
2017-05-21 08:49:17 +00:00
}
#[test]
fn file_bytes() {
let directory = f::Size::Some(1048576);
let expected = TextCell {
width: DisplayWidth::from(9),
contents: vec![
Fixed(66).paint("1,048,576"),
2017-05-21 08:49:17 +00:00
].into(),
};
assert_eq!(expected, directory.render(&TestColours, SizeFormat::JustBytes, &NumericLocale::english()))
2017-05-21 08:49:17 +00:00
}
#[test]
fn device_ids() {
let directory = f::Size::DeviceIDs(f::DeviceIDs { major: 10, minor: 80 });
let expected = TextCell {
width: DisplayWidth::from(5),
contents: vec![
Blue.on(Red).paint("10"),
Green.italic().paint(","),
Cyan.on(Yellow).paint("80"),
].into(),
};
assert_eq!(expected, directory.render(&TestColours, SizeFormat::JustBytes, &NumericLocale::english()))
2017-05-21 08:49:17 +00:00
}
}