#[derive(Default, Clone, Debug, PartialEq, Eq)]
pub(crate) struct AnsiBytes {
parser: anstyle_parse::Parser,
capture: AnsiCapture,
}
impl AnsiBytes {
pub(crate) fn new() -> Self {
Default::default()
}
pub(crate) fn extract_next<'s>(&'s mut self, bytes: &'s [u8]) -> AnsiBytesIter<'s> {
self.capture.reset();
self.capture.printable.reserve(bytes.len());
AnsiBytesIter {
bytes,
parser: &mut self.parser,
capture: &mut self.capture,
}
}
}
#[derive(Debug, PartialEq, Eq)]
pub(crate) struct AnsiBytesIter<'s> {
bytes: &'s [u8],
parser: &'s mut anstyle_parse::Parser,
capture: &'s mut AnsiCapture,
}
impl Iterator for AnsiBytesIter<'_> {
type Item = Element;
#[inline]
fn next(&mut self) -> Option<Self::Item> {
next_bytes(&mut self.bytes, self.parser, self.capture)
}
}
#[inline]
fn next_bytes(
bytes: &mut &[u8],
parser: &mut anstyle_parse::Parser,
capture: &mut AnsiCapture,
) -> Option<Element> {
capture.reset();
while capture.ready.is_none() {
let byte = if let Some((byte, remainder)) = (*bytes).split_first() {
*bytes = remainder;
*byte
} else {
break;
};
parser.advance(capture, byte);
}
if capture.printable.is_empty() {
return None;
}
let (style, url) = capture.ready.clone().unwrap_or((capture.style, None));
Some(Element {
text: std::mem::take(&mut capture.printable),
style,
url,
})
}
#[derive(Default, Clone, Debug, PartialEq, Eq)]
struct AnsiCapture {
style: anstyle::Style,
printable: String,
hyperlink: Option<String>,
ready: Option<(anstyle::Style, Option<String>)>,
}
impl AnsiCapture {
fn reset(&mut self) {
self.ready = None;
}
}
impl anstyle_parse::Perform for AnsiCapture {
fn print(&mut self, c: char) {
self.printable.push(c);
}
fn execute(&mut self, byte: u8) {
if byte.is_ascii_whitespace() {
self.printable.push(byte as char);
}
}
fn csi_dispatch(
&mut self,
params: &anstyle_parse::Params,
_intermediates: &[u8],
ignore: bool,
action: u8,
) {
if ignore {
return;
}
if action != b'm' {
return;
}
let mut style = self.style;
let mut state = CsiState::Normal;
let mut r = None;
let mut g = None;
let mut color_target = ColorTarget::Fg;
for param in params {
for value in param {
match (state, *value) {
(CsiState::Normal, 0) => {
style = anstyle::Style::default();
break;
}
(CsiState::Normal, 1) => {
style = style.bold();
break;
}
(CsiState::Normal, 2) => {
style = style.dimmed();
break;
}
(CsiState::Normal, 3) => {
style = style.italic();
break;
}
(CsiState::Normal, 4) => {
style = style.underline();
state = CsiState::Underline;
}
(CsiState::Normal, 21) => {
style |= anstyle::Effects::DOUBLE_UNDERLINE;
break;
}
(CsiState::Normal, 7) => {
style = style.invert();
break;
}
(CsiState::Normal, 8) => {
style = style.hidden();
break;
}
(CsiState::Normal, 9) => {
style = style.strikethrough();
break;
}
(CsiState::Normal, 30..=37) => {
let color = to_ansi_color(value - 30).expect("within 4-bit range");
style = style.fg_color(Some(color.into()));
break;
}
(CsiState::Normal, 38) => {
color_target = ColorTarget::Fg;
state = CsiState::PrepareCustomColor;
}
(CsiState::Normal, 39) => {
style = style.fg_color(None);
break;
}
(CsiState::Normal, 40..=47) => {
let color = to_ansi_color(value - 40).expect("within 4-bit range");
style = style.bg_color(Some(color.into()));
break;
}
(CsiState::Normal, 48) => {
color_target = ColorTarget::Bg;
state = CsiState::PrepareCustomColor;
}
(CsiState::Normal, 49) => {
style = style.bg_color(None);
break;
}
(CsiState::Normal, 58) => {
color_target = ColorTarget::Underline;
state = CsiState::PrepareCustomColor;
}
(CsiState::Normal, 90..=97) => {
let color = to_ansi_color(value - 90)
.expect("within 4-bit range")
.bright(true);
style = style.fg_color(Some(color.into()));
break;
}
(CsiState::Normal, 100..=107) => {
let color = to_ansi_color(value - 100)
.expect("within 4-bit range")
.bright(true);
style = style.bg_color(Some(color.into()));
break;
}
(CsiState::PrepareCustomColor, 5) => {
state = CsiState::Ansi256;
}
(CsiState::PrepareCustomColor, 2) => {
state = CsiState::Rgb;
r = None;
g = None;
}
(CsiState::Ansi256, n) => {
let color = anstyle::Ansi256Color(n as u8);
style = match color_target {
ColorTarget::Fg => style.fg_color(Some(color.into())),
ColorTarget::Bg => style.bg_color(Some(color.into())),
ColorTarget::Underline => style.underline_color(Some(color.into())),
};
break;
}
(CsiState::Rgb, b) => match (r, g) {
(None, _) => {
r = Some(b);
}
(Some(_), None) => {
g = Some(b);
}
(Some(r), Some(g)) => {
let color = anstyle::RgbColor(r as u8, g as u8, b as u8);
style = match color_target {
ColorTarget::Fg => style.fg_color(Some(color.into())),
ColorTarget::Bg => style.bg_color(Some(color.into())),
ColorTarget::Underline => style.underline_color(Some(color.into())),
};
break;
}
},
(CsiState::Underline, 0) => {
style =
style.effects(style.get_effects().remove(anstyle::Effects::UNDERLINE));
}
(CsiState::Underline, 1) => {
}
(CsiState::Underline, 2) => {
style = style
.effects(style.get_effects().remove(anstyle::Effects::UNDERLINE))
| anstyle::Effects::DOUBLE_UNDERLINE;
}
(CsiState::Underline, 3) => {
style = style
.effects(style.get_effects().remove(anstyle::Effects::UNDERLINE))
| anstyle::Effects::CURLY_UNDERLINE;
}
(CsiState::Underline, 4) => {
style = style
.effects(style.get_effects().remove(anstyle::Effects::UNDERLINE))
| anstyle::Effects::DOTTED_UNDERLINE;
}
(CsiState::Underline, 5) => {
style = style
.effects(style.get_effects().remove(anstyle::Effects::UNDERLINE))
| anstyle::Effects::DASHED_UNDERLINE;
}
_ => {
break;
}
}
}
}
if style != self.style && !self.printable.is_empty() {
self.ready = Some((self.style, self.hyperlink.clone()));
}
self.style = style;
}
fn osc_dispatch(&mut self, params: &[&[u8]], _bell_terminated: bool) {
let mut state = OscState::Normal;
for value in params {
match (state, value) {
(OscState::Normal, &[b'8']) => {
state = OscState::HyperlinkParams;
}
(OscState::HyperlinkParams, _) => {
state = OscState::HyperlinkUri;
}
(OscState::HyperlinkUri, &[]) => {
if self.hyperlink.is_some() {
self.ready = Some((self.style, std::mem::take(&mut self.hyperlink)));
}
break;
}
(OscState::HyperlinkUri, uri) => {
let hyperlink = uri.iter().map(|b| *b as char).collect::<String>();
self.hyperlink = Some(hyperlink);
if !self.printable.is_empty() {
self.ready = Some((self.style, None));
}
break;
}
_ => {
break;
}
}
}
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub(crate) struct Element {
pub(crate) text: String,
pub(crate) style: anstyle::Style,
pub(crate) url: Option<String>,
}
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
enum CsiState {
Normal,
PrepareCustomColor,
Ansi256,
Rgb,
Underline,
}
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
enum OscState {
Normal,
HyperlinkParams,
HyperlinkUri,
}
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
enum ColorTarget {
Fg,
Bg,
Underline,
}
fn to_ansi_color(digit: u16) -> Option<anstyle::AnsiColor> {
match digit {
0 => Some(anstyle::AnsiColor::Black),
1 => Some(anstyle::AnsiColor::Red),
2 => Some(anstyle::AnsiColor::Green),
3 => Some(anstyle::AnsiColor::Yellow),
4 => Some(anstyle::AnsiColor::Blue),
5 => Some(anstyle::AnsiColor::Magenta),
6 => Some(anstyle::AnsiColor::Cyan),
7 => Some(anstyle::AnsiColor::White),
_ => None,
}
}
#[cfg(test)]
mod test {
use super::*;
use proptest::prelude::*;
const URL: &str = "https://example.com";
#[track_caller]
fn verify(input: &str, expected: Vec<Element>) {
let expected = expected.into_iter().collect::<Vec<_>>();
let mut state = AnsiBytes::new();
let actual = state.extract_next(input.as_bytes()).collect::<Vec<_>>();
assert_eq!(expected, actual, "{input:?}");
}
fn hyperlink(input: &str, url: &str) -> String {
format!("\x1B]8;;{url}\x1B\\{input}\x1B]8;;\x1B\\")
}
#[test]
fn start() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!("{green_on_red}Hello{green_on_red:#} world!");
let expected = vec![
Element {
text: "Hello".to_owned(),
style: green_on_red,
url: None,
},
Element {
text: " world!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
#[test]
fn middle() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!("Hello {green_on_red}world{green_on_red:#}!");
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: None,
},
Element {
text: "world".to_owned(),
style: green_on_red,
url: None,
},
Element {
text: "!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
#[test]
fn end() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!("Hello {green_on_red}world!{green_on_red:#}");
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: None,
},
Element {
text: "world!".to_owned(),
style: green_on_red,
url: None,
},
];
verify(&input, expected);
}
#[test]
fn ansi256_colors() {
let ansi_11 = anstyle::Ansi256Color(11).on_default();
let input = format!("Hello {ansi_11}world{ansi_11:#}!");
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: None,
},
Element {
text: "world".to_owned(),
style: ansi_11,
url: None,
},
Element {
text: "!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
#[test]
fn hyperlink_start() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!(
"{green_on_red}{}{green_on_red:#} world!",
hyperlink("Hello", URL)
);
let expected = vec![
Element {
text: "Hello".to_owned(),
style: green_on_red,
url: Some(URL.to_owned()),
},
Element {
text: " world!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
#[test]
fn hyperlink_middle() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!(
"Hello {green_on_red}{}{green_on_red:#}!",
hyperlink("world", URL)
);
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: None,
},
Element {
text: "world".to_owned(),
style: green_on_red,
url: Some(URL.to_owned()),
},
Element {
text: "!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
#[test]
fn hyperlink_end() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!(
"Hello {green_on_red}{}{green_on_red:#}",
hyperlink("world!", URL)
);
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: None,
},
Element {
text: "world!".to_owned(),
style: green_on_red,
url: Some(URL.to_owned()),
},
];
verify(&input, expected);
}
#[test]
fn hyperlink_ansi256_colors() {
let ansi_11 = anstyle::Ansi256Color(11).on_default();
let input = format!("Hello {ansi_11}{}{ansi_11:#}!", hyperlink("world", URL));
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: None,
},
Element {
text: "world".to_owned(),
style: ansi_11,
url: Some(URL.to_owned()),
},
Element {
text: "!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
#[test]
fn style_mid_hyperlink_text() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let styled_str = format!("Hello {green_on_red}world{green_on_red:#}!");
let input = hyperlink(&styled_str, URL);
let expected = vec![
Element {
text: "Hello ".to_owned(),
style: anstyle::Style::default(),
url: Some(URL.to_owned()),
},
Element {
text: "world".to_owned(),
style: green_on_red,
url: Some(URL.to_owned()),
},
Element {
text: "!".to_owned(),
style: anstyle::Style::default(),
url: Some(URL.to_owned()),
},
];
verify(&input, expected);
}
#[test]
fn hyperlink_empty() {
let green_on_red = anstyle::AnsiColor::Green.on(anstyle::AnsiColor::Red);
let input = format!(
"{green_on_red}{}{green_on_red:#} world!",
hyperlink("Hello", "")
);
let expected = vec![
Element {
text: "Hello".to_owned(),
style: green_on_red,
url: None,
},
Element {
text: " world!".to_owned(),
style: anstyle::Style::default(),
url: None,
},
];
verify(&input, expected);
}
proptest! {
#[test]
#[cfg_attr(miri, ignore)] fn wincon_no_escapes(s in "\\PC*") {
let expected = if s.is_empty() {
vec![]
} else {
vec![Element {
text: s.clone(),
style: anstyle::Style::default(),
url: None,
}]
};
let mut state = AnsiBytes::new();
let actual = state.extract_next(s.as_bytes()).collect::<Vec<_>>();
assert_eq!(expected, actual);
}
}
}