Code viewer for World: v1.0 Pool Predictor Prototype

// Cloned by lillisl2 on 6 Mar 2019 from World "v1.0 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 1 Mar 2019 from World "v0.98 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 28 Feb 2019 from World "v0.97 Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 


// Cloned by lillisl2 on 27 Feb 2019 from World "v0.95 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 25 Feb 2019 from World "v0.94 Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 


// Cloned by lillisl2 on 25 Feb 2019 from World "v0.94 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 25 Feb 2019 from World "v0.92 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 20 Feb 2019 from World "v0.91 Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 


// Cloned by lillisl2 on 20 Feb 2019 from World "v0.9 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 20 Feb 2019 from World "v0.8 Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 


// Cloned by lillisl2 on 19 Feb 2019 from World "v0.7 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 18 Feb 2019 from World "v0.6 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 16 Feb 2019 from World "v0.5 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 14 Feb 2019 from World "v0.44 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 12 Feb 2019 from World "v0.43 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 5 Feb 2019 from World "v0.3 Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 


// Cloned by lillisl2 on 4 Feb 2019 from World "v0.2 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 


// Cloned by Ian Gilligan on 1 Feb 2019 from World "lillisl2 0.2 Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 


// Cloned by lillisl2 on 1 Feb 2019 from World "0.1 Pool Predictor Prototype" by Ian Gilligan 
// Please leave this clone trail here.
 
// Cloned by Ian Gilligan on 30 Jan 2019 from World "Pool Predictor Prototype" by lillisl2 
// Please leave this clone trail here.
 
// Cloned by lillisl2 on 28 Jan 2019 from World "Cloned Bouncy Balls" by Ian Gilligan 
// Please leave this clone trail here.
 
// Cloned by Ian Gilligan on 28 Jan 2019 from World "Bouncy Balls" by Starter user 
// Please leave this clone trail here.

// ==== Starter World ===============================================================================================
// (c) Ancient Brain Ltd. All rights reserved.
// This code is only for use on the Ancient Brain site.
// This code may be freely copied and edited by anyone on the Ancient Brain site.
// This code may not be copied, re-published or used on any other website.
// To include a run of this code on another website, see the "Embed code" links provided on the Ancient Brain site.
// ==================================================================================================================

// Physijs based World
// Bouncy balls 

// ===================================================================================================================
// === Start of tweaker's box ======================================================================================== 
// ===================================================================================================================

// The easiest things to modify are in this box.
// You should be able to change things in this box without being a JavaScript programmer.
// Go ahead and change some of these. What's the worst that could happen?
AB.maxSteps                 = 100000; 
AB.screenshotStep           = 50;   
AB.clockTick                = 50;
  	// Take screenshot on this step. (All resources should have finished loading.) Default 50.


// ===================================================================================================================
// === Variables & Constants =========================================================================================
// ===================================================================================================================

const OBJPATH = "/uploads/gilligi2/";	                // path of OBJ and MTL 
const OBJNAME = "pooltable.obj";
const MTLNAME = "pooltable.mtl";
const SOUND_COLLISION = "/uploads/gilligi2/pool_sound.mp3";
const SOUND_GOOD = "/uploads/gilligi2/good_shot_sound.mp3";
const SOUND_BAD = "/uploads/gilligi2/bad_shot_sound.mp3";
const SOUND_CLICK = "/uploads/gilligi2/button_click_sound.mp3";

const MODELLENGTH 			= 11054;							
const MODELWIDTH			= 6279; 
const SCALE                 = 0.009;

const BALLSIZE              = 2;                        // size of each ball
const BALLPOS               = BALLSIZE * 100; 		    // x,z start position of balls is random in this interval
const BALLHEIGHT            = BALLSIZE * 2; 		    // y start position of balls 
const HEAVYIMPACT           = 1;     			        // heavy impacts have y velocity greater than this 

const GROUNDSIZE            = 150;                      // size of table
const startRadius           = MODELWIDTH * 0.02;        // camera start distance
const maxRadius             = MODELLENGTH * 0.03;       // camera max distance
	
const SKYCOLOR              = 0xccddff;                 //colour of sky / background (white)

// friction and restitution between 0 and 1: 
const GROUND_FRICTION 		= 0.99;                     // friction of ground
const GROUND_RESTITUTION 	= 0.7;                      // restitution of ground
const WALL_FRICTION 		= 0.99;                     // friction of wall
const WALL_RESTITUTION  	= 0.9;                      // restitution of wall
const BALL_FRICTION 		= 0.99;                     // friction of ball
const BALL_RESTITUTION 		= 0.8;                      // restitution of ball

var   LIMIT                 = [7,7,1,1];                // limit values of each type of balls [yellow, red, black, white]
var   BALL_CHOICE           = 0;                        // the ball that the user has currently selected
const BALL_TYPE             = [                         // what each value of ball choice (0-3) represent
                            "Yellow",
                            "Red", 
                            "Black",
                            "White"];
                            
// define gravity along x,y,z dimensions:
const gravity               = new THREE.Vector3 (0, -150, 0);
const BALL_MASS             = 10;                       // mass of ball

const BASE_POWER            = 500;
var   CURRENT_POWER         = 0;
var   MIN_POWER             = 1;
var   MAX_POWER             = 5;
var   SAVED_POWER           = 0;

var   difficulty            = "Easy";
var   YOURCOLOR             = "None";

var   CALCULATED            = false;
var   CALCULATING           = false;

// == Balls Placed ==
var   BALL_COUNT            = 0;
var   BALLS                 = [];
var   WHITE                 = -1;

// == Ball Colors ==
const BALL_COLORS           = {
                            0 : "Yellow",
                            1 : "Red",
                            2 : "Black",
                            3 : "White"};
const BALL_COLORS_R         = {
                            "Yellow" : 0,
                            "Red"    : 1,
                            "Black"  : 2,
                            "White"  : 3};

// == Ball Movement ==
var   OBSTRUCTION           = false;
var   isSpace               = true;
var   movFin                = false;
var   MOVING                = false;
var   last_velocity         = 0;
const SLOWFACTOR            = 0.1;

// == Score Calculation ==
var   POCKETED              = false;
var   WHITE_POT             = false;
var   BLACK_POT             = false;
var   firstHit              = true;
var   YOURSCORE             = 0;
var   OTHERSCORE            = 0;
var   FOUL                  = false;
var   setUpScene;
var   SCORE                 = 50;
var   ANGLE;

// == Shot Choice ==
var   BALLS_TO_POT          = 0;
var   BALLS_ATTEMPTED       = 0;
var   SHOT_PICKED           = 0;

// == Camera ==
ABWorld.drawCameraControls  = false; 
var cameraPocketCounter = 0;

// == UI ==
var   PAGE_SELECT = 1;
var   CURRENT_PAGE = 0;
const line4_1 = ("<p align=\"center\"><button id='yellowButton'>__</button><button id='redButton'>__</button> <button id='blackButton'>__</button> <button id='whiteButton'>__</button></p>");
const line4_3 = ("<p align=\"center\"><button id='easyButton'>Easy</button> <button id='mediumButton'>Medium</button> <button id='hardButton'>Hard</button></p>");
const line6_1 = ("<p align=\"center\"><button id='yourNButton'>__</button> <button id='yourYButton'>__</button> <button id='yourRButton'>__</button></p>");
const line6_2 = ("<p align=\"center\"><button id='decreasePowerMinButton'>- Min</button> <button id='increasePowerMinButton'>+ Min</button> <button id='decreasePowerMaxButton'>- Max</button> <button id='increasePowerMaxButton'>+ Max</button></p> ");
const line6_3 = ("<p align=\"center\"><button id='cameraFlatButton'>Flat Down View</button> <button id='cameraCornerButton'>Pocket View</button></p> ");

// == Save ==
BALL_LAYOUT = [];
BEST_SCORES = [];

var toggleBasic = false;
var toggledBasic = false;



// ===================================================================================================================
// === End of tweaker's box ==========================================================================================
// ===================================================================================================================

// You will need to be some sort of JavaScript programmer to change things below the tweaker's box.

// ===================================================================================================================
// === Inits & Loads =================================================================================================
// ===================================================================================================================


