1// Copyright (c) 2012 The Chromium Authors. All rights reserved. 2// Use of this source code is governed by a BSD-style license that can be 3// found in the LICENSE file. 4 5/** 6 * TablePrinter is a helper to format a table as ASCII art or an HTML table. 7 * 8 * Usage: call addRow() and addCell() repeatedly to specify the data. 9 * 10 * addHeaderCell() can optionally be called to specify header cells for a 11 * single header row. The header row appears at the top of an HTML formatted 12 * table, and uses thead and th tags. In ascii tables, the header is separated 13 * from the table body by a partial row of dashes. 14 * 15 * setTitle() can optionally be used to set a title that is displayed before 16 * the header row. In HTML tables, it uses the title class and in ascii tables 17 * it's between two rows of dashes. 18 * 19 * Once all the fields have been input, call toText() to format it as text or 20 * toHTML() to format it as HTML. 21 */ 22var TablePrinter = (function() { 23 'use strict'; 24 25 /** 26 * @constructor 27 */ 28 function TablePrinter() { 29 this.rows_ = []; 30 this.hasHeaderRow_ = false; 31 this.title_ = null; 32 // Number of cells automatically added at the start of new rows. 33 this.newRowCellIndent_ = 0; 34 } 35 36 TablePrinter.prototype = { 37 /** 38 * Sets the number of blank cells to add after each call to addRow. 39 */ 40 setNewRowCellIndent: function(newRowCellIndent) { 41 this.newRowCellIndent_ = newRowCellIndent; 42 }, 43 44 /** 45 * Starts a new row. 46 */ 47 addRow: function() { 48 this.rows_.push([]); 49 for (var i = 0; i < this.newRowCellIndent_; ++i) 50 this.addCell(''); 51 }, 52 53 /** 54 * Adds a column to the current row, setting its value to cellText. 55 * 56 * @return {!TablePrinterCell} the cell that was added. 57 */ 58 addCell: function(cellText) { 59 var r = this.rows_[this.rows_.length - 1]; 60 var cell = new TablePrinterCell(cellText); 61 r.push(cell); 62 return cell; 63 }, 64 65 /** 66 * Sets the title displayed at the top of a table. Titles are optional. 67 */ 68 setTitle: function(title) { 69 this.title_ = title; 70 }, 71 72 /** 73 * Adds a header row, if not already present, and adds a new column to it, 74 * setting its contents to |headerText|. 75 * 76 * @return {!TablePrinterCell} the cell that was added. 77 */ 78 addHeaderCell: function(headerText) { 79 // Insert empty new row at start of |rows_| if currently no header row. 80 if (!this.hasHeaderRow_) { 81 this.rows_.splice(0, 0, []); 82 this.hasHeaderRow_ = true; 83 } 84 var cell = new TablePrinterCell(headerText); 85 this.rows_[0].push(cell); 86 return cell; 87 }, 88 89 /** 90 * Returns the maximum number of columns this table contains. 91 */ 92 getNumColumns: function() { 93 var numColumns = 0; 94 for (var i = 0; i < this.rows_.length; ++i) { 95 numColumns = Math.max(numColumns, this.rows_[i].length); 96 } 97 return numColumns; 98 }, 99 100 /** 101 * Returns the cell at position (rowIndex, columnIndex), or null if there is 102 * no such cell. 103 */ 104 getCell_: function(rowIndex, columnIndex) { 105 if (rowIndex >= this.rows_.length) 106 return null; 107 var row = this.rows_[rowIndex]; 108 if (columnIndex >= row.length) 109 return null; 110 return row[columnIndex]; 111 }, 112 113 /** 114 * Returns true if searchString can be found entirely within a cell. 115 * Case insensitive. 116 * 117 * @param {string} string String to search for, must be lowercase. 118 * @return {boolean} True if some cell contains searchString. 119 */ 120 search: function(searchString) { 121 var numColumns = this.getNumColumns(); 122 for (var r = 0; r < this.rows_.length; ++r) { 123 for (var c = 0; c < numColumns; ++c) { 124 var cell = this.getCell_(r, c); 125 if (!cell) 126 continue; 127 if (cell.text.toLowerCase().indexOf(searchString) != -1) 128 return true; 129 } 130 } 131 return false; 132 }, 133 134 /** 135 * Prints a formatted text representation of the table data to the 136 * node |parent|. |spacing| indicates number of extra spaces, if any, 137 * to add between columns. 138 */ 139 toText: function(spacing, parent) { 140 var pre = addNode(parent, 'pre'); 141 var numColumns = this.getNumColumns(); 142 143 // Figure out the maximum width of each column. 144 var columnWidths = []; 145 columnWidths.length = numColumns; 146 for (var i = 0; i < numColumns; ++i) 147 columnWidths[i] = 0; 148 149 // If header row is present, temporarily add a spacer row to |rows_|. 150 if (this.hasHeaderRow_) { 151 var headerSpacerRow = []; 152 for (var c = 0; c < numColumns; ++c) { 153 var cell = this.getCell_(0, c); 154 if (!cell) 155 continue; 156 var spacerStr = makeRepeatedString('-', cell.text.length); 157 headerSpacerRow.push(new TablePrinterCell(spacerStr)); 158 } 159 this.rows_.splice(1, 0, headerSpacerRow); 160 } 161 162 var numRows = this.rows_.length; 163 for (var c = 0; c < numColumns; ++c) { 164 for (var r = 0; r < numRows; ++r) { 165 var cell = this.getCell_(r, c); 166 if (cell && !cell.allowOverflow) { 167 columnWidths[c] = Math.max(columnWidths[c], cell.text.length); 168 } 169 } 170 } 171 172 var out = []; 173 174 // Print title, if present. 175 if (this.title_) { 176 var titleSpacerStr = makeRepeatedString('-', this.title_.length); 177 out.push(titleSpacerStr); 178 out.push('\n'); 179 out.push(this.title_); 180 out.push('\n'); 181 out.push(titleSpacerStr); 182 out.push('\n'); 183 } 184 185 // Print each row. 186 var spacingStr = makeRepeatedString(' ', spacing); 187 for (var r = 0; r < numRows; ++r) { 188 for (var c = 0; c < numColumns; ++c) { 189 var cell = this.getCell_(r, c); 190 if (cell) { 191 // Pad the cell with spaces to make it fit the maximum column width. 192 var padding = columnWidths[c] - cell.text.length; 193 var paddingStr = makeRepeatedString(' ', padding); 194 195 if (cell.alignRight) 196 out.push(paddingStr); 197 if (cell.link) { 198 // Output all previous text, and clear |out|. 199 addTextNode(pre, out.join('')); 200 out = []; 201 202 var linkNode = addNodeWithText(pre, 'a', cell.text); 203 linkNode.href = cell.link; 204 } else { 205 out.push(cell.text); 206 } 207 if (!cell.alignRight) 208 out.push(paddingStr); 209 out.push(spacingStr); 210 } 211 } 212 out.push('\n'); 213 } 214 215 // Remove spacer row under the header row, if one was added. 216 if (this.hasHeaderRow_) 217 this.rows_.splice(1, 1); 218 219 addTextNode(pre, out.join('')); 220 }, 221 222 /** 223 * Adds a new HTML table to the node |parent| using the specified style. 224 */ 225 toHTML: function(parent, style) { 226 var numRows = this.rows_.length; 227 var numColumns = this.getNumColumns(); 228 229 var table = addNode(parent, 'table'); 230 table.setAttribute('class', style); 231 232 var thead = addNode(table, 'thead'); 233 var tbody = addNode(table, 'tbody'); 234 235 // Add title, if needed. 236 if (this.title_) { 237 var tableTitleRow = addNode(thead, 'tr'); 238 var tableTitle = addNodeWithText(tableTitleRow, 'th', this.title_); 239 tableTitle.colSpan = numColumns; 240 tableTitle.classList.add('title'); 241 } 242 243 // Fill table body, adding header row first, if needed. 244 for (var r = 0; r < numRows; ++r) { 245 var cellType; 246 var row; 247 if (r == 0 && this.hasHeaderRow_) { 248 row = addNode(thead, 'tr'); 249 cellType = 'th'; 250 } else { 251 row = addNode(tbody, 'tr'); 252 cellType = 'td'; 253 } 254 for (var c = 0; c < numColumns; ++c) { 255 var cell = this.getCell_(r, c); 256 if (cell) { 257 var tableCell = addNode(row, cellType, cell.text); 258 if (cell.alignRight) 259 tableCell.alignRight = true; 260 // If allowing overflow on the rightmost cell of a row, 261 // make the cell span the rest of the columns. Otherwise, 262 // ignore the flag. 263 if (cell.allowOverflow && !this.getCell_(r, c + 1)) 264 tableCell.colSpan = numColumns - c; 265 if (cell.link) { 266 var linkNode = addNodeWithText(tableCell, 'a', cell.text); 267 linkNode.href = cell.link; 268 } else { 269 addTextNode(tableCell, cell.text); 270 } 271 } 272 } 273 } 274 return table; 275 } 276 }; 277 278 /** 279 * Links are only used in HTML tables. 280 */ 281 function TablePrinterCell(value) { 282 this.text = '' + value; 283 this.link = null; 284 this.alignRight = false; 285 this.allowOverflow = false; 286 } 287 288 return TablePrinter; 289})(); 290