Add a stateful discord activity system
This commit is contained in:
parent
467a525d01
commit
683d90fe1d
@ -12,4 +12,6 @@ log = "0.4.14"
|
|||||||
profiling = "1.0.5"
|
profiling = "1.0.5"
|
||||||
serde = { version = "1.0.136", features = ["derive"] }
|
serde = { version = "1.0.136", features = ["derive"] }
|
||||||
serde_json = "1.0.79"
|
serde_json = "1.0.79"
|
||||||
directories = "4.0.1"
|
directories = "4.0.1"
|
||||||
|
chrono = { verison = "0.4.19", features = ["serde"] }
|
||||||
|
discord-sdk = "0.3.0"
|
4
game/game_logic/src/discord/mod.rs
Normal file
4
game/game_logic/src/discord/mod.rs
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
//! This module contains code needed for interacting with a local Discord instance.
|
||||||
|
|
||||||
|
mod signal;
|
||||||
|
pub use signal::DiscordRpcSignal;
|
94
game/game_logic/src/discord/signal.rs
Normal file
94
game/game_logic/src/discord/signal.rs
Normal file
@ -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<Utc>),
|
||||||
|
|
||||||
|
/// 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<String>,
|
||||||
|
},
|
||||||
|
|
||||||
|
/// 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<chrono::DateTime<Utc>>,
|
||||||
|
game_end_timer: Option<chrono::DateTime<Utc>>,
|
||||||
|
game_details: Option<String>,
|
||||||
|
game_party_status: Option<String>,
|
||||||
|
game_assets: Option<Assets>,
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
@ -4,6 +4,7 @@ use std::borrow::Borrow;
|
|||||||
|
|
||||||
mod persistent;
|
mod persistent;
|
||||||
mod rendering;
|
mod rendering;
|
||||||
|
mod discord;
|
||||||
|
|
||||||
/// This is the game logic entrypoint. Despite being async,
|
/// This is the game logic entrypoint. Despite being async,
|
||||||
/// this is expected to block the main thread for rendering and stuff.
|
/// this is expected to block the main thread for rendering and stuff.
|
||||||
|
@ -1,11 +1,13 @@
|
|||||||
use raylib::RaylibBuilder;
|
use raylib::RaylibBuilder;
|
||||||
|
|
||||||
|
/// Will begin rendering graphics. Returns when the window closes
|
||||||
pub fn handle_graphics_blocking<ConfigBuilder>(config: ConfigBuilder, target_frames_per_second: u32)
|
pub fn handle_graphics_blocking<ConfigBuilder>(config: ConfigBuilder, target_frames_per_second: u32)
|
||||||
where
|
where
|
||||||
ConfigBuilder: FnOnce(&mut RaylibBuilder),
|
ConfigBuilder: FnOnce(&mut RaylibBuilder),
|
||||||
{
|
{
|
||||||
// Let the caller configure Raylib's internal window stuff
|
// Let the caller configure Raylib's internal window stuff
|
||||||
let (mut raylib_handle, raylib_thread) = {
|
let (mut raylib_handle, raylib_thread) = {
|
||||||
|
log::trace!("Configuring Raylib");
|
||||||
let mut builder = raylib::init();
|
let mut builder = raylib::init();
|
||||||
config(&mut builder);
|
config(&mut builder);
|
||||||
builder.build()
|
builder.build()
|
||||||
@ -16,5 +18,11 @@ where
|
|||||||
raylib_handle.set_target_fps(target_frames_per_second);
|
raylib_handle.set_target_fps(target_frames_per_second);
|
||||||
|
|
||||||
// Run the event loop
|
// 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");
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user