This commit is contained in:
rsninja722 2021-04-24 16:02:46 -04:00
parent 6fd77dffcf
commit b6bcdad88c
2 changed files with 83 additions and 31 deletions

File diff suppressed because one or more lines are too long

View File

@ -8,10 +8,20 @@ const FISH_FOLLOW_PLAYER_SPEED: f32 = 1.8;
const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = FISH_FOLLOW_PLAYER_SPEED * 3.0; const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = FISH_FOLLOW_PLAYER_SPEED * 3.0;
const FISH_ATTACH_RADIUS: f32 = 20.0; const FISH_ATTACH_RADIUS: f32 = 20.0;
const FISH_VISION: f32 = 25.0;
const FISH_MAX_SPEED: f32 = 2.0;
const FISH_MAX_FORCE: f32 = 0.05;
const FISH_FACTOR_ATTRACTION: f32 = 1.0;
const FISH_FACTOR_PLAYER: f32 = 0.1;
const FISH_FACTOR_COHESION: f32 = 0.1;
const FISH_SEPARATION_DISTANCE: f32 = 15.0;
const FISH_FACTOR_SEPARATION: f32 = 1.5;
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct FishEntity { pub struct FishEntity {
position: Vector2, position: Vector2,
direction: Vector2, direction: Vector2,
velocity: Vector2,
pub following_player: bool, pub following_player: bool,
size: Vector2, size: Vector2,
rng: ThreadRng rng: ThreadRng
@ -22,6 +32,7 @@ impl FishEntity {
Self { Self {
position: position, position: position,
direction: Vector2::zero(), direction: Vector2::zero(),
velocity: Vector2::zero(),
following_player: false, following_player: false,
size: Vector2 { x: 5.0, y: 8.0 }, size: Vector2 { x: 5.0, y: 8.0 },
rng: rand::thread_rng() rng: rand::thread_rng()
@ -37,33 +48,80 @@ impl FishEntity {
} }
pub fn handle_follow_player(&mut self, player: &Player, dt: f64, other_fish: &Vec<FishEntity>) { pub fn handle_follow_player(&mut self, player: &Player, dt: f64, other_fish: &Vec<FishEntity>) {
// Distance and direction to player let mut acceleration: Vector2 = Vector2::zero();
let dist_to_player = player.position - self.position;
let dist_to_player_lin = self.position.distance_to(player.position); let mut steer: Vector2 = Vector2::zero();
let mut direction_to_player = dist_to_player; let mut count1: u16 = 0;
direction_to_player.normalize(); let mut sum1: Vector2 = Vector2::zero();
let mut count2: u16 = 0;
// Fish movement let mut sum2: Vector2 = Vector2::zero();
let movement; let mut count3: u16 = 0;
// separation
// Random variance for i in other_fish {
let variance = self.rng.gen_range(500.0..1000.0) / 1000.0; let dist = (self.position - i.position).length();
if dist < FISH_SEPARATION_DISTANCE && dist > 0.0 {
// If the fish is double its follow distance from the player let mut diff: Vector2 = self.position - i.position;
if dist_to_player_lin.abs() > (FISH_FOLLOW_PLAYER_DISTANCE * 2.0) { diff.normalize();
movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST * variance; diff /= dist;
} else { steer += diff;
// Move slowly in the direction of the player unless too close count1 += 1;
if dist_to_player_lin.abs() > FISH_FOLLOW_PLAYER_DISTANCE { }
movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED * variance; if dist < FISH_VISION && dist > 0.0 {
} else { sum1 += i.direction;
movement = Vector2::zero(); count2 += 1;
sum2 += i.position;
count3 += 1;
} }
} }
if count1 > 0 {
steer /= count1 as f32;
}
if steer.x != 0.0 || steer.y != 0.0 {
steer.normalize();
steer *= FISH_MAX_SPEED;
steer -= self.velocity;
steer.x = f32::min(f32::max(steer.x, -FISH_MAX_FORCE), FISH_MAX_FORCE);
steer.y = f32::min(f32::max(steer.y, -FISH_MAX_FORCE), FISH_MAX_FORCE);
acceleration += steer * FISH_FACTOR_SEPARATION;
}
// attraction
if count2 > 0 {
sum1 /= count2 as f32;
sum1.normalize();
sum1 *= FISH_MAX_SPEED;
sum1 -= self.velocity;
sum1.x = f32::min(f32::max(sum1.x, -FISH_MAX_FORCE), FISH_MAX_FORCE);
sum1.y = f32::min(f32::max(sum1.y, -FISH_MAX_FORCE), FISH_MAX_FORCE);
acceleration += sum1 * FISH_FACTOR_ATTRACTION;
}
// cohesion
if count3 > 0 {
sum2 /= count3 as f32;
let mut desired: Vector2 = sum2 - self.position;
desired.normalize();
desired *= FISH_MAX_SPEED;
desired.x = f32::min(f32::max(desired.x, -FISH_MAX_FORCE), FISH_MAX_FORCE);
desired.y = f32::min(f32::max(desired.y, -FISH_MAX_FORCE), FISH_MAX_FORCE);
acceleration += desired * FISH_FACTOR_COHESION;
}
// turn to player
let mut player_factor: Vector2 = player.position - self.position;
player_factor.normalize();
acceleration += player_factor * FISH_FACTOR_PLAYER;
// Move the fish // Move the fish
self.direction = direction_to_player; self.direction = self.velocity.normalized();
self.position += movement; self.velocity += acceleration;
self.velocity.x = f32::min(f32::max(self.velocity.x, -FISH_MAX_SPEED), FISH_MAX_SPEED);
self.velocity.y = f32::min(f32::max(self.velocity.y, -FISH_MAX_SPEED), FISH_MAX_SPEED);
self.position += self.velocity;
} }
pub fn handle_free_movement(&mut self, player: &mut Player, dt: f64) { pub fn handle_free_movement(&mut self, player: &mut Player, dt: f64) {
@ -76,6 +134,7 @@ impl FishEntity {
// Handle player picking up fish // Handle player picking up fish
if player.position.distance_to(self.position).abs() <= player.size.y * 2.2 { if player.position.distance_to(self.position).abs() <= player.size.y * 2.2 {
self.following_player = true; self.following_player = true;
self.velocity = self.direction.normalized();
// Add currency to the player // Add currency to the player
player.coins += 1; player.coins += 1;