195 lines
6.3 KiB
C#
195 lines
6.3 KiB
C#
using System.Linq;
|
|
using Content.Shared.Actions;
|
|
using Content.Shared.Charges.Components;
|
|
using Content.Shared.Charges.Systems;
|
|
using Content.Shared.Interaction;
|
|
using Robust.Shared.Random;
|
|
using Robust.Shared.Timing;
|
|
|
|
namespace Content.Server.Actions;
|
|
|
|
/// <summary>
|
|
/// This System handled interactions for the <see cref="ActionOnInteractComponent"/>.
|
|
/// </summary>
|
|
public sealed class ActionOnInteractSystem : EntitySystem
|
|
{
|
|
[Dependency] private readonly IRobustRandom _random = default!;
|
|
[Dependency] private readonly IGameTiming _timing = default!;
|
|
[Dependency] private readonly SharedActionsSystem _actions = default!;
|
|
[Dependency] private readonly ActionContainerSystem _actionContainer = default!;
|
|
[Dependency] private readonly SharedChargesSystem _charges = default!;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeLocalEvent<ActionOnInteractComponent, ActivateInWorldEvent>(OnActivate);
|
|
SubscribeLocalEvent<ActionOnInteractComponent, AfterInteractEvent>(OnAfterInteract);
|
|
SubscribeLocalEvent<ActionOnInteractComponent, MapInitEvent>(OnMapInit);
|
|
}
|
|
|
|
private void OnMapInit(EntityUid uid, ActionOnInteractComponent component, MapInitEvent args)
|
|
{
|
|
if (component.Actions == null)
|
|
return;
|
|
|
|
var comp = EnsureComp<ActionsContainerComponent>(uid);
|
|
foreach (var id in component.Actions)
|
|
{
|
|
_actionContainer.AddAction(uid, id, comp);
|
|
}
|
|
}
|
|
|
|
private void OnActivate(EntityUid uid, ActionOnInteractComponent component, ActivateInWorldEvent args)
|
|
{
|
|
if (args.Handled || !args.Complex)
|
|
return;
|
|
|
|
if (component.ActionEntities is not {} actionEnts)
|
|
{
|
|
if (!TryComp<ActionsContainerComponent>(uid, out var actionsContainerComponent))
|
|
return;
|
|
|
|
actionEnts = actionsContainerComponent.Container.ContainedEntities.ToList();
|
|
}
|
|
|
|
var options = GetValidActions<InstantActionComponent>(actionEnts);
|
|
if (options.Count == 0)
|
|
return;
|
|
|
|
if (!TryUseCharge((uid, component)))
|
|
return;
|
|
|
|
var (actId, act) = _random.Pick(options);
|
|
_actions.PerformAction(args.User, null, actId, act, act.Event, _timing.CurTime, false);
|
|
args.Handled = true;
|
|
}
|
|
|
|
private void OnAfterInteract(EntityUid uid, ActionOnInteractComponent component, AfterInteractEvent args)
|
|
{
|
|
if (args.Handled)
|
|
return;
|
|
|
|
if (component.ActionEntities is not {} actionEnts)
|
|
{
|
|
if (!TryComp<ActionsContainerComponent>(uid, out var actionsContainerComponent))
|
|
return;
|
|
|
|
actionEnts = actionsContainerComponent.Container.ContainedEntities.ToList();
|
|
}
|
|
|
|
// First, try entity target actions
|
|
if (args.Target != null)
|
|
{
|
|
var entOptions = GetValidActions<EntityTargetActionComponent>(actionEnts, args.CanReach);
|
|
for (var i = entOptions.Count - 1; i >= 0; i--)
|
|
{
|
|
var action = entOptions[i];
|
|
if (!_actions.ValidateEntityTarget(args.User, args.Target.Value, action))
|
|
entOptions.RemoveAt(i);
|
|
}
|
|
|
|
if (entOptions.Count > 0)
|
|
{
|
|
if (!TryUseCharge((uid, component)))
|
|
return;
|
|
|
|
var (entActId, entAct) = _random.Pick(entOptions);
|
|
if (entAct.Event != null)
|
|
{
|
|
entAct.Event.Target = args.Target.Value;
|
|
}
|
|
|
|
_actions.PerformAction(args.User, null, entActId, entAct, entAct.Event, _timing.CurTime, false);
|
|
args.Handled = true;
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Then EntityWorld target actions
|
|
var entWorldOptions = GetValidActions<EntityWorldTargetActionComponent>(actionEnts, args.CanReach);
|
|
for (var i = entWorldOptions.Count - 1; i >= 0; i--)
|
|
{
|
|
var action = entWorldOptions[i];
|
|
if (!_actions.ValidateEntityWorldTarget(args.User, args.Target, args.ClickLocation, action))
|
|
entWorldOptions.RemoveAt(i);
|
|
}
|
|
|
|
if (entWorldOptions.Count > 0)
|
|
{
|
|
if (!TryUseCharge((uid, component)))
|
|
return;
|
|
|
|
var (entActId, entAct) = _random.Pick(entWorldOptions);
|
|
if (entAct.Event != null)
|
|
{
|
|
entAct.Event.Entity = args.Target;
|
|
entAct.Event.Coords = args.ClickLocation;
|
|
}
|
|
|
|
_actions.PerformAction(args.User, null, entActId, entAct, entAct.Event, _timing.CurTime, false);
|
|
args.Handled = true;
|
|
return;
|
|
}
|
|
|
|
// else: try world target actions
|
|
var options = GetValidActions<WorldTargetActionComponent>(component.ActionEntities, args.CanReach);
|
|
for (var i = options.Count - 1; i >= 0; i--)
|
|
{
|
|
var action = options[i];
|
|
if (!_actions.ValidateWorldTarget(args.User, args.ClickLocation, action))
|
|
options.RemoveAt(i);
|
|
}
|
|
|
|
if (options.Count == 0)
|
|
return;
|
|
|
|
if (!TryUseCharge((uid, component)))
|
|
return;
|
|
|
|
var (actId, act) = _random.Pick(options);
|
|
if (act.Event != null)
|
|
{
|
|
act.Event.Target = args.ClickLocation;
|
|
}
|
|
|
|
_actions.PerformAction(args.User, null, actId, act, act.Event, _timing.CurTime, false);
|
|
args.Handled = true;
|
|
}
|
|
|
|
private List<(EntityUid Id, T Comp)> GetValidActions<T>(List<EntityUid>? actions, bool canReach = true) where T : BaseActionComponent
|
|
{
|
|
var valid = new List<(EntityUid Id, T Comp)>();
|
|
|
|
if (actions == null)
|
|
return valid;
|
|
|
|
foreach (var id in actions)
|
|
{
|
|
if (!_actions.TryGetActionData(id, out var baseAction) ||
|
|
baseAction as T is not { } action ||
|
|
!_actions.ValidAction(action, canReach))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
valid.Add((id, action));
|
|
}
|
|
|
|
return valid;
|
|
}
|
|
|
|
private bool TryUseCharge(Entity<ActionOnInteractComponent> ent)
|
|
{
|
|
if (!ent.Comp.RequiresCharge)
|
|
return true;
|
|
|
|
Entity<LimitedChargesComponent?> charges = ent.Owner;
|
|
if (_charges.IsEmpty(charges))
|
|
return false;
|
|
|
|
_charges.TryUseCharge(charges);
|
|
return true;
|
|
}
|
|
}
|