From bcc57d93fd6829ed848ec9a38787fde74f8ea6fa Mon Sep 17 00:00:00 2001 From: Sooraj Sannabhadti <2005sooraj@gmail.com> Date: Tue, 13 Aug 2024 20:35:34 +0800 Subject: [PATCH] Added boids flocking algorithm code --- EnemyBoid.js | 187 ++++++++++++++++++++++++++++++++++++ EnemyBoidOptimised.js | 154 ++++++++++++++++++++++++++++++ FriendlyBoid.js | 200 ++++++++++++++++++++++++++++++++++++++ FriendlyBoidOptimised.js | 161 +++++++++++++++++++++++++++++++ PointOfInterest.js | 25 +++++ index.html | 5 + sketch.js | 201 +++++++++++++++++++++++++++++++++++++++ style.css | 1 + 8 files changed, 934 insertions(+) create mode 100644 EnemyBoid.js create mode 100644 EnemyBoidOptimised.js create mode 100644 FriendlyBoid.js create mode 100644 FriendlyBoidOptimised.js create mode 100644 PointOfInterest.js diff --git a/EnemyBoid.js b/EnemyBoid.js new file mode 100644 index 0000000..8eedfb2 --- /dev/null +++ b/EnemyBoid.js @@ -0,0 +1,187 @@ +class EnemyBoid { + constructor(x, y) { + this.position = createVector(x, y); + this.velocity = createVector(0, 0); + + this.trail = []; + for (let i = 0; i < 100; i++) { + this.trail.push(this.position.copy()); + } + } + + update(trail = false, update = true) { + if (!update) { + return; + } + + // Separation + let close_dx = 0; + let close_dy = 0; + + for (let boid of enemy_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < avoidance_range) { + close_dx += this.position.x - boid.position.x; + close_dy += this.position.y - boid.position.y; + } + } + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(avoidance_constant)); + + + // Alignment + let avg_dx = 0; + let avg_dy = 0; + let total_neighbours = 0; + + for (let boid of enemy_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < visible_range) { + avg_dx += boid.velocity.x; + avg_dy += boid.velocity.y; + + total_neighbours++; + } + } + } + + if (total_neighbours > 0) { + avg_dx /= total_neighbours; + avg_dy /= total_neighbours; + } + + this.velocity.add(createVector(avg_dx, avg_dy).normalize().mult(alignment_constant)); + + + // Cohesion + total_neighbours = 0; + let avg_x = 0; + let avg_y = 0; + + for (let boid of enemy_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < visible_range) { + avg_x += boid.position.x; + avg_y += boid.position.y; + + total_neighbours++; + } + } + } + + if (total_neighbours > 0) { + avg_x /= total_neighbours; + avg_y /= total_neighbours; + } + + this.velocity.add(createVector(avg_x - this.position.x, avg_y - this.position.y).normalize().mult(cohesion_constant)); + + + // POI Repulsion + close_dx = 0; + close_dy = 0; + + for (let poi of pointsOfInterest) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < poi_protection_range) { + close_dx += this.position.x - poi.position.x; + close_dy += this.position.y - poi.position.y; + } + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(poi_avoidance_constant)); + + + // Enemy Boid Attack + let friendly_boid_dx = 0; + let friendly_boid_dy = 0; + for (let poi of friendly_boids) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < visible_range) { + friendly_boid_dx += poi.position.x - this.position.x; + friendly_boid_dy += poi.position.y - this.position.y; + } + } + + this.velocity.add(createVector(friendly_boid_dx, friendly_boid_dy).normalize().mult(enemy_boid_attack_constant)); + + + // Turn around if it's going out of bounds + if (this.position.x < margin) { + this.velocity.x += turn_constant; + } else if (this.position.x > width - margin) { + this.velocity.x -= turn_constant; + } + + if (this.position.y < margin) { + this.velocity.y += turn_constant; + } else if (this.position.y > height - margin) { + this.velocity.y -= turn_constant; + } + + + // Limit speed + if (this.velocity.mag() > max_speed) { + this.velocity.normalize().mult(max_speed); + } else if (this.velocity.mag() < min_speed) { + this.velocity.normalize().mult(min_speed); + } + + // Update the trail + if (trail) { + this.trail.shift(); + this.trail.push(this.position.copy()); + } + + // Move the boid + this.position.add(this.velocity); + } + + draw(trail = false, shape = "Triangle", direction = false, range = false, avoid = false) { + // Draw the boid as a sphere with a line indicating its direction + if (shape == "Circle") { + stroke(255, 0, 0); + strokeWeight(15); + point(this.position.x, this.position.y); + } else { + // Draw the boid as a triangle + strokeWeight(0); + fill(255, 0, 0); + let angle = this.velocity.heading(); + let a = createVector(0, -30).rotate(angle + PI / 2).add(this.position); + let b = createVector(10, 0).rotate(angle + PI / 2).add(this.position); + let c = createVector(-10, 0).rotate(angle + PI / 2).add(this.position); + triangle(a.x, a.y, b.x, b.y, c.x, c.y); + } + + if (direction) { + stroke(0, 0, 255); + strokeWeight(2); + line(this.position.x, this.position.y, this.position.x + (this.velocity.x * 5), this.position.y + (this.velocity.y * 5)); + } + + // Add a trail behind the boid + if (trail) { + for (let i = 0; i < this.trail.length; i++) { + let alpha = map(i, 0, this.trail.length, 0, 255); + fill(255, 255, 255, alpha); + stroke(255, 255, 255, alpha); + strokeWeight(5); + point(this.trail[i].x, this.trail[i].y); + } + } + + if (range) { + noFill(); + stroke(0, 255, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, visible_range * 2); + } + + if (avoid) { + noFill(); + stroke(255, 0, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, avoidance_range * 2); + } + } +} \ No newline at end of file diff --git a/EnemyBoidOptimised.js b/EnemyBoidOptimised.js new file mode 100644 index 0000000..0b83081 --- /dev/null +++ b/EnemyBoidOptimised.js @@ -0,0 +1,154 @@ +class EnemyBoidOptimised { + constructor(x, y) { + this.position = createVector(x, y); + this.velocity = createVector(0, 0); + + this.trail = []; + for (let i = 0; i < 100; i++) { + this.trail.push(this.position.copy()); + } + } + + update(trail = false, update = true) { + if (!update) { + return; + } + + let close_dx = 0; + let close_dy = 0; + let avg_dx = 0; + let avg_dy = 0; + let avg_x = 0; + let avg_y = 0; + let total_neighbours = 0; + let poi_avoid_dx = 0; + let poi_avoid_dy = 0; + let friendly_boid_dx = 0; + let friendly_boid_dy = 0; + + for (let boid of enemy_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < avoidance_range) { + close_dx += this.position.x - boid.position.x; + close_dy += this.position.y - boid.position.y; + } + + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < visible_range) { + avg_dx += boid.velocity.x; + avg_dy += boid.velocity.y; + + avg_x += boid.position.x; + avg_y += boid.position.y; + + total_neighbours++; + } + } + } + + for (let poi of pointsOfInterest) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < poi_protection_range) { + poi_avoid_dx += this.position.x - poi.position.x; + poi_avoid_dy += this.position.y - poi.position.y; + } + } + + for (let poi of friendly_boids) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < visible_range) { + friendly_boid_dx += poi.position.x - this.position.x; + friendly_boid_dy += poi.position.y - this.position.y; + } + } + + if (total_neighbours > 0) { + avg_dx /= total_neighbours; + avg_dy /= total_neighbours; + + avg_x /= total_neighbours; + avg_y /= total_neighbours; + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(avoidance_constant)); + this.velocity.add(createVector(avg_dx, avg_dy).normalize().mult(alignment_constant)); + this.velocity.add(createVector(avg_x - this.position.x, avg_y - this.position.y).normalize().mult(cohesion_constant)); + this.velocity.add(createVector(poi_avoid_dx, poi_avoid_dy).normalize().mult(poi_avoidance_constant)); + this.velocity.add(createVector(friendly_boid_dx, friendly_boid_dy).normalize().mult(enemy_boid_attack_constant)); + + // Turn around if it's going out of bounds + if (this.position.x < margin) { + this.velocity.x += turn_constant; + } else if (this.position.x > width - margin) { + this.velocity.x -= turn_constant; + } + + if (this.position.y < margin) { + this.velocity.y += turn_constant; + } else if (this.position.y > height - margin) { + this.velocity.y -= turn_constant; + } + + // Limit speed + if (this.velocity.mag() > max_speed) { + this.velocity.normalize().mult(max_speed); + } else if (this.velocity.mag() < min_speed) { + this.velocity.normalize().mult(min_speed); + } + + // Update the trail + if (trail) { + this.trail.shift(); + this.trail.push(this.position.copy()); + } + + // Move the boid + this.position.add(this.velocity); + } + + draw(trail = false, shape = "Triangle", direction = false, range = false, avoid = false) { + // Draw the boid as a sphere with a line indicating its direction + if (shape == "Circle") { + stroke(255, 0, 0); + strokeWeight(15); + point(this.position.x, this.position.y); + } else { + // Draw the boid as a triangle + strokeWeight(0); + fill(255, 0, 0); + let angle = this.velocity.heading(); + let a = createVector(0, -30).rotate(angle + PI / 2).add(this.position); + let b = createVector(10, 0).rotate(angle + PI / 2).add(this.position); + let c = createVector(-10, 0).rotate(angle + PI / 2).add(this.position); + triangle(a.x, a.y, b.x, b.y, c.x, c.y); + } + + if (direction) { + stroke(0, 0, 255); + strokeWeight(2); + line(this.position.x, this.position.y, this.position.x + (this.velocity.x * 5), this.position.y + (this.velocity.y * 5)); + } + + // Add a trail behind the boid + if (trail) { + for (let i = 0; i < this.trail.length; i++) { + let alpha = map(i, 0, this.trail.length, 0, 255); + fill(255, 255, 255, alpha); + stroke(255, 255, 255, alpha); + strokeWeight(5); + point(this.trail[i].x, this.trail[i].y); + } + } + + if (range) { + noFill(); + stroke(0, 255, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, visible_range * 2); + } + + if (avoid) { + noFill(); + stroke(255, 0, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, avoidance_range * 2); + } + } +} \ No newline at end of file diff --git a/FriendlyBoid.js b/FriendlyBoid.js new file mode 100644 index 0000000..64e79ee --- /dev/null +++ b/FriendlyBoid.js @@ -0,0 +1,200 @@ +class FriendlyBoid { + constructor(x, y) { + this.position = createVector(x, y); + this.velocity = createVector(0, 0); + + this.trail = []; + for (let i = 0; i < 100; i++) { + this.trail.push(this.position.copy()); + } + } + + update(trail = false, update = true) { + if (!update) { + return; + } + + // Separation + let close_dx = 0; + let close_dy = 0; + + for (let boid of friendly_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < avoidance_range) { + close_dx += this.position.x - boid.position.x; + close_dy += this.position.y - boid.position.y; + } + } + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(avoidance_constant)); + + + // Alignment + let avg_dx = 0; + let avg_dy = 0; + let total_neighbours = 0; + + for (let boid of friendly_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < visible_range) { + avg_dx += boid.velocity.x; + avg_dy += boid.velocity.y; + + total_neighbours++; + } + } + } + + if (total_neighbours > 0) { + avg_dx /= total_neighbours; + avg_dy /= total_neighbours; + } + + this.velocity.add(createVector(avg_dx, avg_dy).normalize().mult(alignment_constant)); + + + // Cohesion + total_neighbours = 0; + let avg_x = 0; + let avg_y = 0; + + for (let boid of friendly_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < visible_range) { + avg_x += boid.position.x; + avg_y += boid.position.y; + + total_neighbours++; + } + } + } + + if (total_neighbours > 0) { + avg_x /= total_neighbours; + avg_y /= total_neighbours; + } + + this.velocity.add(createVector(avg_x - this.position.x, avg_y - this.position.y).normalize().mult(cohesion_constant)); + + + // POI Attraction + let poi_dx = 0; + let poi_dy = 0; + for (let poi of pointsOfInterest) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < poi_attraction_range) { + poi_dx += poi.position.x - this.position.x; + poi_dy += poi.position.y - this.position.y; + } + } + + this.velocity.add(createVector(poi_dx, poi_dy).normalize().mult(interest_constant)); + + // POI Repulsion + close_dx = 0; + close_dy = 0; + + for (let poi of pointsOfInterest) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < poi_protection_range) { + close_dx += this.position.x - poi.position.x; + close_dy += this.position.y - poi.position.y; + } + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(poi_avoidance_constant)); + + + // Enemy Boid Avoidance + close_dx = 0; + close_dy = 0; + + for (let boid of enemy_boids) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < enemy_avoidance_range) { + close_dx += (this.position.x - boid.position.x); + close_dy += (this.position.y - boid.position.y); + } + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(enemy_avoidance_constant)); + + + // Turn around if it's going out of bounds + if (this.position.x < margin) { + this.velocity.x += turn_constant; + } else if (this.position.x > width - margin) { + this.velocity.x -= turn_constant; + } + + if (this.position.y < margin) { + this.velocity.y += turn_constant; + } else if (this.position.y > height - margin) { + this.velocity.y -= turn_constant; + } + + + // Limit speed + if (this.velocity.mag() > max_speed) { + this.velocity.normalize().mult(max_speed); + } else if (this.velocity.mag() < min_speed) { + this.velocity.normalize().mult(min_speed); + } + + // Update the trail + if (trail) { + this.trail.shift(); + this.trail.push(this.position.copy()); + } + + // Move the boid + this.position.add(this.velocity); + } + + draw(trail = false, shape = "Triangle", direction = false, range = false, avoid = false) { + // Draw the boid as a sphere with a line indicating its direction + if (shape == "Circle") { + stroke(0, 255, 0); + strokeWeight(15); + point(this.position.x, this.position.y); + } else { + // Draw the boid as a triangle + strokeWeight(0); + fill(0, 255, 0); + let angle = this.velocity.heading(); + let a = createVector(0, -30).rotate(angle + PI / 2).add(this.position); + let b = createVector(10, 0).rotate(angle + PI / 2).add(this.position); + let c = createVector(-10, 0).rotate(angle + PI / 2).add(this.position); + triangle(a.x, a.y, b.x, b.y, c.x, c.y); + } + + if (direction) { + stroke(0, 0, 255); + strokeWeight(2); + line(this.position.x, this.position.y, this.position.x + (this.velocity.x * 5), this.position.y + (this.velocity.y * 5)); + } + + // Add a trail behind the boid + if (trail) { + for (let i = 0; i < this.trail.length; i++) { + let alpha = map(i, 0, this.trail.length, 0, 255); + fill(255, 255, 255, alpha); + stroke(255, 255, 255, alpha); + strokeWeight(5); + point(this.trail[i].x, this.trail[i].y); + } + } + + if (range) { + noFill(); + stroke(0, 255, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, visible_range * 2); + } + + if (avoid) { + noFill(); + stroke(255, 0, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, avoidance_range * 2); + } + } +} \ No newline at end of file diff --git a/FriendlyBoidOptimised.js b/FriendlyBoidOptimised.js new file mode 100644 index 0000000..3e91021 --- /dev/null +++ b/FriendlyBoidOptimised.js @@ -0,0 +1,161 @@ +class FriendlyBoidOptimised { + constructor(x, y) { + this.position = createVector(x, y); + this.velocity = createVector(0, 0); + + this.trail = []; + for (let i = 0; i < 100; i++) { + this.trail.push(this.position.copy()); + } + } + + update(trail = false, update = true) { + if (!update) { + return; + } + + let close_dx = 0; + let close_dy = 0; + let avg_dx = 0; + let avg_dy = 0; + let avg_x = 0; + let avg_y = 0; + let total_neighbours = 0; + let poi_avoid_dx = 0; + let poi_avoid_dy = 0; + let poi_attract_dx = 0; + let poi_attract_dy = 0; + let enemy_close_dx = 0; + let enemy_close_dy = 0; + + for (let boid of friendly_boids) { + if (boid != this) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < avoidance_range) { + close_dx += this.position.x - boid.position.x; + close_dy += this.position.y - boid.position.y; + } + + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < visible_range) { + avg_dx += boid.velocity.x; + avg_dy += boid.velocity.y; + + avg_x += boid.position.x; + avg_y += boid.position.y; + + total_neighbours++; + } + } + } + + for (let poi of pointsOfInterest) { + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < poi_attraction_range) { + poi_attract_dx += poi.position.x - this.position.x; + poi_attract_dy += poi.position.y - this.position.y; + } + if (dist(this.position.x, this.position.y, poi.position.x, poi.position.y) < poi_protection_range) { + poi_avoid_dx += this.position.x - poi.position.x; + poi_avoid_dy += this.position.y - poi.position.y; + } + } + + for (let boid of enemy_boids) { + if (dist(this.position.x, this.position.y, boid.position.x, boid.position.y) < enemy_avoidance_range) { + enemy_close_dx += (this.position.x - boid.position.x); + enemy_close_dy += (this.position.y - boid.position.y); + } + } + + if (total_neighbours > 0) { + avg_dx /= total_neighbours; + avg_dy /= total_neighbours; + + avg_x /= total_neighbours; + avg_y /= total_neighbours; + } + + this.velocity.add(createVector(close_dx, close_dy).normalize().mult(avoidance_constant)); + this.velocity.add(createVector(avg_dx, avg_dy).normalize().mult(alignment_constant)); + this.velocity.add(createVector(avg_x - this.position.x, avg_y - this.position.y).normalize().mult(cohesion_constant)); + this.velocity.add(createVector(poi_attract_dx, poi_attract_dy).normalize().mult(interest_constant)); + this.velocity.add(createVector(poi_avoid_dx, poi_avoid_dy).normalize().mult(poi_avoidance_constant)); + this.velocity.add(createVector(enemy_close_dx, enemy_close_dy).normalize().mult(enemy_avoidance_constant)); + + // Turn around if it's going out of bounds + if (this.position.x < margin) { + this.velocity.x += turn_constant; + } else if (this.position.x > width - margin) { + this.velocity.x -= turn_constant; + } + + if (this.position.y < margin) { + this.velocity.y += turn_constant; + } else if (this.position.y > height - margin) { + this.velocity.y -= turn_constant; + } + + // Limit speed + if (this.velocity.mag() > max_speed) { + this.velocity.normalize().mult(max_speed); + } else if (this.velocity.mag() < min_speed) { + this.velocity.normalize().mult(min_speed); + } + + // Update the trail + if (trail) { + this.trail.shift(); + this.trail.push(this.position.copy()); + } + + // Move the boid + this.position.add(this.velocity); + } + + draw(trail = false, shape = "Triangle", direction = false, range = false, avoid = false) { + // Draw the boid as a sphere with a line indicating its direction + if (shape == "Circle") { + stroke(0, 255, 0); + strokeWeight(15); + point(this.position.x, this.position.y); + } else { + // Draw the boid as a triangle + strokeWeight(0); + fill(0, 255, 0); + let angle = this.velocity.heading(); + let a = createVector(0, -30).rotate(angle + PI / 2).add(this.position); + let b = createVector(10, 0).rotate(angle + PI / 2).add(this.position); + let c = createVector(-10, 0).rotate(angle + PI / 2).add(this.position); + triangle(a.x, a.y, b.x, b.y, c.x, c.y); + } + + if (direction) { + stroke(0, 0, 255); + strokeWeight(2); + line(this.position.x, this.position.y, this.position.x + (this.velocity.x * 5), this.position.y + (this.velocity.y * 5)); + } + + // Add a trail behind the boid + if (trail) { + for (let i = 0; i < this.trail.length; i++) { + let alpha = map(i, 0, this.trail.length, 0, 255); + fill(255, 255, 255, alpha); + stroke(255, 255, 255, alpha); + strokeWeight(5); + point(this.trail[i].x, this.trail[i].y); + } + } + + if (range) { + noFill(); + stroke(0, 255, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, visible_range * 2); + } + + if (avoid) { + noFill(); + stroke(255, 0, 0, 50); + strokeWeight(2); + ellipse(this.position.x, this.position.y, avoidance_range * 2); + } + } +} \ No newline at end of file diff --git a/PointOfInterest.js b/PointOfInterest.js new file mode 100644 index 0000000..bfc9bc6 --- /dev/null +++ b/PointOfInterest.js @@ -0,0 +1,25 @@ +class PointOfInterest { + constructor(x, y) { + this.position = createVector(x, y); + } + + draw(protect_range = false, attract_range = false) { + stroke(255, 255, 0); + strokeWeight(10); + point(this.position.x, this.position.y); + + if (protect_range) { + noFill(); + stroke(255, 255, 0, 100); + strokeWeight(2); + ellipse(this.position.x, this.position.y, poi_protection_range * 2); + } + + if (attract_range) { + noFill(); + stroke(255, 255, 0, 100); + strokeWeight(2); + ellipse(this.position.x, this.position.y, poi_attraction_range * 2); + } + } +} \ No newline at end of file diff --git a/index.html b/index.html index 45cdefb..195a972 100644 --- a/index.html +++ b/index.html @@ -14,5 +14,10 @@
+ + + + +