var resourcesLoaded = false;
var splashClicked = false;

function World() 
{ 
	var ground_texture;
	var self = this;
	var GROUND;
	var table;
	var p_height = 25;
	
	// --- Cue -----------------------------------------------------------------------  
	var cue_material = Physijs.createMaterial(new THREE.MeshStandardMaterial({color : "#c96120"}));
    var cue = new THREE.Mesh(new THREE.CylinderGeometry(0.5,1.5,100, 100), cue_material , 0);
    cue.position.x = -51;
    cue.position.y = 4;
    cue.rotation.z = (Math.PI / 2) * 3 - 0.05;
    var pivot = new THREE.Mesh(new THREE.SphereGeometry(0.001, 10, 10), cue_material, 0);
    pivot.add(cue);

	// --- Ball ----------------------------------------------------------------------
	var ball_material_0 = Physijs.createMaterial(new THREE.MeshLambertMaterial({color : "yellow"}), BALL_FRICTION, BALL_RESTITUTION);
	var ball_material_1 = Physijs.createMaterial(new THREE.MeshLambertMaterial({color : "red"}), BALL_FRICTION, BALL_RESTITUTION);
	var ball_material_2 = Physijs.createMaterial(new THREE.MeshLambertMaterial({color : "black"}), BALL_FRICTION, BALL_RESTITUTION);
	var ball_material_3 = Physijs.createMaterial(new THREE.MeshLambertMaterial({color : "white"}), BALL_FRICTION, BALL_RESTITUTION);
	var ball_geometry = new THREE.SphereGeometry(BALLSIZE, 100, 100);
	
	// --- Space ---------------------------------------------------------------------
	var space_material = Physijs.createMaterial(new THREE.MeshStandardMaterial({color : "#c96120"}));
	
function loadResources()		// asynchronous file loads - call initScene() when all finished 
{
    var m = new THREE.MTLLoader();
	m.setTexturePath (OBJPATH);
	m.setPath        (OBJPATH);
	
	m.load ( MTLNAME, function ( materials ) 
	{
		materials.preload(); 
		var o = new THREE.OBJLoader();
		o.setMaterials ( materials );
		o.setPath ( OBJPATH );
		
		o.load ( OBJNAME, function ( object ) 
		{
			table = object;
			if ( asynchFinished() )	initScene();		 
		});
	});}

function asynchFinished()		// all file loads returned 
{
	if (! table){
	    return false;}
	return true;}
	
function initScene()			// all file loads have returned 
{
	// --- Light ------------------------------------------------------------------
	var light = new THREE.DirectionalLight(0xFFFFFF, 0.4);
	// close to origin, high up, works best for shadows
	light.position.set(0, 40, 0);
	light.target.position.copy(ABWorld.scene.position);
	light.castShadow = true;
	// how far away to draw shadows:
	light.shadow.camera.left    = -20;
	light.shadow.camera.right   =  20;
	light.shadow.camera.bottom  = -20;
	light.shadow.camera.top     =  20;
    // higher quality shadows at expense of computation time:
	light.shadow.mapSize.width  = 2048;
	light.shadow.mapSize.height = 2048;
	light.shadow.bias           = -0.1;
	ABWorld.scene.add(light);
	var light1 = new THREE.DirectionalLight(0xFFFFFF, 0.4);
	var light2 = new THREE.DirectionalLight(0xFFFFFF, 0.4);
	var light3 = new THREE.DirectionalLight(0xFFFFFF, 0.4);
	var light4 = new THREE.DirectionalLight(0xFFFFFF, 0.4);
	light1.position.set(100, 30, 0);
	light2.position.set(-100, 30, 0);
	light3.position.set(0, 30, 100);
	light4.position.set(0, 30, -100);
	ABWorld.scene.add(light1);
	ABWorld.scene.add(light2);
	ABWorld.scene.add(light3);
	ABWorld.scene.add(light4);

	// --- Table -------------------------------------------------------------------
	table.position.y = 0;
    table.position.x = -15;
    table.position.z = 0; 
    ABWorld.scene.add(table);

	// --- Ground ------------------------------------------------------------------
	var ground_material = Physijs.createMaterial(new THREE.MeshStandardMaterial({wireframe : true}), GROUND_FRICTION, GROUND_RESTITUTION);
	// ground as plane allows bounces but seems to be infinite plane
	var ground = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.93, 1, MODELWIDTH * SCALE * 0.68), ground_material, 0);
	ground.position.y = p_height;
	ground.collisions = 0;
	ground.receiveShadow = true;
	GROUND = ground;
	
	// --- Floor -------------------------------------------------------------------
    var floor_material = Physijs.createMaterial(new THREE.MeshStandardMaterial({color: "#a0192b"}), 0.1, 0.1);
	var floor = new Physijs.PlaneMesh(new THREE.PlaneGeometry(1000, 1000), floor_material);
    floor.rotation.x = (Math.PI / 2) * 3;
    floor.receiveShadow = true;
    floor.addEventListener('collision', function(other_object, relative_velocity, relative_rotation, contact_normal) 
	{
	    POCKETED = false;
	    FOUL = true;
        removeBall(other_object);
	});
	ABWorld.scene.add(floor);

	// --- Wall --------------------------------------------------------------------
	var wall_material = Physijs.createMaterial(new THREE.MeshStandardMaterial({wireframe : true}), WALL_FRICTION, WALL_RESTITUTION);
	var wall1 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 1.03, 5.9, MODELWIDTH * SCALE * 0.055), wall_material, 0);
	wall1.position.z = 27.5;
    wall1.collisions = 0;
	wall1.receiveShadow = true;
	var wall2 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 1.03, 5.9, MODELWIDTH * SCALE * 0.055), wall_material, 0);
	wall2.position.z = -27.7;
    wall2.collisions = 0;
	wall2.receiveShadow = true;
	var wall3 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.02, 5.9, MODELWIDTH * SCALE * 0.92), wall_material, 0);
	wall3.position.x = 52.2;
	wall3.collisions = 0;
	wall3.receiveShadow = true;
	var wall4 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.02, 5.9, MODELWIDTH * SCALE * 0.92), wall_material, 0);
	wall4.position.x = -52.2;
    wall4.collisions = 0;
	wall4.receiveShadow = true;

    // --- Bumper --------------------------------------------------------------------
    var bumper1 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.33, 5.9, MODELWIDTH * SCALE * 0.075 * 1.65), wall_material, 0);
	bumper1.position.z = 23.5;
	bumper1.position.x = 23.5;
	bumper1.position.y = -0.1;
	bumper1.collisions = 0;
	bumper1.receiveShadow = true;
    var bumper2 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.33, 5.9, MODELWIDTH * SCALE * 0.075 * 1.65), wall_material, 0);
	bumper2.position.z = 23.5;
	bumper2.position.x = -23;
	bumper2.collisions = 0;
	bumper2.receiveShadow = true;
	var bumper3 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.33, 5.9, MODELWIDTH * SCALE * 0.075 * 1.58), wall_material, 0);
	bumper3.position.z = -23.5;
	bumper3.position.x = 23.5;
	bumper3.position.y = -0.1;
	bumper3.collisions = 0;
	bumper3.receiveShadow = true;
	var bumper4 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.33, 5.9, MODELWIDTH * SCALE * 0.075 * 1.58), wall_material, 0);
	bumper4.position.z = -23.5;
	bumper4.position.x = -23;
	bumper4.collisions = 0;
	bumper4.receiveShadow = true;
	var bumper5 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.047, 5.9, MODELWIDTH * SCALE * 0.56), wall_material, 0);
	bumper5.position.z = 0;
	bumper5.position.x = 49;
	bumper5.receiveShadow = true;
	var bumper6 = new Physijs.BoxMesh(new THREE.BoxGeometry(MODELLENGTH * SCALE * 0.047, 5.9, MODELWIDTH * SCALE * 0.56), wall_material, 0);
	bumper6.position.z = 0;
	bumper6.position.x = -49;
	bumper6.receiveShadow = true;
	ground.add(wall1);
	ground.add(wall2);
	ground.add(wall3);
	ground.add(wall4);
	ground.add(bumper1);
	ground.add(bumper2);
	ground.add(bumper3);
	ground.add(bumper4);
	ground.add(bumper5);
	ground.add(bumper6);

    // --- Bumper Edge ---------------------------------------------------------------
    var bumperedge1 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge1.position.z = 23;
	bumperedge1.position.x = 7.8;
	bumperedge1.receiveShadow = true;
    var bumperedge2 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge2.position.z = 23;
	bumperedge2.position.x = -7.5;
	bumperedge2.receiveShadow = true;
    var bumperedge3 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge3.position.z = -23.1;
	bumperedge3.position.x = 7.8;
	bumperedge3.receiveShadow = true;
    var bumperedge4 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge4.position.z = -23.1;
	bumperedge4.position.x = -7.5;
	bumperedge4.receiveShadow = true;
	var bumperedge5 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge5.position.z = 23;
	bumperedge5.position.x = 39.1;
	bumperedge5.receiveShadow = true;
    var bumperedge6 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge6.position.z = 23;
	bumperedge6.position.x = -39;
	bumperedge6.receiveShadow = true;
    var bumperedge7 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge7.position.z = -23.1;
	bumperedge7.position.x = 39.1;
	bumperedge7.receiveShadow = true;
    var bumperedge8 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge8.position.z = -23.1;
	bumperedge8.position.x = -39;
	bumperedge8.receiveShadow = true;
    var bumperedge9 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge9.position.z = 14.2;
	bumperedge9.position.x = 49.6;
	bumperedge9.receiveShadow = true;
    var bumperedge10 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge10.position.z = 14.2;
	bumperedge10.position.x = -49.6;
	bumperedge10.receiveShadow = true;
    var bumperedge11 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge11.position.z = -14.2;
	bumperedge11.position.x = 49.6;
	bumperedge11.receiveShadow = true;
    var bumperedge12 = new Physijs.CylinderMesh( new THREE.CylinderGeometry( 3, 2.9, 5.7, 100 ), wall_material );
    bumperedge12.position.z = -14.2;
	bumperedge12.position.x = -49.6;
	bumperedge12.receiveShadow = true;   
	
    ground.add(bumperedge1);
    ground.add(bumperedge2);
    ground.add(bumperedge3);
    ground.add(bumperedge4);
    ground.add(bumperedge5);
    ground.add(bumperedge6);
    ground.add(bumperedge7);
    ground.add(bumperedge8);
    ground.add(bumperedge9);
    ground.add(bumperedge10);
    ground.add(bumperedge11);
    ground.add(bumperedge12);
    
    ABWorld.scene.add(ground);

    // --- Pocket --------------------------------------------------------------------
	var pocket_material = Physijs.createMaterial(new THREE.MeshStandardMaterial());
    var pocket = new Physijs.SphereMesh(new THREE.SphereGeometry(1), pocket_material, 0);
    pocket.position.y = -4;
    var pocket1 = new Physijs.SphereMesh(new THREE.SphereGeometry(4, 15, 15), pocket_material, 0);
    pocket1.position.y = p_height;
    pocket1.position.z = 21.5;
    pocket1.position.x = 47;
    var pocket2 = new Physijs.SphereMesh(new THREE.SphereGeometry(4, 15, 15), pocket_material, 0);
    pocket2.position.y = p_height;
    pocket2.position.z = -21.5;
    pocket2.position.x = 47;
    var pocket3 = new Physijs.SphereMesh(new THREE.SphereGeometry(4, 15, 15), pocket_material, 0);
    pocket3.position.y = p_height;
    pocket3.position.z = 24;
    var pocket4 = new Physijs.SphereMesh(new THREE.SphereGeometry(4, 15, 15), pocket_material, 0);
    pocket4.position.y = p_height;
    pocket4.position.z = 21.5;
    pocket4.position.x = -47;
    var pocket5 = new Physijs.SphereMesh(new THREE.SphereGeometry(4, 15, 15), pocket_material, 0);
    pocket5.position.y = p_height;
    pocket5.position.z = -21.5;
    pocket5.position.x = -47;
    var pocket6 = new Physijs.SphereMesh(new THREE.SphereGeometry(4, 15, 15), pocket_material, 0);
    pocket6.position.y = p_height;
    pocket6.position.z = -24;
    pocket.add(pocket1);
    pocket.add(pocket2);
    pocket.add(pocket3);
    pocket.add(pocket4);
    pocket.add(pocket5);
    pocket.add(pocket6);
    pocket.addEventListener('collision', function(other_object, relative_velocity, relative_rotation, contact_normal) 
	{
	    POCKETED = true;
        removeBall(other_object);
	});
	pocket.visible = false;
    ABWorld.scene.add(pocket);
	ABWorld.render();
	console.log("Resources loaded.");
	resourcesLoaded = true;}

