asciiflow2/js-lib/state.js

245 lines
6.9 KiB
JavaScript
Raw Normal View History

/**
2014-01-19 14:23:17 +00:00
* Holds the entire state of the diagram as a 2D array of cells
* and provides methods to modify the current state.
2014-01-12 10:37:38 +00:00
*
* @constructor
*/
2014-01-09 20:18:46 +00:00
ascii.State = function() {
/** @type {Array.<Array.<ascii.Cell>>} */
this.cells = new Array(MAX_GRID_WIDTH);
2014-01-21 22:34:56 +00:00
/** @type {Array.<ascii.MappedCell>} */
this.scratchCells = new Array();
2014-01-21 22:34:56 +00:00
/** @type {Array.<Array.<ascii.MappedValue>>} */
this.undoStates = new Array();
for (var i = 0; i < this.cells.length; i++) {
this.cells[i] = new Array(MAX_GRID_HEIGHT);
for (var j = 0; j < this.cells[i].length; j++) {
2014-01-09 20:18:46 +00:00
this.cells[i][j] = new ascii.Cell();
}
}
};
2014-01-22 22:32:16 +00:00
/**
* This clears the entire state, but is undoable.
*/
ascii.State.prototype.clear = function() {
2014-01-22 22:32:16 +00:00
for (var i = 0; i < this.cells.length; i++) {
for (var j = 0; j < this.cells[i].length; j++) {
var position = new ascii.Vector(i, j);
if(this.cells[i][j].getRawValue() != null) {
this.drawValue(new ascii.Vector(i, j), ERASE_CHAR);
}
}
}
this.commitDraw();
};
2014-01-12 10:37:38 +00:00
/**
* Returns the cell at the given coordinates.
*
* @param {ascii.Vector} vector
2014-01-12 10:48:39 +00:00
* @return {ascii.Cell}
2014-01-12 10:37:38 +00:00
*/
ascii.State.prototype.getCell = function(vector) {
return this.cells[vector.x][vector.y];
};
2014-01-12 11:09:55 +00:00
/**
* Sets the cells value at the given position. Probably shouldn't
* be used directly in many cases. Used drawValue instead.
2014-01-12 11:09:55 +00:00
*
* @param {ascii.Vector} position
* @param {?string} value
2014-01-12 11:09:55 +00:00
*/
ascii.State.prototype.setValue = function(position, value) {
this.getCell(position).value = value;
};
2014-01-19 14:23:17 +00:00
/**
* Sets the cells scratch (uncommitted) value at the given position.
*
* @param {ascii.Vector} position
* @param {?string} value
*/
ascii.State.prototype.drawValue = function(position, value) {
var cell = this.getCell(position);
2014-01-21 22:34:56 +00:00
this.scratchCells.push(new ascii.MappedCell(position, cell));
cell.scratchValue = value;
};
/**
* Sets the cells scratch (uncommitted) value at the given position
* iff the value is different to what it already is.
*
* @param {ascii.Vector} position
* @param {?string} value
*/
ascii.State.prototype.drawValueIncremental = function(position, value) {
if (this.getCell(position).getRawValue() != value) {
this.drawValue(position, value);
}
};
/**
* Clears the current drawing scratchpad.
*/
ascii.State.prototype.clearDraw = function() {
for (var i in this.scratchCells) {
2014-01-21 22:34:56 +00:00
this.scratchCells[i].cell.scratchValue = null;
}
this.scratchCells.length = 0;
};
/**
* Returns the draw value of a cell at the given position.
*
* @param {ascii.Vector} position
* @return {?string}
*/
ascii.State.prototype.getDrawValue = function(position) {
var cell = this.getCell(position);
2014-01-19 14:23:17 +00:00
var value = cell.scratchValue != null ? cell.scratchValue : cell.value;
if (value != SPECIAL_VALUE) {
return value;
}
// Magic time.
var context = this.getContext(position);
2014-01-19 14:23:17 +00:00
if (context.left && context.right && !context.up && !context.down) {
return SPECIAL_LINE_H;
}
if (!context.left && !context.right && context.up && context.down) {
return SPECIAL_LINE_V;
}
if (context.left && context.right && context.up && context.down) {
return SPECIAL_LINE_H;
}
return SPECIAL_VALUE;
};
/**
* @param {ascii.Vector} position
* @return {ascii.CellContext}
*/
ascii.State.prototype.getContext = function(position) {
2014-01-19 14:23:17 +00:00
var left = this.getCell(position.add(new ascii.Vector(-1, 0))).isSpecial();
var right = this.getCell(position.add(new ascii.Vector(1, 0))).isSpecial();
var up = this.getCell(position.add(new ascii.Vector(0, -1))).isSpecial();
var down = this.getCell(position.add(new ascii.Vector(0, 1))).isSpecial();
return new ascii.CellContext(left, right, up, down);
};
/**
* Ends the current draw, commiting anything currently drawn the scratchpad.
2014-01-21 22:34:56 +00:00
* @param {boolean=} opt_skipSave
*/
2014-01-21 22:34:56 +00:00
ascii.State.prototype.commitDraw = function(opt_skipSave) {
var oldValues = [];
// Dedupe the scratch values, or this causes havoc for history management.
var positions = this.scratchCells.map(function (value) {
return value.position.x.toString() + value.position.y.toString();
});
var scratchCellsUnique = this.scratchCells.filter(function (value, index, arr) {
return positions.indexOf(positions[index]) == index;
});
this.scratchCells.length = 0;
for (var i in scratchCellsUnique) {
var position = scratchCellsUnique[i].position;
var cell = scratchCellsUnique[i].cell;
// Push the effective old value unto the array.
oldValues.push(new ascii.MappedValue(position, cell.value != null ? cell.value : ' '));
var newValue = cell.getRawValue();
2014-01-22 22:32:16 +00:00
if (newValue == ERASE_CHAR || newValue == ' ') {
newValue = null;
}
2014-01-21 22:34:56 +00:00
cell.scratchValue = null;
cell.value = newValue;
}
// Don't save a new state if we are undoing an old one.
if (!opt_skipSave && oldValues.length > 0) {
2014-01-22 21:28:15 +00:00
// If we have too many undo states, clear one out.
if(this.undoStates.length > MAX_UNDO) {
this.undoStates.shift();
}
2014-01-21 22:34:56 +00:00
this.undoStates.push(oldValues);
}
};
2014-01-21 23:31:58 +00:00
/**
* Undoes the last committed state.
*/
2014-01-21 22:34:56 +00:00
ascii.State.prototype.undo = function() {
if (this.undoStates.length == 0) { return; }
2014-01-21 23:31:58 +00:00
2014-01-21 22:34:56 +00:00
var lastState = this.undoStates.pop();
for (var i in lastState) {
var mappedValue = lastState[i];
this.drawValue(mappedValue.position, mappedValue.value);
}
this.commitDraw(true);
};
2014-01-21 23:31:58 +00:00
/**
* Outputs the entire contents of the diagram as text.
* @return {string}
*/
ascii.State.prototype.outputText = function() {
// Find the first/last cells in the diagram so we don't output everything.
var start = new ascii.Vector(Number.MAX_VALUE, Number.MAX_VALUE);
var end = new ascii.Vector(-1, -1);
for (var i = 0; i < this.cells.length; i++) {
for (var j = 0; j < this.cells[i].length; j++) {
if (this.cells[i][j].getRawValue() != null) {
if (i < start.x) { start.x = i; }
if (j < start.y) { start.y = j; }
if (i > end.x) { end.x = i; }
if (j > end.y) { end.y = j; }
}
}
}
if (end.x < 0) { return '' };
var output = '';
for (var j = start.y; j <= end.y; j++) {
var line = '';
for (var i = start.x; i <= end.x; i++) {
var val = this.getDrawValue(new ascii.Vector(i, j));
line += (val == null ? ' ' : val);
}
// Trim end whitespace.
output += line.replace('\\s+$/g', '') + '\n';
}
return output;
};
2014-01-22 22:32:16 +00:00
/**
* Loads the given text into the diagram starting at the given offset.
*/
ascii.State.prototype.fromText = function(value, offset) {
var lines = value.split('\n');
for (var j = 0; j < lines.length; j++) {
var line = lines[j];
for (var i = 0; i < line.length; i++) {
var char = line.charAt(i);
// Convert special output back to special chars.
// TODO: This is a horrible hack, need to handle multiple special chars
// correctly and preserve them through line drawing etc.
if (char == SPECIAL_LINE_H || char == SPECIAL_LINE_V) {
char = SPECIAL_VALUE;
}
this.drawValue(new ascii.Vector(i, j).add(offset), char);
2014-01-22 22:32:16 +00:00
}
}
this.commitDraw();
};