diff --git a/Cargo.toml b/Cargo.toml index ed60464..853d3d3 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -17,4 +17,5 @@ parry2d = "0.4.0" log = "0.4.14" env_logger = "0.8.3" nalgebra = "0.26.1" -tiled = "0.9.4" \ No newline at end of file +rand = "0.8.3" +tiled = "0.9.4" diff --git a/assets/worlds/mainworld.json b/assets/worlds/mainworld.json index f34b38e..45a78a5 100644 --- a/assets/worlds/mainworld.json +++ b/assets/worlds/mainworld.json @@ -2,5 +2,15 @@ "end_position": { "x": 10000.0, "y": 10000.0 - } + }, + "fish": [ + { + "x": 500.0, + "y": 300.0 + }, + { + "x": 800.0, + "y": 200.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..869fb61 --- /dev/null +++ b/src/entities/fish.rs @@ -0,0 +1,130 @@ +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 = 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, + rng: ThreadRng +} + +impl FishEntity { + pub fn new(position: Vector2) -> Self { + Self { + position: position, + direction: Vector2::zero(), + following_player: false, + size: Vector2 { x: 5.0, y: 8.0 }, + rng: rand::thread_rng() + } + } + + 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, 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); + let mut direction_to_player = dist_to_player; + direction_to_player.normalize(); + + // 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 * 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 * variance; + } else { + movement = Vector2::zero(); + } + } + + // Move the fish + self.direction = direction_to_player; + self.position += movement; + } + + pub fn handle_free_movement(&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); + 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) { + if self.following_player { + self.handle_follow_player(player, dt); + } else { + 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(); + + // 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 67f4c0b..0016bc3 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,6 +5,7 @@ mod resources; mod player; mod world; mod pallette; +mod entities; mod items; use gamecore::{GameCore, GameProgress, GameState}; 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) } }