/* <copyright>
This file contains proprietary software owned by Motorola Mobility, Inc.<br/>
No rights, expressed or implied, whatsoever to this software are provided by Motorola Mobility, Inc. hereunder.<br/>
(c) Copyright 2011 Motorola Mobility, Inc.  All Rights Reserved.
</copyright> */


/*
 *	Creates a unique node id
 */
 nodeIdGen = {};
 
 nodeIdGen.counter = 0;
 
 nodeIdGen.getId = function()
 {
	return  "gen_" + nodeIdGen.counter++;
 }

/*
 *	Constructs a new transform node
 */
function createTransformNode(nodeName)
{
	node = { 'name': nodeName };
	
	node.transformNodeTemplate = new transformNodeTemplate(node);
	
	return node;
}

/*
 *	Constructs a new material node
 */
function createMaterialNode(nodeName)
{
	node = { 'name': nodeName };
	
	node.materialNodeTemplate = new materialNodeTemplate(node);
	
	return node;
}

/*
 *	Constructs a new mesh node
 */
function createMeshNode(nodeName, primitive)
{
	meshNode = { 'mesh':{}, 'meshNodeTemplate':{} };
	
	var renderer = g_Engine.getContext().renderer;
		
	if(!primitive.built)
    {
        renderer.createPrimitive(primitive);
    }

    var model = g_meshMan.getModelByName(nodeName);
    if(!model)
    {
        meshNode.mesh.meshNodeTemplate = new meshNodeTemplate(meshNode.mesh, primitive, nodeName);

        g_meshMan.modelMap[nodeName] = meshNode.mesh;
        return meshNode; // --early out--
    }
    else if(!renderer.buffers[model.primitive.buffersID])
	{
		renderer.createPrimitive(model.primitive);
	}
	
    meshNode.mesh.meshNodeTemplate = new meshNodeTemplate(meshNode.mesh, model.primitive, nodeName);
	
	
	return meshNode;
}

/*
 *	Construct a light node
 */
function createLightNode(nodeName)
{
    node = { 'name': nodeName };
	node.lightNodeTemplate = new lightNodeTemplate(node);
	
	return node;
}

/*
 *	creates a specialized mesh node representing a screen aligned quad with identity transform
 */
function createScreenQuadNode()
{
	var trNode = createTransformNode();
	trNode.attachMeshNode("screenQuad", createScreenAlignedQuad());
	return trNode; 
}

function verifyTransformNode( node )
{
    if(node.transformNodeTemplate == undefined)
    {
        node.transformNodeTemplate = new transformNodeTemplate(node);
    }
}

function verifyMaterialNode( node )
{
    if(node.materialNodeTemplate == undefined)
    {
        node.materialNodeTemplate = new materialNodeTemplate(node);
    }
}

function verifyLightNode( node )
{
    if(node.lightNodeTemplate == undefined)
    {
        node.lightNodeTemplate = new lightNodeTemplate(node);
    }
}


/*
 *	Takes an object and attachs transform node
 *	functions and fields if they are not defined
 */
