using Content.Shared.Administration.Logs;
using Content.Shared.Charges.Components;
using Content.Shared.Charges.Systems;
using Content.Shared.Database;
using Content.Shared.DoAfter;
using Content.Shared.Examine;
using Content.Shared.Interaction;
using Content.Shared.Popups;
using Content.Shared.SprayPainter.Components;
using Content.Shared.SprayPainter.Prototypes;
using Content.Shared.Verbs;
using Robust.Shared.Audio.Systems;
using Robust.Shared.Prototypes;
using Robust.Shared.Timing;
using Robust.Shared.Utility;
using System.Linq;
namespace Content.Shared.SprayPainter;
///
/// System for painting paintable objects using a spray painter.
/// Pipes are handled serverside since AtmosPipeColorSystem is server only.
///
public abstract class SharedSprayPainterSystem : EntitySystem
{
[Dependency] private readonly IGameTiming _timing = default!;
[Dependency] protected readonly IPrototypeManager Proto = default!;
[Dependency] protected readonly ISharedAdminLogManager AdminLogger = default!;
[Dependency] protected readonly SharedAppearanceSystem Appearance = default!;
[Dependency] protected readonly SharedAudioSystem Audio = default!;
[Dependency] protected readonly SharedChargesSystem Charges = default!;
[Dependency] protected readonly SharedDoAfterSystem DoAfter = default!;
[Dependency] private readonly SharedPopupSystem _popup = default!;
public override void Initialize()
{
base.Initialize();
SubscribeLocalEvent(OnMapInit);
SubscribeLocalEvent(OnPainterDoAfter);
SubscribeLocalEvent>(OnPainterGetAltVerbs);
SubscribeLocalEvent(OnPaintableInteract);
SubscribeLocalEvent(OnPainedExamined);
Subs.BuiEvents(SprayPainterUiKey.Key,
subs =>
{
subs.Event(OnSetPaintable);
subs.Event(OnSetPipeColor);
subs.Event(OnTabChanged);
subs.Event(OnSetDecal);
subs.Event(OnSetDecalColor);
subs.Event(OnSetDecalAngle);
subs.Event(OnSetDecalSnap);
});
}
private void OnMapInit(Entity ent, ref MapInitEvent args)
{
bool stylesByGroupPopulated = false;
foreach (var groupProto in Proto.EnumeratePrototypes())
{
ent.Comp.StylesByGroup[groupProto.ID] = groupProto.DefaultStyle;
stylesByGroupPopulated = true;
}
if (stylesByGroupPopulated)
Dirty(ent);
if (ent.Comp.ColorPalette.Count > 0)
SetPipeColor(ent, ent.Comp.ColorPalette.First().Key);
}
private void SetPipeColor(Entity ent, string? paletteKey)
{
if (paletteKey == null || paletteKey == ent.Comp.PickedColor)
return;
if (!ent.Comp.ColorPalette.ContainsKey(paletteKey))
return;
ent.Comp.PickedColor = paletteKey;
Dirty(ent);
UpdateUi(ent);
}
#region Interaction
private void OnPainterDoAfter(Entity ent, ref SprayPainterDoAfterEvent args)
{
if (args.Handled || args.Cancelled)
return;
if (args.Args.Target is not { } target)
return;
if (!HasComp(target))
return;
Appearance.SetData(target, PaintableVisuals.Prototype, args.Prototype);
Audio.PlayPredicted(ent.Comp.SpraySound, ent, args.Args.User);
Charges.TryUseCharges(new Entity(ent, EnsureComp(ent)), args.Cost);
var paintedComponent = EnsureComp(target);
paintedComponent.DryTime = _timing.CurTime + ent.Comp.FreshPaintDuration;
Dirty(target, paintedComponent);
var ev = new EntityPaintedEvent(
User: args.User,
Tool: ent,
Prototype: args.Prototype,
Group: args.Group);
RaiseLocalEvent(target, ref ev);
AdminLogger.Add(LogType.Action,
LogImpact.Low,
$"{ToPrettyString(args.Args.User):user} painted {ToPrettyString(args.Args.Target.Value):target}");
args.Handled = true;
}
private void OnPainterGetAltVerbs(Entity ent, ref GetVerbsEvent args)
{
if (!args.CanAccess || !args.CanInteract || !args.Using.HasValue)
return;
var user = args.User;
AlternativeVerb verb = new()
{
Text = Loc.GetString("spray-painter-verb-toggle-decals"),
Icon = new SpriteSpecifier.Texture(new ResPath("/Textures/Interface/VerbIcons/settings.svg.192dpi.png")),
Act = () => TogglePaintDecals(ent, user),
Impact = LogImpact.Low
};
args.Verbs.Add(verb);
}
///
/// Toggles whether clicking on the floor paints a decal or not.
///
private void TogglePaintDecals(Entity ent, EntityUid user)
{
if (!_timing.IsFirstTimePredicted)
return;
var pitch = 1.0f;
switch (ent.Comp.DecalMode)
{
case DecalPaintMode.Off:
default:
ent.Comp.DecalMode = DecalPaintMode.Add;
pitch = 1.0f;
break;
case DecalPaintMode.Add:
ent.Comp.DecalMode = DecalPaintMode.Remove;
pitch = 1.2f;
break;
case DecalPaintMode.Remove:
ent.Comp.DecalMode = DecalPaintMode.Off;
pitch = 0.8f;
break;
}
Dirty(ent);
// Make the machine beep.
Audio.PlayPredicted(ent.Comp.SoundSwitchDecalMode, ent, user, ent.Comp.SoundSwitchDecalMode.Params.WithPitchScale(pitch));
}
///
/// Handles spray paint interactions with an object.
/// An object must belong to a spray paintable group to be painted, and the painter must have sufficient ammo to paint it.
///
private void OnPaintableInteract(Entity ent, ref InteractUsingEvent args)
{
if (args.Handled)
return;
if (!TryComp(args.Used, out var painter))
return;
if (ent.Comp.Group is not { } group
|| !painter.StylesByGroup.TryGetValue(group, out var selectedStyle)
|| !Proto.Resolve(group, out PaintableGroupPrototype? targetGroup))
return;
// Valid paint target.
args.Handled = true;
if (TryComp(args.Used, out var charges)
&& charges.LastCharges < targetGroup.Cost)
{
var msg = Loc.GetString("spray-painter-interact-no-charges");
_popup.PopupClient(msg, args.User, args.User);
return;
}
if (!targetGroup.Styles.TryGetValue(selectedStyle, out var proto))
{
var msg = Loc.GetString("spray-painter-style-not-available");
_popup.PopupClient(msg, args.User, args.User);
return;
}
var doAfterEventArgs = new DoAfterArgs(EntityManager,
args.User,
targetGroup.Time,
new SprayPainterDoAfterEvent(proto, group, targetGroup.Cost),
args.Used,
target: ent,
used: args.Used)
{
BreakOnMove = true,
BreakOnDamage = true,
NeedHand = true,
};
if (!DoAfter.TryStartDoAfter(doAfterEventArgs, out _))
return;
// Log the attempt
AdminLogger.Add(LogType.Action,
LogImpact.Low,
$"{ToPrettyString(args.User):user} is painting {ToPrettyString(ent):target} to '{selectedStyle}' at {Transform(ent).Coordinates:targetlocation}");
}
///
/// Prints out if an object has been painted recently.
///
private void OnPainedExamined(Entity ent, ref ExaminedEvent args)
{
// If the paint's dried, it isn't detectable.
if (_timing.CurTime > ent.Comp.DryTime)
return;
args.PushText(Loc.GetString("spray-painter-on-examined-painted-message"));
}
#endregion Interaction
#region UI
///
/// Sets the style that a particular type of paintable object (e.g. lockers) should be painted in.
///
private void OnSetPaintable(Entity ent, ref SprayPainterSetPaintableStyleMessage args)
{
if (!ent.Comp.StylesByGroup.ContainsKey(args.Group))
return;
ent.Comp.StylesByGroup[args.Group] = args.Style;
Dirty(ent);
UpdateUi(ent);
}
///
/// Changes the color to paint pipes in.
///
private void OnSetPipeColor(Entity ent, ref SprayPainterSetPipeColorMessage args)
{
SetPipeColor(ent, args.Key);
}
///
/// Tracks the tab the spray painter was on.
///
private void OnTabChanged(Entity ent, ref SprayPainterTabChangedMessage args)
{
ent.Comp.SelectedTab = args.Index;
Dirty(ent);
}
///
/// Sets the decal prototype to paint.
///
private void OnSetDecal(Entity ent, ref SprayPainterSetDecalMessage args)
{
ent.Comp.SelectedDecal = args.DecalPrototype;
Dirty(ent);
UpdateUi(ent);
}
///
/// Sets the angle to paint decals at.
///
private void OnSetDecalAngle(Entity ent, ref SprayPainterSetDecalAngleMessage args)
{
ent.Comp.SelectedDecalAngle = args.Angle;
Dirty(ent);
UpdateUi(ent);
}
///
/// Enables or disables snap-to-grid when painting decals.
///
private void OnSetDecalSnap(Entity ent, ref SprayPainterSetDecalSnapMessage args)
{
ent.Comp.SnapDecals = args.Snap;
Dirty(ent);
UpdateUi(ent);
}
///
/// Sets the decal to paint on the ground.
///
private void OnSetDecalColor(Entity ent, ref SprayPainterSetDecalColorMessage args)
{
ent.Comp.SelectedDecalColor = args.Color;
Dirty(ent);
UpdateUi(ent);
}
protected virtual void UpdateUi(Entity ent)
{
}
#endregion
}