(function () { var ns = $.namespace("pskl.model"); ns.Frame = function (width, height) { if (width && height) { this.width = width; this.height = height; this.pixels = ns.Frame.createEmptyPixelGrid_(width, height); this.previousStates = [this.getPixels()]; this.stateIndex = 0; } else { throw 'Bad arguments in pskl.model.Frame constructor : ' + width + ', ' + height; } }; ns.Frame.fromPixelGrid = function (pixels) { if (pixels.length && pixels[0].length) { var w = pixels.length, h = pixels[0].length; var frame = new pskl.model.Frame(w, h); frame.setPixels(pixels); return frame; } else { throw 'Bad arguments in pskl.model.Frame.fromPixelGrid : ' + pixels; } }; ns.Frame.createEmptyPixelGrid_ = function (width, height) { var pixels = []; //new Array(width); for (var columnIndex=0; columnIndex < width; columnIndex++) { var columnArray = []; for(var heightIndex = 0; heightIndex < height; heightIndex++) { columnArray.push(Constants.TRANSPARENT_COLOR); } pixels[columnIndex] = columnArray; } return pixels; }; ns.Frame.createEmptyFromFrame = function (frame) { return new ns.Frame(frame.getWidth(), frame.getHeight()); }; ns.Frame.prototype.clone = function () { var clone = new ns.Frame(this.width, this.height); clone.setPixels(this.getPixels()); return clone; }; /** * Returns a copy of the pixels used by the frame */ ns.Frame.prototype.getPixels = function () { return this.clonePixels_(this.pixels); }; /** * Copies the passed pixels into the frame. */ ns.Frame.prototype.setPixels = function (pixels) { this.pixels = this.clonePixels_(pixels); }; ns.Frame.prototype.clear = function () { var pixels = ns.Frame.createEmptyPixelGrid_(this.getWidth(), this.getHeight()); this.setPixels(pixels); }; /** * Clone a set of pixels. Should be static utility method * @private */ ns.Frame.prototype.clonePixels_ = function (pixels) { var clonedPixels = []; for (var col = 0 ; col < pixels.length ; col++) { clonedPixels[col] = pixels[col].slice(0 , pixels[col].length); } return clonedPixels; }; ns.Frame.prototype.serialize = function () { return JSON.stringify(this.pixels); }; ns.Frame.prototype.setPixel = function (col, row, color) { this.pixels[col][row] = color; }; ns.Frame.prototype.getPixel = function (col, row) { return this.pixels[col][row]; }; ns.Frame.prototype.forEachPixel = function (callback) { for (var col = 0 ; col < this.getWidth() ; col++) { for (var row = 0 ; row < this.getHeight() ; row++) { callback(this.getPixel(col, row), col, row); } } }; ns.Frame.prototype.getWidth = function () { return this.width; }; ns.Frame.prototype.getHeight = function () { return this.height; }; ns.Frame.prototype.containsPixel = function (col, row) { return col >= 0 && row >= 0 && col < this.pixels.length && row < this.pixels[0].length; }; ns.Frame.prototype.saveState = function () { // remove all states past current state this.previousStates.length = this.stateIndex + 1; // push new state this.previousStates.push(this.getPixels()); // set the stateIndex to latest saved state this.stateIndex = this.previousStates.length - 1; }; ns.Frame.prototype.loadPreviousState = function () { if (this.stateIndex > 0) { this.stateIndex--; this.setPixels(this.previousStates[this.stateIndex]); } }; ns.Frame.prototype.loadNextState = function () { if (this.stateIndex < this.previousStates.length - 1) { this.stateIndex++; this.setPixels(this.previousStates[this.stateIndex]); } }; ns.Frame.prototype.isSameSize = function (otherFrame) { return this.getHeight() == otherFrame.getHeight() && this.getWidth() == otherFrame.getWidth(); }; })();