transformNodeTemplate = function(trNode)
{
    // Lots of checking for things that might exist and adding them when they don't
    
    /* ------------------------------------------- */
	if(!trNode.children)
	{
		trNode.children = [];
	}
	
	if(!trNode.local)
	{
		trNode.local = mat4.identity();
	}
	
	if(!trNode.world)
	{
		trNode.world = mat4.identity();
	}
	
	if(!trNode.id)
	{
		trNode.id = nodeIdGen.getId();
	}
	
	if(!trNode.name)
	{
		trNode.name = "xfrmNode" + trNode.id;
	}
	
	if(!trNode.parent)
	{
	    trNode.parent = null;
	}
		
	if(!trNode.meshes)
	{
		trNode.meshes = [];
	}
	
	if(!trNode.nodeType)
	{
		trNode.nodeType = rdgeConstants.nodeType.TRNODE;
	}
	
    /* ------------------------------------------- */
    
    // Adding functions to the node none of these exist from export process	
	/*
	 *	Attaches a material to a node
	 */
	trNode.attachMaterial = function(matNode)
	{
	    verifyMaterialNode(matNode);
	    
		this.materialNode = matNode;
	}
	
	/*
	 *	@param node - the node to attach, can optionally be a node name paired with a primitive that will be built into a meshNode
	 *	@param primitive - an optional parameter that must be supplied if the node is a name and not an object
	 */
	trNode.attachMeshNode = function( node, primitive )
	{
		if(typeof node == "string")
		{
		    node = createMeshNode(node, primitive)
	    }
	    if(trNode.materialNode == undefined)
		{
			trNode.materialNode = createMaterialNode(trNode.name + "|defMaterial");
		}
	    
		trNode.meshes.push( {'mesh':{'name':node.mesh.attribs.name, 'id':node.mesh.attribs.id}});
	}
	
	/*
	 *	Inserts a node as a child of this node
	 */
	trNode.insertAsChild = function(transNode)
	{
	    if(this == transNode)
	        return;
	        
	   verifyTransformNode(transNode);
	    
	    transNode.parent = this;
	    this.children.push({transformNode:transNode});
	}
	
	/*
	 *	Inserts a node as the parent of this node
	 */
	trNode.insertAsParent = function(transNode)
	{
	    if(this == transNode)
	        return;
	        
	    verifyTransformNode(transNode);
	    
	    if(this.parent)
	    {
	        // remove this node from current parents list
	        var len = this.parent.children.length;
	        for(var i = 0; i < len;  ++i)
	        {
	            if(this.parent.children[i].transformNode != undefined)
	            {
	                tr = this.parent.children[i].transformNode;
	                if(tr.id == this.id)
	                {
	                    // removes 1 item starting from i
	                    this.parent.children.splice(i,1);
	                    break;
	                }
	            }
	            
	        }
	        
	        // set the new parents parent
	        transNode.parent = this.parent;
	        
	        // push passed in node into parents list
	        this.parent.children.push({transformNode:transNode});
	        
	        // set the passed in node as the current parent
	        this.parent = transNode;
	    }
	    
	    // add this node to the passed in nodes child list
	    transNode.children.push({transformNode:this});
	}

}


