commit
12ed150a17
@ -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"
|
||||
rand = "0.8.3"
|
||||
tiled = "0.9.4"
|
||||
|
@ -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
|
||||
}
|
||||
]
|
||||
}
|
130
src/entities/fish.rs
Normal file
130
src/entities/fish.rs
Normal file
@ -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<Vector2>) -> Vec<Self> {
|
||||
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<RaylibDrawHandle>) {
|
||||
// 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,
|
||||
);
|
||||
}
|
||||
}
|
1
src/entities/mod.rs
Normal file
1
src/entities/mod.rs
Normal file
@ -0,0 +1 @@
|
||||
pub mod fish;
|
@ -1 +1,2 @@
|
||||
pub mod profiler;
|
||||
pub mod profiler;
|
||||
pub mod triangles;
|
14
src/lib/utils/triangles.rs
Normal file
14
src/lib/utils/triangles.rs
Normal file
@ -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()),
|
||||
};
|
||||
|
||||
}
|
@ -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<GameState> {
|
||||
// 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);
|
||||
}
|
||||
|
@ -5,6 +5,7 @@ mod resources;
|
||||
mod player;
|
||||
mod world;
|
||||
mod pallette;
|
||||
mod entities;
|
||||
mod items;
|
||||
|
||||
use gamecore::{GameCore, GameProgress, GameState};
|
||||
|
17
src/world.rs
17
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<Vector2>,
|
||||
|
||||
#[serde(skip)]
|
||||
pub fish: Vec<FishEntity>
|
||||
}
|
||||
|
||||
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)
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user