diff --git a/assets/common/abilities/unique/wendigomagic/firebomb.ron b/assets/common/abilities/unique/wendigomagic/frostbomb.ron similarity index 78% rename from assets/common/abilities/unique/wendigomagic/firebomb.ron rename to assets/common/abilities/unique/wendigomagic/frostbomb.ron index 16444d31a3..9f53c6deb9 100644 --- a/assets/common/abilities/unique/wendigomagic/firebomb.ron +++ b/assets/common/abilities/unique/wendigomagic/frostbomb.ron @@ -2,12 +2,11 @@ BasicRanged( energy_cost: 0, buildup_duration: 0.5, recover_duration: 0.35, - projectile: Fireball( + projectile: Frostball( damage: 100.0, radius: 5.0, - energy_regen: 50, ), - projectile_body: Object(BoltFire), + projectile_body: Object(BoltFire), // TODO: Get ice projectile model /*projectile_light: Some(LightEmitter { col: (1.0, 0.75, 0.11).into(), ..Default::default() diff --git a/assets/common/abilities/weapon_ability_manifest.ron b/assets/common/abilities/weapon_ability_manifest.ron index 5d08c7c957..91b8b12126 100644 --- a/assets/common/abilities/weapon_ability_manifest.ron +++ b/assets/common/abilities/weapon_ability_manifest.ron @@ -93,7 +93,7 @@ abilities: [], ), Unique(WendigoMagic): ( - primary: "common.abilities.unique.wendigomagic.firebomb", + primary: "common.abilities.unique.wendigomagic.frostbomb", secondary: "common.abilities.unique.wendigomagic.singlestrike", skills: [], ), diff --git a/assets/voxygen/shaders/particle-vert.glsl b/assets/voxygen/shaders/particle-vert.glsl index 95cfc1ecaf..f33ece8ede 100644 --- a/assets/voxygen/shaders/particle-vert.glsl +++ b/assets/voxygen/shaders/particle-vert.glsl @@ -58,6 +58,7 @@ const int FIRE_SHOCKWAVE = 16; const int FIRE_BOWL = 17; const int SNOW = 18; const int EXPLOSION = 19; +const int ICE = 20; // meters per second squared (acceleration) const float earth_gravity = 9.807; @@ -324,7 +325,7 @@ void main() { attr = Attr( spiral_motion(inst_dir, 0.3 * (floor(2 * rand0 + 0.5) - 0.5) * min(linear_scale(10), 1), lifetime / inst_lifespan, 10.0, inst_time), vec3((1.7 - 0.7 * abs(floor(2 * rand0 - 0.5) + 0.5)) * (1.5 + 0.5 * sin(tick.x * 10 - lifetime * 4))), - vec4(vec3(0.4, 1.6 + 0.3 * sin(tick.x * 10 - lifetime * 3 + 4), 1.0 + 0.15 * sin(tick.x * 5 - lifetime * 5)), start_end(1.0, 0.0) /*0.3*/), + vec4(vec3(0.4, 1.6 + 0.3 * sin(tick.x * 10 - lifetime * 3 + 4), 1.0 + 0.15 * sin(tick.x * 5 - lifetime * 5)), 1 /*0.3*/), spin_in_axis(inst_dir, tick.z) ); } else if (inst_mode == ENERGY_NATURE) { @@ -333,7 +334,7 @@ void main() { inst_dir * slow_end(0.03) + spiral_motion(vec3(rand1, rand2, rand3), 0.2 * (rand4 + 1.3) * slow_end(0.02), percent() * 3 * (rand4 + 4.0) + rand0, 1.0, 0.0), vec3(1.0), - vec4(vec3(0, 2.5, 1.5 + rand7 * 0.7), start_end(1.0, 0.0)), + vec4(vec3(0, 2.5, 1.5 + rand7 * 0.7), 1), spin_in_axis(vec3(rand6, rand7, rand8), rand9 * 3) ); } else if (inst_mode == FLAMETHROWER) { @@ -341,7 +342,7 @@ void main() { attr = Attr( (inst_dir * slow_end(1.5)) + vec3(rand0, rand1, rand2) * (lifetime * 5 + 0.25), vec3((2.5 * (1 - slow_start(0.3)))), - vec4(3, 1.6 + rand5 * 0.3 - 0.4 * percent(), 0.2, start_end(1.0, 0.0)), + vec4(3, 1.6 + rand5 * 0.3 - 0.4 * percent(), 0.2, 1), spin_in_axis(vec3(rand6, rand7, rand8), percent() * 10 + 3 * rand9) ); } else if (inst_mode == EXPLOSION) { @@ -349,7 +350,15 @@ void main() { attr = Attr( inst_dir * ((rand0+1.0)/2 + 0.4) * slow_end(2.0) + 0.3 * grav_vel(earth_gravity), vec3((3 * (1 - slow_start(0.1)))), - vec4(3, 1.6 + rand5 * 0.3 - 0.4 * percent(), 0.2, start_end(1.0, 0.0)), + vec4(3, 1.6 + rand5 * 0.3 - 0.4 * percent(), 0.2, 1), + spin_in_axis(vec3(rand6, rand7, rand8), percent() * 10 + 3 * rand9) + ); + } else if (inst_mode == ICE) { + f_reflect = 0.0; // Ice doesn't reflect to look like magic + attr = Attr( + inst_dir * ((rand0+1.0)/2 + 0.4) * slow_end(2.0) + 0.3 * grav_vel(earth_gravity), + vec3((3 * (1 - slow_start(0.1)))), + vec4(0.2, 1.6 + rand5 * 0.3 - 0.4 * percent(), 3, 1), spin_in_axis(vec3(rand6, rand7, rand8), percent() * 10 + 3 * rand9) ); } else if (inst_mode == FIRE_SHOCKWAVE) { @@ -357,7 +366,7 @@ void main() { attr = Attr( vec3(rand0, rand1, lifetime * 10 + rand2), vec3((5 * (1 - slow_start(0.5)))), - vec4(3, 1.6 + rand5 * 0.3 - 0.4 * percent(), 0.2, start_end(1.0, 0.0)), + vec4(3, 1.6 + rand5 * 0.3 - 0.4 * percent(), 0.2, 1), spin_in_axis(vec3(rand3, rand4, rand5), rand6) ); } else { diff --git a/common/src/comp/projectile.rs b/common/src/comp/projectile.rs index cd7a6ad56d..86f118cfe8 100644 --- a/common/src/comp/projectile.rs +++ b/common/src/comp/projectile.rs @@ -3,6 +3,7 @@ use crate::{ Attack, AttackDamage, AttackEffect, CombatBuff, CombatEffect, CombatRequirement, Damage, DamageSource, GroupTarget, Knockback, KnockbackDir, }, + comp::item::Reagent, uid::Uid, Explosion, RadiusEffect, }; @@ -49,6 +50,10 @@ pub enum ProjectileConstructor { radius: f32, energy_regen: f32, }, + Frostball { + damage: f32, + radius: f32, + }, Firebolt { damage: f32, energy_regen: f32, @@ -124,6 +129,34 @@ impl ProjectileConstructor { RadiusEffect::TerrainDestruction(2.0), ], radius, + reagent: Some(Reagent::Red), + }; + Projectile { + hit_solid: vec![Effect::Explode(explosion.clone()), Effect::Vanish], + hit_entity: vec![Effect::Explode(explosion), Effect::Vanish], + time_left: Duration::from_secs(10), + owner, + ignore_group: true, + } + }, + Frostball { + damage, + radius, + } => { + let damage = AttackDamage::new( + Damage { + source: DamageSource::Explosion, + value: damage, + }, + Some(GroupTarget::OutOfGroup), + ); + let attack = Attack::default().with_damage(damage); + let explosion = Explosion { + effects: vec![ + RadiusEffect::Attack(attack), + ], + radius, + reagent: Some(Reagent::Blue), }; Projectile { hit_solid: vec![Effect::Explode(explosion.clone()), Effect::Vanish], @@ -173,6 +206,7 @@ impl ProjectileConstructor { let explosion = Explosion { effects: vec![RadiusEffect::Attack(attack)], radius, + reagent: Some(Reagent::Green), }; Projectile { hit_solid: vec![Effect::Explode(explosion.clone()), Effect::Vanish], @@ -219,6 +253,14 @@ impl ProjectileConstructor { *energy_regen *= regen; *radius *= range; }, + Frostball { + ref mut damage, + ref mut radius, + .. + } => { + *damage *= power; + *radius *= range; + }, Firebolt { ref mut damage, ref mut energy_regen, diff --git a/common/src/event.rs b/common/src/event.rs index e528677b50..c38855cd21 100644 --- a/common/src/event.rs +++ b/common/src/event.rs @@ -39,7 +39,6 @@ pub enum ServerEvent { pos: Vec3, explosion: Explosion, owner: Option, - reagent: Option, }, Damage { entity: EcsEntity, diff --git a/common/src/explosion.rs b/common/src/explosion.rs index 04296a856b..692c08e97a 100644 --- a/common/src/explosion.rs +++ b/common/src/explosion.rs @@ -1,10 +1,11 @@ -use crate::{combat::Attack, effect::Effect}; +use crate::{combat::Attack, effect::Effect, comp::item::Reagent}; use serde::{Deserialize, Serialize}; #[derive(Clone, Debug, Serialize, Deserialize)] pub struct Explosion { pub effects: Vec, pub radius: f32, + pub reagent: Option, } #[derive(Clone, Debug, Serialize, Deserialize)] diff --git a/common/sys/src/projectile.rs b/common/sys/src/projectile.rs index fa7dfa4476..329ad2df36 100644 --- a/common/sys/src/projectile.rs +++ b/common/sys/src/projectile.rs @@ -127,7 +127,6 @@ impl<'a> System<'a> for Sys { pos: pos.0, explosion: e, owner: projectile.owner, - reagent: None, }) }, projectile::Effect::Vanish => { @@ -163,7 +162,6 @@ impl<'a> System<'a> for Sys { pos: pos.0, explosion: e, owner: projectile.owner, - reagent: None, }) }, projectile::Effect::Vanish => { diff --git a/server/src/cmd.rs b/server/src/cmd.rs index 79e3047371..0cd6ed1c7d 100644 --- a/server/src/cmd.rs +++ b/server/src/cmd.rs @@ -1374,9 +1374,9 @@ fn handle_explosion( RadiusEffect::TerrainDestruction(power), ], radius: 3.0 * power, + reagent: None, }, owner: ecs.read_storage::().get(target).copied(), - reagent: None, }) }, None => server.notify_client( diff --git a/server/src/events/entity_manipulation.rs b/server/src/events/entity_manipulation.rs index 900ec70fa8..551b524eb2 100644 --- a/server/src/events/entity_manipulation.rs +++ b/server/src/events/entity_manipulation.rs @@ -29,7 +29,6 @@ use common::{ }; use common_net::{msg::ServerGeneral, sync::WorldSyncExt}; use common_sys::state::BlockChange; -use comp::item::Reagent; use hashbrown::HashSet; use rand::prelude::*; use specs::{join::Join, saveload::MarkerAllocator, Entity as EcsEntity, WorldExt}; @@ -577,19 +576,13 @@ pub fn handle_explosion( pos: Vec3, explosion: Explosion, owner: Option, - reagent: Option, ) { // Go through all other entities let ecs = &server.state.ecs(); // Add an outcome - // Uses radius as outcome power, makes negative if explosion has healing effect - let outcome_power = explosion.radius - * if explosion.effects.iter().any(|e| matches!(e, RadiusEffect::Attack(a) if a.effects().any(|e| matches!(e.effect(), combat::CombatEffect::Heal(h) if *h > 0.0)))) { - -1.0 - } else { - 1.0 - }; + // Uses radius as outcome power for now + let outcome_power = explosion.radius; ecs.write_resource::>() .push(Outcome::Explosion { pos, @@ -599,7 +592,7 @@ pub fn handle_explosion( .effects .iter() .any(|e| matches!(e, RadiusEffect::Attack(_))), - reagent, + reagent: explosion.reagent, }); let owner_entity = owner.and_then(|uid| { ecs.read_resource::() diff --git a/server/src/events/mod.rs b/server/src/events/mod.rs index a83689be38..f8adcacfa7 100644 --- a/server/src/events/mod.rs +++ b/server/src/events/mod.rs @@ -64,8 +64,7 @@ impl Server { pos, explosion, owner, - reagent, - } => handle_explosion(&self, pos, explosion, owner, reagent), + } => handle_explosion(&self, pos, explosion, owner), ServerEvent::Shoot { entity, dir, diff --git a/server/src/sys/object.rs b/server/src/sys/object.rs index afb3574a0f..d2577c0bbc 100644 --- a/server/src/sys/object.rs +++ b/server/src/sys/object.rs @@ -60,9 +60,9 @@ impl<'a> System<'a> for Sys { RadiusEffect::TerrainDestruction(4.0), ], radius: 12.0, + reagent: None, }, owner: *owner, - reagent: None, }); } }, @@ -87,9 +87,9 @@ impl<'a> System<'a> for Sys { RadiusEffect::TerrainDestruction(4.0), ], radius: 12.0, + reagent: Some(*reagent), }, owner: *owner, - reagent: Some(*reagent), }); } }, diff --git a/voxygen/src/audio/sfx/mod.rs b/voxygen/src/audio/sfx/mod.rs index a4afb5a4bb..47183dd40b 100644 --- a/voxygen/src/audio/sfx/mod.rs +++ b/voxygen/src/audio/sfx/mod.rs @@ -91,7 +91,7 @@ use client::Client; use common::{ assets::{self, AssetExt, AssetHandle}, comp::{ - item::{ItemKind, ToolKind}, + item::{ItemKind, Reagent, ToolKind}, object, Body, CharacterAbilityType, InventoryUpdateEvent, }, outcome::Outcome, @@ -298,9 +298,10 @@ impl SfxMgr { pos, power, is_attack, + reagent, .. } => { - let file_ref = if *is_attack && *power < 0.0 { + let file_ref = if *is_attack && matches!(reagent, Some(Reagent::Green)) { "voxygen.audio.sfx.abilities.heal_bomb" } else { "voxygen.audio.sfx.explosion" diff --git a/voxygen/src/render/pipelines/particle.rs b/voxygen/src/render/pipelines/particle.rs index c5db9acfda..d5e2cee267 100644 --- a/voxygen/src/render/pipelines/particle.rs +++ b/voxygen/src/render/pipelines/particle.rs @@ -116,6 +116,7 @@ pub enum ParticleMode { FireBowl = 17, Snow = 18, Explosion = 19, + Ice = 20, } impl ParticleMode { diff --git a/voxygen/src/scene/mod.rs b/voxygen/src/scene/mod.rs index 339917b173..580eebd820 100644 --- a/voxygen/src/scene/mod.rs +++ b/voxygen/src/scene/mod.rs @@ -406,9 +406,9 @@ impl Scene { Outcome::Explosion { pos, power, - radius: _, - is_attack: _, + is_attack, reagent, + .. } => self.event_lights.push(EventLight { light: Light::new( *pos, @@ -416,20 +416,19 @@ impl Scene { Some(Reagent::Blue) => Rgb::new(0.15, 0.4, 1.0), Some(Reagent::Green) => Rgb::new(0.0, 1.0, 0.0), Some(Reagent::Purple) => Rgb::new(0.7, 0.0, 1.0), - Some(Reagent::Red) => Rgb::new(1.0, 0.0, 0.0), - Some(Reagent::Yellow) => Rgb::new(1.0, 1.0, 0.0), - None => { - if *power < 0.0 { - Rgb::new(0.0, 1.0, 0.0) - } else { - Rgb::new(1.0, 0.5, 0.0) - } + Some(Reagent::Red) => if *is_attack { + Rgb::new(1.0, 0.5, 0.0) + } else { + Rgb::new(1.0, 0.0, 0.0) }, + Some(Reagent::Yellow) => Rgb::new(1.0, 1.0, 0.0), + None => Rgb::new(1.0, 0.5, 0.0), }, - power.abs() - * match reagent { - Some(_) => 5.0, - None => 2.5, + power + * if *is_attack || reagent.is_none() { + 2.5 + } else { + 5.0 }, ), timeout: match reagent { diff --git a/voxygen/src/scene/particle.rs b/voxygen/src/scene/particle.rs index 62da9320a7..95c36b1fe3 100644 --- a/voxygen/src/scene/particle.rs +++ b/voxygen/src/scene/particle.rs @@ -62,38 +62,59 @@ impl ParticleMgr { reagent, } => { if *is_attack { - if *power < 0.0 { - self.particles.resize_with( - self.particles.len() + (60.0 * power.abs()) as usize, - || { - Particle::new_directed( - Duration::from_secs_f32(rng.gen_range(0.2..3.0)), - time, - ParticleMode::EnergyNature, - *pos, - *pos + Vec3::::zero() - .map(|_| rng.gen_range(-1.0..1.0)) - .normalized() - * rng.gen_range(1.0..*radius), - ) - }, - ); - } else { - self.particles.resize_with( - self.particles.len() + (75.0 * power.abs()) as usize, - || { - Particle::new_directed( - Duration::from_millis(500), - time, - ParticleMode::Explosion, - *pos, - *pos + Vec3::::zero() - .map(|_| rng.gen_range(-1.0..1.0)) - .normalized() - * *radius, - ) - }, - ); + match reagent { + Some(Reagent::Green) => { + self.particles.resize_with( + self.particles.len() + (60.0 * power.abs()) as usize, + || { + Particle::new_directed( + Duration::from_secs_f32(rng.gen_range(0.2..3.0)), + time, + ParticleMode::EnergyNature, + *pos, + *pos + Vec3::::zero() + .map(|_| rng.gen_range(-1.0..1.0)) + .normalized() + * rng.gen_range(1.0..*radius), + ) + }, + ); + }, + Some(Reagent::Red) => { + self.particles.resize_with( + self.particles.len() + (75.0 * power.abs()) as usize, + || { + Particle::new_directed( + Duration::from_millis(500), + time, + ParticleMode::Explosion, + *pos, + *pos + Vec3::::zero() + .map(|_| rng.gen_range(-1.0..1.0)) + .normalized() + * *radius, + ) + }, + ); + }, + Some(Reagent::Blue) => { + self.particles.resize_with( + self.particles.len() + (75.0 * power.abs()) as usize, + || { + Particle::new_directed( + Duration::from_millis(500), + time, + ParticleMode::Ice, + *pos, + *pos + Vec3::::zero() + .map(|_| rng.gen_range(-1.0..1.0)) + .normalized() + * *radius, + ) + }, + ); + }, + _ => {}, } } else { self.particles.resize_with(