// add material handlers to a material node
materialNodeTemplate = function(matNode)
{
	// enumerate constants

	// type definitions
	TEX_DIF  = 0;
	TEX_SPEC = 1;
	TEX_NORM = 2;
	TEX_GLOW = 3;
	
	if(!matNode.nodeType)
	{
		matNode.nodeType = rdgeConstants.nodeType.MATNODE;
	}
	
	MATERIAL_MAX_LIGHTS = rdgeConstants.MAX_MATERIAL_LIGHTS;
	
	if(!matNode.lightChannel)
	{
		matNode.lightChannel = 
		[
			null,
			null,
			null,
			null
		];
	}
	
	/*
	 *	Material categories determine sorting
	 */
	if(!matNode.sortCategory)
	{
		matNode.sortCategory = rdgeConstants.categoryEnumeration.OPAQUE;
	}
	
	/*
	 *	every node has an id either generated by export or generated here
	 */
	if(!matNode.id)
	{
		matNode.id = nodeIdGen.getId();
	}
	
	/*
	 *	every node has an name either setin art pipeline or generated here
	 */
	if(!matNode.name)
	{
		matNode.name = "matNode" + matNode.id;
	}
	
	/*
	 *	Default list of textures if nothing is set
	 */
	if(!matNode.textureList)
	{
		var renderer = g_Engine.getContext().renderer;
	    matNode.textureList = 
	    [
	        {'name':"colMap",	'handle':renderer.getTextureByName("assets/images/white"),         'unit': TEX_DIF,    "type":UNIFORMTYPE.TEXTURE2D},
		    {'name':"envMap",	'handle':renderer.getTextureByName("assets/images/material_paint"),'unit': TEX_SPEC,   "type":UNIFORMTYPE.TEXTURE2D},
		    {'name':"normalMap",'handle':renderer.getTextureByName("assets/images/blue"),          'unit': TEX_NORM,   "type":UNIFORMTYPE.TEXTURE2D},
		    {'name':"glowMap",	'handle':renderer.getTextureByName("assets/images/black"),         'unit': TEX_GLOW,   "type":UNIFORMTYPE.TEXTURE2D}
	    ];
	}
	
	if(!matNode.uniforms)
	{
	    matNode.uniforms = [];
	}
	
	matNode.setTexture = function(texType, texName)
	{
		var renderer = g_Engine.getContext().renderer;
		this.textureList[texType].handle = renderer.getTextureByName("assets/images/" + texName);
	    this.textureList[texType].unit = texType;
		this.textureList[texType].type = UNIFORMTYPE.TEXTURE2D;
		
	}

	matNode.setDiffuseTexture = function(texName)
	{
		this.setTexture(TEX_DIF, texName);
	}
	
	matNode.setSpecTexture = function(texName)
	{
		this.setTexture(TEX_SPEC, texName);
	}
	
	matNode.setNormalTexture = function(texName)
	{
		this.setTexture(TEX_NORM, texName);
	}
	
	matNode.setGlowTexture = function(texName)
	{
		this.setTexture(TEX_GLOW, texName);
	}
	
	matNode.setUniform = function(name, arrValue)
	{
		var len = this.uniforms.length;
		for(var i = 0; i < len; ++i)
		{
			if(this.uniforms[i].name == name)
			{
				this.uniforms[i].value = arrValue;
				return;
			}
		}
		
		window.console.log("Could not find uniform: " + name);
	}
	
	matNode.setShader = function( jshaderObject )
	{
		this.shaderProgram = jshaderObject;
	}
	
	matNode.setSortCategory = function( materialCat )
	{
		matNode.sortCategory = materialCat;
	}
	
	/*
	 * Sets a light channel reference to a lightNode
	 * @param channelNumber a number indicating whick light to turn on (0 - 3), or an array of numbers if multiple lights being set
	 * @param lightNode - a refernce to a light node object or an array lights
	 */
	matNode.enableLightChannel = function( channelNumber, lightNode )
	{
		verifyLightNode(lightNode);
		
		// set an array
		if(typeof channelNumber == "object")
		{
			var len = channelNumber.length;
			var maxLight = lightNode.length != undefined ? lightNode.length : 0;
			for(var i = 0; i < len; ++i)
			{
			
				matNode.lightChannel[channelNumber] = maxLight > 0 ? lightNode[ Math.min(i, maxLight - 1)] : lightNode;
			}
		}
		else // set an individual light
		{
			if(channelNumber < MATERIAL_MAX_LIGHTS)
				matNode.lightChannel[channelNumber] = lightNode;
		}
	}

	
	matNode.disableLightChannel = function( channelNumber )
	{
		if(typeof channelNumber != "object")
		{
			var len = channelNumber.length;
			
			for(var i = 0; i < len; ++i)
			{
				if(channelNumber[i] , MATERIAL_MAX_LIGHTS)
					matNode.lightChannel[channelNumber[i]] = null;
			}
		}
		else
		{
			if(channelNumber < MATERIAL_MAX_LIGHTS)
				matNode.lightChannel[channelNumber] = null;
		}
		
	}
	
	matNode.disableAllLights = function()
	{
		for(var i = 0; i < MATERIAL_MAX_LIGHTS; ++i)
		{
			matNode.lightChannel[i] = null;
		}
	}
	
		
	matNode.toJSON = function()
	{
		var jsonObj = {'jsonExportName':"materialNode"};
		for(var member in this)
		{
			jsonObj[member] = this[member];
			
			if(member === "textureList")
			{
				var texList = jsonObj[member];
				for(var i = 0, len = texList.length; i < len; ++i)
				{
					texList[i].handle.image = texList[i].handle.lookUpName; 
				}
			}
			else if(member === "shaderProgram")
			{
				// test that the shader hasn't already been exported
				if(typeof jsonObj[member] != "string")
				{
					jsonObj[member] = jsonObj[member].exportShader();
				}
			}
		}
		
		return jsonObj;
	}
}


