/**
 * Created by aghassaei on 1/16/15.
 */


Lattice = Backbone.Model.extend({

    defaults: {
        scale: window.defaultLatticeScale,
        nodes: [],
        cells: [[[null]]],//3D matrix containing all cells and null, dynamic size
        inverseCells: [[[null]]],//3d matrix containing all inverse cells and null, dynamic size
        cellsMin: {x:0, y:0, z:0},//min position of cells matrix
        cellsMax: {x:0, y:0, z:0},//max position of cells matrix
        numCells: 0,
        basePlane: null//plane to build from
    },

    //pass in fillGeometry

    initialize: function(options){

        //bind events
        this.listenTo(this, "change:scale", this._scaleDidChange);
    },

    ////////////////////////////////////////////////////////////////////////////////////
    ///////////////////////////////ADD/REMOVE CELLS/////////////////////////////////////
    ////////////////////////////////////////////////////////////////////////////////////

    addCellsInRange: function(range){//add a block of cells
        var scale = this.get("scale");
        var cells = this.get("cells");
        this._checkForMatrixExpansion(cells, range.max, range.min);

        var cellsMin = this.get("cellsMin");
        var relativeMin = this._subtract(range.min, cellsMin);
        var relativeMax = this._subtract(range.max, this.get("cellsMin"));

        for (var x=relativeMin.x;x<=relativeMax.x;x++){
            for (var y=relativeMin.y;y<=relativeMax.y;y++){
                for (var z=relativeMin.z;z<=relativeMax.z;z++){
                    if (!cells[x][y][z]) {
                        cells[x][y][z] = this._makeCellForLatticeType(this._add({x:x, y:y, z:z}, cellsMin), scale);
                        this.set("numCells", this.get("numCells")+1);
                    } else console.warn("already a cell there");
                }
            }
        }
        window.three.render();
    },

    addCellAtIndex: function(indices){

        var scale = this.get("scale");
        var cells = this.get("cells");
        this._checkForMatrixExpansion(cells, indices, indices);

        var index = this._subtract(indices, this.get("cellsMin"));
        if (!cells[index.x][index.y][index.z]) {
            cells[index.x][index.y][index.z] = this._makeCellForLatticeType(indices, scale);
            this.set("numCells", this.get("numCells")+1);
            window.three.render();
        } else console.warn("already a cell there");

    },

    removeCellAtIndex: function(indices){

        var index = this._subtract(indices, this.get("cellsMin"));
        var cells = this.get("cells");
        if (index.x<cells.length && index.y<cells[0].length && index.z<cells[0][0].length){
            this.removeCell(cells[index.x][index.y][index.z]);
        }
    },

    removeCell: function(cell){
        if (!cell) return;
        var index = this._subtract(cell.indices, this.get("cellsMin"));
        var cells = this.get("cells");
        cell.destroy();
        cells[index.x][index.y][index.z] = null;

        //todo shrink cells matrix if needed

        this.set("numCells", this.get("numCells")-1);
        window.three.render();
    },

    clearCells: function(){
        this._iterCells(this.get("cells"), function(cell){
            if (cell) cell.destroy();
        });
        this.set("cells", [[[null]]]);
        this.set("cellsMax", {x:0, y:0, z:0});
        this.set("cellsMin", {x:0, y:0, z:0});
        this.set("nodes", []);
        this.set("numCells", 0);
        if (this.get("basePlane")) this.get("basePlane").set("zIndex", 0);
        window.three.render();
    },

    ////////////////////////////////////////////////////////////////////////////////////
    ///////////////////////////////FILL GEOMETRY////////////////////////////////////////
    ////////////////////////////////////////////////////////////////////////////////////

    subtractMesh: function(mesh){

        var scale = this.getScale();
        var yscale = scale/2*Math.sqrt(3);
        var zScale = this.get("scale")*2/Math.sqrt(6);

        var cells = this.get("cells");
        var cellsMin = this.get("cellsMin");

        var allVertexPos = mesh.geometry.attributes.position.array;

        var zHeight = 0;
        for (var x=0;x<cells.length;x++){
            for (var y=0;y<cells[0].length;y++){
                var firstCell = null;
                for (var z=0;z<cells[0][0].length;z++){
                    firstCell = cells[x][y][z];
                    if (firstCell) break;
                }
                if (!firstCell) continue;//nothing in col

                var origin = firstCell._calcPosition(0, this._add({x:x,y:y,z:z}, cellsMin));
                zHeight = this._findIntersectionsInWindow(scale/2, yscale/2, origin, allVertexPos) || zHeight;

                zHeight = Math.floor(zHeight/zScale);
                for (var z=0;z<zHeight;z++){
                    var cell = cells[x][y][z];
                    if (cell) cell.destroy();
                    cells[x][y][z] = null;
                }
            }

        }
        window.three.render();
    },

    _findIntersectionsInWindow: function(windowX, windowY, origin, allVertexPos){
        for (var i=0;i<allVertexPos.length;i+=3){
            if (allVertexPos[i] > origin.x-windowX && allVertexPos[i] < origin.x+windowX
                && allVertexPos[i+1] > origin.y-windowY && allVertexPos[i+1] < origin.y+windowY){
                return allVertexPos[i+2];
            }
        }
        return null
    },


    ////////////////////////////////////////////////////////////////////////////////////
    ///////////////////////////////CELLS ARRAY//////////////////////////////////////////
    ////////////////////////////////////////////////////////////////////////////////////

    _checkForMatrixExpansion: function(cells, indicesMax, indicesMin){

        var lastMax = this.get("cellsMax");
        var lastMin = this.get("cellsMin");
        var newMax = this._updateCellsMax(indicesMax, lastMax);
        var newMin = this._updateCellsMin(indicesMin, lastMin);
        if (newMax) {
            this._expandCellsArray(cells, this._subtract(newMax, lastMax), false);
            this.set("cellsMax", newMax);
        }
        if (newMin) {
            this._expandCellsArray(cells, this._subtract(lastMin, newMin), true);
            this.set("cellsMin", newMin);
        }
    },

    _expandCellsArray: function(cells, expansion, fromFront){

        _.each(_.keys(expansion), function(key){
            if (expansion[key] == 0) return;//no expansion on this axis

            var cellsX = cells.length;
            var cellsY = cells[0].length;
            var cellsZ = cells[0][0].length;

            if (key=="x"){
                for (var x=0;x<expansion[key];x++){
                    var newLayer = [];
                    for (var y=0;y<cellsY;y++){
                        var newCol = [];
                        for (var z=0;z<cellsZ;z++){
                            newCol.push(null);
                        }
                        newLayer.push(newCol);
                    }
                    if (fromFront) cells.unshift(newLayer);
                    else cells.push(newLayer);
                }
            } else if (key=="y"){
                for (var x=0;x<cellsX;x++){
                    for (var y=0;y<expansion[key];y++){
                        var newCol = [];
                        for (var z=0;z<cellsZ;z++){
                            newCol.push(null);
                        }
                        if (fromFront) cells[x].unshift(newCol);
                        else cells[x].push(newCol);
                    }
                }
            } else if (key=="z"){
                for (var x=0;x<cellsX;x++){
                    for (var y=0;y<cellsY;y++){
                        for (var z=0;z<expansion[key];z++){
                            if (fromFront) cells[x][y].unshift(null);
                            else cells[x][y].push(null);
                        }
                    }
                }
            }
        });
    },

    _updateCellsMin: function(newPosition, currentMin){
        var newMin = {};
        var hasChanged = false;
        _.each(_.keys(newPosition), function(key){
            if (newPosition[key]<currentMin[key]){
                hasChanged = true;
                newMin[key] = newPosition[key];
            } else {
                newMin[key] = currentMin[key];
            }
        });
        if (hasChanged) return newMin;
        return false;
    },

    _updateCellsMax: function(newPosition, currentMax){
        var newMax = {};
        var hasChanged = false;
        _.each(_.keys(newPosition), function(key){
            if (newPosition[key]>currentMax[key]){
                hasChanged = true;
                newMax[key] = newPosition[key];
            } else {
                newMax[key] = currentMax[key];
            }
        });
        if (hasChanged) return newMax;
        return false;
    },

    _subtract: function(pos1, pos2){
        return {x:pos1.x-pos2.x, y:pos1.y-pos2.y, z:pos1.z-pos2.z};
    },

    _add: function(pos1, pos2){
        return {x:pos1.x+pos2.x, y:pos1.y+pos2.y, z:pos1.z+pos2.z};
    },

    ////////////////////////////////////////////////////////////////////////////////////
    ////////////////////////////////////EVENTS//////////////////////////////////////////
    ////////////////////////////////////////////////////////////////////////////////////

    cellModeDidChange: function(mode){
        this._iterCells(this.get("cells"), function(cell){
            if (cell && cell.drawForMode) cell.drawForMode(mode);
        });
        window.three.render();
    },

    _scaleDidChange: function(){
        var scale = this.get("scale");
        this.get("basePlane").updateScale(scale);
        this._iterCells(this.get("cells"), function(cell){
            if (cell) cell.updateForScale(scale);
        });
        window.three.render();
    },

    previewScaleChange: function(scale){
        this.get("basePlane").updateScale(scale);
    },

    ////////////////////////////////////////////////////////////////////////////////////
    ///////////////////////////////CONNECTION TYPE//////////////////////////////////////
    ////////////////////////////////////////////////////////////////////////////////////

    updateLatticeType: function(cellType, connectionType){
        this.clearCells();
        if (this._undo) this._undo();
        if (cellType == "octa"){
            if (connectionType == "face"){
                _.extend(this, this.OctaFaceLattice);
            } else if (connectionType == "edge"){
                _.extend(this, this.OctaEdgeLattice);
            } else if (connectionType == "edgeRot"){

            } else if (connectionType == "vertex"){
                _.extend(this, this.OctaVertexLattice);
            }
        } else if (cellType == "cube"){

        }
        this._initLatticeType();
    },

    ////////////////////////////////////////////////////////////////////////////////////
    ////////////////////////////////////UTILS///////////////////////////////////////////
    ////////////////////////////////////////////////////////////////////////////////////

    _iterCells: function(cells, callback){
        _.each(cells, function(cellLayer){
            _.each(cellLayer, function(cellColumn){
                _.each(cellColumn, function(cell){
                    callback(cell, cellColumn, cellLayer);
                });
            });

        });
    },

    getScale: function(){
        return this.get("scale");
    },

////////////////////////////////////////////////////////////////////////////////////////
//////////////////////////////FACE CONN OCTA LATTICE////////////////////////////////////
////////////////////////////////////////////////////////////////////////////////////////


    OctaFaceLattice: {

        _initLatticeType: function(){

            //bind events
            this.listenTo(this, "change:columnSeparation", this._changeColSeparation);

            if (this.get("basePlane")) this.get("basePlane").destroy();
            this.set("basePlane", new OctaBasePlane({scale:this.get("scale")}));
            this.set("columnSeparation", 0.0);
        },

        _changeColSeparation: function(){
            var colSep = this.get("columnSeparation");
            var scale = this.get("scale");
            this.get("basePlane").updateColSeparation(colSep);
            this._iterCells(this.get("cells"), function(cell){
                if (cell) cell.updateForScale(scale);
            });
            window.three.render();
        },

        getIndexForPosition: function(absPosition){

            //calc indices in cell matrix
            var scale = this.get("scale");
            var colSep = this.get("columnSeparation");
            var latticeScale = scale*(1+2*colSep);
            var octHeight = 2*scale/Math.sqrt(6);
            var triHeight = latticeScale/2*Math.sqrt(3);
            var position = {};
            position.x = Math.round(absPosition.x/latticeScale);
            position.y = Math.round(absPosition.y/triHeight);
            position.z = Math.round(absPosition.z/octHeight)-1;
            if (position.z%2 == 1) position.y += 1;

            return position;
        },

        getScale: function(){
            return this.get("scale")*(1.0+2*this.get("columnSeparation"));
        },

        _makeCellForLatticeType: function(indices, scale){
            return new DMASideOctaCell(indices, scale, this);
        },

        _undo: function(){//remove all the mixins, this will help with debugging later
            this._initLatticeType = null;
            this._changeColSeparation = null;
            this.addCellAtPosition = null;
            this.getScale = null;
            this._makeCellForLatticeType = null;
            this.stopListening(this, "columnSeparation");
            this.set("columnSeparation", null);
            this._undo = null;
        }

    },

////////////////////////////////////////////////////////////////////////////////////////
//////////////////////////////EDGE CONN OCTA LATTICE////////////////////////////////////
////////////////////////////////////////////////////////////////////////////////////////

    OctaEdgeLattice: {

        _initLatticeType: function(){

            //bind events
            this.listenTo(this, "change:columnSeparation", this._changeColSeparation);

            if (this.get("basePlane")) this.get("basePlane").destroy();
            this.set("basePlane", new OctaBasePlane({scale:this.get("scale")}));
            this.set("columnSeparation", 0.0);
        },

        _changeColSeparation: function(){
            var colSep = this.get("columnSeparation");
            var scale = this.get("scale");
            this.get("basePlane").updateColSeparation(colSep);
            this._iterCells(this.get("cells"), function(cell){
                if (cell) cell.updateForScale(scale);
            });
            window.three.render();
        },

        addCellAtPosition: function(absPosition){

            //calc indices in cell matrix
            var scale = this.get("scale");
            var colSep = this.get("columnSeparation");
            var latticeScale = scale*(1+2*colSep);
            var octHeight = 2*scale/Math.sqrt(6);
            var triHeight = latticeScale/2*Math.sqrt(3);
            var position = {};
            position.x = Math.round(absPosition.x/latticeScale);
            position.y = Math.round(absPosition.y/triHeight);
            position.z = Math.round(absPosition.z/octHeight);
            if (position.z%2 == 1) position.y += 1;

            this.addCellAtIndex(position);
        },

        getScale: function(){
            return this.get("scale")*(1.0+2*this.get("columnSeparation"));
        },

        _makeCellForLatticeType: function(indices, scale){
            return new DMASideOctaCell(indices, scale, this);
        },

        _undo: function(){//remove all the mixins, this will help with debugging later
            this._initLatticeType = null;
            this._changeColSeparation = null;
            this.addCellAtPosition = null;
            this.getScale = null;
            this._makeCellForLatticeType = null;
            this.stopListening(this, "columnSeparation");
            this.set("columnSeparation", null);
            this._undo = null;
        }

    },


////////////////////////////////////////////////////////////////////////////////////////
//////////////////////////////VERTEX CONN OCTA LATTICE//////////////////////////////////
////////////////////////////////////////////////////////////////////////////////////////

    OctaVertexLattice: {

        _initLatticeType: function(){

            //bind events

            if (this.get("basePlane")) this.get("basePlane").destroy();
            this.set("basePlane", new SquareBasePlane({scale:this.get("scale")}));
        },

        addCellAtPosition: function(absPosition){

            //calc indices in cell matrix
            var scale = this.get("scale");
            var colSep = this.get("columnSeparation");
            var latticeScale = scale*(1+2*colSep);
            var octHeight = 2*scale/Math.sqrt(6);
            var triHeight = latticeScale/2*Math.sqrt(3);
            var position = {};
            position.x = Math.round(absPosition.x/latticeScale);
            position.y = Math.round(absPosition.y/triHeight);
            position.z = Math.round(absPosition.z/octHeight);
            if (position.z%2 == 1) position.y += 1;

            this.addCellAtIndex(position);
        },

        getScale: function(){
            return this.get("scale");
        },

        _makeCellForLatticeType: function(indices, scale){
            return new DMASideOctaCell(indices, scale, this);
        },

        _undo: function(){//remove all the mixins, this will help with debugging later
            this._initLatticeType = null;
            this.addCellAtPosition = null;
            this.getScale = null;
            this._makeCellForLatticeType = null;
            this.stopListening(this, "columnSeparation");
            this.set("columnSeparation", null);
            this._undo = null;
        }

    }



////////////////////////////////////////////////////////////////////////////////////////
//////////////////////////////CUBE LATTICE//////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////////////////



});