Very basic collisions with the world (janky)

This commit is contained in:
Evan Pratten 2021-10-02 16:53:50 -04:00
parent a76dd7de72
commit f1e10be5da
6 changed files with 67 additions and 17 deletions

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

After

Width:  |  Height:  |  Size: 16 KiB

View File

@ -2,11 +2,17 @@ use std::ops::Mul;
use raylib::math::{Rectangle, Vector2}; use raylib::math::{Rectangle, Vector2};
use crate::scenes::ingame_scene::world::WORLD_LEVEL_X_OFFSET;
use super::{CharacterState, MainCharacter}; use super::{CharacterState, MainCharacter};
pub const GRAVITY_PPS: f32 = 2.0; pub const GRAVITY_PPS: f32 = 2.0;
pub fn modify_player_based_on_forces(player: &mut MainCharacter) -> Result<(), ()> { pub fn modify_player_based_on_forces(
player: &mut MainCharacter,
colliders: &Vec<Rectangle>,
level_height_offset: f32,
) -> Result<(), ()> {
// Modify the player's velocity by the forces // Modify the player's velocity by the forces
player.movement_force += player.base_velocity; player.movement_force += player.base_velocity;
player.velocity = player.movement_force; player.velocity = player.movement_force;
@ -14,7 +20,7 @@ pub fn modify_player_based_on_forces(player: &mut MainCharacter) -> Result<(), (
// Predict the player's position next frame // Predict the player's position next frame
let predicted_player_position = player.position + player.velocity; let predicted_player_position = player.position + player.velocity;
// Calculate a bounding rect around the player // Calculate a bounding rect around the player both now, and one frame in the future
let player_rect = Rectangle::new( let player_rect = Rectangle::new(
predicted_player_position.x - (player.size.x / 2.0), predicted_player_position.x - (player.size.x / 2.0),
predicted_player_position.y - (player.size.x / 2.0), predicted_player_position.y - (player.size.x / 2.0),
@ -25,10 +31,24 @@ pub fn modify_player_based_on_forces(player: &mut MainCharacter) -> Result<(), (
// Calculate a generic "floor" to always collide with // Calculate a generic "floor" to always collide with
let floor_rect = Rectangle::new(f32::MIN, 0.0, f32::MAX, 1.0); let floor_rect = Rectangle::new(f32::MIN, 0.0, f32::MAX, 1.0);
// Check collision conditions
let check_player_colliding_with_floor = || floor_rect.check_collision_recs(&player_rect);
let check_player_colliding_with_floor_next_frame =
|| player_rect.y + player_rect.height > floor_rect.y;
let check_player_colliding_with_colliders = || {
colliders.iter().any(|rect| {
let mut translated_rect = rect.clone();
translated_rect.y += level_height_offset;
translated_rect.x += WORLD_LEVEL_X_OFFSET;
translated_rect.check_collision_recs(&player_rect)
})
};
// If the player is colliding, only apply the x force // If the player is colliding, only apply the x force
if (floor_rect.check_collision_recs(&player_rect) if (check_player_colliding_with_floor()
|| player_rect.y + player_rect.height > floor_rect.y) || check_player_colliding_with_floor_next_frame()
&& player.velocity.y > 0.0 || check_player_colliding_with_colliders())
&& player.velocity.y != 0.0
{ {
player.velocity.y = 0.0; player.velocity.y = 0.0;
@ -36,11 +56,17 @@ pub fn modify_player_based_on_forces(player: &mut MainCharacter) -> Result<(), (
if player.current_state == CharacterState::Jumping if player.current_state == CharacterState::Jumping
|| player.current_state == CharacterState::Dashing || player.current_state == CharacterState::Dashing
{ {
player.update_player(Some(CharacterState::Running)); player.update_player(
Some(CharacterState::Running),
colliders,
level_height_offset,
);
return Ok(()); return Ok(());
} }
} }
// Check sideways collisions
// Finally apply the velocity to the player // Finally apply the velocity to the player
player.position += player.velocity; player.position += player.velocity;

View File

@ -2,7 +2,10 @@ pub mod collisions;
pub mod render; pub mod render;
use chrono::{DateTime, Utc}; use chrono::{DateTime, Utc};
use raylib::{math::Vector2, texture::Texture2D}; use raylib::{
math::{Rectangle, Vector2},
texture::Texture2D,
};
use crate::utilities::anim_render::AnimatedSpriteSheet; use crate::utilities::anim_render::AnimatedSpriteSheet;
@ -48,7 +51,12 @@ impl MainCharacter {
} }
} }
pub fn update_player(&mut self, state: Option<CharacterState>) { pub fn update_player(
&mut self,
state: Option<CharacterState>,
colliders: &Vec<Rectangle>,
level_height_offset: f32,
) {
if let Some(state) = state { if let Some(state) = state {
// Update the internal state // Update the internal state
if state != self.current_state { if state != self.current_state {
@ -65,6 +73,6 @@ impl MainCharacter {
} }
// Update the player based on the new velocity // Update the player based on the new velocity
modify_player_based_on_forces(self).unwrap(); modify_player_based_on_forces(self, colliders, level_height_offset).unwrap();
} }
} }

View File

@ -18,7 +18,7 @@ use tracing::{debug, trace};
mod hud; mod hud;
pub mod level; pub mod level;
mod update; mod update;
mod world; pub mod world;
#[derive(Debug)] #[derive(Debug)]
pub struct InGameScreen { pub struct InGameScreen {
@ -43,7 +43,7 @@ impl InGameScreen {
rotation: 0.0, rotation: 0.0,
zoom: 1.0, zoom: 1.0,
}, },
player: MainCharacter::new(Vector2::new(0.0, -80.0), player_sprite_sheet), player: MainCharacter::new(Vector2::new(0.0, -85.0), player_sprite_sheet),
world_background: WorldPaintTexture::new(background_texture), world_background: WorldPaintTexture::new(background_texture),
levels, levels,
current_level_idx: 0, current_level_idx: 0,
@ -61,7 +61,12 @@ impl Action<Scenes, ScreenError, GameContext> for InGameScreen {
debug!("Running InGameScreen for the first time"); debug!("Running InGameScreen for the first time");
// Set the player to running // Set the player to running
self.player.update_player(Some(CharacterState::Running)); let cur_level = self.levels.get(self.current_level_idx).unwrap();
self.player.update_player(
Some(CharacterState::Running),
&cur_level.colliders,
-cur_level.platform_tex.height as f32,
);
Ok(()) Ok(())
} }

View File

@ -17,6 +17,11 @@ impl FrameUpdate for InGameScreen {
config: &GameConfig, config: &GameConfig,
) { ) {
puffin::profile_function!(); puffin::profile_function!();
// Get the current level
let cur_level = self.levels.get(self.current_level_idx).unwrap();
// Set the camera's offset based on screen size // Set the camera's offset based on screen size
self.camera.offset = raylib.get_screen_size().div(Vector2::new(2.0, 1.05)); self.camera.offset = raylib.get_screen_size().div(Vector2::new(2.0, 1.05));
self.camera.target = Vector2::new(self.player.position.x, self.camera.target.y); self.camera.target = Vector2::new(self.player.position.x, self.camera.target.y);
@ -28,16 +33,20 @@ impl FrameUpdate for InGameScreen {
&& !(self.player.current_state == CharacterState::Dashing); && !(self.player.current_state == CharacterState::Dashing);
if is_jump { if is_jump {
self.player.update_player(Some(CharacterState::Jumping)); self.player.update_player(Some(CharacterState::Jumping), &cur_level.colliders,
-cur_level.platform_tex.height as f32,);
} else if is_dash { } else if is_dash {
self.player.update_player(Some(CharacterState::Dashing)); self.player.update_player(Some(CharacterState::Dashing), &cur_level.colliders,
-cur_level.platform_tex.height as f32,);
} else { } else {
if self.player.current_state != CharacterState::Jumping if self.player.current_state != CharacterState::Jumping
&& self.player.current_state != CharacterState::Dashing && self.player.current_state != CharacterState::Dashing
{ {
self.player.update_player(Some(CharacterState::Running)); self.player.update_player(Some(CharacterState::Running), &cur_level.colliders,
-cur_level.platform_tex.height as f32,);
} else { } else {
self.player.update_player(None); self.player.update_player(None, &cur_level.colliders,
-cur_level.platform_tex.height as f32,);
} }
} }
} }

View File

@ -8,6 +8,8 @@ use crate::{
}; };
use raylib::prelude::*; use raylib::prelude::*;
pub const WORLD_LEVEL_X_OFFSET: f32 = 200.0;
impl WorldSpaceRender for InGameScreen { impl WorldSpaceRender for InGameScreen {
fn render_world_space( fn render_world_space(
&self, &self,
@ -23,7 +25,7 @@ impl WorldSpaceRender for InGameScreen {
// self.world_background.render(raylib, Vector2::new(0.0, -1080.0), &self.camera); // self.world_background.render(raylib, Vector2::new(0.0, -1080.0), &self.camera);
// Render the platform layer // Render the platform layer
raylib.draw_texture_v(&cur_level.platform_tex, Vector2::new(-10.0, -cur_level.platform_tex.height as f32), Color::WHITE); raylib.draw_texture_v(&cur_level.platform_tex, Vector2::new(WORLD_LEVEL_X_OFFSET, -cur_level.platform_tex.height as f32), Color::WHITE);
// Render the floor as a line // Render the floor as a line
let screen_world_zero = raylib.get_screen_to_world2D(Vector2::zero(), self.camera); let screen_world_zero = raylib.get_screen_to_world2D(Vector2::zero(), self.camera);