veloren/common/src/comp/ability.rs

981 lines
36 KiB
Rust
Raw Normal View History

use crate::{
assets::{self, Asset},
2020-03-24 19:31:54 +00:00
comp::{
item::{armor::Protection, tool::AbilityMap, Item, ItemKind},
projectile::ProjectileConstructor,
Body, CharacterState, EnergySource, Gravity, LightEmitter, StateUpdate,
2020-03-24 19:31:54 +00:00
},
states::{
behavior::JoinData,
utils::{AbilityKey, StageSection},
*,
},
2020-10-18 18:21:58 +00:00
Knockback,
};
use arraygen::Arraygen;
use serde::{Deserialize, Serialize};
2020-03-22 04:49:32 +00:00
use specs::{Component, FlaggedStorage};
use specs_idvs::IdvStorage;
use std::{fs::File, io::BufReader, time::Duration};
use vek::Vec3;
2020-01-01 17:16:29 +00:00
#[derive(Copy, Clone, Hash, Eq, PartialEq, Debug, Serialize, Deserialize)]
pub enum CharacterAbilityType {
BasicMelee,
BasicRanged,
Boost,
2020-10-14 22:30:58 +00:00
ChargedMelee(StageSection),
2020-07-26 03:06:53 +00:00
ChargedRanged,
2020-10-14 22:30:58 +00:00
DashMelee(StageSection),
BasicBlock,
2020-09-13 01:33:46 +00:00
ComboMelee(StageSection, u32),
2020-10-14 22:30:58 +00:00
LeapMelee(StageSection),
SpinMelee(StageSection),
Shockwave,
BasicBeam,
RepeaterRanged,
}
impl From<&CharacterState> for CharacterAbilityType {
fn from(state: &CharacterState) -> Self {
match state {
CharacterState::BasicMelee(_) => Self::BasicMelee,
CharacterState::BasicRanged(_) => Self::BasicRanged,
CharacterState::Boost(_) => Self::Boost,
2020-10-14 22:30:58 +00:00
CharacterState::DashMelee(data) => Self::DashMelee(data.stage_section),
CharacterState::BasicBlock => Self::BasicBlock,
2020-10-14 22:30:58 +00:00
CharacterState::LeapMelee(data) => Self::LeapMelee(data.stage_section),
2020-09-13 01:33:46 +00:00
CharacterState::ComboMelee(data) => Self::ComboMelee(data.stage_section, data.stage),
2020-10-14 22:30:58 +00:00
CharacterState::SpinMelee(data) => Self::SpinMelee(data.stage_section),
CharacterState::ChargedMelee(data) => Self::ChargedMelee(data.stage_section),
2020-07-26 03:06:53 +00:00
CharacterState::ChargedRanged(_) => Self::ChargedRanged,
CharacterState::Shockwave(_) => Self::ChargedRanged,
CharacterState::BasicBeam(_) => Self::BasicBeam,
CharacterState::RepeaterRanged(_) => Self::RepeaterRanged,
_ => Self::BasicMelee,
}
}
}
#[derive(Clone, PartialEq, Debug, Serialize, Deserialize)]
2020-03-14 15:40:29 +00:00
pub enum CharacterAbility {
BasicMelee {
2020-03-24 21:03:11 +00:00
energy_cost: u32,
buildup_duration: u64,
swing_duration: u64,
recover_duration: u64,
base_damage: u32,
knockback: f32,
range: f32,
max_angle: f32,
2020-03-12 14:36:02 +00:00
},
BasicRanged {
2020-03-24 19:09:23 +00:00
energy_cost: u32,
buildup_duration: u64,
recover_duration: u64,
projectile: ProjectileConstructor,
2020-03-22 19:39:50 +00:00
projectile_body: Body,
2020-03-24 19:31:54 +00:00
projectile_light: Option<LightEmitter>,
projectile_gravity: Option<Gravity>,
projectile_speed: f32,
can_continue: bool,
2020-03-22 19:39:50 +00:00
},
RepeaterRanged {
energy_cost: u32,
movement_duration: u64,
buildup_duration: u64,
shoot_duration: u64,
recover_duration: u64,
2020-09-28 02:38:23 +00:00
leap: Option<f32>,
projectile: ProjectileConstructor,
projectile_body: Body,
projectile_light: Option<LightEmitter>,
projectile_gravity: Option<Gravity>,
projectile_speed: f32,
reps_remaining: u32,
},
Boost {
movement_duration: u64,
only_up: bool,
},
2020-03-16 15:34:53 +00:00
DashMelee {
energy_cost: u32,
base_damage: u32,
max_damage: u32,
base_knockback: f32,
max_knockback: f32,
range: f32,
angle: f32,
energy_drain: u32,
forward_speed: f32,
buildup_duration: u64,
charge_duration: u64,
swing_duration: u64,
recover_duration: u64,
infinite_charge: bool,
is_interruptible: bool,
2020-03-16 15:34:53 +00:00
},
2020-03-07 18:15:02 +00:00
BasicBlock,
Roll {
energy_cost: u32,
buildup_duration: u64,
movement_duration: u64,
recover_duration: u64,
roll_strength: f32,
},
ComboMelee {
stage_data: Vec<combo_melee::Stage<u64>>,
initial_energy_gain: u32,
max_energy_gain: u32,
energy_increase: u32,
speed_increase: f32,
max_speed_increase: f32,
is_interruptible: bool,
2020-03-19 22:40:03 +00:00
},
LeapMelee {
energy_cost: u32,
buildup_duration: u64,
movement_duration: u64,
swing_duration: u64,
recover_duration: u64,
base_damage: u32,
range: f32,
max_angle: f32,
2020-09-25 05:51:26 +00:00
knockback: f32,
2020-09-28 23:55:38 +00:00
forward_leap_strength: f32,
vertical_leap_strength: f32,
},
2020-07-08 19:58:41 +00:00
SpinMelee {
buildup_duration: u64,
swing_duration: u64,
recover_duration: u64,
2020-07-08 19:58:41 +00:00
base_damage: u32,
knockback: f32,
range: f32,
energy_cost: u32,
is_infinite: bool,
is_helicopter: bool,
is_interruptible: bool,
forward_speed: f32,
num_spins: u32,
2020-07-08 19:58:41 +00:00
},
ChargedMelee {
energy_cost: u32,
energy_drain: u32,
initial_damage: u32,
max_damage: u32,
initial_knockback: f32,
max_knockback: f32,
range: f32,
max_angle: f32,
speed: f32,
charge_duration: u64,
swing_duration: u64,
hit_timing: f32,
recover_duration: u64,
},
2020-07-26 03:06:53 +00:00
ChargedRanged {
energy_cost: u32,
energy_drain: u32,
initial_damage: u32,
max_damage: u32,
initial_knockback: f32,
max_knockback: f32,
speed: f32,
buildup_duration: u64,
charge_duration: u64,
recover_duration: u64,
2020-07-26 03:06:53 +00:00
projectile_body: Body,
projectile_light: Option<LightEmitter>,
projectile_gravity: Option<Gravity>,
initial_projectile_speed: f32,
max_projectile_speed: f32,
2020-07-26 03:06:53 +00:00
},
Shockwave {
2020-08-08 20:53:55 +00:00
energy_cost: u32,
buildup_duration: u64,
swing_duration: u64,
recover_duration: u64,
2020-08-08 20:53:55 +00:00
damage: u32,
2020-10-18 18:21:58 +00:00
knockback: Knockback,
2020-08-08 20:53:55 +00:00
shockwave_angle: f32,
2020-10-12 22:55:55 +00:00
shockwave_vertical_angle: f32,
2020-08-08 20:53:55 +00:00
shockwave_speed: f32,
shockwave_duration: u64,
2020-09-19 16:55:31 +00:00
requires_ground: bool,
2020-10-09 17:42:15 +00:00
move_efficiency: f32,
2020-08-08 20:53:55 +00:00
},
BasicBeam {
buildup_duration: u64,
recover_duration: u64,
beam_duration: u64,
base_hps: u32,
base_dps: u32,
2020-08-31 21:55:38 +00:00
tick_rate: f32,
range: f32,
max_angle: f32,
lifesteal_eff: f32,
energy_regen: u32,
energy_cost: u32,
energy_drain: u32,
},
2020-01-01 17:16:29 +00:00
}
impl Default for CharacterAbility {
fn default() -> Self {
CharacterAbility::BasicMelee {
energy_cost: 0,
buildup_duration: 250,
swing_duration: 250,
recover_duration: 500,
base_damage: 10,
knockback: 0.0,
range: 3.5,
max_angle: 15.0,
}
}
}
impl Asset for CharacterAbility {
const ENDINGS: &'static [&'static str] = &["ron"];
fn parse(buf_reader: BufReader<File>, _specifier: &str) -> Result<Self, assets::Error> {
ron::de::from_reader(buf_reader).map_err(assets::Error::parse_error)
}
}
2020-03-17 14:01:41 +00:00
impl CharacterAbility {
2020-03-21 22:55:20 +00:00
/// Attempts to fulfill requirements, mutating `update` (taking energy) if
/// applicable.
pub fn requirements_paid(&self, data: &JoinData, update: &mut StateUpdate) -> bool {
2020-03-17 14:01:41 +00:00
match self {
2020-11-05 20:22:30 +00:00
CharacterAbility::Roll { energy_cost, .. } => {
2020-03-17 14:01:41 +00:00
data.physics.on_ground
&& data.vel.0.xy().magnitude_squared() > 0.5
2020-03-17 14:01:41 +00:00
&& update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
2020-03-22 19:39:50 +00:00
.is_ok()
},
CharacterAbility::DashMelee { energy_cost, .. } => update
2020-03-26 13:46:08 +00:00
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
2020-03-26 13:46:08 +00:00
.is_ok(),
CharacterAbility::BasicMelee { energy_cost, .. } => update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
CharacterAbility::BasicRanged { energy_cost, .. } => update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
CharacterAbility::LeapMelee { energy_cost, .. } => {
update.vel.0.z >= 0.0
&& update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok()
},
2020-07-08 19:58:41 +00:00
CharacterAbility::SpinMelee { energy_cost, .. } => update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
2020-07-26 03:06:53 +00:00
CharacterAbility::ChargedRanged { energy_cost, .. } => update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
CharacterAbility::ChargedMelee { energy_cost, .. } => update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
CharacterAbility::RepeaterRanged {
energy_cost, leap, ..
} => {
(leap.is_none() || update.vel.0.z >= 0.0)
&& update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok()
},
CharacterAbility::Shockwave { energy_cost, .. } => update
2020-08-08 20:53:55 +00:00
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
2020-03-17 14:01:41 +00:00
_ => true,
}
}
fn default_roll() -> CharacterAbility {
CharacterAbility::Roll {
energy_cost: 100,
buildup_duration: 100,
movement_duration: 250,
recover_duration: 150,
roll_strength: 2.5,
}
}
2020-11-13 03:50:40 +00:00
pub fn adjusted_by_stats(mut self, power: f32, speed: f32) -> Self {
use CharacterAbility::*;
match self {
BasicMelee {
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut base_damage,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*swing_duration = (*swing_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
*base_damage = (*base_damage as f32 * power) as u32;
},
BasicRanged {
ref mut buildup_duration,
ref mut recover_duration,
ref mut projectile,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
2020-11-13 03:50:40 +00:00
*projectile = projectile.modified_projectile(power);
},
RepeaterRanged {
ref mut movement_duration,
ref mut buildup_duration,
ref mut shoot_duration,
ref mut recover_duration,
ref mut projectile,
..
} => {
*movement_duration = (*movement_duration as f32 / speed) as u64;
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*shoot_duration = (*shoot_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
2020-11-13 03:50:40 +00:00
*projectile = projectile.modified_projectile(power);
},
Boost {
ref mut movement_duration,
..
} => {
*movement_duration = (*movement_duration as f32 / speed) as u64;
},
DashMelee {
ref mut base_damage,
ref mut max_damage,
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
..
} => {
*base_damage = (*base_damage as f32 * power) as u32;
*max_damage = (*max_damage as f32 * power) as u32;
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*swing_duration = (*swing_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
},
BasicBlock => {},
Roll {
ref mut buildup_duration,
ref mut movement_duration,
ref mut recover_duration,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*movement_duration = (*movement_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
},
ComboMelee {
ref mut stage_data, ..
} => {
*stage_data = stage_data
.iter_mut()
2020-11-13 03:50:40 +00:00
.map(|s| s.adjusted_by_stats(power, speed))
.collect();
},
LeapMelee {
ref mut buildup_duration,
ref mut movement_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut base_damage,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*movement_duration = (*movement_duration as f32 / speed) as u64;
*swing_duration = (*swing_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
*base_damage = (*base_damage as f32 * power) as u32;
},
SpinMelee {
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut base_damage,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*swing_duration = (*swing_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
*base_damage = (*base_damage as f32 * power) as u32;
},
ChargedMelee {
ref mut initial_damage,
ref mut max_damage,
speed: ref mut ability_speed,
ref mut charge_duration,
ref mut swing_duration,
ref mut recover_duration,
..
} => {
*initial_damage = (*initial_damage as f32 * power) as u32;
*max_damage = (*max_damage as f32 * power) as u32;
*ability_speed *= speed;
*charge_duration = (*charge_duration as f32 / speed) as u64;
*swing_duration = (*swing_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
},
ChargedRanged {
ref mut initial_damage,
ref mut max_damage,
speed: ref mut ability_speed,
ref mut buildup_duration,
ref mut charge_duration,
ref mut recover_duration,
..
} => {
*initial_damage = (*initial_damage as f32 * power) as u32;
*max_damage = (*max_damage as f32 * power) as u32;
*ability_speed *= speed;
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*charge_duration = (*charge_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
},
Shockwave {
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut damage,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*swing_duration = (*swing_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
*damage = (*damage as f32 * power) as u32;
},
BasicBeam {
ref mut buildup_duration,
ref mut recover_duration,
ref mut base_hps,
ref mut base_dps,
ref mut tick_rate,
..
} => {
*buildup_duration = (*buildup_duration as f32 / speed) as u64;
*recover_duration = (*recover_duration as f32 / speed) as u64;
*base_hps = (*base_hps as f32 * power) as u32;
*base_dps = (*base_dps as f32 * power) as u32;
*tick_rate *= speed;
},
}
self
}
pub fn get_energy_cost(&self) -> u32 {
use CharacterAbility::*;
match self {
BasicMelee { energy_cost, .. }
| BasicRanged { energy_cost, .. }
| RepeaterRanged { energy_cost, .. }
| DashMelee { energy_cost, .. }
| Roll { energy_cost, .. }
| LeapMelee { energy_cost, .. }
| SpinMelee { energy_cost, .. }
| ChargedMelee { energy_cost, .. }
| ChargedRanged { energy_cost, .. }
| Shockwave { energy_cost, .. }
| BasicBeam { energy_cost, .. } => *energy_cost,
_ => 0,
}
}
2020-03-17 14:01:41 +00:00
}
#[derive(Clone, PartialEq, Debug, Serialize, Deserialize)]
pub struct ItemConfig {
pub item: Item,
pub ability1: Option<CharacterAbility>,
pub ability2: Option<CharacterAbility>,
pub ability3: Option<CharacterAbility>,
pub block_ability: Option<CharacterAbility>,
pub dodge_ability: Option<CharacterAbility>,
}
impl From<(Item, &AbilityMap)> for ItemConfig {
fn from((item, map): (Item, &AbilityMap)) -> Self {
if let ItemKind::Tool(tool) = &item.kind() {
let abilities = tool.get_abilities(map);
return ItemConfig {
item,
ability1: Some(abilities.primary),
ability2: Some(abilities.secondary),
ability3: abilities.skills.get(0).cloned(),
block_ability: None,
dodge_ability: Some(CharacterAbility::default_roll()),
};
}
unimplemented!("ItemConfig is currently only supported for Tools")
}
}
#[derive(Arraygen, Clone, PartialEq, Default, Debug, Serialize, Deserialize)]
#[gen_array(pub fn get_armor: &Option<Item>)]
pub struct Loadout {
pub active_item: Option<ItemConfig>,
pub second_item: Option<ItemConfig>,
pub lantern: Option<Item>,
2020-08-24 01:31:05 +00:00
pub glider: Option<Item>,
#[in_array(get_armor)]
pub shoulder: Option<Item>,
#[in_array(get_armor)]
pub chest: Option<Item>,
#[in_array(get_armor)]
pub belt: Option<Item>,
#[in_array(get_armor)]
pub hand: Option<Item>,
#[in_array(get_armor)]
pub pants: Option<Item>,
#[in_array(get_armor)]
pub foot: Option<Item>,
#[in_array(get_armor)]
2020-04-06 16:11:05 +00:00
pub back: Option<Item>,
#[in_array(get_armor)]
2020-04-06 20:03:46 +00:00
pub ring: Option<Item>,
#[in_array(get_armor)]
2020-04-06 20:03:46 +00:00
pub neck: Option<Item>,
#[in_array(get_armor)]
2020-04-06 20:03:46 +00:00
pub head: Option<Item>,
#[in_array(get_armor)]
2020-04-06 20:03:46 +00:00
pub tabard: Option<Item>,
2020-01-01 17:16:29 +00:00
}
impl Loadout {
pub fn get_damage_reduction(&self) -> f32 {
let protection = self
.get_armor()
.iter()
.flat_map(|armor| armor.as_ref())
.filter_map(|item| {
if let ItemKind::Armor(armor) = &item.kind() {
Some(armor.get_protection())
} else {
None
}
})
.map(|protection| match protection {
Protection::Normal(protection) => Some(protection),
Protection::Invincible => None,
})
.sum::<Option<f32>>();
match protection {
Some(dr) => dr / (60.0 + dr.abs()),
None => 1.0,
}
}
}
impl From<(&CharacterAbility, AbilityKey)> for CharacterState {
fn from((ability, key): (&CharacterAbility, AbilityKey)) -> Self {
2020-03-14 15:40:29 +00:00
match ability {
CharacterAbility::BasicMelee {
2020-03-14 15:40:29 +00:00
buildup_duration,
swing_duration,
2020-03-14 15:40:29 +00:00
recover_duration,
base_damage,
knockback,
range,
max_angle,
2020-03-24 21:03:11 +00:00
energy_cost: _,
} => CharacterState::BasicMelee(basic_melee::Data {
static_data: basic_melee::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
swing_duration: Duration::from_millis(*swing_duration),
recover_duration: Duration::from_millis(*recover_duration),
base_damage: *base_damage,
knockback: *knockback,
range: *range,
max_angle: *max_angle,
2020-11-20 17:30:48 +00:00
ability_key: key,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-03-14 15:40:29 +00:00
exhausted: false,
}),
CharacterAbility::BasicRanged {
buildup_duration,
2020-03-14 15:40:29 +00:00
recover_duration,
projectile,
projectile_body,
2020-03-24 19:31:54 +00:00
projectile_light,
projectile_gravity,
projectile_speed,
can_continue,
2020-03-24 19:09:23 +00:00
energy_cost: _,
} => CharacterState::BasicRanged(basic_ranged::Data {
static_data: basic_ranged::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
recover_duration: Duration::from_millis(*recover_duration),
projectile: *projectile,
projectile_body: *projectile_body,
projectile_light: *projectile_light,
projectile_gravity: *projectile_gravity,
projectile_speed: *projectile_speed,
can_continue: *can_continue,
ability_key: key,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
exhausted: false,
continue_next: false,
}),
CharacterAbility::Boost {
movement_duration,
only_up,
} => CharacterState::Boost(boost::Data {
static_data: boost::StaticData {
movement_duration: Duration::from_millis(*movement_duration),
only_up: *only_up,
},
timer: Duration::default(),
}),
2020-03-16 15:34:53 +00:00
CharacterAbility::DashMelee {
energy_cost: _,
base_damage,
max_damage,
base_knockback,
max_knockback,
range,
angle,
energy_drain,
forward_speed,
2020-03-16 15:34:53 +00:00
buildup_duration,
charge_duration,
swing_duration,
2020-03-16 15:34:53 +00:00
recover_duration,
infinite_charge,
is_interruptible,
2020-03-16 15:34:53 +00:00
} => CharacterState::DashMelee(dash_melee::Data {
2020-09-11 13:59:45 +00:00
static_data: dash_melee::StaticData {
base_damage: *base_damage,
max_damage: *max_damage,
base_knockback: *base_knockback,
max_knockback: *max_knockback,
range: *range,
angle: *angle,
energy_drain: *energy_drain,
forward_speed: *forward_speed,
infinite_charge: *infinite_charge,
buildup_duration: Duration::from_millis(*buildup_duration),
charge_duration: Duration::from_millis(*charge_duration),
swing_duration: Duration::from_millis(*swing_duration),
recover_duration: Duration::from_millis(*recover_duration),
is_interruptible: *is_interruptible,
2020-10-31 18:44:00 +00:00
ability_key: key,
2020-09-11 13:59:45 +00:00
},
2020-10-18 16:46:28 +00:00
auto_charge: false,
timer: Duration::default(),
2020-10-18 16:46:28 +00:00
refresh_timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-09-11 13:59:45 +00:00
exhausted: false,
2020-03-16 15:34:53 +00:00
}),
CharacterAbility::BasicBlock => CharacterState::BasicBlock,
CharacterAbility::Roll {
energy_cost: _,
buildup_duration,
movement_duration,
recover_duration,
roll_strength,
} => CharacterState::Roll(roll::Data {
static_data: roll::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
movement_duration: Duration::from_millis(*movement_duration),
recover_duration: Duration::from_millis(*recover_duration),
roll_strength: *roll_strength,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-03-26 13:46:08 +00:00
was_wielded: false, // false by default. utils might set it to true
was_sneak: false,
was_combo: None,
2020-03-14 21:17:27 +00:00
}),
CharacterAbility::ComboMelee {
stage_data,
initial_energy_gain,
max_energy_gain,
energy_increase,
speed_increase,
max_speed_increase,
is_interruptible,
} => CharacterState::ComboMelee(combo_melee::Data {
2020-09-21 22:38:01 +00:00
static_data: combo_melee::StaticData {
num_stages: stage_data.len() as u32,
2020-11-13 03:50:40 +00:00
stage_data: stage_data.iter().map(|stage| stage.to_duration()).collect(),
2020-09-21 22:38:01 +00:00
initial_energy_gain: *initial_energy_gain,
max_energy_gain: *max_energy_gain,
energy_increase: *energy_increase,
speed_increase: 1.0 - *speed_increase,
max_speed_increase: *max_speed_increase - 1.0,
is_interruptible: *is_interruptible,
2020-10-31 18:44:00 +00:00
ability_key: key,
2020-09-21 22:38:01 +00:00
},
stage: 1,
combo: 0,
timer: Duration::default(),
stage_section: StageSection::Buildup,
next_stage: false,
2020-03-27 17:40:15 +00:00
}),
CharacterAbility::LeapMelee {
energy_cost: _,
buildup_duration,
2020-09-28 23:55:38 +00:00
movement_duration,
swing_duration,
recover_duration,
base_damage,
2020-09-28 23:55:38 +00:00
knockback,
range,
max_angle,
2020-09-28 23:55:38 +00:00
forward_leap_strength,
vertical_leap_strength,
} => CharacterState::LeapMelee(leap_melee::Data {
2020-09-28 23:55:38 +00:00
static_data: leap_melee::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
movement_duration: Duration::from_millis(*movement_duration),
swing_duration: Duration::from_millis(*swing_duration),
recover_duration: Duration::from_millis(*recover_duration),
2020-09-28 23:55:38 +00:00
base_damage: *base_damage,
knockback: *knockback,
range: *range,
max_angle: *max_angle,
forward_leap_strength: *forward_leap_strength,
vertical_leap_strength: *vertical_leap_strength,
2020-11-20 17:30:48 +00:00
ability_key: key,
2020-09-28 23:55:38 +00:00
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
exhausted: false,
}),
2020-07-08 19:58:41 +00:00
CharacterAbility::SpinMelee {
buildup_duration,
swing_duration,
2020-07-08 19:58:41 +00:00
recover_duration,
base_damage,
knockback,
range,
energy_cost,
is_infinite,
is_helicopter,
is_interruptible,
forward_speed,
num_spins,
2020-07-08 19:58:41 +00:00
} => CharacterState::SpinMelee(spin_melee::Data {
static_data: spin_melee::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
swing_duration: Duration::from_millis(*swing_duration),
recover_duration: Duration::from_millis(*recover_duration),
base_damage: *base_damage,
knockback: *knockback,
range: *range,
energy_cost: *energy_cost,
is_infinite: *is_infinite,
is_helicopter: *is_helicopter,
is_interruptible: *is_interruptible,
forward_speed: *forward_speed,
num_spins: *num_spins,
2020-10-31 18:44:00 +00:00
ability_key: key,
},
timer: Duration::default(),
spins_remaining: *num_spins - 1,
stage_section: StageSection::Buildup,
2020-09-18 19:27:02 +00:00
exhausted: false,
2020-07-08 19:58:41 +00:00
}),
CharacterAbility::ChargedMelee {
2020-10-06 05:21:22 +00:00
energy_cost,
energy_drain,
initial_damage,
max_damage,
initial_knockback,
max_knockback,
speed,
charge_duration,
2020-09-28 01:58:49 +00:00
swing_duration,
hit_timing,
recover_duration,
range,
max_angle,
} => CharacterState::ChargedMelee(charged_melee::Data {
2020-09-28 01:58:49 +00:00
static_data: charged_melee::StaticData {
2020-10-06 05:21:22 +00:00
energy_cost: *energy_cost,
2020-09-28 01:58:49 +00:00
energy_drain: *energy_drain,
initial_damage: *initial_damage,
max_damage: *max_damage,
initial_knockback: *initial_knockback,
max_knockback: *max_knockback,
speed: *speed,
2020-09-28 01:58:49 +00:00
range: *range,
max_angle: *max_angle,
charge_duration: Duration::from_millis(*charge_duration),
swing_duration: Duration::from_millis(*swing_duration),
hit_timing: *hit_timing,
recover_duration: Duration::from_millis(*recover_duration),
2020-10-31 18:44:00 +00:00
ability_key: key,
2020-09-28 01:58:49 +00:00
},
stage_section: StageSection::Charge,
2020-09-28 01:58:49 +00:00
timer: Duration::default(),
exhausted: false,
2020-09-28 01:58:49 +00:00
charge_amount: 0.0,
}),
2020-07-26 03:06:53 +00:00
CharacterAbility::ChargedRanged {
energy_cost: _,
energy_drain,
initial_damage,
max_damage,
initial_knockback,
max_knockback,
speed,
buildup_duration,
2020-07-26 03:06:53 +00:00
charge_duration,
recover_duration,
projectile_body,
projectile_light,
projectile_gravity,
initial_projectile_speed,
max_projectile_speed,
2020-07-26 03:06:53 +00:00
} => CharacterState::ChargedRanged(charged_ranged::Data {
static_data: charged_ranged::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
charge_duration: Duration::from_millis(*charge_duration),
recover_duration: Duration::from_millis(*recover_duration),
energy_drain: *energy_drain,
initial_damage: *initial_damage,
max_damage: *max_damage,
speed: *speed,
initial_knockback: *initial_knockback,
max_knockback: *max_knockback,
projectile_body: *projectile_body,
projectile_light: *projectile_light,
projectile_gravity: *projectile_gravity,
initial_projectile_speed: *initial_projectile_speed,
max_projectile_speed: *max_projectile_speed,
2020-10-31 18:44:00 +00:00
ability_key: key,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-07-26 03:06:53 +00:00
exhausted: false,
}),
CharacterAbility::RepeaterRanged {
energy_cost: _,
movement_duration,
2020-09-28 02:38:23 +00:00
buildup_duration,
shoot_duration,
recover_duration,
2020-09-28 02:38:23 +00:00
leap,
projectile,
projectile_body,
projectile_light,
projectile_gravity,
projectile_speed,
reps_remaining,
} => CharacterState::RepeaterRanged(repeater_ranged::Data {
2020-09-28 02:38:23 +00:00
static_data: repeater_ranged::StaticData {
movement_duration: Duration::from_millis(*movement_duration),
buildup_duration: Duration::from_millis(*buildup_duration),
shoot_duration: Duration::from_millis(*shoot_duration),
recover_duration: Duration::from_millis(*recover_duration),
2020-09-28 02:38:23 +00:00
leap: *leap,
projectile: *projectile,
2020-09-28 02:38:23 +00:00
projectile_body: *projectile_body,
projectile_light: *projectile_light,
projectile_gravity: *projectile_gravity,
projectile_speed: *projectile_speed,
2020-11-20 17:30:48 +00:00
ability_key: key,
2020-09-28 02:38:23 +00:00
},
timer: Duration::default(),
stage_section: StageSection::Movement,
reps_remaining: *reps_remaining,
}),
CharacterAbility::Shockwave {
2020-08-08 20:53:55 +00:00
energy_cost: _,
buildup_duration,
swing_duration,
2020-08-08 20:53:55 +00:00
recover_duration,
damage,
knockback,
shockwave_angle,
2020-10-12 22:55:55 +00:00
shockwave_vertical_angle,
2020-08-08 20:53:55 +00:00
shockwave_speed,
shockwave_duration,
2020-09-19 16:55:31 +00:00
requires_ground,
2020-10-09 17:42:15 +00:00
move_efficiency,
} => CharacterState::Shockwave(shockwave::Data {
static_data: shockwave::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
swing_duration: Duration::from_millis(*swing_duration),
recover_duration: Duration::from_millis(*recover_duration),
damage: *damage,
knockback: *knockback,
shockwave_angle: *shockwave_angle,
2020-10-12 22:55:55 +00:00
shockwave_vertical_angle: *shockwave_vertical_angle,
shockwave_speed: *shockwave_speed,
shockwave_duration: Duration::from_millis(*shockwave_duration),
requires_ground: *requires_ground,
2020-10-09 17:42:15 +00:00
move_efficiency: *move_efficiency,
2020-11-20 17:30:48 +00:00
ability_key: key,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-08-08 20:53:55 +00:00
}),
CharacterAbility::BasicBeam {
buildup_duration,
recover_duration,
beam_duration,
base_hps,
base_dps,
2020-08-31 21:55:38 +00:00
tick_rate,
range,
max_angle,
lifesteal_eff,
energy_regen,
energy_cost,
energy_drain,
} => CharacterState::BasicBeam(basic_beam::Data {
static_data: basic_beam::StaticData {
buildup_duration: Duration::from_millis(*buildup_duration),
recover_duration: Duration::from_millis(*recover_duration),
beam_duration: Duration::from_millis(*beam_duration),
base_hps: *base_hps,
base_dps: *base_dps,
tick_rate: *tick_rate,
range: *range,
max_angle: *max_angle,
lifesteal_eff: *lifesteal_eff,
energy_regen: *energy_regen,
energy_cost: *energy_cost,
energy_drain: *energy_drain,
ability_key: key,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
particle_ori: None::<Vec3<f32>>,
offset: Vec3::zero(),
}),
2020-02-03 10:54:50 +00:00
}
}
}
impl Component for Loadout {
type Storage = FlaggedStorage<Self, IdvStorage<Self>>;
2020-01-01 17:16:29 +00:00
}