From d7fa4e093d9f2715bd60aa56323c5458f5b86956 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ludvig=20B=C3=B6klin?= Date: Sat, 22 May 2021 17:56:13 +0000 Subject: [PATCH] Adjusted masses; less excessive knockbacks; prevent loot shooting off --- common/src/comp/body.rs | 50 ++++++++++---- common/src/comp/body/object.rs | 2 +- common/src/comp/fluid_dynamics.rs | 85 ++++++++++++++---------- common/systems/src/phys.rs | 31 ++++++--- server/src/events/entity_manipulation.rs | 3 +- 5 files changed, 113 insertions(+), 58 deletions(-) diff --git a/common/src/comp/body.rs b/common/src/comp/body.rs index 21411bd035..82b0d3d46f 100644 --- a/common/src/comp/body.rs +++ b/common/src/comp/body.rs @@ -184,18 +184,24 @@ impl Body { }, Body::BipedSmall(_) => 50.0, - // ravens are 0.69-2 kg, crows are 0.51 kg on average - Body::BirdMedium(_) => 1.0, - Body::BirdLarge(_) => 200.0, + // ravens are 0.69-2 kg, crows are 0.51 kg on average. + Body::BirdMedium(body) => match body.species { + bird_medium::Species::Chicken => 2.0, // ~✅ Red junglefowl are 1-1.5 kg + bird_medium::Species::Duck => 2.0, + bird_medium::Species::Eagle => 10.0, // ~✅ Steller's sea eagle are 5-9 kg + bird_medium::Species::Goose => 3.5, // ~✅ Swan geese are 2.8-3.5 kg + bird_medium::Species::Owl => 2.0, + bird_medium::Species::Parrot => 2.0, + bird_medium::Species::Peacock => 5.0, + }, + Body::BirdLarge(_) => 100.0, Body::Dragon(_) => 20_000.0, - Body::FishMedium(_) => 2.5, + Body::FishMedium(_) => 5.0, Body::FishSmall(_) => 1.0, Body::Golem(_) => 10_000.0, Body::Humanoid(humanoid) => { - // humanoids are quite a bit larger than in real life, so we multiply their mass - // to scale it up proportionally (remember cube law) - 1.0 * match (humanoid.species, humanoid.body_type) { + match (humanoid.species, humanoid.body_type) { (humanoid::Species::Orc, humanoid::BodyType::Male) => 120.0, (humanoid::Species::Orc, humanoid::BodyType::Female) => 120.0, (humanoid::Species::Human, humanoid::BodyType::Male) => 77.0, // ~✅ @@ -240,13 +246,33 @@ impl Body { _ => 200.0, }, Body::QuadrupedSmall(body) => match body.species { - quadruped_small::Species::Batfox => 50.0, + quadruped_small::Species::Axolotl => 1.0, + quadruped_small::Species::Batfox => 10.0, + quadruped_small::Species::Beaver => 10.0, quadruped_small::Species::Boar => 80.0, // ~✅ (60-100 kg) - quadruped_small::Species::Dodarock => 150.0, - quadruped_small::Species::Holladon => 150.0, + quadruped_small::Species::Cat => 4.0, // ~✅ (4-5 kg) + quadruped_small::Species::Dodarock => 500.0, + quadruped_small::Species::Dog => 30.0, // ~✅ (German Shepherd: 30-40 kg) + quadruped_small::Species::Fox => 10.0, + quadruped_small::Species::Frog => 1.0, + quadruped_small::Species::Fungome => 10.0, + quadruped_small::Species::Gecko => 1.0, + quadruped_small::Species::Goat => 50.0, + quadruped_small::Species::Hare => 10.0, + quadruped_small::Species::Holladon => 60.0, quadruped_small::Species::Hyena => 70.0, // ~✅ (vaguely) - quadruped_small::Species::Truffler => 150.0, - _ => 80.0, + quadruped_small::Species::Jackalope => 10.0, + quadruped_small::Species::Pig => 20.0, + quadruped_small::Species::Porcupine => 5.0, + quadruped_small::Species::Quokka => 10.0, + quadruped_small::Species::Rabbit => 2.0, + quadruped_small::Species::Raccoon => 30.0, + quadruped_small::Species::Rat => 1.0, + quadruped_small::Species::Sheep => 50.0, + quadruped_small::Species::Skunk => 5.0, + quadruped_small::Species::Squirrel => 1.0, + quadruped_small::Species::Truffler => 70.0, + quadruped_small::Species::Turtle => 40.0, }, Body::Theropod(body) => match body.species { // for reference, elephants are in the range of 2.6-6.9 tons diff --git a/common/src/comp/body/object.rs b/common/src/comp/body/object.rs index d5a00fcdc2..6de2c5ca86 100644 --- a/common/src/comp/body/object.rs +++ b/common/src/comp/body/object.rs @@ -349,7 +349,7 @@ impl Body { Body::BoltFire => Vec3::new(0.1, 0.1, 0.1), Body::Crossbow => Vec3::new(3.0, 3.0, 1.5), Body::HaniwaSentry => Vec3::new(0.8, 0.8, 1.4), - _ => Vec3::broadcast(0.2), + _ => Vec3::broadcast(0.5), } } } diff --git a/common/src/comp/fluid_dynamics.rs b/common/src/comp/fluid_dynamics.rs index d80a277997..fc85b415d4 100644 --- a/common/src/comp/fluid_dynamics.rs +++ b/common/src/comp/fluid_dynamics.rs @@ -1,6 +1,6 @@ use super::{ body::{object, Body}, - CharacterState, Density, Ori, Vel, + Density, Ori, Vel, }; use crate::{ consts::{AIR_DENSITY, WATER_DENSITY}, @@ -87,12 +87,18 @@ impl Default for Fluid { } } +pub struct Wings { + pub aspect_ratio: f32, + pub planform_area: f32, + pub ori: Ori, +} + impl Body { pub fn aerodynamic_forces( &self, rel_flow: &Vel, fluid_density: f32, - character_state: Option<&CharacterState>, + wings: Option<&Wings>, ) -> Vec3 { let v_sq = rel_flow.0.magnitude_squared(); if v_sq < 0.25 { @@ -103,31 +109,27 @@ impl Body { // All the coefficients come pre-multiplied by their reference area 0.5 * fluid_density * v_sq - * character_state - .and_then(|cs| match cs { - CharacterState::Glide(data) => { - Some((data.aspect_ratio, data.planform_area, data.ori)) - }, - _ => None, - }) - .map(|(ar, area, ori)| { - if ar > 25.0 { + * match wings { + Some(&Wings { + aspect_ratio, + planform_area, + ori, + }) => { + if aspect_ratio > 25.0 { tracing::warn!( "Calculating lift for wings with an aspect ratio of {}. The \ formulas are only valid for aspect ratios below 25.", - ar + aspect_ratio ) }; - (ar.min(24.0), area, ori) - }) - .map(|(ar, area, ori)| { + let ar = aspect_ratio.min(24.0); // We have an elliptical wing; proceed to calculate its lift and drag // aoa will be positive when we're pitched up and negative otherwise let aoa = angle_of_attack(&ori, &rel_flow_dir); // c_l will be positive when aoa is positive (we have positive lift, // producing an upward force) and negative otherwise - let c_l = lift_coefficient(ar, area, aoa); + let c_l = lift_coefficient(ar, planform_area, aoa); // lift dir will be orthogonal to the local relative flow vector. // Local relative flow is the resulting vector of (relative) freestream @@ -151,22 +153,26 @@ impl Body { ori.pitched_down(aoa_eff).up() }; - // drag coefficient due to lift + // drag coefficient let c_d = { // Oswald's efficiency factor (empirically derived--very magical) // (this definition should not be used for aspect ratios > 25) let e = 1.78 * (1.0 - 0.045 * ar.powf(0.68)) - 0.64; + // induced drag coefficient (drag due to lift) + let cdi = c_l.powi(2) / (PI * e * ar); - zero_lift_drag_coefficient(area) - + self.parasite_drag_coefficient() - + c_l.powi(2) / (PI * e * ar) + zero_lift_drag_coefficient(planform_area) + + self.parasite_drag_coefficient(wings) + + cdi }; debug_assert!(c_d.is_sign_positive()); debug_assert!(c_l.is_sign_positive() || aoa.is_sign_negative()); c_l * *lift_dir + c_d * *rel_flow_dir - }) - .unwrap_or_else(|| self.parasite_drag_coefficient() * *rel_flow_dir) + }, + + _ => self.parasite_drag_coefficient(wings) * *rel_flow_dir, + } } } @@ -174,14 +180,15 @@ impl Body { /// Skin friction is the drag arising from the shear forces between a fluid /// and a surface, while pressure drag is due to flow separation. Both are /// viscous effects. - fn parasite_drag_coefficient(&self) -> f32 { + fn parasite_drag_coefficient(&self, wings: Option<&Wings>) -> f32 { // Reference area and drag coefficient assumes best-case scenario of the // orientation producing least amount of drag match self { // Cross-section, head/feet first Body::BipedLarge(_) | Body::BipedSmall(_) | Body::Golem(_) | Body::Humanoid(_) => { let dim = self.dimensions().xy().map(|a| a * 0.5); - 0.7 * PI * dim.x * dim.y + const CD: f32 = 0.7; + CD * PI * dim.x * dim.y }, // Cross-section, nose/tail first @@ -190,7 +197,7 @@ impl Body { | Body::QuadrupedSmall(_) | Body::QuadrupedLow(_) => { let dim = self.dimensions().map(|a| a * 0.5); - let cd = if matches!(self, Body::QuadrupedLow(_)) { + let cd: f32 = if matches!(self, Body::QuadrupedLow(_)) { 0.7 } else { 1.0 @@ -201,12 +208,16 @@ impl Body { // Cross-section, zero-lift angle; exclude the wings (width * 0.2) Body::BirdMedium(_) | Body::BirdLarge(_) | Body::Dragon(_) => { let dim = self.dimensions().map(|a| a * 0.5); - // "Field Estimates of Body Drag Coefficient on the Basis of Dives in Passerine - // Birds", Anders Hedenström and Felix Liechti, 2001 - let cd = match self { - Body::BirdLarge(_) | Body::BirdMedium(_) => 0.2, - // arbitrary - _ => 0.7, + let cd: f32 = if wings.is_none() { + 0.7 + } else { + // "Field Estimates of Body Drag Coefficient on the Basis of Dives in Passerine + // Birds", Anders Hedenström and Felix Liechti, 2001 + match self { + Body::BirdLarge(_) | Body::BirdMedium(_) => 0.2, + // arbitrary + _ => 0.7, + } }; cd * PI * dim.x * 0.2 * dim.z }, @@ -216,7 +227,8 @@ impl Body { let dim = self.dimensions().map(|a| a * 0.5); // "A Simple Method to Determine Drag Coefficients in Aquatic Animals", // D. Bilo and W. Nachtigall, 1980 - 0.031 * PI * dim.x * 0.2 * dim.z + const CD: f32 = 0.031; + CD * PI * dim.x * 0.2 * dim.z }, Body::Object(object) => match object { @@ -232,7 +244,8 @@ impl Body { | object::Body::FireworkYellow | object::Body::MultiArrow => { let dim = self.dimensions().map(|a| a * 0.5); - 0.02 * PI * dim.x * dim.z + const CD: f32 = 0.02; + CD * PI * dim.x * dim.z }, // spherical-ish objects @@ -250,12 +263,14 @@ impl Body { | object::Body::Pumpkin4 | object::Body::Pumpkin5 => { let dim = self.dimensions().map(|a| a * 0.5); - 0.5 * PI * dim.x * dim.z + const CD: f32 = 0.5; + CD * PI * dim.x * dim.z }, _ => { let dim = self.dimensions(); - 2.0 * (PI / 6.0 * dim.x * dim.y * dim.z).powf(2.0 / 3.0) + const CD: f32 = 2.0; + CD * (PI / 6.0 * dim.x * dim.y * dim.z).powf(2.0 / 3.0) }, }, diff --git a/common/systems/src/phys.rs b/common/systems/src/phys.rs index 65cddd045d..6b20254f94 100644 --- a/common/systems/src/phys.rs +++ b/common/systems/src/phys.rs @@ -1,14 +1,15 @@ use common::{ comp::{ body::ship::figuredata::{VoxelCollider, VOXEL_COLLIDER_MANIFEST}, - BeamSegment, Body, CharacterState, Collider, Density, Fluid, Mass, Mounting, Ori, - PhysicsState, Pos, PosVelDefer, PreviousPhysCache, Projectile, Scale, Shockwave, Sticky, - Vel, + fluid_dynamics::{Fluid, Wings}, + BeamSegment, Body, CharacterState, Collider, Density, Mass, Mounting, Ori, PhysicsState, + Pos, PosVelDefer, PreviousPhysCache, Projectile, Scale, Shockwave, Sticky, Vel, }, consts::{AIR_DENSITY, FRIC_GROUND, GRAVITY}, event::{EventBus, ServerEvent}, outcome::Outcome, resources::DeltaTime, + states, terrain::{Block, TerrainGrid}, uid::Uid, util::{Projection, SpatialGrid}, @@ -42,10 +43,9 @@ fn fluid_density(height: f32, fluid: &Fluid) -> Density { fn integrate_forces( dt: &DeltaTime, mut vel: Vel, - body: &Body, + (body, wings): (&Body, Option<&Wings>), density: &Density, mass: &Mass, - character_state: Option<&CharacterState>, fluid: &Fluid, gravity: f32, ) -> Vel { @@ -59,7 +59,7 @@ fn integrate_forces( // Aerodynamic/hydrodynamic forces if !rel_flow.0.is_approx_zero() { debug_assert!(!rel_flow.0.map(|a| a.is_nan()).reduce_or()); - let impulse = dt.0 * body.aerodynamic_forces(&rel_flow, fluid_density.0, character_state); + let impulse = dt.0 * body.aerodynamic_forces(&rel_flow, fluid_density.0, wings); debug_assert!(!impulse.map(|a| a.is_nan()).reduce_or()); if !impulse.is_approx_zero() { let new_v = vel.0 + impulse / mass.0; @@ -71,7 +71,7 @@ fn integrate_forces( if new_v.dot(vel.0) < 0.0 { // Multiply by a factor to prevent full stop, as this can cause things to get // stuck in high-density medium - vel.0 -= vel.0.projected(&impulse) * 0.7; + vel.0 -= vel.0.projected(&impulse) * 0.9; } else { vel.0 = new_v; } @@ -616,13 +616,26 @@ impl<'a> PhysicsData<'a> { vel.0.z -= dt.0 * GRAVITY; }, Some(fluid) => { + let wings = match character_state { + Some(&CharacterState::Glide(states::glide::Data { + aspect_ratio, + planform_area, + ori, + .. + })) => Some(Wings { + aspect_ratio, + planform_area, + ori, + }), + + _ => None, + }; vel.0 = integrate_forces( &dt, *vel, - body, + (body, wings.as_ref()), density, mass, - character_state, &fluid, GRAVITY, ) diff --git a/server/src/events/entity_manipulation.rs b/server/src/events/entity_manipulation.rs index 33fcc05826..13c2ceb532 100644 --- a/server/src/events/entity_manipulation.rs +++ b/server/src/events/entity_manipulation.rs @@ -75,7 +75,8 @@ pub fn handle_knockback(server: &Server, entity: EcsEntity, impulse: Vec3) 0.4 }; if let Some(mass) = ecs.read_storage::().get(entity) { - impulse /= mass.0; + // we go easy on the little ones (because they fly so far) + impulse /= mass.0.max(40.0); } let mut velocities = ecs.write_storage::(); if let Some(vel) = velocities.get_mut(entity) {