Fix knockback for light entities; add wing state; increase max drag

This commit is contained in:
Ludvig Böklin 2021-05-15 13:16:57 +02:00
parent 8702a5eb9a
commit 9e9b82b6b8
5 changed files with 125 additions and 48 deletions

View File

@ -184,18 +184,16 @@ impl Body {
}, },
Body::BipedSmall(_) => 50.0, Body::BipedSmall(_) => 50.0,
// ravens are 0.69-2 kg, crows are 0.51 kg on average // Ravens are 0.69-2 kg, crows are 0.51 kg on average. Our birds are quite large
Body::BirdMedium(_) => 1.0, Body::BirdMedium(_) => 3.0,
Body::BirdLarge(_) => 200.0, Body::BirdLarge(_) => 50.0,
Body::Dragon(_) => 20_000.0, Body::Dragon(_) => 20_000.0,
Body::FishMedium(_) => 2.5, Body::FishMedium(_) => 5.0,
Body::FishSmall(_) => 1.0, Body::FishSmall(_) => 1.0,
Body::Golem(_) => 10_000.0, Body::Golem(_) => 10_000.0,
Body::Humanoid(humanoid) => { Body::Humanoid(humanoid) => {
// humanoids are quite a bit larger than in real life, so we multiply their mass match (humanoid.species, humanoid.body_type) {
// to scale it up proportionally (remember cube law)
1.0 * match (humanoid.species, humanoid.body_type) {
(humanoid::Species::Orc, humanoid::BodyType::Male) => 120.0, (humanoid::Species::Orc, humanoid::BodyType::Male) => 120.0,
(humanoid::Species::Orc, humanoid::BodyType::Female) => 120.0, (humanoid::Species::Orc, humanoid::BodyType::Female) => 120.0,
(humanoid::Species::Human, humanoid::BodyType::Male) => 77.0, // ~✅ (humanoid::Species::Human, humanoid::BodyType::Male) => 77.0, // ~✅
@ -240,13 +238,33 @@ impl Body {
_ => 200.0, _ => 200.0,
}, },
Body::QuadrupedSmall(body) => match body.species { 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::Boar => 80.0, // ~✅ (60-100 kg)
quadruped_small::Species::Dodarock => 150.0, quadruped_small::Species::Cat => 4.0, // ~✅ (4-5 kg)
quadruped_small::Species::Holladon => 150.0, 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::Hyena => 70.0, // ~✅ (vaguely)
quadruped_small::Species::Truffler => 150.0, quadruped_small::Species::Jackalope => 10.0,
_ => 80.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 { Body::Theropod(body) => match body.species {
// for reference, elephants are in the range of 2.6-6.9 tons // for reference, elephants are in the range of 2.6-6.9 tons

View File

@ -349,7 +349,7 @@ impl Body {
Body::BoltFire => Vec3::new(0.1, 0.1, 0.1), Body::BoltFire => Vec3::new(0.1, 0.1, 0.1),
Body::Crossbow => Vec3::new(3.0, 3.0, 1.5), Body::Crossbow => Vec3::new(3.0, 3.0, 1.5),
Body::HaniwaSentry => Vec3::new(0.8, 0.8, 1.4), Body::HaniwaSentry => Vec3::new(0.8, 0.8, 1.4),
_ => Vec3::broadcast(0.2), _ => Vec3::broadcast(0.5),
} }
} }
} }

View File

