This repository has been archived on 2021-10-11. You can view files and clone it, but cannot push or open issues or pull requests.

224 lines
6.9 KiB
Rust

#![feature(derive_default_enum)]
#![deny(unsafe_code)]
#![warn(
clippy::all,
clippy::await_holding_lock,
clippy::char_lit_as_u8,
clippy::checked_conversions,
clippy::dbg_macro,
clippy::debug_assert_with_mut_call,
clippy::doc_markdown,
clippy::empty_enum,
clippy::enum_glob_use,
clippy::exit,
clippy::expl_impl_clone_on_copy,
clippy::explicit_deref_methods,
clippy::explicit_into_iter_loop,
clippy::fallible_impl_from,
clippy::filter_map_next,
clippy::float_cmp_const,
clippy::fn_params_excessive_bools,
clippy::if_let_mutex,
clippy::implicit_clone,
clippy::imprecise_flops,
clippy::inefficient_to_string,
clippy::invalid_upcast_comparisons,
clippy::large_types_passed_by_value,
clippy::let_unit_value,
clippy::linkedlist,
clippy::lossy_float_literal,
clippy::macro_use_imports,
clippy::manual_ok_or,
clippy::map_err_ignore,
clippy::map_flatten,
clippy::map_unwrap_or,
clippy::match_on_vec_items,
clippy::match_same_arms,
clippy::match_wildcard_for_single_variants,
clippy::mem_forget,
clippy::mismatched_target_os,
clippy::mut_mut,
clippy::mutex_integer,
clippy::needless_borrow,
clippy::needless_continue,
clippy::option_option,
clippy::path_buf_push_overwrite,
clippy::ptr_as_ptr,
clippy::ref_option_ref,
clippy::rest_pat_in_fully_bound_structs,
clippy::same_functions_in_if_condition,
clippy::semicolon_if_nothing_returned,
clippy::string_add_assign,
clippy::string_add,
clippy::string_lit_as_bytes,
clippy::string_to_string,
clippy::todo,
clippy::trait_duplication_in_bounds,
clippy::unimplemented,
clippy::unnested_or_patterns,
clippy::unused_self,
clippy::useless_transmute,
clippy::verbose_file_reads,
clippy::zero_sized_map_values,
future_incompatible,
nonstandard_style,
rust_2018_idioms
)]
#![feature(custom_inner_attributes)]
#![clippy::msrv = "1.57.0"]
use std::cell::RefCell;
use discord_sdk::activity::ActivityBuilder;
use raylib::prelude::*;
use tracing::{error, info};
use utilities::discord::DiscordConfig;
use crate::{
context::GameContext,
discord_rpc::{maybe_set_discord_presence, try_connect_to_local_discord},
scenes::build_screen_state_machine,
utilities::shaders::{
shader::ShaderWrapper,
util::{dynamic_screen_texture::DynScreenTexture, render_texture::render_to_texture},
},
};
#[macro_use]
extern crate thiserror;
#[macro_use]
extern crate serde;
mod context;
mod discord_rpc;
mod scenes;
mod utilities;
pub use utilities::{datastore::StaticGameData, game_config::GameConfig};
/// The game entrypoint
pub async fn game_begin(game_config: &GameConfig) -> Result<(), Box<dyn std::error::Error>> {
// Set up profiling
#[cfg(debug_assertions)]
let _puffin_server =
puffin_http::Server::new(&format!("0.0.0.0:{}", puffin_http::DEFAULT_PORT)).unwrap();
puffin::set_scopes_on(true);
// Attempt to connect to a locally running Discord instance for rich presence access
let discord_config = DiscordConfig::load(
StaticGameData::get("configs/discord.json").expect("Failed to load discord.json"),
)
.unwrap();
let discord_rpc = match try_connect_to_local_discord(&discord_config).await {
Ok(client) => Some(client),
Err(err) => match err {
utilities::discord::rpc::DiscordError::ConnectionTimeoutError(time) => {
error!(
"Could not find or connect to a local Discord instance after {} seconds",
time
);
None
}
_ => panic!("Failed to connect to Discord: {}", err),
},
};
maybe_set_discord_presence(
&discord_rpc,
ActivityBuilder::default().details("Testing..."),
)
.await
.unwrap();
// Get the main state machine
let mut game_state_machine = build_screen_state_machine().unwrap();
let context;
let raylib_thread;
{
// Set up FFI access to raylib
let (mut rl, thread) = raylib::init()
.size(640, 480)
.title(&game_config.name)
.vsync()
.msaa_4x()
.resizable()
.build();
rl.set_exit_key(None);
raylib_thread = thread;
// Build the game context
context = Box::new(GameContext::new(RefCell::new(rl.into())));
}
// Create a dynamic texture to draw to for processing by shaders
info!("Allocating a resizable texture for the screen");
let mut dynamic_texture =
DynScreenTexture::new(&mut context.renderer.borrow_mut(), &raylib_thread)?;
// Load the pixel art shader
info!("Loading the pixel art shader");
let mut pixel_shader = ShaderWrapper::new(
None,
Some(StaticGameData::get("shaders/pixelart.fs")).expect("Failed to load pixelart.fs"),
vec!["viewport"],
&mut context.renderer.borrow_mut(),
&raylib_thread,
)?;
info!("Starting the render loop");
while !context.renderer.borrow().window_should_close() {
// Profile the main game loop
puffin::profile_scope!("main_loop");
puffin::GlobalProfiler::lock().new_frame();
// Update the GPU texture that we draw to. This handles screen resizing and some other stuff
dynamic_texture
.update(&mut context.renderer.borrow_mut(), &raylib_thread)
.unwrap();
// Switch into draw mode the unsafe way (using unsafe code here to avoid borrow checker hell)
#[allow(unsafe_code)]
unsafe {
raylib::ffi::BeginDrawing();
}
// Fetch the screen size once to work with in render code
let screen_size = Vector2::new(
context.renderer.borrow().get_screen_width() as f32,
context.renderer.borrow().get_screen_height() as f32,
);
// Update the pixel shader to correctly handle the screen size
pixel_shader.set_variable("viewport", screen_size)?;
// Render the game via the pixel shader
render_to_texture(&mut dynamic_texture, || {
// Profile the internal render code
puffin::profile_scope!("internal_shaded_render");
// Run a state machine iteration
let result = game_state_machine.run(&context);
if let Err(err) = result {
error!("Main state machine encountered an error while running!");
error!("Main thread crash!!");
error!("Cannot recover from error");
panic!("{:?}", err);
}
});
// Send the texture to the GPU to be drawn
pixel_shader.process_texture_and_render(
&mut context.renderer.borrow_mut(),
&raylib_thread,
&dynamic_texture,
);
// We MUST end draw mode
#[allow(unsafe_code)]
unsafe {
raylib::ffi::EndDrawing();
}
}
Ok(())
}