#![warn(missing_docs,
unused_extern_crates,
unused_import_braces,
unused_qualifications)]
extern crate unicode_width;
extern crate term;
extern crate atty;
#[cfg(feature = "csv")]
extern crate csv;
#[macro_use]
extern crate lazy_static;
extern crate encode_unicode;
use std::io::{self, Write, Error};
#[cfg(feature = "csv")]
use std::io::Read;
use std::fmt;
#[cfg(feature = "csv")]
use std::path::Path;
use std::iter::{FromIterator, IntoIterator};
use std::slice::{Iter, IterMut};
use std::ops::{Index, IndexMut};
use std::mem::transmute;
pub use term::{Attr, color};
pub(crate) use term::{Terminal, stdout};
pub mod cell;
pub mod row;
pub mod format;
mod utils;
use row::Row;
use cell::Cell;
use format::{TableFormat, LinePosition, consts};
use utils::StringWriter;
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
pub struct Table {
format: Box<TableFormat>,
titles: Box<Option<Row>>,
rows: Vec<Row>,
}
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
pub struct TableSlice<'a> {
format: &'a TableFormat,
titles: &'a Option<Row>,
rows: &'a [Row],
}
impl<'a> TableSlice<'a> {
pub fn get_column_num(&self) -> usize {
let mut cnum = 0;
for r in self.rows {
let l = r.len();
if l > cnum {
cnum = l;
}
}
cnum
}
pub fn len(&self) -> usize {
self.rows.len()
}
pub fn is_empty(&self) -> bool {
self.rows.is_empty()
}
pub fn get_row(&self, row: usize) -> Option<&Row> {
self.rows.get(row)
}
fn get_column_width(&self, col_idx: usize) -> usize {
let mut width = match *self.titles {
Some(ref t) => t.get_cell_width(col_idx),
None => 0,
};
for r in self.rows {
let l = r.get_cell_width(col_idx);
if l > width {
width = l;
}
}
width
}
fn get_all_column_width(&self) -> Vec<usize> {
let colnum = self.get_column_num();
let mut col_width = vec![0usize; colnum];
for i in 0..colnum {
col_width[i] = self.get_column_width(i);
}
col_width
}
pub fn column_iter(&self, column: usize) -> ColumnIter {
ColumnIter(self.rows.iter(), column)
}
pub fn row_iter(&self) -> Iter<Row> {
self.rows.iter()
}
fn __print<T: Write + ?Sized, F>(&self, out: &mut T, f: F) -> Result<(), Error>
where F: Fn(&Row, &mut T, &TableFormat, &[usize]) -> Result<(), Error>
{
let col_width = self.get_all_column_width();
self.format
.print_line_separator(out, &col_width, LinePosition::Top)?;
if let Some(ref t) = *self.titles {
f(t, out, self.format, &col_width)?;
self.format
.print_line_separator(out, &col_width, LinePosition::Title)?;
}
let mut iter = self.rows.into_iter().peekable();
while let Some(r) = iter.next() {
f(r, out, self.format, &col_width)?;
if iter.peek().is_some() {
self.format
.print_line_separator(out, &col_width, LinePosition::Intern)?;
}
}
self.format
.print_line_separator(out, &col_width, LinePosition::Bottom)?;
out.flush()
}
pub fn print<T: Write + ?Sized>(&self, out: &mut T) -> Result<(), Error> {
self.__print(out, Row::print)
}
pub fn print_term<T: Terminal + ?Sized>(&self, out: &mut T) -> Result<(), Error> {
self.__print(out, Row::print_term)
}
pub fn print_tty(&self, force_colorize: bool) {
let r = match (stdout(), atty::is(atty::Stream::Stdout) || force_colorize) {
(Some(mut o), true) => self.print_term(&mut *o),
_ => self.print(&mut io::stdout()),
};
if let Err(e) = r {
panic!("Cannot print table to standard output : {}", e);
}
}
pub fn printstd(&self) {
self.print_tty(false);
}
#[cfg(feature = "csv")]
pub fn to_csv<W: Write>(&self, w: W) -> csv::Result<csv::Writer<W>> {
self.to_csv_writer(csv::Writer::from_writer(w))
}
#[cfg(feature = "csv")]
pub fn to_csv_writer<W: Write>(&self,
mut writer: csv::Writer<W>)
-> csv::Result<csv::Writer<W>> {
for title in self.titles {
writer.write(title.iter().map(|c| c.get_content()))?;
}
for row in self.rows {
writer.write(row.iter().map(|c| c.get_content()))?;
}
writer.flush()?;
Ok(writer)
}
}
impl<'a> IntoIterator for &'a TableSlice<'a> {
type Item = &'a Row;
type IntoIter = Iter<'a, Row>;
fn into_iter(self) -> Self::IntoIter {
self.row_iter()
}
}
impl Table {
pub fn new() -> Table {
Self::init(Vec::new())
}
pub fn init(rows: Vec<Row>) -> Table {
Table {
rows: rows,
titles: Box::new(None),
format: Box::new(*consts::FORMAT_DEFAULT),
}
}
#[cfg(feature = "csv")]
pub fn from_csv_string(csv_s: &str) -> csv::Result<Table> {
Ok(Table::from_csv(&mut csv::Reader::from_string(csv_s).has_headers(false)))
}
#[cfg(feature = "csv")]
pub fn from_csv_file<P: AsRef<Path>>(csv_p: P) -> csv::Result<Table> {
Ok(Table::from_csv(&mut csv::Reader::from_file(csv_p)?.has_headers(false)))
}
#[cfg(feature = "csv")]
pub fn from_csv<R: Read>(reader: &mut csv::Reader<R>) -> Table {
Table::init(reader
.records()
.map(|row| {
Row::new(row.unwrap()
.into_iter()
.map(|cell| Cell::new(&cell))
.collect())
})
.collect())
}
pub fn set_format(&mut self, format: TableFormat) {
*self.format = format;
}
pub fn get_format(&mut self) -> &mut TableFormat {
&mut self.format
}
pub fn get_column_num(&self) -> usize {
self.as_ref().get_column_num()
}
pub fn len(&self) -> usize {
self.rows.len()
}
pub fn is_empty(&self) -> bool {
self.rows.is_empty()
}
pub fn set_titles(&mut self, titles: Row) {
*self.titles = Some(titles);
}
pub fn unset_titles(&mut self) {
*self.titles = None;
}
pub fn get_mut_row(&mut self, row: usize) -> Option<&mut Row> {
self.rows.get_mut(row)
}
pub fn get_row(&self, row: usize) -> Option<&Row> {
self.rows.get(row)
}
pub fn add_row(&mut self, row: Row) -> &mut Row {
self.rows.push(row);
let l = self.rows.len() - 1;
&mut self.rows[l]
}
pub fn add_empty_row(&mut self) -> &mut Row {
self.add_row(Row::default())
}
pub fn insert_row(&mut self, index: usize, row: Row) -> &mut Row {
if index < self.rows.len() {
self.rows.insert(index, row);
&mut self.rows[index]
} else {
self.add_row(row)
}
}
pub fn set_element(&mut self, element: &str, column: usize, row: usize) -> Result<(), &str> {
let rowline = self.get_mut_row(row).ok_or("Cannot find row")?;
rowline.set_cell(Cell::new(element), column)
}
pub fn remove_row(&mut self, index: usize) {
if index < self.rows.len() {
self.rows.remove(index);
}
}
pub fn column_iter(&self, column: usize) -> ColumnIter {
ColumnIter(self.rows.iter(), column)
}
pub fn column_iter_mut(&mut self, column: usize) -> ColumnIterMut {
ColumnIterMut(self.rows.iter_mut(), column)
}
pub fn row_iter(&self) -> Iter<Row> {
self.rows.iter()
}
pub fn row_iter_mut(&mut self) -> IterMut<Row> {
self.rows.iter_mut()
}
pub fn print<T: Write + ?Sized>(&self, out: &mut T) -> Result<(), Error> {
self.as_ref().print(out)
}
pub fn print_term<T: Terminal + ?Sized>(&self, out: &mut T) -> Result<(), Error> {
self.as_ref().print_term(out)
}
pub fn print_tty(&self, force_colorize: bool) {
self.as_ref().print_tty(force_colorize);
}
pub fn printstd(&self) {
self.as_ref().printstd();
}
#[cfg(feature = "csv")]
pub fn to_csv<W: Write>(&self, w: W) -> csv::Result<csv::Writer<W>> {
self.as_ref().to_csv(w)
}
#[cfg(feature = "csv")]
pub fn to_csv_writer<W: Write>(&self, writer: csv::Writer<W>) -> csv::Result<csv::Writer<W>> {
self.as_ref().to_csv_writer(writer)
}
}
impl Index<usize> for Table {
type Output = Row;
fn index(&self, idx: usize) -> &Self::Output {
&self.rows[idx]
}
}
impl<'a> Index<usize> for TableSlice<'a> {
type Output = Row;
fn index(&self, idx: usize) -> &Self::Output {
&self.rows[idx]
}
}
impl IndexMut<usize> for Table {
fn index_mut(&mut self, idx: usize) -> &mut Self::Output {
&mut self.rows[idx]
}
}
impl fmt::Display for Table {
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
self.as_ref().fmt(fmt)
}
}
impl<'a> fmt::Display for TableSlice<'a> {
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
let mut writer = StringWriter::new();
if self.print(&mut writer).is_err() {
return Err(fmt::Error);
}
fmt.write_str(writer.as_string())
}
}
impl<B: ToString, A: IntoIterator<Item = B>> FromIterator<A> for Table {
fn from_iter<T>(iterator: T) -> Table
where T: IntoIterator<Item = A>
{
Self::init(iterator.into_iter().map(Row::from).collect())
}
}
impl FromIterator<Row> for Table {
fn from_iter<T>(iterator: T) -> Table
where T: IntoIterator<Item = Row>
{
Self::init(iterator.into_iter().collect())
}
}
impl<T, A, B> From<T> for Table
where B: ToString,
A: IntoIterator<Item = B>,
T: IntoIterator<Item = A>
{
fn from(it: T) -> Table {
Self::from_iter(it)
}
}
impl<'a> IntoIterator for &'a Table {
type Item = &'a Row;
type IntoIter = Iter<'a, Row>;
fn into_iter(self) -> Self::IntoIter {
self.as_ref().row_iter()
}
}
impl<'a> IntoIterator for &'a mut Table {
type Item = &'a mut Row;
type IntoIter = IterMut<'a, Row>;
fn into_iter(self) -> Self::IntoIter {
self.row_iter_mut()
}
}
impl <A: Into<Row>> Extend<A> for Table {
fn extend<T: IntoIterator<Item=A>>(&mut self, iter: T) {
self.rows.extend(iter.into_iter().map(|r| r.into()));
}
}
pub struct ColumnIter<'a>(Iter<'a, Row>, usize);
impl<'a> Iterator for ColumnIter<'a> {
type Item = &'a Cell;
fn next(&mut self) -> Option<&'a Cell> {
self.0.next().and_then(|row| row.get_cell(self.1))
}
}
pub struct ColumnIterMut<'a>(IterMut<'a, Row>, usize);
impl<'a> Iterator for ColumnIterMut<'a> {
type Item = &'a mut Cell;
fn next(&mut self) -> Option<&'a mut Cell> {
self.0.next().and_then(|row| row.get_mut_cell(self.1))
}
}
impl<'a> AsRef<TableSlice<'a>> for TableSlice<'a> {
fn as_ref(&self) -> &TableSlice<'a> {
self
}
}
impl<'a> AsRef<TableSlice<'a>> for Table {
fn as_ref(&self) -> &TableSlice<'a> {
unsafe {
let s = &mut *((self as *const Table) as *mut Table);
s.rows.shrink_to_fit();
transmute(self)
}
}
}
pub trait Slice<'a, E> {
type Output: 'a;
fn slice(&'a self, arg: E) -> Self::Output;
}
impl<'a, T, E> Slice<'a, E> for T
where T: AsRef<TableSlice<'a>>,
[Row]: Index<E, Output = [Row]>
{
type Output = TableSlice<'a>;
fn slice(&'a self, arg: E) -> Self::Output {
let sl = self.as_ref();
TableSlice {
format: sl.format,
titles: sl.titles,
rows: sl.rows.index(arg),
}
}
}
#[macro_export]
macro_rules! table {
($([$($content:tt)*]), *) => (
$crate::Table::init(vec![$(row![$($content)*]), *])
);
}
#[macro_export]
macro_rules! ptable {
($($content:tt)*) => (
{
let tab = table!($($content)*);
tab.printstd();
tab
}
);
}
#[cfg(test)]
mod tests {
use Table;
use Slice;
use row::Row;
use cell::Cell;
use format;
use format::consts::{FORMAT_DEFAULT, FORMAT_NO_LINESEP, FORMAT_NO_COLSEP, FORMAT_CLEAN};
#[test]
fn table() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
let out = "\
+-----+----+-----+
| t1 | t2 | t3 |
+=====+====+=====+
| a | bc | def |
+-----+----+-----+
| def | bc | a |
+-----+----+-----+
";
assert_eq!(table.to_string().replace("\r\n", "\n"), out);
table.unset_titles();
let out = "\
+-----+----+-----+
| a | bc | def |
+-----+----+-----+
| def | bc | a |
+-----+----+-----+
";
assert_eq!(table.to_string().replace("\r\n", "\n"), out);
}
#[test]
fn index() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
assert_eq!(table[1][1].get_content(), "bc");
table[1][1] = Cell::new("newval");
assert_eq!(table[1][1].get_content(), "newval");
let out = "\
+-----+--------+-----+
| t1 | t2 | t3 |
+=====+========+=====+
| a | bc | def |
+-----+--------+-----+
| def | newval | a |
+-----+--------+-----+
";
assert_eq!(table.to_string().replace("\r\n", "\n"), out);
}
#[test]
fn table_size() {
let mut table = Table::new();
assert!(table.is_empty());
assert!(table.as_ref().is_empty());
assert_eq!(table.len(), 0);
assert_eq!(table.as_ref().len(), 0);
assert_eq!(table.get_column_num(), 0);
assert_eq!(table.as_ref().get_column_num(), 0);
table.add_empty_row();
assert!(!table.is_empty());
assert!(!table.as_ref().is_empty());
assert_eq!(table.len(), 1);
assert_eq!(table.as_ref().len(), 1);
assert_eq!(table.get_column_num(), 0);
assert_eq!(table.as_ref().get_column_num(), 0);
table[0].add_cell(Cell::default());
assert_eq!(table.get_column_num(), 1);
assert_eq!(table.as_ref().get_column_num(), 1);
}
#[test]
fn get_row() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
assert!(table.get_row(12).is_none());
assert!(table.get_row(1).is_some());
assert_eq!(table.get_row(1).unwrap()[0].get_content(), "def");
assert!(table.get_mut_row(12).is_none());
assert!(table.get_mut_row(1).is_some());
table.get_mut_row(1).unwrap().add_cell(Cell::new("z"));
assert_eq!(table.get_row(1).unwrap()[3].get_content(), "z");
}
#[test]
fn add_empty_row() {
let mut table = Table::new();
assert_eq!(table.len(), 0);
table.add_empty_row();
assert_eq!(table.len(), 1);
assert_eq!(table[0].len(), 0);
}
#[test]
fn remove_row() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.remove_row(12);
assert_eq!(table.len(), 2);
table.remove_row(0);
assert_eq!(table.len(), 1);
assert_eq!(table[0][0].get_content(), "def");
}
#[test]
fn insert_row() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.insert_row(12,
Row::new(vec![Cell::new("1"), Cell::new("2"), Cell::new("3")]));
assert_eq!(table.len(), 3);
assert_eq!(table[2][1].get_content(), "2");
table.insert_row(1,
Row::new(vec![Cell::new("3"), Cell::new("4"), Cell::new("5")]));
assert_eq!(table.len(), 4);
assert_eq!(table[1][1].get_content(), "4");
assert_eq!(table[2][1].get_content(), "bc");
}
#[test]
fn set_element() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
assert!(table.set_element("foo", 12, 12).is_err());
assert!(table.set_element("foo", 1, 1).is_ok());
assert_eq!(table[1][1].get_content(), "foo");
}
#[test]
fn no_linesep() {
let mut table = Table::new();
table.set_format(*FORMAT_NO_LINESEP);
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
assert_eq!(table[1][1].get_content(), "bc");
table[1][1] = Cell::new("newval");
assert_eq!(table[1][1].get_content(), "newval");
let out = "\
+-----+--------+-----+
| t1 | t2 | t3 |
| a | bc | def |
| def | newval | a |
+-----+--------+-----+
";
assert_eq!(table.to_string().replace("\r\n", "\n"), out);
}
#[test]
fn no_colsep() {
let mut table = Table::new();
table.set_format(*FORMAT_NO_COLSEP);
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
assert_eq!(table[1][1].get_content(), "bc");
table[1][1] = Cell::new("newval");
assert_eq!(table[1][1].get_content(), "newval");
let out = "\
------------------
t1 t2 t3 \n\
==================
a bc def \n\
------------------
def newval a \n\
------------------
";
println!("{}", out);
println!("____");
println!("{}", table.to_string().replace("\r\n", "\n"));
assert_eq!(table.to_string().replace("\r\n", "\n"), out);
}
#[test]
fn clean() {
let mut table = Table::new();
table.set_format(*FORMAT_CLEAN);
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
assert_eq!(table[1][1].get_content(), "bc");
table[1][1] = Cell::new("newval");
assert_eq!(table[1][1].get_content(), "newval");
let out = "\
\u{0020}t1 t2 t3 \n\
\u{0020}a bc def \n\
\u{0020}def newval a \n\
";
println!("{}", out);
println!("____");
println!("{}", table.to_string().replace("\r\n", "\n"));
assert_eq!(out, table.to_string().replace("\r\n", "\n"));
}
#[test]
fn padding() {
let mut table = Table::new();
let mut format = *FORMAT_DEFAULT;
format.padding(2, 2);
table.set_format(format);
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
assert_eq!(table[1][1].get_content(), "bc");
table[1][1] = Cell::new("newval");
assert_eq!(table[1][1].get_content(), "newval");
let out = "\
+-------+----------+-------+
| t1 | t2 | t3 |
+=======+==========+=======+
| a | bc | def |
+-------+----------+-------+
| def | newval | a |
+-------+----------+-------+
";
println!("{}", out);
println!("____");
println!("{}", table.to_string().replace("\r\n", "\n"));
assert_eq!(out, table.to_string().replace("\r\n", "\n"));
}
#[test]
fn indent() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("a"), Cell::new("bc"), Cell::new("def")]));
table.add_row(Row::new(vec![Cell::new("def"), Cell::new("bc"), Cell::new("a")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
table.get_format().indent(8);
let out = " +-----+----+-----+
| t1 | t2 | t3 |
+=====+====+=====+
| a | bc | def |
+-----+----+-----+
| def | bc | a |
+-----+----+-----+
";
assert_eq!(table.to_string().replace("\r\n", "\n"), out);
}
#[test]
fn slices() {
let mut table = Table::new();
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
table.add_row(Row::new(vec![Cell::new("0"), Cell::new("0"), Cell::new("0")]));
table.add_row(Row::new(vec![Cell::new("1"), Cell::new("1"), Cell::new("1")]));
table.add_row(Row::new(vec![Cell::new("2"), Cell::new("2"), Cell::new("2")]));
table.add_row(Row::new(vec![Cell::new("3"), Cell::new("3"), Cell::new("3")]));
table.add_row(Row::new(vec![Cell::new("4"), Cell::new("4"), Cell::new("4")]));
table.add_row(Row::new(vec![Cell::new("5"), Cell::new("5"), Cell::new("5")]));
let out = "\
+----+----+----+
| t1 | t2 | t3 |
+====+====+====+
| 1 | 1 | 1 |
+----+----+----+
| 2 | 2 | 2 |
+----+----+----+
| 3 | 3 | 3 |
+----+----+----+
";
let slice = table.slice(..);
let slice = slice.slice(1..);
let slice = slice.slice(..3);
assert_eq!(out, slice.to_string().replace("\r\n", "\n"));
assert_eq!(out, table.slice(1..4).to_string().replace("\r\n", "\n"));
}
#[test]
fn test_unicode_separators() {
let mut table = Table::new();
table.set_format(format::FormatBuilder::new()
.column_separator('│')
.borders('│')
.separators(&[format::LinePosition::Top],
format::LineSeparator::new('─',
'┬',
'┌',
'┐'))
.separators(&[format::LinePosition::Intern],
format::LineSeparator::new('─',
'┼',
'├',
'┤'))
.separators(&[format::LinePosition::Bottom],
format::LineSeparator::new('─',
'┴',
'└',
'┘'))
.padding(1, 1)
.build());
table.add_row(Row::new(vec![Cell::new("1"), Cell::new("1"), Cell::new("1")]));
table.add_row(Row::new(vec![Cell::new("2"), Cell::new("2"), Cell::new("2")]));
table.set_titles(Row::new(vec![Cell::new("t1"), Cell::new("t2"), Cell::new("t3")]));
let out = "\
┌────┬────┬────┐
│ t1 │ t2 │ t3 │
├────┼────┼────┤
│ 1 │ 1 │ 1 │
├────┼────┼────┤
│ 2 │ 2 │ 2 │
└────┴────┴────┘
";
println!("{}", out);
println!("____");
println!("{}", table.to_string().replace("\r\n", "\n"));
assert_eq!(out, table.to_string().replace("\r\n", "\n"));
}
#[cfg(feature = "csv")]
mod csv {
use Table;
use row::Row;
use cell::Cell;
static CSV_S: &'static str = "ABC,DEFG,HIJKLMN\n\
foobar,bar,foo\n\
foobar2,bar2,foo2\n";
fn test_table() -> Table {
let mut table = Table::new();
table
.add_row(Row::new(vec![Cell::new("ABC"), Cell::new("DEFG"), Cell::new("HIJKLMN")]));
table.add_row(Row::new(vec![Cell::new("foobar"), Cell::new("bar"), Cell::new("foo")]));
table.add_row(Row::new(vec![Cell::new("foobar2"),
Cell::new("bar2"),
Cell::new("foo2")]));
table
}
#[test]
fn from() {
assert_eq!(test_table().to_string().replace("\r\n", "\n"),
Table::from_csv_string(CSV_S)
.unwrap()
.to_string()
.replace("\r\n", "\n"));
}
#[test]
fn to() {
assert_eq!(test_table().to_csv(Vec::new()).unwrap().as_string(), CSV_S);
}
#[test]
fn trans() {
assert_eq!(Table::from_csv_string(test_table()
.to_csv(Vec::new())
.unwrap()
.as_string())
.unwrap()
.to_string()
.replace("\r\n", "\n"),
test_table().to_string().replace("\r\n", "\n"));
}
#[test]
fn extend_table() {
let mut table = Table::new();
table.add_row(Row::new(vec![Cell::new("ABC"), Cell::new("DEFG"), Cell::new("HIJKLMN")]));
table.extend(vec![vec!["A", "B", "C"]]);
let t2 = table.clone();
table.extend(t2.rows);
assert_eq!(table.get_row(1).unwrap().get_cell(2).unwrap().get_content(), "C");
assert_eq!(table.get_row(2).unwrap().get_cell(1).unwrap().get_content(), "DEFG");
}
}
}