veloren/common/src/comp/ability.rs

2012 lines
79 KiB
Rust
Raw Normal View History

use crate::{
assets::{self, Asset},
combat::{self, CombatEffect, DamageKind, Knockback},
2020-03-24 19:31:54 +00:00
comp::{
2021-07-06 01:32:12 +00:00
self, aura, beam, buff,
inventory::item::tool::{Stats, ToolKind},
projectile::ProjectileConstructor,
skills, Body, CharacterState, EnergySource, LightEmitter, StateUpdate,
2020-03-24 19:31:54 +00:00
},
states::{
behavior::JoinData,
utils::{AbilityInfo, StageSection},
*,
},
2021-06-19 18:53:23 +00:00
terrain::SpriteKind,
};
use serde::{Deserialize, Serialize};
2021-08-16 09:16:20 +00:00
use std::{convert::TryFrom, time::Duration};
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,
2021-03-17 23:54:49 +00:00
BasicAura,
SelfBuff,
}
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),
2021-04-10 03:40:20 +00:00
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::Shockwave,
CharacterState::BasicBeam(_) => Self::BasicBeam,
CharacterState::RepeaterRanged(_) => Self::RepeaterRanged,
2021-03-17 23:54:49 +00:00
CharacterState::BasicAura(_) => Self::BasicAura,
CharacterState::SelfBuff(_) => Self::SelfBuff,
_ => Self::BasicMelee,
}
}
}
#[derive(Clone, PartialEq, Debug, Serialize, Deserialize)]
2020-03-14 15:40:29 +00:00
pub enum CharacterAbility {
BasicMelee {
energy_cost: f32,
buildup_duration: f32,
swing_duration: f32,
recover_duration: f32,
base_damage: f32,
base_poise_damage: f32,
knockback: Knockback,
range: f32,
max_angle: f32,
damage_effect: Option<CombatEffect>,
damage_kind: DamageKind,
2020-03-12 14:36:02 +00:00
},
BasicRanged {
energy_cost: f32,
buildup_duration: f32,
recover_duration: f32,
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_speed: f32,
2021-05-16 03:09:39 +00:00
num_projectiles: u32,
projectile_spread: f32,
2020-03-22 19:39:50 +00:00
},
RepeaterRanged {
energy_cost: f32,
buildup_duration: f32,
shoot_duration: f32,
recover_duration: f32,
2021-05-16 03:09:39 +00:00
max_speed: f32,
half_speed_at: u32,
projectile: ProjectileConstructor,
projectile_body: Body,
projectile_light: Option<LightEmitter>,
projectile_speed: f32,
},
Boost {
movement_duration: f32,
only_up: bool,
speed: f32,
max_exit_velocity: f32,
},
2020-03-16 15:34:53 +00:00
DashMelee {
energy_cost: f32,
base_damage: f32,
scaled_damage: f32,
base_poise_damage: f32,
scaled_poise_damage: f32,
base_knockback: f32,
scaled_knockback: f32,
range: f32,
angle: f32,
energy_drain: f32,
forward_speed: f32,
buildup_duration: f32,
charge_duration: f32,
swing_duration: f32,
recover_duration: f32,
ori_modifier: f32,
2021-04-18 22:09:57 +00:00
charge_through: bool,
is_interruptible: bool,
damage_kind: DamageKind,
2021-07-10 16:04:12 +00:00
damage_effect: Option<CombatEffect>,
2020-03-16 15:34:53 +00:00
},
2021-04-10 03:40:20 +00:00
BasicBlock {
buildup_duration: f32,
recover_duration: f32,
max_angle: f32,
block_strength: f32,
2021-04-13 05:46:42 +00:00
energy_cost: f32,
2021-04-10 03:40:20 +00:00
},
Roll {
energy_cost: f32,
buildup_duration: f32,
movement_duration: f32,
recover_duration: f32,
roll_strength: f32,
immune_melee: bool,
},
ComboMelee {
stage_data: Vec<combo_melee::Stage<f32>>,
initial_energy_gain: f32,
max_energy_gain: f32,
energy_increase: f32,
speed_increase: f32,
max_speed_increase: f32,
scales_from_combo: u32,
is_interruptible: bool,
2021-04-30 02:59:29 +00:00
ori_modifier: f32,
2020-03-19 22:40:03 +00:00
},
LeapMelee {
energy_cost: f32,
buildup_duration: f32,
movement_duration: f32,
swing_duration: f32,
recover_duration: f32,
base_damage: f32,
base_poise_damage: f32,
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,
damage_kind: DamageKind,
2021-07-10 16:04:12 +00:00
damage_effect: Option<CombatEffect>,
},
2020-07-08 19:58:41 +00:00
SpinMelee {
buildup_duration: f32,
swing_duration: f32,
recover_duration: f32,
base_damage: f32,
base_poise_damage: f32,
knockback: Knockback,
range: f32,
damage_effect: Option<CombatEffect>,
energy_cost: f32,
is_infinite: bool,
2021-01-21 22:45:03 +00:00
movement_behavior: spin_melee::MovementBehavior,
is_interruptible: bool,
forward_speed: f32,
num_spins: u32,
specifier: Option<spin_melee::FrontendSpecifier>,
target: Option<combat::GroupTarget>,
damage_kind: DamageKind,
2020-07-08 19:58:41 +00:00
},
ChargedMelee {
energy_cost: f32,
energy_drain: f32,
initial_damage: f32,
scaled_damage: f32,
initial_poise_damage: f32,
scaled_poise_damage: f32,
initial_knockback: f32,
scaled_knockback: f32,
range: f32,
max_angle: f32,
charge_duration: f32,
swing_duration: f32,
hit_timing: f32,
recover_duration: f32,
specifier: Option<charged_melee::FrontendSpecifier>,
damage_kind: DamageKind,
2021-07-10 16:04:12 +00:00
damage_effect: Option<CombatEffect>,
},
2020-07-26 03:06:53 +00:00
ChargedRanged {
energy_cost: f32,
energy_drain: f32,
2021-05-16 03:09:39 +00:00
initial_regen: f32,
scaled_regen: f32,
initial_damage: f32,
scaled_damage: f32,
2020-07-26 03:06:53 +00:00
initial_knockback: f32,
scaled_knockback: f32,
buildup_duration: f32,
charge_duration: f32,
recover_duration: f32,
2020-07-26 03:06:53 +00:00
projectile_body: Body,
projectile_light: Option<LightEmitter>,
initial_projectile_speed: f32,
scaled_projectile_speed: f32,
2020-12-23 02:28:55 +00:00
move_speed: f32,
2020-07-26 03:06:53 +00:00
},
Shockwave {
energy_cost: f32,
buildup_duration: f32,
swing_duration: f32,
recover_duration: f32,
damage: f32,
poise_damage: f32,
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: f32,
2020-09-19 16:55:31 +00:00
requires_ground: bool,
2020-10-09 17:42:15 +00:00
move_efficiency: f32,
damage_kind: DamageKind,
specifier: comp::shockwave::FrontendSpecifier,
2021-07-10 16:04:12 +00:00
damage_effect: Option<CombatEffect>,
2020-08-08 20:53:55 +00:00
},
BasicBeam {
buildup_duration: f32,
recover_duration: f32,
beam_duration: f32,
damage: f32,
2020-08-31 21:55:38 +00:00
tick_rate: f32,
range: f32,
max_angle: f32,
damage_effect: Option<CombatEffect>,
energy_regen: f32,
energy_drain: f32,
ori_rate: f32,
specifier: beam::FrontendSpecifier,
},
BasicAura {
2021-03-01 20:44:29 +00:00
buildup_duration: f32,
cast_duration: f32,
recover_duration: f32,
targets: combat::GroupTarget,
aura: aura::AuraBuffConstructor,
aura_duration: f32,
2021-03-01 20:44:29 +00:00
range: f32,
energy_cost: f32,
2021-07-10 07:10:31 +00:00
scales_with_combo: bool,
specifier: aura::Specifier,
},
Blink {
buildup_duration: f32,
recover_duration: f32,
max_range: f32,
},
BasicSummon {
buildup_duration: f32,
cast_duration: f32,
recover_duration: f32,
summon_amount: u32,
2021-06-03 22:42:50 +00:00
summon_distance: (f32, f32),
summon_info: basic_summon::SummonInfo,
2021-06-03 22:42:50 +00:00
duration: Option<Duration>,
},
SelfBuff {
buildup_duration: f32,
cast_duration: f32,
recover_duration: f32,
buff_kind: buff::BuffKind,
buff_strength: f32,
buff_duration: Option<f32>,
energy_cost: f32,
},
2021-06-19 18:53:23 +00:00
SpriteSummon {
buildup_duration: f32,
cast_duration: f32,
recover_duration: f32,
sprite: SpriteKind,
summon_distance: (f32, f32),
sparseness: f64,
2021-06-19 18:53:23 +00:00
},
2020-01-01 17:16:29 +00:00
}
impl Default for CharacterAbility {
fn default() -> Self {
CharacterAbility::BasicMelee {
energy_cost: 0.0,
buildup_duration: 0.25,
swing_duration: 0.25,
recover_duration: 0.5,
base_damage: 10.0,
base_poise_damage: 0.0,
knockback: Knockback {
strength: 0.0,
direction: combat::KnockbackDir::Away,
},
range: 3.5,
max_angle: 15.0,
damage_effect: None,
damage_kind: DamageKind::Crushing,
}
}
}
impl Asset for CharacterAbility {
2020-12-12 22:14:24 +00:00
type Loader = assets::RonLoader;
2020-12-13 01:09:57 +00:00
const EXTENSION: &'static str = "ron";
}
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, .. } => {
2021-06-20 03:51:04 +00:00
data.physics.on_ground.is_some()
&& data.inputs.move_dir.magnitude_squared() > 0.25
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()
},
2021-03-01 20:44:29 +00:00
CharacterAbility::DashMelee { energy_cost, .. }
| CharacterAbility::BasicMelee { energy_cost, .. }
| CharacterAbility::BasicRanged { energy_cost, .. }
| CharacterAbility::SpinMelee { energy_cost, .. }
| CharacterAbility::ChargedRanged { energy_cost, .. }
| CharacterAbility::ChargedMelee { energy_cost, .. }
| CharacterAbility::Shockwave { energy_cost, .. }
| CharacterAbility::BasicBlock { energy_cost, .. }
| CharacterAbility::SelfBuff { energy_cost, .. } => update
2020-03-26 13:46:08 +00:00
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok(),
2021-05-16 03:09:39 +00:00
// Consumes energy within state, so value only checked before entering state
CharacterAbility::RepeaterRanged { energy_cost, .. } => {
update.energy.current() as f32 >= *energy_cost
},
2021-03-01 20:44:29 +00:00
CharacterAbility::LeapMelee { energy_cost, .. } => {
update.vel.0.z >= 0.0
&& update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok()
},
2021-07-10 07:10:31 +00:00
CharacterAbility::BasicAura {
energy_cost,
scales_with_combo,
..
} => {
((*scales_with_combo && data.combo.counter() > 0) | !*scales_with_combo)
&& update
.energy
.try_change_by(-(*energy_cost as i32), EnergySource::Ability)
.is_ok()
},
CharacterAbility::ComboMelee { .. }
| CharacterAbility::Boost { .. }
| CharacterAbility::BasicBeam { .. }
| CharacterAbility::Blink { .. }
2021-06-19 18:53:23 +00:00
| CharacterAbility::BasicSummon { .. }
| CharacterAbility::SpriteSummon { .. } => true,
2020-03-17 14:01:41 +00:00
}
}
pub fn default_roll() -> CharacterAbility {
CharacterAbility::Roll {
2021-04-05 21:50:53 +00:00
energy_cost: 120.0,
buildup_duration: 0.05,
2021-04-05 21:50:53 +00:00
movement_duration: 0.33,
recover_duration: 0.125,
2021-04-05 21:50:53 +00:00
roll_strength: 2.0,
immune_melee: true,
}
}
2021-04-10 03:40:20 +00:00
pub fn default_block() -> CharacterAbility {
CharacterAbility::BasicBlock {
buildup_duration: 0.35,
recover_duration: 0.3,
2021-04-10 03:40:20 +00:00
max_angle: 60.0,
block_strength: 0.5,
2021-04-13 05:46:42 +00:00
energy_cost: 50.0,
2021-04-10 03:40:20 +00:00
}
}
2021-07-06 01:32:12 +00:00
pub fn adjusted_by_stats(mut self, stats: Stats) -> Self {
use CharacterAbility::*;
match self {
BasicMelee {
ref mut energy_cost,
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut base_damage,
2020-12-16 23:30:33 +00:00
ref mut base_poise_damage,
knockback: _,
2021-07-06 01:32:12 +00:00
ref mut range,
max_angle: _,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
damage_kind: _,
} => {
*buildup_duration /= stats.speed;
*swing_duration /= stats.speed;
*recover_duration /= stats.speed;
*base_damage *= stats.power;
*base_poise_damage *= stats.effect_power;
*range *= stats.range;
*energy_cost /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
BasicRanged {
ref mut energy_cost,
ref mut buildup_duration,
ref mut recover_duration,
ref mut projectile,
projectile_body: _,
projectile_light: _,
2021-07-06 01:32:12 +00:00
ref mut projectile_speed,
num_projectiles: _,
projectile_spread: _,
} => {
*buildup_duration /= stats.speed;
*recover_duration /= stats.speed;
*projectile = projectile.modified_projectile(stats.power, 1_f32, 1_f32);
*projectile_speed *= stats.range;
*energy_cost /= stats.energy_efficiency;
},
RepeaterRanged {
ref mut energy_cost,
ref mut buildup_duration,
ref mut shoot_duration,
ref mut recover_duration,
max_speed: _,
half_speed_at: _,
ref mut projectile,
projectile_body: _,
projectile_light: _,
2021-07-06 01:32:12 +00:00
ref mut projectile_speed,
} => {
*buildup_duration /= stats.speed;
*shoot_duration /= stats.speed;
*recover_duration /= stats.speed;
*projectile = projectile.modified_projectile(stats.power, 1_f32, 1_f32);
*projectile_speed *= stats.range;
*energy_cost /= stats.energy_efficiency;
},
Boost {
ref mut movement_duration,
only_up: _,
speed: ref mut boost_speed,
max_exit_velocity: _,
} => {
*movement_duration /= stats.speed;
*boost_speed *= stats.power;
},
DashMelee {
ref mut energy_cost,
ref mut base_damage,
ref mut scaled_damage,
2020-12-16 23:30:33 +00:00
ref mut base_poise_damage,
ref mut scaled_poise_damage,
base_knockback: _,
scaled_knockback: _,
ref mut range,
angle: _,
ref mut energy_drain,
forward_speed: _,
ref mut buildup_duration,
charge_duration: _,
ref mut swing_duration,
ref mut recover_duration,
ori_modifier: _,
charge_through: _,
is_interruptible: _,
damage_kind: _,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
} => {
*base_damage *= stats.power;
*scaled_damage *= stats.power;
*base_poise_damage *= stats.effect_power;
*scaled_poise_damage *= stats.effect_power;
*buildup_duration /= stats.speed;
*swing_duration /= stats.speed;
*recover_duration /= stats.speed;
*range *= stats.range;
*energy_cost /= stats.energy_efficiency;
*energy_drain /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
2021-04-10 03:40:20 +00:00
BasicBlock {
ref mut buildup_duration,
ref mut recover_duration,
// Do we want angle to be adjusted by range?
max_angle: _,
// Block strength explicitly not modified by power, that will be a separate stat
block_strength: _,
ref mut energy_cost,
2021-04-10 03:40:20 +00:00
} => {
*buildup_duration /= stats.speed;
*recover_duration /= stats.speed;
*energy_cost /= stats.energy_efficiency;
2021-04-10 03:40:20 +00:00
},
Roll {
ref mut energy_cost,
ref mut buildup_duration,
ref mut movement_duration,
ref mut recover_duration,
roll_strength: _,
immune_melee: _,
} => {
*buildup_duration /= stats.speed;
*movement_duration /= stats.speed;
*recover_duration /= stats.speed;
*energy_cost /= stats.energy_efficiency;
},
ComboMelee {
ref mut stage_data,
initial_energy_gain: _,
max_energy_gain: _,
energy_increase: _,
speed_increase: _,
max_speed_increase: _,
scales_from_combo: _,
is_interruptible: _,
ori_modifier: _,
} => {
*stage_data = stage_data
.iter_mut()
2021-07-06 01:32:12 +00:00
.map(|s| s.adjusted_by_stats(stats))
.collect();
},
LeapMelee {
ref mut energy_cost,
ref mut buildup_duration,
movement_duration: _,
ref mut swing_duration,
ref mut recover_duration,
ref mut base_damage,
2020-12-16 23:30:33 +00:00
ref mut base_poise_damage,
2021-07-06 01:32:12 +00:00
ref mut range,
max_angle: _,
knockback: _,
forward_leap_strength: _,
vertical_leap_strength: _,
damage_kind: _,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
} => {
*buildup_duration /= stats.speed;
*swing_duration /= stats.speed;
*recover_duration /= stats.speed;
*base_damage *= stats.power;
*base_poise_damage *= stats.effect_power;
*range *= stats.range;
*energy_cost /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
SpinMelee {
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut base_damage,
2020-12-16 23:30:33 +00:00
ref mut base_poise_damage,
knockback: _,
2021-07-06 01:32:12 +00:00
ref mut range,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
ref mut energy_cost,
is_infinite: _,
movement_behavior: _,
is_interruptible: _,
forward_speed: _,
num_spins: _,
specifier: _,
target: _,
damage_kind: _,
} => {
*buildup_duration /= stats.speed;
*swing_duration /= stats.speed;
*recover_duration /= stats.speed;
*base_damage *= stats.power;
*base_poise_damage *= stats.effect_power;
*range *= stats.range;
*energy_cost /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
ChargedMelee {
ref mut energy_cost,
ref mut energy_drain,
ref mut initial_damage,
ref mut scaled_damage,
2020-12-16 23:30:33 +00:00
ref mut initial_poise_damage,
ref mut scaled_poise_damage,
initial_knockback: _,
scaled_knockback: _,
ref mut range,
max_angle: _,
charge_duration: _,
ref mut swing_duration,
hit_timing: _,
ref mut recover_duration,
specifier: _,
damage_kind: _,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
} => {
*initial_damage *= stats.power;
*scaled_damage *= stats.power;
*initial_poise_damage *= stats.effect_power;
*scaled_poise_damage *= stats.effect_power;
*swing_duration /= stats.speed;
*recover_duration /= stats.speed;
*range *= stats.range;
*energy_cost /= stats.energy_efficiency;
*energy_drain /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
ChargedRanged {
ref mut energy_cost,
ref mut energy_drain,
initial_regen: _,
scaled_regen: _,
ref mut initial_damage,
ref mut scaled_damage,
initial_knockback: _,
scaled_knockback: _,
ref mut buildup_duration,
charge_duration: _,
ref mut recover_duration,
projectile_body: _,
projectile_light: _,
2021-07-06 01:32:12 +00:00
ref mut initial_projectile_speed,
ref mut scaled_projectile_speed,
move_speed: _,
} => {
*initial_damage *= stats.power;
*scaled_damage *= stats.power;
*buildup_duration /= stats.speed;
*recover_duration /= stats.speed;
*initial_projectile_speed *= stats.range;
*scaled_projectile_speed *= stats.range;
*energy_cost /= stats.energy_efficiency;
*energy_drain /= stats.energy_efficiency;
},
Shockwave {
ref mut energy_cost,
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
ref mut damage,
2020-12-16 23:30:33 +00:00
ref mut poise_damage,
knockback: _,
shockwave_angle: _,
shockwave_vertical_angle: _,
shockwave_speed: _,
2021-07-06 01:32:12 +00:00
ref mut shockwave_duration,
requires_ground: _,
move_efficiency: _,
damage_kind: _,
specifier: _,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
} => {
*buildup_duration /= stats.speed;
*swing_duration /= stats.speed;
*recover_duration /= stats.speed;
*damage *= stats.power;
*poise_damage *= stats.effect_power;
*shockwave_duration *= stats.range;
*energy_cost /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
BasicBeam {
ref mut buildup_duration,
ref mut recover_duration,
ref mut beam_duration,
ref mut damage,
ref mut tick_rate,
2021-07-06 01:32:12 +00:00
ref mut range,
max_angle: _,
2021-07-10 16:04:12 +00:00
ref mut damage_effect,
energy_regen: _,
ref mut energy_drain,
ori_rate: _,
specifier: _,
} => {
*buildup_duration /= stats.speed;
*recover_duration /= stats.speed;
*damage *= stats.power;
*tick_rate *= stats.speed;
*range *= stats.range;
// Duration modified to keep velocity constant
*beam_duration *= stats.range;
*energy_drain /= stats.energy_efficiency;
2021-07-10 16:04:12 +00:00
if let Some(CombatEffect::Buff(combat::CombatBuff {
kind: _,
dur_secs: _,
strength,
chance: _,
})) = damage_effect
{
*strength *= stats.buff_strength;
}
},
BasicAura {
2021-03-01 20:44:29 +00:00
ref mut buildup_duration,
ref mut cast_duration,
2021-03-01 20:44:29 +00:00
ref mut recover_duration,
targets: _,
aura:
aura::AuraBuffConstructor {
kind: _,
ref mut strength,
duration: _,
category: _,
},
aura_duration: _,
2021-07-06 01:32:12 +00:00
ref mut range,
ref mut energy_cost,
2021-07-10 07:10:31 +00:00
scales_with_combo: _,
specifier: _,
2021-03-01 20:44:29 +00:00
} => {
*buildup_duration /= stats.speed;
*cast_duration /= stats.speed;
*recover_duration /= stats.speed;
2021-07-10 16:04:12 +00:00
// Do we want to make buff_strength affect this instead of power?
// Look into during modular weapon transition
*strength *= stats.power;
*range *= stats.range;
*energy_cost /= stats.energy_efficiency;
2021-03-01 20:44:29 +00:00
},
Blink {
ref mut buildup_duration,
ref mut recover_duration,
2021-07-06 01:32:12 +00:00
ref mut max_range,
} => {
*buildup_duration /= stats.speed;
*recover_duration /= stats.speed;
*max_range *= stats.range;
},
BasicSummon {
ref mut buildup_duration,
ref mut cast_duration,
ref mut recover_duration,
summon_amount: _,
summon_distance: (ref mut inner_dist, ref mut outer_dist),
summon_info: _,
duration: _,
} => {
// TODO: Figure out how/if power should affect this
*buildup_duration /= stats.speed;
*cast_duration /= stats.speed;
*recover_duration /= stats.speed;
*inner_dist *= stats.range;
*outer_dist *= stats.range;
},
SelfBuff {
ref mut buildup_duration,
ref mut cast_duration,
ref mut recover_duration,
buff_kind: _,
ref mut buff_strength,
buff_duration: _,
ref mut energy_cost,
} => {
2021-07-10 16:04:12 +00:00
// Do we want to make buff_strength affect this instead of power?
// Look into during modular weapon transition
*buff_strength *= stats.power;
*buildup_duration /= stats.speed;
*cast_duration /= stats.speed;
*recover_duration /= stats.speed;
*energy_cost /= stats.energy_efficiency;
},
2021-06-19 18:53:23 +00:00
SpriteSummon {
ref mut buildup_duration,
ref mut cast_duration,
ref mut recover_duration,
sprite: _,
summon_distance: (ref mut inner_dist, ref mut outer_dist),
sparseness: _,
2021-06-19 18:53:23 +00:00
} => {
// TODO: Figure out how/if power should affect this
*buildup_duration /= stats.speed;
*cast_duration /= stats.speed;
*recover_duration /= stats.speed;
*inner_dist *= stats.range;
*outer_dist *= stats.range;
2021-06-19 18:53:23 +00:00
},
}
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, .. }
2021-04-13 05:46:42 +00:00
| BasicAura { energy_cost, .. }
| BasicBlock { energy_cost, .. }
| SelfBuff { energy_cost, .. } => *energy_cost as u32,
BasicBeam { energy_drain, .. } => {
if *energy_drain > f32::EPSILON {
1
} else {
0
}
},
2021-06-19 18:53:23 +00:00
Boost { .. }
| ComboMelee { .. }
| Blink { .. }
| BasicSummon { .. }
| SpriteSummon { .. } => 0,
}
}
2020-12-07 03:35:29 +00:00
2021-08-16 09:16:20 +00:00
#[must_use = "method returns new ability and doesn't mutate the original value"]
#[warn(clippy::pedantic)]
2020-12-07 03:35:29 +00:00
pub fn adjusted_by_skills(
mut self,
skillset: &skills::SkillSet,
2020-12-07 03:35:29 +00:00
tool: Option<ToolKind>,
) -> Self {
2021-08-16 09:16:20 +00:00
use skills::Skill;
match tool {
2021-08-16 09:16:20 +00:00
Some(ToolKind::Sword) => self.adjusted_by_sword_skills(skillset),
Some(ToolKind::Axe) => self.adjusted_by_axe_skills(skillset),
Some(ToolKind::Hammer) => self.adjusted_by_hammer_skills(skillset),
Some(ToolKind::Bow) => self.adjusted_by_bow_skills(skillset),
Some(ToolKind::Staff) => self.adjusted_by_staff_skills(skillset),
Some(ToolKind::Sceptre) => self.adjusted_by_sceptre_skills(skillset),
2021-06-09 05:14:20 +00:00
Some(ToolKind::Pick) => {
2021-08-16 09:16:20 +00:00
use skills::MiningSkill::Speed;
if let CharacterAbility::BasicMelee {
2021-06-09 05:14:20 +00:00
ref mut buildup_duration,
ref mut swing_duration,
ref mut recover_duration,
..
} = self
{
2021-08-16 09:16:20 +00:00
if let Ok(Some(level)) = skillset.skill_level(Skill::Pick(Speed)) {
2021-06-09 05:14:20 +00:00
let speed = 1.1_f32.powi(level.into());
*buildup_duration /= speed;
*swing_duration /= speed;
*recover_duration /= speed;
}
}
},
2021-03-20 23:35:34 +00:00
None => {
if let CharacterAbility::Roll {
ref mut energy_cost,
ref mut roll_strength,
ref mut movement_duration,
..
} = self
{
2021-08-16 09:16:20 +00:00
use skills::RollSkill::{Cost, Duration, Strength};
2021-03-20 23:35:34 +00:00
if let Ok(Some(level)) = skillset.skill_level(Skill::Roll(Cost)) {
2021-04-05 21:50:53 +00:00
*energy_cost *= 0.9_f32.powi(level.into());
2021-03-20 23:35:34 +00:00
}
if let Ok(Some(level)) = skillset.skill_level(Skill::Roll(Strength)) {
2021-04-05 21:50:53 +00:00
*roll_strength *= 1.1_f32.powi(level.into());
2021-03-20 23:35:34 +00:00
}
if let Ok(Some(level)) = skillset.skill_level(Skill::Roll(Duration)) {
2021-04-05 21:50:53 +00:00
*movement_duration *= 1.1_f32.powi(level.into());
2021-03-20 23:35:34 +00:00
}
}
},
Some(_) => {},
2020-12-07 03:35:29 +00:00
}
self
}
2021-08-16 09:16:20 +00:00
#[warn(clippy::pedantic)]
fn adjusted_by_sword_skills(&mut self, skillset: &skills::SkillSet) {
#![allow(clippy::enum_glob_use)]
use skills::{Skill::Sword, SwordSkill::*};
match self {
CharacterAbility::ComboMelee {
ref mut is_interruptible,
ref mut speed_increase,
ref mut max_speed_increase,
ref stage_data,
ref mut max_energy_gain,
ref mut scales_from_combo,
..
} => {
*is_interruptible = skillset.has_skill(Sword(InterruptingAttacks));
if skillset.has_skill(Sword(TsCombo)) {
let speed_segments = Sword(TsSpeed)
.max_level()
.map_or(1.0, |l| f32::from(l) + 1.0);
let speed_level = f32::from(skillset.skill_level_or(Sword(TsSpeed), 0));
*speed_increase = (speed_level + 1.0) / speed_segments;
*max_speed_increase = (speed_level + 1.0) / speed_segments;
} else {
*speed_increase = 0.0;
*max_speed_increase = 0.0;
}
let energy_level = skillset.skill_level_or(Sword(TsRegen), 0);
let stages = u16::try_from(stage_data.len())
.expect("number of stages can't be more than u16");
*max_energy_gain *= f32::from((energy_level + 1) * stages - 1)
* f32::from(stages - 1)
/ f32::from(Sword(TsRegen).max_level().unwrap() + 1);
*scales_from_combo = skillset.skill_level_or(Sword(TsDamage), 0).into();
},
CharacterAbility::DashMelee {
ref mut is_interruptible,
ref mut energy_cost,
ref mut energy_drain,
ref mut base_damage,
ref mut scaled_damage,
ref mut forward_speed,
ref mut charge_through,
..
} => {
*is_interruptible = skillset.has_skill(Sword(InterruptingAttacks));
if let Ok(Some(level)) = skillset.skill_level(Sword(DCost)) {
*energy_cost *= 0.75_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sword(DDrain)) {
*energy_drain *= 0.75_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sword(DDamage)) {
*base_damage *= 1.2_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sword(DScaling)) {
*scaled_damage *= 1.2_f32.powi(level.into());
}
if skillset.has_skill(Sword(DSpeed)) {
*forward_speed *= 1.15;
}
*charge_through = skillset.has_skill(Sword(DInfinite));
},
CharacterAbility::SpinMelee {
ref mut is_interruptible,
ref mut base_damage,
ref mut swing_duration,
ref mut energy_cost,
ref mut num_spins,
..
} => {
*is_interruptible = skillset.has_skill(Sword(InterruptingAttacks));
if let Ok(Some(level)) = skillset.skill_level(Sword(SDamage)) {
*base_damage *= 1.4_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sword(SSpeed)) {
*swing_duration *= 0.8_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sword(SCost)) {
*energy_cost *= 0.75_f32.powi(level.into());
}
let spin_level = skillset.skill_level_or(Sword(SSpins), 0);
*num_spins = u32::from(spin_level) + 1;
},
_ => {},
}
}
#[warn(clippy::pedantic)]
fn adjusted_by_axe_skills(&mut self, skillset: &skills::SkillSet) {
#![allow(clippy::enum_glob_use)]
use skills::{AxeSkill::*, Skill::Axe};
match self {
CharacterAbility::ComboMelee {
ref mut speed_increase,
ref mut max_speed_increase,
ref mut stage_data,
ref mut max_energy_gain,
ref mut scales_from_combo,
..
} => {
if !skillset.has_skill(Axe(DsCombo)) {
stage_data.pop();
}
let speed_segments = f32::from(Axe(DsSpeed).max_level().unwrap_or(1));
let speed_level = f32::from(skillset.skill_level_or(Axe(DsSpeed), 0));
*speed_increase *= speed_level / speed_segments;
*max_speed_increase *= speed_level / speed_segments;
let energy_level = skillset.skill_level_or(Axe(DsRegen), 0);
let stages = u16::try_from(stage_data.len())
.expect("number of stages can't be more than u16");
*max_energy_gain *= f32::from((energy_level + 1) * stages - 1).max(1.0)
* f32::from(stages - 1).max(1.0)
/ f32::from(Axe(DsRegen).max_level().unwrap() + 1);
*scales_from_combo = skillset.skill_level_or(Axe(DsDamage), 0).into();
},
CharacterAbility::SpinMelee {
ref mut base_damage,
ref mut swing_duration,
ref mut energy_cost,
ref mut is_infinite,
ref mut movement_behavior,
..
} => {
*is_infinite = skillset.has_skill(Axe(SInfinite));
*movement_behavior = if skillset.has_skill(Axe(SHelicopter)) {
spin_melee::MovementBehavior::AxeHover
} else {
spin_melee::MovementBehavior::ForwardGround
};
if let Ok(Some(level)) = skillset.skill_level(Axe(SDamage)) {
*base_damage *= 1.3_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Axe(SSpeed)) {
*swing_duration *= 0.8_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Axe(SCost)) {
*energy_cost *= 0.75_f32.powi(level.into());
}
},
CharacterAbility::LeapMelee {
ref mut base_damage,
ref mut knockback,
ref mut energy_cost,
ref mut forward_leap_strength,
ref mut vertical_leap_strength,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Axe(LDamage)) {
*base_damage *= 1.35_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Axe(LKnockback)) {
*knockback *= 1.4_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Axe(LCost)) {
*energy_cost *= 0.75_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Axe(LDistance)) {
*forward_leap_strength *= 1.2_f32.powi(level.into());
*vertical_leap_strength *= 1.2_f32.powi(level.into());
}
},
_ => {},
}
}
#[warn(clippy::pedantic)]
fn adjusted_by_hammer_skills(&mut self, skillset: &skills::SkillSet) {
#![allow(clippy::enum_glob_use)]
use skills::{HammerSkill::*, Skill::Hammer};
match self {
CharacterAbility::ComboMelee {
ref mut speed_increase,
ref mut max_speed_increase,
ref mut stage_data,
ref mut max_energy_gain,
ref mut scales_from_combo,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Hammer(SsKnockback)) {
*stage_data = (*stage_data)
.iter()
.map(|s| s.modify_strike(1.5_f32.powi(level.into())))
.collect::<Vec<_>>();
}
let speed_segments = f32::from(Hammer(SsSpeed).max_level().unwrap_or(1));
let speed_level = f32::from(skillset.skill_level_or(Hammer(SsSpeed), 0));
*speed_increase *= speed_level / speed_segments;
*max_speed_increase *= speed_level / speed_segments;
let energy_level = skillset.skill_level_or(Hammer(SsRegen), 0);
let stages = u16::try_from(stage_data.len())
.expect("number of stages can't be more than u16");
*max_energy_gain *= f32::from((energy_level + 1) * stages)
/ f32::from((Hammer(SsRegen).max_level().unwrap() + 1) * stages);
*scales_from_combo = skillset.skill_level_or(Hammer(SsDamage), 0).into();
},
CharacterAbility::ChargedMelee {
ref mut scaled_damage,
ref mut scaled_knockback,
ref mut energy_drain,
ref mut charge_duration,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Hammer(CDamage)) {
*scaled_damage *= 1.25_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(CKnockback)) {
*scaled_knockback *= 1.5_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(CDrain)) {
*energy_drain *= 0.75_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(CSpeed)) {
*charge_duration /= 1.25_f32.powi(level.into());
}
},
CharacterAbility::LeapMelee {
ref mut base_damage,
ref mut knockback,
ref mut energy_cost,
ref mut forward_leap_strength,
ref mut vertical_leap_strength,
ref mut range,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Hammer(LDamage)) {
*base_damage *= 1.4_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(LKnockback)) {
*knockback *= 1.5_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(LCost)) {
*energy_cost *= 0.75_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(LDistance)) {
*forward_leap_strength *= 1.25_f32.powi(level.into());
*vertical_leap_strength *= 1.25_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Hammer(LRange)) {
*range += 1.0 * f32::from(level);
}
},
_ => {},
}
}
#[warn(clippy::pedantic)]
fn adjusted_by_bow_skills(&mut self, skillset: &skills::SkillSet) {
#![allow(clippy::enum_glob_use)]
use skills::{BowSkill::*, Skill::Bow};
match self {
CharacterAbility::ChargedRanged {
ref mut initial_damage,
ref mut scaled_damage,
ref mut initial_regen,
ref mut scaled_regen,
ref mut initial_knockback,
ref mut scaled_knockback,
ref mut move_speed,
ref mut initial_projectile_speed,
ref mut scaled_projectile_speed,
ref mut charge_duration,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Bow(ProjSpeed)) {
let projectile_speed_scaling = 1.2_f32.powi(level.into());
*initial_projectile_speed *= projectile_speed_scaling;
*scaled_projectile_speed *= projectile_speed_scaling;
}
if let Ok(Some(level)) = skillset.skill_level(Bow(CDamage)) {
let damage_scaling = 1.2_f32.powi(level.into());
*initial_damage *= damage_scaling;
*scaled_damage *= damage_scaling;
}
if let Ok(Some(level)) = skillset.skill_level(Bow(CRegen)) {
let regen_scaling = 1.2_f32.powi(level.into());
*initial_regen *= regen_scaling;
*scaled_regen *= regen_scaling;
}
if let Ok(Some(level)) = skillset.skill_level(Bow(CKnockback)) {
let knockback_scaling = 1.2_f32.powi(level.into());
*initial_knockback *= knockback_scaling;
*scaled_knockback *= knockback_scaling;
}
if let Ok(Some(level)) = skillset.skill_level(Bow(CSpeed)) {
*charge_duration /= 1.1_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Bow(CMove)) {
*move_speed *= 1.1_f32.powi(level.into());
}
},
CharacterAbility::RepeaterRanged {
ref mut energy_cost,
ref mut projectile,
ref mut max_speed,
ref mut projectile_speed,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Bow(ProjSpeed)) {
*projectile_speed *= 1.2_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Bow(RDamage)) {
let power = 1.2_f32.powi(level.into());
*projectile = projectile.modified_projectile(power, 1_f32, 1_f32);
}
if let Ok(Some(level)) = skillset.skill_level(Bow(RCost)) {
*energy_cost *= 0.8_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Bow(RSpeed)) {
*max_speed *= 1.2_f32.powi(level.into());
}
},
CharacterAbility::BasicRanged {
ref mut projectile,
ref mut energy_cost,
ref mut num_projectiles,
ref mut projectile_spread,
ref mut projectile_speed,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Bow(ProjSpeed)) {
*projectile_speed *= 1.2_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Bow(SDamage)) {
let power = 1.2_f32.powi(level.into());
*projectile = projectile.modified_projectile(power, 1_f32, 1_f32);
}
if let Ok(Some(level)) = skillset.skill_level(Bow(SCost)) {
*energy_cost *= 0.8_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Bow(SArrows)) {
*num_projectiles += u32::from(level);
}
if let Ok(Some(level)) = skillset.skill_level(Bow(SSpread)) {
*projectile_spread *= 0.8_f32.powi(level.into());
}
},
_ => {},
}
}
#[warn(clippy::pedantic)]
fn adjusted_by_staff_skills(&mut self, skillset: &skills::SkillSet) {
#![allow(clippy::enum_glob_use)]
use skills::{Skill::Staff, StaffSkill::*};
match self {
CharacterAbility::BasicRanged {
ref mut projectile, ..
} => {
let damage_level = skillset.skill_level_or(Staff(BDamage), 0);
let regen_level = skillset.skill_level_or(Staff(BRegen), 0);
let range_level = skillset.skill_level_or(Staff(BRadius), 0);
let power = 1.2_f32.powi(damage_level.into());
let regen = 1.2_f32.powi(regen_level.into());
let range = 1.15_f32.powi(range_level.into());
*projectile = projectile.modified_projectile(power, regen, range);
},
CharacterAbility::BasicBeam {
ref mut damage,
ref mut range,
ref mut energy_drain,
ref mut beam_duration,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Staff(FDamage)) {
*damage *= 1.3_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Staff(FRange)) {
let range_mod = 1.25_f32.powi(level.into());
*range *= range_mod;
// Duration modified to keep velocity constant
*beam_duration *= range_mod;
}
if let Ok(Some(level)) = skillset.skill_level(Staff(FDrain)) {
*energy_drain *= 0.8_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Staff(FVelocity)) {
let velocity_increase = 1.25_f32.powi(level.into());
let duration_mod = 1.0 / (1.0 + velocity_increase);
*beam_duration *= duration_mod;
}
},
CharacterAbility::Shockwave {
ref mut damage,
ref mut knockback,
ref mut shockwave_duration,
ref mut energy_cost,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Staff(SDamage)) {
*damage *= 1.3_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Staff(SKnockback)) {
*knockback = knockback.modify_strength(1.3_f32.powi(level.into()));
}
if let Ok(Some(level)) = skillset.skill_level(Staff(SRange)) {
*shockwave_duration *= 1.2_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Staff(SCost)) {
*energy_cost *= 0.8_f32.powi(level.into());
}
},
_ => {},
}
}
#[warn(clippy::pedantic)]
fn adjusted_by_sceptre_skills(&mut self, skillset: &skills::SkillSet) {
#![allow(clippy::enum_glob_use)]
use skills::{SceptreSkill::*, Skill::Sceptre};
match self {
CharacterAbility::BasicBeam {
ref mut damage,
ref mut range,
ref mut beam_duration,
ref mut damage_effect,
ref mut energy_regen,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Sceptre(LDamage)) {
*damage *= 1.2_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(LRange)) {
let range_mod = 1.2_f32.powi(level.into());
*range *= range_mod;
// Duration modified to keep velocity constant
*beam_duration *= range_mod;
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(LRegen)) {
*energy_regen *= 1.2_f32.powi(level.into());
}
if let (Ok(Some(level)), Some(CombatEffect::Lifesteal(ref mut lifesteal))) =
(skillset.skill_level(Sceptre(LLifesteal)), damage_effect)
{
*lifesteal *= 1.15_f32.powi(level.into());
}
},
CharacterAbility::BasicAura {
ref mut aura,
ref mut range,
ref mut energy_cost,
specifier: aura::Specifier::WardingAura,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Sceptre(AStrength)) {
aura.strength *= 1.15_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(ADuration)) {
aura.duration.map(|dur| dur * 1.2_f32.powi(level.into()));
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(ARange)) {
*range *= 1.25_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(ACost)) {
*energy_cost *= 0.85_f32.powi(level.into());
}
},
CharacterAbility::BasicAura {
ref mut aura,
ref mut range,
ref mut energy_cost,
specifier: aura::Specifier::HealingAura,
..
} => {
if let Ok(Some(level)) = skillset.skill_level(Sceptre(HHeal)) {
aura.strength *= 1.15_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(HDuration)) {
aura.duration.map(|dur| dur * 1.2_f32.powi(level.into()));
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(HRange)) {
*range *= 1.25_f32.powi(level.into());
}
if let Ok(Some(level)) = skillset.skill_level(Sceptre(HCost)) {
*energy_cost *= 0.85_f32.powi(level.into());
}
},
_ => {},
}
}
2020-03-17 14:01:41 +00:00
}
impl From<(&CharacterAbility, AbilityInfo, &JoinData<'_>)> for CharacterState {
fn from((ability, ability_info, data): (&CharacterAbility, AbilityInfo, &JoinData)) -> 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,
base_poise_damage,
knockback,
range,
max_angle,
damage_effect,
2020-03-24 21:03:11 +00:00
energy_cost: _,
damage_kind,
} => CharacterState::BasicMelee(basic_melee::Data {
static_data: basic_melee::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
swing_duration: Duration::from_secs_f32(*swing_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
base_damage: *base_damage,
base_poise_damage: *base_poise_damage,
knockback: *knockback,
range: *range,
max_angle: *max_angle,
damage_effect: *damage_effect,
ability_info,
damage_kind: *damage_kind,
},
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_speed,
2020-03-24 19:09:23 +00:00
energy_cost: _,
2021-05-16 03:09:39 +00:00
num_projectiles,
projectile_spread,
} => CharacterState::BasicRanged(basic_ranged::Data {
static_data: basic_ranged::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
projectile: *projectile,
projectile_body: *projectile_body,
projectile_light: *projectile_light,
projectile_speed: *projectile_speed,
2021-05-16 03:09:39 +00:00
num_projectiles: *num_projectiles,
projectile_spread: *projectile_spread,
ability_info,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
exhausted: false,
}),
CharacterAbility::Boost {
movement_duration,
only_up,
speed,
max_exit_velocity,
} => CharacterState::Boost(boost::Data {
static_data: boost::StaticData {
movement_duration: Duration::from_secs_f32(*movement_duration),
only_up: *only_up,
speed: *speed,
max_exit_velocity: *max_exit_velocity,
ability_info,
},
timer: Duration::default(),
}),
2020-03-16 15:34:53 +00:00
CharacterAbility::DashMelee {
energy_cost: _,
base_damage,
scaled_damage,
base_poise_damage,
2020-12-16 23:30:33 +00:00
scaled_poise_damage,
base_knockback,
scaled_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,
ori_modifier,
2021-04-18 22:09:57 +00:00
charge_through,
is_interruptible,
damage_kind,
2021-07-10 16:04:12 +00:00
damage_effect,
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,
scaled_damage: *scaled_damage,
base_poise_damage: *base_poise_damage,
2020-12-16 23:30:33 +00:00
scaled_poise_damage: *scaled_poise_damage,
2020-09-11 13:59:45 +00:00
base_knockback: *base_knockback,
scaled_knockback: *scaled_knockback,
2020-09-11 13:59:45 +00:00
range: *range,
angle: *angle,
energy_drain: *energy_drain,
forward_speed: *forward_speed,
2021-04-18 22:09:57 +00:00
charge_through: *charge_through,
buildup_duration: Duration::from_secs_f32(*buildup_duration),
charge_duration: Duration::from_secs_f32(*charge_duration),
swing_duration: Duration::from_secs_f32(*swing_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
ori_modifier: *ori_modifier,
is_interruptible: *is_interruptible,
2021-07-10 16:04:12 +00:00
damage_effect: *damage_effect,
ability_info,
damage_kind: *damage_kind,
2020-09-11 13:59:45 +00:00
},
2020-10-18 16:46:28 +00:00
auto_charge: false,
timer: Duration::default(),
2021-04-18 22:09:57 +00:00
charge_end_timer: Duration::from_secs_f32(*charge_duration),
stage_section: StageSection::Buildup,
2020-09-11 13:59:45 +00:00
exhausted: false,
2020-03-16 15:34:53 +00:00
}),
2021-04-10 03:40:20 +00:00
CharacterAbility::BasicBlock {
buildup_duration,
recover_duration,
max_angle,
block_strength,
2021-04-13 05:46:42 +00:00
energy_cost: _,
2021-04-10 03:40:20 +00:00
} => CharacterState::BasicBlock(basic_block::Data {
static_data: basic_block::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
max_angle: *max_angle,
block_strength: *block_strength,
ability_info,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
}),
CharacterAbility::Roll {
energy_cost: _,
buildup_duration,
movement_duration,
recover_duration,
roll_strength,
immune_melee,
} => CharacterState::Roll(roll::Data {
static_data: roll::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
movement_duration: Duration::from_secs_f32(*movement_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
roll_strength: *roll_strength,
immune_melee: *immune_melee,
ability_info,
},
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,
scales_from_combo,
is_interruptible,
2021-04-30 02:59:29 +00:00
ori_modifier,
} => 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,
2020-12-07 03:35:29 +00:00
max_speed_increase: *max_speed_increase,
scales_from_combo: *scales_from_combo,
2020-09-21 22:38:01 +00:00
is_interruptible: *is_interruptible,
2021-04-30 02:59:29 +00:00
ori_modifier: *ori_modifier as f32,
ability_info,
2020-09-21 22:38:01 +00:00
},
exhausted: false,
stage: 1,
timer: Duration::default(),
stage_section: StageSection::Buildup,
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,
base_poise_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,
damage_kind,
2021-07-10 16:04:12 +00:00
damage_effect,
} => CharacterState::LeapMelee(leap_melee::Data {
2020-09-28 23:55:38 +00:00
static_data: leap_melee::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
movement_duration: Duration::from_secs_f32(*movement_duration),
swing_duration: Duration::from_secs_f32(*swing_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
2020-09-28 23:55:38 +00:00
base_damage: *base_damage,
base_poise_damage: *base_poise_damage,
2020-09-28 23:55:38 +00:00
knockback: *knockback,
range: *range,
max_angle: *max_angle,
forward_leap_strength: *forward_leap_strength,
vertical_leap_strength: *vertical_leap_strength,
2021-07-10 16:04:12 +00:00
damage_effect: *damage_effect,
ability_info,
damage_kind: *damage_kind,
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,
base_poise_damage,
knockback,
range,
damage_effect,
energy_cost,
is_infinite,
2021-01-21 22:45:03 +00:00
movement_behavior,
is_interruptible,
forward_speed,
num_spins,
specifier,
target,
damage_kind,
2020-07-08 19:58:41 +00:00
} => CharacterState::SpinMelee(spin_melee::Data {
static_data: spin_melee::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
swing_duration: Duration::from_secs_f32(*swing_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
base_damage: *base_damage,
2020-12-16 23:30:33 +00:00
base_poise_damage: *base_poise_damage,
knockback: *knockback,
range: *range,
damage_effect: *damage_effect,
energy_cost: *energy_cost,
is_infinite: *is_infinite,
2021-01-21 22:45:03 +00:00
movement_behavior: *movement_behavior,
is_interruptible: *is_interruptible,
forward_speed: *forward_speed,
num_spins: *num_spins,
target: *target,
ability_info,
specifier: *specifier,
damage_kind: *damage_kind,
},
timer: Duration::default(),
2021-03-29 14:44:46 +00:00
consecutive_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,
scaled_damage,
initial_poise_damage,
2020-12-16 23:30:33 +00:00
scaled_poise_damage,
initial_knockback,
scaled_knockback,
charge_duration,
2020-09-28 01:58:49 +00:00
swing_duration,
hit_timing,
recover_duration,
range,
max_angle,
specifier,
damage_kind,
2021-07-10 16:04:12 +00:00
damage_effect,
} => 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,
scaled_damage: *scaled_damage,
initial_poise_damage: *initial_poise_damage,
2020-12-16 23:30:33 +00:00
scaled_poise_damage: *scaled_poise_damage,
2020-09-28 01:58:49 +00:00
initial_knockback: *initial_knockback,
scaled_knockback: *scaled_knockback,
2020-09-28 01:58:49 +00:00
range: *range,
max_angle: *max_angle,
charge_duration: Duration::from_secs_f32(*charge_duration),
swing_duration: Duration::from_secs_f32(*swing_duration),
hit_timing: *hit_timing,
recover_duration: Duration::from_secs_f32(*recover_duration),
2021-07-10 16:04:12 +00:00
damage_effect: *damage_effect,
ability_info,
specifier: *specifier,
damage_kind: *damage_kind,
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,
2021-05-16 03:09:39 +00:00
initial_regen,
scaled_regen,
2020-07-26 03:06:53 +00:00
initial_damage,
scaled_damage,
2020-07-26 03:06:53 +00:00
initial_knockback,
scaled_knockback,
buildup_duration,
2020-07-26 03:06:53 +00:00
charge_duration,
recover_duration,
projectile_body,
projectile_light,
initial_projectile_speed,
scaled_projectile_speed,
2020-12-23 02:28:55 +00:00
move_speed,
2020-07-26 03:06:53 +00:00
} => CharacterState::ChargedRanged(charged_ranged::Data {
static_data: charged_ranged::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
charge_duration: Duration::from_secs_f32(*charge_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
energy_drain: *energy_drain,
2021-05-16 03:09:39 +00:00
initial_regen: *initial_regen,
scaled_regen: *scaled_regen,
initial_damage: *initial_damage,
scaled_damage: *scaled_damage,
initial_knockback: *initial_knockback,
scaled_knockback: *scaled_knockback,
projectile_body: *projectile_body,
projectile_light: *projectile_light,
initial_projectile_speed: *initial_projectile_speed,
scaled_projectile_speed: *scaled_projectile_speed,
2020-12-23 02:28:55 +00:00
move_speed: *move_speed,
ability_info,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-07-26 03:06:53 +00:00
exhausted: false,
}),
CharacterAbility::RepeaterRanged {
2021-05-16 03:09:39 +00:00
energy_cost,
2020-09-28 02:38:23 +00:00
buildup_duration,
shoot_duration,
recover_duration,
2021-05-16 03:09:39 +00:00
max_speed,
half_speed_at,
projectile,
projectile_body,
projectile_light,
projectile_speed,
} => CharacterState::RepeaterRanged(repeater_ranged::Data {
2020-09-28 02:38:23 +00:00
static_data: repeater_ranged::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
shoot_duration: Duration::from_secs_f32(*shoot_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
2021-05-16 03:09:39 +00:00
energy_cost: *energy_cost,
// 1.0 is subtracted as 1.0 is added in state file
max_speed: *max_speed - 1.0,
half_speed_at: *half_speed_at,
projectile: *projectile,
2020-09-28 02:38:23 +00:00
projectile_body: *projectile_body,
projectile_light: *projectile_light,
projectile_speed: *projectile_speed,
ability_info,
2020-09-28 02:38:23 +00:00
},
timer: Duration::default(),
2021-05-16 03:09:39 +00:00
stage_section: StageSection::Buildup,
projectiles_fired: 0,
speed: 1.0,
}),
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,
poise_damage,
2020-08-08 20:53:55 +00:00
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,
damage_kind,
specifier,
2021-07-10 16:04:12 +00:00
damage_effect,
} => CharacterState::Shockwave(shockwave::Data {
static_data: shockwave::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
swing_duration: Duration::from_secs_f32(*swing_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
damage: *damage,
poise_damage: *poise_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_secs_f32(*shockwave_duration),
requires_ground: *requires_ground,
2020-10-09 17:42:15 +00:00
move_efficiency: *move_efficiency,
2021-07-10 16:04:12 +00:00
damage_effect: *damage_effect,
ability_info,
damage_kind: *damage_kind,
specifier: *specifier,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
2020-08-08 20:53:55 +00:00
}),
CharacterAbility::BasicBeam {
buildup_duration,
recover_duration,
beam_duration,
damage,
2020-08-31 21:55:38 +00:00
tick_rate,
range,
max_angle,
damage_effect,
energy_regen,
energy_drain,
ori_rate,
specifier,
} => CharacterState::BasicBeam(basic_beam::Data {
static_data: basic_beam::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
beam_duration: Duration::from_secs_f32(*beam_duration),
damage: *damage,
tick_rate: *tick_rate,
range: *range,
max_angle: *max_angle,
damage_effect: *damage_effect,
energy_regen: *energy_regen,
energy_drain: *energy_drain,
ability_info,
ori_rate: *ori_rate,
specifier: *specifier,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
}),
CharacterAbility::BasicAura {
2021-03-01 20:44:29 +00:00
buildup_duration,
cast_duration,
recover_duration,
targets,
aura,
aura_duration,
2021-03-01 20:44:29 +00:00
range,
energy_cost: _,
2021-07-10 07:10:31 +00:00
scales_with_combo,
specifier,
} => CharacterState::BasicAura(basic_aura::Data {
static_data: basic_aura::StaticData {
2021-03-01 20:44:29 +00:00
buildup_duration: Duration::from_secs_f32(*buildup_duration),
cast_duration: Duration::from_secs_f32(*cast_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
targets: *targets,
aura: *aura,
aura_duration: Duration::from_secs_f32(*aura_duration),
2021-03-01 20:44:29 +00:00
range: *range,
ability_info,
2021-07-10 07:10:31 +00:00
scales_with_combo: *scales_with_combo,
combo_at_cast: data.combo.counter(),
specifier: *specifier,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
}),
CharacterAbility::Blink {
buildup_duration,
recover_duration,
max_range,
} => CharacterState::Blink(blink::Data {
static_data: blink::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
max_range: *max_range,
ability_info,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
}),
CharacterAbility::BasicSummon {
buildup_duration,
cast_duration,
recover_duration,
summon_amount,
2021-06-03 22:42:50 +00:00
summon_distance,
summon_info,
2021-06-03 22:42:50 +00:00
duration,
} => CharacterState::BasicSummon(basic_summon::Data {
static_data: basic_summon::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
cast_duration: Duration::from_secs_f32(*cast_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
summon_amount: *summon_amount,
2021-06-03 22:42:50 +00:00
summon_distance: *summon_distance,
summon_info: *summon_info,
ability_info,
2021-06-03 22:42:50 +00:00
duration: *duration,
},
summon_count: 0,
timer: Duration::default(),
stage_section: StageSection::Buildup,
}),
CharacterAbility::SelfBuff {
buildup_duration,
cast_duration,
recover_duration,
buff_kind,
buff_strength,
buff_duration,
energy_cost: _,
} => CharacterState::SelfBuff(self_buff::Data {
static_data: self_buff::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
cast_duration: Duration::from_secs_f32(*cast_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
buff_kind: *buff_kind,
buff_strength: *buff_strength,
buff_duration: buff_duration.map(Duration::from_secs_f32),
ability_info,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
}),
2021-06-19 18:53:23 +00:00
CharacterAbility::SpriteSummon {
buildup_duration,
cast_duration,
recover_duration,
sprite,
summon_distance,
sparseness,
2021-06-19 18:53:23 +00:00
} => CharacterState::SpriteSummon(sprite_summon::Data {
static_data: sprite_summon::StaticData {
buildup_duration: Duration::from_secs_f32(*buildup_duration),
cast_duration: Duration::from_secs_f32(*cast_duration),
recover_duration: Duration::from_secs_f32(*recover_duration),
sprite: *sprite,
summon_distance: *summon_distance,
sparseness: *sparseness,
2021-06-19 18:53:23 +00:00
ability_info,
},
timer: Duration::default(),
stage_section: StageSection::Buildup,
achieved_radius: summon_distance.0.floor() as i32 - 1,
2021-06-19 18:53:23 +00:00
}),
2020-02-03 10:54:50 +00:00
}
}
}