1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
use std::{borrow::Cow, fmt::Write as WriteStr};
use termcolor::NoColor;
#[cfg(feature = "svg")]
use crate::write::{SvgLine, SvgWriter};
use crate::{
utils::{normalize_newlines, WriteAdapter},
write::HtmlWriter,
TermError,
};
mod parser;
#[cfg(test)]
mod tests;
pub(crate) use self::parser::TermOutputParser;
/// Marker trait for supported types of terminal output.
pub trait TermOutput: Clone + Send + Sync + 'static {}
/// Output captured from the terminal.
#[derive(Debug, Clone)]
pub struct Captured(String);
impl AsRef<str> for Captured {
fn as_ref(&self) -> &str {
&self.0
}
}
impl From<String> for Captured {
fn from(raw: String) -> Self {
// Normalize newlines to `\n`.
Self(match normalize_newlines(&raw) {
Cow::Owned(normalized) => normalized,
Cow::Borrowed(_) => raw,
})
}
}
impl Captured {
pub(crate) fn write_as_html(
&self,
output: &mut dyn WriteStr,
wrap_width: Option<usize>,
) -> Result<(), TermError> {
let mut html_writer = HtmlWriter::new(output, wrap_width);
TermOutputParser::new(&mut html_writer).parse(self.0.as_bytes())
}
#[cfg(feature = "svg")]
pub(crate) fn write_as_svg(
&self,
wrap_width: Option<usize>,
) -> Result<Vec<SvgLine>, TermError> {
let mut svg_writer = SvgWriter::new(wrap_width);
TermOutputParser::new(&mut svg_writer).parse(self.0.as_bytes())?;
Ok(svg_writer.into_lines())
}
/// Converts this terminal output to an HTML string.
///
/// The conversion applies styles by wrapping colored / styled text into `span`s with
/// the following `class`es:
///
/// - `bold`, `italic`, `dimmed`, `underline` are self-explanatory
/// - `fg0`, `fg1`, ..., `fg15` are used to indicate indexed 4-bit ANSI color of the text.
/// Indexes 0..=7 correspond to the ordinary color variations, and 8..=15
/// to the intense ones.
/// - `bg0`, `bg1`, ..., `bg15` work similarly, but for the background color instead of
/// text color.
///
/// Indexed ANSI colors with indexes >15 and ANSI RGB colors are rendered using the `style`
/// attribute.
///
/// The output string retains whitespace of the input. Hence, it needs to be wrapped
/// into a `pre` element or an element with the [`white-space`] CSS property set to `pre`
/// in order to be displayed properly.
///
/// # Errors
///
/// Returns an error if there was an issue processing output.
///
/// [`white-space`]: https://developer.mozilla.org/en-US/docs/Web/CSS/white-space
pub fn to_html(&self) -> Result<String, TermError> {
let mut output = String::with_capacity(self.0.len());
self.write_as_html(&mut output, None)?;
Ok(output)
}
fn write_as_plaintext(&self, output: &mut dyn WriteStr) -> Result<(), TermError> {
let mut plaintext_writer = NoColor::new(WriteAdapter::new(output));
TermOutputParser::new(&mut plaintext_writer).parse(self.0.as_bytes())
}
/// Converts this terminal output to plaintext.
///
/// # Errors
///
/// Returns an error if there was an issue processing output.
pub fn to_plaintext(&self) -> Result<String, TermError> {
let mut output = String::with_capacity(self.0.len());
self.write_as_plaintext(&mut output)?;
Ok(output)
}
}
impl TermOutput for Captured {}