2014-01-08 23:06:08 +00:00
|
|
|
/**
|
2014-01-19 14:23:17 +00:00
|
|
|
* Common classes and constants.
|
2014-01-08 23:06:08 +00:00
|
|
|
*/
|
|
|
|
|
2014-02-09 18:00:55 +00:00
|
|
|
// Define namespace for closure compiler but don't make it a requirement.
|
|
|
|
try {
|
|
|
|
goog.provide('ascii');
|
2014-02-20 00:10:20 +00:00
|
|
|
throw 1;
|
2014-02-09 18:00:55 +00:00
|
|
|
} catch (e) {
|
2014-02-23 19:06:09 +00:00
|
|
|
/** type {Object} */
|
2014-02-20 00:10:20 +00:00
|
|
|
window.ascii = window.ascii || {};
|
2014-02-09 18:00:55 +00:00
|
|
|
}
|
2014-01-19 14:23:17 +00:00
|
|
|
|
2014-01-22 23:20:17 +00:00
|
|
|
/** @const */ var MAX_GRID_WIDTH = 2000;
|
|
|
|
/** @const */ var MAX_GRID_HEIGHT = 600;
|
|
|
|
|
2014-01-19 14:23:17 +00:00
|
|
|
/** @const */ var SPECIAL_VALUE = '+';
|
2014-04-27 12:20:23 +00:00
|
|
|
/** @const */ var ALT_SPECIAL_VALUE = '^';
|
|
|
|
/** @const */ var SPECIAL_ARROW_LEFT = '<';
|
|
|
|
/** @const */ var SPECIAL_ARROW_UP = '^';
|
|
|
|
/** @const */ var SPECIAL_ARROW_RIGHT = '>';
|
|
|
|
/** @const */ var SPECIAL_ARROW_DOWN = 'v';
|
2014-04-27 13:30:54 +00:00
|
|
|
/** @const */ var SPECIAL_VALUES = ['+', '\u2012', '\u2013', '-', '|'];
|
2014-04-27 12:20:23 +00:00
|
|
|
/** @const */ var ALT_SPECIAL_VALUES = ['>', '<', '^', 'v'];
|
|
|
|
/** @const */ var ALL_SPECIAL_VALUES = SPECIAL_VALUES.concat(ALT_SPECIAL_VALUES);
|
2014-01-19 14:23:17 +00:00
|
|
|
|
2014-01-21 22:34:56 +00:00
|
|
|
/** @const */ var MAX_UNDO = 50;
|
|
|
|
|
2014-04-27 12:27:00 +00:00
|
|
|
/** @const */ var SPECIAL_LINE_H = '-';
|
2014-01-19 14:23:17 +00:00
|
|
|
/** @const */ var SPECIAL_LINE_V = '|';
|
|
|
|
|
2014-01-22 21:28:15 +00:00
|
|
|
/** @const */ var ERASE_CHAR = '\u2009';
|
|
|
|
|
2014-03-26 19:48:41 +00:00
|
|
|
/** @const */ var DRAG_LATENCY = 150; // Milliseconds.
|
|
|
|
/** @const */ var DRAG_ACCURACY = 6; // Pixels.
|
2014-01-19 14:23:17 +00:00
|
|
|
|
2014-01-22 23:20:17 +00:00
|
|
|
/** @const */ var CHAR_PIXELS_H = 9;
|
|
|
|
/** @const */ var CHAR_PIXELS_V = 17;
|
2014-01-22 21:07:46 +00:00
|
|
|
|
2014-03-10 22:54:02 +00:00
|
|
|
/** @const */ var RENDER_PADDING_CELLS = 3;
|
2014-01-19 14:23:17 +00:00
|
|
|
|
2014-01-21 21:30:47 +00:00
|
|
|
/** @const */ var KEY_RETURN = '<enter>';
|
|
|
|
/** @const */ var KEY_BACKSPACE = '<backspace>';
|
|
|
|
/** @const */ var KEY_COPY = '<copy>';
|
|
|
|
/** @const */ var KEY_PASTE = '<paste>';
|
|
|
|
/** @const */ var KEY_CUT = '<cut>';
|
|
|
|
/** @const */ var KEY_UP = '<up>';
|
|
|
|
/** @const */ var KEY_DOWN = '<down>';
|
|
|
|
/** @const */ var KEY_LEFT = '<left>';
|
|
|
|
/** @const */ var KEY_RIGHT = '<right>';
|
|
|
|
|
2014-02-20 00:10:20 +00:00
|
|
|
// http://stackoverflow.com/questions/4817029/whats-the-best-way-to-detect-a-touch-screen-device-using-javascript
|
2014-02-23 19:06:09 +00:00
|
|
|
/** @const */ var TOUCH_ENABLED =
|
2014-02-20 00:10:20 +00:00
|
|
|
'ontouchstart' in window ||
|
|
|
|
'onmsgesturechange' in window;
|
|
|
|
|
2014-01-08 23:06:08 +00:00
|
|
|
/**
|
2014-01-19 14:23:17 +00:00
|
|
|
* Stores a 2D vector.
|
|
|
|
*
|
2014-01-08 23:06:08 +00:00
|
|
|
* @constructor
|
2014-01-12 10:37:38 +00:00
|
|
|
* @param {number} x
|
|
|
|
* @param {number} y
|
2014-01-08 23:06:08 +00:00
|
|
|
*/
|
2014-01-11 16:40:01 +00:00
|
|
|
ascii.Vector = function(x, y) {
|
2014-01-08 23:06:08 +00:00
|
|
|
/** type {Number} */ this.x = x;
|
|
|
|
/** type {Number} */ this.y = y;
|
|
|
|
};
|
2014-01-09 20:18:46 +00:00
|
|
|
|
2014-01-12 10:37:38 +00:00
|
|
|
/**
|
|
|
|
* @param {ascii.Vector} other
|
|
|
|
* @return {boolean}
|
|
|
|
*/
|
2014-01-11 16:40:01 +00:00
|
|
|
ascii.Vector.prototype.equals = function(other) {
|
2014-01-12 14:38:56 +00:00
|
|
|
return (other != null) && (this.x == other.x) && (this.y == other.y);
|
2014-01-09 20:18:46 +00:00
|
|
|
};
|
2014-01-11 16:40:01 +00:00
|
|
|
|
2014-01-12 10:37:38 +00:00
|
|
|
/**
|
|
|
|
* @param {ascii.Vector} other
|
2014-01-12 10:48:39 +00:00
|
|
|
* @return {ascii.Vector}
|
2014-01-12 10:37:38 +00:00
|
|
|
*/
|
2014-01-11 16:40:01 +00:00
|
|
|
ascii.Vector.prototype.subtract = function(other) {
|
|
|
|
return new ascii.Vector(this.x - other.x, this.y - other.y);
|
|
|
|
};
|
|
|
|
|
2014-01-12 10:37:38 +00:00
|
|
|
/**
|
|
|
|
* @param {ascii.Vector} other
|
2014-01-12 10:48:39 +00:00
|
|
|
* @return {ascii.Vector}
|
2014-01-12 10:37:38 +00:00
|
|
|
*/
|
2014-01-11 16:40:01 +00:00
|
|
|
ascii.Vector.prototype.add = function(other) {
|
|
|
|
return new ascii.Vector(this.x + other.x, this.y + other.y);
|
|
|
|
};
|
|
|
|
|
2014-01-18 00:28:46 +00:00
|
|
|
/**
|
|
|
|
* @return {ascii.Vector}
|
|
|
|
*/
|
|
|
|
ascii.Vector.prototype.clone = function() {
|
|
|
|
return new ascii.Vector(this.x, this.y);
|
|
|
|
};
|
|
|
|
|
2014-01-11 16:40:01 +00:00
|
|
|
/** @return {number} */
|
|
|
|
ascii.Vector.prototype.length = function() {
|
|
|
|
return Math.sqrt(this.x * this.x + this.y * this.y);
|
|
|
|
};
|
2014-01-11 17:41:44 +00:00
|
|
|
|
2014-01-12 10:37:38 +00:00
|
|
|
/**
|
|
|
|
* @param {number} scale
|
|
|
|
* @return {ascii.Vector}
|
|
|
|
*/
|
2014-01-11 17:41:44 +00:00
|
|
|
ascii.Vector.prototype.scale = function(scale) {
|
|
|
|
return new ascii.Vector(this.x * scale, this.y * scale);
|
|
|
|
};
|
|
|
|
|
2014-04-07 22:28:20 +00:00
|
|
|
/**
|
|
|
|
* Represents a box with normalized position vectors.
|
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
* @param {ascii.Vector} a
|
|
|
|
* @param {ascii.Vector} b
|
|
|
|
*/
|
|
|
|
ascii.Box = function(a, b) {
|
|
|
|
/** type {Number} */ this.startX = Math.min(a.x, b.x);
|
|
|
|
/** type {Number} */ this.startY = Math.min(a.y, b.y);
|
|
|
|
/** type {Number} */ this.endX = Math.max(a.x, b.x);
|
|
|
|
/** type {Number} */ this.endY = Math.max(a.y, b.y);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** @return {ascii.Vector} */
|
|
|
|
ascii.Box.prototype.topLeft = function() {
|
|
|
|
return new ascii.Vector(this.startX, this.startY);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** @return {ascii.Vector} */
|
|
|
|
ascii.Box.prototype.bottomRight = function() {
|
|
|
|
return new ascii.Vector(this.endX, this.endY);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** @return {boolean} */
|
|
|
|
ascii.Box.prototype.contains = function(position) {
|
|
|
|
return position.x >= this.startX && position.x <= this.endX && position.y >= this.startY && position.y <= this.endY;
|
|
|
|
};
|
|
|
|
|
2014-02-20 00:10:20 +00:00
|
|
|
/** @const */ var DIR_LEFT = new ascii.Vector(-1, 0);
|
|
|
|
/** @const */ var DIR_RIGHT = new ascii.Vector(1, 0);
|
|
|
|
/** @const */ var DIR_UP = new ascii.Vector(0, -1);
|
|
|
|
/** @const */ var DIR_DOWN = new ascii.Vector(0, 1);
|
|
|
|
|
2014-02-23 19:06:09 +00:00
|
|
|
/** @const */ var DIRECTIONS = [DIR_LEFT, DIR_RIGHT, DIR_UP, DIR_DOWN];
|
2014-02-20 00:10:20 +00:00
|
|
|
|
2014-01-19 14:23:17 +00:00
|
|
|
/**
|
|
|
|
* An individual cell within the diagram and it's current value.
|
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
*/
|
|
|
|
ascii.Cell = function() {
|
|
|
|
/** @type {?string} */ this.value = null;
|
|
|
|
/** @type {?string} */ this.scratchValue = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** @return {?string} */
|
|
|
|
ascii.Cell.prototype.getRawValue = function() {
|
|
|
|
return (this.scratchValue != null ? this.scratchValue : this.value);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** @return {boolean} */
|
|
|
|
ascii.Cell.prototype.isSpecial = function() {
|
2014-04-27 12:20:23 +00:00
|
|
|
return ALL_SPECIAL_VALUES.indexOf(this.getRawValue()) != -1;
|
2014-01-19 14:23:17 +00:00
|
|
|
};
|
|
|
|
|
2015-03-24 21:45:41 +00:00
|
|
|
/** @return {boolean} */
|
|
|
|
ascii.Cell.prototype.isEmpty = function() {
|
|
|
|
return this.value == null && this.scratchValue == null;
|
|
|
|
};
|
|
|
|
|
2014-01-20 21:58:25 +00:00
|
|
|
/** @return {boolean} */
|
|
|
|
ascii.Cell.prototype.hasScratch = function() {
|
|
|
|
return this.scratchValue != null;
|
|
|
|
};
|
|
|
|
|
2014-01-22 21:28:15 +00:00
|
|
|
/** @return {boolean} */
|
|
|
|
ascii.Cell.prototype.isErase = function() {
|
|
|
|
return this.scratchValue == ERASE_CHAR;
|
|
|
|
};
|
|
|
|
|
2014-01-19 14:23:17 +00:00
|
|
|
/**
|
|
|
|
* The context for a cell, i.e. the status of the cells around it.
|
|
|
|
*
|
|
|
|
* @param {boolean} left
|
|
|
|
* @param {boolean} right
|
|
|
|
* @param {boolean} up
|
|
|
|
* @param {boolean} down
|
|
|
|
* @constructor
|
|
|
|
*/
|
|
|
|
ascii.CellContext = function(left, right, up, down) {
|
|
|
|
/** @type {boolean} */ this.left = left;
|
|
|
|
/** @type {boolean} */ this.right = right;
|
|
|
|
/** @type {boolean} */ this.up = up;
|
|
|
|
/** @type {boolean} */ this.down = down;
|
2014-04-27 13:30:54 +00:00
|
|
|
/** @type {boolean} */ this.leftup = false;
|
|
|
|
/** @type {boolean} */ this.rightup = false;
|
|
|
|
/** @type {boolean} */ this.leftdown = false;
|
|
|
|
/** @type {boolean} */ this.rightdown = false;
|
2014-01-19 14:23:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the total number of surrounding special cells.
|
|
|
|
* @return {number}
|
|
|
|
*/
|
|
|
|
ascii.CellContext.prototype.sum = function() {
|
|
|
|
return this.left + this.right + this.up + this.down;
|
|
|
|
};
|
2014-01-21 22:34:56 +00:00
|
|
|
|
2014-04-27 13:30:54 +00:00
|
|
|
/**
|
|
|
|
* Returns the total number of surrounding special cells.
|
|
|
|
* @return {number}
|
|
|
|
*/
|
|
|
|
ascii.CellContext.prototype.extendedSum = function() {
|
|
|
|
return this.left + this.right + this.up + this.down + this.leftup + this.leftdown + this.rightup + this.rightdown;
|
|
|
|
};
|
|
|
|
|
2014-01-21 22:34:56 +00:00
|
|
|
/**
|
|
|
|
* A pair of a vector and a string value. Used in history management.
|
|
|
|
* @constructor
|
|
|
|
* @struct
|
2014-02-23 19:06:09 +00:00
|
|
|
* @param {ascii.Vector} position
|
2014-04-07 22:28:20 +00:00
|
|
|
* @param {string|null} value
|
2014-01-21 22:34:56 +00:00
|
|
|
*/
|
|
|
|
ascii.MappedValue = function(position, value) {
|
|
|
|
this.position = position;
|
|
|
|
this.value = value;
|
2014-02-23 19:06:09 +00:00
|
|
|
};
|
2014-01-21 22:34:56 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A pair of a vector and a cell. Used in history management.
|
|
|
|
* @constructor
|
|
|
|
* @struct
|
2014-02-23 19:06:09 +00:00
|
|
|
* @param {ascii.Vector} position
|
|
|
|
* @param {ascii.Cell} cell
|
2014-01-21 22:34:56 +00:00
|
|
|
*/
|
|
|
|
ascii.MappedCell = function(position, cell) {
|
|
|
|
this.position = position;
|
|
|
|
this.cell = cell;
|
2014-02-23 19:06:09 +00:00
|
|
|
};
|