No longer living in borrow checker hell for gfx
This commit is contained in:
parent
36f0744e31
commit
d2372772f0
@ -1,14 +1,16 @@
|
|||||||
use raylib::{RaylibHandle, prelude::{RaylibDrawHandle, RaylibMode2D}};
|
use raylib::{RaylibHandle, prelude::{RaylibDrawHandle, RaylibMode2D}};
|
||||||
|
|
||||||
|
use crate::utilities::non_ref_raylib::HackedRaylibHandle;
|
||||||
|
|
||||||
|
|
||||||
pub trait FrameUpdate {
|
pub trait FrameUpdate {
|
||||||
fn update(&mut self, raylib: &RaylibHandle, delta_seconds: f32);
|
fn update(&mut self, raylib: &RaylibHandle, delta_seconds: f32);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait ScreenSpaceRender {
|
pub trait ScreenSpaceRender {
|
||||||
fn render_screen_space(&self, raylib: &mut RaylibDrawHandle);
|
fn render_screen_space(&self, raylib: &mut HackedRaylibHandle);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait WorldSpaceRender {
|
pub trait WorldSpaceRender {
|
||||||
fn render_world_space(&self, raylib: &mut RaylibMode2D<RaylibDrawHandle>);
|
fn render_world_space(&self, raylib: &mut RaylibMode2D<HackedRaylibHandle>);
|
||||||
}
|
}
|
||||||
|
110
game/src/lib.rs
110
game/src/lib.rs
@ -1,6 +1,12 @@
|
|||||||
#![feature(derive_default_enum)]
|
#![feature(derive_default_enum)]
|
||||||
|
|
||||||
use std::cell::RefCell;
|
use std::{
|
||||||
|
borrow::BorrowMut,
|
||||||
|
cell::{Cell, RefCell},
|
||||||
|
ops::Deref,
|
||||||
|
rc::Rc,
|
||||||
|
sync::Arc,
|
||||||
|
};
|
||||||
|
|
||||||
use discord_sdk::activity::ActivityBuilder;
|
use discord_sdk::activity::ActivityBuilder;
|
||||||
use raylib::prelude::*;
|
use raylib::prelude::*;
|
||||||
@ -14,10 +20,13 @@ use utilities::{
|
|||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
context::GameContext,
|
context::GameContext,
|
||||||
scenes::{build_screen_state_machine, RenderContext},
|
scenes::build_screen_state_machine,
|
||||||
utilities::shaders::{
|
utilities::{
|
||||||
shader::ShaderWrapper,
|
non_ref_raylib::HackedRaylibHandle,
|
||||||
util::{dynamic_screen_texture::DynScreenTexture, render_texture::render_to_texture},
|
shaders::{
|
||||||
|
shader::ShaderWrapper,
|
||||||
|
util::{dynamic_screen_texture::DynScreenTexture, render_texture::render_to_texture},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -73,21 +82,28 @@ pub async fn game_begin() {
|
|||||||
build_screen_state_machine().expect("Could not init state main state machine");
|
build_screen_state_machine().expect("Could not init state main state machine");
|
||||||
|
|
||||||
// Build the game context
|
// Build the game context
|
||||||
let mut context = RefCell::new(GameContext::new());
|
let mut context = Rc::new(RefCell::new(GameContext::new()));
|
||||||
|
|
||||||
let (mut rl, thread) = raylib::init()
|
let mut raylib_handle: RefCell<HackedRaylibHandle>;
|
||||||
.size(640, 480)
|
let raylib_thread;
|
||||||
.title(&game_config.name)
|
{
|
||||||
.vsync()
|
let (mut rl, thread) = raylib::init()
|
||||||
.msaa_4x()
|
.size(640, 480)
|
||||||
.resizable()
|
.title(&game_config.name)
|
||||||
.build();
|
.vsync()
|
||||||
rl.set_exit_key(None);
|
.msaa_4x()
|
||||||
|
.resizable()
|
||||||
|
.build();
|
||||||
|
rl.set_exit_key(None);
|
||||||
|
raylib_handle = RefCell::new(rl.into());
|
||||||
|
raylib_thread = thread;
|
||||||
|
}
|
||||||
|
|
||||||
// Create a dynamic texture to draw to for processing by shaders
|
// Create a dynamic texture to draw to for processing by shaders
|
||||||
info!("Allocating a resizable texture for the screen");
|
info!("Allocating a resizable texture for the screen");
|
||||||
let mut dynamic_texture =
|
let mut dynamic_texture =
|
||||||
DynScreenTexture::new(&mut rl, &thread).expect("Failed to allocate a screen texture");
|
DynScreenTexture::new(&mut raylib_handle.borrow_mut(), &raylib_thread)
|
||||||
|
.expect("Failed to allocate a screen texture");
|
||||||
|
|
||||||
// Load the pixel art shader
|
// Load the pixel art shader
|
||||||
info!("Loading the pixel art shader");
|
info!("Loading the pixel art shader");
|
||||||
@ -95,51 +111,63 @@ pub async fn game_begin() {
|
|||||||
None,
|
None,
|
||||||
Some(StaticGameData::get("shaders/pixelart.fs")).expect("Failed to load pixelart.fs"),
|
Some(StaticGameData::get("shaders/pixelart.fs")).expect("Failed to load pixelart.fs"),
|
||||||
vec!["viewport"],
|
vec!["viewport"],
|
||||||
&mut rl,
|
&mut raylib_handle.borrow_mut(),
|
||||||
&thread,
|
&raylib_thread,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
info!("Starting the render loop");
|
info!("Starting the render loop");
|
||||||
while !rl.window_should_close() {
|
while !raylib_handle.borrow().window_should_close() {
|
||||||
// Profile the main game loop
|
// Profile the main game loop
|
||||||
puffin::profile_scope!("main_loop");
|
puffin::profile_scope!("main_loop");
|
||||||
puffin::GlobalProfiler::lock().new_frame();
|
puffin::GlobalProfiler::lock().new_frame();
|
||||||
|
|
||||||
// Update the GPU texture that we draw to. This handles screen resizing and some other stuff
|
// Update the GPU texture that we draw to. This handles screen resizing and some other stuff
|
||||||
dynamic_texture.update(&mut rl, &thread).unwrap();
|
dynamic_texture
|
||||||
|
.update(&mut raylib_handle.borrow_mut(), &raylib_thread)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
// Switch into draw mode
|
// Switch into draw mode (using unsafe code here to avoid borrow checker hell)
|
||||||
let mut d = rl.begin_drawing(&thread);
|
unsafe {
|
||||||
|
raylib::ffi::BeginDrawing();
|
||||||
|
}
|
||||||
|
// let mut d = rl.begin_drawing(&thread);
|
||||||
|
|
||||||
// Fetch the screen size once to work with in render code
|
// Fetch the screen size once to work with in render code
|
||||||
let screen_size = Vector2::new(d.get_screen_width() as f32, d.get_screen_height() as f32);
|
let screen_size = Vector2::new(
|
||||||
|
raylib_handle.borrow().get_screen_width() as f32,
|
||||||
|
raylib_handle.borrow().get_screen_height() as f32,
|
||||||
|
);
|
||||||
|
|
||||||
// Update the pixel shader to correctly handle the screen size
|
// Update the pixel shader to correctly handle the screen size
|
||||||
pixel_shader.set_variable("viewport", screen_size).unwrap();
|
pixel_shader.set_variable("viewport", screen_size).unwrap();
|
||||||
|
|
||||||
// Build render context
|
// Render the game via the pixel shader
|
||||||
{
|
render_to_texture(&mut dynamic_texture, || {
|
||||||
let render_ctx = RefCell::new((RefCell::new(d), context));
|
// Profile the internal render code
|
||||||
|
puffin::profile_scope!("internal_shaded_render");
|
||||||
|
|
||||||
// Render the game via the pixel shader
|
// Run a state machine iteration
|
||||||
render_to_texture(&mut dynamic_texture, || {
|
let result = game_state_machine.run(&raylib_handle);
|
||||||
// Profile the internal render code
|
|
||||||
puffin::profile_scope!("internal_shaded_render");
|
|
||||||
|
|
||||||
// Run a state machine iteration
|
if let Err(err) = result {
|
||||||
let result = game_state_machine.run(&render_ctx);
|
error!("Main state machine encountered an error while running!");
|
||||||
|
error!("Main thread crash!!");
|
||||||
if let Err(err) = result {
|
error!("Cannot recover from error");
|
||||||
error!("Main state machine encountered an error while running!");
|
panic!("{:?}", err);
|
||||||
error!("Main thread crash!!");
|
}
|
||||||
error!("Cannot recover from error");
|
});
|
||||||
panic!("{:?}", err);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Send the texture to the GPU to be drawn
|
// Send the texture to the GPU to be drawn
|
||||||
pixel_shader.process_texture_and_render(&mut d, &thread, &dynamic_texture);
|
pixel_shader.process_texture_and_render(
|
||||||
|
&mut raylib_handle.borrow_mut(),
|
||||||
|
&raylib_thread,
|
||||||
|
&dynamic_texture,
|
||||||
|
);
|
||||||
|
|
||||||
|
// We MUST end draw mode
|
||||||
|
unsafe {
|
||||||
|
raylib::ffi::EndDrawing();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,15 +1,12 @@
|
|||||||
use std::cell::RefCell;
|
use std::cell::{Cell, RefCell};
|
||||||
|
|
||||||
use dirty_fsm::{Action, ActionFlag};
|
use dirty_fsm::{Action, ActionFlag};
|
||||||
use raylib::{
|
use raylib::{color::Color, prelude::RaylibDraw, RaylibHandle};
|
||||||
color::Color,
|
|
||||||
prelude::{RaylibDraw, RaylibDrawHandle},
|
|
||||||
};
|
|
||||||
use tracing::{debug, error, info, trace};
|
use tracing::{debug, error, info, trace};
|
||||||
|
|
||||||
use crate::{context::GameContext, gfx::render_layer::ScreenSpaceRender};
|
use crate::{gfx::render_layer::ScreenSpaceRender, utilities::non_ref_raylib::HackedRaylibHandle};
|
||||||
|
|
||||||
use super::{RenderContext, Scenes, ScreenError};
|
use super::{Scenes, ScreenError};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct FsmErrorScreen {}
|
pub struct FsmErrorScreen {}
|
||||||
@ -21,16 +18,13 @@ impl FsmErrorScreen {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Rl> Action<Scenes, ScreenError, RefCell<(RefCell<Rl>, RefCell<GameContext>)>> for FsmErrorScreen
|
impl Action<Scenes, ScreenError, RefCell<HackedRaylibHandle>> for FsmErrorScreen {
|
||||||
where
|
|
||||||
Rl: RaylibDraw,
|
|
||||||
{
|
|
||||||
fn on_register(&mut self) -> Result<(), ScreenError> {
|
fn on_register(&mut self) -> Result<(), ScreenError> {
|
||||||
debug!("Registered");
|
debug!("Registered");
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn on_first_run(&mut self, context: &RefCell<(RefCell<Rl>, RefCell<GameContext>)>) -> Result<(), ScreenError> {
|
fn on_first_run(&mut self, context: &RefCell<HackedRaylibHandle>) -> Result<(), ScreenError> {
|
||||||
debug!("Running FsmErrorScreen for the first time");
|
debug!("Running FsmErrorScreen for the first time");
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -38,9 +32,10 @@ where
|
|||||||
fn execute(
|
fn execute(
|
||||||
&mut self,
|
&mut self,
|
||||||
delta: &chrono::Duration,
|
delta: &chrono::Duration,
|
||||||
context: &RefCell<(RefCell<Rl>, RefCell<GameContext>)>,
|
context: &RefCell<HackedRaylibHandle>,
|
||||||
) -> Result<dirty_fsm::ActionFlag<Scenes>, ScreenError> {
|
) -> Result<dirty_fsm::ActionFlag<Scenes>, ScreenError> {
|
||||||
trace!("execute() called on FsmErrorScreen, but we have not logic");
|
trace!("execute() called on FsmErrorScreen");
|
||||||
|
self.render_screen_space(&mut context.borrow_mut());
|
||||||
Ok(ActionFlag::Continue)
|
Ok(ActionFlag::Continue)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -51,7 +46,7 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl ScreenSpaceRender for FsmErrorScreen {
|
impl ScreenSpaceRender for FsmErrorScreen {
|
||||||
fn render_screen_space(&self, raylib: &mut raylib::prelude::RaylibDrawHandle) {
|
fn render_screen_space(&self, raylib: &mut HackedRaylibHandle) {
|
||||||
raylib.clear_background(Color::RED);
|
raylib.clear_background(Color::RED);
|
||||||
|
|
||||||
// Render a warning message
|
// Render a warning message
|
||||||
|
@ -1,20 +1,24 @@
|
|||||||
use std::cell::{RefCell, RefMut};
|
use std::{
|
||||||
|
borrow::Borrow,
|
||||||
|
cell::{Cell, RefCell, RefMut},
|
||||||
|
rc::Rc,
|
||||||
|
};
|
||||||
|
|
||||||
use dirty_fsm::{Action, StateMachine};
|
use dirty_fsm::{Action, StateMachine};
|
||||||
use raylib::{RaylibHandle, prelude::{RaylibDraw, RaylibDrawHandle}};
|
use raylib::{
|
||||||
|
prelude::{RaylibDraw, RaylibDrawHandle},
|
||||||
use crate::{
|
RaylibHandle,
|
||||||
context::GameContext,
|
|
||||||
gfx::render_layer::{FrameUpdate, ScreenSpaceRender, WorldSpaceRender},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::{context::GameContext, gfx::render_layer::{FrameUpdate, ScreenSpaceRender, WorldSpaceRender}, utilities::non_ref_raylib::HackedRaylibHandle};
|
||||||
|
|
||||||
use self::fsm_error_screen::FsmErrorScreen;
|
use self::fsm_error_screen::FsmErrorScreen;
|
||||||
|
|
||||||
pub mod fsm_error_screen;
|
pub mod fsm_error_screen;
|
||||||
// pub mod loading_screen;
|
// pub mod loading_screen;
|
||||||
|
|
||||||
/// Data passed to all scenes upon render
|
/// Data passed to all scenes upon render
|
||||||
pub type RenderContext<'a, 'b> = (&'b mut RaylibDrawHandle<'a>, &'b mut GameContext);
|
// pub type RenderContext<'a, 'b> = (&'b mut RaylibDrawHandle<'a>, &'b mut GameContext);
|
||||||
|
|
||||||
/// Defines all scenes
|
/// Defines all scenes
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default, Hash)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default, Hash)]
|
||||||
@ -29,8 +33,11 @@ pub enum Scenes {
|
|||||||
pub enum ScreenError {}
|
pub enum ScreenError {}
|
||||||
|
|
||||||
/// Build the state machine for all scenes
|
/// Build the state machine for all scenes
|
||||||
pub fn build_screen_state_machine<Rl>(
|
pub fn build_screen_state_machine() -> Result<
|
||||||
) -> Result<StateMachine<Scenes, ScreenError, RefCell<(RefCell<Rl>, RefCell<GameContext>)>>, ScreenError> where Rl: RaylibDraw {
|
// StateMachine<Scenes, ScreenError, RefCell<(NonRefDrawHandle, Rc<RefCell<GameContext>>)>>,
|
||||||
|
StateMachine<Scenes, ScreenError, RefCell<HackedRaylibHandle>>,
|
||||||
|
ScreenError,
|
||||||
|
> {
|
||||||
let mut machine = StateMachine::new();
|
let mut machine = StateMachine::new();
|
||||||
machine.add_action(Scenes::FsmErrorScreen, FsmErrorScreen::new())?;
|
machine.add_action(Scenes::FsmErrorScreen, FsmErrorScreen::new())?;
|
||||||
Ok(machine)
|
Ok(machine)
|
||||||
|
@ -3,3 +3,4 @@ pub mod datastore;
|
|||||||
pub mod game_config;
|
pub mod game_config;
|
||||||
pub mod math;
|
pub mod math;
|
||||||
pub mod shaders;
|
pub mod shaders;
|
||||||
|
pub mod non_ref_raylib;
|
||||||
|
27
game/src/utilities/non_ref_raylib.rs
Normal file
27
game/src/utilities/non_ref_raylib.rs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
use std::{borrow::Borrow, cell::{Cell, RefCell, RefMut}, ops::{Deref, DerefMut}, rc::Rc, sync::Arc};
|
||||||
|
|
||||||
|
use raylib::{prelude::RaylibDraw, RaylibHandle};
|
||||||
|
|
||||||
|
pub struct HackedRaylibHandle(RaylibHandle);
|
||||||
|
|
||||||
|
impl RaylibDraw for HackedRaylibHandle {}
|
||||||
|
|
||||||
|
impl From<RaylibHandle> for HackedRaylibHandle {
|
||||||
|
fn from(handle: RaylibHandle) -> Self {
|
||||||
|
Self(handle)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for HackedRaylibHandle {
|
||||||
|
type Target = RaylibHandle;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DerefMut for HackedRaylibHandle {
|
||||||
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||||
|
&mut self.0
|
||||||
|
}
|
||||||
|
}
|
@ -14,6 +14,8 @@ use raylib::{
|
|||||||
};
|
};
|
||||||
use rust_embed::EmbeddedFile;
|
use rust_embed::EmbeddedFile;
|
||||||
|
|
||||||
|
use crate::utilities::non_ref_raylib::HackedRaylibHandle;
|
||||||
|
|
||||||
#[derive(Debug, Error)]
|
#[derive(Debug, Error)]
|
||||||
pub enum ShaderError {
|
pub enum ShaderError {
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
@ -76,13 +78,12 @@ impl ShaderWrapper {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Handles rendering a texture to the screen via the shader. If run inside another shader context, this *should* chain with it.
|
/// Handles rendering a texture to the screen via the shader. If run inside another shader context, this *should* chain with it.
|
||||||
pub fn process_texture_and_render<H>(
|
pub fn process_texture_and_render(
|
||||||
&self,
|
&self,
|
||||||
raylib: &mut H,
|
raylib: &mut HackedRaylibHandle,
|
||||||
_thread: &RaylibThread,
|
_thread: &RaylibThread,
|
||||||
texture: &RenderTexture2D,
|
texture: &RenderTexture2D,
|
||||||
) where
|
)
|
||||||
H: RaylibShaderModeExt + RaylibDraw,
|
|
||||||
{
|
{
|
||||||
puffin::profile_function!();
|
puffin::profile_function!();
|
||||||
// Create a shader context to work under
|
// Create a shader context to work under
|
||||||
|
Reference in New Issue
Block a user