clap/src/fmt.rs

184 lines
4.8 KiB
Rust
Raw Normal View History

2016-09-05 19:29:40 +00:00
#[cfg(all(feature = "color", not(target_os = "windows")))]
use ansi_term::ANSIString;
#[cfg(all(feature = "color", not(target_os = "windows")))]
2015-10-28 14:23:59 +00:00
use ansi_term::Colour::{Green, Red, Yellow};
#[cfg(feature = "color")]
use atty;
2016-09-05 21:03:45 +00:00
use std::fmt;
use std::env;
#[doc(hidden)]
#[derive(Debug, Copy, Clone, PartialEq)]
pub enum ColorWhen {
Auto,
Always,
2016-09-05 19:29:40 +00:00
Never,
}
#[cfg(feature = "color")]
pub fn is_a_tty(stderr: bool) -> bool {
debugln!("is_a_tty: stderr={:?}", stderr);
let stream = if stderr {
atty::Stream::Stderr
} else {
atty::Stream::Stdout
};
atty::is(stream)
}
#[cfg(not(feature = "color"))]
pub fn is_a_tty(_: bool) -> bool {
debugln!("is_a_tty;");
false
}
2017-05-27 16:10:42 +00:00
pub fn is_term_dumb() -> bool { env::var("TERM").ok() == Some(String::from("dumb")) }
#[doc(hidden)]
pub struct ColorizerOption {
pub use_stderr: bool,
pub when: ColorWhen,
}
#[doc(hidden)]
pub struct Colorizer {
use_stderr: bool,
when: ColorWhen,
}
macro_rules! color {
($_self:ident, $c:ident, $m:expr) => {
match $_self.when {
ColorWhen::Auto => Format::$c($m),
ColorWhen::Always => Format::$c($m),
ColorWhen::Never => Format::None($m),
}
};
}
impl Colorizer {
2017-05-27 16:10:42 +00:00
pub fn new(option: ColorizerOption) -> Colorizer {
let is_a_tty = is_a_tty(option.use_stderr);
let is_term_dumb = is_term_dumb();
Colorizer {
2017-05-27 16:10:42 +00:00
use_stderr: option.use_stderr,
when: if is_a_tty && !is_term_dumb {
option.when
} else {
ColorWhen::Never
2017-05-27 16:10:42 +00:00
},
}
}
2016-09-05 19:29:40 +00:00
pub fn good<T>(&self, msg: T) -> Format<T>
where T: fmt::Display + AsRef<str>
{
debugln!("Colorizer::good;");
color!(self, Good, msg)
}
2016-09-05 19:29:40 +00:00
pub fn warning<T>(&self, msg: T) -> Format<T>
where T: fmt::Display + AsRef<str>
{
debugln!("Colorizer::warning;");
color!(self, Warning, msg)
}
2016-09-05 19:29:40 +00:00
pub fn error<T>(&self, msg: T) -> Format<T>
where T: fmt::Display + AsRef<str>
{
debugln!("Colorizer::error;");
color!(self, Error, msg)
}
2016-09-05 19:29:40 +00:00
pub fn none<T>(&self, msg: T) -> Format<T>
where T: fmt::Display + AsRef<str>
{
debugln!("Colorizer::none;");
Format::None(msg)
}
}
impl Default for Colorizer {
fn default() -> Self {
2017-05-27 16:10:42 +00:00
Colorizer::new(ColorizerOption {
use_stderr: true,
when: ColorWhen::Auto,
})
}
}
/// Defines styles for different types of error messages. Defaults to Error=Red, Warning=Yellow,
/// and Good=Green
#[derive(Debug)]
2016-05-02 16:48:47 +00:00
#[doc(hidden)]
pub enum Format<T> {
/// Defines the style used for errors, defaults to Red
2015-09-07 01:07:46 +00:00
Error(T),
/// Defines the style used for warnings, defaults to Yellow
2015-09-07 01:07:46 +00:00
Warning(T),
/// Defines the style used for good values, defaults to Green
2015-09-07 01:07:46 +00:00
Good(T),
/// Defines no formatting style
None(T),
}
#[cfg(all(feature = "color", not(target_os = "windows")))]
impl<T: AsRef<str>> Format<T> {
fn format(&self) -> ANSIString {
match *self {
Format::Error(ref e) => Red.bold().paint(e.as_ref()),
Format::Warning(ref e) => Yellow.paint(e.as_ref()),
Format::Good(ref e) => Green.paint(e.as_ref()),
Format::None(ref e) => ANSIString::from(e.as_ref()),
}
}
}
#[cfg(any(not(feature = "color"), target_os = "windows"))]
#[cfg_attr(feature="lints", allow(match_same_arms))]
impl<T: fmt::Display> Format<T> {
fn format(&self) -> &T {
match *self {
Format::Error(ref e) => e,
Format::Warning(ref e) => e,
Format::Good(ref e) => e,
Format::None(ref e) => e,
}
}
}
#[cfg(all(feature = "color", not(target_os = "windows")))]
impl<T: AsRef<str>> fmt::Display for Format<T> {
2016-11-20 19:41:15 +00:00
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{}", &self.format()) }
}
#[cfg(any(not(feature = "color"), target_os = "windows"))]
impl<T: fmt::Display> fmt::Display for Format<T> {
2016-11-20 19:41:15 +00:00
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{}", &self.format()) }
}
2015-09-04 15:38:48 +00:00
#[cfg(all(test, feature = "color", not(target_os = "windows")))]
2015-09-04 15:38:48 +00:00
mod test {
use ansi_term::ANSIString;
2016-09-05 19:29:40 +00:00
use ansi_term::Colour::{Green, Red, Yellow};
use super::Format;
2015-09-04 15:38:48 +00:00
#[test]
fn colored_output() {
let err = Format::Error("error");
2015-10-28 14:23:59 +00:00
assert_eq!(&*format!("{}", err),
&*format!("{}", Red.bold().paint("error")));
2015-09-04 15:38:48 +00:00
let good = Format::Good("good");
assert_eq!(&*format!("{}", good), &*format!("{}", Green.paint("good")));
let warn = Format::Warning("warn");
assert_eq!(&*format!("{}", warn), &*format!("{}", Yellow.paint("warn")));
let none = Format::None("none");
2016-09-05 19:29:40 +00:00
assert_eq!(&*format!("{}", none),
&*format!("{}", ANSIString::from("none")));
2015-09-04 15:38:48 +00:00
}
2015-09-07 01:07:46 +00:00
}