fix weather_zone command, and only send local wind once every 30 ticks

This commit is contained in:
Isse 2024-01-23 22:21:33 +01:00
parent b55038a526
commit 8f15dedc3b
6 changed files with 75 additions and 24 deletions

View File

@ -180,10 +180,29 @@ pub struct SiteInfoRich {
struct WeatherLerp { struct WeatherLerp {
old: (SharedWeatherGrid, Instant), old: (SharedWeatherGrid, Instant),
new: (SharedWeatherGrid, Instant), new: (SharedWeatherGrid, Instant),
local_weather: Weather, old_local_wind: (Vec2<f32>, Instant),
new_local_wind: (Vec2<f32>, Instant),
local_wind: Vec2<f32>,
} }
impl WeatherLerp { impl WeatherLerp {
fn local_wind_update(&mut self, wind: Vec2<f32>) {
self.old_local_wind = mem::replace(&mut self.new_local_wind, (wind, Instant::now()));
}
fn update_local_wind(&mut self) {
// Assumes updates are regular
let t = (self.new_local_wind.1.elapsed().as_secs_f32()
/ self
.new_local_wind
.1
.duration_since(self.old_local_wind.1)
.as_secs_f32())
.clamp(0.0, 1.0);
self.local_wind = Vec2::lerp_unclamped(self.old_local_wind.0, self.new_local_wind.0, t);
}
fn weather_update(&mut self, weather: SharedWeatherGrid) { fn weather_update(&mut self, weather: SharedWeatherGrid) {
self.old = mem::replace(&mut self.new, (weather, Instant::now())); self.old = mem::replace(&mut self.new, (weather, Instant::now()));
} }
@ -192,6 +211,7 @@ impl WeatherLerp {
// that updates come at regular intervals. // that updates come at regular intervals.
fn update(&mut self, to_update: &mut WeatherGrid) { fn update(&mut self, to_update: &mut WeatherGrid) {
prof_span!("WeatherLerp::update"); prof_span!("WeatherLerp::update");
self.update_local_wind();
let old = &self.old.0; let old = &self.old.0;
let new = &self.new.0; let new = &self.new.0;
if new.size() == Vec2::zero() { if new.size() == Vec2::zero() {
@ -218,10 +238,14 @@ impl WeatherLerp {
impl Default for WeatherLerp { impl Default for WeatherLerp {
fn default() -> Self { fn default() -> Self {
let old = Instant::now();
let new = Instant::now();
Self { Self {
old: (SharedWeatherGrid::new(Vec2::zero()), Instant::now()), old: (SharedWeatherGrid::new(Vec2::zero()), old),
new: (SharedWeatherGrid::new(Vec2::zero()), Instant::now()), new: (SharedWeatherGrid::new(Vec2::zero()), new),
local_weather: Weather::default(), old_local_wind: (Vec2::zero(), old),
new_local_wind: (Vec2::zero(), new),
local_wind: Vec2::zero(),
} }
} }
} }
@ -1715,7 +1739,15 @@ impl Client {
.map(|v| v.0) .map(|v| v.0)
} }
pub fn weather_at_player(&self) -> Weather { self.weather.local_weather } pub fn weather_at_player(&self) -> Weather {
self.position()
.map(|p| {
let mut weather = self.state.weather_at(p.xy());
weather.wind = self.weather.local_wind;
weather
})
.unwrap_or_default()
}
pub fn current_chunk(&self) -> Option<Arc<TerrainChunk>> { pub fn current_chunk(&self) -> Option<Arc<TerrainChunk>> {
let chunk_pos = Vec2::from(self.position()?) let chunk_pos = Vec2::from(self.position()?)
@ -2536,8 +2568,8 @@ impl Client {
ServerGeneral::WeatherUpdate(weather) => { ServerGeneral::WeatherUpdate(weather) => {
self.weather.weather_update(weather); self.weather.weather_update(weather);
}, },
ServerGeneral::LocalWeatherUpdate(weather) => { ServerGeneral::LocalWindUpdate(wind) => {
self.weather.local_weather = weather; self.weather.local_wind_update(wind);
}, },
ServerGeneral::SpectatePosition(pos) => { ServerGeneral::SpectatePosition(pos) => {
frontend_events.push(Event::SpectatePosition(pos)); frontend_events.push(Event::SpectatePosition(pos));

View File

@ -17,7 +17,7 @@ use common::{
trade::{PendingTrade, SitePrices, TradeId, TradeResult}, trade::{PendingTrade, SitePrices, TradeId, TradeResult},
uid::Uid, uid::Uid,
uuid::Uuid, uuid::Uuid,
weather::{SharedWeatherGrid, Weather}, weather::SharedWeatherGrid,
}; };
use hashbrown::HashMap; use hashbrown::HashMap;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
@ -215,7 +215,7 @@ pub enum ServerGeneral {
SiteEconomy(EconomyInfo), SiteEconomy(EconomyInfo),
MapMarker(comp::MapMarkerUpdate), MapMarker(comp::MapMarkerUpdate),
WeatherUpdate(SharedWeatherGrid), WeatherUpdate(SharedWeatherGrid),
LocalWeatherUpdate(Weather), LocalWindUpdate(Vec2<f32>),
/// Suggest the client to spectate a position. Called after client has /// Suggest the client to spectate a position. Called after client has
/// requested teleport etc. /// requested teleport etc.
SpectatePosition(Vec3<f32>), SpectatePosition(Vec3<f32>),
@ -340,7 +340,7 @@ impl ServerMsg {
| ServerGeneral::SiteEconomy(_) | ServerGeneral::SiteEconomy(_)
| ServerGeneral::MapMarker(_) | ServerGeneral::MapMarker(_)
| ServerGeneral::WeatherUpdate(_) | ServerGeneral::WeatherUpdate(_)
| ServerGeneral::LocalWeatherUpdate(_) | ServerGeneral::LocalWindUpdate(_)
| ServerGeneral::SpectatePosition(_) => { | ServerGeneral::SpectatePosition(_) => {
c_type == ClientType::Game && presence.is_some() c_type == ClientType::Game && presence.is_some()
}, },

View File

@ -192,7 +192,7 @@ impl Client {
| ServerGeneral::FinishedTrade(_) | ServerGeneral::FinishedTrade(_)
| ServerGeneral::MapMarker(_) | ServerGeneral::MapMarker(_)
| ServerGeneral::WeatherUpdate(_) | ServerGeneral::WeatherUpdate(_)
| ServerGeneral::LocalWeatherUpdate(_) | ServerGeneral::LocalWindUpdate(_)
| ServerGeneral::SpectatePosition(_) => { | ServerGeneral::SpectatePosition(_) => {
PreparedMsg::new(2, &g, &self.in_game_stream_params) PreparedMsg::new(2, &g, &self.in_game_stream_params)
}, },

View File

@ -10,7 +10,7 @@ use crate::{
SettingError, WhitelistInfo, WhitelistRecord, SettingError, WhitelistInfo, WhitelistRecord,
}, },
sys::terrain::NpcData, sys::terrain::NpcData,
weather::WeatherSim, weather::WeatherJob,
wiring, wiring,
wiring::OutputFormula, wiring::OutputFormula,
Server, Settings, StateExt, Server, Settings, StateExt,
@ -4474,11 +4474,14 @@ fn handle_weather_zone(
let mut add_zone = |weather: weather::Weather| { let mut add_zone = |weather: weather::Weather| {
if let Ok(pos) = position(server, client, "player") { if let Ok(pos) = position(server, client, "player") {
let pos = pos.0.xy() / weather::CELL_SIZE as f32; let pos = pos.0.xy() / weather::CELL_SIZE as f32;
server if let Some(weather_job) = server
.state .state
.ecs_mut() .ecs_mut()
.write_resource::<WeatherSim>() .write_resource::<Option<WeatherJob>>()
.add_zone(weather, pos, radius, time); .as_mut()
{
weather_job.queue_zone(weather, pos, radius, time);
}
} }
}; };
match name.as_str() { match name.as_str() {

View File

@ -5,7 +5,7 @@ use specs::DispatcherBuilder;
mod sim; mod sim;
mod tick; mod tick;
pub use sim::WeatherSim; pub use tick::WeatherJob;
/// How often the weather is updated, in seconds /// How often the weather is updated, in seconds
const WEATHER_DT: f32 = 5.0; const WEATHER_DT: f32 = 5.0;
@ -18,8 +18,6 @@ pub fn add_server_systems(dispatch_builder: &mut DispatcherBuilder) {
pub fn init(state: &mut State) { pub fn init(state: &mut State) {
use crate::weather::sim::LightningCells; use crate::weather::sim::LightningCells;
use self::tick::WeatherJob;
state.ecs_mut().insert(None::<WeatherJob>); state.ecs_mut().insert(None::<WeatherJob>);
state.ecs_mut().insert(LightningCells::default()); state.ecs_mut().insert(LightningCells::default());
} }

View File

@ -4,16 +4,17 @@ use common::{
outcome::Outcome, outcome::Outcome,
resources::{DeltaTime, ProgramTime, TimeOfDay}, resources::{DeltaTime, ProgramTime, TimeOfDay},
slowjob::{SlowJob, SlowJobPool}, slowjob::{SlowJob, SlowJobPool},
weather::{SharedWeatherGrid, WeatherGrid}, weather::{SharedWeatherGrid, Weather, WeatherGrid},
}; };
use common_ecs::{Origin, Phase, System}; use common_ecs::{Origin, Phase, System};
use common_net::msg::ServerGeneral; use common_net::msg::ServerGeneral;
use rand::{seq::SliceRandom, thread_rng, Rng}; use rand::{seq::SliceRandom, thread_rng, Rng};
use specs::{Join, Read, ReadExpect, ReadStorage, Write, WriteExpect}; use specs::{Entities, Join, Read, ReadExpect, ReadStorage, Write, WriteExpect};
use std::{mem, sync::Arc}; use std::{mem, sync::Arc};
use vek::Vec2;
use world::World; use world::World;
use crate::client::Client; use crate::{client::Client, Tick};
use super::{ use super::{
sim::{LightningCells, WeatherSim}, sim::{LightningCells, WeatherSim},
@ -31,6 +32,13 @@ pub struct WeatherJob {
weather_tx: crossbeam_channel::Sender<(WeatherGrid, LightningCells, WeatherSim)>, weather_tx: crossbeam_channel::Sender<(WeatherGrid, LightningCells, WeatherSim)>,
weather_rx: crossbeam_channel::Receiver<(WeatherGrid, LightningCells, WeatherSim)>, weather_rx: crossbeam_channel::Receiver<(WeatherGrid, LightningCells, WeatherSim)>,
state: WeatherJobState, state: WeatherJobState,
qeued_zones: Vec<(Weather, Vec2<f32>, f32, f32)>,
}
impl WeatherJob {
pub fn queue_zone(&mut self, weather: Weather, pos: Vec2<f32>, radius: f32, time: f32) {
self.qeued_zones.push((weather, pos, radius, time))
}
} }
#[derive(Default)] #[derive(Default)]
@ -38,8 +46,10 @@ pub struct Sys;
impl<'a> System<'a> for Sys { impl<'a> System<'a> for Sys {
type SystemData = ( type SystemData = (
Entities<'a>,
Read<'a, TimeOfDay>, Read<'a, TimeOfDay>,
Read<'a, ProgramTime>, Read<'a, ProgramTime>,
Read<'a, Tick>,
Read<'a, DeltaTime>, Read<'a, DeltaTime>,
Write<'a, LightningCells>, Write<'a, LightningCells>,
Write<'a, Option<WeatherJob>>, Write<'a, Option<WeatherJob>>,
@ -58,8 +68,10 @@ impl<'a> System<'a> for Sys {
fn run( fn run(
_job: &mut common_ecs::Job<Self>, _job: &mut common_ecs::Job<Self>,
( (
entities,
game_time, game_time,
program_time, program_time,
tick,
delta_time, delta_time,
mut lightning_cells, mut lightning_cells,
mut weather_job, mut weather_job,
@ -87,6 +99,7 @@ impl<'a> System<'a> for Sys {
weather_tx, weather_tx,
weather_rx, weather_rx,
state: WeatherJobState::Idle(sim), state: WeatherJobState::Idle(sim),
qeued_zones: Vec::new(),
}); });
None None
@ -118,6 +131,9 @@ impl<'a> System<'a> for Sys {
let weather_tx = weather_job.weather_tx.clone(); let weather_tx = weather_job.weather_tx.clone();
let game_time = *game_time; let game_time = *game_time;
for (weather, pos, radius, time) in weather_job.qeued_zones.drain(..) {
sim.add_zone(weather, pos, radius, time)
}
let job = slow_job_pool.spawn("WEATHER", move || { let job = slow_job_pool.spawn("WEATHER", move || {
let mut grid = WeatherGrid::new(sim.size()); let mut grid = WeatherGrid::new(sim.size());
let lightning_cells = sim.tick(game_time, &mut grid); let lightning_cells = sim.tick(game_time, &mut grid);
@ -148,9 +164,11 @@ impl<'a> System<'a> for Sys {
}); });
} }
for (client, pos) in (&clients, &positions).join() { for (entity, client, pos) in (&entities, &clients, &positions).join() {
let weather = grid.get_interpolated(pos.0.xy()); if entity.id() as u64 % 30 == tick.0 % 30 {
client.send_fallible(ServerGeneral::LocalWeatherUpdate(weather)); let weather = grid.get_interpolated(pos.0.xy());
client.send_fallible(ServerGeneral::LocalWindUpdate(weather.wind));
}
} }
} }
} }