mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
214 lines
8.1 KiB
Rust
214 lines
8.1 KiB
Rust
use super::utils::*;
|
|
use crate::{
|
|
comp::{
|
|
character_state::OutputEvents, fluid_dynamics::angle_of_attack, inventory::slot::EquipSlot,
|
|
CharacterState, Ori, StateUpdate, Vel,
|
|
},
|
|
event::LocalEvent,
|
|
outcome::Outcome,
|
|
states::{
|
|
behavior::{CharacterBehavior, JoinData},
|
|
glide_wield, idle,
|
|
},
|
|
util::{Dir, Plane, Projection},
|
|
};
|
|
use serde::{Deserialize, Serialize};
|
|
use std::{f32::consts::PI, time::Duration};
|
|
use vek::*;
|
|
|
|
const PITCH_SLOW_TIME: f32 = 0.5;
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize)]
|
|
pub struct Data {
|
|
/// The aspect ratio is the ratio of the span squared to actual planform
|
|
/// area
|
|
pub aspect_ratio: f32,
|
|
pub planform_area: f32,
|
|
pub ori: Ori,
|
|
last_vel: Vel,
|
|
pub timer: Duration,
|
|
inputs_disabled: bool,
|
|
}
|
|
|
|
impl Data {
|
|
/// A glider is modelled as an elliptical wing and has a span length
|
|
/// (distance from wing tip to wing tip) and a chord length (distance from
|
|
/// leading edge to trailing edge through its centre) measured in block
|
|
/// units.
|
|
///
|
|
/// https://en.wikipedia.org/wiki/Elliptical_wing
|
|
pub fn new(span_length: f32, chord_length: f32, ori: Ori) -> Self {
|
|
let planform_area = PI * chord_length * span_length * 0.25;
|
|
let aspect_ratio = span_length.powi(2) / planform_area;
|
|
Self {
|
|
aspect_ratio,
|
|
planform_area,
|
|
ori,
|
|
last_vel: Vel::zero(),
|
|
timer: Duration::default(),
|
|
inputs_disabled: true,
|
|
}
|
|
}
|
|
|
|
fn tgt_dir(&self, data: &JoinData) -> Dir {
|
|
let move_dir = if self.inputs_disabled {
|
|
Vec2::zero()
|
|
} else {
|
|
data.inputs.move_dir
|
|
};
|
|
let look_ori = Ori::from(data.inputs.look_dir);
|
|
look_ori
|
|
.yawed_right(PI / 3.0 * look_ori.right().xy().dot(move_dir))
|
|
.pitched_up(PI * 0.04)
|
|
.pitched_down(
|
|
data.inputs
|
|
.look_dir
|
|
.xy()
|
|
.try_normalized()
|
|
.map_or(0.0, |ld| {
|
|
PI * 0.1 * ld.dot(move_dir) * self.timer.as_secs_f32().min(PITCH_SLOW_TIME)
|
|
/ PITCH_SLOW_TIME
|
|
}),
|
|
)
|
|
.look_dir()
|
|
}
|
|
}
|
|
|
|
impl CharacterBehavior for Data {
|
|
fn behavior(&self, data: &JoinData, _: &mut OutputEvents) -> StateUpdate {
|
|
let mut update = StateUpdate::from(data);
|
|
|
|
// If player is on ground, end glide
|
|
if data.physics.on_ground.is_some()
|
|
&& (data.vel.0 - data.physics.ground_vel).magnitude_squared() < 2_f32.powi(2)
|
|
{
|
|
update.character = CharacterState::GlideWield(glide_wield::Data::from(data));
|
|
} else if data.physics.in_liquid().is_some()
|
|
|| data
|
|
.inventory
|
|
.and_then(|inv| inv.equipped(EquipSlot::Glider))
|
|
.is_none()
|
|
{
|
|
update.character = CharacterState::Idle(idle::Data::default());
|
|
} else if !handle_climb(data, &mut update) {
|
|
let air_flow = data
|
|
.physics
|
|
.in_fluid
|
|
.map(|fluid| fluid.relative_flow(data.vel))
|
|
.unwrap_or_default();
|
|
|
|
let inputs_disabled = self.inputs_disabled && !data.inputs.move_dir.is_approx_zero();
|
|
|
|
let ori = {
|
|
let slerp_s = {
|
|
let angle = self.ori.look_dir().angle_between(*data.inputs.look_dir);
|
|
let rate = 0.4 * PI / angle;
|
|
(data.dt.0 * rate).min(1.0)
|
|
};
|
|
|
|
Dir::from_unnormalized(air_flow.0)
|
|
.map(|flow_dir| {
|
|
let tgt_dir = self.tgt_dir(data);
|
|
let tgt_dir_ori = Ori::from(tgt_dir);
|
|
let tgt_dir_up = tgt_dir_ori.up();
|
|
// The desired up vector of our glider.
|
|
// We begin by projecting the flow dir on the plane with the normal of
|
|
// our tgt_dir to get an idea of how it will hit the glider
|
|
let tgt_up = flow_dir
|
|
.projected(&Plane::from(tgt_dir))
|
|
.map(|d| {
|
|
let d = if d.dot(*tgt_dir_up).is_sign_negative() {
|
|
// when the final direction of flow is downward we don't roll
|
|
// upside down but instead mirror the target up vector
|
|
Quaternion::rotation_3d(PI, *tgt_dir_ori.right()) * d
|
|
} else {
|
|
d
|
|
};
|
|
// slerp from untilted up towards the direction by a factor of
|
|
// lateral wind to prevent overly reactive adjustments
|
|
let lateral_wind_speed =
|
|
air_flow.0.projected(&self.ori.right()).magnitude();
|
|
tgt_dir_up.slerped_to(d, lateral_wind_speed / 15.0)
|
|
})
|
|
.unwrap_or_else(Dir::up);
|
|
let global_roll = tgt_dir_up.rotation_between(tgt_up);
|
|
let global_pitch = angle_of_attack(&tgt_dir_ori, &flow_dir)
|
|
* self.timer.as_secs_f32().min(PITCH_SLOW_TIME)
|
|
/ PITCH_SLOW_TIME;
|
|
|
|
self.ori.slerped_towards(
|
|
tgt_dir_ori.prerotated(global_roll).pitched_up(global_pitch),
|
|
slerp_s,
|
|
)
|
|
})
|
|
.unwrap_or_else(|| self.ori.slerped_towards(self.ori.uprighted(), slerp_s))
|
|
};
|
|
|
|
update.ori = {
|
|
let slerp_s = {
|
|
let angle = data.ori.look_dir().angle_between(*data.inputs.look_dir);
|
|
let rate = 0.2 * data.body.base_ori_rate() * PI / angle;
|
|
(data.dt.0 * rate).min(1.0)
|
|
};
|
|
|
|
let rot_from_drag = {
|
|
let speed_factor =
|
|
air_flow.0.magnitude_squared().min(40_f32.powi(2)) / 40_f32.powi(2);
|
|
|
|
Quaternion::rotation_3d(
|
|
-PI / 2.0 * speed_factor,
|
|
ori.up()
|
|
.cross(air_flow.0)
|
|
.try_normalized()
|
|
.unwrap_or_else(|| *data.ori.right()),
|
|
)
|
|
};
|
|
|
|
let rot_from_accel = {
|
|
let accel = data.vel.0 - self.last_vel.0;
|
|
let accel_factor = accel.magnitude_squared().min(1.0) / 1.0;
|
|
|
|
Quaternion::rotation_3d(
|
|
PI / 2.0
|
|
* accel_factor
|
|
* if data.physics.on_ground.is_some() {
|
|
-1.0
|
|
} else {
|
|
1.0
|
|
},
|
|
ori.up()
|
|
.cross(accel)
|
|
.try_normalized()
|
|
.unwrap_or_else(|| *data.ori.right()),
|
|
)
|
|
};
|
|
|
|
update.ori.slerped_towards(
|
|
ori.to_horizontal()
|
|
.prerotated(rot_from_drag * rot_from_accel),
|
|
slerp_s,
|
|
)
|
|
};
|
|
update.character = CharacterState::Glide(Self {
|
|
ori,
|
|
last_vel: *data.vel,
|
|
timer: tick_attack_or_default(data, self.timer, None),
|
|
inputs_disabled,
|
|
..*self
|
|
});
|
|
}
|
|
|
|
update
|
|
}
|
|
|
|
fn unwield(&self, data: &JoinData, output_events: &mut OutputEvents) -> StateUpdate {
|
|
let mut update = StateUpdate::from(data);
|
|
output_events.emit_local(LocalEvent::CreateOutcome(Outcome::Glider {
|
|
pos: data.pos.0,
|
|
wielded: false,
|
|
}));
|
|
update.character = CharacterState::Idle(idle::Data::default());
|
|
update
|
|
}
|
|
}
|