2020-03-12 14:25:06 +00:00
|
|
|
use crate::{
|
2020-03-19 22:40:03 +00:00
|
|
|
comp::{Attacking, CharacterState, EnergySource, StateUpdate},
|
2020-03-12 14:25:06 +00:00
|
|
|
states::utils::*,
|
2020-03-14 21:17:27 +00:00
|
|
|
sys::character_behavior::{CharacterBehavior, JoinData},
|
2020-03-12 14:25:06 +00:00
|
|
|
};
|
|
|
|
use std::{collections::VecDeque, time::Duration};
|
2020-03-19 22:40:03 +00:00
|
|
|
use vek::vec::Vec2;
|
2020-03-12 14:25:06 +00:00
|
|
|
|
2020-03-12 15:22:42 +00:00
|
|
|
// In millis
|
|
|
|
const STAGE_DURATION: u64 = 600;
|
|
|
|
|
2020-03-19 22:40:03 +00:00
|
|
|
const BASE_ACCEL: f32 = 200.0;
|
|
|
|
const BASE_SPEED: f32 = 250.0;
|
2020-03-12 15:22:42 +00:00
|
|
|
/// ### A sequence of 3 incrementally increasing attacks.
|
2020-03-12 14:25:06 +00:00
|
|
|
///
|
2020-03-12 15:22:42 +00:00
|
|
|
/// While holding down the `primary` button, perform a series of 3 attacks,
|
|
|
|
/// each one pushes the player forward as the character steps into the swings.
|
2020-03-12 14:25:06 +00:00
|
|
|
/// The player can let go of the left mouse button at any time
|
|
|
|
/// and stop their attacks by interrupting the attack animation.
|
2020-03-14 21:17:27 +00:00
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize, Eq, Hash)]
|
|
|
|
pub struct Data {
|
|
|
|
/// The tool this state will read to handle damage, etc.
|
2020-03-19 22:40:03 +00:00
|
|
|
pub base_damage: u32,
|
2020-03-14 21:17:27 +00:00
|
|
|
/// `int` denoting what stage (of 3) the attack is in.
|
|
|
|
pub stage: i8,
|
|
|
|
/// How long current stage has been active
|
|
|
|
pub stage_time_active: Duration,
|
|
|
|
/// Whether current stage has exhausted its attack
|
2020-03-19 22:40:03 +00:00
|
|
|
pub stage_exhausted: bool,
|
2020-03-14 21:17:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl CharacterBehavior for Data {
|
|
|
|
fn behavior(&self, data: &JoinData) -> StateUpdate {
|
|
|
|
let mut update = StateUpdate {
|
|
|
|
pos: *data.pos,
|
|
|
|
vel: *data.vel,
|
|
|
|
ori: *data.ori,
|
|
|
|
energy: *data.energy,
|
2020-03-16 11:32:57 +00:00
|
|
|
character: data.character.clone(),
|
2020-03-14 21:17:27 +00:00
|
|
|
local_events: VecDeque::new(),
|
|
|
|
server_events: VecDeque::new(),
|
|
|
|
};
|
2020-03-12 14:25:06 +00:00
|
|
|
|
2020-03-14 21:17:27 +00:00
|
|
|
let new_stage_time_active = self
|
|
|
|
.stage_time_active
|
2020-03-12 14:25:06 +00:00
|
|
|
.checked_add(Duration::from_secs_f32(data.dt.0))
|
|
|
|
.unwrap_or(Duration::default());
|
|
|
|
|
2020-03-12 15:22:42 +00:00
|
|
|
// If player stops holding input,
|
2020-03-12 14:25:06 +00:00
|
|
|
if !data.inputs.primary.is_pressed() {
|
2020-03-19 22:40:03 +00:00
|
|
|
// Done
|
|
|
|
update.character = CharacterState::Wielding;
|
|
|
|
// Make sure attack component is removed
|
|
|
|
data.updater.remove::<Attacking>(data.entity);
|
2020-03-12 15:22:42 +00:00
|
|
|
return update;
|
2020-03-12 14:25:06 +00:00
|
|
|
}
|
|
|
|
|
2020-03-14 21:17:27 +00:00
|
|
|
if self.stage < 3 {
|
2020-03-12 15:22:42 +00:00
|
|
|
if new_stage_time_active < Duration::from_millis(STAGE_DURATION / 3) {
|
|
|
|
// Move player forward while in first third of each stage
|
2020-03-19 22:40:03 +00:00
|
|
|
// Move player according to move_dir
|
|
|
|
if update.vel.0.magnitude_squared() < BASE_SPEED.powf(2.0) {
|
|
|
|
update.vel.0 =
|
|
|
|
update.vel.0 + Vec2::broadcast(data.dt.0) * data.ori.0 * BASE_ACCEL;
|
|
|
|
let mag2 = update.vel.0.magnitude_squared();
|
|
|
|
if mag2 > BASE_SPEED.powf(2.0) {
|
|
|
|
update.vel.0 = update.vel.0.normalized() * BASE_SPEED;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
update.character = CharacterState::TripleStrike(Data {
|
|
|
|
base_damage: self.base_damage,
|
|
|
|
stage: self.stage,
|
|
|
|
stage_time_active: new_stage_time_active,
|
|
|
|
stage_exhausted: false,
|
|
|
|
});
|
2020-03-12 15:22:42 +00:00
|
|
|
} else if new_stage_time_active > Duration::from_millis(STAGE_DURATION / 2)
|
2020-03-19 22:40:03 +00:00
|
|
|
&& !self.stage_exhausted
|
2020-03-12 15:22:42 +00:00
|
|
|
{
|
2020-03-19 22:40:03 +00:00
|
|
|
// Allow player to influence orientation a little
|
|
|
|
handle_move(data, &mut update);
|
|
|
|
|
2020-03-12 15:22:42 +00:00
|
|
|
// Try to deal damage in second half of stage
|
2020-03-19 22:40:03 +00:00
|
|
|
data.updater.insert(data.entity, Attacking {
|
|
|
|
base_damage: self.base_damage * (self.stage as u32 + 1),
|
|
|
|
max_angle: 180_f32.to_radians(),
|
|
|
|
applied: false,
|
|
|
|
hit_count: 0,
|
|
|
|
});
|
|
|
|
|
|
|
|
update.character = CharacterState::TripleStrike(Data {
|
|
|
|
base_damage: self.base_damage,
|
|
|
|
stage: self.stage,
|
|
|
|
stage_time_active: new_stage_time_active,
|
|
|
|
stage_exhausted: true,
|
|
|
|
});
|
|
|
|
} else if new_stage_time_active > Duration::from_millis(STAGE_DURATION) {
|
|
|
|
// Allow player to influence orientation a little
|
|
|
|
handle_move(data, &mut update);
|
|
|
|
|
|
|
|
update.character = CharacterState::TripleStrike(Data {
|
|
|
|
base_damage: self.base_damage,
|
|
|
|
stage: self.stage + 1,
|
|
|
|
stage_time_active: Duration::default(),
|
|
|
|
stage_exhausted: false,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
update.character = CharacterState::TripleStrike(Data {
|
|
|
|
base_damage: self.base_damage,
|
|
|
|
stage: self.stage,
|
|
|
|
stage_time_active: new_stage_time_active,
|
|
|
|
stage_exhausted: self.stage_exhausted,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Done
|
|
|
|
update.character = CharacterState::Wielding;
|
|
|
|
// Make sure attack component is removed
|
|
|
|
data.updater.remove::<Attacking>(data.entity);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grant energy on successful hit
|
|
|
|
if let Some(attack) = data.attacking {
|
|
|
|
if attack.applied && attack.hit_count > 0 {
|
|
|
|
data.updater.remove::<Attacking>(data.entity);
|
|
|
|
update.energy.change_by(100, EnergySource::HitEnemy);
|
2020-03-12 15:22:42 +00:00
|
|
|
}
|
2020-03-12 14:25:06 +00:00
|
|
|
}
|
|
|
|
|
2020-03-14 21:17:27 +00:00
|
|
|
update
|
|
|
|
}
|
2020-03-12 14:25:06 +00:00
|
|
|
}
|