// ===================================================================================================================
// === Button Functions ==============================================================================================
// ===================================================================================================================

function decreasePowerMin(){
    if (MIN_POWER > 1) {MIN_POWER -= 1;}
}

function increasePowerMin(){
    if (MIN_POWER < MAX_POWER) {MIN_POWER += 1;}
}

function decreasePowerMax(){
    if (MIN_POWER < MAX_POWER) {MAX_POWER -= 1;}
}
    
function increasePowerMax(){
    if (MAX_POWER < 5) {MAX_POWER += 1;}
}

function selectBall(x)
{
    BALL_CHOICE = x;
}
    
function setDifficulty(x){
    difficulty = x;
}

function setColor(x){
    YOURCOLOR = x;
}

function pageChange(x){
    PAGE_SELECT = x;
}

function cameraFlatDown(){
    ABWorld.camera.position.set(0,100,0);
}

function cameraSomeCornerPocket(){

    if (cameraPocketCounter % 4 === 0){
        ABWorld.camera.position.set(50,50,30);
    }
    else if (cameraPocketCounter % 4 == 1){
        ABWorld.camera.position.set(50,50,-30);
    }
    else if(cameraPocketCounter % 4 == 2){
        ABWorld.camera.position.set(-50,50,-30);
    }
    else if(cameraPocketCounter % 4 == 3){
       ABWorld.camera.position.set(-50,50, 30); 
    }
    cameraPocketCounter = cameraPocketCounter + 1;
}

// ===================================================================================================================
// === Shot Calculations =============================================================================================
// ===================================================================================================================


function calculateShot()
{
    if(validSetUp() === true){
        CALCULATING = true;
        initOptions();
        prepareHit();
    }
}

function prepareHit()
{
    if (YOURCOLOR == "None"){
        BALLS_TO_POT = 14;
    }
    else if (YOURCOLOR != "None"){
        BALLS_TO_POT = colorCount(YOURCOLOR);
        if (BALLS_TO_POT === 0){
            BALLS_TO_POT = 1;
        }
    }
    BALLS_ATTEMPTED = 0;
    CURRENT_POWER = MIN_POWER;
    if (toggledBasic === false){
        for (var i in BALLS){
            var x = BALLS[i].position.x;
            var z = BALLS[i].position.z;
            var name = BALLS[i].name;
            BALL_LAYOUT.push([x, p_height + 0.5, z, BALL_COLORS_R[name]]);
        }
    }
    hitBall();

}

function hitBall()
{
    movFin = false;
    MOVING = true;
    ABWorld.scene.addEventListener("update", function(){
        if (movementFinished()){
            if (CALCULATED === false){
                SCORE = calculateScore(YOURSCORE, OTHERSCORE, WHITE_POT, BLACK_POT);
                console.log("Score for this turn: " + SCORE + ".");
            }
            clearTable();
        }
    });
    firstHit = true;
    if (CALCULATED === false){
        ANGLE = ballChoice();
    }
    if (isSpace === true){
        cue = makeCue(ANGLE);
        WHITE.applyCentralImpulse(ANGLE);
        removeCue(cue);
    }
    else{firstHit = false}
}

function clearTable()
{
    while (BALLS.length !== 0){
        LIMIT[BALL_COLORS_R[BALLS[0].name]] += 1;
        BALLS[0].name = "None";
        BALLS[0].geometry.dispose();
        BALLS[0].material.dispose();
        ABWorld.scene.remove(BALLS[0]);
        BALLS.splice(0, 1);
    }
    reset();
}