meshNodeTemplate = function( meshNode, primitive, meshName )
{
    if(!primitive.built)
    {
        renderer.createPrimitive(primitive);
    }
    
    if(!meshNode.nodeType)
	{
		meshNode.nodeType = rdgeConstants.nodeType.MESHNODE;
	}
    
    if(!meshNode.attribs)
    {
        var newID = nodeIdGen.getId();
        
        meshNode.attribs = { 'id': newID,
        'indexCount': primitive.indexCount,
        'name': meshName,
        'vertCount': primitive.posCount};
        
        meshNode.name = meshName;
    }
    
    if(!meshNode.bbox)
    {
        meshNode.bbox = new box();
    }
    
    meshNode.data = null;
    
    meshNode.primitive = primitive;
    
    // setup bounding box
    var numPositions = primitive.posCount;
  
    if(numPositions > 0)
    {
        var positions = primitive.positions;
        
        var idx = 0;
        while (idx < numPositions - 2)
        {
          var thisPos = [positions[idx+0], positions[idx+1], positions[idx+2]];
          meshNode.bbox.addVec3(thisPos);
          idx += 3;
        }
    }
    else
    {
		window.console.error("mesh " + meshNode.attribs.name + ": bounding volume not created");
    }
}

lightNodeTemplate = function(lightNode)
{
	if(!lightNode.nodeType)
	{
		lightNode.nodeType = rdgeConstants.nodeType.LIGHTNODE;
	}
	
    if(!lightNode.id)
    {
        lightNode.id = nodeIdGen.getId();
    }
    
    if(!lightNode.name)
    {
        lightNode.name = "light_" + lightNode.id;
    }
    
    if(!lightNode.typeName)
    {
        lightNode.typeName = "dir_light";
    }
    
    if(!lightNode.castShadow)
    {
        lightNode.castShadow = false;
    }
    
    if(!lightNode.depthMapBias)
    {
        lightNode.depthMapBias = 0.0179;
    }
    
    if(!lightNode.depthMapSize)
    {
        lightNode.depthMapSize = 1024;
    }
    
    if(!lightNode.coneAngle)
    {
        lightNode.coneAngle = 0.707;
    }
    
    if(!lightNode.penumbraAngle)
    {
        lightNode.coneAngle = 0.0;
    }
    
    if(!lightNode.dropOff)
    {
        lightNode.coneAngle = 0.025;
    }
    
    if(!lightNode.color)
    {
        lightNode.color = [1,1,1,1];
    }
    
    if(!lightNode.dir)
    {
        lightNode.dir = [1,-1,1];
    }
    
    if(!lightNode.links)
    {
        lightNode.links = [];
    }
    
    if(!lightNode.position)
    {
        lightNode.position = [0,0,0];
    }
    
    if(!lightNode.lightDiffuse)
    {
        lightNode.lightDiffuse = [1,1,1,1];
    }
    
    if(!lightNode.lightAmbient)
    {
        lightNode.lightAmbient = [0.5,0.5,0.5,1.0];
    }
    
    if(!lightNode.lightSpecular)
    {
        lightNode.lightSpecular = [1,1,1,1];
    }
        
    lightNode.setPosition = function( pos )
    {
		for(var i = 0; i < 3; i++)
		{
			this.position[i] = pos[i];
		}
    }
    
    lightNode.setDiffuseColor = function( color )
    {
		for(var i = 0; i < 4; i++)
		{
			this.lightDiffuse[i] = color[i];
		}
    }
    
    lightNode.setAmbientColor = function( color )
    {
		for(var i = 0; i < 4; i++)
		{
			this.lightAmbient[i] = color[i];
		}
    }
    
    lightNode.setSpecularColor = function( color )
    {
		for(var i = 0; i < 4; i++)
		{
			this.lightSpecular[i] = color[i];
		}
    }
}