Skip to content

Commit 00c7aa2

Browse files
committed
synchronized output
1 parent 9b533cc commit 00c7aa2

File tree

4 files changed

+142
-39
lines changed

4 files changed

+142
-39
lines changed

src/draw_target.rs

+35-3
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ use console::Term;
1010
use instant::Instant;
1111

1212
use crate::multi::{MultiProgressAlignment, MultiState};
13-
use crate::TermLike;
13+
use crate::{term_like, TermLike};
1414

1515
/// Target for draw operations
1616
///
@@ -69,9 +69,12 @@ impl ProgressDrawTarget {
6969
///
7070
/// Will panic if `refresh_rate` is `0`.
7171
pub fn term(term: Term, refresh_rate: u8) -> Self {
72+
let supports_ansi_codes = term.supports_ansi_codes();
73+
7274
Self {
7375
kind: TargetKind::Term {
7476
term,
77+
supports_ansi_codes,
7578
last_line_count: 0,
7679
rate_limiter: RateLimiter::new(refresh_rate),
7780
draw_state: DrawState::default(),
@@ -81,9 +84,12 @@ impl ProgressDrawTarget {
8184

8285
/// Draw to a boxed object that implements the [`TermLike`] trait.
8386
pub fn term_like(term_like: Box<dyn TermLike>) -> Self {
87+
let supports_ansi_codes = term_like.supports_ansi_codes();
88+
8489
Self {
8590
kind: TargetKind::TermLike {
8691
inner: term_like,
92+
supports_ansi_codes,
8793
last_line_count: 0,
8894
rate_limiter: None,
8995
draw_state: DrawState::default(),
@@ -94,9 +100,12 @@ impl ProgressDrawTarget {
94100
/// Draw to a boxed object that implements the [`TermLike`] trait,
95101
/// with a specific refresh rate.
96102
pub fn term_like_with_hz(term_like: Box<dyn TermLike>, refresh_rate: u8) -> Self {
103+
let supports_ansi_codes = term_like.supports_ansi_codes();
104+
97105
Self {
98106
kind: TargetKind::TermLike {
99107
inner: term_like,
108+
supports_ansi_codes,
100109
last_line_count: 0,
101110
rate_limiter: Option::from(RateLimiter::new(refresh_rate)),
102111
draw_state: DrawState::default(),
@@ -149,6 +158,7 @@ impl ProgressDrawTarget {
149158
match &mut self.kind {
150159
TargetKind::Term {
151160
term,
161+
supports_ansi_codes,
152162
last_line_count,
153163
rate_limiter,
154164
draw_state,
@@ -160,6 +170,7 @@ impl ProgressDrawTarget {
160170
match force_draw || rate_limiter.allow(now) {
161171
true => Some(Drawable::Term {
162172
term,
173+
supports_ansi_codes: *supports_ansi_codes,
163174
last_line_count,
164175
draw_state,
165176
}),
@@ -177,12 +188,14 @@ impl ProgressDrawTarget {
177188
}
178189
TargetKind::TermLike {
179190
inner,
191+
supports_ansi_codes,
180192
last_line_count,
181193
rate_limiter,
182194
draw_state,
183195
} => match force_draw || rate_limiter.as_mut().map_or(true, |r| r.allow(now)) {
184196
true => Some(Drawable::TermLike {
185197
term_like: &**inner,
198+
supports_ansi_codes: *supports_ansi_codes,
186199
last_line_count,
187200
draw_state,
188201
}),
@@ -228,6 +241,7 @@ impl ProgressDrawTarget {
228241
enum TargetKind {
229242
Term {
230243
term: Term,
244+
supports_ansi_codes: bool,
231245
last_line_count: usize,
232246
rate_limiter: RateLimiter,
233247
draw_state: DrawState,
@@ -239,6 +253,7 @@ enum TargetKind {
239253
Hidden,
240254
TermLike {
241255
inner: Box<dyn TermLike>,
256+
supports_ansi_codes: bool,
242257
last_line_count: usize,
243258
rate_limiter: Option<RateLimiter>,
244259
draw_state: DrawState,
@@ -268,6 +283,7 @@ impl TargetKind {
268283
pub(crate) enum Drawable<'a> {
269284
Term {
270285
term: &'a Term,
286+
supports_ansi_codes: bool,
271287
last_line_count: &'a mut usize,
272288
draw_state: &'a mut DrawState,
273289
},
@@ -279,6 +295,7 @@ pub(crate) enum Drawable<'a> {
279295
},
280296
TermLike {
281297
term_like: &'a dyn TermLike,
298+
supports_ansi_codes: bool,
282299
last_line_count: &'a mut usize,
283300
draw_state: &'a mut DrawState,
284301
},
@@ -324,9 +341,10 @@ impl<'a> Drawable<'a> {
324341
match self {
325342
Drawable::Term {
326343
term,
344+
supports_ansi_codes,
327345
last_line_count,
328346
draw_state,
329-
} => draw_state.draw_to_term(term, last_line_count),
347+
} => draw_state.draw_to_term(term, supports_ansi_codes, last_line_count),
330348
Drawable::Multi {
331349
mut state,
332350
force_draw,
@@ -335,9 +353,10 @@ impl<'a> Drawable<'a> {
335353
} => state.draw(force_draw, None, now),
336354
Drawable::TermLike {
337355
term_like,
356+
supports_ansi_codes,
338357
last_line_count,
339358
draw_state,
340-
} => draw_state.draw_to_term(term_like, last_line_count),
359+
} => draw_state.draw_to_term(term_like, supports_ansi_codes, last_line_count),
341360
}
342361
}
343362
}
@@ -464,12 +483,20 @@ impl DrawState {
464483
fn draw_to_term(
465484
&mut self,
466485
term: &(impl TermLike + ?Sized),
486+
supports_ansi_codes: bool,
467487
last_line_count: &mut usize,
468488
) -> io::Result<()> {
469489
if panicking() {
470490
return Ok(());
471491
}
472492

493+
// Begin synchronized update
494+
let sync_guard = if supports_ansi_codes {
495+
Some(term_like::SyncGuard::begin_sync(term)?)
496+
} else {
497+
None
498+
};
499+
473500
if !self.lines.is_empty() && self.move_cursor {
474501
term.move_cursor_up(*last_line_count)?;
475502
} else {
@@ -542,6 +569,11 @@ impl DrawState {
542569
}
543570
term.write_str(&" ".repeat(last_line_filler))?;
544571

572+
// End synchronized update
573+
if let Some(sync_guard) = sync_guard {
574+
sync_guard.finish_sync()?;
575+
}
576+
545577
term.flush()?;
546578
*last_line_count = real_len - orphan_visual_line_count + shift;
547579
Ok(())

src/in_memory.rs

+39-6
Original file line numberDiff line numberDiff line change
@@ -13,14 +13,16 @@ use crate::TermLike;
1313
#[derive(Debug, Clone)]
1414
pub struct InMemoryTerm {
1515
state: Arc<Mutex<InMemoryTermState>>,
16+
supports_ansi_codes: bool,
1617
}
1718

1819
impl InMemoryTerm {
19-
pub fn new(rows: u16, cols: u16) -> InMemoryTerm {
20+
pub fn new(rows: u16, cols: u16, supports_ansi_codes: bool) -> InMemoryTerm {
2021
assert!(rows > 0, "rows must be > 0");
2122
assert!(cols > 0, "cols must be > 0");
2223
InMemoryTerm {
2324
state: Arc::new(Mutex::new(InMemoryTermState::new(rows, cols))),
25+
supports_ansi_codes,
2426
}
2527
}
2628

@@ -190,6 +192,10 @@ impl TermLike for InMemoryTerm {
190192
state.history.push(Move::Flush);
191193
state.parser.flush()
192194
}
195+
196+
fn supports_ansi_codes(&self) -> bool {
197+
self.supports_ansi_codes
198+
}
193199
}
194200

195201
struct InMemoryTermState {
@@ -234,6 +240,8 @@ enum Move {
234240

235241
#[cfg(test)]
236242
mod test {
243+
use crate::term_like;
244+
237245
use super::*;
238246

239247
fn cursor_pos(in_mem: &InMemoryTerm) -> (u16, u16) {
@@ -248,7 +256,7 @@ mod test {
248256

249257
#[test]
250258
fn line_wrapping() {
251-
let in_mem = InMemoryTerm::new(10, 5);
259+
let in_mem = InMemoryTerm::new(10, 5, false);
252260
assert_eq!(cursor_pos(&in_mem), (0, 0));
253261

254262
in_mem.write_str("ABCDE").unwrap();
@@ -282,7 +290,7 @@ mod test {
282290

283291
#[test]
284292
fn write_line() {
285-
let in_mem = InMemoryTerm::new(10, 5);
293+
let in_mem = InMemoryTerm::new(10, 5, false);
286294
assert_eq!(cursor_pos(&in_mem), (0, 0));
287295

288296
in_mem.write_line("A").unwrap();
@@ -318,7 +326,7 @@ NewLine
318326

319327
#[test]
320328
fn basic_functionality() {
321-
let in_mem = InMemoryTerm::new(10, 80);
329+
let in_mem = InMemoryTerm::new(10, 80, false);
322330

323331
in_mem.write_line("This is a test line").unwrap();
324332
assert_eq!(in_mem.contents(), "This is a test line");
@@ -352,7 +360,7 @@ Str("TEST")
352360

353361
#[test]
354362
fn newlines() {
355-
let in_mem = InMemoryTerm::new(10, 10);
363+
let in_mem = InMemoryTerm::new(10, 10, false);
356364
in_mem.write_line("LINE ONE").unwrap();
357365
in_mem.write_line("LINE TWO").unwrap();
358366
in_mem.write_line("").unwrap();
@@ -376,7 +384,7 @@ NewLine
376384

377385
#[test]
378386
fn cursor_zero_movement() {
379-
let in_mem = InMemoryTerm::new(10, 80);
387+
let in_mem = InMemoryTerm::new(10, 80, false);
380388
in_mem.write_line("LINE ONE").unwrap();
381389
assert_eq!(cursor_pos(&in_mem), (1, 0));
382390

@@ -396,4 +404,29 @@ NewLine
396404
in_mem.move_cursor_right(0).unwrap();
397405
assert_eq!(cursor_pos(&in_mem), (1, 1));
398406
}
407+
408+
#[test]
409+
fn sync_update() {
410+
let in_mem = InMemoryTerm::new(10, 80, true);
411+
assert_eq!(cursor_pos(&in_mem), (0, 0));
412+
413+
let sync_guard = term_like::SyncGuard::begin_sync(&in_mem).unwrap();
414+
in_mem.write_line("LINE ONE").unwrap();
415+
assert_eq!(cursor_pos(&in_mem), (1, 0));
416+
assert_eq!(
417+
in_mem.moves_since_last_check(),
418+
r#"Str("\u{1b}[?2026h")
419+
Str("LINE ONE")
420+
NewLine
421+
"#
422+
);
423+
424+
sync_guard.finish_sync().unwrap();
425+
assert_eq!(cursor_pos(&in_mem), (1, 0));
426+
assert_eq!(
427+
in_mem.moves_since_last_check(),
428+
r#"Str("\u{1b}[?2026l")
429+
"#
430+
);
431+
}
399432
}

src/term_like.rs

+38
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use std::cell::Cell;
12
use std::fmt::Debug;
23
use std::io;
34

@@ -34,6 +35,9 @@ pub trait TermLike: Debug + Send + Sync {
3435
fn clear_line(&self) -> io::Result<()>;
3536

3637
fn flush(&self) -> io::Result<()>;
38+
39+
// Whether ANSI escape sequences are supported
40+
fn supports_ansi_codes(&self) -> bool;
3741
}
3842

3943
impl TermLike for Term {
@@ -76,4 +80,38 @@ impl TermLike for Term {
7680
fn flush(&self) -> io::Result<()> {
7781
self.flush()
7882
}
83+
84+
fn supports_ansi_codes(&self) -> bool {
85+
self.features().colors_supported()
86+
}
87+
}
88+
89+
pub(crate) struct SyncGuard<'a, T: TermLike + ?Sized> {
90+
term_like: Cell<Option<&'a T>>,
91+
}
92+
93+
impl<'a, T: TermLike + ?Sized> SyncGuard<'a, T> {
94+
pub(crate) fn begin_sync(term_like: &'a T) -> io::Result<Self> {
95+
term_like.write_str("\x1b[?2026h")?;
96+
Ok(Self {
97+
term_like: Cell::new(Some(term_like)),
98+
})
99+
}
100+
101+
pub(crate) fn finish_sync(self) -> io::Result<()> {
102+
self.finish_sync_inner()
103+
}
104+
105+
fn finish_sync_inner(&self) -> io::Result<()> {
106+
if let Some(term_like) = self.term_like.take() {
107+
term_like.write_str("\x1b[?2026l")?;
108+
}
109+
Ok(())
110+
}
111+
}
112+
113+
impl<T: TermLike + ?Sized> Drop for SyncGuard<'_, T> {
114+
fn drop(&mut self) {
115+
let _ = self.finish_sync_inner();
116+
}
79117
}

0 commit comments

Comments
 (0)