Merge remote-tracking branch 'origin/master' into rsninja/fixes
This commit is contained in:
commit
b72051ebbe
@ -1,4 +1,4 @@
|
|||||||
# Ludum Dare 50: *unnamed game*
|
# Ludum Dare 50: *Melting Point*
|
||||||
[](https://github.com/Ewpratten/ludum-dare-50/actions/workflows/build.yml)
|
[](https://github.com/Ewpratten/ludum-dare-50/actions/workflows/build.yml)
|
||||||
|
|
||||||
## Navigating this repository
|
## Navigating this repository
|
||||||
|
BIN
game/dist/assets/cut/cut_intro/cut_intro.png
vendored
Normal file
BIN
game/dist/assets/cut/cut_intro/cut_intro.png
vendored
Normal file
Binary file not shown.
After Width: | Height: | Size: 54 KiB |
Before Width: | Height: | Size: 47 KiB After Width: | Height: | Size: 47 KiB |
5
game/dist/project-constants.json
vendored
5
game/dist/project-constants.json
vendored
@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"game_name": "Ludum Dare 50",
|
"game_name": "Melting Point",
|
||||||
"base_window_size": [
|
"base_window_size": [
|
||||||
1080,
|
1080,
|
||||||
720
|
720
|
||||||
@ -21,6 +21,7 @@
|
|||||||
"max_velocity": 3,
|
"max_velocity": 3,
|
||||||
"acceleration": 2,
|
"acceleration": 2,
|
||||||
"deceleration": 1,
|
"deceleration": 1,
|
||||||
"start_size": 0.8
|
"start_size": 0.8,
|
||||||
|
"melt_speed": 0.00944443
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,8 @@
|
|||||||
//! - If you want to add data to the save state file or settings file, check out the [`persistent`](persistent/index.html) module.
|
//! - If you want to add data to the save state file or settings file, check out the [`persistent`](persistent/index.html) module.
|
||||||
#![doc(issue_tracker_base_url = "https://github.com/Ewpratten/ludum-dare-50/issues/")]
|
#![doc(issue_tracker_base_url = "https://github.com/Ewpratten/ludum-dare-50/issues/")]
|
||||||
|
|
||||||
|
use discord_sdk::activity::Assets;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
asset_manager::load_json_structure,
|
asset_manager::load_json_structure,
|
||||||
discord::{DiscordRpcSignal, DiscordRpcThreadHandle},
|
discord::{DiscordRpcSignal, DiscordRpcThreadHandle},
|
||||||
@ -69,6 +71,12 @@ pub async fn entrypoint(force_recreate_savefiles: bool) {
|
|||||||
.expect("Failed to connect to Discord RPC");
|
.expect("Failed to connect to Discord RPC");
|
||||||
let event_loop_discord_tx = discord.get_channel();
|
let event_loop_discord_tx = discord.get_channel();
|
||||||
let discord_task_handle = discord.begin_thread_non_blocking();
|
let discord_task_handle = discord.begin_thread_non_blocking();
|
||||||
|
event_loop_discord_tx
|
||||||
|
.send(DiscordRpcSignal::ChangeAssets(Assets::default().large(
|
||||||
|
project_constants.discord.artwork.get("logo").unwrap(),
|
||||||
|
Some(""),
|
||||||
|
)))
|
||||||
|
.await.unwrap();
|
||||||
|
|
||||||
// Blocking call to the graphics rendering loop.
|
// Blocking call to the graphics rendering loop.
|
||||||
rendering::event_loop::handle_graphics_blocking(
|
rendering::event_loop::handle_graphics_blocking(
|
||||||
|
@ -1,22 +1,53 @@
|
|||||||
use nalgebra as na;
|
use nalgebra as na;
|
||||||
|
|
||||||
|
use raylib::prelude::*;
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
use crate::asset_manager::load_texture_from_internal_data;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
pub struct Player {
|
pub struct Player {
|
||||||
pub position: na::Vector2<f32>,
|
pub position: na::Vector2<f32>,
|
||||||
pub velocity: na::Vector2<f32>,
|
pub velocity: na::Vector2<f32>,
|
||||||
pub size: f32,
|
pub size: f32,
|
||||||
|
pub active_texture: i32,
|
||||||
|
pub textures: Vec<Texture2D>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Player {
|
impl Player {
|
||||||
|
|
||||||
/// Construct a new player.
|
/// Construct a new player.
|
||||||
pub fn new(position: na::Vector2<f32>) -> Self {
|
pub fn new(
|
||||||
|
raylib_handle: &mut raylib::RaylibHandle,
|
||||||
|
thread: &raylib::RaylibThread,
|
||||||
|
position: na::Vector2<f32>,
|
||||||
|
) -> Self {
|
||||||
|
// Load all the textures
|
||||||
|
let textures = vec![
|
||||||
|
load_texture_from_internal_data(
|
||||||
|
raylib_handle,
|
||||||
|
thread,
|
||||||
|
"assets/chr/chr_cubee/chr_cubeeLarge.png",
|
||||||
|
)
|
||||||
|
.unwrap(),
|
||||||
|
load_texture_from_internal_data(
|
||||||
|
raylib_handle,
|
||||||
|
thread,
|
||||||
|
"assets/chr/chr_cubee/chr_cubeeMedium.png",
|
||||||
|
)
|
||||||
|
.unwrap(),
|
||||||
|
load_texture_from_internal_data(
|
||||||
|
raylib_handle,
|
||||||
|
thread,
|
||||||
|
"assets/chr/chr_cubee/chr_cubeeSmall.png",
|
||||||
|
)
|
||||||
|
.unwrap(),
|
||||||
|
];
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
position,
|
position,
|
||||||
velocity: na::Vector2::zeros(),
|
velocity: na::Vector2::zeros(),
|
||||||
size: 1.0,
|
size: 1.0,
|
||||||
|
active_texture: 0,
|
||||||
|
textures,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use std::collections::HashMap;
|
|
||||||
use nalgebra as na;
|
use nalgebra as na;
|
||||||
use raylib::{texture::Texture2D, RaylibHandle, RaylibThread};
|
use raylib::{texture::Texture2D, RaylibHandle, RaylibThread};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
asset_manager::{load_json_structure, load_texture_from_internal_data},
|
asset_manager::{load_json_structure, load_texture_from_internal_data},
|
||||||
@ -34,6 +34,8 @@ pub struct WorldObjectPackage {
|
|||||||
pub top_animated_textures: HashMap<String, AnimatedTexture>,
|
pub top_animated_textures: HashMap<String, AnimatedTexture>,
|
||||||
/// A list of colliders in the world. We pre-solve these to make comput happy :)
|
/// A list of colliders in the world. We pre-solve these to make comput happy :)
|
||||||
pub world_space_colliders: Vec<WorldSpaceObjectCollider>,
|
pub world_space_colliders: Vec<WorldSpaceObjectCollider>,
|
||||||
|
/// A list of footprints in the world. We pre-solve these to make comput happy :)
|
||||||
|
pub world_space_footprints: Vec<WorldSpaceObjectCollider>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WorldObjectPackage {
|
impl WorldObjectPackage {
|
||||||
@ -52,6 +54,7 @@ impl WorldObjectPackage {
|
|||||||
let mut bottom_animated_textures = HashMap::new();
|
let mut bottom_animated_textures = HashMap::new();
|
||||||
let mut top_animated_textures = HashMap::new();
|
let mut top_animated_textures = HashMap::new();
|
||||||
let mut world_space_colliders: Vec<WorldSpaceObjectCollider> = Vec::new();
|
let mut world_space_colliders: Vec<WorldSpaceObjectCollider> = Vec::new();
|
||||||
|
let mut world_space_footprints: Vec<WorldSpaceObjectCollider> = Vec::new();
|
||||||
for reference in &object_references {
|
for reference in &object_references {
|
||||||
// If this is a new object, load it.
|
// If this is a new object, load it.
|
||||||
let object_key = reference.into_key();
|
let object_key = reference.into_key();
|
||||||
@ -118,6 +121,24 @@ impl WorldObjectPackage {
|
|||||||
world_space_colliders.push(world_space_collider);
|
world_space_colliders.push(world_space_collider);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Keep track of all the footprints in the world
|
||||||
|
for collider in &object_definition.footprint {
|
||||||
|
// Get the object's position
|
||||||
|
let object_position = reference.get_world_space_position();
|
||||||
|
|
||||||
|
// Convert the collider's position to world space
|
||||||
|
let mut world_space_collider = WorldSpaceObjectCollider {
|
||||||
|
position: (object_position + collider.position),
|
||||||
|
size: collider.size,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Invert the Y axis of the collider's position
|
||||||
|
world_space_collider.position.y = -world_space_collider.position.y;
|
||||||
|
|
||||||
|
// Add the collider to the list
|
||||||
|
world_space_footprints.push(world_space_collider);
|
||||||
|
}
|
||||||
|
|
||||||
// Store the object definition
|
// Store the object definition
|
||||||
object_definitions.insert(object_key.to_string(), object_definition);
|
object_definitions.insert(object_key.to_string(), object_definition);
|
||||||
}
|
}
|
||||||
@ -131,6 +152,7 @@ impl WorldObjectPackage {
|
|||||||
bottom_animated_textures,
|
bottom_animated_textures,
|
||||||
top_animated_textures,
|
top_animated_textures,
|
||||||
world_space_colliders,
|
world_space_colliders,
|
||||||
|
world_space_footprints,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,6 +43,9 @@ pub struct PlayerConstants {
|
|||||||
|
|
||||||
/// Starting size of player in tiles
|
/// Starting size of player in tiles
|
||||||
pub start_size: f32,
|
pub start_size: f32,
|
||||||
|
|
||||||
|
/// Base melting speed in percent per second
|
||||||
|
pub melt_speed: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This structure is filled with the contents of `dist/project-constants.json` at runtime
|
/// This structure is filled with the contents of `dist/project-constants.json` at runtime
|
||||||
|
@ -50,7 +50,7 @@ pub async fn handle_graphics_blocking<ConfigBuilder>(
|
|||||||
// Set up audio
|
// Set up audio
|
||||||
debug!("Set up Audio");
|
debug!("Set up Audio");
|
||||||
let audio_subsystem = RaylibAudio::init_audio_device();
|
let audio_subsystem = RaylibAudio::init_audio_device();
|
||||||
audio_subsystem.set_master_volume(0.4);
|
audio_subsystem.set_master_volume(game_settings.volume.unwrap_or(0.5));
|
||||||
|
|
||||||
// Set up the internal screens
|
// Set up the internal screens
|
||||||
let mut loading_screen = crate::rendering::screens::loading_screen::LoadingScreen::new();
|
let mut loading_screen = crate::rendering::screens::loading_screen::LoadingScreen::new();
|
||||||
|
@ -279,6 +279,9 @@ impl MapRenderer {
|
|||||||
camera,
|
camera,
|
||||||
);
|
);
|
||||||
let player_position = na::Vector2::new(player_position.x, player_position.y * -1.0);
|
let player_position = na::Vector2::new(player_position.x, player_position.y * -1.0);
|
||||||
|
// Get the tile width and height
|
||||||
|
let tile_width = 128;
|
||||||
|
let tile_height = 128;
|
||||||
|
|
||||||
// Handle each layer from the bottom up
|
// Handle each layer from the bottom up
|
||||||
for layer in self.map.layers() {
|
for layer in self.map.layers() {
|
||||||
@ -289,10 +292,6 @@ impl MapRenderer {
|
|||||||
let mut sampler_x = 0;
|
let mut sampler_x = 0;
|
||||||
let mut sampler_y = 0;
|
let mut sampler_y = 0;
|
||||||
|
|
||||||
// Get the tile width and height
|
|
||||||
let tile_width = 128;
|
|
||||||
let tile_height = 128;
|
|
||||||
|
|
||||||
// Loop until we have covered all tiles on the screen
|
// Loop until we have covered all tiles on the screen
|
||||||
for y in (world_win_top_left.y as i64)..(world_win_bottom_right.y as i64) {
|
for y in (world_win_top_left.y as i64)..(world_win_bottom_right.y as i64) {
|
||||||
// Convert the pixel coordinates to tile coordinates
|
// Convert the pixel coordinates to tile coordinates
|
||||||
@ -331,157 +330,178 @@ impl MapRenderer {
|
|||||||
Color::WHITE,
|
Color::WHITE,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tiled::LayerType::ObjectLayer(_) => todo!(),
|
||||||
|
tiled::LayerType::ImageLayer(_) => todo!(),
|
||||||
|
tiled::LayerType::GroupLayer(_) => todo!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check if there is an object at this tile
|
// Keep track of our sampler X and Y values
|
||||||
for obj_ref in &self.world_objects.object_references {
|
let mut sampler_x = 0;
|
||||||
if obj_ref.get_tile_space_position().x == sampler_x as f32
|
let mut sampler_y = 0;
|
||||||
&& obj_ref.get_tile_space_position().y
|
|
||||||
== sampler_y as f32
|
|
||||||
{
|
|
||||||
// Get access to the actual object definition
|
|
||||||
let object_key = obj_ref.into_key();
|
|
||||||
// debug!("Found object: {}", object_key);
|
|
||||||
let obj_def = self
|
|
||||||
.world_objects
|
|
||||||
.object_definitions
|
|
||||||
.get(&object_key)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
// We need to render the base layer of the object
|
// Loop until we have covered all tiles on the screen
|
||||||
if obj_def.bottom_texture.animated.unwrap_or(false) {
|
for y in (world_win_top_left.y as i64)..(world_win_bottom_right.y as i64) {
|
||||||
let tex = self
|
// Convert the pixel coordinates to tile coordinates
|
||||||
.world_objects
|
let tile_y = (y as f32 / tile_height as f32).floor() as i32;
|
||||||
.bottom_animated_textures
|
|
||||||
.get_mut(&object_key)
|
|
||||||
.unwrap();
|
|
||||||
tex.render_automatic(
|
|
||||||
draw_handle,
|
|
||||||
obj_ref.get_world_space_position()
|
|
||||||
- (tex.size() / 2.0),
|
|
||||||
None,
|
|
||||||
Some(tex.size() / 2.0),
|
|
||||||
Some(obj_ref.rotation_degrees),
|
|
||||||
None,
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
let tex = self
|
|
||||||
.world_objects
|
|
||||||
.bottom_static_textures
|
|
||||||
.get_mut(&object_key)
|
|
||||||
.unwrap();
|
|
||||||
let p: Vector2 =
|
|
||||||
obj_ref.get_world_space_position().into();
|
|
||||||
let r1 = Rectangle {
|
|
||||||
x: 0.0,
|
|
||||||
y: 0.0,
|
|
||||||
width: tex.width as f32,
|
|
||||||
height: tex.height as f32,
|
|
||||||
};
|
|
||||||
let r2 = Rectangle {
|
|
||||||
x: p.x,
|
|
||||||
y: p.y,
|
|
||||||
width: tex.width as f32,
|
|
||||||
height: tex.height as f32,
|
|
||||||
};
|
|
||||||
|
|
||||||
draw_handle.draw_texture_pro(
|
// If we are looking at a new tile, update the sampler
|
||||||
&tex,
|
if sampler_y != tile_y {
|
||||||
r1,
|
sampler_y = tile_y;
|
||||||
r2,
|
|
||||||
Vector2::new(
|
|
||||||
tex.width as f32 / 2.0,
|
|
||||||
tex.height as f32 / 2.0,
|
|
||||||
),
|
|
||||||
obj_ref.rotation_degrees,
|
|
||||||
Color::WHITE,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If needed we can render the top layer of the object
|
for x in (world_win_top_left.x as i64)..(world_win_bottom_right.x as i64) {
|
||||||
if let Some(top_texture) = &obj_def.top_texture {
|
// Convert the pixel coordinates to tile coordinates
|
||||||
// We need to detect if the player is in the footprint of the object
|
let tile_x = (x as f32 / tile_width as f32).floor() as i32;
|
||||||
let mut tint = Color::WHITE;
|
// debug!("Tile: ({}, {})", tile_x, tile_y);
|
||||||
if let Some(footprint_radius) =
|
|
||||||
obj_def.visualization_radius
|
|
||||||
{
|
|
||||||
let player_dist_to_object = (obj_ref
|
|
||||||
.get_world_space_position()
|
|
||||||
- player_position)
|
|
||||||
.norm();
|
|
||||||
// debug!(
|
|
||||||
// "Player dist to object: {}",
|
|
||||||
// player_dist_to_object
|
|
||||||
// );
|
|
||||||
if player_dist_to_object <= footprint_radius {
|
|
||||||
tint.a = 128;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if top_texture.animated.unwrap_or(false) {
|
// If we are looking at a new tile, update the sampler
|
||||||
let tex = self
|
if sampler_x != tile_x {
|
||||||
.world_objects
|
sampler_x = tile_x;
|
||||||
.top_animated_textures
|
|
||||||
.get_mut(&object_key)
|
|
||||||
.unwrap();
|
|
||||||
tex.render_automatic(
|
|
||||||
draw_handle,
|
|
||||||
obj_ref.get_world_space_position()
|
|
||||||
- (tex.size() / 2.0),
|
|
||||||
None,
|
|
||||||
Some(tex.size() / 2.0),
|
|
||||||
Some(obj_ref.rotation_degrees),
|
|
||||||
Some(tint),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
let tex = self
|
|
||||||
.world_objects
|
|
||||||
.top_static_textures
|
|
||||||
.get_mut(&object_key)
|
|
||||||
.unwrap();
|
|
||||||
let p: Vector2 =
|
|
||||||
obj_ref.get_world_space_position().into();
|
|
||||||
let r1 = Rectangle {
|
|
||||||
x: 0.0,
|
|
||||||
y: 0.0,
|
|
||||||
width: tex.width as f32,
|
|
||||||
height: tex.height as f32,
|
|
||||||
};
|
|
||||||
let r2 = Rectangle {
|
|
||||||
x: p.x,
|
|
||||||
y: p.y,
|
|
||||||
width: tex.width as f32,
|
|
||||||
height: tex.height as f32,
|
|
||||||
};
|
|
||||||
|
|
||||||
draw_handle.draw_texture_pro(
|
// Check if there is an object at this tile
|
||||||
&tex,
|
for obj_ref in &self.world_objects.object_references {
|
||||||
r1,
|
if obj_ref.get_tile_space_position().x == sampler_x as f32
|
||||||
r2,
|
&& obj_ref.get_tile_space_position().y == sampler_y as f32
|
||||||
Vector2::new(
|
{
|
||||||
tex.width as f32 / 2.0,
|
// Get access to the actual object definition
|
||||||
tex.height as f32 / 2.0,
|
let object_key = obj_ref.into_key();
|
||||||
),
|
// debug!("Found object: {}", object_key);
|
||||||
obj_ref.rotation_degrees,
|
let obj_def = self
|
||||||
tint,
|
.world_objects
|
||||||
);
|
.object_definitions
|
||||||
}
|
.get(&object_key)
|
||||||
}
|
.unwrap();
|
||||||
|
|
||||||
|
// We need to render the base layer of the object
|
||||||
|
if obj_def.bottom_texture.animated.unwrap_or(false) {
|
||||||
|
let tex = self
|
||||||
|
.world_objects
|
||||||
|
.bottom_animated_textures
|
||||||
|
.get_mut(&object_key)
|
||||||
|
.unwrap();
|
||||||
|
tex.render_automatic(
|
||||||
|
draw_handle,
|
||||||
|
obj_ref.get_world_space_position() - (tex.size() / 2.0),
|
||||||
|
None,
|
||||||
|
Some(tex.size() / 2.0),
|
||||||
|
Some(obj_ref.rotation_degrees),
|
||||||
|
None,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
let tex = self
|
||||||
|
.world_objects
|
||||||
|
.bottom_static_textures
|
||||||
|
.get_mut(&object_key)
|
||||||
|
.unwrap();
|
||||||
|
let p: Vector2 = obj_ref.get_world_space_position().into();
|
||||||
|
let r1 = Rectangle {
|
||||||
|
x: 0.0,
|
||||||
|
y: 0.0,
|
||||||
|
width: tex.width as f32,
|
||||||
|
height: tex.height as f32,
|
||||||
|
};
|
||||||
|
let r2 = Rectangle {
|
||||||
|
x: p.x,
|
||||||
|
y: p.y,
|
||||||
|
width: tex.width as f32,
|
||||||
|
height: tex.height as f32,
|
||||||
|
};
|
||||||
|
|
||||||
|
draw_handle.draw_texture_pro(
|
||||||
|
&tex,
|
||||||
|
r1,
|
||||||
|
r2,
|
||||||
|
Vector2::new(
|
||||||
|
tex.width as f32 / 2.0,
|
||||||
|
tex.height as f32 / 2.0,
|
||||||
|
),
|
||||||
|
obj_ref.rotation_degrees,
|
||||||
|
Color::WHITE,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If needed we can render the top layer of the object
|
||||||
|
if let Some(top_texture) = &obj_def.top_texture {
|
||||||
|
// We need to detect if the player is in the footprint of the object
|
||||||
|
let mut tint = Color::WHITE;
|
||||||
|
if let Some(footprint_radius) = obj_def.visualization_radius {
|
||||||
|
let player_dist_to_object =
|
||||||
|
(obj_ref.get_world_space_position() - player_position)
|
||||||
|
.norm();
|
||||||
|
// debug!(
|
||||||
|
// "Player dist to object: {}",
|
||||||
|
// player_dist_to_object
|
||||||
|
// );
|
||||||
|
if player_dist_to_object <= footprint_radius {
|
||||||
|
tint.a = 128;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if show_debug_grid {
|
if top_texture.animated.unwrap_or(false) {
|
||||||
draw_handle.draw_rectangle_lines(
|
let tex = self
|
||||||
tile_x * tile_width as i32,
|
.world_objects
|
||||||
tile_y * tile_height as i32,
|
.top_animated_textures
|
||||||
self.map.tile_width as i32,
|
.get_mut(&object_key)
|
||||||
self.map.tile_height as i32,
|
.unwrap();
|
||||||
Color::RED,
|
tex.render_automatic(
|
||||||
|
draw_handle,
|
||||||
|
obj_ref.get_world_space_position() - (tex.size() / 2.0),
|
||||||
|
None,
|
||||||
|
Some(tex.size() / 2.0),
|
||||||
|
Some(obj_ref.rotation_degrees),
|
||||||
|
Some(tint),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
let tex = self
|
||||||
|
.world_objects
|
||||||
|
.top_static_textures
|
||||||
|
.get_mut(&object_key)
|
||||||
|
.unwrap();
|
||||||
|
let p: Vector2 = obj_ref.get_world_space_position().into();
|
||||||
|
let r1 = Rectangle {
|
||||||
|
x: 0.0,
|
||||||
|
y: 0.0,
|
||||||
|
width: tex.width as f32,
|
||||||
|
height: tex.height as f32,
|
||||||
|
};
|
||||||
|
let r2 = Rectangle {
|
||||||
|
x: p.x,
|
||||||
|
y: p.y,
|
||||||
|
width: tex.width as f32,
|
||||||
|
height: tex.height as f32,
|
||||||
|
};
|
||||||
|
|
||||||
|
draw_handle.draw_texture_pro(
|
||||||
|
&tex,
|
||||||
|
r1,
|
||||||
|
r2,
|
||||||
|
Vector2::new(
|
||||||
|
tex.width as f32 / 2.0,
|
||||||
|
tex.height as f32 / 2.0,
|
||||||
|
),
|
||||||
|
obj_ref.rotation_degrees,
|
||||||
|
tint,
|
||||||
);
|
);
|
||||||
draw_handle.draw_pixel(x as i32, y as i32, Color::BLUE);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if show_debug_grid {
|
||||||
|
draw_handle.draw_rectangle_lines(
|
||||||
|
tile_x * tile_width as i32,
|
||||||
|
tile_y * tile_height as i32,
|
||||||
|
self.map.tile_width as i32,
|
||||||
|
self.map.tile_height as i32,
|
||||||
|
Color::RED,
|
||||||
|
);
|
||||||
|
draw_handle.draw_pixel(x as i32, y as i32, Color::BLUE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// for collider in &self.world_objects.world_space_colliders {
|
// for collider in &self.world_objects.world_space_colliders {
|
||||||
// draw_handle.draw_rectangle_lines(
|
// draw_handle.draw_rectangle_lines(
|
||||||
@ -493,9 +513,6 @@ impl MapRenderer {
|
|||||||
// );
|
// );
|
||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
tiled::LayerType::ObjectLayer(_) => todo!(),
|
|
||||||
tiled::LayerType::ImageLayer(_) => todo!(),
|
|
||||||
tiled::LayerType::GroupLayer(_) => todo!(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ use raylib::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
asset_manager::load_texture_from_internal_data,
|
||||||
discord::{DiscordChannel, DiscordRpcSignal},
|
discord::{DiscordChannel, DiscordRpcSignal},
|
||||||
global_resource_package::GlobalResources,
|
global_resource_package::GlobalResources,
|
||||||
persistent::settings::PersistentGameSettings,
|
persistent::settings::PersistentGameSettings,
|
||||||
@ -17,9 +18,24 @@ use crate::{
|
|||||||
|
|
||||||
use super::main_menu::MenuStateSignal;
|
use super::main_menu::MenuStateSignal;
|
||||||
|
|
||||||
|
const MIWU_WHITE: Color = Color {
|
||||||
|
r: 247,
|
||||||
|
g: 239,
|
||||||
|
b: 231,
|
||||||
|
a: 255,
|
||||||
|
};
|
||||||
|
const MIWU_WHITE_V2: Color = Color {
|
||||||
|
r: 255,
|
||||||
|
g: 245,
|
||||||
|
b: 228,
|
||||||
|
a: 255,
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct CutScenes {
|
pub struct CutScenes {
|
||||||
show_debug_info: bool,
|
show_debug_info: bool,
|
||||||
|
intro_art: Texture2D,
|
||||||
|
melted_art: Texture2D,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CutScenes {
|
impl CutScenes {
|
||||||
@ -30,8 +46,24 @@ impl CutScenes {
|
|||||||
constants: &ProjectConstants,
|
constants: &ProjectConstants,
|
||||||
game_settings: &mut PersistentGameSettings,
|
game_settings: &mut PersistentGameSettings,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
|
// Load art
|
||||||
|
let intro_art = load_texture_from_internal_data(
|
||||||
|
raylib_handle,
|
||||||
|
thread,
|
||||||
|
"assets/cut/cut_intro/cut_intro.png",
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
let melted_art = load_texture_from_internal_data(
|
||||||
|
raylib_handle,
|
||||||
|
thread,
|
||||||
|
"assets/cut/cut_melty/cut_melty.png",
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
show_debug_info: false,
|
show_debug_info: false,
|
||||||
|
intro_art,
|
||||||
|
melted_art,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -48,7 +80,7 @@ impl CutScenes {
|
|||||||
let mut draw = raylib.begin_drawing(rl_thread);
|
let mut draw = raylib.begin_drawing(rl_thread);
|
||||||
|
|
||||||
// Clear the screen
|
// Clear the screen
|
||||||
draw.clear_background(Color::WHITE);
|
draw.clear_background(MIWU_WHITE);
|
||||||
|
|
||||||
//Obtain mouse position
|
//Obtain mouse position
|
||||||
let mouse_x = draw.get_mouse_x();
|
let mouse_x = draw.get_mouse_x();
|
||||||
@ -71,8 +103,40 @@ impl CutScenes {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Title
|
// Title
|
||||||
draw.draw_text("INTRO CUTSCENE GOES HERE", 100, 90, 60, Color::BLACK);
|
// draw.draw_text("INTRO CUTSCENE GOES HERE", 100, 90, 60, Color::BLACK);
|
||||||
draw.draw_text("Press SPACE to skip", 100, 600, 20, Color::BLACK);
|
// draw.draw_text("Press SPACE to skip", 100, 600, 20, Color::BLACK);
|
||||||
|
|
||||||
|
let screen_height = draw.get_screen_height();
|
||||||
|
let screen_width = draw.get_screen_width();
|
||||||
|
|
||||||
|
// Build a rect for the texture
|
||||||
|
let tex_rect = Rectangle::new(
|
||||||
|
0.0,
|
||||||
|
0.0,
|
||||||
|
self.intro_art.width as f32,
|
||||||
|
self.intro_art.height as f32,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Draw the texture to the center of the screen.
|
||||||
|
// Keep in mind, textures are drawn from the top left
|
||||||
|
// corner, so we need to offset the rect by half the
|
||||||
|
// texture's width and height.
|
||||||
|
let dest_rect = Rectangle::new(
|
||||||
|
(screen_width / 2) as f32 - (tex_rect.width / 2.0),
|
||||||
|
(screen_height / 2) as f32 - (tex_rect.height / 2.0),
|
||||||
|
tex_rect.width,
|
||||||
|
tex_rect.height,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Draw the texture
|
||||||
|
draw.draw_texture_pro(
|
||||||
|
&self.intro_art,
|
||||||
|
&tex_rect,
|
||||||
|
&dest_rect,
|
||||||
|
Vector2::zero(),
|
||||||
|
0.0,
|
||||||
|
Color::WHITE,
|
||||||
|
);
|
||||||
|
|
||||||
// Let the user leave this cutscene by pressing space
|
// Let the user leave this cutscene by pressing space
|
||||||
if draw.is_key_pressed(KeyboardKey::KEY_SPACE) {
|
if draw.is_key_pressed(KeyboardKey::KEY_SPACE) {
|
||||||
@ -99,7 +163,7 @@ impl CutScenes {
|
|||||||
let mut draw = raylib.begin_drawing(rl_thread);
|
let mut draw = raylib.begin_drawing(rl_thread);
|
||||||
|
|
||||||
// Clear the screen
|
// Clear the screen
|
||||||
draw.clear_background(Color::WHITE);
|
draw.clear_background(MIWU_WHITE_V2);
|
||||||
|
|
||||||
//Obtain mouse position
|
//Obtain mouse position
|
||||||
let mouse_x = draw.get_mouse_x();
|
let mouse_x = draw.get_mouse_x();
|
||||||
@ -121,16 +185,48 @@ impl CutScenes {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Title
|
// // Title
|
||||||
draw.draw_text("MELTY CUTSCENE GOES HERE", 100, 90, 60, Color::BLACK);
|
// draw.draw_text("MELTY CUTSCENE GOES HERE", 100, 90, 60, Color::BLACK);
|
||||||
draw.draw_text(
|
// draw.draw_text(
|
||||||
&format!("This took you {} seconds", playtime.num_seconds()),
|
// &format!("This took you {} seconds", playtime.num_seconds()),
|
||||||
100,
|
// 100,
|
||||||
600,
|
// 600,
|
||||||
20,
|
// 20,
|
||||||
Color::BLACK,
|
// Color::BLACK,
|
||||||
|
// );
|
||||||
|
// draw.draw_text("Press SPACE to skip", 100, 680, 20, Color::BLACK);
|
||||||
|
|
||||||
|
let screen_height = draw.get_screen_height();
|
||||||
|
let screen_width = draw.get_screen_width();
|
||||||
|
|
||||||
|
// Build a rect for the texture
|
||||||
|
let tex_rect = Rectangle::new(
|
||||||
|
0.0,
|
||||||
|
0.0,
|
||||||
|
self.melted_art.width as f32,
|
||||||
|
self.melted_art.height as f32,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Draw the texture to the center of the screen.
|
||||||
|
// Keep in mind, textures are drawn from the top left
|
||||||
|
// corner, so we need to offset the rect by half the
|
||||||
|
// texture's width and height.
|
||||||
|
let dest_rect = Rectangle::new(
|
||||||
|
(screen_width / 2) as f32 - (tex_rect.width / 2.0),
|
||||||
|
(screen_height / 2) as f32 - (tex_rect.height / 2.0),
|
||||||
|
tex_rect.width,
|
||||||
|
tex_rect.height,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Draw the texture
|
||||||
|
draw.draw_texture_pro(
|
||||||
|
&self.melted_art,
|
||||||
|
&tex_rect,
|
||||||
|
&dest_rect,
|
||||||
|
Vector2::zero(),
|
||||||
|
0.0,
|
||||||
|
Color::WHITE,
|
||||||
);
|
);
|
||||||
draw.draw_text("Press SPACE to skip", 100, 680, 20, Color::BLACK);
|
|
||||||
|
|
||||||
// Let the user leave this cutscene by pressing space
|
// Let the user leave this cutscene by pressing space
|
||||||
if draw.is_key_pressed(KeyboardKey::KEY_SPACE) {
|
if draw.is_key_pressed(KeyboardKey::KEY_SPACE) {
|
||||||
@ -158,7 +254,7 @@ impl CutScenes {
|
|||||||
let mut draw = raylib.begin_drawing(rl_thread);
|
let mut draw = raylib.begin_drawing(rl_thread);
|
||||||
|
|
||||||
// Clear the screen
|
// Clear the screen
|
||||||
draw.clear_background(Color::WHITE);
|
draw.clear_background(MIWU_WHITE);
|
||||||
|
|
||||||
//Obtain mouse position
|
//Obtain mouse position
|
||||||
let mouse_x = draw.get_mouse_x();
|
let mouse_x = draw.get_mouse_x();
|
||||||
@ -217,7 +313,7 @@ impl CutScenes {
|
|||||||
let mut draw = raylib.begin_drawing(rl_thread);
|
let mut draw = raylib.begin_drawing(rl_thread);
|
||||||
|
|
||||||
// Clear the screen
|
// Clear the screen
|
||||||
draw.clear_background(Color::WHITE);
|
draw.clear_background(MIWU_WHITE);
|
||||||
|
|
||||||
//Obtain mouse position
|
//Obtain mouse position
|
||||||
let mouse_x = draw.get_mouse_x();
|
let mouse_x = draw.get_mouse_x();
|
||||||
|
@ -60,18 +60,25 @@ impl PlayableScene {
|
|||||||
let game_soundtrack =
|
let game_soundtrack =
|
||||||
load_music_from_internal_data(thread, "assets/audio/gameSoundtrack.mp3").unwrap();
|
load_music_from_internal_data(thread, "assets/audio/gameSoundtrack.mp3").unwrap();
|
||||||
|
|
||||||
|
// Load the player
|
||||||
|
let player = Player::new(
|
||||||
|
raylib_handle,
|
||||||
|
thread,
|
||||||
|
player_start_position,
|
||||||
|
);
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
has_updated_discord_rpc: false,
|
has_updated_discord_rpc: false,
|
||||||
player_start_position,
|
player_start_position,
|
||||||
player: Player::new(player_start_position),
|
|
||||||
world_map: map_renderer,
|
world_map: map_renderer,
|
||||||
|
player,
|
||||||
camera: raylib::camera::Camera2D {
|
camera: raylib::camera::Camera2D {
|
||||||
target: raylib::math::Vector2 { x: 0.0, y: 0.0 },
|
target: raylib::math::Vector2 { x: 0.0, y: 0.0 },
|
||||||
offset: raylib::math::Vector2 { x: 0.0, y: 0.0 },
|
offset: raylib::math::Vector2 { x: 0.0, y: 0.0 },
|
||||||
rotation: 0.0,
|
rotation: 0.0,
|
||||||
zoom: 1.0,
|
zoom: 1.0,
|
||||||
},
|
},
|
||||||
last_update: SystemTime::UNIX_EPOCH,
|
last_update: SystemTime::now(),
|
||||||
game_soundtrack,
|
game_soundtrack,
|
||||||
world_colliders,
|
world_colliders,
|
||||||
show_debug_info: false,
|
show_debug_info: false,
|
||||||
@ -127,13 +134,18 @@ impl PlayableScene {
|
|||||||
self.draw_ui(&mut draw, constants);
|
self.draw_ui(&mut draw, constants);
|
||||||
|
|
||||||
// NOTE: If you want to trigger a cutscene, do it here by using one of:
|
// NOTE: If you want to trigger a cutscene, do it here by using one of:
|
||||||
// return MenuStateSignal::DoMeltedDeathCutscene {
|
|
||||||
// playtime: Utc::now().signed_duration_since(self.play_start_time),
|
|
||||||
// };
|
|
||||||
// return MenuStateSignal::DoOceanCutscene {
|
// return MenuStateSignal::DoOceanCutscene {
|
||||||
// playtime: Utc::now().signed_duration_since(self.play_start_time),
|
// playtime: Utc::now().signed_duration_since(self.play_start_time),
|
||||||
// };
|
// };
|
||||||
|
|
||||||
|
// Handle Losing
|
||||||
|
if self.player.size < 0.15 {
|
||||||
|
return MenuStateSignal::DoMeltedDeathCutscene {
|
||||||
|
playtime: Utc::now().signed_duration_since(self.play_start_time),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
// Handle winning
|
// Handle winning
|
||||||
if self
|
if self
|
||||||
.world_map
|
.world_map
|
||||||
@ -181,6 +193,14 @@ impl PlayableScene {
|
|||||||
let mouse_x = draw.get_mouse_x();
|
let mouse_x = draw.get_mouse_x();
|
||||||
let mouse_y = draw.get_mouse_y();
|
let mouse_y = draw.get_mouse_y();
|
||||||
|
|
||||||
|
let current_temperature = self.world_map.sample_temperature_at(
|
||||||
|
self.player.position.component_mul(&na::Vector2::new(1.0, -1.0))
|
||||||
|
);
|
||||||
|
let mut current_temperature_val: f32 = -247.51879;
|
||||||
|
if let Some(val) = current_temperature {
|
||||||
|
current_temperature_val = val;
|
||||||
|
}
|
||||||
|
|
||||||
// Optionally display debug info
|
// Optionally display debug info
|
||||||
if draw.is_key_pressed(KeyboardKey::KEY_F3) {
|
if draw.is_key_pressed(KeyboardKey::KEY_F3) {
|
||||||
self.show_debug_info = !self.show_debug_info;
|
self.show_debug_info = !self.show_debug_info;
|
||||||
@ -224,6 +244,12 @@ impl PlayableScene {
|
|||||||
// 32,
|
// 32,
|
||||||
// Color::BLACK,
|
// Color::BLACK,
|
||||||
// );
|
// );
|
||||||
|
let melt_amount = (current_temperature_val)/(-247.51879);
|
||||||
|
|
||||||
|
draw.draw_text(
|
||||||
|
format!("Funny Temperature: ({})[{}]", current_temperature_val, melt_amount).as_str(),
|
||||||
|
10, 10, 20, Color::PAPAYAWHIP
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Physics
|
// Physics
|
||||||
@ -242,11 +268,11 @@ impl PlayableScene {
|
|||||||
|
|
||||||
let player = &mut self.player;
|
let player = &mut self.player;
|
||||||
|
|
||||||
// NOTE: This is how to check friction and temperature
|
|
||||||
let current_friction = self.world_map.sample_friction_at(player.position);
|
let current_friction = self.world_map.sample_friction_at(player.position);
|
||||||
let current_temperature = self.world_map.sample_temperature_at(player.position);
|
let current_temperature = self.world_map.sample_temperature_at(
|
||||||
|
player.position.component_mul(&na::Vector2::new(1.0, -1.0))
|
||||||
|
);
|
||||||
let map_size = self.world_map.get_map_size();
|
let map_size = self.world_map.get_map_size();
|
||||||
// TODO: You can access the colission list with: self.world_colliders
|
|
||||||
|
|
||||||
// Get input direction components
|
// Get input direction components
|
||||||
let h_axis = raylib.is_key_down(KeyboardKey::KEY_D) as i8
|
let h_axis = raylib.is_key_down(KeyboardKey::KEY_D) as i8
|
||||||
@ -347,6 +373,15 @@ impl PlayableScene {
|
|||||||
player.velocity.y = 0.0;
|
player.velocity.y = 0.0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mut current_temperature_val: f32 = -247.51879;
|
||||||
|
if let Some(val) = current_temperature {
|
||||||
|
current_temperature_val = val - 273.15;
|
||||||
|
}
|
||||||
|
|
||||||
|
let melt_amount = constants.player.melt_speed * (current_temperature_val)/(-247.51879);
|
||||||
|
|
||||||
|
player.size -= melt_amount * delta_time;
|
||||||
|
|
||||||
self.update_camera(raylib);
|
self.update_camera(raylib);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user