API Docs for: 1.0.2
Show:

File: GIScene\Scene.js

/**
 * The Scene Object
 *
 * @namespace GIScene
 * @class Scene
 * @constructor
 * @extends THREE.EventDispatcher
 * @param {String} containerDivId The id of an HTMLDivElement which will be used as container for the webgl canvas.
 * @param {Object} [config] Allowed
 * values for config are:
 *
 * 	cameratype:	'perspective'|'ortho'
 *  near: {Number}
 *  far:  {Numver}
 *  fov:  {Number}
 * 
 *  width:	{Integer}
 *  height: {Integer}
 * 
 *  animate:{Boolean}
 *  
 *  center: 				{THREE.Vector3}
 *  positionFromCenter: 	{THREE.Vector3}
 *  projection: 			{String}
 *  units:					{String}
 *  offset:					{GIScene.Coordinate3}
 *  skyColor:				{THREE.Color}
 *  fog:					{THREE.Fog}
 */

GIScene.Scene = function(containerDivId, config) {

	/**
	 * The default config which is used when no config object is provided.
	 *
	 * @private
	 * @property defaults
	 * @type Object
	 **/
	var defaults = {

		//canvas options
		width : null,
		height : null,

		//render options
		animate : true, 
		fog : null,

		//scene options
		center : new THREE.Vector3(0, 0, 0),
		projection : 'EPSG:32616', //not yet used UTM16N???-->EPSG:32616
		units : 'm', //not yet used
		offset : new GIScene.Coordinate3(0,0,0), 
		//new THREE.Vector3(0,0,0), //new THREE.Vector3(269500, 550, -1641500), //not yet used offset is used to keep coordinates short.
		skyColor : new THREE.Color().setStyle('lightskyblue').getHex(),

		//camera options
		cameratype : 'perspective',
		near : 0.1,
		far : 1000,
		fov : 45, //for perspective camera only
		positionFromCenter : new THREE.Vector3(0, 0, 10),

		//data options
		// url : null,
		// format : null,
		// verticalAxis : "Y"

	};

	/**
	 * The config which is used to initialize the Scene. Merged from defaults and passed config Object.
	 *
	 * @property config
	 * @type Object
	 */
	this.config = GIScene.Utils.mergeObjects(defaults, config || {});


	//make this class an EventDispatcher
	//THREE.EventDispatcher.call( this );

	//canvas properties
	this.containerDiv = document.getElementById(containerDivId);
	this.canvas = null;
	//get size from config else from containerDiv-CSS Style else use default 500x500 px
	var width = null;
	width = (this.config.width) ? this.config.width :  this.containerDiv.clientWidth ;
	
	var height = null;
	height = (this.config.height) ?  this.config.height : (this.containerDiv.clientHeight > 0) ?  this.containerDiv.clientHeight :  500;

	//render properties
	var animationFrameId;

	//scene properties
	/**
	 * The scenegraph root node. Add {THREE.Object3D}-objects to this node. See THREE.js docs for further information
	 * @property root
	 * @type {THREE.Scene}
	 */
	this.root = null;
	/**
	 * The active camera. {THREE.CombinedCamera} can be switched from perspective to orthographic and vice versa
	 * @property camera
	 * @type {THREE.CombinedCamera}
	 */
	this.camera = null;
	//Camera gets a target object later in initScene()
	
	//THREE.Scene and THREE.Orthocamera to render Sprites in screen coordinate space
	this.spriteRoot = null;
	this.spriteCamera = null;
	
	this.lights = [];
	this.renderer = null;
	this.effectComposer = null;
	this.fog = null;
	this.layers = [];
	//Layers should add {THREE.Scene} objects to this.root
	this.center = new THREE.Vector3(0, 0, 0);

	//controls
	this.controls = [];

	//clock to be independent of framerates
	this.clock = new THREE.Clock();
	this.delta = null; //global, automatically updated in startAnimation() every frame

	this.init = function() {

		//initCanvas
		this.initCanvas();

		//initScene
		this.initScene();

		//animate: start renderLoop or render once
		(this.config.animate) ? this.startAnimation() : this.renderer.render(this.root, this.camera);

	};

	this.initCanvas = function() {
		//create canvas in div use div use size from config else from div element else
		this.canvas = document.createElement('canvas');
		this.canvas.className = "giscene_canvas";
		//remove line height from canvas containterDiv to fit the canvas exactly to its wrapper div
		this.containerDiv.style.lineHeight = 0;
		
		
		
		// this.canvas.style.width  = "100%";//width + "px";//
		// this.canvas.style.height = "100%";//height + "px";//
		this.containerDiv.appendChild(this.canvas);
		// this.containerDiv.style.width  = width + "px";
		// this.containerDiv.style.height = height + "px";
	};

	this.initScene = function(config) {
		//create THREE.Scene
		this.root = new THREE.Scene();
		this.root.name = 'root';
		
		//optionally add fog
		if(this.config.fog){this.root.fog = this.fog = this.config.fog.clone();}
		
		//initCamera
		//this.camera = (this.config.cameratype == 'ortho') ? new THREE.OrthographicCamera(width / -2, width / 2, height / 2, height / -2, this.config.near, this.config.far) : new THREE.PerspectiveCamera(this.config.fov, width / height, this.config.near, this.config.far);
		this.camera = new THREE.CombinedCamera(width, height, this.config.fov, this.config.near, this.config.far, this.config.near, this.config.far );
		this.camera.name = 'THREE.CombinedCamera';
		this.camera.target = new THREE.Object3D();
		this.camera.target.name = "Camera target";
		this.camera.add(this.camera.target);
		if(this.config.cameratype == 'ortho'){this.camera.toOrthographic();}

		/**
		 * Fires whenever camera position changes
		 *
		 * @event cameraChange
		 */
		var oldCamPos = new THREE.Vector3(0, 0, 0);
		var oldCamRot = new THREE.Vector3(0, 0, 0);
		var cameraHasChanged = function() {
			if (!oldCamPos.equals(this.camera.position) || !oldCamRot.equals(this.camera.rotation)) {
				
				this.dispatchEvent({type : 'cameraChange'});
				oldCamPos = this.camera.position.clone();
				oldCamRot = this.camera.rotation.clone();
			};
			
		}.bind(this);
		this.addEventListener('afterRender', cameraHasChanged);
		//this.addEventListener('beforeRender', cameraHasChanged);
		
		//init sprite scene and camera
		this.spriteRoot = new THREE.Scene();
		// this.spriteCamera = new THREE.OrthographicCamera(0,width,0,height,-100,100 ); //l,r,t,b,near,far
		var width1_2 = width/2;
		var heigh1_2 = height/2;
		this.spriteCamera = new THREE.OrthographicCamera(-width1_2,width1_2,heigh1_2,-heigh1_2,-100,100 ); //l,r,t,b,near,far
		// this.spriteRoot.add(this.spriteCamera);
		// this.camera.add(this.spriteCamera);
		
		//initLights
		this.directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);//new THREE.DirectionalLight(0xffffff, 0.5);
		this.directionalLight.name = "THREE.DirectionalLight";
		//color, intensity //position will be set by Control.CameraLight
		this.ambientLight = new THREE.AmbientLight(0xcccccc); //cc ^= 0.8   new THREE.AmbientLight(0xffffff);
		this.ambientLight.name = "THREE.AmbientLight";

		//experimental spot light
		this.headLight = new THREE.SpotLight(0xffffff,0.3,0,Math.PI/2,20);
		this.camera.add(this.headLight);
		this.headLight.target.position.setZ(-1000);
		this.camera.target.add(this.headLight.target);

		//initRenderer
		this.renderer = new THREE.WebGLRenderer({
			antialias : true,
			precision : "highp",
			canvas : this.canvas,
			devicePixelRatio: 1, //when browser is in zoom mode this ratio is changed and renderer.setSize uses it 
		});
		
		this.renderer.setSize(width, height);
		this.renderer.setClearColor(this.config.skyColor);
		this.renderer.autoClear = false;
		
		// init effectComposer
		this.effectComposer = new THREE.EffectComposer(this.renderer);
		this.effectComposer.setSize(width, height);
		
		// add objects to the scene
		this.root.add(this.camera);
		this.root.add(this.ambientLight);
		this.root.add(this.directionalLight);

		//setCenter
		// this.camera.position.add(this.config.positionFromCenter);
		// this.camera.target.position.setZ(-this.config.positionFromCenter.length());
		this.setCenter(this.config.center, this.config.positionFromCenter);
		
		//add Logo
		var gisceneLogoTexture = new THREE.Texture(null);
		var gisceneLogoImage = new Image();
		gisceneLogoImage.onload = function(e) {
			gisceneLogoTexture.image = gisceneLogoImage;
			gisceneLogoTexture.needsUpdate = true;
		};
		// dataURL from poweredbygiscenegiscience.png
		// gisceneLogoImage.src = "";	
		// dataURL from poweredbygiscenegiscience2.png
		gisceneLogoImage.src = "";
		var gisceneLogoMaterial = new THREE.SpriteMaterial({ map : gisceneLogoTexture, opacity:1  });
		var gisceneLogo = new THREE.Sprite(gisceneLogoMaterial);
		gisceneLogo.scale.set( 158, 20, 1 );
		gisceneLogo.position.set((this.canvas.width/2)-79,(-this.canvas.height/2)+10,0);
		this.spriteRoot.add(gisceneLogo);
		//will be called in this.onResize
		this.logoOnResize = function() {
			gisceneLogo.position.set((this.canvas.width/2)-79,(-this.canvas.height/2)+10,0);
		}.bind(this);
		

	};

	/**
	 * Start animation frame and render loop
	 *
	 * @method startAnimation
	 */
	this.startAnimation = function() {
		animationFrameId = requestAnimationFrame(this.startAnimation.bind(this));
		
		this.delta = this.clock.getDelta();
		
		TWEEN.update();
		
		this.renderer.clear(true,true,true); //color, depth, stencil; autoclear is false to render sprites on top of scene
		/**
		 * beforeRender Event will be executed directly before the render call in the render loop
		 * @event beforeRender
		 */
		this.dispatchEvent({
			type : 'beforeRender'
		});
		/**
		 * beforeRender2 Event will be executed directly before the render call in the render loop
		 * @event beforeRender
		 */
		this.dispatchEvent({
			type : 'beforeRender2'
		});
		
		//mcaTWEEN.update();	
		
		if(this.effectComposer.passes.length>0){
			this.effectComposer.render();
		}else{
			this.renderer.render(this.root, this.camera);
		}
		
		this.renderer.clear(false,true,false); //render sprites always on top, therefore clear depth buffer
		this.renderer.render(this.spriteRoot, this.spriteCamera);
		
		this.dispatchEvent({
			type : 'afterRender'
		});
	};

	/**
	 * Stop animation frame and render loop
	 *
	 * @method stopAnimation
	 */
	this.stopAnimation = function() {
		cancelAnimationFrame(animationFrameId);
	};

	this.setCenter_old = function(vector, positionFromCenter) {
		var translation = vector.clone().sub(this.center);
		//
		// //keep light direction but translate light.target
		// var lightWorld 		 = this.directionalLight.parent.localToWorld(this.directionalLight.position.clone());
		// var lightTargetWorld = this.directionalLight.target.parent.localToWorld(this.directionalLight.target.position.clone());
		// var lightWorldNew 	 = lightWorld.add(translation);
		// this.directionalLight.position = this.directionalLight.parent.worldToLocal(lightWorldNew);
		//
		// this.directionalLight.target.position = vector.clone();

		//scene center
		this.center = vector.clone();
		//focus camera on center
		if (positionFromCenter) {
			//translate and focus camera on center
			// console.log(positionFromCenter.length());
			this.camera.position = vector.clone().add(positionFromCenter);
			this.camera.lookAt(this.center);
			this.camera.target.position.setZ(-positionFromCenter.length());
			
		} else {
			//just translate
			this.camera.position.add(translation);
		}

		// var positionFromCenter = positionFromCenter || this.camera.position.clone().sub(this.camera.target.parent.localToWorld(this.camera.target.position.clone())); //new THREE.Vector3(0,0,0);
		// this.camera.position = vector.clone().add(positionFromCenter);
		// this.camera.target.position = positionFromCenter.clone().multiplyScalar(-1);
		//this.camera.lookAt(this.center);
		// //this.camera.target.position = this.camera.target.parent.worldToLocal(vector.clone());
		this.dispatchEvent({
			type : 'center',
			content : {
				center : vector.clone(),
				translation : translation,
				positionFromCenter : positionFromCenter
			}
		});
	};
	
	
	/**
	 * Jump to another place in the scene.
	 * 
	 * @method setCenter
	 * @param {THREE.Vector3} vector vector in scene coordinates
	 * @param {THREE.Vector3} positionFromCenter vector to place the camera at some distance from the point of interest (vector)
	 * @param {Number} duration duration in milliseconds. if undefined or > 0 an animation is perforemed from the current to the specified new center. Set this to 0 to jump immediately to the new location.
	 */
	var _setCenterActive=false;
	var _setCenterTween = null;
	this.setCenter = function(vector, positionFromCenter, duration) {
		// stop exisitng tween when a new one comes during animation time
		if(_setCenterActive && _setCenterTween){_setCenterTween.stop();_setCenterActive=false;}
		
		if(!_setCenterActive){
				_setCenterActive = true;
		//var active = "false"; // avoid multiple calls at a time
		var duration = (duration == null)?1000:duration;
		var oldCamTargetPos	= this.center;//this.camera.localToWorld(this.camera.target.position.clone());
		var newCamTargetPos	= vector.clone();
		var positionFromCenter = positionFromCenter || this.camera.position.clone().sub( oldCamTargetPos );
		var oldCamPos		= this.camera.position.clone();
		var newCamPos 		= vector.clone().add(positionFromCenter);
		var lookAtVector	= new THREE.Vector3();
		var numDecimals = 3;
		
		var tweenValues = {
			tx: oldCamTargetPos.x,
			ty: oldCamTargetPos.y,
			tz: oldCamTargetPos.z,
			cx: oldCamPos.x,
			cy: oldCamPos.y,
			cz: oldCamPos.z
		};
		var targetValues = {
			tx: newCamTargetPos.x,
			ty: newCamTargetPos.y,
			tz: newCamTargetPos.z,
			cx: newCamPos.x,
			cy: newCamPos.y,
			cz: newCamPos.z
		};
		//scene center
		this.center = vector.clone();
		// console.log("Scene.setCenter():duration: "+duration);
		if(duration == 0){
			this.camera.position.set(parseFloat(targetValues.cx.toFixed(numDecimals)), parseFloat(targetValues.cy.toFixed(numDecimals)), parseFloat(targetValues.cz.toFixed(numDecimals)));
			this.camera.lookAt(lookAtVector.set(parseFloat(targetValues.tx.toFixed(numDecimals)),parseFloat(targetValues.ty.toFixed(numDecimals)),parseFloat(targetValues.tz.toFixed(numDecimals))));
			this.camera.target.position.setZ(-positionFromCenter.length());
			/**
			 * Fires after setCenter() completed
			 * @event center
			 */
			this.dispatchEvent({
				type : 'center',
				content : {
					center : vector.clone(),
					translation : oldCamTargetPos.clone().sub(newCamTargetPos),
					positionFromCenter : positionFromCenter
				}
			});
			_setCenterActive = false;
		}
		else {
			
				//tween camera target position and camera position
				_setCenterTween = new TWEEN.Tween(tweenValues)
											.to(targetValues,duration)
											.easing( TWEEN.Easing.Quartic.Out )
											.onUpdate(function(){
												this.camera.position.set(parseFloat(tweenValues.cx.toFixed(numDecimals)), parseFloat(tweenValues.cy.toFixed(numDecimals)), parseFloat(tweenValues.cz.toFixed(numDecimals)));
												this.camera.lookAt(lookAtVector.set(parseFloat(tweenValues.tx.toFixed(numDecimals)),parseFloat(tweenValues.ty.toFixed(numDecimals)),parseFloat(tweenValues.tz.toFixed(numDecimals))));
											}.bind(this))
											.onComplete(function(){
												this.camera.target.position.setZ(-positionFromCenter.length());
												/**
												 * Fires after setCenter() completed
												 * @event center
												 */
												this.dispatchEvent({
													type : 'center',
													content : {
														center : vector.clone(),
														translation : oldCamTargetPos.clone().sub(newCamTargetPos),
														positionFromCenter : positionFromCenter
													}
												});
												
												_setCenterActive = false;
												
												}.bind(this))
											.start();		
				
			}
		

		//this is done every frame update by OrbitZoomPan-Control
		// if (this.camera.inOrthographicMode) {
			// this.camera.toOrthographic();
		// }
		
		}	
	};
	
	/**
	 * Function to move camera to predefined relative positions from Scene.center
	 * 
	 * @method viewFrom
	 * @param {String} relativePosition allowed values are 'left','right','front','back','top','bottom'
	 */
	this.viewFrom = function(relativePosition) {
		var distanceToCenter = this.camera.position.clone().sub( this.center ).length();
		var direction = new THREE.Vector3(0,0,1); //default 'front'
		switch (relativePosition) {
			case 'left': 
				direction = new THREE.Vector3(-1,0,0);
				break;
			case 'right':
				direction = new THREE.Vector3(1,0,0);
				break;
			case 'front':
				direction = new THREE.Vector3(0,0,1);
				break;
			case 'back':
				direction = new THREE.Vector3(0,0,-1);
				break;
			case 'top':
				direction = new THREE.Vector3(0,1,0.01);
				break;	
			case 'bottom':
				direction = new THREE.Vector3(0,-1,0.01);
				break;	
			default:
				console.error('Value "' + relativePosition + '" not supported!');	
		}
		
		this.setCenter(this.center, direction.clone().multiplyScalar(distanceToCenter));
	};

	/**
	 * Adds a control to the scene
	 *
	 * @method addControl
	 */
	this.addControl = function(control) {
		// add Control to controls array
		this.controls.push(control);
		control.setScene(this);

	};

	/**
	 * Removes a control from the scene
	 *
	 * @method removeControl
	 */
	this.removeControl = function(control) {
		control.deactivate();
		var x = 0;
		while (x < this.controls.length) {
			if (control === this.controls[x]) {
				this.controls.splice(x, 1);
			}
			x++;
		}
		control.setScene(null);
	};

	/**
	 * Adds a layer to the scene
	 *
	 * @method addLayer
	 */
	this.addLayer = function(layer) {
		this.layers.push(layer);
		layer.setScene(this);
		this.root.add(layer.root);
		/**
		 * @event addlayer 
		 */
		this.dispatchEvent({ type: 'addlayer', content: layer });
	};

	/**
	 * Removes a layer from the scene
	 *
	 * @method removeLayer
	 */
	this.removeLayer = function(layer) {
		/**
		 * @event removelayer 
		 */
		this.dispatchEvent({ type: 'beforeremovelayer', content: layer });
		this.root.remove(layer.root);
		var x = 0;
		while (x < this.layers.length) {
			if (layer === this.layers[x]) {
				this.layers.splice(x, 1);
			}
			x++;
		}
		layer.setScene(null);
		/**
		 * @event removelayer 
		 */
		this.dispatchEvent({ type: 'removelayer', content: layer });
	};
	
	/**
	 * Deletes all layer objects and the layer from memory
	 * 
	 * @method disposeLayer
	 * @param {GIScene.Layer} layer
	 * 
	 *  @TODO remove all working materials first
	 *  @TODO removeEventListeners
	 */
	this.disposeLayer = function(layer) {
		this.removeLayer(layer);
		
		var objects = layer.root.getDescendants(); 
		objects.push(layer.root);
		//get all geoms, materials, textures
		var materials = [], textures = [];
		
		objects.forEach(function(object){
			//@TODO Docs fire event to clear references in other objects, eg. selectables in Control.Select
			this.dispatchEvent( { type: 'beforeDisposeObject', content: object } );
			
			if(object.geometry){
				object.geometry.dispose();
				delete object.geometry;
			}
			
			//get materials
			if (object.material) {
				if (object.material instanceof THREE.MeshFaceMaterial) {
					object.material.materials.forEach(function(material) {
						if (! GIScene.Utils.arrayContains(materials, material)) {
							materials.push(material);
						};
					});
				} else {

					if (! GIScene.Utils.arrayContains(materials, object.material)) {
						materials.push(object.material);
					};
				}
			}
			delete object.material;
			
			
			// // get textures
			// materials.forEach(function(material) {
				// var maps = ["map", "lightMap", "bumpMap", "normalMap", "specularMap", "envMap"];
				// for (var i = 0, j = maps.length; i < j; i++) {
					// if (material[maps[i]] && material[maps[i]] != null && ! GIScene.Utils.arrayContains(textures, material[maps[i]])) {
						// textures.push(material[maps[i]]);
					// };
				// };
			// }); 
			
			//dispose objects
			if(object.dispose)object.dispose();
			object = null;
		}.bind(this)
		);
		
		//get further materials from styles
		layer.styles.forEach(function(style){
			if (style.material) {
				if (style.material instanceof THREE.MeshFaceMaterial) {
					style.material.materials.forEach(function(material) {
						if (! GIScene.Utils.arrayContains(materials, material)) {
							materials.push(material);
						};
					});
				} else {

					if (! GIScene.Utils.arrayContains(materials, style.material)) {
						materials.push(style.material);
					};
				}
			}
			
		});
		
		// get textures
			materials.forEach(function(material) {
				var maps = ["map", "lightMap", "bumpMap", "normalMap", "specularMap", "envMap", "texture"]; //texture exists in RasterOverlayMaterial
				for (var i = 0, j = maps.length; i < j; i++) {
					if (material[maps[i]] && material[maps[i]] != null && ! GIScene.Utils.arrayContains(textures, material[maps[i]])) {
						textures.push(material[maps[i]]);
					};
				};
			}); 
		
		//dispose textures, materials
		textures.forEach(function (texture) {
		  texture.dispose();
		});
		textures = null;
		
		materials.forEach(function (material) {
		  material.dispose();
		});
		materials = null;
		
		objects = null;
		
		//remove THREE.EventDispatcher listeners
		for (type in layer._listeners){
			delete layer._listeners[type];
		}
		for (type in layer.loader._listeners){
			delete layer.loader._listeners[type];
		}
		
		layer = null;
	};
	
	/**
	 * Set the wireframe property for the whole scene
	 * @method setWireframe
	 * @param {Mixed} wireframeMode can be {String} 'default' | {boolean} true | {boolean} false
	 */
	
	this.setWireframe = function(wireframeMode) {
		this.root.traverse(function(object) {
			
			GIScene.Utils.WorkingMaterial.setWireframe(object,wireframeMode);
			
			// if (object.material) {
				// if(!object.userData.originalMaterial){
					// // if no working material exists create one 
					// object.userData.originalMaterial = object.material;
					// object.material = object.userData.originalMaterial.clone();
				// }
				// if(object.userData.workingMaterialFlags && (object.userData.workingMaterialFlags & GIScene.WORKINGMATERIALFLAGS.WIRE) != 0){
					// //if wire flag exist remove and check if workingmaterial is still in use, and change state or material
					// object.userData.workingMaterialFlags ^= GIScene.WORKINGMATERIALFLAGS.WIRE;
					// if(object.userData.workingMaterialFlags == 0){
						// //change back to original material
						// object.material = object.userData.originalMaterial;
						// object.userData.originalMaterial = null;
						// delete object.userData.originalMaterial;
					// }
					// else{
						// // set original state
// 						
						// //for multimaterial objects
						// if("materials" in object.material){
							// object.material.materials.forEach(
								// function (e,i,a){
									// a[i].wireframe = isTrue;
								// }
							// );
						// }
// 						
						// // for single material objects
						// else {
							// object.material.wireframe = isTrue;
						// }
					// }
				// }
				// else{
					// //set wireflag and change wireframe state
					// //set wire flag
					// object.userData.workingMaterialFlags = ("workingMaterialFlags" in object.userData)? object.userData.workingMaterialFlags ^ GIScene.WORKINGMATERIALFLAGS.WIRE : GIScene.WORKINGMATERIALFLAGS.WIRE ;
					// //change wireframe state
// 					
					// //for multimaterial objects
						// if("materials" in object.material){
							// object.material.materials.forEach(
								// function (e,i,a){
									// a[i].wireframe = isTrue;
								// }
							// );
						// }
// 						
						// // for single material objects
						// else {
							// object.material.wireframe = isTrue;
						// }
				// }
// 				
			// }
		});
	}; 

	/**
	 * Sets texturing of on or off for the whole scene
	 * @method setTexturing
	 * @param {Mixed} texturingMode can be: {String} 'default' | {boolean} true | {boolean} false
	 */
	this.setTexturing = function(texturingMode) {
		this.root.traverse(function(object) {
			
			GIScene.Utils.WorkingMaterial.setTexturing(object, texturingMode);
			
			// if (object.material && !((object instanceof THREE.Sprite)||(object instanceof THREE.ParticleSystem))) {
				// if(!object.userData.originalMaterial){
					// // if no working material exists create one 
					// object.userData.originalMaterial = object.material;
					// object.material = object.userData.originalMaterial.clone();
				// }
				// if(object.userData.workingMaterialFlags && (object.userData.workingMaterialFlags & GIScene.WORKINGMATERIALFLAGS.MAP) != 0){
					// //if texture flag exist remove and check if workingmaterial is still in use, and change state or material
					// object.userData.workingMaterialFlags ^= GIScene.WORKINGMATERIALFLAGS.MAP;
					// if(object.userData.workingMaterialFlags == 0){
						// //change back to original material
						// object.material = object.userData.originalMaterial;
						// object.userData.originalMaterial = null;
						// delete object.userData.originalMaterial;
					// }
					// else{
						// // set original state
// 						
						// //for multimaterial objects
						// if("materials" in object.material){
							// object.material.materials.forEach(
								// function (e,i,a){
									// a[i].map = (object.userData.originalMaterial.materials[i].map && object.userData.originalMaterial.materials[i].map != null)? object.userData.originalMaterial.materials[i].map.clone() : null;
									// if(a[i].map != null)a[i].map.needsUpdate = true;
									// a[i].needsUpdate = true;
								// }
							// );
						// }
// 						
						// // for single material objects
						// else{
							// object.material.map = (object.userData.originalMaterial.map && object.userData.originalMaterial.map != null)? object.userData.originalMaterial.map.clone() : null;
							// if(object.material.map != null)object.material.map.needsUpdate = true;
							// object.material.needsUpdate = true;
						// }
// 						
					// }
				// }
				// else{
					// //set mapflag and change map state
					// //set map flag
					// object.userData.workingMaterialFlags = ("workingMaterialFlags" in object.userData)? object.userData.workingMaterialFlags ^ GIScene.WORKINGMATERIALFLAGS.MAP : GIScene.WORKINGMATERIALFLAGS.MAP ;
// 					
					// //change map state
// 					
					// //for multimaterial objects
					// if("materials" in object.material){
						// object.material.materials.forEach(
							// function (e,i,a){
								// a[i].map = null;
								// a[i].needsUpdate = true;
							// }
						// );
					// }
// 					
					// // for single material objects
					// else{
						// object.material.map = null;
						// object.material.needsUpdate =true;
					// }
				// }
			// }
		});
	};
	
	/**
	 * Sets the face culling mode for the whole scene
	 * 
	 * @method setFaceCulling
	 * @param {THREE.FrontSide || THREE.BackSide || THREE.DoubleSide || 'default'} faceCullingMode
	 */
	
	this.setFaceCulling = function(faceCullingMode) { //use THREE.FrontSide || THREE.BackSide || THREE.DoubleSide || 'default'
		this.root.traverse(function(object) {
			
			GIScene.Utils.WorkingMaterial.setFaceCulling(object,faceCullingMode);
			
			// if (object.material && !((object instanceof THREE.Sprite)||(object instanceof THREE.ParticleSystem))) {
				// if(!object.userData.originalMaterial){
					// // if no working material exists create one 
					// object.userData.originalMaterial = object.material;
					// object.material = object.userData.originalMaterial.clone();
					// //set flag
					// object.userData.workingMaterialFlags = GIScene.WORKINGMATERIALFLAGS.SIDE;
				// }
// 				
				// //no wm --> create and set flag and mode
// 					
				// //else wm exists --> check if new mode == original
				// /*else*/ if (faceCullingMode == object.userData.originalMaterial.side || faceCullingMode == 'default'){
					// //remove (toggle) flag
					// object.userData.workingMaterialFlags ^= GIScene.WORKINGMATERIALFLAGS.SIDE;	
				// }
// 				
				// //set property
// 				
				// function setFC(material, faceCullingMode){
					// material.side = faceCullingMode;
					// material.needsUpdate = true;
				// }
// 				
				// //for multimaterial objects
						// if("materials" in object.material){
							// object.material.materials.forEach(
								// function (e,i,a){
									// if(faceCullingMode == 'default'){
										// var FCValue = object.userData.originalMaterial.materials[i].side;
										// // a[i].side = object.userData.originalMaterial.materials[i].side;
										// // a[i].needsUpdate = true;
									// }
									// else{
										// var FCValue = faceCullingMode;
										// // a[i].side = faceCullingMode;
										// // a[i].needsUpdate = true;
									// }
									// setFC(a[i],FCValue);
								// }
							// );
						// }
// 						
						// // for single material objects
						// else {
							// if(faceCullingMode == 'default'){
								// var FCValue = object.userData.originalMaterial.side;
								// // object.material.side = object.userData.originalMaterial.side;
								// // object.material.needsUpdate = true;
							// }
							// else{
								// var FCValue = faceCullingMode;
								// // object.material.side = faceCullingMode;
								// // object.material.needsUpdate = true;
							// }
							// setFC(object.material, FCValue);
						// }
// 				
				// //check if wm still in use --> false remove wm and switch to orignal
				// if(object.userData.workingMaterialFlags == 0){
						// //change back to original material
						// object.material = object.userData.originalMaterial;
						// object.userData.originalMaterial = null;
						// delete object.userData.originalMaterial;
				// }
// 				
			// }
		}
		);
	};
	
	/**
	 * Sets the vertex color mode for the whole scene
	 * 
	 * @method setVertexColors
	 * @param {THREE.NoColors || THREE.FaceColors || THREE.VertexColors || 'default'} vertexColorMode
	 */
	
	this.setVertexColors = function(vertexColorMode) { 
		this.root.traverse(function(object) {
			
			GIScene.Utils.WorkingMaterial.setVertexColors(object,vertexColorMode);
			
			// console.log('VC',object.userData.workingMaterialFlags);
			// if (object.material && !((object instanceof THREE.Sprite)||(object instanceof THREE.ParticleSystem))) {
				// if(!object.userData.originalMaterial){
					// // if no working material exists create one 
					// object.userData.originalMaterial = object.material;
					// object.material = object.userData.originalMaterial.clone();
					// //set flag
					// // console.log("set workingmaterial vertexColors");
					// object.userData.workingMaterialFlags = 0;//GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;
					// // console.log(object.userData.workingMaterialFlags);
				// }
// 				
				// //no wm --> create and set flag and mode
// 					
				// //else wm exists --> check if new mode == original
				// // /*else*/ if (vertexColorMode == object.userData.originalMaterial.vertexColors || vertexColorMode == 'default'){
// 					
// 							 
							 // //if default clear flag: object.userData.workingMaterialFlags &= ~GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;
							 // if (vertexColorMode == 'default'){ object.userData.workingMaterialFlags &= ~GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS; }
							 // //else
							 // else{
								 // //if multi
								 // if ("materials" in object.material){
								 	// //check isHomogenous
								 	// var isHomogenous = null;
								 	// for (var i=1; i < object.userData.originalMaterial.materials.length; i++){
								 		// isHomogenous = (object.userData.originalMaterial.materials[0].vertexColors == object.userData.originalMaterial.materials[i].vertexColors);
								 		// if(!isHomogenous){break;}
								 	// }
								 	// //if isHomogenous
								 	// if(isHomogenous){
								 		// //if newVC != object.userData.originalMaterial.materials[0].vertexColors : set flag |=
								 		// if(vertexColorMode != object.userData.originalMaterial.materials[0].vertexColors){ object.userData.workingMaterialFlags |= GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS; }
								 		// //else clear flag &= ~
								 		// else{object.userData.workingMaterialFlags &= ~GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;}
								 	// }
								 	// //else (not homogenous)
								 	// else{	
								 		// //set flag: set flag |=
								 		// object.userData.workingMaterialFlags |= GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;
								 	// }
								 // }
								 // //else (single)
								 // else{
								 	// //if newVC != object.userData.originalMaterial.vertexColors : set flag |=
								 	// if(vertexColorMode != object.userData.originalMaterial.vertexColors){ object.userData.workingMaterialFlags |= GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS; }
							 		// //else clear flag &= ~
							 		// else{object.userData.workingMaterialFlags &= ~GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;}
								 // }
							 // }
// 							 
// 							 
							 // //check multiIsHomogenous
							 // //if multiIsHomogenous == false: setMask : object.userData.workingMaterialFlags |= GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;
							 // //else 
							 // //if newVCMode != 
// 							 
							 // // var newVCMode = (vertexColorMode == 'default')? object.userData.originalMaterial.vertexColors : vertexColorMode;
					// // /*else*/ if (newVCMode != object.userData.originalMaterial.vertexColors || vertexColorMode == 'default'){
					// //remove (toggle) flag
					// // console.log("remove workingmaterial vertexColors");
					// // object.userData.workingMaterialFlags ^= GIScene.WORKINGMATERIALFLAGS.VERTEXCOLORS;	
					// // console.log(object.userData.workingMaterialFlags);
				// // }
				// //set property
// 				
				// //for multimaterial objects
				// if("materials" in object.material){
					// object.material.materials.forEach(
								// function (e,i,a){
									// if(vertexColorMode == 'default'){
										// a[i].vertexColors = object.userData.originalMaterial.materials[i].vertexColors;
										// a[i].color = object.userData.originalMaterial.materials[i].color;
										// a[i].ambient = object.userData.originalMaterial.materials[i].ambient;
										// a[i].needsUpdate = true;
									// }
									// else{
										// if (object.geometry.faces[0].vertexColors.length > 0) {
// 											
											// a[i].vertexColors = vertexColorMode;
// 											
											// if(vertexColorMode == THREE.NoColors){
												// a[i].color = new THREE.Color(0xFFFFFF);
												// a[i].ambient = new THREE.Color(0x999999);
											// }else
											// {
												// a[i].color = new THREE.Color(0xFFFFFF);
												// a[i].ambient = new THREE.Color(0xFFFFFF);
											// }
// 											
											// a[i].needsUpdate = true;
// 											
										// }
									// }
								// }
					// )
				// }
				// else {
					// //for single material models
					// if(vertexColorMode == 'default'){
						// object.material.vertexColors = object.userData.originalMaterial.vertexColors;
						// object.material.color = object.userData.originalMaterial.color;
						// object.material.ambient = object.userData.originalMaterial.ambient;
						// object.material.needsUpdate = true;
					// }
					// else{
						// if (object.geometry.faces[0].vertexColors.length > 0) {
// 							
							// object.material.vertexColors = vertexColorMode;
// 							
							// if(vertexColorMode == THREE.NoColors){
								// object.material.color = new THREE.Color(0xFFFFFF);
								// object.material.ambient = new THREE.Color(0x999999);
							// }else
							// {
								// object.material.color = new THREE.Color(0xFFFFFF);
								// object.material.ambient = new THREE.Color(0xFFFFFF);
							// }
// 							
							// object.material.needsUpdate = true;
// 							
						// }
					// }
				// }
// 				
				// //check if wm still in use --> false remove wm and switch to orignal
				// if(object.userData.workingMaterialFlags == 0){
						// //change back to original material
						// object.material = object.userData.originalMaterial;
						// object.userData.originalMaterial = null;
						// delete object.userData.originalMaterial;
				// }
			// }
		}
		);
	};
	
	/**
	 * Sets the shading mode for the whole scene
	 * 
	 * @method setShading
	 * @param { THREE.FlatShading || THREE.SmoothShading || 'default' } shadingMode
	 */
	this.setShading = function(shadingMode) { //use THREE.FlatShading || THREE.SmoothShading || 'default'
		this.root.traverse(function(object) {
			
			GIScene.Utils.WorkingMaterial.setShading(object,shadingMode);
			
			// if (object.material && !((object instanceof THREE.Sprite)||(object instanceof THREE.ParticleSystem))) {
				// if(!object.userData.originalMaterial){
					// // if no working material exists create one 
					// object.userData.originalMaterial = object.material;
					// object.material = object.userData.originalMaterial.clone();
					// //set flag
					// object.userData.workingMaterialFlags = GIScene.WORKINGMATERIALFLAGS.SHADING;
				// }
// 				
				// //no wm --> create and set flag and mode
// 					
				// //else wm exists --> check if new mode == original
				// /*else*/ if (shadingMode == object.userData.originalMaterial.shading || shadingMode == 'default'){
					// //remove (toggle) flag
					// object.userData.workingMaterialFlags ^= GIScene.WORKINGMATERIALFLAGS.SHADING;	
				// }
// 				
				// //store originalVertexNormals the first time we are changing this property
				// if(!("originalVertexNormals" in object.userData) ) {
					// // console.log('store originalVertexNormals');
					// object.userData.originalVertexNormals = [];
					// for (var i=0,l=object.geometry.faces.length; i<l ;i++){
						// // object.userData.originalVertexNormals.push(object.geometry.faces[i].vertexNormals);
						// if(object.geometry.faces[i].vertexNormals.length != 0){
							// object.userData.originalVertexNormals[object.geometry.faces[i].a] = object.geometry.faces[i].vertexNormals[0].clone();
							// object.userData.originalVertexNormals[object.geometry.faces[i].b] = object.geometry.faces[i].vertexNormals[1].clone();
							// object.userData.originalVertexNormals[object.geometry.faces[i].c] = object.geometry.faces[i].vertexNormals[2].clone();	
						// }
					// }
				// }
// 				
				// //set property
				// if(shadingMode == 'default'){
					// //restore originalVertexNormals
					// for (var i=0,l=object.geometry.faces.length; i<l ;i++){
						// if(object.userData.originalVertexNormals[object.geometry.faces[i].a] == undefined){
							// object.geometry.faces[i].vertexNormals = [];
						// }
						// else{
							// object.geometry.faces[i].vertexNormals[0] = object.userData.originalVertexNormals[object.geometry.faces[i].a];
							// object.geometry.faces[i].vertexNormals[1] = object.userData.originalVertexNormals[object.geometry.faces[i].b];
							// object.geometry.faces[i].vertexNormals[2] = object.userData.originalVertexNormals[object.geometry.faces[i].c];	
						// }	
					// }
					// // console.log('deleting originalVertexNormals');
					// object.geometry.__tmpVertices = undefined; //has been set by Geometry.computeVertexNormals()
					// delete object.userData.originalVertexNormals;
// 					
						// //for multimaterial objects
						// if("materials" in object.material){
							// object.material.materials.forEach(
								// function (e,i,a){
									// a[i].shading = object.userData.originalMaterial.materials[i].shading;
									// object.geometry.normalsNeedUpdate = true;
									// a[i].needsUpdate = true;
								// }
							// )
						// }
// 						
						// //for single materials
						// else{
							// object.material.shading = object.userData.originalMaterial.shading;
							// object.geometry.normalsNeedUpdate = true;
							// object.material.needsUpdate = true;
						// }
				// }
// 				
				// // other than 'default'
				// else{
					// if( object.geometry.faces[0].vertexNormals.length == 0 ){
						// object.geometry.computeVertexNormals();
					// }
// 					
					// //for multimaterial objects
						// if("materials" in object.material){
							// object.material.materials.forEach(
								// function (e,i,a){
									// a[i].shading = shadingMode;
									// object.geometry.normalsNeedUpdate = true;
									// a[i].needsUpdate = true;
								// }
							// )
						// }
// 						
						// //for single materials
						// else{
							// object.material.shading = shadingMode;
							// object.geometry.normalsNeedUpdate = true;
							// object.material.needsUpdate = true;
						// }
				// }
// 					
				// //check if wm still in use --> false remove wm and switch to orignal
				// if(object.userData.workingMaterialFlags == 0){
						// //change back to original material
						// object.material = object.userData.originalMaterial;
						// object.userData.originalMaterial = null;
						// delete object.userData.originalMaterial;
				// }
// 				
			// }
		}
		);
	};
	
	/**
	 * get all descendants of the layers. Optionally the layers can be filtered by an array filter function.
	 * 
	 * @method getLayerDescendants
	 * @param {Function} filter An array filter function that is applied to each layer object of the scene. Should return true or false. 
	 * @example
	 * 	//get only descendants of layers that are not Helper Layers
	 * 	scene.getLayerDescendants(function(e,i,a){
	 * 		return !(e instanceof GIScene.Layer.Helper);
	 * 		}
	 *	);
	 *  
	 */
	this.getLayerDescendants = function(filter){
		var layerDescendants = [];
		var layers = this.layers;
		
		if(typeof filter == 'function'){
			layers = layers.filter(filter);
		}
		
		for(var i=0,j=layers.length; i<j; i++){
		  layers[i].root.getDescendants(layerDescendants);
		};
		
		return layerDescendants;
	};
	
	/**
	 * @method getLayerById
	 * @param {String} id
	 * @return {GIScene.Layer} result returns the layer found by id or null
	 */
	this.getLayerById = function(id) {
		var result = null;
		for (var i=0,j=this.layers.length;i<j;i++){
			if(this.layers[i].id.toString() === id.toString()) return this.layers[i];
		}
		return result;
	};
	
	/* experimental function */
	this.sortFacesFromCamera = function() {
		//sorting function to sort face arrays according to the distance from the camera to minimize opacity rendering problems
		var sortingFunction = function (elementA, elementB){
		    var distA = this.camera.position.distanceTo(elementA);
		    var distB = this.camera.position.distanceTo(elementB);
		    // return distA - distB;
		    return distB - distA; //sort from far to near
		}.bind(this);
		
		//merge all geoms
		var mergedGeom = new THREE.Geometry();
		this.root.traverse(function(object) {
			if (object.geometry && object instanceof THREE.Mesh){
				THREE.GeometryUtils.merge(mergedGeom, object.geometry);
			}
		}.bind(this));
		
		//sort triangles
		mergedGeom.faces.sort(sortingFunction);
		// this.root.traverse(function(object) {
			// if (object.geometry && object instanceof THREE.Mesh){
				// object.geometry.faces.sort(sortingFunction);
			// }
		// });
		mergedMat = new THREE.MeshLambertMaterial({
			color : 0xD2B48C, //0xFFFF66(gelb),
			ambient : 0x8B7355, //0x7B7B33,
			emissive : 0x000000,
			depthTest: false,
			depthWrite: true,
			opacity:0.5,
			transparent:true
		});
		
		mergedMesh = new THREE.Mesh(mergedGeom, mergedMat);
		
		mergedLayer = new GIScene.Layer();
		mergedLayer.root.add(mergedMesh);
		this.disposeLayer(this.layers[0]);
		this.addLayer(mergedLayer);
		
		
	};

	var debugView = false;
	this.toggleDebugView = function() {
		if(debugView){
			debugView = false;
			this.camera.target.remove(this.camera.target.children[1]);
		}
		else{
			debugView = true;
			var sphereGeom1 = new THREE.SphereGeometry(0.1,8,8);
			var camTargetDummy = new THREE.Mesh(sphereGeom1);
			this.camera.target.add(camTargetDummy);
			
			var sphereGeom2 = new THREE.SphereGeometry(0.2,10,10);
			var pivotLightDummy = new THREE.Mesh(sphereGeom2);
			cameraLightControl.pivotLight.add(pivotLightDummy);
		}
	};
	
	//start auto initialization
	this.init();
	
	//set viewport on resize
	this.onResize = function(event) {
		var w = this.containerDiv.clientWidth;
		var h = this.containerDiv.clientHeight;
		this.renderer.setSize( w, h );
		this.effectComposer.setSize( w, h );
		this.camera.setSize( w, h );
		this.camera.updateProjectionMatrix();
		
		this.spriteCamera.left = -w/2;
		this.spriteCamera.right = w/2;
		this.spriteCamera.top = h/2;
		this.spriteCamera.bottom = -h/2;
		this.spriteCamera.updateProjectionMatrix();
		
		//update logo position
		this.logoOnResize();
	}.bind(this);
	window.addEventListener('resize', this.onResize, false);
	
	this.onUnload = function(event) {
		//@TODO free memory before reload
		
		this.stopAnimation();
		
		//deactivate all controls to removeEventListeners on canvas etc
		for(var i=0,j=this.controls.length; i<j; i++){
		  if(this.controls[i].isActive)this.controls[i].deactivate();
		};
		
		//dispose all layers
		for(var i=0,j=this.layers.length; i<j; i++){
		  this.disposeLayer(this.layers[i]); 
		};
		
		this.containerDiv.innerHTML = "";
		
		for (props in this){
			delete this[props];
		}
		
		
		
	}.bind(this);
	window.addEventListener('unload', this.onUnload, false);
};

//Provide EventDispatcher Functions
GIScene.Scene.prototype = {

	constructor : GIScene.Scene,
	
	/**
	 * get Objects by a evaluation function which recursively tries to match the objects of the scene
	 * 
	 * @method getObjectsBy
	 * @param {Function} callback
	 * @return {Array} matches
	 */
	getObjectsBy : function(callback) {
		return GIScene.Utils.getObjectsBy(this.root,callback);
	},

	/**
	 * inherited from THREE.EventDispatcher
	 *
	 * @method addEventListener
	 * @param {String} eventType
	 * @param {Function} listener function, to be called when the event is dispatched
	 */
	addEventListener : THREE.EventDispatcher.prototype.addEventListener,
	
	/**
	 * inherited from THREE.EventDispatcher
	 *
	 * @method hasEventListener
	 * @param {String} eventType
	 * @param {Function} listener function, to be removed from the event
	 */
	hasEventListener : THREE.EventDispatcher.prototype.hasEventListener,
	
	/**
	 * inherited from THREE.EventDispatcher
	 *
	 * @method removeEventListener
	 * @param {String} eventType
	 * @param {Function} listener function, to be removed from the event
	 */
	removeEventListener : THREE.EventDispatcher.prototype.removeEventListener,
	
	/**
	 * inherited from THREE.EventDispatcher
	 *
	 * @method dispatchEvent
	 * @param {Object} event
	 * @example this.dispatchEvent( { type: 'beforeRender', content: anythingToBePassedToTheListeners } );
	 *
	 */
	dispatchEvent : THREE.EventDispatcher.prototype.dispatchEvent

};