function reset()
{
    BALL_COUNT = 0;
    WHITE_POT = false;
    BLACK_POT = false;
    YOURSCORE = 0;
    OTHERSCORE = 0;
    FOUL = false;
    if (toggledBasic === true){
        YOURCOLOR = "None";
        basicSetUp();
    }
    else{
        for (var i in BALL_LAYOUT){
            createBall(BALL_LAYOUT[i][0], BALL_LAYOUT[i][1], BALL_LAYOUT[i][2], BALL_LAYOUT[i][3]);
        }
        MOVING = false;
        $("#simulate").css({"background-color" : "red" , "color" : "white", "padding" : "15px", "font-size" : "25px", "border-radius" : "10px", "font-weight" : "bold"});
        if (CALCULATING === true && BALLS_TO_POT > BALLS_ATTEMPTED){
            hitBall();
        }
        else if (CALCULATING === true && BALLS_TO_POT == BALLS_ATTEMPTED){
            CALCULATING = false;
            CALCULATED = true;
            console.log("Finished Calculations");
            console.log("Best Scores: ", BEST_SCORES);
            initPostOptions();
        }
    }
}

function calculateScore(x, y, w, b)
{
    var score = 50;
    if ((FOUL === true || w === true) && y === 0){
        score += (-8) + (x * 7);
    }
    else{
        score += (-9 * y) + (x * 7);
    }
    if (b === true && (w === true || YOURCOLOR == "None" || LIMIT[BALL_COLORS_R[YOURCOLOR]] != 7)){
        score = -1;
    }
    else if (b === true){
        score = 100;
    }
    if (BEST_SCORES.length < 3){
        BEST_SCORES.push([score, ANGLE, SAVED_POWER]);
    }
    else{
        chooseBest(score, ANGLE, SAVED_POWER);
    }
    return(score);
}

function chooseBest(s, a, p)
{
    var min = 0;
    for (var i in BEST_SCORES){
        if (BEST_SCORES[i][0] < BEST_SCORES[min][0]){
            min = i;
        }
    }
    if (BEST_SCORES[min][0] < s){
        BEST_SCORES[min] = [s, a, p];
        soundGood();
    }
    else{soundBad()}
}

function validSetUp()
{
    if (LIMIT[2] + LIMIT[3] > 0){
        errorSplash("Invalid Set Up! Black or white ball missing.");
        return(false);
    }
    else if (YOURCOLOR == "None" && LIMIT[0] + LIMIT[1] + LIMIT[2] + LIMIT[3] > 0){
        errorSplash("Invalid Set Up! Incorrect setting of your color, should be Yellow or Red.");
        return(false);
    }
    else if ((YOURCOLOR == "Red" || YOURCOLOR == "Yellow") && LIMIT[0] + LIMIT[1] === 0){
        errorSplash("Invalid Set Up! Incorrect setting of your color, should be None.");
        return(false);
    }
    else{
        return(true);
    }
}

function colorCount(color)
{
    var x = 0;
    for (var i in BALLS){
        if (BALLS[i].name == color){
            x += 1;
        }
    }
    return(x);
}

function makeCue(angle)
{
    firstHit = true;
    if (WHITE != -1){
        pivot.position.x = WHITE.position.x;
        pivot.position.y = WHITE.position.y;
        pivot.position.z = WHITE.position.z;
        var rotate = -(Math.atan((angle.z - WHITE.position.z)/(angle.x - WHITE.position.x)));
        if((angle.x - WHITE.position.x) < 0){
            if ((angle.z - WHITE.position.z) < 0){
                rotate -= Math.PI;
            }
            else if ((angle.z - WHITE.position.z) > 0){
                rotate += Math.PI;
            }
            else {rotate = Math.PI}
        }
        pivot.rotation.y = rotate;
        ABWorld.scene.add(pivot);
        return pivot;
    }
}

function waiting(ms) {
    return new Promise(resolve => setTimeout(resolve, ms));
}

async function removeCue(object){
        await waiting(2000);
        object.geometry.dispose();
        object.material.dispose();
        ABWorld.scene.remove(object);    
}


function isValid(ball){
    if (YOURCOLOR == "None"){
        if (ball.name == "Yellow" || ball.name == "Red"){
            console.log("Valid Selection");
            return(true);
        }
    }
    else if (ball.name == YOURCOLOR){
        console.log("Valid Selection");
        return(true);
    }
    else if (ball.name == "Black" && LIMIT[BALL_COLORS_R[YOURCOLOR]] == 7){
        return(true);
    }
    return(false);
}

function errorSplash(msg){
    AB.newSplash ("<p align=\"center\">Error!</p><p align=\"center\"> "+ msg +"</p>");
    $("#splashbutton").click (function()        
        {
                AB.removeSplash();                      // remove splash screen 
                soundClick();
                splashClicked = true;
        });
}

function ballChoice()
{
    // choose a valid ball to calculate angle
    var valid_balls = 0;
    for (var i in BALLS){
        if (isValid(BALLS[i])){
            if (valid_balls == BALLS_ATTEMPTED){
                var angle = pickAngle(BALLS[i]);
                console.log(angle);
                if (CURRENT_POWER == MAX_POWER){
                    BALLS_ATTEMPTED += 1;
                    CURRENT_POWER = MIN_POWER;
                }
                else{
                    CURRENT_POWER += 1;
                }
                return(angle);
            }
            valid_balls += 1;
        }
    }
}

function pickAngle(ball)
{
    // calculate angle to hit ball
    var length = (Math.sqrt(Math.pow((WHITE.position.x - ball.position.x), 2) + Math.pow((WHITE.position.z - ball.position.z), 2)));
    if (checkSpace(ball, (BASE_POWER * CURRENT_POWER * ((ball.position.x - WHITE.position.x) / length)), (BASE_POWER * CURRENT_POWER * ((ball.position.z - WHITE.position.z) / length))) === true){
        SAVED_POWER = CURRENT_POWER;
        isSpace = true;
        return(new THREE.Vector3 ((BASE_POWER * CURRENT_POWER * ((ball.position.x - WHITE.position.x) / length)), 0, (BASE_POWER * CURRENT_POWER * ((ball.position.z - WHITE.position.z) / length))));
    }
    else{
        isSpace = false;
        return(new THREE.Vector3 (0, 0, 0));
    }
}

function checkSpace(ball, x, z)
{
    OBSTRUCTION = false;
    var length = (Math.sqrt(Math.pow((WHITE.position.x - ball.position.x), 2) + Math.pow((WHITE.position.z - ball.position.z), 2))) - 5;
    var space = new Physijs.BoxMesh(new THREE.BoxGeometry(length, 1, 4), space_material, 0.00001);
    space.position.x = (ball.position.x + WHITE.position.x) / 2;
    space.position.y = p_height + 4;
    space.position.z = (ball.position.z + WHITE.position.z) / 2;
    space.rotation.y = -(Math.atan((z - WHITE.position.z)/(x - WHITE.position.x)));
    space.addEventListener('collision', function(other_object, relative_velocity, relative_rotation, contact_normal) 
	{
	    if (other_object.name == "Red" || other_object.name == "Yellow" || other_object.name == "Black"){
	        console.log("IT BAD", other_object.name);
	        OBSTRUCTION = true;
	        space.geometry.dispose();
            space.material.dispose();
            ABWorld.scene.remove(space);
	    }
	});
    ABWorld.scene.add(space);
    console.log(OBSTRUCTION, "obstructor");
    if (OBSTRUCTION === false){
        space.geometry.dispose();
        space.material.dispose();
        ABWorld.scene.remove(space);
        return(true);}
    else {return false}
}

function movementFinished()
{
    if (firstHit === false && movFin === false){
        var vel = 0;
        for (var x in BALLS){
            vel += BALLS[x].getAngularVelocity().length();
        }
        if (vel < 0.2){
            console.log("Movement Finished!");
            movFin = true;
            return(true);
        }
        return(false);
    }
}

function simulateShot()
{
    if (MOVING === false){
        ANGLE = BEST_SCORES[SHOT_PICKED][1];
        hitBall();
    }
    else {errorSplash("Wait for current simulation to end.");}
}

// ===================================================================================================================
// === Ball Creation =================================================================================================
// ===================================================================================================================


