* 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>
229 lines
9.3 KiB
C#
229 lines
9.3 KiB
C#
using System.Diagnostics.CodeAnalysis;
|
|
using System.Linq;
|
|
using Content.Server.Administration.Logs;
|
|
using Content.Server.Atmos.EntitySystems;
|
|
using Content.Server.Body.Systems;
|
|
using Content.Server.Construction;
|
|
using Content.Server.Destructible.Thresholds;
|
|
using Content.Server.Destructible.Thresholds.Behaviors;
|
|
using Content.Server.Explosion.EntitySystems;
|
|
using Content.Server.Fluids.EntitySystems;
|
|
using Content.Server.Stack;
|
|
using Content.Shared.Chemistry.EntitySystems;
|
|
using Content.Shared.Damage;
|
|
using Content.Shared.Damage.Systems;
|
|
using Content.Shared.Database;
|
|
using Content.Shared.Destructible;
|
|
using Content.Shared.Destructible.Thresholds.Triggers;
|
|
using Content.Shared.FixedPoint;
|
|
using Content.Shared.Humanoid;
|
|
using Content.Shared.Trigger.Systems;
|
|
using JetBrains.Annotations;
|
|
using Robust.Server.Audio;
|
|
using Robust.Shared.Containers;
|
|
using Robust.Shared.Prototypes;
|
|
using Robust.Shared.Random;
|
|
|
|
namespace Content.Server.Destructible
|
|
{
|
|
[UsedImplicitly]
|
|
public sealed partial class DestructibleSystem : SharedDestructibleSystem
|
|
{
|
|
[Dependency] public readonly IRobustRandom Random = default!;
|
|
public new IEntityManager EntityManager => base.EntityManager;
|
|
|
|
[Dependency] public readonly AtmosphereSystem AtmosphereSystem = default!;
|
|
[Dependency] public readonly AudioSystem AudioSystem = default!;
|
|
[Dependency] public readonly BodySystem BodySystem = default!;
|
|
[Dependency] public readonly ConstructionSystem ConstructionSystem = default!;
|
|
[Dependency] public readonly ExplosionSystem ExplosionSystem = default!;
|
|
[Dependency] public readonly StackSystem StackSystem = default!;
|
|
[Dependency] public readonly TriggerSystem TriggerSystem = default!;
|
|
[Dependency] public readonly SharedSolutionContainerSystem SolutionContainerSystem = default!;
|
|
[Dependency] public readonly PuddleSystem PuddleSystem = default!;
|
|
[Dependency] public readonly SharedContainerSystem ContainerSystem = default!;
|
|
[Dependency] public readonly IPrototypeManager PrototypeManager = default!;
|
|
[Dependency] public readonly IAdminLogManager AdminLogger = default!;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
SubscribeLocalEvent<DestructibleComponent, DamageChangedEvent>(OnDamageChanged);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Check if any thresholds were reached. if they were, execute them.
|
|
/// </summary>
|
|
private void OnDamageChanged(EntityUid uid, DestructibleComponent component, DamageChangedEvent args)
|
|
{
|
|
component.IsBroken = false;
|
|
|
|
foreach (var threshold in component.Thresholds)
|
|
{
|
|
if (Triggered(threshold, (uid, args.Damageable)))
|
|
{
|
|
RaiseLocalEvent(uid, new DamageThresholdReached(component, threshold), true);
|
|
|
|
var logImpact = LogImpact.Low;
|
|
// Convert behaviors into string for logs
|
|
var triggeredBehaviors = string.Join(", ", threshold.Behaviors.Select(b =>
|
|
{
|
|
if (logImpact <= b.Impact)
|
|
logImpact = b.Impact;
|
|
if (b is DoActsBehavior doActsBehavior)
|
|
{
|
|
return $"{b.GetType().Name}:{doActsBehavior.Acts.ToString()}";
|
|
}
|
|
return b.GetType().Name;
|
|
}));
|
|
|
|
// If it doesn't have a humanoid component, it's probably not particularly notable?
|
|
if (logImpact > LogImpact.Medium && !HasComp<HumanoidAppearanceComponent>(uid))
|
|
logImpact = LogImpact.Medium;
|
|
|
|
if (args.Origin != null)
|
|
{
|
|
AdminLogger.Add(LogType.Damaged,
|
|
logImpact,
|
|
$"{ToPrettyString(args.Origin.Value):actor} caused {ToPrettyString(uid):subject} to trigger [{triggeredBehaviors}]");
|
|
}
|
|
else
|
|
{
|
|
AdminLogger.Add(LogType.Damaged,
|
|
logImpact,
|
|
$"Unknown damage source caused {ToPrettyString(uid):subject} to trigger [{triggeredBehaviors}]");
|
|
}
|
|
|
|
Execute(threshold, uid, args.Origin);
|
|
}
|
|
|
|
if (threshold.OldTriggered)
|
|
{
|
|
component.IsBroken |= threshold.Behaviors.Any(b => b is DoActsBehavior doActsBehavior &&
|
|
(doActsBehavior.HasAct(ThresholdActs.Breakage) || doActsBehavior.HasAct(ThresholdActs.Destruction)));
|
|
}
|
|
|
|
// if destruction behavior (or some other deletion effect) occurred, don't run other triggers.
|
|
if (EntityManager.IsQueuedForDeletion(uid) || Deleted(uid))
|
|
return;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Check if the given threshold should trigger.
|
|
/// </summary>
|
|
public bool Triggered(DamageThreshold threshold, Entity<Shared.Damage.Components.DamageableComponent> owner)
|
|
{
|
|
if (threshold.Trigger == null)
|
|
return false;
|
|
|
|
if (threshold.Triggered && threshold.TriggersOnce)
|
|
return false;
|
|
|
|
if (threshold.OldTriggered)
|
|
{
|
|
threshold.OldTriggered = threshold.Trigger.Reached(owner, this);
|
|
return false;
|
|
}
|
|
|
|
if (!threshold.Trigger.Reached(owner, this))
|
|
return false;
|
|
|
|
threshold.OldTriggered = true;
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Check if the conditions for the given threshold are currently true.
|
|
/// </summary>
|
|
public bool Reached(DamageThreshold threshold, Entity<Shared.Damage.Components.DamageableComponent> owner)
|
|
{
|
|
if (threshold.Trigger == null)
|
|
return false;
|
|
|
|
return threshold.Trigger.Reached(owner, this);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Triggers this threshold.
|
|
/// </summary>
|
|
/// <param name="owner">The entity that owns this threshold.</param>
|
|
/// <param name="cause">The entity that caused this threshold to trigger.</param>
|
|
public void Execute(DamageThreshold threshold, EntityUid owner, EntityUid? cause = null)
|
|
{
|
|
threshold.Triggered = true;
|
|
|
|
foreach (var behavior in threshold.Behaviors)
|
|
{
|
|
// The owner has been deleted. We stop execution of behaviors here.
|
|
if (!Exists(owner))
|
|
return;
|
|
|
|
// TODO: Replace with EntityEffects.
|
|
behavior.Execute(owner, this, cause);
|
|
}
|
|
}
|
|
|
|
public bool TryGetDestroyedAt(Entity<DestructibleComponent?> ent, [NotNullWhen(true)] out FixedPoint2? destroyedAt)
|
|
{
|
|
destroyedAt = null;
|
|
if (!Resolve(ent, ref ent.Comp, false))
|
|
return false;
|
|
|
|
destroyedAt = DestroyedAt(ent, ent.Comp);
|
|
return true;
|
|
}
|
|
|
|
// FFS this shouldn't be this hard. Maybe this should just be a field of the destructible component. Its not
|
|
// like there is currently any entity that is NOT just destroyed upon reaching a total-damage value.
|
|
/// <summary>
|
|
/// Figure out how much damage an entity needs to have in order to be destroyed.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This assumes that this entity has some sort of destruction or breakage behavior triggered by a
|
|
/// total-damage threshold.
|
|
/// </remarks>
|
|
public FixedPoint2 DestroyedAt(EntityUid uid, DestructibleComponent? destructible = null)
|
|
{
|
|
if (!Resolve(uid, ref destructible, logMissing: false))
|
|
return FixedPoint2.MaxValue;
|
|
|
|
// We have nested for loops here, but the vast majority of components only have one threshold with 1-3 behaviors.
|
|
// Really, this should probably just be a property of the damageable component.
|
|
var damageNeeded = FixedPoint2.MaxValue;
|
|
foreach (var threshold in destructible.Thresholds)
|
|
{
|
|
if (threshold.Trigger is not DamageTrigger trigger)
|
|
continue;
|
|
|
|
foreach (var behavior in threshold.Behaviors)
|
|
{
|
|
if (behavior is DoActsBehavior actBehavior &&
|
|
actBehavior.HasAct(ThresholdActs.Destruction | ThresholdActs.Breakage))
|
|
{
|
|
damageNeeded = FixedPoint2.Min(damageNeeded, trigger.Damage);
|
|
}
|
|
}
|
|
}
|
|
return damageNeeded;
|
|
}
|
|
}
|
|
|
|
// Currently only used for destructible integration tests. Unless other uses are found for this, maybe this should just be removed and the tests redone.
|
|
/// <summary>
|
|
/// Event raised when a <see cref="DamageThreshold"/> is reached.
|
|
/// </summary>
|
|
public sealed class DamageThresholdReached : EntityEventArgs
|
|
{
|
|
public readonly DestructibleComponent Parent;
|
|
|
|
public readonly DamageThreshold Threshold;
|
|
|
|
public DamageThresholdReached(DestructibleComponent parent, DamageThreshold threshold)
|
|
{
|
|
Parent = parent;
|
|
Threshold = threshold;
|
|
}
|
|
}
|
|
}
|