Damage on high speed effect (#18566)
This commit is contained in:
@@ -1,33 +1,40 @@
|
|||||||
|
using Content.Server.Damage.Systems;
|
||||||
using Content.Shared.Damage;
|
using Content.Shared.Damage;
|
||||||
using Robust.Shared.Audio;
|
using Robust.Shared.Audio;
|
||||||
|
using Robust.Shared.Serialization.TypeSerializers.Implementations.Custom;
|
||||||
|
|
||||||
|
namespace Content.Server.Damage.Components;
|
||||||
|
|
||||||
namespace Content.Server.Damage.Components
|
|
||||||
{
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Should the entity take damage / be stunned if colliding at a speed above MinimumSpeed?
|
/// Should the entity take damage / be stunned if colliding at a speed above MinimumSpeed?
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[RegisterComponent]
|
[RegisterComponent, Access(typeof(DamageOnHighSpeedImpactSystem))]
|
||||||
internal sealed class DamageOnHighSpeedImpactComponent : Component
|
public sealed class DamageOnHighSpeedImpactComponent : Component
|
||||||
{
|
{
|
||||||
[DataField("minimumSpeed")]
|
[DataField("minimumSpeed"), ViewVariables(VVAccess.ReadWrite)]
|
||||||
public float MinimumSpeed { get; set; } = 20f;
|
public float MinimumSpeed = 20f;
|
||||||
[DataField("factor")]
|
|
||||||
public float Factor { get; set; } = 0.5f;
|
|
||||||
[DataField("soundHit", required: true)]
|
|
||||||
public SoundSpecifier SoundHit { get; set; } = default!;
|
|
||||||
[DataField("stunChance")]
|
|
||||||
public float StunChance { get; set; } = 0.25f;
|
|
||||||
[DataField("stunMinimumDamage")]
|
|
||||||
public int StunMinimumDamage { get; set; } = 10;
|
|
||||||
[DataField("stunSeconds")]
|
|
||||||
public float StunSeconds { get; set; } = 1f;
|
|
||||||
[DataField("damageCooldown")]
|
|
||||||
public float DamageCooldown { get; set; } = 2f;
|
|
||||||
|
|
||||||
internal TimeSpan LastHit = TimeSpan.Zero;
|
[DataField("speedDamageFactor"), ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public float SpeedDamageFactor = 0.5f;
|
||||||
|
|
||||||
[DataField("damage", required: true)]
|
[DataField("soundHit", required: true), ViewVariables(VVAccess.ReadWrite)]
|
||||||
[ViewVariables(VVAccess.ReadWrite)]
|
public SoundSpecifier SoundHit = default!;
|
||||||
|
|
||||||
|
[DataField("stunChance"), ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public float StunChance = 0.25f;
|
||||||
|
|
||||||
|
[DataField("stunMinimumDamage"), ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public int StunMinimumDamage = 10;
|
||||||
|
|
||||||
|
[DataField("stunSeconds"), ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public float StunSeconds = 1f;
|
||||||
|
|
||||||
|
[DataField("damageCooldown"), ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public float DamageCooldown = 2f;
|
||||||
|
|
||||||
|
[DataField("lastHit", customTypeSerializer: typeof(TimeOffsetSerializer)), ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public TimeSpan LastHit = TimeSpan.Zero;
|
||||||
|
|
||||||
|
[DataField("damage", required: true), ViewVariables(VVAccess.ReadWrite)]
|
||||||
public DamageSpecifier Damage = default!;
|
public DamageSpecifier Damage = default!;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,24 +1,22 @@
|
|||||||
using Content.Server.Damage.Components;
|
using Content.Server.Damage.Components;
|
||||||
using Content.Server.Stunnable;
|
using Content.Server.Stunnable;
|
||||||
using Content.Shared.Audio;
|
|
||||||
using Content.Shared.Damage;
|
using Content.Shared.Damage;
|
||||||
using JetBrains.Annotations;
|
using Content.Shared.Effects;
|
||||||
using Robust.Shared.Audio;
|
using Robust.Shared.Audio;
|
||||||
using Robust.Shared.Physics.Dynamics;
|
|
||||||
using Robust.Shared.Physics.Events;
|
using Robust.Shared.Physics.Events;
|
||||||
using Robust.Shared.Player;
|
using Robust.Shared.Player;
|
||||||
using Robust.Shared.Random;
|
using Robust.Shared.Random;
|
||||||
using Robust.Shared.Timing;
|
using Robust.Shared.Timing;
|
||||||
|
|
||||||
namespace Content.Server.Damage.Systems
|
namespace Content.Server.Damage.Systems;
|
||||||
{
|
|
||||||
[UsedImplicitly]
|
public sealed class DamageOnHighSpeedImpactSystem : EntitySystem
|
||||||
internal sealed class DamageOnHighSpeedImpactSystem: EntitySystem
|
|
||||||
{
|
{
|
||||||
[Dependency] private readonly IRobustRandom _robustRandom = default!;
|
[Dependency] private readonly IRobustRandom _robustRandom = default!;
|
||||||
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
||||||
[Dependency] private readonly DamageableSystem _damageableSystem = default!;
|
[Dependency] private readonly DamageableSystem _damageable = default!;
|
||||||
[Dependency] private readonly StunSystem _stunSystem = default!;
|
[Dependency] private readonly StunSystem _stun = default!;
|
||||||
|
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
||||||
|
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
@@ -28,28 +26,30 @@ namespace Content.Server.Damage.Systems
|
|||||||
|
|
||||||
private void HandleCollide(EntityUid uid, DamageOnHighSpeedImpactComponent component, ref StartCollideEvent args)
|
private void HandleCollide(EntityUid uid, DamageOnHighSpeedImpactComponent component, ref StartCollideEvent args)
|
||||||
{
|
{
|
||||||
|
if (!args.OurFixture.Hard || !args.OtherFixture.Hard)
|
||||||
|
return;
|
||||||
|
|
||||||
if (!EntityManager.HasComponent<DamageableComponent>(uid))
|
if (!EntityManager.HasComponent<DamageableComponent>(uid))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var otherBody = args.OtherEntity;
|
|
||||||
var speed = args.OurBody.LinearVelocity.Length();
|
var speed = args.OurBody.LinearVelocity.Length();
|
||||||
|
|
||||||
if (speed < component.MinimumSpeed)
|
if (speed < component.MinimumSpeed)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
SoundSystem.Play(component.SoundHit.GetSound(), Filter.Pvs(otherBody), otherBody, AudioHelpers.WithVariation(0.125f).WithVolume(-0.125f));
|
|
||||||
|
|
||||||
if ((_gameTiming.CurTime - component.LastHit).TotalSeconds < component.DamageCooldown)
|
if ((_gameTiming.CurTime - component.LastHit).TotalSeconds < component.DamageCooldown)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
component.LastHit = _gameTiming.CurTime;
|
component.LastHit = _gameTiming.CurTime;
|
||||||
|
|
||||||
if (_robustRandom.Prob(component.StunChance))
|
if (_robustRandom.Prob(component.StunChance))
|
||||||
_stunSystem.TryStun(uid, TimeSpan.FromSeconds(component.StunSeconds), true);
|
_stun.TryStun(uid, TimeSpan.FromSeconds(component.StunSeconds), true);
|
||||||
|
|
||||||
var damageScale = (speed / component.MinimumSpeed) * component.Factor;
|
var damageScale = component.SpeedDamageFactor * speed / component.MinimumSpeed;
|
||||||
|
|
||||||
_damageableSystem.TryChangeDamage(uid, component.Damage * damageScale);
|
_damageable.TryChangeDamage(uid, component.Damage * damageScale);
|
||||||
}
|
|
||||||
|
_audio.PlayPvs(component.SoundHit, uid, AudioParams.Default.WithVariation(0.125f).WithVolume(-0.125f));
|
||||||
|
RaiseNetworkEvent(new ColorFlashEffectEvent(Color.Red, new List<EntityUid> { uid }), Filter.Pvs(uid, entityManager: EntityManager));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user