diff --git a/crates/anstyle-wincon/src/stream.rs b/crates/anstyle-wincon/src/stream.rs index 14b7cf2d..d1d232b0 100644 --- a/crates/anstyle-wincon/src/stream.rs +++ b/crates/anstyle-wincon/src/stream.rs @@ -105,29 +105,25 @@ impl WinconStream for std::fs::File { /// Write colored text to the screen #[derive(Clone, Debug)] -pub(crate) struct ConsoleState { - initial_fg: Option, - initial_bg: Option, - last_fg: Option, - last_bg: Option, +pub(crate) enum ConsoleState { + Wincon(WinconAdapter), + Pass(PassThroughAdapter), } impl ConsoleState { pub(crate) fn new( stream: &S, ) -> std::io::Result { - let (initial_fg, initial_bg) = match stream.get_colors() { - Ok(ok) => ok, + let adapter = match stream.get_colors() { + Ok((Some(initial_fg), Some(initial_bg))) => { + Self::Wincon(WinconAdapter::new(initial_fg, initial_bg)) + } + Ok(_) => Self::Pass(PassThroughAdapter), Err(err) => { return Err(err); } }; - Ok(Self { - initial_fg, - initial_bg, - last_fg: initial_fg, - last_bg: initial_bg, - }) + Ok(adapter) } pub(crate) fn write( @@ -146,17 +142,63 @@ impl ConsoleState { &mut self, stream: &mut S, ) -> std::io::Result<()> { - self.apply(stream, self.initial_fg, self.initial_bg) + self.apply(stream, None, None) + } + + fn apply( + &mut self, + stream: &mut S, + fg: Option, + bg: Option, + ) -> std::io::Result<()> { + match self { + Self::Wincon(adapter) => adapter.apply(stream, fg, bg), + Self::Pass(adapter) => adapter.apply(stream, fg, bg), + } } +} +#[derive(Clone, Debug)] +pub(crate) struct PassThroughAdapter; + +impl PassThroughAdapter { fn apply( &mut self, stream: &mut S, fg: Option, bg: Option, ) -> std::io::Result<()> { - let fg = fg.or(self.initial_fg); - let bg = bg.or(self.initial_bg); + stream.set_colors(fg, bg)?; + Ok(()) + } +} + +#[derive(Clone, Debug)] +pub(crate) struct WinconAdapter { + initial_fg: anstyle::AnsiColor, + initial_bg: anstyle::AnsiColor, + last_fg: anstyle::AnsiColor, + last_bg: anstyle::AnsiColor, +} + +impl WinconAdapter { + fn new(initial_fg: anstyle::AnsiColor, initial_bg: anstyle::AnsiColor) -> Self { + Self { + initial_fg, + initial_bg, + last_fg: initial_fg, + last_bg: initial_bg, + } + } + + fn apply( + &mut self, + stream: &mut S, + fg: Option, + bg: Option, + ) -> std::io::Result<()> { + let fg = fg.unwrap_or(self.initial_fg); + let bg = bg.unwrap_or(self.initial_bg); if fg == self.last_fg && bg == self.last_bg { return Ok(()); } @@ -164,7 +206,7 @@ impl ConsoleState { // Ensure everything is written with the last set of colors before applying the next set stream.flush()?; - stream.set_colors(fg, bg)?; + stream.set_colors(Some(fg), Some(bg))?; self.last_fg = fg; self.last_bg = bg;