This commit is contained in:
Adam 2024-08-14 00:16:54 -04:00
parent 54d16b2c01
commit bfc4600396
6 changed files with 240 additions and 12 deletions

View file

@ -5,7 +5,7 @@ use lib::*;
#[component] #[component]
pub fn Game() -> impl IntoView { pub fn Game() -> impl IntoView {
// let websocket = expect_context::<WebSocketContext>(); // let websocket = expect_context::<WebSocketContext>();
let game_meta = expect_context::<ReadSignal<Option<GameMeta>>>(); let game_meta = expect_context::<ReadSignal<Option<GameStateMeta>>>();
let (game_name, set_game_name) = create_signal("".to_string()); let (game_name, set_game_name) = create_signal("".to_string());
let (game_host, set_game_host) = create_signal("".to_string()); let (game_host, set_game_host) = create_signal("".to_string());

View file

@ -73,7 +73,7 @@ pub fn Websocket() -> impl IntoView {
let (chat_update, set_chat_update) = create_signal::<Option<ChatUpdate>>(Option::None); let (chat_update, set_chat_update) = create_signal::<Option<ChatUpdate>>(Option::None);
let (chat_message, set_chat_message) = create_signal::<Option<ChatMessage>>(Option::None); let (chat_message, set_chat_message) = create_signal::<Option<ChatMessage>>(Option::None);
let (active_games, set_active_games) = create_signal::<Vec<GameBrowserMeta>>(vec![]); let (active_games, set_active_games) = create_signal::<Vec<GameBrowserMeta>>(vec![]);
let (current_game, set_current_game) = create_signal::<Option<GameMeta>>(Option::None); let (current_game, set_current_game) = create_signal::<Option<GameStateMeta>>(Option::None);
let (card_packs_meta, set_card_packs_meta) = create_signal::<CardPacksMeta>(CardPacksMeta { let (card_packs_meta, set_card_packs_meta) = create_signal::<CardPacksMeta>(CardPacksMeta {
official_meta: vec![], official_meta: vec![],
unofficial_meta: vec![], unofficial_meta: vec![],
@ -84,7 +84,7 @@ pub fn Websocket() -> impl IntoView {
provide_context::<ReadSignal<Option<ChatUpdate>>>(chat_update); provide_context::<ReadSignal<Option<ChatUpdate>>>(chat_update);
provide_context::<ReadSignal<Option<ChatMessage>>>(chat_message); provide_context::<ReadSignal<Option<ChatMessage>>>(chat_message);
provide_context::<ReadSignal<Vec<GameBrowserMeta>>>(active_games); provide_context::<ReadSignal<Vec<GameBrowserMeta>>>(active_games);
provide_context::<ReadSignal<Option<GameMeta>>>(current_game); provide_context::<ReadSignal<Option<GameStateMeta>>>(current_game);
provide_context::<ReadSignal<CardPacksMeta>>(card_packs_meta); provide_context::<ReadSignal<CardPacksMeta>>(card_packs_meta);
provide_context::<ReadSignal<Option<ServerStateSummary>>>(state_summary); provide_context::<ReadSignal<Option<ServerStateSummary>>>(state_summary);
@ -107,7 +107,7 @@ pub fn Websocket() -> impl IntoView {
set_chat_update(Some(chat_update)); set_chat_update(Some(chat_update));
} else if let Ok(games_update) = from_str::<GamesUpdate>(message) { } else if let Ok(games_update) = from_str::<GamesUpdate>(message) {
set_active_games(games_update.games); set_active_games(games_update.games);
} else if let Ok(game_update) = from_str::<GameMeta>(message) { } else if let Ok(game_update) = from_str::<GameStateMeta>(message) {
set_current_game(Some(game_update)); set_current_game(Some(game_update));
} else if let Ok(packs_meta_update) = from_str::<CardPacksMeta>(message) { } else if let Ok(packs_meta_update) = from_str::<CardPacksMeta>(message) {
set_card_packs_meta(packs_meta_update); set_card_packs_meta(packs_meta_update);

View file

@ -8,7 +8,7 @@ pub struct GameJoinRequest {
/// Game meta /// Game meta
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
pub struct GameMeta { pub struct GameStateMeta {
pub uuid: String, pub uuid: String,
pub name: String, pub name: String,
pub host: String, pub host: String,

View file

@ -280,7 +280,7 @@ impl Game {
} }
/// Create a new player and add them to the game. /// Create a new player and add them to the game.
fn create_player(&mut self, user: Arc<RwLock<User>>) -> Result<()> { pub fn create_player(&mut self, user: Arc<RwLock<User>>) -> Result<()> {
let mut new_player = Player { let mut new_player = Player {
white: vec![], white: vec![],
black: vec![], black: vec![],
@ -310,8 +310,8 @@ pub enum GameHandlerMessage {
new_game: NewGameRequest, new_game: NewGameRequest,
}, },
JoinGame { JoinGame {
user: Arc<User>, addr: SocketAddr,
game_name: String, id: String,
}, },
} }
pub struct GameHandler { pub struct GameHandler {
@ -326,12 +326,224 @@ impl GameHandler {
pub async fn handle(&self, message: GameHandlerMessage) { pub async fn handle(&self, message: GameHandlerMessage) {
match message { match message {
GameHandlerMessage::NewGame { addr, new_game } => self.new_game(addr, new_game).await, GameHandlerMessage::NewGame { addr, new_game } => self.new_game(addr, new_game).await,
GameHandlerMessage::JoinGame { addr, id } => self.join_game(addr, id).await,
_ => { _ => {
tracing::debug!("Unhandled at game handler"); tracing::debug!("Unhandled at game handler");
} }
} }
} }
async fn join_game(&self, addr: SocketAddr, id: String) {
// Create player
self.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.write()
.unwrap()
.create_player(
self.state
.online_users
.read()
.unwrap()
.get(&addr)
.unwrap()
.clone(),
)
.unwrap();
// Create update for user's game view
let mut black_card = ("Error".to_string(), 0u8);
if let Some(ref current_black) = self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.current_black
{
black_card = (current_black.text.to_owned(), current_black.pick)
}
let meta = GameStateMeta {
uuid: id.clone(),
name: self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.name
.clone(),
host: self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.host
.read()
.unwrap()
.name
.clone(),
players: self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.players
.iter()
.map(|player| {
self.state
.user_uuid
.read()
.unwrap()
.get(player.0)
.unwrap()
.read()
.unwrap()
.name
.clone()
})
.collect(),
czar: self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.host
.read()
.unwrap()
.name
.clone(),
black: black_card,
white: self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.players
.get(
&self
.state
.online_users
.read()
.unwrap()
.get(&addr)
.unwrap()
.read()
.unwrap()
.uuid,
)
.unwrap()
.white
.iter()
.map(|card| card.text.clone())
.collect(),
packs: self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.packs
.clone(),
};
// Send updates for all players
for player in self
.state
.games
.read()
.unwrap()
.get(&id)
.unwrap()
.read()
.unwrap()
.players
.iter()
{
// // Send user's update
// self.state
// .user_uuid
// .read()
// .unwrap()
// .get(&player.0)
// .unwrap()
// .read()
// .unwrap()
// .tx
// .send(serde_json::to_string(&meta).unwrap())
// .await
// .unwrap();
tracing::debug! {"y"};
}
// let _ = self
// .state
// .games
// .read()
// .unwrap()
// .get(&id)
// .unwrap()
// .read()
// .unwrap()
// .players
// .iter()
// .map(async |player: (&Uuid, &Player)| {
// // Send user's update
// self.state
// .user_uuid
// .read()
// .unwrap()
// .get(player.0)
// .unwrap()
// .read()
// .unwrap()
// .tx
// .send(serde_json::to_string(&meta).unwrap())
// .await
// .unwrap();
// })
// .count();
// Broadcast game browser update
self.state
.broadcast_tx
.send(meta_games_browser_update(&self.state))
.unwrap();
// Broadcast server meta update
self.state
.broadcast_tx
.send(meta_server_summary_update(&self.state))
.unwrap();
}
async fn new_game(&self, addr: SocketAddr, new_game: NewGameRequest) { async fn new_game(&self, addr: SocketAddr, new_game: NewGameRequest) {
if new_game.packs.is_empty() { if new_game.packs.is_empty() {
tracing::debug!("Cards are empty"); tracing::debug!("Cards are empty");
@ -354,7 +566,7 @@ impl GameHandler {
packs: new_game.packs, packs: new_game.packs,
}; };
// create game // Create game
if let Ok(new_game_object) = Game::new(self.state.clone(), manifest) { if let Ok(new_game_object) = Game::new(self.state.clone(), manifest) {
let tx = self let tx = self
.state .state
@ -368,12 +580,12 @@ impl GameHandler {
.tx .tx
.clone(); .clone();
// Create update for user's game view
let mut black_card = ("Error".to_string(), 0u8); let mut black_card = ("Error".to_string(), 0u8);
if let Some(ref current_black) = new_game_object.current_black { if let Some(ref current_black) = new_game_object.current_black {
black_card = (current_black.text.to_owned(), current_black.pick) black_card = (current_black.text.to_owned(), current_black.pick)
} }
let meta = GameStateMeta {
let meta = GameMeta {
uuid: new_game_object.uuid.to_string(), uuid: new_game_object.uuid.to_string(),
name: new_game_object.name.clone(), name: new_game_object.name.clone(),
host: new_game_object.host.read().unwrap().name.clone(), host: new_game_object.host.read().unwrap().name.clone(),
@ -406,17 +618,24 @@ impl GameHandler {
packs: new_game_object.packs.clone(), packs: new_game_object.packs.clone(),
}; };
// Send user's update
tx.send(serde_json::to_string(&meta).unwrap()) tx.send(serde_json::to_string(&meta).unwrap())
.await .await
.unwrap(); .unwrap();
// Add game to active list
self.state.games.write().unwrap().insert( self.state.games.write().unwrap().insert(
new_game_object.name.clone(), new_game_object.uuid.to_string(),
Arc::new(RwLock::new(new_game_object)), Arc::new(RwLock::new(new_game_object)),
); );
// Broadcast game browser update
self.state self.state
.broadcast_tx .broadcast_tx
.send(meta_games_browser_update(&self.state)) .send(meta_games_browser_update(&self.state))
.unwrap(); .unwrap();
// Broadcast server meta update
self.state self.state
.broadcast_tx .broadcast_tx
.send(meta_server_summary_update(&self.state)) .send(meta_server_summary_update(&self.state))

View file

@ -1,4 +1,5 @@
#![feature(if_let_guard)] #![feature(if_let_guard)]
#![feature(async_closure)]
use crate::game_handler::*; use crate::game_handler::*;
use anyhow::{Context, Result}; use anyhow::{Context, Result};

View file

@ -47,6 +47,14 @@ impl MessageHandler {
.unwrap(); .unwrap();
} }
_join_game_request if let Ok(join_request) = from_str::<GameJoinRequest>(&text) => {
self.state
.games_tx
.send(GameHandlerMessage::JoinGame { addr, id: join_request.id })
.await
.unwrap();
}
_ => tracing::debug!("Unhandled text from {}", addr), _ => tracing::debug!("Unhandled text from {}", addr),
}, },
Message::Binary(data) => tracing::debug!("{} sent binary: {:?}", addr, data), Message::Binary(data) => tracing::debug!("{} sent binary: {:?}", addr, data),