tabled 0.20.0

An easy to use library for pretty print tables of Rust `struct`s and `enum`s.
Documentation
use crate::{
    grid::config::{ColoredConfig, SpannedConfig},
    grid::records::{ExactRecords, Records, RecordsMut, Resizable},
    settings::TableOption,
};

/// A horizontal/column span from 0 to a count rows.
#[derive(Debug)]
pub struct HorizontalPanel<S> {
    text: S,
    row: usize,
}

impl<S> HorizontalPanel<S> {
    /// Creates a new horizontal panel.
    pub fn new(row: usize, text: S) -> Self {
        Self { row, text }
    }
}

impl<S, R, D> TableOption<R, ColoredConfig, D> for HorizontalPanel<S>
where
    S: AsRef<str>,
    R: Records + ExactRecords + Resizable + RecordsMut<String>,
{
    fn change(self, records: &mut R, cfg: &mut ColoredConfig, _: &mut D) {
        let count_rows = records.count_rows();
        let count_cols = records.count_columns();

        if self.row > count_rows {
            return;
        }

        let is_intersect_vertical_span = (0..records.count_columns())
            .any(|col| cfg.is_cell_covered_by_row_span((self.row, col).into()));
        if is_intersect_vertical_span {
            return;
        }

        move_rows_aside(records, self.row);
        move_row_spans(cfg, self.row);

        let text = self.text.as_ref().to_owned();
        records.set((self.row, 0).into(), text);

        cfg.set_column_span((self.row, 0).into(), count_cols);
    }
}

fn move_rows_aside<R: ExactRecords + Resizable>(records: &mut R, row: usize) {
    records.push_row();

    let count_rows = records.count_rows();

    let shift_count = count_rows - row;
    for i in 1..shift_count {
        let row = count_rows - i;
        records.swap_row(row, row - 1);
    }
}

fn move_row_spans(cfg: &mut SpannedConfig, target_row: usize) {
    for (p, span) in cfg.get_column_spans() {
        if p.row < target_row {
            continue;
        }

        cfg.set_column_span(p, 1);
        cfg.set_column_span(p + (1, 0), span);
    }

    for (p, span) in cfg.get_row_spans() {
        if p.row < target_row {
            continue;
        }

        cfg.set_row_span(p, 1);
        cfg.set_row_span(p + (1, 0), span);
    }
}