mirror of
https://github.com/nikolassv/bartib
synced 2024-12-11 20:52:26 +00:00
implemented wrapping of cell contents in activity lists
This commit is contained in:
parent
3386e94095
commit
e12a143265
4 changed files with 109 additions and 17 deletions
61
Cargo.lock
generated
61
Cargo.lock
generated
|
@ -2,6 +2,15 @@
|
||||||
# It is not intended for manual editing.
|
# It is not intended for manual editing.
|
||||||
version = 3
|
version = 3
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "aho-corasick"
|
||||||
|
version = "0.7.18"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1e37cfd5e7657ada45f742d6e99ca5788580b5c529dc78faf11ece6dc702656f"
|
||||||
|
dependencies = [
|
||||||
|
"memchr",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ansi_term"
|
name = "ansi_term"
|
||||||
version = "0.11.0"
|
version = "0.11.0"
|
||||||
|
@ -43,6 +52,7 @@ dependencies = [
|
||||||
"clap",
|
"clap",
|
||||||
"nu-ansi-term",
|
"nu-ansi-term",
|
||||||
"term_size",
|
"term_size",
|
||||||
|
"textwrap 0.14.2",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -75,7 +85,7 @@ dependencies = [
|
||||||
"atty",
|
"atty",
|
||||||
"bitflags",
|
"bitflags",
|
||||||
"strsim",
|
"strsim",
|
||||||
"textwrap",
|
"textwrap 0.11.0",
|
||||||
"unicode-width",
|
"unicode-width",
|
||||||
"vec_map",
|
"vec_map",
|
||||||
]
|
]
|
||||||
|
@ -120,6 +130,12 @@ version = "0.2.86"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "b7282d924be3275cec7f6756ff4121987bc6481325397dde6ba3e7802b1a8b1c"
|
checksum = "b7282d924be3275cec7f6756ff4121987bc6481325397dde6ba3e7802b1a8b1c"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "memchr"
|
||||||
|
version = "2.4.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "308cc39be01b73d0d18f82a0e7b2a3df85245f84af96fdddc5d202d27e47b86a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nu-ansi-term"
|
name = "nu-ansi-term"
|
||||||
version = "0.34.0"
|
version = "0.34.0"
|
||||||
|
@ -174,6 +190,29 @@ dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "regex"
|
||||||
|
version = "1.5.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d07a8629359eb56f1e2fb1652bb04212c072a87ba68546a04065d525673ac461"
|
||||||
|
dependencies = [
|
||||||
|
"aho-corasick",
|
||||||
|
"memchr",
|
||||||
|
"regex-syntax",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "regex-syntax"
|
||||||
|
version = "0.6.25"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f497285884f3fcff424ffc933e56d7cbca511def0c9831a7f9b5f6153e3cc89b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "smawk"
|
||||||
|
version = "0.3.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f67ad224767faa3c7d8b6d91985b78e70a1324408abcb1cfcc2be4c06bc06043"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "strsim"
|
name = "strsim"
|
||||||
version = "0.8.0"
|
version = "0.8.0"
|
||||||
|
@ -211,6 +250,17 @@ dependencies = [
|
||||||
"unicode-width",
|
"unicode-width",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "textwrap"
|
||||||
|
version = "0.14.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "0066c8d12af8b5acd21e00547c3797fde4e8677254a7ee429176ccebbe93dd80"
|
||||||
|
dependencies = [
|
||||||
|
"smawk",
|
||||||
|
"unicode-linebreak",
|
||||||
|
"unicode-width",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "thiserror"
|
name = "thiserror"
|
||||||
version = "1.0.24"
|
version = "1.0.24"
|
||||||
|
@ -242,6 +292,15 @@ dependencies = [
|
||||||
"winapi 0.3.9",
|
"winapi 0.3.9",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "unicode-linebreak"
|
||||||
|
version = "0.1.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3a52dcaab0c48d931f7cc8ef826fa51690a08e1ea55117ef26f89864f532383f"
|
||||||
|
dependencies = [
|
||||||
|
"regex",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicode-width"
|
name = "unicode-width"
|
||||||
version = "0.1.8"
|
version = "0.1.8"
|
||||||
|
|
|
@ -12,4 +12,5 @@ clap = "~2.33"
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
anyhow = "1.0.42"
|
anyhow = "1.0.42"
|
||||||
nu-ansi-term = "0.34.0"
|
nu-ansi-term = "0.34.0"
|
||||||
term_size = "1.0.0-beta1"
|
term_size = "1.0.0-beta1"
|
||||||
|
textwrap = "0.14.2"
|
|
@ -1,3 +1,4 @@
|
||||||
pub static FORMAT_DATETIME: &str = "%F %R";
|
pub static FORMAT_DATETIME: &str = "%F %R";
|
||||||
pub static FORMAT_TIME: &str = "%R";
|
pub static FORMAT_TIME: &str = "%R";
|
||||||
pub static FORMAT_DATE: &str = "%F";
|
pub static FORMAT_DATE: &str = "%F";
|
||||||
|
pub static DEFAULT_WIDTH : usize = 80;
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
use std::cmp;
|
use std::cmp;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::str;
|
use std::str;
|
||||||
|
use std::borrow::Cow;
|
||||||
|
|
||||||
use nu_ansi_term::Style;
|
use nu_ansi_term::Style;
|
||||||
|
use textwrap;
|
||||||
|
use crate::conf;
|
||||||
|
|
||||||
pub enum Wrap {
|
pub enum Wrap {
|
||||||
Wrap,
|
Wrap,
|
||||||
|
@ -74,6 +77,11 @@ impl Table {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Calculates the widths for the columns in a table
|
||||||
|
|
||||||
|
If the width of the longest line in the table exceeds the maximum width for the output
|
||||||
|
all the wrapable columns shrink to an acceptable size.
|
||||||
|
*/
|
||||||
fn get_column_width(&self, max_width: usize) -> Vec<usize> {
|
fn get_column_width(&self, max_width: usize) -> Vec<usize> {
|
||||||
let mut max_column_width = self.get_max_column_width();
|
let mut max_column_width = self.get_max_column_width();
|
||||||
|
|
||||||
|
@ -82,10 +90,11 @@ impl Table {
|
||||||
let mut number_of_wrappable_columns : usize = columns_wrap.iter().filter(|w| matches!(w, Wrap::Wrap)).count();
|
let mut number_of_wrappable_columns : usize = columns_wrap.iter().filter(|w| matches!(w, Wrap::Wrap)).count();
|
||||||
|
|
||||||
if width <= max_width || number_of_wrappable_columns == 0 {
|
if width <= max_width || number_of_wrappable_columns == 0 {
|
||||||
// we do or can not wrap
|
// we do not need to or can not wrap
|
||||||
return max_column_width;
|
return max_column_width;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// the total width of the columns that we may not wrap
|
||||||
let unwrapable_width : usize = max_column_width.iter().zip(columns_wrap.iter())
|
let unwrapable_width : usize = max_column_width.iter().zip(columns_wrap.iter())
|
||||||
.filter(|(_, wrap)| matches!(wrap, Wrap::NoWrap))
|
.filter(|(_, wrap)| matches!(wrap, Wrap::NoWrap))
|
||||||
.map(|(width, _)| width)
|
.map(|(width, _)| width)
|
||||||
|
@ -96,14 +105,14 @@ impl Table {
|
||||||
return max_column_width;
|
return max_column_width;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut available_width_for_wrappable_columns = max_width - unwrapable_width;
|
|
||||||
|
|
||||||
// we start with a width of 0 for all the wrapable columns
|
// we start with a width of 0 for all the wrapable columns
|
||||||
let mut column_width : Vec<usize> = max_column_width.iter().zip(columns_wrap.iter())
|
let mut column_width : Vec<usize> = max_column_width.iter().zip(columns_wrap.iter())
|
||||||
.map(|(width, wrap)| if matches!(wrap, Wrap::NoWrap) { width.clone() } else { 0 })
|
.map(|(width, wrap)| if matches!(wrap, Wrap::NoWrap) { width.clone() } else { 0 })
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
// then we distribute the available width to the wrappable columns
|
// then we distribute the available width to the wrappable columns
|
||||||
|
let mut available_width_for_wrappable_columns = max_width - unwrapable_width;
|
||||||
|
|
||||||
while available_width_for_wrappable_columns > 0 && number_of_wrappable_columns > 0 {
|
while available_width_for_wrappable_columns > 0 && number_of_wrappable_columns > 0 {
|
||||||
|
|
||||||
// the maximum additional width we give each column in this round
|
// the maximum additional width we give each column in this round
|
||||||
|
@ -123,7 +132,7 @@ impl Table {
|
||||||
available_width_for_wrappable_columns -= *max_width - *width;
|
available_width_for_wrappable_columns -= *max_width - *width;
|
||||||
*width = *max_width;
|
*width = *max_width;
|
||||||
|
|
||||||
// this row won't need any more width
|
// this column won't need any more width
|
||||||
number_of_wrappable_columns -= 1;
|
number_of_wrappable_columns -= 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -157,7 +166,11 @@ impl Table {
|
||||||
|
|
||||||
impl fmt::Display for Table {
|
impl fmt::Display for Table {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
let column_width = self.get_column_width(1000); // TODO
|
|
||||||
|
let terminal_width = term_size::dimensions_stdout().map(|d| d.0)
|
||||||
|
.unwrap_or(conf::DEFAULT_WIDTH);
|
||||||
|
|
||||||
|
let column_width = self.get_column_width(terminal_width - self.columns.len());
|
||||||
|
|
||||||
let labels : Vec<&String> = self.columns.iter().map(|c| &c.label).collect();
|
let labels : Vec<&String> = self.columns.iter().map(|c| &c.label).collect();
|
||||||
|
|
||||||
|
@ -211,15 +224,33 @@ fn write_cells<T: AsRef<str> + std::fmt::Display>(
|
||||||
column_width: &[usize],
|
column_width: &[usize],
|
||||||
style: Option<Style>,
|
style: Option<Style>,
|
||||||
) -> fmt::Result {
|
) -> fmt::Result {
|
||||||
let cells_with_width: Vec<(Option<&usize>, &str)> = cells
|
|
||||||
|
let wrapped_cells : Vec<Vec<Cow<str>>> = cells
|
||||||
.iter()
|
.iter()
|
||||||
.map(|cell| cell.as_ref())
|
|
||||||
.enumerate()
|
.enumerate()
|
||||||
.map(|(i, cell)| (column_width.get(i), cell))
|
.map(|(i, c)| match column_width.get(i) {
|
||||||
|
Some(s) => textwrap::wrap(c.as_ref(), textwrap::Options::new(*s)),
|
||||||
|
None => {
|
||||||
|
let mut lines = Vec::new();
|
||||||
|
lines.push(Cow::from(c.as_ref()));
|
||||||
|
lines
|
||||||
|
}
|
||||||
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
for (width, cell) in cells_with_width {
|
let most_lines : usize = wrapped_cells.iter().map(|c| c.len()).max().unwrap_or(1);
|
||||||
write_with_width_and_style(f, cell, width, style)?;
|
|
||||||
|
for line in 0..most_lines {
|
||||||
|
for (width, wrapped_cell) in column_width.iter().zip(wrapped_cells.iter()) {
|
||||||
|
|
||||||
|
match wrapped_cell.get(line) {
|
||||||
|
Some(c) => write_with_width_and_style(f, c, width, style)?,
|
||||||
|
None => write!(f, "{} ", "\u{a0}".repeat(*width))?
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let is_last_line = line + 1 < most_lines;
|
||||||
|
if is_last_line { writeln!(f)?; }
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -228,17 +259,17 @@ fn write_cells<T: AsRef<str> + std::fmt::Display>(
|
||||||
fn write_with_width_and_style(
|
fn write_with_width_and_style(
|
||||||
f: &mut fmt::Formatter<'_>,
|
f: &mut fmt::Formatter<'_>,
|
||||||
content: &str,
|
content: &str,
|
||||||
opt_width: Option<&usize>,
|
width: &usize,
|
||||||
opt_style: Option<Style>,
|
opt_style: Option<Style>,
|
||||||
) -> fmt::Result {
|
) -> fmt::Result {
|
||||||
let content_length = content.chars().count();
|
|
||||||
let style_prefix = opt_style.map_or("".to_string(), |style| style.prefix().to_string());
|
let style_prefix = opt_style.map_or("".to_string(), |style| style.prefix().to_string());
|
||||||
let style_suffix = opt_style.map_or("".to_string(), |style| style.suffix().to_string());
|
let style_suffix = opt_style.map_or("".to_string(), |style| style.suffix().to_string());
|
||||||
let width = opt_width.unwrap_or(&content_length);
|
|
||||||
|
|
||||||
|
// cells are filled with non-breaking white space. Contrary to normal spaces non-breaking white
|
||||||
|
// space will be styled (e.g. underlined)
|
||||||
write!(
|
write!(
|
||||||
f,
|
f,
|
||||||
"{prefix}{content:<width$}{suffix} ",
|
"{prefix}{content:\u{a0}<width$}{suffix} ",
|
||||||
prefix = style_prefix,
|
prefix = style_prefix,
|
||||||
content = content,
|
content = content,
|
||||||
width = width,
|
width = width,
|
||||||
|
@ -371,7 +402,7 @@ mod tests {
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
format!("{}", t),
|
format!("{}", t),
|
||||||
"\u{1b}[4ma \u{1b}[0m \u{1b}[4mb \u{1b}[0m \u{1b}[4mc \u{1b}[0m \nabc defg \na b cdef \n"
|
"\u{1b}[4ma\u{a0}\u{a0}\u{1b}[0m \u{1b}[4mb\u{a0}\u{a0}\u{a0}\u{1b}[0m \u{1b}[4mc\u{a0}\u{a0}\u{a0}\u{1b}[0m \nabc defg \na\u{a0}\u{a0} b\u{a0}\u{a0}\u{a0} cdef \n"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue