API Docs for: 1.0.2
Show:

File: GIScene\Control\CameraLight.js

/**
 * A control which adds a sprite to the viewport for interactive movement of a directional light source.
 * The Light will always stay relative to the camera. This ensures to keep a chosen lighting angle while moving the camera.
 * Changing interactively the lighting angle gives the user a good impression of surface characteristics.
 * 
 * @namespace GIScene
 * @class Control.CameraLight
 * @constructor
 * @extends GIScene.Control
 * @param {THREE.Camera} camera
 * @param {THREE.DirectionalLight} light 
 * @param {Object} [config] config.<properties> are: maxHorizontalAngle, maxVerticalAngle both Numbers in degrees between 0..90
 * 
 */

GIScene.Control.CameraLight = function (camera, light, config){
	
	//make this a control
	GIScene.Control.call(this);
	
	var defaults = {
		maxHorizontalAngle	: 65,
		maxVerticalAngle	: 65
	};
	
	/**
	 * The config which is used to initialize the CameraLight-Control. Merged from defaults and passed config Object.
	 * 
	 * @property config
	 * @type Object
	 */
	this.config = GIScene.Utils.mergeObjects(defaults, config || {});
	this.camera = camera;
	this.light = light || new THREE.DirectionalLight(0xffffff, 0.5);
	this.pivotLight = new THREE.Object3D();
	this.domElement = null; //will be set in this.activate()
	var degrees = new THREE.Vector2(0,0); //illumination angles
	
	this.maxHorizontalAngle	= this.config.maxHorizontalAngle; 
	this.maxVerticalAngle	= this.config.maxVerticalAngle;
	
	
	var STATE = { NONE : -1, PAN: 1 }; 
	var state = STATE.NONE;
	var isMouseover = false;
	
	
	//var spriteTexture = THREE.ImageUtils.loadTexture( GIScene.LIBRARYPATH + GIScene.RESOURCESPATH.replace(/([^\/])$/, "$1/") +"resources/images/sprite1.png" );
	// spriteTexture.flipY=false;
	var imgDataURL = "";


	var spriteTexture = new THREE.Texture(null);
	var spriteImg = new Image();
		spriteImg.onload = function() {
			spriteTexture.image = spriteImg;
			spriteTexture.needsUpdate = true;
		};
		spriteImg.src = imgDataURL;
	
	var spriteMaterial = new THREE.SpriteMaterial({ map : spriteTexture, opacity:0.7  });
	this.sprite = new THREE.Sprite(spriteMaterial);
	this.sprite.scale.set( 50, 50, 1 );
	
	
	var panStart, panEnd, panDelta;
	
	/**
	 * Moves the UI graphic sprite and changes the light position accordingly
	 * 
	 * @method panSprite
	 * @param {THREE.Vector2()} panDelta A Vector used to translate the UI-graphic in Pixels  
	 */
	this.panSprite = function (panDelta) {
		
		var p = this.sprite.position;
		//pan sprite
		p.set(p.x+panDelta.x, p.y-panDelta.y, p.z);
		//move light		
		this.updateLightPosition();
		
	};
	
	/**
	 * Updates the light position to the correct angle according to sprite.position
	 * 
	 * @method updateLightPosition 
	 */
	this.updateLightPosition = function(){
		
		var normalizedDeviceCoordinate = this.sprite.position.clone();
		normalizedDeviceCoordinate.x /= this.scene.canvas.width/2;
		normalizedDeviceCoordinate.y /= this.scene.canvas.height/2;
		
		degrees = normalizedDeviceCoordinate.clone().set(normalizedDeviceCoordinate.x * this.maxHorizontalAngle, normalizedDeviceCoordinate.y * -this.maxVerticalAngle);
		this.pivotLight.rotation.set(THREE.Math.degToRad(degrees.y),THREE.Math.degToRad(degrees.x),0);
		
	};
	
	/**
	 * Returns the position of the light in polar angles relative to the light.target and the camera.position
	 * 
	 * @method getIlluminationAngles
	 * @return {Object} illuminationAngles an object with two properties: "theta" and "phi" containing the actual values in degrees
	 */
	this.getIlluminationAngles = function(){
		var illumAngles = {
			'theta' : degrees.x,
			'phi'	: degrees.y
		};
		return illumAngles;
	};
	
	// this.update = function () {
		// //currently not in use
	// };
	
	var onMouseDown = function(event){
		
		if(isMouseover){
			/**
			 * Fires on mouse down but before mouse moved
			 * 
			 * @event panstart 
			 */
			this.dispatchEvent( {type:'panstart'} );
			state = STATE.PAN;
			var relativeScreenCoords = GIScene.Utils.getRelativeScreenCoordsFromDOMEvent(this.domElement, event);
			panStart = relativeScreenCoords;
			
			this.domElement.addEventListener( 'mouseup', onMouseUp, false );
		}
	}.bind(this);
	
	var onMouseMove = function(event){
		
		var viewportCoords = GIScene.Utils.getViewportCoordsFromDOMEvent(this.domElement, event);
		viewportCoords.x *= this.scene.canvas.width/2;
		viewportCoords.y *= this.scene.canvas.height/2;
		
		var distance = viewportCoords.distanceTo(this.sprite.position);
		
		if (distance <= Math.max(this.sprite.scale.x, this.sprite.scale.y)/2){
			if(isMouseover == false){
				//over
				this.sprite.material.opacity = 1;
			}
			
			isMouseover = true;
			
		} else {
			if(isMouseover == true){
				//out
				this.sprite.material.opacity = 0.7;
			}
			isMouseover = false;
		}
		if (state == STATE.PAN){
			
			panEnd = GIScene.Utils.getRelativeScreenCoordsFromDOMEvent(this.domElement, event);
			panDelta = panEnd.clone().sub(panStart);
			this.panSprite(panDelta);
			panStart = panEnd;
			
			/**
			 * Fires while panning
			 * 
			 * @event pan
			 */
			this.dispatchEvent( {type:'pan'} );
		}
	}.bind(this);
	
	var onMouseUp = function(event){
		
		if(state == STATE.PAN){
			/**
			 * Fires on mouse up after a pan operation
			 * 
			 * @event panend 
			 */
			
			this.dispatchEvent( {type:'panend'} );
		}
		
		state = STATE.NONE;
		
		this.domElement.removeEventListener( 'mouseup', onMouseUp, false );
	}.bind(this);
	
	
	var onCameraChange = function(event){
		
		// var camTargetWorld = this.camera.localToWorld(this.camera.target.position.clone());
		// this.light.target.position = camTargetWorld;
		// this.light.position.setZ(this.camera.target.position.z);
		
		// this.light.position.setZ(this.camera.position.distanceTo(camTargetWorld));	
		// if(state != STATE.PAN)return; //@TODO Käs
		// this.light.target.position = camTargetWorld;
		
	}.bind(this);
	
	/**
	 * Activates this Control
	 * 
	 * @method activate
	 *  
	 */
	this.activate = function(){
		if(this.isActive) return;
		
		//var camTargetWorld = this.camera.target.parent.localToWorld(this.camera.target.position.clone());
		var camTargetWorld = new THREE.Vector3().getPositionFromMatrix(this.camera.target.matrixWorld);
		
		//reset pivot
		this.pivotLight.rotation.set(0,0,0);
		//pivot Light should be identical with camera target position. simply add it as child;
		this.camera.target.add(this.pivotLight);
		
		this.light.position.set(0,0,this.camera.position.distanceTo(camTargetWorld));
		this.pivotLight.add(this.light);
		
		//dont update light.target manually
		//this.light.target.position.copy(camTargetWorld); //= camTargetWorld.clone(); // this only works for OrbitZoomPan: this.scene.center;
		//instead add to cam target
		this.light.target.position.set(0,0,0);
		this.camera.target.add( this.light.target );
		
		this.domElement = this.scene.canvas;
		// var initialSpritePosition = GIScene.Utils.transformViewportCoordsToRelativeScreenCoords(new THREE.Vector2(0,0),this.domElement.offsetWidth, this.domElement.offsetHeight);
		// this.sprite.position.set(initialSpritePosition.x,initialSpritePosition.y,0);
		this.sprite.position.set(0,0,0);
		this.scene.spriteRoot.add(this.sprite);
		
		//events
		this.domElement.addEventListener( 'contextmenu', function ( event ) { event.preventDefault(); }, false );
		this.domElement.addEventListener( 'mousemove', onMouseMove, false );
		this.domElement.addEventListener( 'mousedown', onMouseDown, false );
		
		//this.scene.addEventListener('cameraChange', onCameraChange);
		
		//call super class method
		GIScene.Control.prototype.activate.call(this);
	};
	
	/**
	 * Deactivates this Control
	 * 
	 * @method deactivate
	 *  
	 */
	this.deactivate = function(){
		if(!this.isActive) return;
		
		this.scene.spriteRoot.remove(this.sprite);
		
		//detach light.target from camera.target
		THREE.SceneUtils.detach(this.light.target, this.camera.target, this.scene.root);
		
		//detach light from pivot (keep last light world-position)
		THREE.SceneUtils.detach(this.light, this.light.parent, this.scene.root);
		
		
		//remove pivot from camera target
		this.camera.target.remove(this.pivotLight);
		
		
		this.domElement.removeEventListener( 'contextmenu', function ( event ) { event.preventDefault(); }, false );
		this.domElement.removeEventListener( 'mousemove', onMouseMove, false );
		this.domElement.removeEventListener( 'mousedown', onMouseDown, false );
		
		//this.scene.removeEventListener('cameraChange', onCameraChange);
		
		//call super class method
		GIScene.Control.prototype.deactivate.call(this);
	};
	
	
};

//Inherit from GIScene.Control
GIScene.Control.CameraLight.prototype = Object.create( GIScene.Control.prototype );