2012-09-07 02:18:59 +04:00
|
|
|
(function () {
|
2014-04-17 03:27:49 +04:00
|
|
|
var ns = $.namespace('pskl.service');
|
|
|
|
|
2014-04-19 18:01:51 +04:00
|
|
|
var SNAPSHOT_PERIOD = 50;
|
2014-04-20 15:15:30 +04:00
|
|
|
var LOAD_STATE_INTERVAL = 50;
|
2014-04-17 03:27:49 +04:00
|
|
|
|
2013-09-22 23:02:43 +04:00
|
|
|
ns.HistoryService = function (piskelController) {
|
2013-09-29 01:52:51 +04:00
|
|
|
this.piskelController = piskelController;
|
2014-04-17 03:27:49 +04:00
|
|
|
this.stateQueue = [];
|
|
|
|
this.currentIndex = -1;
|
2014-04-23 01:57:30 +04:00
|
|
|
this.saveState__b = this.onSaveStateEvent.bind(this);
|
2014-04-17 03:27:49 +04:00
|
|
|
|
2014-04-20 15:15:30 +04:00
|
|
|
this.lastLoadState = -1;
|
2012-09-11 01:26:12 +04:00
|
|
|
};
|
2012-09-07 02:18:59 +04:00
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
ns.HistoryService.SNAPSHOT = 'SNAPSHOT';
|
|
|
|
ns.HistoryService.REPLAY = 'REPLAY';
|
2013-09-29 01:52:51 +04:00
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
ns.HistoryService.prototype.init = function () {
|
2014-04-17 03:27:49 +04:00
|
|
|
$.subscribe(Events.PISKEL_SAVE_STATE, this.saveState__b);
|
2013-11-20 02:46:33 +04:00
|
|
|
|
|
|
|
pskl.app.shortcutService.addShortcut('ctrl+Z', this.undo.bind(this));
|
|
|
|
pskl.app.shortcutService.addShortcut('ctrl+Y', this.redo.bind(this));
|
2014-04-23 01:57:30 +04:00
|
|
|
|
|
|
|
this.saveState({
|
|
|
|
type : ns.HistoryService.SNAPSHOT
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
ns.HistoryService.prototype.onSaveStateEvent = function (evt, stateInfo) {
|
|
|
|
this.saveState(stateInfo);
|
2013-08-10 14:11:16 +04:00
|
|
|
};
|
2012-09-07 02:18:59 +04:00
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
ns.HistoryService.prototype.saveState = function (stateInfo) {
|
2014-04-17 03:27:49 +04:00
|
|
|
this.stateQueue = this.stateQueue.slice(0, this.currentIndex + 1);
|
|
|
|
this.currentIndex = this.currentIndex + 1;
|
|
|
|
|
|
|
|
var state = {
|
2014-04-23 01:57:30 +04:00
|
|
|
action : stateInfo,
|
2014-04-17 03:27:49 +04:00
|
|
|
frameIndex : this.piskelController.currentFrameIndex,
|
|
|
|
layerIndex : this.piskelController.currentLayerIndex
|
|
|
|
};
|
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
if (stateInfo.type === ns.HistoryService.SNAPSHOT || this.currentIndex % SNAPSHOT_PERIOD === 0) {
|
|
|
|
state.piskel = this.piskelController.serialize(true);
|
2014-04-17 03:27:49 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
this.stateQueue.push(state);
|
2012-09-11 01:26:12 +04:00
|
|
|
};
|
2012-09-07 02:18:59 +04:00
|
|
|
|
2012-09-16 15:10:05 +04:00
|
|
|
ns.HistoryService.prototype.undo = function () {
|
2014-04-20 15:15:30 +04:00
|
|
|
this.loadState(this.currentIndex - 1);
|
2012-09-11 01:26:12 +04:00
|
|
|
};
|
|
|
|
|
2012-09-16 15:10:05 +04:00
|
|
|
ns.HistoryService.prototype.redo = function () {
|
2014-04-20 15:15:30 +04:00
|
|
|
this.loadState(this.currentIndex + 1);
|
2014-04-17 03:27:49 +04:00
|
|
|
};
|
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
ns.HistoryService.prototype.isLoadStateAllowed_ = function (index) {
|
|
|
|
var timeOk = (Date.now() - this.lastLoadState) > LOAD_STATE_INTERVAL;
|
|
|
|
var indexInRange = index >= 0 && index < this.stateQueue.length;
|
|
|
|
return timeOk && indexInRange;
|
|
|
|
};
|
|
|
|
|
|
|
|
ns.HistoryService.prototype.getPreviousSnapshotIndex_ = function (index) {
|
2014-06-27 04:08:00 +04:00
|
|
|
var counter = 0;
|
2014-04-23 01:57:30 +04:00
|
|
|
while (this.stateQueue[index] && !this.stateQueue[index].piskel) {
|
|
|
|
index = index - 1;
|
2014-06-27 04:08:00 +04:00
|
|
|
if(++counter > 2*SNAPSHOT_PERIOD) {
|
|
|
|
break;
|
|
|
|
}
|
2014-04-23 01:57:30 +04:00
|
|
|
}
|
|
|
|
return index;
|
|
|
|
};
|
|
|
|
|
2014-04-17 03:27:49 +04:00
|
|
|
ns.HistoryService.prototype.loadState = function (index) {
|
2014-06-27 04:08:00 +04:00
|
|
|
try {
|
|
|
|
if (this.isLoadStateAllowed_(index)) {
|
|
|
|
this.lastLoadState = Date.now();
|
|
|
|
|
|
|
|
var snapshotIndex = this.getPreviousSnapshotIndex_(index);
|
|
|
|
if (snapshotIndex < 0) {
|
|
|
|
throw 'Could not find previous SNAPSHOT saved in history stateQueue';
|
|
|
|
}
|
|
|
|
var serializedPiskel = this.getSnapshotFromState_(snapshotIndex);
|
|
|
|
var onPiskelLoadedCb = this.onPiskelLoaded_.bind(this, index, snapshotIndex);
|
|
|
|
pskl.utils.serialization.Deserializer.deserialize(serializedPiskel, onPiskelLoadedCb);
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
window.console.error("[CRITICAL ERROR] : Unable to load a history state.");
|
|
|
|
window.console.error("Can you open an issue on http://github.com/juliandescottes/piskel or contact @piskelapp on twitter ? Thanks !");
|
|
|
|
window.console.error("Thanks !");
|
|
|
|
if (typeof e === "string") {
|
|
|
|
window.console.error(e);
|
|
|
|
} else {
|
|
|
|
window.console.error(e.message);
|
|
|
|
window.console.error(e.stack);
|
2014-04-17 03:27:49 +04:00
|
|
|
}
|
2014-04-23 01:57:30 +04:00
|
|
|
}
|
|
|
|
};
|
2014-04-17 03:27:49 +04:00
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
ns.HistoryService.prototype.getSnapshotFromState_ = function (stateIndex) {
|
|
|
|
var state = this.stateQueue[stateIndex];
|
|
|
|
var piskelSnapshot = state.piskel;
|
2014-04-17 03:27:49 +04:00
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
// If the snapshot is stringified, parse it and backup the result for faster access next time
|
|
|
|
// FIXME : Memory consumption might go crazy if we keep unpacking big piskels indefinitely
|
|
|
|
// ==> should ensure I remove some of them :)
|
|
|
|
if (typeof piskelSnapshot === "string") {
|
|
|
|
piskelSnapshot = JSON.parse(piskelSnapshot);
|
|
|
|
state.piskel = piskelSnapshot;
|
2014-04-17 03:27:49 +04:00
|
|
|
}
|
2014-04-20 15:15:30 +04:00
|
|
|
|
2014-04-23 01:57:30 +04:00
|
|
|
return piskelSnapshot;
|
2014-04-20 15:15:30 +04:00
|
|
|
};
|
2014-04-17 03:27:49 +04:00
|
|
|
|
2014-05-05 00:58:36 +04:00
|
|
|
ns.HistoryService.prototype.onPiskelLoaded_ = function (index, snapshotIndex, piskel) {
|
|
|
|
var originalSize = this.getPiskelSize_();
|
|
|
|
piskel.setDescriptor(this.piskelController.piskel.getDescriptor());
|
|
|
|
this.piskelController.setPiskel(piskel);
|
2014-04-17 03:27:49 +04:00
|
|
|
|
|
|
|
for (var i = snapshotIndex + 1 ; i <= index ; i++) {
|
|
|
|
var state = this.stateQueue[i];
|
2014-04-18 15:13:42 +04:00
|
|
|
this.setupState(state);
|
2014-04-17 03:27:49 +04:00
|
|
|
this.replayState(state);
|
|
|
|
}
|
|
|
|
|
2014-05-17 13:03:18 +04:00
|
|
|
var lastState = this.stateQueue[index+1];
|
|
|
|
if (lastState) {
|
|
|
|
this.setupState(lastState);
|
|
|
|
}
|
2014-04-20 15:15:30 +04:00
|
|
|
this.currentIndex = index;
|
2013-09-29 02:01:18 +04:00
|
|
|
$.publish(Events.PISKEL_RESET);
|
2014-05-05 00:58:36 +04:00
|
|
|
if (originalSize !== this.getPiskelSize_()) {
|
|
|
|
$.publish(Events.FRAME_SIZE_CHANGED);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ns.HistoryService.prototype.getPiskelSize_ = function () {
|
|
|
|
return this.piskelController.getWidth() + 'x' + this.piskelController.getHeight();
|
2014-04-17 03:27:49 +04:00
|
|
|
};
|
|
|
|
|
2014-04-18 15:13:42 +04:00
|
|
|
ns.HistoryService.prototype.setupState = function (state) {
|
|
|
|
this.piskelController.setCurrentFrameIndex(state.frameIndex);
|
|
|
|
this.piskelController.setCurrentLayerIndex(state.layerIndex);
|
|
|
|
};
|
|
|
|
|
2014-04-17 03:27:49 +04:00
|
|
|
ns.HistoryService.prototype.replayState = function (state) {
|
2014-04-20 15:15:30 +04:00
|
|
|
var action = state.action;
|
|
|
|
var type = action.type;
|
|
|
|
var layer = this.piskelController.getLayerAt(state.layerIndex);
|
|
|
|
var frame = layer.getFrameAt(state.frameIndex);
|
|
|
|
action.scope.replay(frame, action.replay);
|
2012-09-11 01:26:12 +04:00
|
|
|
};
|
|
|
|
|
2012-09-07 02:18:59 +04:00
|
|
|
})();
|