function basicSetUp()
{
    if (BALL_COUNT === 0){
        createBall(-30, p_height + 0.5, 0, 3); //white
        createBall(30, p_height + 0.5, 0, 2); //black
        createBall(23, p_height + 0.5, 0, 1); //red
        createBall(26.5, p_height + 0.5, -2, 1); //red
        createBall(30, p_height + 0.5, 4, 1); //red
        createBall(33.5, p_height + 0.5, 2, 1); //red
        createBall(33.5, p_height + 0.5, -6, 1); // red
        createBall(37, p_height + 0.5, -4, 1); //red
        createBall(37, p_height + 0.5, 8, 1); //red
        createBall(26.5, p_height + 0.5, 2, 0); //yellow
        createBall(30, p_height + 0.5, -4, 0); //yellow
        createBall(33.5, p_height + 0.5, -2, 0); //yellow
        createBall(33.5, p_height + 0.5, 6, 0); //yellow
        createBall(37, p_height + 0.5, 0, 0); //yellow
        createBall(37, p_height + 0.5, 4, 0); //yellow
        createBall(37, p_height + 0.5, -8, 0); //yellow
        MOVING = false;
        
    }
    if (CALCULATING === true && BALLS_TO_POT > BALLS_ATTEMPTED){
        hitBall();
    }
    else if (CALCULATING === true && BALLS_TO_POT == BALLS_ATTEMPTED){
            CALCULATING = false;
            CALCULATED = true;
            console.log("Finished Calculations");
            console.log("Best Scores: ", BEST_SCORES);
            initPostOptions();

    }
}

function createBall(x, y, z, CURRENT_COLOR) 	 
{
	
 if ( resourcesLoaded && LIMIT[CURRENT_COLOR] !== 0 && splashClicked)		  
 {  
    var ball;
    // choose ball color
    if (CURRENT_COLOR === 0){
        ball = new Physijs.SphereMesh(ball_geometry, ball_material_0, BALL_MASS);
    }
    else if (CURRENT_COLOR == 1){
        ball = new Physijs.SphereMesh(ball_geometry, ball_material_1, BALL_MASS);
    }
    else if (CURRENT_COLOR == 2){
        ball = new Physijs.SphereMesh(ball_geometry, ball_material_2, BALL_MASS);
    }
    else{
        ball = new Physijs.SphereMesh(ball_geometry, ball_material_3, BALL_MASS);
    }
    ball.name = BALL_COLORS[CURRENT_COLOR];
	ball.collisions = 0;
	ball.castShadow = true;
	//ball.position.set(1, 40, 1);
	ball.position.set(x, y + 2, z);
	ball.addEventListener('collision', function(other_object, relative_velocity, relative_rotation, contact_normal) 
	{
	    if (other_object != GROUND){
	    	var mainImpact = relative_velocity.y;		// impact in direction of gravity
 		    if (Math.abs(mainImpact) > HEAVYIMPACT)			// main impact, not lesser ones as it settles  
		    {
			    soundCollision();
		    }
	    }
	    if (ball == WHITE && other_object != GROUND){

	        if (firstHit === true){
	            if(other_object.name == YOURCOLOR || YOURCOLOR == "None"){
	                console.log("Valid Hit");
	            }
	            else if (LIMIT[BALL_COLORS_R[YOURCOLOR]] == 7 && other_object.name == "Black"){
	                console.log("Valid Hit");
	            }
	            else{
	                console.log("Invalid Hit");
	                FOUL = true;
	            }
	            firstHit = false;

	       }
        }

	});
	BALLS.push(ball);
	if (CURRENT_COLOR == 3){
	    WHITE = ball;
	}
	BALL_COUNT += 1;
	ABWorld.scene.add(ball);
	ball.setDamping(0.3, 0.3);
	LIMIT[CURRENT_COLOR] -= 1;
 }
}

function removeBall(ball)
{
    var index = BALLS.indexOf(ball);
    if (index > -1) {
        BALLS.splice(index, 1);
    }
    if (POCKETED === true){
        if (ball.name == YOURCOLOR){
            console.log("Good Shot!");
            YOURSCORE += 1;
        }
        else if (YOURCOLOR == "None" && (ball.name !== "Black" && ball.name !== "White")){
            console.log("You are now " + ball.name + "s.");
            YOURCOLOR = ball.name;
            YOURSCORE += 1;
        }
        else if (ball.name == "Black"){
            BLACK_POT = true;
        }
        else if (ball.name == "White"){
            WHITE_POT = true;
            console.log("Foul Shot.");
        }
        else{
            console.log("Bad Shot :(");
            OTHERSCORE += 1;
        }
    }
    LIMIT[BALL_COLORS_R[ball.name]] += 1;
    ball.name = "None";
    ball.geometry.dispose();
    ball.material.dispose();
    ABWorld.scene.remove(ball);

}

// ===================================================================================================================
// === Keyboard & Mouse Control ======================================================================================
// ===================================================================================================================


var OURKEYS = [49, 50, 51, 52];
function ourKeys(event) {return(OURKEYS.includes(event.keyCode));}
function keyHandler(event)           
{
    if      (! ABRun.runReady)          {return true;}            // not ready yet 
    // if not handling this key, send it to default: 
    if      (! ourKeys(event))          {return true;}
    // else handle it and prevent default:
    if      ( event.keyCode == 49 )     {selectBall(0);}
    else if ( event.keyCode == 50 )     {selectBall(1);} 
    else if ( event.keyCode == 51 )     {selectBall(2);}
    else if ( event.keyCode == 52 )     {selectBall(3);} 
    event.stopPropagation(); event.preventDefault(); return false;
}

var startX, startY; 
var dragevents;         // number of events in the current drag 

function click(event)
{
    console.log(event.x, event.y);
    var object = (ABWorld.hitsObjectPoint(event.x, event.y, GROUND));  // returns null if no hit
    if (object !== null){
        createBall(object.x, object.y, object.z, BALL_CHOICE);}
}

//for future drag functionality
function initTouch (event)
{
    console.log(event);
    var object = (ABWorld.hitsObjectPoint(x, y, GROUND));  // returns null if no hit
    if (object !== null){
        createBall(object.x, object.y, object.z, BALL_CHOICE);}
}

// ===================================================================================================================
// === Unit Tests  ===================================================================================================
// ===================================================================================================================

function All_tests(){
    test_suite_decreasePowerMin();
    test_suite_increasePowerMin();
    test_suite_decreasePowerMax();
    test_suite_increasePowerMax();
    test_suite_selectBall();
    test_suite_setColor();
    test_suite_pageChange();
    
}



function test_suite_decreasePowerMin(){
    // this suite will test the value of Min_POWER Variable which changed when the "-MIN button" is clicked
   test_DPMin_1();
   test_DPMin_2();
   test_DPMin_3();
}

function test_DPMin_1(){
    MIN_POWER = 3;
    decreasePowerMin();
    try { 
        if(MIN_POWER != 2) throw "MIN_POWER not decremented";
        
    }
    catch(e){
        console.log("ERROR in test_DPMin_1: " + e );
    }
} 
function test_DPMin_2(){
    MIN_POWER = 1;
    decreasePowerMin();
    try { 
        if(MIN_POWER > 1) throw "MIN_POWER is less then 1";
        
    }
    catch(e){
        console.log("ERROR in test_DPMin_2: " + e );
    }
} 
function test_DPMin_3(){
    MIN_POWER = 5;
    decreasePowerMin();
    decreasePowerMin();
    decreasePowerMin();
    try { 
        if(MIN_POWER != 2) throw "MIN_POWER wasn't decremented 3 times";
        
    }
    catch(e){
        console.log("ERROR in test_DPMin_3: " + e );
    }
} 

function test_suite_increasePowerMin(){
    // this suite will test the value of Min_POWER Variable which changed when the "+MIN button" is clicked and doesnt exceed MAX_POWER
   test_IPMin_1();
   test_IPMin_2();
   test_IPMin_3();
}

