From 683d90fe1db821f4008e4abf530cd916faac003c Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Fri, 18 Mar 2022 12:10:46 -0400 Subject: [PATCH] Add a stateful discord activity system --- game/game_logic/Cargo.toml | 4 +- game/game_logic/src/discord/mod.rs | 4 + game/game_logic/src/discord/signal.rs | 94 +++++++++++++++++++++ game/game_logic/src/lib.rs | 1 + game/game_logic/src/rendering/event_loop.rs | 10 ++- 5 files changed, 111 insertions(+), 2 deletions(-) create mode 100644 game/game_logic/src/discord/mod.rs create mode 100644 game/game_logic/src/discord/signal.rs diff --git a/game/game_logic/Cargo.toml b/game/game_logic/Cargo.toml index 071d3ac1..555c17e1 100644 --- a/game/game_logic/Cargo.toml +++ b/game/game_logic/Cargo.toml @@ -12,4 +12,6 @@ log = "0.4.14" profiling = "1.0.5" serde = { version = "1.0.136", features = ["derive"] } serde_json = "1.0.79" -directories = "4.0.1" \ No newline at end of file +directories = "4.0.1" +chrono = { verison = "0.4.19", features = ["serde"] } +discord-sdk = "0.3.0" \ No newline at end of file diff --git a/game/game_logic/src/discord/mod.rs b/game/game_logic/src/discord/mod.rs new file mode 100644 index 00000000..74413437 --- /dev/null +++ b/game/game_logic/src/discord/mod.rs @@ -0,0 +1,4 @@ +//! This module contains code needed for interacting with a local Discord instance. + +mod signal; +pub use signal::DiscordRpcSignal; diff --git a/game/game_logic/src/discord/signal.rs b/game/game_logic/src/discord/signal.rs new file mode 100644 index 00000000..2bb9004c --- /dev/null +++ b/game/game_logic/src/discord/signal.rs @@ -0,0 +1,94 @@ +//! This file contains a system for signaling Discord RPC context changes between threads. +//! +//! ## Description +//! +//! The idea is that the thread containing the Discord RPC client can hold a `StatefulDiscordRpcSignalHandler` as a stateful context. +//! The game thread can then send `DiscordRpcSignal` values through an `mpsc` sender, which will be received by the Discord RPC client thread. + +use chrono::Utc; +use discord_sdk::activity::{ActivityBuilder, Assets, IntoTimestamp}; + +/// Definitions of signals that can be sent to the Discord RPC thread to control how discord displays game status. +pub enum DiscordRpcSignal { + /// Signal to begin a game timer (Discord will display `XX:XX elapsed`) + BeginGameTimer, + + /// Signal to end a game timer + EndGameTimer, + + /// Signal to begin a countdown timer (Discord will display `XX:XX left`) + SetGameTimeRemainingTimestamp(chrono::DateTime), + + /// Signal to clear the game remaining timer + ClearGameTimeRemaining, + + /// Signal to set the details in the info card + ChangeDetails { + /// What the player is doing, eg. “Exploring the Wilds of Outland”. + /// + /// Limited to 128 bytes. + details: String, + + /// The user’s currenty party status, eg. “Playing Solo”. + /// + /// Limited to 128 bytes. + party_status: Option, + }, + + /// Signal to change the graphical assets in the info card + ChangeAssets(Assets), +} + +/// A struct that can keep track of incoming signals and their effect on Discord +#[derive(Default, Debug, Clone)] +pub struct StatefulDiscordRpcSignalHandler { + game_start_timer: Option>, + game_end_timer: Option>, + game_details: Option, + game_party_status: Option, + game_assets: Option, +} + +impl StatefulDiscordRpcSignalHandler { + + /// Apply a signal to generate a new activity + pub fn apply(&mut self, signal: DiscordRpcSignal) -> ActivityBuilder { + // Fill in the data based on the contents of the signal + match signal { + DiscordRpcSignal::BeginGameTimer => self.game_start_timer = Some(chrono::Utc::now()), + DiscordRpcSignal::EndGameTimer => self.game_start_timer = None, + DiscordRpcSignal::SetGameTimeRemainingTimestamp(timestamp) => { + self.game_end_timer = Some(timestamp) + } + DiscordRpcSignal::ClearGameTimeRemaining => self.game_end_timer = None, + DiscordRpcSignal::ChangeDetails { + details, + party_status, + } => { + self.game_details = Some(details); + self.game_party_status = party_status; + } + DiscordRpcSignal::ChangeAssets(assets) => self.game_assets = Some(assets), + } + + // Decide how to build the Discord RPC activity + let mut builder = ActivityBuilder::default(); + if let Some(start_time) = &self.game_start_timer { + builder = builder.start_timestamp(start_time.timestamp()); + } + if let Some(end_time) = &self.game_end_timer { + builder = builder.end_timestamp(end_time.timestamp()); + } + if let Some(details) = &self.game_details { + builder = builder.details(details); + } + if let Some(party_status) = &self.game_party_status { + builder = builder.state(party_status); + } + if let Some(assets) = &self.game_assets { + builder = builder.assets(assets.clone()); + } + + return builder; + } +} diff --git a/game/game_logic/src/lib.rs b/game/game_logic/src/lib.rs index 679b3df1..b1f3eb44 100644 --- a/game/game_logic/src/lib.rs +++ b/game/game_logic/src/lib.rs @@ -4,6 +4,7 @@ use std::borrow::Borrow; mod persistent; mod rendering; +mod discord; /// This is the game logic entrypoint. Despite being async, /// this is expected to block the main thread for rendering and stuff. diff --git a/game/game_logic/src/rendering/event_loop.rs b/game/game_logic/src/rendering/event_loop.rs index cfb693a3..d0dc3279 100644 --- a/game/game_logic/src/rendering/event_loop.rs +++ b/game/game_logic/src/rendering/event_loop.rs @@ -1,11 +1,13 @@ use raylib::RaylibBuilder; +/// Will begin rendering graphics. Returns when the window closes pub fn handle_graphics_blocking(config: ConfigBuilder, target_frames_per_second: u32) where ConfigBuilder: FnOnce(&mut RaylibBuilder), { // Let the caller configure Raylib's internal window stuff let (mut raylib_handle, raylib_thread) = { + log::trace!("Configuring Raylib"); let mut builder = raylib::init(); config(&mut builder); builder.build() @@ -16,5 +18,11 @@ where raylib_handle.set_target_fps(target_frames_per_second); // Run the event loop - while !raylib_handle.window_should_close() {} + log::trace!("Running event loop"); + while !raylib_handle.window_should_close() { + + // Tell the profiler that we ended the frame + profiling::finish_frame!(); + } + log::trace!("Event loop ended"); }