188 lines
6.7 KiB
C#
188 lines
6.7 KiB
C#
using Content.Shared.Actions;
|
|
using Content.Shared.Hands.Components;
|
|
using Content.Shared.Hands.EntitySystems;
|
|
using Content.Shared.Popups;
|
|
using Content.Shared.Projectiles;
|
|
using Robust.Shared.GameStates;
|
|
using Robust.Shared.Player;
|
|
|
|
namespace Content.Shared.ItemRecall;
|
|
|
|
/// <summary>
|
|
/// System for handling the ItemRecall ability for wizards.
|
|
/// </summary>
|
|
public abstract partial class SharedItemRecallSystem : EntitySystem
|
|
{
|
|
[Dependency] private readonly ISharedPlayerManager _player = default!;
|
|
[Dependency] private readonly SharedPvsOverrideSystem _pvs = default!;
|
|
[Dependency] private readonly SharedActionsSystem _actions = default!;
|
|
[Dependency] private readonly SharedHandsSystem _hands = default!;
|
|
[Dependency] private readonly MetaDataSystem _metaData = default!;
|
|
[Dependency] private readonly SharedPopupSystem _popups = default!;
|
|
[Dependency] private readonly SharedProjectileSystem _proj = default!;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeLocalEvent<ItemRecallComponent, MapInitEvent>(OnMapInit);
|
|
SubscribeLocalEvent<ItemRecallComponent, OnItemRecallActionEvent>(OnItemRecallActionUse);
|
|
|
|
SubscribeLocalEvent<RecallMarkerComponent, ComponentShutdown>(OnRecallMarkerShutdown);
|
|
}
|
|
|
|
private void OnMapInit(Entity<ItemRecallComponent> ent, ref MapInitEvent args)
|
|
{
|
|
ent.Comp.InitialName = Name(ent);
|
|
ent.Comp.InitialDescription = Description(ent);
|
|
}
|
|
|
|
private void OnItemRecallActionUse(Entity<ItemRecallComponent> ent, ref OnItemRecallActionEvent args)
|
|
{
|
|
if (ent.Comp.MarkedEntity == null)
|
|
{
|
|
if (!TryComp<HandsComponent>(args.Performer, out var hands))
|
|
return;
|
|
|
|
var markItem = _hands.GetActiveItem((args.Performer, hands));
|
|
|
|
if (markItem == null)
|
|
{
|
|
_popups.PopupClient(Loc.GetString("item-recall-item-mark-empty"), args.Performer, args.Performer);
|
|
return;
|
|
}
|
|
|
|
if (HasComp<RecallMarkerComponent>(markItem))
|
|
{
|
|
_popups.PopupClient(Loc.GetString("item-recall-item-already-marked", ("item", markItem)), args.Performer, args.Performer);
|
|
return;
|
|
}
|
|
|
|
_popups.PopupClient(Loc.GetString("item-recall-item-marked", ("item", markItem.Value)), args.Performer, args.Performer);
|
|
TryMarkItem(ent, markItem.Value);
|
|
return;
|
|
}
|
|
|
|
RecallItem(ent.Comp.MarkedEntity.Value);
|
|
args.Handled = true;
|
|
}
|
|
|
|
private void RecallItem(Entity<RecallMarkerComponent?> ent)
|
|
{
|
|
if (!Resolve(ent.Owner, ref ent.Comp, false))
|
|
return;
|
|
|
|
if (!TryComp<InstantActionComponent>(ent.Comp.MarkedByAction, out var instantAction))
|
|
return;
|
|
|
|
var actionOwner = instantAction.AttachedEntity;
|
|
|
|
if (actionOwner == null)
|
|
return;
|
|
|
|
if (TryComp<EmbeddableProjectileComponent>(ent, out var projectile))
|
|
_proj.UnEmbed(ent, projectile, actionOwner.Value);
|
|
|
|
_popups.PopupPredicted(Loc.GetString("item-recall-item-summon", ("item", ent)), actionOwner.Value, actionOwner.Value);
|
|
|
|
_hands.TryForcePickupAnyHand(actionOwner.Value, ent);
|
|
}
|
|
|
|
private void OnRecallMarkerShutdown(Entity<RecallMarkerComponent> ent, ref ComponentShutdown args)
|
|
{
|
|
TryUnmarkItem(ent);
|
|
}
|
|
|
|
private void TryMarkItem(Entity<ItemRecallComponent> ent, EntityUid item)
|
|
{
|
|
if (!TryComp<InstantActionComponent>(ent, out var instantAction))
|
|
return;
|
|
|
|
var actionOwner = instantAction.AttachedEntity;
|
|
|
|
if (actionOwner == null)
|
|
return;
|
|
|
|
AddToPvsOverride(item, actionOwner.Value);
|
|
|
|
var marker = AddComp<RecallMarkerComponent>(item);
|
|
ent.Comp.MarkedEntity = item;
|
|
Dirty(ent);
|
|
|
|
marker.MarkedByAction = ent.Owner;
|
|
|
|
UpdateActionAppearance(ent);
|
|
Dirty(item, marker);
|
|
}
|
|
|
|
private void TryUnmarkItem(EntityUid item)
|
|
{
|
|
if (!TryComp<RecallMarkerComponent>(item, out var marker))
|
|
return;
|
|
|
|
if (!TryComp<InstantActionComponent>(marker.MarkedByAction, out var instantAction))
|
|
return;
|
|
|
|
if (TryComp<ItemRecallComponent>(marker.MarkedByAction, out var action))
|
|
{
|
|
// For some reason client thinks the station grid owns the action on client and this doesn't work. It doesn't work in PopupEntity(mispredicts) and PopupPredicted either(doesnt show).
|
|
// I don't have the heart to move this code to server because of this small thing.
|
|
// This line will only do something once that is fixed.
|
|
if (instantAction.AttachedEntity != null)
|
|
{
|
|
_popups.PopupClient(Loc.GetString("item-recall-item-unmark", ("item", item)), instantAction.AttachedEntity.Value, instantAction.AttachedEntity.Value, PopupType.MediumCaution);
|
|
RemoveFromPvsOverride(item, instantAction.AttachedEntity.Value);
|
|
}
|
|
|
|
action.MarkedEntity = null;
|
|
UpdateActionAppearance((marker.MarkedByAction.Value, action));
|
|
Dirty(marker.MarkedByAction.Value, action);
|
|
}
|
|
|
|
RemCompDeferred<RecallMarkerComponent>(item);
|
|
}
|
|
|
|
private void UpdateActionAppearance(Entity<ItemRecallComponent> action)
|
|
{
|
|
if (!TryComp<InstantActionComponent>(action, out var instantAction))
|
|
return;
|
|
|
|
if (action.Comp.MarkedEntity == null)
|
|
{
|
|
if (action.Comp.InitialName != null)
|
|
_metaData.SetEntityName(action, action.Comp.InitialName);
|
|
if (action.Comp.InitialDescription != null)
|
|
_metaData.SetEntityDescription(action, action.Comp.InitialDescription);
|
|
_actions.SetEntityIcon(action, null, instantAction);
|
|
}
|
|
else
|
|
{
|
|
if (action.Comp.WhileMarkedName != null)
|
|
_metaData.SetEntityName(action, Loc.GetString(action.Comp.WhileMarkedName,
|
|
("item", action.Comp.MarkedEntity.Value)));
|
|
|
|
if (action.Comp.WhileMarkedDescription != null)
|
|
_metaData.SetEntityDescription(action, Loc.GetString(action.Comp.WhileMarkedDescription,
|
|
("item", action.Comp.MarkedEntity.Value)));
|
|
|
|
_actions.SetEntityIcon(action, action.Comp.MarkedEntity, instantAction);
|
|
}
|
|
}
|
|
|
|
private void AddToPvsOverride(EntityUid uid, EntityUid user)
|
|
{
|
|
if (!_player.TryGetSessionByEntity(user, out var mindSession))
|
|
return;
|
|
|
|
_pvs.AddSessionOverride(uid, mindSession);
|
|
}
|
|
|
|
private void RemoveFromPvsOverride(EntityUid uid, EntityUid user)
|
|
{
|
|
if (!_player.TryGetSessionByEntity(user, out var mindSession))
|
|
return;
|
|
|
|
_pvs.RemoveSessionOverride(uid, mindSession);
|
|
}
|
|
}
|