@ -1,6 +1,6 @@
use super::{ use super::{
body::{object, Body}, body::{object, Body},
CharacterState, Density, Ori, Vel, Density, Ori, Vel,
}; };
use crate::{ use crate::{
consts::{AIR_DENSITY, WATER_DENSITY}, consts::{AIR_DENSITY, WATER_DENSITY},
@ -87,12 +87,22 @@ impl Default for Fluid {
} }
} }
pub enum Wings {
Flying,
Gliding {
aspect_ratio: f32,
planform_area: f32,
ori: Ori,
},
Folded,
}
impl Body { impl Body {
pub fn aerodynamic_forces( pub fn aerodynamic_forces(
&self, &self,
rel_flow: &Vel, rel_flow: &Vel,
fluid_density: f32, fluid_density: f32,
character_state: Option<&CharacterState>, wings: Option<&Wings>,
) -> Vec3<f32> { ) -> Vec3<f32> {
let v_sq = rel_flow.0.magnitude_squared(); let v_sq = rel_flow.0.magnitude_squared();
if v_sq < 0.25 { if v_sq < 0.25 {
@ -103,23 +113,24 @@ impl Body {
// All the coefficients come pre-multiplied by their reference area // All the coefficients come pre-multiplied by their reference area
0.5 * fluid_density 0.5 * fluid_density
* v_sq * v_sq
* character_state * wings
.and_then(|cs| match cs { .and_then(|wings| match *wings {
CharacterState::Glide(data) => { Wings::Gliding {
Some((data.aspect_ratio, data.planform_area, data.ori)) 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.",
aspect_ratio
)
};
Some((aspect_ratio.min(24.0), planform_area, ori))
}, },
_ => None, _ => None,
}) })
.map(|(ar, area, ori)| {
if ar > 25.0 {
tracing::warn!(
"Calculating lift for wings with an aspect ratio of {}. The \
formulas are only valid for aspect ratios below 25.",
ar
)
};
(ar.min(24.0), area, ori)
})
.map(|(ar, area, ori)| { .map(|(ar, area, ori)| {
// We have an elliptical wing; proceed to calculate its lift and drag // We have an elliptical wing; proceed to calculate its lift and drag
@ -158,7 +169,7 @@ impl Body {
let e = 1.78 * (1.0 - 0.045 * ar.powf(0.68)) - 0.64; let e = 1.78 * (1.0 - 0.045 * ar.powf(0.68)) - 0.64;
zero_lift_drag_coefficient(area) zero_lift_drag_coefficient(area)
+ self.parasite_drag_coefficient() + self.parasite_drag_coefficient(wings)
+ c_l.powi(2) / (PI * e * ar) + c_l.powi(2) / (PI * e * ar)
}; };
debug_assert!(c_d.is_sign_positive()); debug_assert!(c_d.is_sign_positive());
@ -166,7 +177,7 @@ impl Body {
c_l * *lift_dir + c_d * *rel_flow_dir c_l * *lift_dir + c_d * *rel_flow_dir
}) })
.unwrap_or_else(|| self.parasite_drag_coefficient() * *rel_flow_dir) .unwrap_or_else(|| self.parasite_drag_coefficient(wings) * *rel_flow_dir)
} }
} }
@ -174,7 +185,7 @@ impl Body {
/// Skin friction is the drag arising from the shear forces between a fluid /// 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 /// and a surface, while pressure drag is due to flow separation. Both are
/// viscous effects. /// 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 // Reference area and drag coefficient assumes best-case scenario of the
// orientation producing least amount of drag // orientation producing least amount of drag
match self { match self {
@ -201,12 +212,16 @@ impl Body {
// Cross-section, zero-lift angle; exclude the wings (width * 0.2) // Cross-section, zero-lift angle; exclude the wings (width * 0.2)
Body::BirdMedium(_) | Body::BirdLarge(_) | Body::Dragon(_) => { Body::BirdMedium(_) | Body::BirdLarge(_) | Body::Dragon(_) => {
let dim = self.dimensions().map(|a| a * 0.5); let dim = self.dimensions().map(|a| a * 0.5);
// "Field Estimates of Body Drag Coefficient on the Basis of Dives in Passerine let cd = if matches!(wings, Some(Wings::Folded) | None) {
// Birds", Anders Hedenström and Felix Liechti, 2001 0.7
let cd = match self { } else {
Body::BirdLarge(_) | Body::BirdMedium(_) => 0.2, // "Field Estimates of Body Drag Coefficient on the Basis of Dives in Passerine
// arbitrary // Birds", Anders Hedenström and Felix Liechti, 2001
_ => 0.7, match self {
Body::BirdLarge(_) | Body::BirdMedium(_) => 0.2,
// arbitrary
_ => 0.7,
}
}; };
cd * PI * dim.x * 0.2 * dim.z cd * PI * dim.x * 0.2 * dim.z
}, },

View File

@ -1,14 +1,16 @@
use common::{ use common::{
comp::{ comp::{
body::ship::figuredata::{VoxelCollider, VOXEL_COLLIDER_MANIFEST}, body::ship::figuredata::{VoxelCollider, VOXEL_COLLIDER_MANIFEST},
BeamSegment, Body, CharacterState, Collider, Density, Fluid, Mass, Mounting, Ori, fluid_dynamics::{Fluid, Wings},
PhysicsState, Pos, PosVelDefer, PreviousPhysCache, Projectile, Scale, Shockwave, Sticky, BeamSegment, Body, CharacterState, Collider, Controller, Density, InputKind, Mass,
Vel, Mounting, Ori, PhysicsState, Pos, PosVelDefer, PreviousPhysCache, Projectile, Scale,
Shockwave, Sticky, Vel,
}, },
consts::{AIR_DENSITY, FRIC_GROUND, GRAVITY}, consts::{AIR_DENSITY, FRIC_GROUND, GRAVITY},
event::{EventBus, ServerEvent}, event::{EventBus, ServerEvent},
outcome::Outcome, outcome::Outcome,
resources::DeltaTime, resources::DeltaTime,
states,
terrain::{Block, TerrainGrid}, terrain::{Block, TerrainGrid},
uid::Uid, uid::Uid,
util::{Projection, SpatialGrid}, util::{Projection, SpatialGrid},
@ -38,14 +40,44 @@ fn fluid_density(height: f32, fluid: &Fluid) -> Density {
Density(fluid.density().0 * immersion + AIR_DENSITY * (1.0 - immersion)) Density(fluid.density().0 * immersion + AIR_DENSITY * (1.0 - immersion))
} }
fn get_wings(
character_state: Option<&CharacterState>,
controller: Option<&Controller>,
body: &Body,
) -> Option<Wings> {
match *character_state? {
CharacterState::Glide(states::glide::Data {
aspect_ratio,
planform_area,
ori,
..
}) => Some(Wings::Gliding {
aspect_ratio,
planform_area,
ori,
}),
_ => {
if body.fly_thrust().is_some() {
if controller?.queued_inputs.contains_key(&InputKind::Fly) {
Some(Wings::Flying)
} else {
Some(Wings::Folded)
}
} else {
None
}
},
}
}
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
fn integrate_forces( fn integrate_forces(
dt: &DeltaTime, dt: &DeltaTime,
mut vel: Vel, mut vel: Vel,
body: &Body, (body, wings): (&Body, Option<&Wings>),
density: &Density, density: &Density,
mass: &Mass, mass: &Mass,
character_state: Option<&CharacterState>,
fluid: &Fluid, fluid: &Fluid,
gravity: f32, gravity: f32,
) -> Vel { ) -> Vel {
@ -59,7 +91,7 @@ fn integrate_forces(
// Aerodynamic/hydrodynamic forces // Aerodynamic/hydrodynamic forces
if !rel_flow.0.is_approx_zero() { if !rel_flow.0.is_approx_zero() {
debug_assert!(!rel_flow.0.map(|a| a.is_nan()).reduce_or()); 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()); debug_assert!(!impulse.map(|a| a.is_nan()).reduce_or());
if !impulse.is_approx_zero() { if !impulse.is_approx_zero() {
let new_v = vel.0 + impulse / mass.0; let new_v = vel.0 + impulse / mass.0;
@ -71,7 +103,7 @@ fn integrate_forces(
if new_v.dot(vel.0) < 0.0 { if new_v.dot(vel.0) < 0.0 {
// Multiply by a factor to prevent full stop, as this can cause things to get // Multiply by a factor to prevent full stop, as this can cause things to get
// stuck in high-density medium // stuck in high-density medium
vel.0 -= vel.0.projected(&impulse) * 0.7; vel.0 -= vel.0.projected(&impulse) * 0.9;
} else { } else {
vel.0 = new_v; vel.0 = new_v;
} }
@ -116,6 +148,7 @@ pub struct PhysicsRead<'a> {
stickies: ReadStorage<'a, Sticky>, stickies: ReadStorage<'a, Sticky>,
masses: ReadStorage<'a, Mass>, masses: ReadStorage<'a, Mass>,
colliders: ReadStorage<'a, Collider>, colliders: ReadStorage<'a, Collider>,
controllers: ReadStorage<'a, Controller>,
mountings: ReadStorage<'a, Mounting>, mountings: ReadStorage<'a, Mounting>,
projectiles: ReadStorage<'a, Projectile>, projectiles: ReadStorage<'a, Projectile>,
beams: ReadStorage<'a, BeamSegment>, beams: ReadStorage<'a, BeamSegment>,
@ -574,6 +607,7 @@ impl<'a> PhysicsData<'a> {
&read.bodies, &read.bodies,
read.character_states.maybe(), read.character_states.maybe(),
&write.physics_states, &write.physics_states,
read.controllers.maybe(),
&read.masses, &read.masses,
&read.densities, &read.densities,
!&read.mountings, !&read.mountings,
@ -592,6 +626,7 @@ impl<'a> PhysicsData<'a> {
body, body,
character_state, character_state,
physics_state, physics_state,
controller,
mass, mass,
density, density,
_, _,
@ -616,13 +651,13 @@ impl<'a> PhysicsData<'a> {
vel.0.z -= dt.0 * GRAVITY; vel.0.z -= dt.0 * GRAVITY;
}, },
Some(fluid) => { Some(fluid) => {
let wings = get_wings(character_state, controller, body);
vel.0 = integrate_forces( vel.0 = integrate_forces(
&dt, &dt,
*vel, *vel,
body, (body, wings.as_ref()),
density, density,
mass, mass,
character_state,
&fluid, &fluid,
GRAVITY, GRAVITY,
) )

View File

@ -75,12 +75,21 @@ pub fn handle_knockback(server: &Server, entity: EcsEntity, impulse: Vec3<f32>)
0.4 0.4
}; };
if let Some(mass) = ecs.read_storage::<comp::Mass>().get(entity) { if let Some(mass) = ecs.read_storage::<comp::Mass>().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::<comp::Vel>(); let mut velocities = ecs.write_storage::<comp::Vel>();
if let Some(vel) = velocities.get_mut(entity) { if let Some(vel) = velocities.get_mut(entity) {
vel.0 += impulse; vel.0 += impulse;
} }
/*
// Breaks bird AI (they need to learn how to toggle flight first)
if let Some(controller) = ecs.write_storage::<comp::Controller>().get_mut(entity) {
controller
.actions
.push(comp::ControlAction::CancelInput(comp::InputKind::Fly));
}
*/
if let Some(client) = clients.get(entity) { if let Some(client) = clients.get(entity) {
client.send_fallible(ServerGeneral::Knockback(impulse)); client.send_fallible(ServerGeneral::Knockback(impulse));
} }