From a0735b774baeb38bec679a58b88ed82e375d0900 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Sat, 24 Apr 2021 11:17:34 -0400 Subject: [PATCH] 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) } }