Loading...
Logo
Processing Forum
Hello everyone, I wonder if anyone would help me with an issues I have regarding pathfinding.

I am making a 2D zombie shooter and I am working on making my zombies separate rather than group together. With a lot of help from a friend (who wrote this section of code) I have now got an algorithm that:

Checks for the nearest zombie
Works out the possible move it can make
Checks to see if there is another zombie in the way
If not then it can move 
If there is, see if you can move on the x axis (and vice versa)

BUT! When the program is run, some of the zombies  will not make the required move, so I have zombies dotted around the screen just not moving - unless I make the player move somewhere else to make the zombie move again.

I don't want to paste the whole program into this post as there is too much of it, but have a look at the above function and you may be able to help!

Copy code
  1. void seekTarget() {
  2.     Mover closestZombie = null;
  3.     float distanceToClosestZombie = 10000;  // arbitrary large number
  4.     float distance;
  5.     int tryX = x;
  6.     int tryY = y;
  7.     // find which of the other zombies is nearest to this one,
  8.     // and what distance separates them
  9.     for (Mover m: movers) {
  10.       if (m == this) {
  11.         // don't compare this zombie to itself!
  12.         continue;
  13.       }
  14.       distance = dist(x, y, m.x, m.y);
  15.       if (distance < distanceToClosestZombie) {
  16.         // the zombie being tested is the closest found yet
  17.         closestZombie = m;
  18.         distanceToClosestZombie = distance;
  19.       }
  20.     }
  21.     
  22.     // work out the "ideal" new x and y coordinates, assuming
  23.     // for now that the closest zombie is not in the way
  24.     // (we'll find that out in a moment)
  25.     
  26.       /*if(targetX > x) {
  27.           tryX++;
  28.           //println("working");
  29.       } else if(targetX < x) {
  30.           tryX--;
  31.           //println("working");
  32.       }
  33.     
  34.       if (targetY > y) {
  35.           tryY++;
  36.           //println("working");
  37.       } else if(targetY < y) {
  38.           tryY--;
  39.           //println("working");
  40.       }*/
  41.       
  42.       if(targetX > player.x) {
  43.         //tryX = tryX + int(zombieSpeed);
  44.         tryX--;
  45.         image(zombieLEFT, tryX, tryY);
  46.         println("working");
  47.       } else if(targetX < player.x) {
  48.         tryX++;
  49.         //tryX = tryX - int(zombieSpeed);
  50.         image(zombieRIGHT, tryX, tryY);
  51.       }
  52.       
  53.       if(targetY > player.y) {
  54.         tryY--;
  55.         //tryY = tryY - int(zombieSpeed);
  56.         image(zombieUP, tryX, tryY);
  57.       } else if(targetY < player.y) {
  58.         tryY++;
  59.         //tryY = tryY + int(zombieSpeed);
  60.         image(zombieDOWN, tryX, tryY);
  61.       }
  62.     
  63.     // if the closest zombie is not already too close, make the
  64.     // ideal move computed above
  65.     if (distanceToClosestZombie > MIN_ZOMBIE_SEPARATION) {
  66.       x = tryX;
  67.       y = tryY;
  68.     }
  69.     
  70.     // otherwise, only make a move that will cause the separation
  71.     // between this zombie and its closest neighbour to either
  72.     // increase or remain the same
  73.     else {
  74.       if (dist(tryX, tryY, closestZombie.x, closestZombie.y) >=
  75.           distanceToClosestZombie) {
  76.         // the ideal move increases or doesn't change the separation
  77.         x = tryX;
  78.         y = tryY;
  79.       }
  80.       else if (dist(tryX, y, closestZombie.x, closestZombie.y) >=
  81.           distanceToClosestZombie) {
  82.         // only the horizontal component of the ideal move increases
  83.         // or doesn't change the separation
  84.         x = tryX;
  85.       }
  86.       else if (dist(x, tryY, closestZombie.x, closestZombie.y) >=
  87.           distanceToClosestZombie) {
  88.         // only the vertical component of the ideal move increases
  89.         // or doesn't change the separation
  90.         y = tryY;
  91.       }
  92.     }
  93.   }
The Mover is the zombie :)

Ah - My mate thinks it is to do with the fact that I also have a move() function separate to this, and that I need to integrate it into the above function, below is the other move() function:

Copy code
  1.  void move() {
  2.     int wouldBeX = x;
  3.     int wouldBeY = y;
  4.     
  5.     //Move along X axis to find player
  6.     if (x < player.x) {
  7.       x++;
  8.       image(zombieRIGHT, x, y);
  9.     } 
  10.     else if (x > player.x) {
  11.       x--;
  12.       image(zombieLEFT, x, y);
  13.     }
  14.     
  15.     //Move along Y axis to find player
  16.     if (y < player.y) {
  17.       y++;
  18.       image(zombieDOWN, x, y);
  19.     } 
  20.     else if (y > player.y) {
  21.       y--;
  22.       image(zombieUP, x, y);
  23.     }

  24.     for (Mover m: movers) {
  25.       //tryToMove(m, wouldBeX, wouldBeY);
  26.     }
  27.   }

Thanks!

Replies(4)

Hi,

Maybe you can find help with the A* algorythm, designed for pathfinding. Here's a smart introduction :
A* pathfinding for beginners (naturally, "beginner" isn't pejorative, we're all beginners somewhere... ;)

Hope it helps.


The Code is too long to read and understand completely

1st Remark

you have
Copy code
  1.       if(targetX > player.x) {
and

Copy code
  1.       } else if(targetX < player.x) {
what if they are the same?
same goes for y...

2nd Remark

Did you test this:
Copy code
  1.       if (m == this) {


3rd Remark

try to enter some random



Thanks Docbones - I have thought about the A* pathfinding algorithm but was told it was far too complicated for what I'm trying to do.

Thanks Chrisir - The likelihood of the zombies possible x & y co-ordinates being the same is really minuscule so I haven't coded for that event. As far as the (m == this) goes, that's just so the zombie doesn't test itself when looking for other zombies that are close to it - cos that would be silly :P And what do you mean "try to enter some random"?

But thank you both for replying - I understand the code is really long for one post but I wasn't really sure on how I could change that and still get my problem across :)



As far as the (m == this) goes, that's just so the zombie doesn't test itself

I know - but does it work? Check with println or so.




try to enter some random

that was the idea to have every 5th step or so slightly varied by a random number - so it's a little more random walk / less determinated

see random() in the reference