From a0735b774baeb38bec679a58b88ed82e375d0900 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:17:34 -0400 Subject: [PATCH 1/3] triangle fish --- assets/worlds/mainworld.json | 8 ++- src/entities/fish.rs | 98 ++++++++++++++++++++++++++++++++++++ src/entities/mod.rs | 1 + src/lib/utils/mod.rs | 3 +- src/lib/utils/triangles.rs | 14 ++++++ src/logic/ingame/mod.rs | 14 ++++-- src/main.rs | 1 + src/world.rs | 17 ++++++- 8 files changed, 149 insertions(+), 7 deletions(-) create mode 100644 src/entities/fish.rs create mode 100644 src/entities/mod.rs create mode 100644 src/lib/utils/triangles.rs diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index f34b38e..70fb440 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -2,5 +2,11 @@ "end_position": { "x": 10000.0, "y": 10000.0 - } + }, + "fish": [ + { + "x": 100.0, + "y": 100.0 + } + ] } \ No newline at end of file diff --git a/src/entities/fish.rs b/src/entities/fish.rs new file mode 100644 index 0000000..7b01a6b --- /dev/null +++ b/src/entities/fish.rs @@ -0,0 +1,98 @@ +use raylib::prelude::*; + +use crate::{gamecore::GameCore, lib::utils::triangles::rotate_vector, player::Player}; + +const FISH_FOLLOW_PLAYER_DISTANCE: f32 = 80.0; +const FISH_FOLLOW_PLAYER_SPEED: f32 = 2.0; +const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = 6.0; + +#[derive(Debug, Clone)] +pub struct FishEntity { + position: Vector2, + direction: Vector2, + following_player: bool, + size: Vector2 +} + +impl FishEntity { + pub fn new(position: Vector2) -> Self { + Self { + position: position, + direction: Vector2::zero(), + following_player: true, + size: Vector2 { + x: 5.0, + y: 8.0 + } + } + } + + pub fn new_from_positions(positions: &Vec) -> Vec { + let mut output = Vec::new(); + for position in positions { + output.push(FishEntity::new(*position)); + } + return output; + } + + pub fn handle_follow_player(&mut self, player: &Player) { + // Distance and direction to player + let dist_to_player = player.position - self.position; + let dist_to_player_lin = self.position.distance_to(player.position); + let mut direction_to_player = dist_to_player; + direction_to_player.normalize(); + + // Fish movement + let movement; + + // If the fish is double its follow distance from the player + if dist_to_player_lin.abs() > (FISH_FOLLOW_PLAYER_DISTANCE * 2.0) { + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST; + } else { + // Move slowly in the direction of the player unless too close + if dist_to_player_lin.abs() > FISH_FOLLOW_PLAYER_DISTANCE { + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED; + } else { + movement = Vector2::zero(); + } + } + + // Move the fish + self.direction = direction_to_player; + self.position += movement; + } + + pub fn handle_free_movement(&mut self) {} + + pub fn update_position(&mut self, player: &Player, dt: f64) { + if self.following_player { + self.handle_follow_player(player); + } else { + self.handle_free_movement(); + } + } + + pub fn render(&self, context_2d: &mut RaylibMode2D) { + + // Direction + let direction = Vector2::zero().angle_to(self.direction.normalized()) + (90.0 as f32).to_radians(); + + // Get the corners of the fish + let fish_front = rotate_vector(Vector2 { + x: 0.0, + y: (self.size.y / 2.0) * -1.0, + }, direction); + let fish_bl = rotate_vector(Vector2 { + x: (self.size.x / 2.0) * -1.0, + y: (self.size.y / 2.0), + }, direction); + let fish_br = rotate_vector(Vector2 { + x: (self.size.x / 2.0), + y: (self.size.y / 2.0), + }, direction); + + // Draw the fish as a triangle with rotation + context_2d.draw_triangle(self.position + fish_front, self.position + fish_bl, self.position + fish_br, Color::BLACK); + + } +} diff --git a/src/entities/mod.rs b/src/entities/mod.rs new file mode 100644 index 0000000..e948307 --- /dev/null +++ b/src/entities/mod.rs @@ -0,0 +1 @@ +pub mod fish; \ No newline at end of file diff --git a/src/lib/utils/mod.rs b/src/lib/utils/mod.rs index c570215..9d6dd04 100644 --- a/src/lib/utils/mod.rs +++ b/src/lib/utils/mod.rs @@ -1 +1,2 @@ -pub mod profiler; \ No newline at end of file +pub mod profiler; +pub mod triangles; \ No newline at end of file diff --git a/src/lib/utils/triangles.rs b/src/lib/utils/triangles.rs new file mode 100644 index 0000000..a0f1c10 --- /dev/null +++ b/src/lib/utils/triangles.rs @@ -0,0 +1,14 @@ +use raylib::math::Vector2; + + +pub fn rotate_vector(vector: Vector2, angle_rad: f32) -> Vector2{ + + // let dist = (vector.x * vector.x) + (vector.y * vector.y); + // let angle = (vector.x.abs() / vector.y.abs()).atan(); + // let angle = angle + angle_rad; + return Vector2 { + x: (vector.x * angle_rad.cos()) - (vector.y * angle_rad.sin()), + y: (vector.y * angle_rad.cos()) + (vector.x * angle_rad.sin()), + }; + +} \ No newline at end of file diff --git a/src/logic/ingame/mod.rs b/src/logic/ingame/mod.rs index 7297289..fb93e50 100644 --- a/src/logic/ingame/mod.rs +++ b/src/logic/ingame/mod.rs @@ -1,5 +1,5 @@ -mod playerlogic; mod hud; +mod playerlogic; use raylib::prelude::*; @@ -33,8 +33,6 @@ impl InGameScreen { ) { context_2d.draw_circle(0, 0, 10.0, Color::BLACK); } - - } impl Screen for InGameScreen { @@ -45,6 +43,9 @@ impl Screen for InGameScreen { audio_system: &mut AudioPlayer, game_core: &mut GameCore, ) -> Option { + // Calculate DT + let dt = draw_handle.get_time() - game_core.last_frame_time; + // Clear frame draw_handle.clear_background(Color::BLUE); @@ -71,6 +72,13 @@ impl Screen for InGameScreen { // Render the world self.render_world(&mut context_2d, game_core); + // Render entities + let mut fish = &mut game_core.world.fish; + for fish in fish.iter_mut() { + fish.update_position(&game_core.player, dt); + fish.render(&mut context_2d); + } + // Render Player playerlogic::render_player(&mut context_2d, game_core); } diff --git a/src/main.rs b/src/main.rs index 85ff39b..8fd784b 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,6 +5,7 @@ mod resources; mod player; mod world; mod pallette; +mod entities; use gamecore::{GameCore, GameState}; use lib::{utils::profiler::GameProfiler, wrappers::audio::player::AudioPlayer}; diff --git a/src/world.rs b/src/world.rs index 309fd41..0212480 100644 --- a/src/world.rs +++ b/src/world.rs @@ -5,9 +5,17 @@ use serde::{Deserialize, Serialize}; use std::io::Read; use failure::Error; +use crate::entities::fish::FishEntity; + #[derive(Debug, Serialize, Deserialize, Clone)] pub struct World { - pub end_position: Vector2 + pub end_position: Vector2, + + #[serde(rename = "fish")] + pub fish_positions: Vec, + + #[serde(skip)] + pub fish: Vec } impl World { @@ -17,6 +25,11 @@ impl World { let reader = BufReader::new(file); // Deserialize - Ok(serde_json::from_reader(reader)?) + let mut result: World = serde_json::from_reader(reader)?; + + // Init all fish + result.fish = FishEntity::new_from_positions(&result.fish_positions); + + Ok(result) } } From e3b523ecc152eeb058c79454d0eebab22ac15bb5 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:34:19 -0400 Subject: [PATCH 2/3] fish movement --- Cargo.toml | 1 + assets/worlds/mainworld.json | 8 +++- src/entities/fish.rs | 77 +++++++++++++++++++++++------------- 3 files changed, 56 insertions(+), 30 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 36bf495..6c805d4 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -17,3 +17,4 @@ parry2d = "0.4.0" log = "0.4.14" env_logger = "0.8.3" nalgebra = "0.26.1" +rand = "0.8.3" \ No newline at end of file diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index 70fb440..9d52a17 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -5,8 +5,12 @@ }, "fish": [ { - "x": 100.0, - "y": 100.0 + "x": 500.0, + "y": 500.0 + }, + { + "x": 800.0, + "y": 500.0 } ] } \ No newline at end of file diff --git a/src/entities/fish.rs b/src/entities/fish.rs index 7b01a6b..f2a784c 100644 --- a/src/entities/fish.rs +++ b/src/entities/fish.rs @@ -1,17 +1,20 @@ +use rand::{Rng, prelude::ThreadRng}; use raylib::prelude::*; use crate::{gamecore::GameCore, lib::utils::triangles::rotate_vector, player::Player}; const FISH_FOLLOW_PLAYER_DISTANCE: f32 = 80.0; const FISH_FOLLOW_PLAYER_SPEED: f32 = 2.0; -const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = 6.0; +const FISH_FOLLOW_PLAYER_SPEED_FAST: f32 = FISH_FOLLOW_PLAYER_SPEED * 3.0; +const FISH_ATTACH_RADIUS: f32 = 20.0; #[derive(Debug, Clone)] pub struct FishEntity { position: Vector2, direction: Vector2, following_player: bool, - size: Vector2 + size: Vector2, + rng: ThreadRng } impl FishEntity { @@ -20,10 +23,8 @@ impl FishEntity { position: position, direction: Vector2::zero(), following_player: true, - size: Vector2 { - x: 5.0, - y: 8.0 - } + size: Vector2 { x: 5.0, y: 8.0 }, + rng: rand::thread_rng() } } @@ -35,7 +36,7 @@ impl FishEntity { return output; } - pub fn handle_follow_player(&mut self, player: &Player) { + pub fn handle_follow_player(&mut self, player: &Player, dt: f64) { // Distance and direction to player let dist_to_player = player.position - self.position; let dist_to_player_lin = self.position.distance_to(player.position); @@ -45,13 +46,16 @@ impl FishEntity { // Fish movement let movement; + // Random variance + let variance = self.rng.gen_range(500.0..1000.0) / 1000.0; + // If the fish is double its follow distance from the player if dist_to_player_lin.abs() > (FISH_FOLLOW_PLAYER_DISTANCE * 2.0) { - movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST; + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED_FAST * variance; } else { // Move slowly in the direction of the player unless too close if dist_to_player_lin.abs() > FISH_FOLLOW_PLAYER_DISTANCE { - movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED; + movement = direction_to_player * FISH_FOLLOW_PLAYER_SPEED * variance; } else { movement = Vector2::zero(); } @@ -62,37 +66,54 @@ impl FishEntity { self.position += movement; } - pub fn handle_free_movement(&mut self) {} + pub fn handle_free_movement(&mut self, player: &Player, dt: f64) { + if player.position.distance_to(self.position).abs() <= player.size.y * 2.0 { + self.following_player = true; + } + } pub fn update_position(&mut self, player: &Player, dt: f64) { if self.following_player { - self.handle_follow_player(player); + self.handle_follow_player(player, dt); } else { - self.handle_free_movement(); + self.handle_free_movement(player, dt); } } pub fn render(&self, context_2d: &mut RaylibMode2D) { - // Direction - let direction = Vector2::zero().angle_to(self.direction.normalized()) + (90.0 as f32).to_radians(); + let direction = + Vector2::zero().angle_to(self.direction.normalized()) + (90.0 as f32).to_radians(); // Get the corners of the fish - let fish_front = rotate_vector(Vector2 { - x: 0.0, - y: (self.size.y / 2.0) * -1.0, - }, direction); - let fish_bl = rotate_vector(Vector2 { - x: (self.size.x / 2.0) * -1.0, - y: (self.size.y / 2.0), - }, direction); - let fish_br = rotate_vector(Vector2 { - x: (self.size.x / 2.0), - y: (self.size.y / 2.0), - }, direction); + let fish_front = rotate_vector( + Vector2 { + x: 0.0, + y: (self.size.y / 2.0) * -1.0, + }, + direction, + ); + let fish_bl = rotate_vector( + Vector2 { + x: (self.size.x / 2.0) * -1.0, + y: (self.size.y / 2.0), + }, + direction, + ); + let fish_br = rotate_vector( + Vector2 { + x: (self.size.x / 2.0), + y: (self.size.y / 2.0), + }, + direction, + ); // Draw the fish as a triangle with rotation - context_2d.draw_triangle(self.position + fish_front, self.position + fish_bl, self.position + fish_br, Color::BLACK); - + context_2d.draw_triangle( + self.position + fish_front, + self.position + fish_bl, + self.position + fish_br, + Color::BLACK, + ); } } From 877a5ba4484f70faabe5468a4ab52bec52d99ccf Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:37:17 -0400 Subject: [PATCH 3/3] fish pickup --- assets/worlds/mainworld.json | 4 ++-- src/entities/fish.rs | 15 +++++++++++++-- 2 files changed, 15 insertions(+), 4 deletions(-) diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index 9d52a17..45a78a5 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -6,11 +6,11 @@ "fish": [ { "x": 500.0, - "y": 500.0 + "y": 300.0 }, { "x": 800.0, - "y": 500.0 + "y": 200.0 } ] } \ No newline at end of file diff --git a/src/entities/fish.rs b/src/entities/fish.rs index f2a784c..869fb61 100644 --- a/src/entities/fish.rs +++ b/src/entities/fish.rs @@ -22,7 +22,7 @@ impl FishEntity { Self { position: position, direction: Vector2::zero(), - following_player: true, + following_player: false, size: Vector2 { x: 5.0, y: 8.0 }, rng: rand::thread_rng() } @@ -67,9 +67,20 @@ impl FishEntity { } pub fn handle_free_movement(&mut self, player: &Player, dt: f64) { - if player.position.distance_to(self.position).abs() <= player.size.y * 2.0 { + // Distance and direction to player + let dist_to_player = player.position - self.position; + let dist_to_player_lin = self.position.distance_to(player.position); + let mut direction_to_player = dist_to_player; + direction_to_player.normalize(); + + // Handle player picking up fish + if player.position.distance_to(self.position).abs() <= player.size.y * 1.2 { self.following_player = true; } + + // Look at the player; + self.position = self.position; + self.direction = direction_to_player; } pub fn update_position(&mut self, player: &Player, dt: f64) {