asciiflow2/js-lib/controller.js

263 lines
7.0 KiB
JavaScript
Raw Normal View History

/**
* Different modes of control.
* @const
*/
var Mode = {
2014-02-23 19:06:09 +00:00
NONE: 0,
DRAG: 1,
DRAW: 2
};
/**
* Handles user input events and modifies state.
2014-01-19 14:23:17 +00:00
*
* @constructor
2014-01-12 10:37:38 +00:00
* @param {ascii.View} view
* @param {ascii.State} state
*/
ascii.Controller = function(view, state) {
/** @type {ascii.View} */ this.view = view;
/** @type {ascii.State} */ this.state = state;
2014-01-19 14:23:17 +00:00
/** @type {ascii.DrawFunction} */ this.drawFunction =
new ascii.DrawBox(state);
/** @type {number} */ this.mode = Mode.NONE;
/** @type {ascii.Vector} */ this.dragOrigin;
/** @type {ascii.Vector} */ this.dragOriginCell;
this.installBindings();
};
2014-01-12 10:37:38 +00:00
/**
* @param {ascii.Vector} position
*/
ascii.Controller.prototype.startDraw = function(position) {
this.mode = Mode.DRAW;
this.drawFunction.start(this.view.screenToCell(position));
2014-02-23 19:06:09 +00:00
};
/**
* @param {ascii.Vector} position
*/
ascii.Controller.prototype.startDrag = function(position) {
this.mode = Mode.DRAG;
this.dragOrigin = position;
this.dragOriginCell = this.view.offset;
2014-02-23 19:06:09 +00:00
};
2014-01-08 23:06:08 +00:00
2014-01-12 10:37:38 +00:00
/**
* @param {ascii.Vector} position
*/
ascii.Controller.prototype.handleMove = function(position) {
var moveCell = this.view.screenToCell(position);
2014-01-20 21:36:58 +00:00
// First move event, make sure we don't blow up here.
if (this.lastMoveCell == null) {
this.lastMoveCell = moveCell;
}
// Update the cursor pointer, depending on the draw function.
if (!moveCell.equals(this.lastMoveCell)) {
this.view.canvas.style.cursor = this.drawFunction.getCursor(moveCell);
}
// In drawing mode, so pass the mouse move on, but remove duplicates.
if (this.mode == Mode.DRAW && !moveCell.equals(this.lastMoveCell)) {
this.drawFunction.move(moveCell);
}
// Drag in progress, update the view origin.
if (this.mode == Mode.DRAG) {
this.view.setOffset(this.dragOriginCell.add(
this.dragOrigin
2014-01-12 10:37:38 +00:00
.subtract(position)
2014-01-12 11:09:55 +00:00
.scale(1 / this.view.zoom)));
}
this.lastMoveCell = moveCell;
2014-01-08 23:06:08 +00:00
};
2014-02-23 19:06:09 +00:00
/**
* Ends the current operation.
*/
ascii.Controller.prototype.endAll = function() {
if (this.mode == Mode.DRAW) {
this.drawFunction.end();
}
// Cleanup state.
this.mode = Mode.NONE;
this.dragOrigin = null;
this.dragOriginCell = null;
2014-01-12 14:38:56 +00:00
this.lastMoveCell = null;
};
2014-01-12 10:37:38 +00:00
/**
* Installs input bindings for common use cases devices.
2014-01-12 10:37:38 +00:00
*/
ascii.Controller.prototype.installBindings = function() {
var controller = this;
$(window).resize(function(e) { controller.view.resizeCanvas() });
$('#draw-tools > button.tool').click(function(e) {
$('#text-tool-widget').hide(0);
this.handleDrawButton(e.target.id);
}.bind(this));
2014-01-19 14:23:17 +00:00
$('#file-tools > button.tool').click(function(e) {
this.handleFileButton(e.target.id);
2014-01-19 14:23:17 +00:00
}.bind(this));
$('button.close-dialog-button').click(function(e) {
$('.dialog').removeClass('visible');
2014-01-21 23:31:58 +00:00
}.bind(this));
2014-01-22 22:32:16 +00:00
$('#import-submit-button').click(function(e) {
this.state.clear();
2014-01-22 22:32:16 +00:00
this.state.fromText($('#import-area').val(),
this.view.screenToCell(new ascii.Vector(
2014-03-23 19:55:46 +00:00
this.view.canvas.width / 2,
this.view.canvas.height / 2)));
2014-04-07 22:28:20 +00:00
this.state.commitDraw();
2014-01-22 22:32:16 +00:00
$('#import-area').val('');
2014-03-26 19:53:06 +00:00
$('.dialog').removeClass('visible');
2014-01-22 22:32:16 +00:00
}.bind(this));
$('#use-lines-button').click(function(e) {
2014-03-26 19:53:06 +00:00
$('.dialog').removeClass('visible');
this.view.setUseLines(true);
}.bind(this));
$('#use-ascii-button').click(function(e) {
2014-03-26 19:53:06 +00:00
$('.dialog').removeClass('visible');
this.view.setUseLines(false);
}.bind(this));
$(window).keypress(function(e) {
this.handleKeyPress(e);
}.bind(this));
$(window).keydown(function(e) {
this.handleKeyDown(e);
}.bind(this));
// Bit of a hack, just triggers the text tool to get a new value.
$('#text-tool-input, #freeform-tool-input').keyup(function(){
this.drawFunction.handleKey('');
}.bind(this));
$('#text-tool-input, #freeform-tool-input').change(function(){
this.drawFunction.handleKey('');
}.bind(this));
$('#text-tool-close').click(function(){
$('#text-tool-widget').hide();
this.state.commitDraw();
}.bind(this));
2014-01-19 23:21:53 +00:00
};
2014-01-19 14:23:17 +00:00
2014-01-19 23:21:53 +00:00
/**
* Handles the buttons in the UI.
* @param {string} id The ID of the element clicked.
*/
ascii.Controller.prototype.handleDrawButton = function(id) {
$('#draw-tools > button.tool').removeClass('active');
2014-01-22 22:32:16 +00:00
$('#' + id).toggleClass('active');
$('.dialog').removeClass('visible');
2014-01-20 21:36:58 +00:00
2014-01-19 23:21:53 +00:00
// Install the right draw tool based on button pressed.
if (id == 'box-button') {
this.drawFunction = new ascii.DrawBox(this.state);
}
if (id == 'line-button') {
this.drawFunction = new ascii.DrawLine(this.state, false);
}
if (id == 'arrow-button') {
this.drawFunction = new ascii.DrawLine(this.state, true);
2014-01-19 23:21:53 +00:00
}
if (id == 'freeform-button') {
this.drawFunction = new ascii.DrawFreeform(this.state, "X");
2014-01-19 23:21:53 +00:00
}
if (id == 'erase-button') {
this.drawFunction = new ascii.DrawErase(this.state);
2014-01-19 23:21:53 +00:00
}
if (id == 'move-button') {
2014-01-19 14:23:17 +00:00
this.drawFunction = new ascii.DrawMove(this.state);
2014-01-19 23:21:53 +00:00
}
if (id == 'text-button') {
this.drawFunction = new ascii.DrawText(this.state, this.view);
}
2014-04-07 22:28:20 +00:00
if (id == 'select-button') {
this.drawFunction = new ascii.DrawSelect(this.state);
}
this.state.commitDraw();
this.view.canvas.focus();
};
/**
* Handles the buttons in the UI.
* @param {string} id The ID of the element clicked.
*/
ascii.Controller.prototype.handleFileButton = function(id) {
$('.dialog').removeClass('visible');
$('#' + id + '-dialog').toggleClass('visible');
if (id == 'import-button') {
$('#import-area').val('');
$('#import-area').focus();
}
2014-01-21 23:31:58 +00:00
if (id == 'export-button') {
$('#export-area').val(this.state.outputText());
$('#export-area').select();
2014-01-21 23:31:58 +00:00
}
if (id == 'clear-button') {
this.state.clear();
}
if (id == 'undo-button') {
this.state.undo();
}
2014-03-22 22:30:08 +00:00
if (id == 'redo-button') {
this.state.redo();
}
};
/**
* Handles key presses.
* @param {Object} event
*/
ascii.Controller.prototype.handleKeyPress = function(event) {
2014-02-09 18:34:48 +00:00
if (!event.ctrlKey && !event.metaKey && event.keyCode != 13) {
this.drawFunction.handleKey(String.fromCharCode(event.keyCode));
}
};
/**
* Handles key down events.
* @param {Object} event
*/
ascii.Controller.prototype.handleKeyDown = function(event) {
// Override some special characters so that they can be handled in one place.
var specialKeyCode = null;
if (event.ctrlKey || event.metaKey) {
if (event.keyCode == 67) { specialKeyCode = KEY_COPY; }
if (event.keyCode == 86) { specialKeyCode = KEY_PASTE; }
2014-02-23 17:13:22 +00:00
if (event.keyCode == 90) { this.state.undo(); }
if (event.keyCode == 89) { this.state.redo(); }
if (event.keyCode == 88) { specialKeyCode = KEY_CUT; }
}
if (event.keyCode == 8) { specialKeyCode = KEY_BACKSPACE; }
if (event.keyCode == 13) { specialKeyCode = KEY_RETURN; }
if (event.keyCode == 38) { specialKeyCode = KEY_UP; }
if (event.keyCode == 40) { specialKeyCode = KEY_DOWN; }
if (event.keyCode == 37) { specialKeyCode = KEY_LEFT; }
if (event.keyCode == 39) { specialKeyCode = KEY_RIGHT; }
if (specialKeyCode != null) {
2014-01-21 23:31:58 +00:00
//event.preventDefault();
//event.stopPropagation();
this.drawFunction.handleKey(specialKeyCode);
}
};