Code viewer for World: A star playground (clone b...

// Cloned by K.Ellis on 26 Oct 2019 from World "A star" by "Coding Train" project 
// Please leave this clone trail here.
 

// Port of 
// https://github.com/nature-of-code/NOC-S17-2-Intelligence-Learning/tree/master/week1-graphs/05_astar


// Daniel Shiffman
// Nature of Code: Intelligence and Learning
// https://github.com/shiffman/NOC-S17-2-Intelligence-Learning

// Part 1: https://youtu.be/aKYlikFAV4k
// Part 2: https://youtu.be/EaZxUCWAjb0
// Part 3: https://youtu.be/jwRT4PCT6RU

// is diagonal move allowed 
    const diagonal = true ;
    //const diagonal = false ;

// canvas size 
    const cw = 900;
    const ch = 600; 

// How many columns and rows 
// different each time
var rando = AB.randomIntAtoB ( 3, 6 );
var cols = 3 * rando;
var rows = 3 * rando;

 
// how many walls to make, from 0 to 1 
// different each time
const wallAmount = AB.randomFloatAtoB ( 0, 0.6 );  //KE set walls to zero was 0, 0.6

const backcolor = 'white';
const wallcolor = 'grey';
const pathcolor = 'blue';

const opencolor     = 'lightgreen';
const closedcolor   = 'lightpink';
//const opencolor     = 'white';
//const closedcolor   = 'pink';


// 2D array
var grid = new Array(cols);

// Open and closed set
var openSet = [];
var closedSet = [];

// Start and end
var start;
var end;

// Width and height of each cell of grid
var w, h;

// The road taken
var path = [];

var wallCounter = 0; //KE added to print out number of walls, when goal found



//=== heuristic ===========================
// this must be always optimistic - real time will be this or longer 

function heuristic(a, b) 
{
    if ( diagonal ) return ( dist(a.i, a.j, b.i, b.j) ); // KE distance between 2 points

        // 2D distance
        // dist is a P5 function 

    //else  return ( abs(a.i - b.i) + abs(a.j - b.j) );
    //else return ( dist(a.i, a.j, b.i, b.j) );

        // else not diagonal, can only go across and down 
        // so this is optimistic
        // not this is not optimistic if we can do diagonal move 
}


// Function to delete element from the array
function removeFromArray(arr, elt) 
{
  // Could use indexOf here instead to be more efficient
  for (var i = arr.length - 1; i >= 0; i--) 
    if (arr[i] == elt) 
      arr.splice(i, 1);
}


// Daniel Shiffman
// Nature of Code: Intelligence and Learning
// https://github.com/shiffman/NOC-S17-2-Intelligence-Learning

// Part 1: https://youtu.be/aKYlikFAV4k
// Part 2: https://youtu.be/EaZxUCWAjb0
// Part 3: https://youtu.be/jwRT4PCT6RU

// An object to describe a spot in the grid
function Spot(i, j) 
{

  // Location
  this.i = i;
  this.j = j;

  // f, g, and h values for A*
  this.f = 0;
  this.g = 0;
  this.h = 0;

  // Neighbors
  this.neighbors = [];

  // Where did I come from?
  this.previous = undefined;

  // Am I an wall?
  if (random(1) < wallAmount) 
  {
      this.wall = true; 
      wallCounter++; //KE added for final print out
  }
  else this.wall = false;

  // Display me
  this.show = function(col) 
  {
    if (this.wall) 
    {
        
      fill(wallcolor);
      noStroke();
      
      // wall fills square 
            rect (      this.i * w,             this.j * h,             w,          h           );
        
      // wall only partially fills square 
      //    ellipse (   this.i * w + w / 2,     this.j * h + h / 2,     w * 0.7,    h * 0.7     );

    }
    else if (col)
    {
      fill(col);
      rect(this.i * w, this.j * h, w, h);
    }
  };
  

  // Figure out who my neighbors are
  this.addNeighbors = function(grid) 
  {
    var i = this.i;
    var j = this.j;
    
    if (i < cols - 1)   this.neighbors.push(grid[i + 1][j]);
    if (i > 0)          this.neighbors.push(grid[i - 1][j]);
    if (j < rows - 1)   this.neighbors.push(grid[i][j + 1]);
    if (j > 0)          this.neighbors.push(grid[i][j - 1]);
    
    if (diagonal)
    // diagonals are also neighbours:
    {
        if (i > 0 && j > 0)                 this.neighbors.push(grid[i - 1][j - 1]);
        if (i < cols - 1 && j > 0)          this.neighbors.push(grid[i + 1][j - 1]);
        if (i > 0 && j < rows - 1)          this.neighbors.push(grid[i - 1][j + 1]);
        if (i < cols - 1 && j < rows - 1)   this.neighbors.push(grid[i + 1][j + 1]);
    }
    //console.log('grid ' + grid[i][j]);
    
    
  }
  console.log(grid);
}




function setup() 
{
    // slower frame rate to see how it is working 
   frameRate (1);
    
    
  createCanvas(cw, ch);

  // Grid cell size
  w = width / cols;
  h = height / rows;

  // Making a 2D array
  for (var i = 0; i < cols; i++) 
    grid[i] = new Array(rows);

  for (var i = 0; i < cols; i++) 
    for (var j = 0; j < rows; j++) 
      grid[i][j] = new Spot(i, j);

  // All the neighbors
  for (var i = 0; i < cols; i++) 
    for (var j = 0; j < rows; j++) 
      grid[i][j].addNeighbors(grid);


  // Start and end
  start     = grid[0][0];
  end       = grid[cols - 1][rows - 1];
  console.log('start = ' + start.i + ', ' + start.j); //KE 
  console.log('end = ' + end.i + ', ' + end.j); // KE
  start.wall    = false;
  end.wall      = false;

  // openSet starts with beginning only
  openSet.push(start);
  
    //console.log('start search' + start); // prints object, object
    //console.log('openSet ' + openSet); // prints object, object
    
    console.log(start); // prints spot object for debbug
    console.log(openSet); // prints openSet array for debug

}


function draw() 
// the search goes on over many timesteps 
// each timestep, check one more square and draw current partial solution 
{


  // --- begin still searching -----------------------------
  if (openSet.length > 0) 
  {
     
    // Best next option
    var winner = 0;
    
    for (var i = 0; i < openSet.length; i++) 
        if (openSet[i].f < openSet[winner].f) 
           winner = i;
           var current = openSet[winner];
            
           console.log('winner = g: ' + current.g + ' h: ' + current.h);
    

    // Did I finish?
    if (current === end) 
    {
        noLoop();
        //KE added a lot of print outs to sanity check & to what is happening with demo A*
        console.log("success - found path");
        console.log('start = ' + start.i + ', ' + start.j); //KE always 0,0 in our case
        console.log('end = ' + end.i + ', ' + end.j); // KR currently set to equal cols & rows so this will be something like 14,14 etc
     
        console.log('Total number of spots on grid: ' + (cols*rows));
        console.log('Total number of walls on grid: ' + wallCounter);
        
        console.log('Processed spots left open: ' + (openSet.length - 1)); // KE need to account for start & last spot
        console.log('Processed spots closed: ' + (closedSet.length + 1)); // goal spot not counted, adjust for same
        
        console.log('Optimal Path: ' + path.length); // KE from all the closed set, a subset will be teh chosen joint optimal path
        
        console.log('Unprocessed: ' + ((cols*rows) - ((openSet.length - 1) + (closedSet.length + 1) + (wallCounter))));
        console.log(path);
    }

    // Best option moves from openSet to closedSet
    //KE this can be thought of like a stack or queue mechansim that we are updating
    removeFromArray(openSet, current); 
    closedSet.push(current);

    // Check all the neighbors
    var neighbors = current.neighbors;
    
    
    
    //--- start of for loop -----------
    for (var i = 0; i < neighbors.length; i++) 
    {
      var neighbor = neighbors[i];
    
      // Valid next spot?
      if (!closedSet.includes(neighbor) && !neighbor.wall) 
      {
        var tempG = current.g + heuristic(neighbor, current); 
        
    

        // Is this a better path than before?
        var newPath = false;
        //console.log('path = ' + newPath);
        if (openSet.includes(neighbor)) 
        {
            if (tempG < neighbor.g) 
            {
                neighbor.g = tempG;
                newPath = true;
                console.log('tempG < neighbor.g')
                }
        }
        else 
        {
          neighbor.g = tempG;
          newPath = true;
          openSet.push(neighbor);
        }
        
   
        

        // Yes, it's a better path
        if (newPath) 
        {
          neighbor.h = heuristic(neighbor, end); // KE steps too go to 'end' var i.e. steps to goal from 'n' this spot
          neighbor.f = neighbor.g + neighbor.h; //KE our A* algo
          console.log('f ' + neighbor.f + ' = g ' + neighbor.g + ' + h '+ neighbor.h + '\n'); //KE f(n) = g(n) + h(n) print out
          neighbor.previous = current;
        }
        
      }
    }
    //--- end of for loop -----------
    
  }
  // --- end still searching -----------------------------
  
  
  else 
  {
    console.log('fail - no path exists');
    noLoop();
        console.log('start = ' + start.i + ', ' + start.j); 
        console.log('end = ' + end.i + ', ' + end.j); 
     
        console.log('Total number of spots on grid: ' + (cols*rows));
        console.log('Total number of walls on grid: ' + wallCounter);
        
        console.log('Processed spots left open: ' + (openSet.length)); 
        console.log('Processed spots closed:' + (closedSet.length));
        console.log('Unprocessed: ' + ((cols*rows) - ((openSet.length) + (closedSet.length) + (wallCounter))));
    return;
  }


  // Draw current state of everything
  background(backcolor);

  for (var i = 0; i < cols; i++) 
    for (var j = 0; j < rows; j++) 
      grid[i][j].show();

  for (var i = 0; i < closedSet.length; i++) 
    closedSet[i].show( closedcolor );
   
  for (var i = 0; i < openSet.length; i++) 
    openSet[i].show( opencolor );


  // Find the path by working backwards
  path = [];
  var temp = current;
  path.push(temp);
  while (temp.previous) 
  {
    path.push(temp.previous);
    temp = temp.previous;
  }

  
if (diagonal)
{
  // path as continuous line
  noFill();
  stroke(pathcolor);
  strokeWeight(w / 8);
  beginShape();
  for (var i = 0; i < path.length; i++) 
        vertex (    (path[i].i * w) + w / 2,     (path[i].j * h) + h / 2 );
        endShape();
}
else 
{
    // path as solid blocks 
   for (var i = 0; i < path.length; i++) 
     path[i].show(pathcolor);
}


}