/* eslint-disable linebreak-style */
/* global Spyral, DataTable */
import Chart from './chart.js';
import Util from './util.js';
/**
* The Spyral.Table class in Spyral provides convenience functions for working with tabular
* data.
*
* There are several ways of initializing a Table, here are some of them:
*
* Provide an array of data with 3 rows:
*
* let table = createTable([1,2,3]);
*
*
* Provide a nested array of data with multiple rows:
*
* let table = createTable([[1,2],[3,4]]);
*
* Same nested array, but with a second argument specifying headers
*
* let table = createTable([[1,2],[3,4]], {headers: ["one","two"]});
*
* Create table with comma-separated values:
*
* let table = createTable("one,two\\n1,2\\n3,4");
*
* Create table with tab-separated values
*
* let table = createTable("one\\ttwo\\n1\\t2\\n3\\t4");
*
* Create table with array of objects
*
* let table = createTable([{one:1,two:2},{one:3,two:4}]);
*
* It's also possible simple to create a sorted frequency table from an array of values:
*
* let table = createTable(["one","two","one"], {count: "vertical", headers: ["Term","Count"]})
*
* Working with a Corpus is easy. For instance, we can create a table from the top terms:
*
* loadCorpus("austen").terms({limit:500, stopList: 'auto'}).then(terms => {
* return createTable(terms);
* })
*
* Similarly, we could create a frequency table from the first 1,000 words of the corpus:
*
* loadCorpus("austen").words({limit:1000, docIndex: 0, stopList: 'auto'}).then(words => {
* return createTable(words, {count: "vertical"});
* });
*
* Some of the configuration options are as follows:
*
* * **format**: especially for forcing csv or tsv when the data is a string
* * **hasHeaders**: determines if data has a header row (usually determined automatically)
* * **headers**: a Array of Strings that serve as headers for the table
* * **count**: forces Spyral to create a sorted frequency table from an Array of data, this can be set to "vertical" if the counts are shown vertically or set to true if the counts are shown horizontally
*
* Tables are convenient in Spyral because you can simply show them to preview a version in HTML.
*
* @memberof Spyral
* @class
*/
class Table {
/**
* The Table config object
* @typedef {Object} Spyral.Table~TableConfig
* @property {string} format The format of the provided data, either "tsv" or "csv"
* @property {(Object|Array)} headers The table headers
* @property {boolean} hasHeaders True if the headers are the first item in the data
* @property {string} count Specify "vertical" or "horizontal" to create a table of unique item counts in the provided data
*/
/**
* Create a new Table
* @constructor
* @param {(Object|Array|String|Number)} data An array of data or a string with CSV or TSV.
* @param {Spyral.Table~TableConfig} config an Object for configuring the table initialization
* @returns {Spyral.Table}
*/
constructor(data, config, ...other) {
this._rows = [];
this._headers = {};
this._rowKeyColumnIndex = 0;
if (Util.isPromise(data)) {
throw new Error('Data cannot be a Promise');
}
// we have a configuration object followed by values: create({headers: []}, 1,2,3) …
if (data && typeof data === 'object' && (typeof config === 'string' || typeof config === 'number' || Array.isArray(config))) {
data.rows = [config].concat(other).filter(v => v!==undefined);
config = undefined;
}
// we have a simple variable set of arguments: create(1,2,3) …
if (arguments.length>0 && Array.from(arguments).every(a => a!==undefined && !Array.isArray(a) && typeof a !== 'object')) {
data = [data,config].concat(other).filter(v => v!==undefined);
config = undefined;
}
// could be CSV or TSV
if (Array.isArray(data) && data.length===1 && typeof data[0] === 'string' && (data[0].indexOf(',')>-1 || data[0].indexOf('\t')>-1)) {
data = data[0];
}
// first check if we have a string that might be delimited data
if (data && (typeof data === 'string' || typeof data ==='number')) {
if (typeof data === 'number') {data = String(data);} // convert to string for split
let rows = [];
let format = config && 'format' in config ? config.format : undefined;
data.split(/(\r\n|[\n\v\f\r\x85\u2028\u2029])+/g).forEach((line,i) => {
if (line.trim().length>0) {
let values;
if ((format && format==='tsv') || line.indexOf('\t')>-1) {
values = line.split(/\t/);
} else if ((format && format==='csv') || line.indexOf(',')>-1) {
values = parseCsvLine(line);
} else {
values = [line];
}
// if we can't find any config information for headers then we try to guess
// if the first line doesn't have any numbers - this heuristic may be questionable
if (i===0 && values.every(v => isNaN(v)) &&
((typeof config !== 'object') || (typeof config === 'object' && !('hasHeaders' in config) && !('headers' in config)))) {
this.setHeaders(values);
} else {
rows.push(values.map(v => isNaN(v) ? v : Number(v)));
}
}
});
data = rows;
}
if (data && Array.isArray(data)) {
if (config) {
if (Array.isArray(config)) {
this.setHeaders(config);
} else if (typeof config === 'object') {
if ('headers' in config) {
this.setHeaders(config.headers);
} else if ('hasHeaders' in config && config.hasHeaders) {
this.setHeaders(data.shift());
}
}
}
if (config && 'count' in config && config.count) {
let freqs = Table.counts(data);
if (config.count==='vertical') {
for (let item in freqs) {
this.addRow(item, freqs[item]);
}
this.rowSort((a,b) => Table.cmp(b[1],a[1]));
} else {
this._headers = []; // reset and use the terms as headers
this.addRow(freqs);
this.columnSort((a,b) => Table.cmp(this.cell(0,b),this.cell(0,a)));
}
} else {
this.addRows(data);
}
} else if (data && typeof data === 'object') {
if ('headers' in data && Array.isArray(data.headers)) {
this.setHeaders(data.headers);
} else if ('hasHeaders' in data && 'rows' in data) {
this.setHeaders(data.rows.shift());
}
if ('rows' in data && Array.isArray(data.rows)) {
this.addRows(data.rows);
}
if ('rowKeyColumn' in data) {
if (typeof data.rowKeyColumn === 'number') {
if (data.rowKeyColumn < this.columns()) {
this._rowKeyColumnIndex = data.rowKeyColumn;
} else {
throw new Error('The rowKeyColumn value is higher than the number headers designated: '+data.rowKeyColum);
}
} else if (typeof data.rowKeyColumn === 'string') {
if (data.rowKeyColumn in this._headers) {
this._rowKeyColumnIndex = this._headers[data.rowKeyColumn];
} else {
throw new Error('Unable to find column designated by rowKeyColumn: '+data.rowKeyColumn);
}
}
}
}
}
/**
* Set the headers for the Table
* @param {(Object|Array)} data
* @returns {Spyral.Table}
*/
setHeaders(data) {
if (data && Array.isArray(data)) {
data.forEach(h => this.addColumn(h), this);
} else if (typeof data === 'object') {
if (this.columns()===0 || Object.keys(data).length===this.columns()) {
this._headers = data;
} else {
throw new Error('The number of columns don\'t match: ');
}
} else {
throw new Error('Unrecognized argument for headers, it should be an array or an object.'+data);
}
return this;
}
/**
* Add rows to the Table
* @param {Array} data
* @returns {Spyral.Table}
*/
addRows(data) {
data.forEach(row => this.addRow(row), this);
return this;
}
/**
* Add a row to the Table
* @param {(Array|Object)} data
* @returns {Spyral.Table}
*/
addRow(data, ...other) {
// we have multiple arguments, so call again as an array
if (other.length>0) {
return this.addRow([data].concat(other));
}
this.setRow(this.rows(), data, true);
return this;
}
/**
* Set a row
* @param {(number|string)} ind The row index
* @param {(Object|Array)} data
* @param {boolean} create
* @returns {Spyral.Table}
*/
setRow(ind, data, create) {
let rowIndex = this.getRowIndex(ind, create);
if (rowIndex>=this.rows() && !create) {
throw new Error('Attempt to set row values for a row that does note exist: '+ind+'. Maybe use addRow() instead?');
}
// we have a simple array, so we'll just push to the rows
if (data && Array.isArray(data)) {
if (data.length>this.columns()) {
if (create) {
for (let i = this.columns(); i<data.length; i++) {
this.addColumn();
}
} else {
throw new Error('The row that you\'ve created contains more columns than the current table. Maybe use addColunm() first?');
}
}
data.forEach((d,i) => this.setCell(rowIndex, i, d), this);
}
// we have an object so we'll use the headers
else if (typeof data === 'object') {
for (let column in data) {
if (!this.hasColumn(column)) {
//
}
this.setCell(rowIndex, column, data[column]);
}
}
else if (this.columns()<2 && create) { // hopefully some scalar value
if (this.columns()===0) {
this.addColumn(); // create first column if it doesn't exist
}
this.setCell(rowIndex,0,data);
} else {
throw new Error('setRow() expects an array or an object, maybe setCell()?');
}
return this;
}
/**
* Set a column
* @param {(number|string)} ind The column index
* @param {(Object|Array)} data
* @param {boolean} create
* @returns {Spyral.Table}
*/
setColumn(ind, data, create) {
let columnIndex = this.getColumnIndex(ind, create);
if (columnIndex>=this.columns() && !create) {
throw new Error('Attempt to set column values for a column that does note exist: '+ind+'. Maybe use addColumn() instead?');
}
// we have a simple array, so we'll just push to the rows
if (data && Array.isArray(data)) {
data.forEach((d,i) => this.setCell(i, columnIndex, d, create), this);
}
// we have an object so we'll use the headers
else if (typeof data === 'object') {
for (let row in data) {
this.setCell(row, columnIndex, data[row], create);
}
}
// hope we have a scalar value to assign to the first row
else {
this.setCell(0,columnIndex,data, create);
}
return this;
}
/**
* Add to or set a cell value
* @param {(number|string)} row The row index
* @param {(number|string)} column The column index
* @param {number} value The value to set/add
* @param {boolean} overwrite True to set, false to add to current value
*/
updateCell(row, column, value, overwrite) {
let rowIndex = this.getRowIndex(row, true);
let columnIndex = this.getColumnIndex(column, true);
let val = this.cell(rowIndex, columnIndex);
this._rows[rowIndex][columnIndex] = val && !overwrite ? val+value : value;
return this;
}
/**
* Get the value of a cell
* @param {(number|string)} rowInd The row index
* @param {(number|string)} colInd The column index
* @returns {number}
*/
cell(rowInd, colInd) {
return this._rows[this.getRowIndex(rowInd)][this.getColumnIndex(colInd)];
}
/**
* Set the value of a cell
* @param {(number|string)} row The row index
* @param {(number|string)} column The column index
* @param {number} value The value to set
* @returns {Spyral.Table}
*/
setCell(row, column, value) {
this.updateCell(row,column,value,true);
return this;
}
/**
* Get (and create) the row index
* @param {(number|string)} ind The index
* @param {boolean} create
* @returns {number}
*/
getRowIndex(ind, create) {
if (typeof ind === 'number') {
if (ind < this._rows.length) {
return ind;
} else if (create) {
this._rows[ind] = Array(this.columns());
return ind;
}
throw new Error('The requested row does not exist: '+ind);
} else if (typeof ind === 'string') {
let row = this._rows.findIndex(r => r[this._rowKeyColumnIndex] === ind, this);
if (row>-1) {return row;}
else if (create) {
let arr = Array(this.columns());
arr[this._rowKeyColumnIndex] = ind;
this.addRow(arr);
return this.rows();
}
else {
throw new Error('Unable to find the row named '+ind);
}
}
throw new Error('Please provide a valid row (number or named row)');
}
/**
* Get (and create) the column index
* @param {(number|string)} ind The index
* @param {boolean} create
* @returns {number}
*/
getColumnIndex(ind, create) {
if (typeof ind === 'number') {
if (ind < this.columns()) {
return ind;
} else if (create) {
this.addColumn(ind);
return ind;
}
throw new Error('The requested column does not exist: '+ind);
} else if (typeof ind === 'string') {
if (ind in this._headers) {
return this._headers[ind];
} else if (create) {
this.addColumn({header: ind});
return this._headers[ind];
}
throw new Error('Unable to find column named '+ind);
}
throw new Error('Please provide a valid column (number or named column)');
}
/**
* Add a column (at the specified index)
* @param {(Object|String)} config
* @param {(number|string)} ind
* @returns {Spyral.Table}
*/
addColumn(config, ind) {
// determine col
let col = this.columns(); // default
if (config && typeof config === 'string') {col=config;}
else if (config && (typeof config === 'object') && ('header' in config)) {col = config.header;}
else if (ind!==undefined) {col=ind;}
// check if it exists
if (col in this._headers) {
throw new Error('This column exists already: '+config.header);
}
// add column
let colIndex = this.columns();
this._headers[col] = colIndex;
// determine data
let data = [];
if (config && typeof config === 'object' && 'rows' in config) {data=config.rows;}
else if (Array.isArray(config)) {data = config;}
// make sure we have enough rows for the new data
let columns = this.columns();
while (this._rows.length<data.length) {
this._rows[this._rows.length] = new Array(columns);
}
this._rows.forEach((r,i) => r[colIndex] = data[i]);
return this;
}
/**
* This function returns different values depending on the arguments provided.
* When there are no arguments, it returns the number of rows in this table.
* When the first argument is the boolean value `true` all rows are returned.
* When the first argument is a an array then the rows corresponding to the row
* indices or names are returned. When all arguments except are numbers or strings
* then each of those is returned.
* @param {(Boolean|Array|Number|String)} [inds]
* @param {(Object|Number|String)} [config]
* @returns {(Number|Array)}
*/
rows(inds, config, ...other) {
// return length
if (inds===undefined) {
return this._rows.length;
}
let rows = [];
let asObj = (config && typeof config === 'object' && config.asObj) ||
(other.length>0 && typeof other[other.length-1] === 'object' && other[other.length-1].asObj);
// return all
if (typeof inds === 'boolean' && inds) {
rows = this._rows.map((r,i) => this.row(i, asObj));
}
// return specified rows
else if (Array.isArray(inds)) {
rows = inds.map(ind => this.row(ind));
}
// return specified rows as varargs
else if (typeof inds === 'number' || typeof inds === 'string') {
[inds, config, ...other].every(i => {
if (typeof i === 'number' || typeof i === 'string') {
rows.push(this.row(i, asObj));
return true;
} else {
return false;
}
});
if (other.length>0) { // when config is in last position
if (typeof other[other.length-1] === 'object') {
config = other[other.length-1];
}
}
}
// zip if requested
if (config && typeof config === 'object' && 'zip' in config && config.zip) {
if (rows.length<2) {throw new Error('Only one row available, can\'t zip');}
return Table.zip(rows);
}
else {
return rows;
}
}
/**
* Get the specified row
* @param {(number|string)} ind
* @param {boolean} [asObj]
* @returns {(Object|Number|String)}
*/
row(ind, asObj) {
let row = this._rows[this.getRowIndex(ind)];
if (asObj) {
let obj = {};
for (let key in this._headers) {
obj[key] = row[this._headers[key]];
}
return obj;
} else {
return row;
}
}
/**
* This function returns different values depending on the arguments provided.
* When there are no arguments, it returns the number of columns in this table.
* When the first argument is the boolean value `true` all columns are returned.
* When the first argument is a number a slice of the columns is returned and if
* the second argument is a number it is treated as the length of the slice to
* return (note that it isn't the `end` index like with Array.slice()).
* @param {(Boolean|Array|Number|String)} [inds]
* @param {(Object|Number|String)} [config]
* @returns {(Number|Array)}
*/
columns(inds, config, ...other) {
// return length
if (inds===undefined) {
return Object.keys(this._headers).length;
}
let columns = [];
let asObj = (config && typeof config === 'object' && config.asObj) ||
(other.length>0 && typeof other[other.length-1] === 'object' && other[other.length-1].asObj);
// return all columns
if (typeof inds === 'boolean' && inds) {
for (let i=0, len=this.columns(); i<len; i++) {
columns.push(this.column(i, asObj));
}
}
// return specified columns
else if (Array.isArray(inds)) {
inds.forEach(i => columns.push(this.column(i, asObj)), this);
}
else if (typeof inds === 'number' || typeof inds === 'string') {
[inds, config, ...other].every(i => {
if (typeof i === 'number' || typeof i === 'string') {
columns.push(this.column(i, asObj));
return true;
} else {
return false;
}
});
if (other.length>0) { // when config is in last position
if (typeof other[other.length-1] === 'object') {
config = other[other.length-1];
}
}
}
if (config && typeof config === 'object' && 'zip' in config && config.zip) {
if (columns.length<2) {throw new Error('Only one column available, can\'t zip');}
return Table.zip(columns);
}
else {
return columns;
}
}
/**
* Get the specified column
* @param {(number|string)} ind
* @param {boolean} [asObj]
* @returns {(Object|Number|String)}
*/
column(ind, asObj) {
let column = this.getColumnIndex(ind);
let data = this._rows.forEach(r => r[column]); // TODO
if (asObj) {
let obj = {};
this._rows.forEach(r => {
obj[r[this._rowKeyColumnIndex]] = r[column];
});
return obj;
} else {
return this._rows.map(r => r[column]);
}
}
/**
* Get the specified header
* @param {(number|string)} ind
* @returns {(number|string)}
*/
header(ind) {
let keys = Object.keys(this._headers);
let i = this.getColumnIndex(ind);
return keys[keys.findIndex(k => i===this._headers[k])];
}
/**
* This function returns different values depending on the arguments provided.
* When there are no arguments, it returns the number of headers in this table.
* When the first argument is the boolean value `true` all headers are returned.
* When the first argument is a number a slice of the headers is returned.
* When the first argument is an array the slices specified in the array are returned.
* @param {(Boolean|Array|Number|String)} inds
* @returns {(Number|Array)}
*/
headers(inds, ...other) {
// return length
if (inds===undefined) {
return Object.keys(this._headers).length;
}
// let headers = [];
// return all
if (typeof inds === 'boolean' && inds) {
inds = Array(Object.keys(this._headers).length).fill().map((_,i) => i);
}
// return specified rows
if (Array.isArray(inds)) {
return inds.map(i => this.header(i));
}
// return specified rows as varargs
else if (typeof inds === 'number' || typeof inds === 'string') {
return [inds, ...other].map(i => this.header(i));
}
}
/**
* Does the specified column exist
* @param {(number|string)} ind
* @returns {(number|string)}
*/
hasColumn(ind) {
return ind in this._headers;
}
/**
* Runs the specified function on each row.
* The function is passed the row and the row index.
* @param {Function} fn
*/
forEach(fn) {
this._rows.forEach((r,i) => fn(r,i));
}
/**
* Get the minimum value in the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowMin(ind) {
return Math.min.apply(null, this.row(ind));
}
/**
* Get the maximum value in the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowMax(ind) {
return Math.max.apply(null, this.row(ind));
}
/**
* Get the minimum value in the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnMin(ind) {
return Math.min.apply(null, this.column(ind));
}
/**
* Get the maximum value in the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnMax(ind) {
return Math.max.apply(null, this.column(ind));
}
/**
* Get the sum of the values in the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowSum(ind) {
return Table.sum(this.row(ind));
}
/**
* Get the sum of the values in the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnSum(ind) {
return Table.sum(this.column(ind));
}
/**
* Get the mean of the values in the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowMean(ind) {
return Table.mean(this.row(ind));
}
/**
* Get the mean of the values in the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnMean(ind) {
return Table.mean(this.column(ind));
}
/**
* Get the count of each unique value in the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowCounts(ind) {
return Table.counts(this.row(ind));
}
/**
* Get the count of each unique value in the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnCounts(ind) {
return Table.counts(this.column(ind));
}
/**
* Get the rolling mean for the specified row
* @param {(number|string)} ind
* @param {number} neighbors
* @param {boolean} overwrite
* @returns {Array}
*/
rowRollingMean(ind, neighbors, overwrite) {
let means = Table.rollingMean(this.row(ind), neighbors);
if (overwrite) {
this.setRow(ind, means);
}
return means;
}
/**
* Get the rolling mean for the specified column
* @param {(number|string)} ind
* @param {number} neighbors
* @param {boolean} overwrite
* @returns {Array}
*/
columnRollingMean(ind, neighbors, overwrite) {
let means = Table.rollingMean(this.column(ind), neighbors);
if (overwrite) {
this.setColumn(ind, means);
}
return means;
}
/**
* Get the variance for the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowVariance(ind) {
return Table.variance(this.row(ind));
}
/**
* Get the variance for the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnVariance(ind) {
return Table.variance(this.column(ind));
}
/**
* Get the standard deviation for the specified row
* @param {(number|string)} ind
* @returns {number}
*/
rowStandardDeviation(ind) {
return Table.standardDeviation(this.row(ind));
}
/**
* Get the standard deviation for the specified column
* @param {(number|string)} ind
* @returns {number}
*/
columnStandardDeviation(ind) {
return Table.standardDeviation(this.column(ind));
}
/**
* Get the z scores for the specified row
* @param {(number|string)} ind
* @returns {Array}
*/
rowZScores(ind) {
return Table.zScores(this.row(ind));
}
/**
* Get the z scores for the specified column
* @param {(number|string)} ind
* @returns {Array}
*/
columnZScores(ind) {
return Table.zScores(this.column(ind));
}
/**
* TODO
* Sort the specified rows
* @returns {Spyral.Table}
*/
rowSort(inds, config) {
// no inds, use all columns
if (inds===undefined) {
inds = Array(this.columns()).fill().map((_,i) => i);
}
// wrap a single index as array
if (typeof inds === 'string' || typeof inds === 'number') {
inds = [inds];
}
if (Array.isArray(inds)) {
return this.rowSort((a,b) => {
let ind;
for (let i=0, len=inds.length; i<len; i++) {
ind = this.getColumnIndex(inds[i]);
if (a!==b) {
if (typeof a[ind] === 'string' && typeof b[ind] === 'string') {
return a[ind].localeCompare(b[ind]);
} else {
return a[ind]-b[ind];
}
}
}
return 0;
}, config);
}
if (typeof inds === 'function') {
this._rows.sort((a,b) => {
if (config && 'asObject' in config && config.asObject) {
let c = {};
for (let k in this._headers) {
c[k] = a[this._headers[k]];
}
let d = {};
for (let k in this._headers) {
d[k] = b[this._headers[k]];
}
return inds.apply(this, [c,d]);
} else {
return inds.apply(this, [a,b]);
}
});
if (config && 'reverse' in config && config.reverse) {
this._rows.reverse(); // in place
}
}
return this;
}
/**
* TODO
* Sort the specified columns
* @returns {Spyral.Table}
*/
columnSort(inds, config) {
// no inds, use all columns
if (inds===undefined) {
inds = Array(this.columns()).fill().map((_,i) => i);
}
// wrap a single index as array
if (typeof inds === 'string' || typeof inds === 'number') {
inds = [inds];
}
if (Array.isArray(inds)) {
// convert to column names
let headers = inds.map(ind => this.header(ind));
// make sure we have all columns
Object.keys(this._headers).forEach(h => {
if (!headers.includes(h)) {headers.push(h);}
});
// sort names alphabetically
headers.sort((a,b) => a.localeCompare(b));
// reorder by columns
this._rows = this._rows.map((_,i) => headers.map(h => this.cell(i,h)));
this._headers = {};
headers.forEach((h,i) => this._headers[h]=i);
}
if (typeof inds === 'function') {
let headers = Object.keys(this._headers);
if (config && 'asObject' in headers && headers.asObject) {
headers = headers.map((h,i) => {
return {header: h, data: this._rows.map((r,j) => this.cell(i,j))};
});
}
headers.sort((a,b) => {
return inds.apply(this, [a,b]);
});
headers = headers.map(h => typeof h === 'object' ? h.header : h); // convert back to string
// make sure we have all keys
Object.keys(this._headers).forEach(k => {
if (headers.indexOf(k)===-1) {headers.push(k);}
});
this._rows = this._rows.map((_,i) => headers.map(h => this.cell(i,h)));
this._headers = {};
headers.forEach((h,i) => this._headers[h]=i);
}
}
/**
* Get a CSV representation of the Table
* @param {Object} [config]
* @returns {string}
*/
toCsv(config) {
const cell = function(c) {
let quote = /"/g;
return typeof c === 'string' && (c.indexOf(',')>-1 || c.indexOf('"')>-1) ? '"'+c.replace(quote,'"')+'"' : c;
};
return (config && 'noHeaders' in config && config.noHeaders ? '' : this.headers(true).map(h => cell(h)).join(',') + '\n') +
this._rows.map(row => row.map(c => cell(c)).join(',')).join('\n');
}
/**
* Get a TSV representation of the Table
* @param {Object} [config]
* @returns {string}
*/
toTsv(config) {
return config && 'noHeaders' in config && config.noHeaders ? '' : this.headers(true).join('\t') + '\n' +
this._rows.map(row => row.join('\t')).join('\n');
}
/**
* Set the target's contents to an HTML representation of the Table
* @param {(Function|String|Object)} target
* @param {Object} [config]
* @returns {Spyral.Table}
*/
html(target, config) {
let html = this.toString(config);
if (typeof target === 'function') {
target(html);
} else {
if (typeof target === 'string') {
target = document.querySelector(target);
if (!target) {
throw 'Unable to find specified target: '+target;
}
}
if (typeof target === 'object' && 'innerHTML' in target) {
target.innerHTML = html;
}
}
return this;
}
/**
* Same as {@link toString}.
*/
toHtml(config={}) {
return this.toString(config);
}
/**
* Displays an interactive table using [DataTables]{@link https://datatables.net/}
* @param {HTMLElement} [target]
* @param {Object} config
* @returns {DataTable}
*/
toDataTable(target, config={}) {
if (Util.isNode(target) === false && typeof target === 'object') {
config = target;
target = undefined;
}
if (target === undefined) {
if (typeof Spyral !== 'undefined' && Spyral.Notebook) {
target = Spyral.Notebook.getTarget();
} else {
target = document.createElement('div');
document.body.appendChild(target);
}
} else {
if (Util.isNode(target) && target.isConnected === false) {
throw new Error('The target node does not exist within the document.');
}
}
target = document.body.appendChild(target);
this.html(target, config);
let dataTable = new DataTable(target.firstElementChild);
return dataTable;
}
/**
* Get an HTML representation of the Table
* @param {Object} [config]
* @returns {string}
*/
toString(config={}) {
if (typeof config === 'number') {
config = {limit: config};
}
if ('top' in config && !('limit' in config)) {
config.limit = config.top;
}
if ('limit' in config && !('bottom' in config)) {
config.bottom = 0;
}
if ('bottom' in config && !('limit' in config)) {
config.limit=0;
}
return '<table'+('id' in config ? ' id="'+config.id+'" ' : ' ')+'class="voyantTable">' +
((config && 'caption' in config && typeof config.caption === 'string') ?
'<caption>'+config.caption+'</caption>' : '') +
((config && 'noHeaders' in config && config.noHeaders) ? '' : ('<thead><tr>'+this.headers(true).map(c => '<th>'+c+'</th>').join('')+'</tr></thead>'))+
'<tbody>'+
this._rows.filter((row,i,arr) => ((!('limit' in config) || i<config.limit) || (!('bottom' in config) || i > arr.length-1 - config.bottom)))
.map(row => '<tr>'+row.map(c => '<td>'+(typeof c === 'number' ? c.toLocaleString() : c)+'</td>').join('')+'</tr>').join('') +
'</tbody></table>';
}
/**
* Show a chart representing the Table
* @param {(String|HTMLElement)} [target]
* @param {HighchartsConfig} [config]
* @returns {Highcharts.Chart}
*/
chart(target = undefined, config = {}) {
[target, config] = Chart._handleTargetAndConfig(target, config);
config.chart = config.chart || {};
let columnsCount = this.columns();
let rowsCount = this.rows();
let headers = this.headers(config.columns ? config.columns : true);
let isHeadersCategories = headers.every(h => isNaN(h));
if (isHeadersCategories) {
Chart._setDefaultChartType(config, 'column');
}
// set categories if not set
config.xAxis = config.xAxis || {};
config.xAxis.categories = config.xAxis.categories || headers;
// start filling in series
config.series = config.series || [];
if (!('seriesFrom' in config)) {
// one row, so let's take series from rows
if (rowsCount === 1) {
config.dataFrom = config.dataFrom || 'rows';
} else if (columnsCount === 1 || (!('dataFrom' in config))) {
config.dataFrom = config.dataFrom || 'columns';
}
}
if ('dataFrom' in config) {
if (config.dataFrom === 'rows') {
config.data = {rows:[]};
config.data.rows.push(headers);
config.data.rows = config.data.rows.concat(this.rows(true));
} else if (config.dataFrom === 'columns') {
config.data = {columns:[]};
config.data.columns = config.data.columns.concat(this.columns(true));
if (config.data.columns.length === headers.length) {
headers.forEach((h, i) => {
config.data.columns[i].splice(0, 0, h);
});
}
}
} else if ('seriesFrom' in config) {
if (config.seriesFrom === 'rows') {
this.rows(config.rows ? config.rows : true).forEach((row, i) => {
config.series[i] = config.series[i] || {};
config.series[i].data = headers.map(h => this.cell(i, h));
});
} else if (config.seriesFrom === 'columns') {
this.columns(config.columns ? config.columns : true).forEach((col, i) => {
config.series[i] = config.series[i] || {};
config.series[i].data = [];
for (let r = 0; r < rowsCount; r++) {
config.series[i].data.push(this.cell(r, i));
}
});
}
}
delete config.dataFrom;
delete config.seriesFrom;
return Chart.create(target, config);
}
/**
* Create a new Table
* @param {(Object|Array|String|Number)} data
* @param {Spyral.Table~TableConfig} config
* @returns {Spyral.Table}
* @static
*/
static create(data, config, ...other) {
return new Table(data, config, ...other);
}
/**
* Fetch a Table from a source
* @param {(String|Request)} input
* @param {Object} api
* @param {Object} config
* @returns {Promise}
* @static
*/
static fetch(input, api, config) {
return new Promise((resolve, reject) => {
window.fetch(input, api).then(response => {
if (!response.ok) {throw new Error(response.status + ' ' + response.statusText);}
response.text().then(text => {
resolve(Table.create(text, config || api));
});
});
});
}
/**
* Get the count of each unique value in the data
* @param {Array} data
* @returns {Object}
* @static
*/
static counts(data) {
let vals = {};
data.forEach(v => v in vals ? vals[v]++ : vals[v]=1);
return vals;
}
/**
* Compare two values
* @param {(number|string)} a
* @param {(number|string)} b
* @returns {number}
* @static
*/
static cmp(a, b) {
return typeof a === 'string' && typeof b === 'string' ? a.localeCompare(b) : a-b;
}
/**
* Get the sum of the provided values
* @param {Array} data
* @returns {number}
* @static
*/
static sum(data) {
return data.reduce((a,b) => a+b, 0);
}
/**
* Get the mean of the provided values
* @param {Array} data
* @returns {number}
* @static
*/
static mean(data) {
return Table.sum(data) / data.length;
}
/**
* Get rolling mean for the provided values
* @param {Array} data
* @param {number} neighbors
* @returns {Array}
* @static
*/
static rollingMean(data, neighbors) {
// https://stackoverflow.com/questions/41386083/plot-rolling-moving-average-in-d3-js-v4/41388581#41387286
return data.map((val, idx, arr) => {
let start = Math.max(0, idx - neighbors), end = idx + neighbors;
let subset = arr.slice(start, end + 1);
let sum = subset.reduce((a,b) => a + b);
return sum / subset.length;
});
}
/**
* Get the variance for the provided values
* @param {Array} data
* @returns {number}
* @static
*/
static variance(data) {
let m = Table.mean(data);
return Table.mean(data.map(num => Math.pow(num-m, 2)));
}
/**
* Get the standard deviation for the provided values
* @param {Array} data
* @returns {number}
* @static
*/
static standardDeviation(data) {
return Math.sqrt(Table.variance(data));
}
/**
* Get the z scores for the provided values
* @param {Array} data
* @returns {Array}
* @static
*/
static zScores(data) {
let m = Table.mean(data);
let s = Table.standardDeviation(data);
return data.map(num => (num-m) / s);
}
/**
* Perform a zip operation of the provided arrays. Learn more about zip on [Wikipedia]{@link https://en.wikipedia.org/wiki/Convolution_%28computer_science%29}.
* @param {Array} data
* @returns {Array}
* @static
*/
static zip(...data) {
// we have a single nested array, so let's recall with flattened arguments
if (data.length===1 && Array.isArray(data) && data.every(d => Array.isArray(d))) {
return Table.zip.apply(null, ...data);
}
// allow arrays to be of different lengths
let len = Math.max.apply(null, data.map(d => d.length));
return new Array(len).fill().map((_,i) => data.map(d => d[i]));
}
}
// this seems like a good balance between a built-in flexible parser and a heavier external parser
// https://lowrey.me/parsing-a-csv-file-in-es6-javascript/
const regex = /(?!\s*$)\s*(?:'([^'\\]*(?:\\[\S\s][^'\\]*)*)'|"([^"\\]*(?:\\[\S\s][^"\\]*)*)"|([^,'"\s\\]*(?:\s+[^,'"\s\\]+)*))\s*(?:,|$)/g;
function parseCsvLine(line) {
let arr = [];
line.replace(regex, (m0, m1, m2, m3) => {
if (m1 !== undefined) {
arr.push(m1.replace(/\\'/g, '\''));
} else if (m2 !== undefined) {
arr.push(m2.replace(/\\"/g, '"'));
} else if (m3 !== undefined) {
arr.push(m3);
}
return '';
});
if (/,\s*$/.test(line)) {arr.push('');}
return arr;
}
export default Table;