function test_IPMin_1(){
    MAX_POWER = 5;
    MIN_POWER = 3;
    increasePowerMin();
    try { 
        if(MIN_POWER != 4) throw "MIN_POWER not incremented";
        
    }
    catch(e){
        console.log("ERROR in test_IPMin_1: " + e );
    }
} 
function test_IPMin_2(){
    MAX_POWER = 5;
    MIN_POWER = 5;
    increasePowerMin();
    try { 
        if(MIN_POWER > MAX_POWER ) throw "MIN_POWER is greater then Max_POWER";
        
    }
    catch(e){
        console.log("ERROR in test_IPMin_2: " + e );
    }
} 
function test_IPMin_3(){
    MAX_POWER = 5;
    MIN_POWER = 1;
    increasePowerMin();
    increasePowerMin();
    increasePowerMin();
    try { 
        if(MIN_POWER != 4) throw "MIN_POWER wasn't incremented 3 times";
        
    }
    catch(e){
        console.log("ERROR in test_IPMin_3: " + e );
    }
}

function test_suite_decreasePowerMax(){
    // this suite will test the value of MAX_POWER Variable which changed when the "-MAX button" is clicked and doesnt drop below MIN_POWER
   test_DPMax_1();
   test_DPMax_2();
   test_DPMax_3();
}

function test_DPMax_1(){
    MAX_POWER = 5;
    MIN_POWER = 3;
    decreasePowerMax();
    try { 
        if(MAX_POWER != 4) throw "MAX_POWER not decremented";
        
    }
    catch(e){
        console.log("ERROR in test_DPMax_1: " + e );
    }
} 
function test_DPMax_2(){
    MAX_POWER = 5;
    MIN_POWER = 5;
    decreasePowerMax();
    try { 
        if(MIN_POWER > MAX_POWER ) throw "Max_POWER is less then MIN_POWER";
        
    }
    catch(e){
        console.log("ERROR in test_DPMax_2: " + e );
    }
} 
function test_DPMax_3(){
    MAX_POWER = 5;
    MIN_POWER = 1;
    decreasePowerMax();
    decreasePowerMax();
    decreasePowerMax();
    try { 
        if(MAX_POWER != 2) throw "MIN_POWER wasn't decremented 3 times";
        
    }
    catch(e){
        console.log("ERROR in test_DPMax_3: " + e );
    }
}

function test_suite_increasePowerMax(){
    // this suite will test the value of MAX_POWER Variable which changed when the "+MAX button" is clicked
   test_IPMax_1();
   test_IPMax_2();
   test_IPMax_3();
}

function test_IPMax_1(){
    MAX_POWER = 4;
    MIN_POWER = 1;
    increasePowerMax();
    try { 
        if(MAX_POWER != 5) throw "MAX_POWER not incremented";
        
    }
    catch(e){
        console.log("ERROR in test_IPMax_1: " + e );
    }
} 
function test_IPMax_2(){
    MAX_POWER = 5;
    MIN_POWER = 1;
    increasePowerMax();
    try { 
        if(MAX_POWER > 5) throw "Max_POWER is greater then 5";
        
    }
    catch(e){
        console.log("ERROR in test_IPMax_2: " + e );
    }
} 
function test_IPMax_3(){
    MAX_POWER = 2;
    MIN_POWER = 1;
    increasePowerMax();
    increasePowerMax();
    increasePowerMax();
    try { 
        if(MAX_POWER != 5) throw "MAX_POWER wasn't incremented 3 times";
        
    }
    catch(e){
        console.log("ERROR in test_IPMax_3: " + e );
    }
}

function test_suite_selectBall(){
    // this suite will test the value of BALL_CHOICE Variable matchs the index of BALL_TYPE
   test_selectBall_1();
   test_selectBall_2();
   test_selectBall_3();
   test_selectBall_4();
   test_selectBall_5();
}

function test_selectBall_1(){
    selectBall(0);
    try { 
        if(BALL_TYPE[BALL_CHOICE] != "Yellow") throw "index 0 in BALL_TYPE isn't equal to Yellow";
        
    }
    catch(e){
        console.log("ERROR in test_selectBall_1: " + e );
    }
} 
function test_selectBall_2(){
    selectBall(1);
    try { 
        if(BALL_TYPE[BALL_CHOICE] != "Red") throw "index 1 in BALL_TYPE isn't equal to Red";        
    }
    catch(e){
        console.log("ERROR in test_selectBall_2: " + e );
    }
} 
function test_selectBall_3(){
    selectBall(2);
    try { 
        if(BALL_TYPE[BALL_CHOICE] != "Black") throw "index 2 in BALL_TYPE isn't equal to Black";        
    }
    catch(e){
        console.log("ERROR in test_selectBall_3: " + e );
    }
}

function test_selectBall_4(){
    selectBall(3);
    try { 
        if(BALL_TYPE[BALL_CHOICE] != "White") throw "index 3 in BALL_TYPE isn't equal to White";        
    }
    catch(e){
        console.log("ERROR in test_selectBall_4: " + e );
    }
}
function test_selectBall_5(){
    for (i = 0 ; i<BALL_TYPE.length ;i++ ){
        try {
            if(BALL_TYPE[i] != "Red" && BALL_TYPE[i] != "Yellow" && BALL_TYPE[i] != "White" && BALL_TYPE[i] != "Black") throw "BALL_TYPE contains " + BALL_TYPE[i] +" which is an invalid colour at index" + i;
        
        }
        catch(e){
            console.log("ERROR in test_selectBall_5: " + e );
        }
    }
    
}

function test_suite_setColor(){
    // this suite will test the value of BALL_CHOICE Variable matchs the index of BALL_TYPE
   test_setColor_1();
   test_setColor_2();
   test_setColor_3();
   test_setColor_4();
}

function test_setColor_1(){
    setColor("None");
    try { 
        if(YOURCOLOR != "None") throw "YOURCOLOR isn't equal set to None";
        
    }
    catch(e){
        console.log("ERROR in test_setColor_1: " + e );
    }
} 
function test_setColor_2(){
    setColor("Yellow");
    try { 
        if(YOURCOLOR != "Yellow") throw "YOURCOLOR isn't equal set to Yellow";
        
    }
    catch(e){
        console.log("ERROR in test_setColor_2: " + e );
    }
} 
function test_setColor_3(){
    setColor("Red");
    try { 
        if(YOURCOLOR != "Red") throw "YOURCOLOR isn't equal set to Red";
        
    }
    catch(e){
        console.log("ERROR in test_setColor_3: " + e );
    }
}

function test_setColor_4(){
    try { 
        if(YOURCOLOR != "Red" && YOURCOLOR != "Yellow" && YOURCOLOR != "None" ) throw "YOURCOLOR is set to an invalid string";
        
    }
    catch(e){
        console.log("ERROR in test_setColor_4: " + e );
    }
}

function test_suite_pageChange(){
    // this suite will test the value of PAGE_SELECT Variable 
   test_pageChange_1();
   test_pageChange_2();
   test_pageChange_3();

}

function test_pageChange_1(){
    pageChange(1);
    try { 
        if(PAGE_SELECT != 1) throw "Page wasnt Changed to page 1";
        
    }
    catch(e){
        console.log("ERROR in test_pageChange_1: " + e );
    }
} 
function test_pageChange_2(){
    pageChange(2);
    try { 
        if(PAGE_SELECT != 2) throw "Page wasnt Changed to page 2";
        
    }
    catch(e){
        console.log("ERROR in test_pageChange_2: " + e );
    }
} 
function test_pageChange_3(){
    try { 
        if(PAGE_SELECT != 1 && PAGE_SELECT != 2) throw "Page" +PAGE_SELECT + "doesnt exist" ;        
    }
    catch(e){
        console.log("ERROR in test_pageChange_3: " + e );
    }
}



// ===================================================================================================================
// === User Interface ================================================================================================
// ===================================================================================================================


