128 lines
3.9 KiB
JavaScript
128 lines
3.9 KiB
JavaScript
6 years ago
|
'use strict';
|
||
|
|
||
|
var BaseRenderer = require('power-assert-renderer-base');
|
||
|
var inherits = require('util').inherits;
|
||
|
var forEach = require('core-js/library/fn/array/for-each');
|
||
|
var stringifier = require('stringifier');
|
||
|
var stringWidth = require('power-assert-util-string-width');
|
||
|
var assign = require('core-js/library/fn/object/assign');
|
||
|
var defaultOptions = require('./lib/default-options');
|
||
|
|
||
|
/**
|
||
|
* options.stringify [function]
|
||
|
* options.maxDepth [number]
|
||
|
* options.lineSeparator [string]
|
||
|
* options.anonymous [string]
|
||
|
* options.circular [string]
|
||
|
*
|
||
|
* options.widthOf [function]
|
||
|
* options.ambiguousEastAsianCharWidth [number]
|
||
|
*/
|
||
|
function DiagramRenderer (config) {
|
||
|
BaseRenderer.call(this);
|
||
|
this.config = assign({}, defaultOptions(), config);
|
||
|
this.events = [];
|
||
|
if (typeof this.config.stringify === 'function') {
|
||
|
this.stringify = this.config.stringify;
|
||
|
} else {
|
||
|
this.stringify = stringifier(this.config);
|
||
|
}
|
||
|
if (typeof this.config.widthOf === 'function') {
|
||
|
this.widthOf = this.config.widthOf;
|
||
|
} else {
|
||
|
this.widthOf = (this.config.ambiguousEastAsianCharWidth === 1) ? stringWidth.narrow : stringWidth;
|
||
|
}
|
||
|
this.initialVertivalBarLength = 1;
|
||
|
}
|
||
|
inherits(DiagramRenderer, BaseRenderer);
|
||
|
|
||
|
DiagramRenderer.prototype.onStart = function (context) {
|
||
|
this.assertionLine = context.source.content;
|
||
|
this.initializeRows();
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.onData = function (esNode) {
|
||
|
if (!esNode.isCaptured) {
|
||
|
return;
|
||
|
}
|
||
|
this.events.push({value: esNode.value, leftIndex: esNode.range[0]});
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.onEnd = function () {
|
||
|
this.events.sort(rightToLeft);
|
||
|
this.constructRows(this.events);
|
||
|
var _this = this;
|
||
|
forEach(this.rows, function (columns) {
|
||
|
_this.write(columns.join(''));
|
||
|
});
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.initializeRows = function () {
|
||
|
this.rows = [];
|
||
|
for (var i = 0; i <= this.initialVertivalBarLength; i += 1) {
|
||
|
this.addOneMoreRow();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.newRowFor = function (assertionLine) {
|
||
|
return createRow(this.widthOf(assertionLine), ' ');
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.addOneMoreRow = function () {
|
||
|
this.rows.push(this.newRowFor(this.assertionLine));
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.lastRow = function () {
|
||
|
return this.rows[this.rows.length - 1];
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.renderVerticalBarAt = function (columnIndex) {
|
||
|
var i, lastRowIndex = this.rows.length - 1;
|
||
|
for (i = 0; i < lastRowIndex; i += 1) {
|
||
|
this.rows[i].splice(columnIndex, 1, '|');
|
||
|
}
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.renderValueAt = function (columnIndex, dumpedValue) {
|
||
|
var i, width = this.widthOf(dumpedValue);
|
||
|
for (i = 0; i < width; i += 1) {
|
||
|
this.lastRow().splice(columnIndex + i, 1, dumpedValue.charAt(i));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.isOverlapped = function (prevCapturing, nextCaputuring, dumpedValue) {
|
||
|
return (typeof prevCapturing !== 'undefined') && this.startColumnFor(prevCapturing) <= (this.startColumnFor(nextCaputuring) + this.widthOf(dumpedValue));
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.constructRows = function (capturedEvents) {
|
||
|
var that = this;
|
||
|
var prevCaptured;
|
||
|
forEach(capturedEvents, function (captured) {
|
||
|
var dumpedValue = that.stringify(captured.value);
|
||
|
if (that.isOverlapped(prevCaptured, captured, dumpedValue)) {
|
||
|
that.addOneMoreRow();
|
||
|
}
|
||
|
that.renderVerticalBarAt(that.startColumnFor(captured));
|
||
|
that.renderValueAt(that.startColumnFor(captured), dumpedValue);
|
||
|
prevCaptured = captured;
|
||
|
});
|
||
|
};
|
||
|
|
||
|
DiagramRenderer.prototype.startColumnFor = function (captured) {
|
||
|
return this.widthOf(this.assertionLine.slice(0, captured.leftIndex));
|
||
|
};
|
||
|
|
||
|
function createRow (numCols, initial) {
|
||
|
var row = [], i;
|
||
|
for(i = 0; i < numCols; i += 1) {
|
||
|
row[i] = initial;
|
||
|
}
|
||
|
return row;
|
||
|
}
|
||
|
|
||
|
function rightToLeft (a, b) {
|
||
|
return b.leftIndex - a.leftIndex;
|
||
|
}
|
||
|
|
||
|
module.exports = DiagramRenderer;
|