blob: aef456cbfd85c880acce1852e3dc6a77b9c4e252 [file] [log] [blame]
// Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:math' as math;
/// Print the given [table].
void printTable(List<List<String>> table) {
var buffer = StringBuffer();
buffer.writeTable(table);
print(buffer.toString());
}
/// Return an iterable that will produce all of the integer values between
/// [first] and [last] inclusive.
Iterable<int> range(int first, int last) sync* {
for (var i = first; i < last; i++) {
yield i;
}
}
/// Given a [table] represented as a list of rows, right justify all of the
/// cells in the given [column].
void rightJustifyColumn(List<List<String>> table, int column) {
var width = 0;
for (var i = 0; i < table.length; i++) {
var row = table[i];
width = math.max(width, row[column].length);
}
for (var i = 0; i < table.length; i++) {
var row = table[i];
var cellValue = row[column];
var length = cellValue.length;
if (length < width) {
var padding = ' ' * (width - length);
row[column] = '$padding$cellValue';
}
}
}
/// Given a [table] represented as a list of rows, right justify all of the
/// cells in the given [columns].
void rightJustifyColumns(List<List<String>> table, Iterable<int> columns) {
for (var column in columns) {
rightJustifyColumn(table, column);
}
}
extension OutputUtilities on StringSink {
/// Write the given [table].
///
/// The table is represented as a list or rows, where each row is a list of
/// the contents of the cells in that row.
///
/// Throws an [ArgumentError] if the table is empty or if the rows do not
/// contain the same number of cells.
void writeTable(List<List<String>> table) {
var columnWidths = _computeColumnWidths(table);
for (var row in table) {
var lastNonEmpty = row.length - 1;
while (lastNonEmpty > 0) {
if (row[lastNonEmpty].isNotEmpty) {
break;
}
lastNonEmpty--;
}
for (var i = 0; i <= lastNonEmpty; i++) {
var cellContent = row[i];
var columnWidth = columnWidths[i];
var padding = columnWidth - cellContent.length;
write(cellContent);
if (i < lastNonEmpty) {
write(' ' * (padding + 2));
}
}
writeln();
}
}
/// Return the minimum widths for each of the columns in the given [table].
///
/// The table is represented as a list or rows, where each row is a list of
/// the contents of the cells in that row.
///
/// Throws an [ArgumentError] if the table is empty or if the rows do not
/// contain the same number of cells.
List<int> _computeColumnWidths(List<List<String>> table) {
if (table.isEmpty) {
throw ArgumentError('table cannot be empty');
}
var columnCount = table[0].length;
if (columnCount == 0) {
throw ArgumentError('rows cannot be empty');
}
var columnWidths = List<int>.filled(columnCount, 0);
for (var row in table) {
var rowLength = row.length;
if (rowLength > 0) {
if (rowLength != columnCount) {
throw ArgumentError(
'non-empty rows must contain the same number of columns');
}
for (var i = 0; i < rowLength; i++) {
var cellWidth = row[i].length;
columnWidths[i] = math.max(columnWidths[i], cellWidth);
}
}
}
return columnWidths;
}
}