function initPreOptions()
{
    $("#user_span1").html("<p align=\"center\">Page: <b>"+ PAGE_SELECT + "</b></p>");
    $("#user_span2").html("<p align=\"center\"><button id='page1'>Page 1</button> <button id='page2'>Page 2</button></p>");
    $("#user_span4").html(line4_1);
    $("#user_span6").html(line6_1);
    $("#user_span10").html("<p align=\"center\"><button id='calculateShot'>GO!</button></p>");
    $("#user_span7").html("<p align=\"center\">The Camera Settings :</p>");
    $("#user_span8").html(line6_3);

    $(document).ready(function(){
        //Button Functionality
        $("#page1").click(function(){pageChange(1), soundClick();});
        $("#page2").click(function(){pageChange(2), soundClick();});
        $("#calculateShot").click(function(){calculateShot(), soundClick();});
        $("#cameraFlatButton").click(function(){cameraFlatDown(), soundClick();});
        $("#cameraCornerButton").click(function(){cameraSomeCornerPocket(), soundClick();});
        //Button CSS
        $("#page1").css({"background-color" : "#99ff33" , "border-radius" : "10px" , "padding" : "5px"});
        $("#page2").css({"background-color" : "#99ff33" , "border-radius" : "10px" , "padding" : "5px"});
        $("#calculateShot").css({"background-color" : "red" , "color" : "white", "padding" : "15px", "font-size" : "25px", "border-radius" : "10px", "font-weight" : "bold"});
        $("#cameraFlatButton").css({"background-color" : "white" , "color" : "black" , "border-radius" : "10px" , "padding" : "10px 10px"});
        $("#cameraCornerButton").css({"background-color" : "white" , "color" : "black" , "border-radius" : "10px" , "padding" : "10px 10px"});
    });

}

function initOptions()
{
    $("#user_span1").html("<p align=\"center\"><b>Calculating Table ...</b></p>");
    $("#user_span4").html("<p align=\"center\">Camera</p>");
    $("#user_span5").html(line6_3);
    $("#user_span7").html("<p></p>");
    $("#user_span8").html("<p></p>");
    $("#user_span10").html("<p></p>");
    
    $(document).ready(function(){
        //Button Functionality
        $("#cameraFlatButton").click(function(){cameraFlatDown(), soundClick();});
        $("#cameraCornerButton").click(function(){cameraSomeCornerPocket(), soundClick();});
        //Button CSS
        $("#cameraFlatButton").css({"background-color" : "white" , "color" : "black" , "border-radius" : "10px" , "padding" : "10px 10px"});
        $("#cameraCornerButton").css({"background-color" : "white" , "color" : "black" , "border-radius" : "10px" , "padding" : "10px 10px"});
    }); 
    
 
}

function initPostOptions()
{
    $("#user_span1").html("<p align=\"center\"><b>Finished Calculation</b></p>");
    $("#user_span5").html("<p align=\"center\"><button id='show1'>1</button> <button id='show2'>2</button> <button id='show3'>3</button></p>");
    $("#user_span6").html("<p align=\"center\"><button id='simulate'>Simulate Shot</button></p>");
    $("#user_span7").html("<p></p>");
    $("#user_span8").html("<p></p>");
    $("#user_span9").html("<p></p>");
    $("#user_span10").html("<p></p>");
    
    $(document).ready(function(){
        //Button Functionality
        $("#show1").click(function(){SHOT_PICKED = 0, soundClick();});
        $("#show2").click(function(){SHOT_PICKED = 1, soundClick();});
        $("#show3").click(function(){SHOT_PICKED = 2, soundClick();});
        $("#simulate").click(function(){simulateShot(), soundClick(), $("#simulate").css({"background-color" : "#333333" , "color" : "#aaaaaa", "padding" : "15px", "font-size" : "25px", "border-radius" : "10px", "font-weight" : "bold"});});
        //Button CSS
        $("#show1").css({"background-color" : "black" , "color" : "white" , "padding" : "10px 20px", "font-size" : "16px", "border-radius" : "5px"});
        $("#show2").css({"background-color" : "black" , "color" : "white" , "padding" : "10px 20px", "font-size" : "16px", "border-radius" : "5px"});
        $("#show3").css({"background-color" : "black" , "color" : "white" , "padding" : "10px 20px", "font-size" : "16px", "border-radius" : "5px"});
        $("#simulate").css({"background-color" : "red" , "color" : "white", "padding" : "15px", "font-size" : "25px", "border-radius" : "10px", "font-weight" : "bold"});
    }); 
}

function updatePreHUD()
{
    $("#user_span1").html("<p align=\"center\">Page: <b>"+ PAGE_SELECT + "</b></p>");
    if (PAGE_SELECT == 1){
        $("#user_span3").html("<p align=\"center\">Ball Selected: <b>"+ BALL_TYPE[BALL_CHOICE] + "</b></p>");
        $("#user_span5").html("<p align=\"center\">The Colour you are playing as is set to: <b>" + YOURCOLOR + "</b></p>");
        if (CURRENT_PAGE !== PAGE_SELECT){
            $("#user_span4").html(line4_1);
            $("#user_span6").html(line6_1);
            
            $(document).ready(function(){
                //Button Functionality
                $("#yellowButton").click(function(){selectBall(0), soundClick();});
                $("#redButton").click(function(){selectBall(1), soundClick();});
                $("#blackButton").click(function(){selectBall(2), soundClick();});
                $("#whiteButton").click(function(){selectBall(3), soundClick();});
                $("#yourNButton").click(function(){setDifficulty(setColor("None")), soundClick();});
                $("#yourYButton").click(function(){setDifficulty(setColor("Yellow")), soundClick();});
                $("#yourRButton").click(function(){setDifficulty(setColor("Red")), soundClick();});
                //Button CSS
                $("#yellowButton").css({"background-color" : "yellow" , "color" : "yellow" , "border-radius" : "30px" , "padding" : "10px 10px"});
                $("#redButton").css({"background-color" : "red" , "color" : "red" , "border-radius" : "30px" , "padding" : "10px 10px"});
                $("#blackButton").css({"background-color" : "black" , "color" : "black" , "border-radius" : "30px" , "padding" : "10px 10px"});
                $("#whiteButton").css({"background-color" : "white" , "color" : "white" , "border-radius" : "30px" , "padding" : "10px 10px"});
                $("#yourNButton").css({"background-color" : "#dddddd" , "color" : "#dddddd" , "border-radius" : "30px" , "padding" : "10px 10px"});
                $("#yourYButton").css({"background-color" : "yellow" , "color" : "yellow" , "border-radius" : "30px" , "padding" : "10px 10px"});
                $("#yourRButton").css({"background-color" : "red" , "color" : "red" , "border-radius" : "30px" , "padding" : "10px 10px"});
            });   
        }
    }
    
    else if (PAGE_SELECT == 2){
        $("#user_span3").html("<p align=\"center\">The Difficultly is set to: <b>" + difficulty + "</b></p>");
        $("#user_span5").html("<p align=\"center\">Power: <b>" + MIN_POWER + " to " + MAX_POWER + "</b></p>");
        if (CURRENT_PAGE !== PAGE_SELECT){
            $("#user_span4").html(line4_3);
            $("#user_span6").html(line6_2);
            $(document).ready(function(){
                //Button Functionality
                $("#easyButton").click(function(){setDifficulty("Easy"), soundClick();});
                $("#mediumButton").click(function(){setDifficulty("Medium"), soundClick();});
                $("#hardButton").click(function(){setDifficulty("Hard"), soundClick();});
                $("#decreasePowerMinButton").click(function(){decreasePowerMin(), soundClick();});
                $("#increasePowerMinButton").click(function(){increasePowerMin(), soundClick();});
                $("#decreasePowerMaxButton").click(function(){decreasePowerMax(), soundClick();});
                $("#increasePowerMaxButton").click(function(){increasePowerMax(), soundClick();});
                //Button CSS
                $("#easyButton").css("background-color", "#00ff00");
                $("#mediumButton").css("background-color", "yellow");
                $("#hardButton").css({"background-color" : "red" , "color" : "white"});
                $("#decreasePowerMinButton").css({"background-color" : "black" , "color" : "white" , "border-radius" : "10px" , "padding" : "10px 10px"});
                $("#increasePowerMinButton").css({"background-color" : "black" , "color" : "white" , "border-radius" : "10px" , "padding" : "10px 10px"});
                $("#decreasePowerMaxButton").css({"background-color" : "black" , "color" : "white" , "border-radius" : "10px" , "padding" : "10px 10px"});
                $("#increasePowerMaxButton").css({"background-color" : "black" , "color" : "white" , "border-radius" : "10px" , "padding" : "10px 10px"});
            });   
        }
    }
    CURRENT_PAGE = PAGE_SELECT;
}

