* Update DamageableSystem to modern standards * DamageContainerId -> DamageContainerID with lint flag * Replace strings with protoids * Make CVar subscription declarations all consistently whitespaced * ChangeDamage -> TryChangeDamage, cope with C# jank * Revert event signature changes * Restore a comment * Re-add two queries * Init the queries * Use appearanceQuery in DamageChanged * Use damageableQuery in TryChangeDamage * Use damageableQuery in SetDamageModifierSetId * Final cleanup, fix sandboxing * Rectify ExplosionSystem:::ProcessEntity's call to TryChangeDamage * Re-organize DamageableSystem * first big fuck you breaking change. * THATS A LOT OF DAMAGE!!! * Fix test fails * test fixes 2 * push it --------- Co-authored-by: Princess Cheeseballs <66055347+Pronana@users.noreply.github.com>
105 lines
4.0 KiB
C#
105 lines
4.0 KiB
C#
using Content.Shared.Damage;
|
|
using Content.Shared.Damage.Components;
|
|
using Content.Shared.Damage.Systems;
|
|
using Robust.Shared.Audio.Systems;
|
|
using Robust.Shared.Containers;
|
|
|
|
namespace Content.Shared.Blocking;
|
|
|
|
public sealed partial class BlockingSystem
|
|
{
|
|
[Dependency] private readonly DamageableSystem _damageable = default!;
|
|
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
|
|
|
private void InitializeUser()
|
|
{
|
|
SubscribeLocalEvent<BlockingUserComponent, DamageModifyEvent>(OnUserDamageModified);
|
|
SubscribeLocalEvent<BlockingComponent, DamageModifyEvent>(OnDamageModified);
|
|
|
|
SubscribeLocalEvent<BlockingUserComponent, EntParentChangedMessage>(OnParentChanged);
|
|
SubscribeLocalEvent<BlockingUserComponent, ContainerGettingInsertedAttemptEvent>(OnInsertAttempt);
|
|
SubscribeLocalEvent<BlockingUserComponent, AnchorStateChangedEvent>(OnAnchorChanged);
|
|
SubscribeLocalEvent<BlockingUserComponent, EntityTerminatingEvent>(OnEntityTerminating);
|
|
}
|
|
|
|
private void OnParentChanged(EntityUid uid, BlockingUserComponent component, ref EntParentChangedMessage args)
|
|
{
|
|
UserStopBlocking(uid, component);
|
|
}
|
|
|
|
private void OnInsertAttempt(EntityUid uid, BlockingUserComponent component, ContainerGettingInsertedAttemptEvent args)
|
|
{
|
|
UserStopBlocking(uid, component);
|
|
}
|
|
|
|
private void OnAnchorChanged(EntityUid uid, BlockingUserComponent component, ref AnchorStateChangedEvent args)
|
|
{
|
|
if (args.Anchored)
|
|
return;
|
|
|
|
UserStopBlocking(uid, component);
|
|
}
|
|
|
|
private void OnUserDamageModified(EntityUid uid, BlockingUserComponent component, DamageModifyEvent args)
|
|
{
|
|
if (component.BlockingItem is not { } item || !TryComp<BlockingComponent>(item, out var blocking))
|
|
return;
|
|
|
|
if (args.Damage.GetTotal() <= 0)
|
|
return;
|
|
|
|
// A shield should only block damage it can itself absorb. To determine that we need the Damageable component on it.
|
|
if (!TryComp<DamageableComponent>(item, out var dmgComp))
|
|
return;
|
|
|
|
var blockFraction = blocking.IsBlocking ? blocking.ActiveBlockFraction : blocking.PassiveBlockFraction;
|
|
blockFraction = Math.Clamp(blockFraction, 0, 1);
|
|
_damageable.TryChangeDamage((item, dmgComp), blockFraction * args.OriginalDamage);
|
|
|
|
var modify = new DamageModifierSet();
|
|
foreach (var key in dmgComp.Damage.DamageDict.Keys)
|
|
{
|
|
modify.Coefficients.TryAdd(key, 1 - blockFraction);
|
|
}
|
|
|
|
args.Damage = DamageSpecifier.ApplyModifierSet(args.Damage, modify);
|
|
|
|
if (blocking.IsBlocking && !args.Damage.Equals(args.OriginalDamage))
|
|
{
|
|
_audio.PlayPvs(blocking.BlockSound, uid);
|
|
}
|
|
}
|
|
|
|
private void OnDamageModified(EntityUid uid, BlockingComponent component, DamageModifyEvent args)
|
|
{
|
|
var modifier = component.IsBlocking ? component.ActiveBlockDamageModifier : component.PassiveBlockDamageModifer;
|
|
if (modifier == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
args.Damage = DamageSpecifier.ApplyModifierSet(args.Damage, modifier);
|
|
}
|
|
|
|
private void OnEntityTerminating(EntityUid uid, BlockingUserComponent component, ref EntityTerminatingEvent args)
|
|
{
|
|
if (!TryComp<BlockingComponent>(component.BlockingItem, out var blockingComponent))
|
|
return;
|
|
|
|
StopBlockingHelper(component.BlockingItem.Value, blockingComponent, uid);
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// Check for the shield and has the user stop blocking
|
|
/// Used where you'd like the user to stop blocking, but also don't want to remove the <see cref="BlockingUserComponent"/>
|
|
/// </summary>
|
|
/// <param name="uid">The user blocking</param>
|
|
/// <param name="component">The <see cref="BlockingUserComponent"/></param>
|
|
private void UserStopBlocking(EntityUid uid, BlockingUserComponent component)
|
|
{
|
|
if (TryComp<BlockingComponent>(component.BlockingItem, out var blockComp) && blockComp.IsBlocking)
|
|
StopBlocking(component.BlockingItem.Value, blockComp, uid);
|
|
}
|
|
}
|