Files
OldThink/Content.Server/Fluids/EntitySystems/PuddleSystem.Spillable.cs

189 lines
7.5 KiB
C#
Raw Normal View History

using Content.Server.Chemistry.Containers.EntitySystems;
using Content.Server.Fluids.Components;
2023-04-10 18:56:21 -07:00
using Content.Shared.Chemistry.Components;
using Content.Shared.Chemistry.EntitySystems;
2023-04-10 18:56:21 -07:00
using Content.Shared.Chemistry.Reaction;
using Content.Shared.Chemistry.Reagent;
2023-04-10 15:37:03 +10:00
using Content.Shared.Clothing.Components;
using Content.Shared.CombatMode.Pacification;
2023-04-10 15:37:03 +10:00
using Content.Shared.Database;
using Content.Shared.FixedPoint;
using Content.Shared.Fluids.Components;
2023-04-10 18:56:21 -07:00
using Content.Shared.IdentityManagement;
2023-04-10 15:37:03 +10:00
using Content.Shared.Inventory.Events;
using Content.Shared.Nutrition.EntitySystems;
2023-04-10 18:56:21 -07:00
using Content.Shared.Popups;
2023-04-10 15:37:03 +10:00
using Content.Shared.Spillable;
using Content.Shared.Throwing;
2023-04-10 18:56:21 -07:00
using Content.Shared.Weapons.Melee.Events;
using Robust.Shared.Player;
2023-04-10 15:37:03 +10:00
namespace Content.Server.Fluids.EntitySystems;
public sealed partial class PuddleSystem
{
protected override void InitializeSpillable()
2023-04-10 15:37:03 +10:00
{
base.InitializeSpillable();
2023-04-10 15:37:03 +10:00
SubscribeLocalEvent<SpillableComponent, LandEvent>(SpillOnLand);
// Openable handles the event if it's closed
SubscribeLocalEvent<SpillableComponent, MeleeHitEvent>(SplashOnMeleeHit, after: [typeof(OpenableSystem)]);
2023-04-10 15:37:03 +10:00
SubscribeLocalEvent<SpillableComponent, GotEquippedEvent>(OnGotEquipped);
SubscribeLocalEvent<SpillableComponent, GotUnequippedEvent>(OnGotUnequipped);
SubscribeLocalEvent<SpillableComponent, SolutionContainerOverflowEvent>(OnOverflow);
2023-04-10 15:37:03 +10:00
SubscribeLocalEvent<SpillableComponent, SpillDoAfterEvent>(OnDoAfter);
SubscribeLocalEvent<SpillableComponent, AttemptPacifiedThrowEvent>(OnAttemptPacifiedThrow);
2023-04-10 15:37:03 +10:00
}
private void OnOverflow(Entity<SpillableComponent> entity, ref SolutionContainerOverflowEvent args)
2023-04-10 15:37:03 +10:00
{
if (args.Handled)
return;
2023-04-10 15:37:03 +10:00
TrySpillAt(Transform(entity).Coordinates, args.Overflow, out _);
2023-04-10 15:37:03 +10:00
args.Handled = true;
}
private void SplashOnMeleeHit(Entity<SpillableComponent> entity, ref MeleeHitEvent args)
2023-04-10 18:56:21 -07:00
{
2023-09-23 03:10:04 +01:00
if (args.Handled)
return;
2023-04-10 18:56:21 -07:00
// When attacking someone reactive with a spillable entity,
// splash a little on them (touch react)
// If this also has solution transfer, then assume the transfer amount is how much we want to spill.
// Otherwise let's say they want to spill a quarter of its max volume.
if (!_solutionContainerSystem.TryGetDrainableSolution(entity.Owner, out var soln, out var solution))
2023-04-10 18:56:21 -07:00
return;
var hitCount = args.HitEntities.Count;
var totalSplit = FixedPoint2.Min(solution.MaxVolume * 0.25, solution.Volume);
if (TryComp<SolutionTransferComponent>(entity, out var transfer))
2023-04-10 18:56:21 -07:00
{
totalSplit = FixedPoint2.Min(transfer.TransferAmount, solution.Volume);
}
// a little lame, but reagent quantity is not very balanced and we don't want people
// spilling like 100u of reagent on someone at once!
totalSplit = FixedPoint2.Min(totalSplit, entity.Comp.MaxMeleeSpillAmount);
2023-04-10 18:56:21 -07:00
if (totalSplit == 0)
return;
2023-09-23 03:10:04 +01:00
args.Handled = true;
2023-04-10 18:56:21 -07:00
foreach (var hit in args.HitEntities)
{
if (!HasComp<ReactiveComponent>(hit))
{
hitCount -= 1; // so we don't undershoot solution calculation for actual reactive entities
continue;
}
var splitSolution = _solutionContainerSystem.SplitSolution(soln.Value, totalSplit / hitCount);
2023-04-10 18:56:21 -07:00
_adminLogger.Add(LogType.MeleeHit, $"{ToPrettyString(args.User)} splashed {SolutionContainerSystem.ToPrettyString(splitSolution):solution} from {ToPrettyString(entity.Owner):entity} onto {ToPrettyString(hit):target}");
2023-04-10 18:56:21 -07:00
_reactive.DoEntityReaction(hit, splitSolution, ReactionMethod.Touch);
_popups.PopupEntity(
Loc.GetString("spill-melee-hit-attacker", ("amount", totalSplit / hitCount), ("spillable", entity.Owner),
2023-04-10 18:56:21 -07:00
("target", Identity.Entity(hit, EntityManager))),
hit, args.User);
_popups.PopupEntity(
Loc.GetString("spill-melee-hit-others", ("attacker", args.User), ("spillable", entity.Owner),
2023-04-10 18:56:21 -07:00
("target", Identity.Entity(hit, EntityManager))),
hit, Filter.PvsExcept(args.User), true, PopupType.SmallCaution);
}
}
private void OnGotEquipped(Entity<SpillableComponent> entity, ref GotEquippedEvent args)
2023-04-10 15:37:03 +10:00
{
if (!entity.Comp.SpillWorn)
2023-04-10 15:37:03 +10:00
return;
if (!TryComp(entity, out ClothingComponent? clothing))
2023-04-10 15:37:03 +10:00
return;
// check if entity was actually used as clothing
// not just taken in pockets or something
var isCorrectSlot = clothing.Slots.HasFlag(args.SlotFlags);
if (!isCorrectSlot)
return;
if (!_solutionContainerSystem.TryGetSolution(entity.Owner, entity.Comp.SolutionName, out var soln, out var solution))
2023-04-10 15:37:03 +10:00
return;
// block access to the solution while worn
AddComp<BlockSolutionAccessComponent>(entity);
2023-04-10 15:37:03 +10:00
if (solution.Volume == 0)
return;
// spill all solution on the player
var drainedSolution = _solutionContainerSystem.Drain(entity.Owner, soln.Value, solution.Volume);
TrySplashSpillAt(entity.Owner, Transform(args.Equipee).Coordinates, drainedSolution, out _);
2023-04-10 15:37:03 +10:00
}
private void OnGotUnequipped(Entity<SpillableComponent> entity, ref GotUnequippedEvent args)
{
if (!entity.Comp.SpillWorn)
return;
RemCompDeferred<BlockSolutionAccessComponent>(entity);
}
private void SpillOnLand(Entity<SpillableComponent> entity, ref LandEvent args)
2023-04-10 15:37:03 +10:00
{
if (!_solutionContainerSystem.TryGetSolution(entity.Owner, entity.Comp.SolutionName, out var soln, out var solution))
2023-04-10 15:37:03 +10:00
return;
if (Openable.IsClosed(entity.Owner))
2023-04-10 15:37:03 +10:00
return;
if (!entity.Comp.SpillWhenThrown)
return;
2023-04-10 15:37:03 +10:00
if (args.User != null)
{
_adminLogger.Add(LogType.Landed,
$"{ToPrettyString(entity.Owner):entity} spilled a solution {SolutionContainerSystem.ToPrettyString(solution):solution} on landing");
2023-04-10 15:37:03 +10:00
}
var drainedSolution = _solutionContainerSystem.Drain(entity.Owner, soln.Value, solution.Volume);
TrySplashSpillAt(entity.Owner, Transform(entity).Coordinates, drainedSolution, out _);
2023-04-10 15:37:03 +10:00
}
/// <summary>
/// Prevent Pacified entities from throwing items that can spill liquids.
/// </summary>
private void OnAttemptPacifiedThrow(Entity<SpillableComponent> ent, ref AttemptPacifiedThrowEvent args)
{
// Dont care about closed containers.
if (Openable.IsClosed(ent))
return;
// Dont care about empty containers.
if (!_solutionContainerSystem.TryGetSolution(ent.Owner, ent.Comp.SolutionName, out _, out var solution) || solution.Volume <= 0)
return;
args.Cancel("pacified-cannot-throw-spill");
}
private void OnDoAfter(Entity<SpillableComponent> entity, ref SpillDoAfterEvent args)
2023-04-10 15:37:03 +10:00
{
if (args.Handled || args.Cancelled || args.Args.Target == null)
return;
//solution gone by other means before doafter completes
if (!_solutionContainerSystem.TryGetDrainableSolution(entity.Owner, out var soln, out var solution) || solution.Volume == 0)
2023-04-10 15:37:03 +10:00
return;
var puddleSolution = _solutionContainerSystem.SplitSolution(soln.Value, solution.Volume);
TrySpillAt(Transform(entity).Coordinates, puddleSolution, out _);
2023-04-10 15:37:03 +10:00
args.Handled = true;
}
}