Compare commits

..

4 commits

Author SHA1 Message Date
9ab8db41de
Don't specify peer port
All checks were successful
CI / Formatting (push) Successful in 1m33s
Using a port of 0 and letting the OS choose for us means that we can run multiple peers locally
2025-07-05 23:04:46 -04:00
c29344ba6d
Add Peer UI display 2025-07-05 23:03:52 -04:00
e883f201c0
Add Seed display 2025-07-05 21:36:17 -04:00
58795eb87e
Don't expose heartbeat to app 2025-07-05 19:28:14 -04:00
11 changed files with 123 additions and 13 deletions

4
.vscode/launch.json vendored
View file

@ -11,7 +11,7 @@
"cargo": { "cargo": {
"args": ["build"] "args": ["build"]
}, },
"args": ["--seed=gargamel"], "args": ["--seed=gargamel", "--port=25565"],
"cwd": "${workspaceFolder}", "cwd": "${workspaceFolder}",
"env": { "env": {
"CARGO_MANIFEST_DIR": "${workspaceFolder}", "CARGO_MANIFEST_DIR": "${workspaceFolder}",
@ -30,7 +30,7 @@
"cargo": { "cargo": {
"args": ["build"] "args": ["build"]
}, },
"args": ["--port=25566", "--connect=[::1]:25565"], "args": ["--connect=[::1]:25565"],
"cwd": "${workspaceFolder}", "cwd": "${workspaceFolder}",
"env": { "env": {
"CARGO_MANIFEST_DIR": "${workspaceFolder}", "CARGO_MANIFEST_DIR": "${workspaceFolder}",

View file

@ -6,6 +6,7 @@ mod runtime;
mod seed; mod seed;
mod setup; mod setup;
pub mod state; pub mod state;
mod ui;
#[allow(unused_imports)] #[allow(unused_imports)]
pub mod prelude { pub mod prelude {

View file

@ -11,7 +11,6 @@ pub fn handle_new_peer(
) { ) {
if let Some(seed) = seed { if let Some(seed) = seed {
for peer in new_peers { for peer in new_peers {
warn!("Sending seed to peer: {}", peer.uuid);
outbound.write(OutboundPacket(Packet::new((*seed).into(), peer.uuid))); outbound.write(OutboundPacket(Packet::new((*seed).into(), peer.uuid)));
} }
} }

View file

@ -9,8 +9,11 @@ use super::{
net::{handle_deleted_peer, handle_incoming_packets, handle_new_peer}, net::{handle_deleted_peer, handle_incoming_packets, handle_new_peer},
runtime::{move_camera, move_player, quit, zoom_camera}, runtime::{move_camera, move_player, quit, zoom_camera},
seed::Seed, seed::Seed,
setup::{check_for_seed, setup_balls, setup_from_seed, setup_player, setup_ui, setup_walls}, setup::{
check_for_seed, setup_balls, setup_camera, setup_from_seed, setup_player, setup_walls,
},
state::AppState, state::AppState,
ui::{setup_peer_ui, setup_seed_ui, update_peer_ui, update_peer_ui_timings, update_seed_ui},
}; };
#[derive(Debug, Clone, Copy)] #[derive(Debug, Clone, Copy)]
@ -56,7 +59,7 @@ impl Plugin for GamePlugin {
PhysicsPlugins::default().with_length_unit(50.0), PhysicsPlugins::default().with_length_unit(50.0),
)) ))
.init_state::<AppState>() .init_state::<AppState>()
.add_systems(Startup, setup_ui) .add_systems(Startup, (setup_camera, setup_seed_ui, setup_peer_ui))
.add_systems( .add_systems(
OnEnter(AppState::InGame), OnEnter(AppState::InGame),
( (
@ -81,6 +84,9 @@ impl Plugin for GamePlugin {
( (
((move_player, move_camera).chain(), zoom_camera) ((move_player, move_camera).chain(), zoom_camera)
.run_if(in_state(AppState::InGame)), .run_if(in_state(AppState::InGame)),
update_seed_ui,
(update_peer_ui, update_peer_ui_timings)
.run_if(in_state(NetworkState::MultiPlayer)),
quit.run_if(input_pressed(KeyCode::KeyQ)), quit.run_if(input_pressed(KeyCode::KeyQ)),
), ),
); );

View file

@ -43,7 +43,7 @@ pub fn setup_from_seed(mut commands: Commands, seed: Res<Seed>) {
} }
/// I mean, a camera is technically a user interface, I guess /// I mean, a camera is technically a user interface, I guess
pub fn setup_ui(mut commands: Commands) { pub fn setup_camera(mut commands: Commands) {
commands.spawn((Name::new("Camera"), Camera2d, IsDefaultUiCamera)); commands.spawn((Name::new("Camera"), Camera2d, IsDefaultUiCamera));
} }

102
src/game/ui.rs Normal file
View file

@ -0,0 +1,102 @@
use bevy::prelude::*;
use crate::net::prelude::{Peer, PeerReceiveTiming, PeerSendTiming};
use super::seed::Seed;
#[derive(Component, Debug)]
pub struct SeedUI;
pub fn setup_seed_ui(mut commands: Commands) {
commands
.spawn((
Text::new("Seed: "),
Node {
position_type: PositionType::Absolute,
bottom: Val::Px(5.0),
left: Val::Px(5.0),
..default()
},
))
.with_child((TextSpan::new("<N/A>"), SeedUI));
}
pub fn update_seed_ui(seed: Option<Res<Seed>>, text: Query<&mut TextSpan, With<SeedUI>>) {
if let Some(value) = seed {
if value.is_changed() {
for mut span in text {
let number: u64 = (*value).into();
**span = format!("{}", number);
}
}
}
}
#[derive(Component, Debug)]
pub struct PeerUI;
pub fn setup_peer_ui(mut commands: Commands) {
commands
.spawn((
Node {
display: Display::Flex,
position_type: PositionType::Absolute,
top: Val::Px(5.0),
right: Val::Px(5.0),
row_gap: Val::Px(5.0),
flex_direction: FlexDirection::Column,
align_items: AlignItems::FlexEnd,
justify_content: JustifyContent::FlexStart,
..default()
},
PeerUI,
))
.with_children(|builder| {
builder.spawn(Text::new("Peer Recv Send"));
});
}
#[derive(Component)]
pub struct PeerID(Entity);
pub fn update_peer_ui(
ui: Query<(Entity, &Children), With<PeerUI>>,
rows: Query<&PeerID>,
added: Query<Entity, Added<Peer>>,
mut removed: RemovedComponents<Peer>,
mut commands: Commands,
) {
for (table, children) in ui {
for addition in added {
commands
.entity(table)
.with_child((Text::new("---- ---- ----"), PeerID(addition)));
}
for removal in removed.read() {
for child in children {
if let Ok(id) = rows.get(*child) {
if id.0 == removal {
commands.entity(*child).despawn();
}
}
}
}
}
}
pub fn update_peer_ui_timings(
rows: Query<(&mut Text, &PeerID)>,
peers: Query<(&Peer, &PeerReceiveTiming, &PeerSendTiming)>,
time: Res<Time>,
) {
for (mut row, id) in rows {
if let Ok((peer, recv, send)) = peers.get(id.0) {
**row = format!(
"{} {:.2} {:.2}",
peer.uuid,
(time.elapsed() - recv.time().unwrap_or(default())).as_secs_f64(),
(time.elapsed() - send.time()).as_secs_f64()
)
}
}
}

View file

@ -20,7 +20,7 @@ pub struct AppSettings {
source: Source, source: Source,
/// The port the app should listen for connections on /// The port the app should listen for connections on
#[arg(short, long, default_value = "25565")] #[arg(short, long, default_value = "0")]
port: u16, port: u16,
} }

View file

@ -26,7 +26,9 @@ pub fn handle_network_input(
.into()); .into());
} }
let uuid = Uuid::from_slice(message.split_off(message.len() - 16).as_slice())?; let uuid = Uuid::from_slice(message.split_off(message.len() - 16).as_slice())?;
to_app.write(InboundPacket(Packet::new(message, uuid))); if !message.is_empty() {
to_app.write(InboundPacket(Packet::new(message, uuid)));
}
if let Some(peer_id) = peer_map.get(&uuid) { if let Some(peer_id) = peer_map.get(&uuid) {
let (peer, mut last) = peers.get_mut(*peer_id)?; let (peer, mut last) = peers.get_mut(*peer_id)?;
last.update(&time); last.update(&time);
@ -74,7 +76,7 @@ pub fn heartbeat(
} }
} }
pub fn timeouts( pub fn timeout(
peers: Query<(&Peer, &PeerReceiveTiming)>, peers: Query<(&Peer, &PeerReceiveTiming)>,
time: Res<Time>, time: Res<Time>,
mut delete: EventWriter<PeerChangeEvent>, mut delete: EventWriter<PeerChangeEvent>,

View file

@ -9,6 +9,6 @@ mod thread;
#[allow(unused_imports)] #[allow(unused_imports)]
pub mod prelude { pub mod prelude {
pub use super::packet::{InboundPacket, OutboundPacket, Packet}; pub use super::packet::{InboundPacket, OutboundPacket, Packet};
pub use super::peer::Peer; pub use super::peer::{Peer, PeerReceiveTiming, PeerSendTiming};
pub use super::plugin::{NetIOPlugin, NetworkState}; pub use super::plugin::{NetIOPlugin, NetworkState};
} }

View file

@ -107,7 +107,7 @@ pub fn handle_peer_change(
peer_map.remove(&change.peer); peer_map.remove(&change.peer);
} }
} else if let Some(addr) = change.addr { } else if let Some(addr) = change.addr {
info!("Adding peer {} ({:?})", change.peer, change.addr); info!("Adding peer {} ({})", change.peer, addr);
peer_map.insert( peer_map.insert(
change.peer, change.peer,
commands commands

View file

@ -4,7 +4,7 @@ use bevy::prelude::*;
use uuid::Uuid; use uuid::Uuid;
use super::{ use super::{
io::{Config, handle_network_input, handle_network_output, heartbeat, timeouts}, io::{Config, handle_network_input, handle_network_output, heartbeat, timeout},
packet::{InboundPacket, OutboundPacket}, packet::{InboundPacket, OutboundPacket},
peer::{Peer, PeerChangeEvent, PeerMap, handle_new_peer, handle_peer_change}, peer::{Peer, PeerChangeEvent, PeerMap, handle_new_peer, handle_peer_change},
queues::{NetworkReceive, NetworkSend}, queues::{NetworkReceive, NetworkSend},
@ -40,7 +40,7 @@ impl Plugin for NetIOPlugin {
) )
.add_systems( .add_systems(
FixedUpdate, FixedUpdate,
(heartbeat, timeouts, handle_new_peer).run_if(in_state(NetworkState::MultiPlayer)), (heartbeat, timeout, handle_new_peer).run_if(in_state(NetworkState::MultiPlayer)),
) )
.add_systems( .add_systems(
FixedPostUpdate, FixedPostUpdate,