function updateHUD(){
    $("#user_span2").html("<p align=\"center\">The Colour you are playing as is set to: <b>" + YOURCOLOR + "</b>.</p>");
    $("#user_span3").html("<p align=\"center\"></p>");
    $("#user_span6").html("<p align=\"center\">Balls on Table: <b>" + BALLS.length + "</b>.</p>");
    $("#user_span7").html("<p align=\"center\">Balls Calculated: <b>" + BALLS_ATTEMPTED + "</b> of <b>" + BALLS_TO_POT + "</b>.</p>");
}

function updatePostHUD(){
    $("#user_span2").html("<p align=\"center\">Current Shot: <b>" + ((SHOT_PICKED) + 1) + "</b>.</p>");
    $("#user_span3").html("<p align=\"center\">Shot Score: <b>" + BEST_SCORES[SHOT_PICKED][0] + "</b>.</p>");
    $("#user_span4").html("<p align=\"center\">Shot Power: <b>" + BEST_SCORES[SHOT_PICKED][2] + "</b>.</p>");

    
}

// ===================================================================================================================
// === Ancient Brain Functions =======================================================================================
// ===================================================================================================================

	
this.newRun = function() 
{
	ABWorld.init3d(startRadius, maxRadius, SKYCOLOR); 			// sets up renderer, scene, camera
	initPreOptions();
 // can adjust renderer:
	ABWorld.renderer.shadowMap.enabled = true;
 // scene is Physijs.Scene, not THREE.Scene
 // can adjust scene - change gravity from default:
	ABWorld.scene.setGravity ( gravity );
	loadResources();	// asynchronous file loads 
						// calls initScene() when all done 
	ABWorld.lookat.copy ( ABWorld.scene.position );  
	ABWorld.scene.simulate();		// Physics simulate - runs on independent timer to AB nextStep
	// user actions
    document.onkeydown          = keyHandler;
    document.ondblclick         = click;
    ABHandler.ontouchstart      = initTouch;
    //ABHandler.initMouseDrag   = initDrag;
    //ABHandler.mouseDrag       = drag
    
    All_tests();
};

this.nextStep = function()		// not used 
{
    if (splashClicked === true && toggleBasic === true){
        basicSetUp();
        toggleBasic = false;
        toggledBasic = true;
    }

    if (CALCULATED === false && CALCULATING === false){
        updatePreHUD();}
    else if (CALCULATING === true){
        updateHUD();
    }
    else{
        updatePostHUD();
    }
};

}
	
// ===================================================================================================================
// === Audio =========================================================================================================
// ===================================================================================================================
	
var colaudio1 = new Audio( SOUND_COLLISION );
var colaudio2 = new Audio( SOUND_COLLISION );
var colaudio3 = new Audio( SOUND_COLLISION );
var colaudio4 = new Audio( SOUND_COLLISION );
var colaudio5 = new Audio( SOUND_COLLISION );

var goodaudio1 = new Audio( SOUND_GOOD );
var goodaudio2 = new Audio( SOUND_GOOD );
var goodaudio3 = new Audio( SOUND_GOOD );
var goodaudio4 = new Audio( SOUND_GOOD );
var goodaudio5 = new Audio( SOUND_GOOD );

var badaudio1 = new Audio( SOUND_BAD );
var badaudio2 = new Audio( SOUND_BAD );
var badaudio3 = new Audio( SOUND_BAD );
var badaudio4 = new Audio( SOUND_BAD );
var badaudio5 = new Audio( SOUND_BAD );

var clkaudio1 = new Audio( SOUND_CLICK );
var clkaudio2 = new Audio( SOUND_CLICK );
var clkaudio3 = new Audio( SOUND_CLICK );
var clkaudio4 = new Audio( SOUND_CLICK );
var clkaudio5 = new Audio( SOUND_CLICK );

function soundCollision()		// triggered by JS, not by user interaction 
{
	
	if ( ! AB.audioIsPlaying ( colaudio1 ) ) 	{ colaudio1.play(); return; }
	if ( ! AB.audioIsPlaying ( colaudio2 ) ) 	{ colaudio2.play(); return; }
	if ( ! AB.audioIsPlaying ( colaudio3 ) ) 	{ colaudio3.play(); return; }
	if ( ! AB.audioIsPlaying ( colaudio4 ) ) 	{ colaudio4.play(); return; }
	if ( ! AB.audioIsPlaying ( colaudio5 ) ) 	{ colaudio5.play(); return; }
	
	// else try to make a new one, local variable, without user interaction
	// but this will not play on some browsers:

	var a = new Audio( SOUND_COLLISION );		
	a.play();
}

function soundGood()		// triggered by JS, not by user interaction 
{
	
	if ( ! AB.audioIsPlaying ( goodaudio1 ) ) 	{ goodaudio1.play(); return; }
	if ( ! AB.audioIsPlaying ( goodaudio2 ) ) 	{ goodaudio2.play(); return; }
	if ( ! AB.audioIsPlaying ( goodaudio3 ) ) 	{ goodaudio3.play(); return; }
	if ( ! AB.audioIsPlaying ( goodaudio4 ) ) 	{ goodaudio4.play(); return; }
	if ( ! AB.audioIsPlaying ( goodaudio5 ) ) 	{ goodaudio5.play(); return; }
	
	// else try to make a new one, local variable, without user interaction
	// but this will not play on some browsers:

	var a = new Audio( SOUND_GOOD );		
	a.play();
}

function soundBad()		// triggered by JS, not by user interaction 
{
	
	if ( ! AB.audioIsPlaying ( badaudio1 ) ) 	{ badaudio1.play(); return; }
	if ( ! AB.audioIsPlaying ( badaudio2 ) ) 	{ badaudio2.play(); return; }
	if ( ! AB.audioIsPlaying ( badaudio3 ) ) 	{ badaudio3.play(); return; }
	if ( ! AB.audioIsPlaying ( badaudio4 ) ) 	{ badaudio4.play(); return; }
	if ( ! AB.audioIsPlaying ( badaudio5 ) ) 	{ badaudio5.play(); return; }
	
	// else try to make a new one, local variable, without user interaction
	// but this will not play on some browsers:

	var a = new Audio( SOUND_BAD );		
	a.play();
}

function soundClick()		// triggered by JS, not by user interaction 
{
	
	if ( ! AB.audioIsPlaying ( clkaudio1 ) ) 	{ clkaudio1.play(); return; }
	if ( ! AB.audioIsPlaying ( clkaudio2 ) ) 	{ clkaudio2.play(); return; }
	if ( ! AB.audioIsPlaying ( clkaudio3 ) ) 	{ clkaudio3.play(); return; }
	if ( ! AB.audioIsPlaying ( clkaudio4 ) ) 	{ clkaudio4.play(); return; }
	if ( ! AB.audioIsPlaying ( clkaudio5 ) ) 	{ clkaudio5.play(); return; }
	
	// else try to make a new one, local variable, without user interaction
	// but this will not play on some browsers:

	var a = new Audio( SOUND_CLICK );		
	a.play();
}

// ===================================================================================================================
// === Splash Screens ================================================================================================
// ===================================================================================================================

	AB.newSplash ("<p align=\"center\">Optional Start Settings</p><p align=\"center\">Set up balls in initial break arrangement</p><p align=\"center\"><button id='basicSetUp1'>Initial Break Set Up</button></p>");
        $(document).ready(function(){
        //Button Functionality
        $("#basicSetUp1").click(function(){toggleBasic = true, soundClick(), $("#basicSetUp1").css({"background-color" : "green" , "color" : "white" , "padding" : "15px 30px", "font-size" : "16px", "border-radius" : "5px"});});
        //Button CSS
        $("#basicSetUp1").css({"background-color" : "black" , "color" : "white" , "padding" : "15px 30px", "font-size" : "16px", "border-radius" : "5px"});});  
        // when user clicks/touches button on splash screen, audio starts and run starts:
        $("#splashbutton").click (function()        
        {
                AB.removeSplash();              // remove splash screen
                soundClick();
                splashClicked = true;
        });