Code viewer for World: Enhanced SpaceHero

// Cloned by Mathias Bazin on 6 Jun 2018 from World "SpaceHero" by Ross Franey 
// Please leave this clone trail here.
 

/* 
***Disclaimer - MUSIC ADDED AFTER DEADLINE***

With this game I wanted to do something a little different. I wanted to build
my own gamefrom scratcth rather than using your world as a framework. 

I am very much looking forward to implementing AI into this world. Although the 
time constraints wont allow it for this deadline, I will use the same method as
with my election world, and havethe agent detect 2 spaces ahead, if they are 
rocks, he will move left or right, based on his currentposition in relation to 
the sun. 

As well as the new structure, i wanted to write a world that would give me the 
opportunity to showcasecollision detection and mouse controlled agents. 
This is the world I look forward to working on the most, both with and without 
AI.

The game also allowed me to implement a scailing difficulty level which I could 
not with the first.

Rules: 
Very simple. Collect the sun nodes while avoiding the astroids. 
Every 5 scores difficulty increases.


Enhanced : added visual and audio feedback for collisions and difficulty increases

*/


    const	 	CLOCKTICK 	= 75;					// speed of run - move things every n milliseconds
    const		MAXSTEPS 	= 1000;					// length of a run before final score
	const show3d = true;
	//
	const ACTION_LEFT 		= 0;		   
    const ACTION_RIGHT 		= 1;
    const ACTION_UP 			= 2;		 
    const ACTION_DOWN 		= 3;
    const ACTION_STAYSTILL 		= 4;
    
    const COLLISIONAUDIO = "/uploads/mathias/boom.mp3";
    const DIFFUPAUDIO = "/uploads/mathias/diffUp.mp3"
    
   var camera, scene, renderer, player, gem;
   var mouse = new THREE.Vector2();
   var enemies = [];
   var enemySpeed = 3;
   var scoreDiv = document.getElementById( "score" );
   var bestScoreDiv = document.getElementById( "bestScore" );
   var sphereRadius = 10;
   var enemyRangeX = 550;
   var enemyRangeY = 700;
   var gemRange = 700;
   var score = 0;
   var best = 0;
   var stars = [];
   var starSpeed = 8;
   var difficulty = 0;
   var timeToShake = 0;
   


 	var self = this;						// needed for private fn to call public fn - see below 
 	
 	function shakeCamera()
 	{
 	    if (timeToShake == 0)
 	    {
 	        camera.position.x = 0;
 	        camera.position.y = 0;
 	    }
 	    else if ( timeToShake > 0 )
 	    {
 	        camera.position.x += randomintAtoB(-20,20);
 	        camera.position.y += randomintAtoB(-20,20);
 	        
 	        timeToShake--;
 	    }
 	}
 	
 	function randomfloatAtoB ( A, B )			 
    {
     return ( A + ( Math.random() * (B-A) ) );
    }

 	function randomintAtoB ( A, B )			 
    {
        return  ( Math.round ( randomfloatAtoB ( A, B ) ) );
    }
 	
 	function initWorld()
 	{
	//
	scene = new THREE.Scene();
	camera = new THREE.PerspectiveCamera( 75, window.innerWidth/window.innerHeight, 0.1, 1000 );
	camera.position.z = 500;
	renderer = new THREE.WebGLRenderer();
	renderer.setSize( window.innerWidth, window.innerHeight );
	document.body.appendChild( renderer.domElement );
   
    //
    
    enemies = [];
    stars = [];
   
    enemyRangeX = 550;
	enemyRangeY = 700;
	gemRange = 500;

	geometry = new THREE.SphereGeometry( 15, 20, 20 );
	material = new THREE.MeshLambertMaterial( { map: THREE.ImageUtils.loadTexture('/uploads/rossfraney3/rock.jpg' )} );
	
	geometry2 = new THREE.SphereGeometry( 3, 3, 3 );
	material2 = new THREE.MeshBasicMaterial( { map: THREE.ImageUtils.loadTexture('/uploads/rossfraney3/glass.jpg' )} );

	radius = 1000;
	nboxes = 50;

	for(var i = 0; i<nboxes; i++){
	    var cube = new THREE.Mesh( geometry, material );
	    cube.castShadow = false;
	    cube.receiveShadow = false;
		scene.add( cube );
		cube.position.set(radius/2 - radius * Math.random(), radius/2 - radius * Math.random(), 0.0);
		enemies.push( cube );
	}
	for(var i = 0; i<nboxes*2; i++){
	    var star = new THREE.Mesh( geometry2, material2 );
		scene.add( star );
		star.position.set(radius/2 - radius * Math.random(), radius/2 - radius * Math.random(), 0.0);
		stars.push( star );
	}
	
	//
	
	gem = new THREE.Mesh( geometry, new THREE.MeshLambertMaterial({map: THREE.ImageUtils.loadTexture('/uploads/rossfraney3/fire.jpg'), side:THREE.FrontSide}));
		gem.position.set( gemRange/2 - gemRange * Math.random(),
						 gemRange/2 - gemRange * Math.random(),
						  0.0);
	gem.castShadow = false;
	gem.receiveShadow = false;

	scene.add( gem );

	theagent = new THREE.Mesh( geometry, new THREE.MeshBasicMaterial({map: THREE.ImageUtils.loadTexture('/uploads/rossfraney3/earthbare.jpg'), side:THREE.FrontSide}) );
				scene.add( theagent );
				window.addEventListener( 'mousemove', onMouseMove, false );
	
	//

	function onMouseMove(event) {

	// Update the mouse variable
		event.preventDefault();
		mouse.x = (event.clientX / window.innerWidth) * 2 - 1;
		mouse.y = - (event.clientY / window.innerHeight) * 2 + 1;
	
 	 // Make the sphere follow the mouse
  		var vector = new THREE.Vector3(mouse.x, mouse.y, 0.5);
		vector.unproject( camera );
		var dir = vector.sub( camera.position ).normalize();
		var distance = - camera.position.z / dir.z;
		var pos = camera.position.clone().add( dir.multiplyScalar( distance ) );
		theagent.position.copy(pos);
  
	// Make the sphere follow the mouse
//	mouseMesh.position.set(event.clientX, event.clientY, 0);
	};


	var initSkybox = function() 
    {
        var skyGeometry = new THREE.SphereGeometry ( 3000, 60, 40 );	
        var uniforms = {
	    texture: { type: 't', value: THREE.ImageUtils.loadTexture('sky.jpg')}
        };
        var skyMaterial = new THREE.MeshBasicMaterial({ map: THREE.ImageUtils.loadTexture('/uploads/rossfraney3/ESO.jpg')});
  
        var theskybox = new THREE.Mesh ( skyGeometry, skyMaterial );
        theskybox.scale.set(-5, 5, 5);
        theskybox.eulerOrder = 'XYZ'
        theskybox.renderDepth = 500.0;
        scene.add( theskybox );						
  
        var light = new THREE.PointLight(0xffffff);
        light.position.set(0,250,0);
        scene.add(light); //add light to world (From a nearby star, obviously)
        var x = "<audio  id=theaudio  src=/uploads/starter/SuspenseStrings.mp3   autoplay loop> </audio>" ;
  		$("#user_span2").html( x );
    }	
        initSkybox();

	var render = function () {
	    requestAnimationFrame( render );
        
        shakeCamera();

		for( var i = 0; i < enemies.length; i++ ){
		    if(enemies[i].position.y < -500){
	        //enemies[i].position.x = enemyRangeX/2 - enemyRangeX * Math.random();
	        enemies[i].position.y = 500;
		    } 
		    else{
		     if ( enemies[i].position.distanceTo( theagent.position ) < 2 * sphereRadius) { // if there's a player-enemy collision
		        timeToShake = 10;
			 	score = 0;
			 	let boom = new Audio( COLLISIONAUDIO );
			 	boom.play();
			 	enemySpeed = 3;
			 	starSpeed = 8;
			 	difficulty = 0;
			 	gem.position.x = gemRange/2 - gemRange * Math.random();
 			    gem.position.y = gemRange/2 - gemRange * Math.random();
			 }
			enemies[i].position.y -= enemySpeed;
			}
		}
		for( var i = 0; i < stars.length; i++ ){
		    if(stars[i].position.y < -500){
	        stars[i].position.y = 500;
		    }
		    stars[i].position.y -= starSpeed; 
		}
		gem.rotation.y += .02;
		theagent.rotation.x += .002;
		theagent.rotation.y += .02;

		var status = " <center> <b> Collect the Suns. For each multiple of 5 difficulty will increase.</b></BR>Score: " + score + " &nbsp Best Score: " + best + "  &nbsp Difficulty Level: " + difficulty + "</center>";
 		$("#user_span4").html( status );

 		if(theagent.position.distanceTo( gem.position ) < 2 * sphereRadius){ //agent grabs sun 
 			gem.position.x = gemRange/2 - gemRange * Math.random();
 			gem.position.y = gemRange/2 - gemRange * Math.random();
 			score ++;
 			var y = "<audio  id=theaudio  src=/uploads/rossfraney3/elevatording.mp3  autoplay  > </audio>" ;
  		    $("#user_span3").html( y );
 			if(score % 5 == 0){ //difficulty up !
 			    
 			    let diffUpAudio = new Audio( DIFFUPAUDIO );
 			    diffUpAudio.play()
 				enemySpeed++;
 				starSpeed++;
 				difficulty++;
 			}
 			if(score > best){
 				best = score;
 			}
 		}
		renderer.render(scene, camera);
    };
	render();
 	}
	initWorld();
	
	//