* Suhariki (#4) * init * Porting is complete Still WIP * small stuff * SmallStuff * awawawa * Нас рано * Engi * More luck * removed some shit (cherry picked from commit e1d745655b0e7253ea4e4814b82dd56e90153514) * Самодельные отражайки (#6) * Ghetto Mirror Shield * Protorypes for ghetto mirrors * locale * idk, works fine on my machine * forsenPls (cherry picked from commit 4505739fd9430bcc578da3dec932d9aa7ca933ca) * Шлем из ведра (#7) * Bucket Helmet * Code cleanup * Squashed commit of the following: commit 4505739fd9430bcc578da3dec932d9aa7ca933ca Author: BIGZi0348 <118811750+BIGZi0348@users.noreply.github.com> Date: Sat Sep 21 17:10:02 2024 +0300 Самодельные отражайки (#6) * Ghetto Mirror Shield * Protorypes for ghetto mirrors * locale * idk, works fine on my machine * forsenPls commit 1f066876508dec8d3f21ad753dab9fcc13dc8d1a Author: BIGZi0348 <118811750+BIGZi0348@users.noreply.github.com> Date: Sat Sep 21 14:57:30 2024 +0300 Агрессивный вор (#5) * AggressiveThief * Попытка насрать обратно commit e1d745655b0e7253ea4e4814b82dd56e90153514 Author: BIGZi0348 <118811750+BIGZi0348@users.noreply.github.com> Date: Sat Sep 21 14:46:38 2024 +0300 Suhariki (#4) * init * Porting is complete Still WIP * small stuff * SmallStuff * awawawa * Нас рано * Engi * More luck * removed some shit commit 08061b4f1fbbfd521e3a40ef53d9bc86fa140087 Author: BIGZi0348 <118811750+BIGZi0348@users.noreply.github.com> Date: Sat Sep 21 14:30:31 2024 +0300 Engi rise up (#3) commit 00e2f5c45ca7abfeff1f85d4893352d8875f8560 Author: BIGZi0348 <118811750+BIGZi0348@users.noreply.github.com> Date: Thu Sep 19 19:42:30 2024 +0300 Bigzi upstream1 (#2) * init * Porting is complete Still WIP * Мелочи - Король крыс, диски, правки перевода и респрайт одеяния культа (#686) * Большой Крыс Отдыхает * Фикс размера фейк диска * Доработка диска очков РНД * Скоро перейду на райдер * Переводы * Диски одного размера * Респрайт робы и капюшона культа * Automatic changelog update * fix (#687) * Automatic changelog update * govno HD (#688) * Automatic changelog update * Hotfix loadouts (#689) * govno HD * hotfix loadouts * fuck * Automatic changelog update * Update PULL_REQUEST_TEMPLATE.md * Even smaller stuff (#690) * 45 to 44 * Фикс ошибки перевода * Voice mask translate * Automatic changelog update * small stuff * SmallStuff * awawawa * Локализация шепелявости (#691) * БУМ!!!!!!!! (#692) * Automatic changelog update * Большой ребаланс милишки (#681) * MeleeThrowOnHit rework * buff baseball bat * better mjolnir * telebaton system unhardcode + refactor + transform TelescopicBatonComponent to KnockDownOnHitComponent * fix telebaton prototype * darova * fix KnockDownOnHitSystem * chaplain weapons rebalance * fix nullrod hit sound * BloodstreamSystem cleanup * bleeding rebalance * damage rebalance * small baseball bat fix * Automatic changelog update * Chaplain armor fix (#693) * Фикс описания (#696) * No passengers on this military complex (#695) * Automatic changelog update * Переводы (#698) * Пепеводы * Очепятки * Фикс прожектора и открутка цифр (#697) * Melochi * Фикс хамелеона * Переводы денег мне на карту * Automatic changelog update * Нас рано * Фикс пополнения зарядов РЦД используя стаки ресурсов (#699) * Фиксики * Спасли Валеру от банкротства * Very hot man (#700) * Automatic changelog update * reverted illegal code added by oniks --------- Co-authored-by: RavmorganButOnCocaine <valtos@nextmail.ru> Co-authored-by: Spatison <137375981+Spatison@users.noreply.github.com> Co-authored-by: Valtos <valtos@spaces.ru> Co-authored-by: Jabak <163307958+Jabaks@users.noreply.github.com> Co-authored-by: ThereDrD <88589686+ThereDrD0@users.noreply.github.com> * ВЕДРО!!! * Funny * Oppsie (cherry picked from commit 92ce631b0c0187678b6d1eea34d3c0830b9b0fac) * Yes (#8) (cherry picked from commit 3d083befca212455cffcae56b84ee55050ad62be) * Small Fix То чувство, когда можешь срать прямо в мэйн (cherry picked from commit 8fa7cfb678711a55bf6704ced30f7ed9b4b1e8c4) * bababooey (#9) (cherry picked from commit f7ed9d287f6798c7d39ac437a2f969ec0dc10309) * GoodLooking (#10) (cherry picked from commit 1617dce502b71db22a6b736ac879c04c07ec54a0) * Dadka scazal (#11) (cherry picked from commit 50da866354952628185bb396f359c2f533d57340) * Сортировка эксклюзива енги + респрайт деревянных столов и дверей (#12) * Massive * Чейндлок (cherry picked from commit 4dc165e4c1aa75e8fd22f14db95ac72bf50c77f5) * Big Zi bucket helmet fix * Опять насрал в мастер (cherry picked from commit bd866e0cd87770b1509885f66ec5c562dda3fd91) * Боевые Молоты (#14) (cherry picked from commit 64df6b4b91050273218007365cf81a45c3ea74a8) * хотфикс сухариков (cherry picked from commit a98265c22dc7de467693f501cf3a5d8577e94a54) * funny sword (#15) (cherry picked from commit 231aaa1b33b7e4edf00506c6ebdbd389257cf1ca) * funny hammer (cherry picked from commit 1e58eaa146cc95f34358d97d6c436ba4705d09a1) * Добавил админ логирование PacifiedOnChaplainActionSystem (#19) Update PacifiedOnChaplainActionSystem.cs (cherry picked from commit f6aa42eab7cce294336aec14f7dfb6f0671267cf) * Small refactor (cherry picked from commit b38ad7e7b8afa35a22c84c2fb8836bf13736dfb3) * Команда для отправки цели для станции (#21) * Команда для отправки цели для станции * Правки (cherry picked from commit 93a79c4952e3a75c32ff5cc4d07df4018a9bf7db) * Больше целей (cherry picked from commit c3d34d6a4e000d8af66c8f53d60a898cfae0a454) * Music for events (cherry picked from commit 4a7214634b3e77caefd011bf5900f581055a0ac8) * Санитизация * unused --------- Co-authored-by: BIGZi0348 <118811750+bigzi0348@users.noreply.github.com> Co-authored-by: BIGZi0348 <svalker0348@gmail.com>
660 lines
27 KiB
C#
660 lines
27 KiB
C#
using System.Linq;
|
|
using Content.Server.Administration.Logs;
|
|
using Content.Server.Ensnaring;
|
|
using Content.Shared.CombatMode;
|
|
using Content.Shared.Cuffs;
|
|
using Content.Shared.Cuffs.Components;
|
|
using Content.Shared.Database;
|
|
using Content.Shared.DoAfter;
|
|
using Content.Shared.Ensnaring.Components;
|
|
using Content.Shared.Hands.Components;
|
|
using Content.Shared.Hands.EntitySystems;
|
|
using Content.Shared.IdentityManagement;
|
|
using Content.Shared.Interaction;
|
|
using Content.Shared.Interaction.Events;
|
|
using Content.Shared.Inventory;
|
|
using Content.Shared.Inventory.VirtualItem;
|
|
using Content.Shared.Popups;
|
|
using Content.Shared.Strip;
|
|
using Content.Shared.Strip.Components;
|
|
using Content.Shared.Verbs;
|
|
using Content.Shared._White.MagGloves;
|
|
using Robust.Server.GameObjects;
|
|
using Robust.Shared.Player;
|
|
using Robust.Shared.Utility;
|
|
using Content.Shared._White._Engi.BucketHelmet;
|
|
|
|
namespace Content.Server.Strip
|
|
{
|
|
public sealed class StrippableSystem : SharedStrippableSystem
|
|
{
|
|
[Dependency] private readonly InventorySystem _inventorySystem = default!;
|
|
[Dependency] private readonly EnsnareableSystem _ensnaringSystem = default!;
|
|
[Dependency] private readonly UserInterfaceSystem _userInterfaceSystem = default!;
|
|
|
|
[Dependency] private readonly SharedCuffableSystem _cuffableSystem = default!;
|
|
[Dependency] private readonly SharedDoAfterSystem _doAfterSystem = default!;
|
|
[Dependency] private readonly SharedHandsSystem _handsSystem = default!;
|
|
[Dependency] private readonly SharedPopupSystem _popupSystem = default!;
|
|
|
|
[Dependency] private readonly IAdminLogManager _adminLogger = default!;
|
|
|
|
// TODO: ECS popups. Not all of these have ECS equivalents yet.
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
|
|
SubscribeLocalEvent<StrippableComponent, GetVerbsEvent<Verb>>(AddStripVerb);
|
|
SubscribeLocalEvent<StrippableComponent, GetVerbsEvent<ExamineVerb>>(AddStripExamineVerb);
|
|
SubscribeLocalEvent<StrippableComponent, ActivateInWorldEvent>(OnActivateInWorld);
|
|
|
|
// BUI
|
|
SubscribeLocalEvent<StrippableComponent, StrippingSlotButtonPressed>(OnStripButtonPressed);
|
|
SubscribeLocalEvent<EnsnareableComponent, StrippingEnsnareButtonPressed>(OnStripEnsnareMessage);
|
|
|
|
// DoAfters
|
|
SubscribeLocalEvent<HandsComponent, DoAfterAttemptEvent<StrippableDoAfterEvent>>(OnStrippableDoAfterRunning);
|
|
SubscribeLocalEvent<HandsComponent, StrippableDoAfterEvent>(OnStrippableDoAfterFinished);
|
|
}
|
|
|
|
private void AddStripVerb(EntityUid uid, StrippableComponent component, GetVerbsEvent<Verb> args)
|
|
{
|
|
if (args.Hands == null || !args.CanAccess || !args.CanInteract || args.Target == args.User)
|
|
return;
|
|
|
|
if (!HasComp<ActorComponent>(args.User))
|
|
return;
|
|
|
|
Verb verb = new()
|
|
{
|
|
Text = Loc.GetString("strip-verb-get-data-text"),
|
|
Icon = new SpriteSpecifier.Texture(new("/Textures/Interface/VerbIcons/outfit.svg.192dpi.png")),
|
|
Act = () => StartOpeningStripper(args.User, (uid, component), true),
|
|
};
|
|
|
|
args.Verbs.Add(verb);
|
|
}
|
|
|
|
private void AddStripExamineVerb(EntityUid uid, StrippableComponent component, GetVerbsEvent<ExamineVerb> args)
|
|
{
|
|
if (args.Hands == null || !args.CanAccess || !args.CanInteract || args.Target == args.User)
|
|
return;
|
|
|
|
if (!HasComp<ActorComponent>(args.User))
|
|
return;
|
|
|
|
ExamineVerb verb = new()
|
|
{
|
|
Text = Loc.GetString("strip-verb-get-data-text"),
|
|
Icon = new SpriteSpecifier.Texture(new("/Textures/Interface/VerbIcons/outfit.svg.192dpi.png")),
|
|
Act = () => StartOpeningStripper(args.User, (uid, component), true),
|
|
Category = VerbCategory.Examine,
|
|
};
|
|
|
|
args.Verbs.Add(verb);
|
|
}
|
|
|
|
private void OnActivateInWorld(EntityUid uid, StrippableComponent component, ActivateInWorldEvent args)
|
|
{
|
|
if (args.Target == args.User)
|
|
return;
|
|
|
|
if (!HasComp<ActorComponent>(args.User))
|
|
return;
|
|
|
|
StartOpeningStripper(args.User, (uid, component));
|
|
}
|
|
|
|
public override void StartOpeningStripper(EntityUid user, Entity<StrippableComponent> strippable, bool openInCombat = false)
|
|
{
|
|
base.StartOpeningStripper(user, strippable, openInCombat);
|
|
|
|
if (TryComp<CombatModeComponent>(user, out var mode) && mode.IsInCombatMode && !openInCombat)
|
|
return;
|
|
|
|
if (HasComp<StrippingComponent>(user))
|
|
{
|
|
_userInterfaceSystem.OpenUi(strippable.Owner, StrippingUiKey.Key, user);
|
|
}
|
|
}
|
|
|
|
private void OnStripButtonPressed(Entity<StrippableComponent> strippable, ref StrippingSlotButtonPressed args)
|
|
{
|
|
if (args.Actor is not { Valid: true } user ||
|
|
!TryComp<HandsComponent>(user, out var userHands))
|
|
{
|
|
return;
|
|
}
|
|
|
|
// WD EDIT
|
|
if ((args.Slot == "gloves" || args.IsHand) && TryComp(strippable, out PreventStrippingFromHandsAndGlovesComponent? _))
|
|
{
|
|
var message = Loc.GetString("maggloves-cant-strip");
|
|
_popupSystem.PopupEntity(message, user, user);
|
|
return;
|
|
}
|
|
// WD EDIT END
|
|
|
|
// WD EDIT START
|
|
if (args.Slot == "ears" && TryComp(strippable, out PreventStrippingFromEarsComponent? _))
|
|
{
|
|
var message = Loc.GetString("buckethelmet-cant-strip");
|
|
_popupSystem.PopupEntity(message, user, user);
|
|
return;
|
|
}
|
|
// WD EDIT END
|
|
|
|
|
|
if (args.IsHand)
|
|
{
|
|
StripHand((user, userHands), (strippable.Owner, null), args.Slot, strippable);
|
|
return;
|
|
}
|
|
|
|
if (!TryComp<InventoryComponent>(strippable, out var inventory))
|
|
return;
|
|
|
|
var hasEnt = _inventorySystem.TryGetSlotEntity(strippable, args.Slot, out var held, inventory);
|
|
|
|
if (userHands.ActiveHandEntity != null && !hasEnt)
|
|
StartStripInsertInventory((user, userHands), strippable.Owner, userHands.ActiveHandEntity.Value, args.Slot);
|
|
else if (userHands.ActiveHandEntity == null && hasEnt)
|
|
StartStripRemoveInventory(user, strippable.Owner, held!.Value, args.Slot);
|
|
}
|
|
|
|
private void StripHand(
|
|
Entity<HandsComponent?> user,
|
|
Entity<HandsComponent?> target,
|
|
string handId,
|
|
StrippableComponent? targetStrippable)
|
|
{
|
|
if (!Resolve(user, ref user.Comp) ||
|
|
!Resolve(target, ref target.Comp) ||
|
|
!Resolve(target, ref targetStrippable))
|
|
return;
|
|
|
|
if (!_handsSystem.TryGetHand(target.Owner, handId, out var handSlot))
|
|
return;
|
|
|
|
// Is the target a handcuff?
|
|
if (TryComp<VirtualItemComponent>(handSlot.HeldEntity, out var virtualItem) &&
|
|
TryComp<CuffableComponent>(target.Owner, out var cuffable) &&
|
|
_cuffableSystem.GetAllCuffs(cuffable).Contains(virtualItem.BlockingEntity))
|
|
{
|
|
_cuffableSystem.TryUncuff(target.Owner, user, virtualItem.BlockingEntity, cuffable);
|
|
return;
|
|
}
|
|
|
|
if (user.Comp.ActiveHandEntity != null && handSlot.HeldEntity == null)
|
|
StartStripInsertHand(user, target, user.Comp.ActiveHandEntity.Value, handId, targetStrippable);
|
|
else if (user.Comp.ActiveHandEntity == null && handSlot.HeldEntity != null)
|
|
StartStripRemoveHand(user, target, handSlot.HeldEntity.Value, handId, targetStrippable);
|
|
}
|
|
|
|
private void OnStripEnsnareMessage(EntityUid uid, EnsnareableComponent component, StrippingEnsnareButtonPressed args)
|
|
{
|
|
if (args.Actor is not { Valid: true } user)
|
|
return;
|
|
|
|
foreach (var entity in component.Container.ContainedEntities)
|
|
{
|
|
if (!TryComp<EnsnaringComponent>(entity, out var ensnaring))
|
|
continue;
|
|
|
|
_ensnaringSystem.TryFree(uid, user, entity, ensnaring);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks whether the item is in a user's active hand and whether it can be inserted into the inventory slot.
|
|
/// </summary>
|
|
private bool CanStripInsertInventory(
|
|
Entity<HandsComponent?> user,
|
|
EntityUid target,
|
|
EntityUid held,
|
|
string slot)
|
|
{
|
|
if (!Resolve(user, ref user.Comp))
|
|
return false;
|
|
|
|
if (user.Comp.ActiveHand == null)
|
|
return false;
|
|
|
|
if (user.Comp.ActiveHandEntity == null)
|
|
return false;
|
|
|
|
if (user.Comp.ActiveHandEntity != held)
|
|
return false;
|
|
|
|
if (!_handsSystem.CanDropHeld(user, user.Comp.ActiveHand))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-cannot-drop"), user);
|
|
return false;
|
|
}
|
|
|
|
if (_inventorySystem.TryGetSlotEntity(target, slot, out _))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-item-slot-occupied", ("owner", target)), user);
|
|
return false;
|
|
}
|
|
|
|
if (!_inventorySystem.CanEquip(user, target, held, slot, out _))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-cannot-equip-message", ("owner", target)), user);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins a DoAfter to insert the item in the user's active hand into the inventory slot.
|
|
/// </summary>
|
|
private void StartStripInsertInventory(
|
|
Entity<HandsComponent?> user,
|
|
EntityUid target,
|
|
EntityUid held,
|
|
string slot)
|
|
{
|
|
if (!Resolve(user, ref user.Comp))
|
|
return;
|
|
|
|
if (!CanStripInsertInventory(user, target, held, slot))
|
|
return;
|
|
|
|
if (!_inventorySystem.TryGetSlot(target, slot, out var slotDef))
|
|
{
|
|
Log.Error($"{ToPrettyString(user)} attempted to place an item in a non-existent inventory slot ({slot}) on {ToPrettyString(target)}");
|
|
return;
|
|
}
|
|
|
|
var (time, stealth) = GetStripTimeModifiers(user, target, slotDef.StripTime);
|
|
|
|
if (!stealth)
|
|
_popupSystem.PopupEntity(Loc.GetString("strippable-component-alert-owner-insert", ("user", Identity.Entity(user, EntityManager)), ("item", user.Comp.ActiveHandEntity!.Value)), target, target, PopupType.Large);
|
|
|
|
var prefix = stealth ? "stealthily " : "";
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Low, $"{ToPrettyString(user):actor} is trying to {prefix}place the item {ToPrettyString(held):item} in {ToPrettyString(target):target}'s {slot} slot");
|
|
|
|
var doAfterArgs = new DoAfterArgs(EntityManager, user, time, new StrippableDoAfterEvent(true, true, slot), user, target, held)
|
|
{
|
|
Hidden = stealth,
|
|
AttemptFrequency = AttemptFrequency.EveryTick,
|
|
BreakOnDamage = true,
|
|
BreakOnMove = true,
|
|
NeedHand = true,
|
|
DuplicateCondition = DuplicateConditions.SameTool
|
|
};
|
|
|
|
_doAfterSystem.TryStartDoAfter(doAfterArgs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Inserts the item in the user's active hand into the inventory slot.
|
|
/// </summary>
|
|
private void StripInsertInventory(
|
|
Entity<HandsComponent?> user,
|
|
EntityUid target,
|
|
EntityUid held,
|
|
string slot)
|
|
{
|
|
if (!Resolve(user, ref user.Comp))
|
|
return;
|
|
|
|
if (!CanStripInsertInventory(user, target, held, slot))
|
|
return;
|
|
|
|
if (!_handsSystem.TryDrop(user, handsComp: user.Comp))
|
|
return;
|
|
|
|
_inventorySystem.TryEquip(user, target, held, slot);
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Medium, $"{ToPrettyString(user):actor} has placed the item {ToPrettyString(held):item} in {ToPrettyString(target):target}'s {slot} slot");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks whether the item can be removed from the target's inventory.
|
|
/// </summary>
|
|
private bool CanStripRemoveInventory(
|
|
EntityUid user,
|
|
EntityUid target,
|
|
EntityUid item,
|
|
string slot)
|
|
{
|
|
if (!_inventorySystem.TryGetSlotEntity(target, slot, out var slotItem))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-item-slot-free-message", ("owner", target)), user);
|
|
return false;
|
|
}
|
|
|
|
if (slotItem != item)
|
|
return false;
|
|
|
|
if (!_inventorySystem.CanUnequip(user, target, slot, out var reason))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString(reason), user);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins a DoAfter to remove the item from the target's inventory and insert it in the user's active hand.
|
|
/// </summary>
|
|
private void StartStripRemoveInventory(
|
|
EntityUid user,
|
|
EntityUid target,
|
|
EntityUid item,
|
|
string slot)
|
|
{
|
|
if (!CanStripRemoveInventory(user, target, item, slot))
|
|
return;
|
|
|
|
if (!_inventorySystem.TryGetSlot(target, slot, out var slotDef))
|
|
{
|
|
Log.Error($"{ToPrettyString(user)} attempted to take an item from a non-existent inventory slot ({slot}) on {ToPrettyString(target)}");
|
|
return;
|
|
}
|
|
|
|
var (time, stealth) = GetStripTimeModifiers(user, target, slotDef.StripTime);
|
|
|
|
if (!stealth)
|
|
{
|
|
if (slotDef.StripHidden)
|
|
_popupSystem.PopupEntity(Loc.GetString("strippable-component-alert-owner-hidden", ("slot", slot)), target, target, PopupType.Large);
|
|
else
|
|
_popupSystem.PopupEntity(Loc.GetString("strippable-component-alert-owner", ("user", Identity.Entity(user, EntityManager)), ("item", item)), target, target, PopupType.Large);
|
|
}
|
|
|
|
var prefix = stealth ? "stealthily " : "";
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Low, $"{ToPrettyString(user):actor} is trying to {prefix}strip the item {ToPrettyString(item):item} from {ToPrettyString(target):target}'s {slot} slot");
|
|
|
|
var doAfterArgs = new DoAfterArgs(EntityManager, user, time, new StrippableDoAfterEvent(false, true, slot), user, target, item)
|
|
{
|
|
Hidden = stealth,
|
|
AttemptFrequency = AttemptFrequency.EveryTick,
|
|
BreakOnDamage = true,
|
|
BreakOnMove = true,
|
|
NeedHand = true,
|
|
BreakOnHandChange = false, // Allow simultaneously removing multiple items.
|
|
DuplicateCondition = DuplicateConditions.SameTool
|
|
};
|
|
|
|
_doAfterSystem.TryStartDoAfter(doAfterArgs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Removes the item from the target's inventory and inserts it in the user's active hand.
|
|
/// </summary>
|
|
private void StripRemoveInventory(
|
|
EntityUid user,
|
|
EntityUid target,
|
|
EntityUid item,
|
|
string slot,
|
|
bool stealth)
|
|
{
|
|
if (!CanStripRemoveInventory(user, target, item, slot))
|
|
return;
|
|
|
|
if (!_inventorySystem.TryUnequip(user, target, slot))
|
|
return;
|
|
|
|
RaiseLocalEvent(item, new DroppedEvent(user), true); // Gas tank internals etc.
|
|
|
|
_handsSystem.PickupOrDrop(user, item, animateUser: stealth, animate: stealth);
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Medium, $"{ToPrettyString(user):actor} has stripped the item {ToPrettyString(item):item} from {ToPrettyString(target):target}'s {slot} slot");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks whether the item in the user's active hand can be inserted into one of the target's hands.
|
|
/// </summary>
|
|
private bool CanStripInsertHand(
|
|
Entity<HandsComponent?> user,
|
|
Entity<HandsComponent?> target,
|
|
EntityUid held,
|
|
string handName)
|
|
{
|
|
if (!Resolve(user, ref user.Comp) ||
|
|
!Resolve(target, ref target.Comp))
|
|
return false;
|
|
|
|
if (user.Comp.ActiveHand == null)
|
|
return false;
|
|
|
|
if (user.Comp.ActiveHandEntity == null)
|
|
return false;
|
|
|
|
if (user.Comp.ActiveHandEntity != held)
|
|
return false;
|
|
|
|
if (!_handsSystem.CanDropHeld(user, user.Comp.ActiveHand))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-cannot-drop"), user);
|
|
return false;
|
|
}
|
|
|
|
if (!_handsSystem.TryGetHand(target, handName, out var handSlot, target.Comp) ||
|
|
!_handsSystem.CanPickupToHand(target, user.Comp.ActiveHandEntity.Value, handSlot, checkActionBlocker: false, target.Comp))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-cannot-put-message", ("owner", target)), user);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins a DoAfter to insert the item in the user's active hand into one of the target's hands.
|
|
/// </summary>
|
|
private void StartStripInsertHand(
|
|
Entity<HandsComponent?> user,
|
|
Entity<HandsComponent?> target,
|
|
EntityUid held,
|
|
string handName,
|
|
StrippableComponent? targetStrippable = null)
|
|
{
|
|
if (!Resolve(user, ref user.Comp) ||
|
|
!Resolve(target, ref target.Comp) ||
|
|
!Resolve(target, ref targetStrippable))
|
|
return;
|
|
|
|
if (!CanStripInsertHand(user, target, held, handName))
|
|
return;
|
|
|
|
var (time, stealth) = GetStripTimeModifiers(user, target, targetStrippable.HandStripDelay);
|
|
|
|
var prefix = stealth ? "stealthily " : "";
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Low, $"{ToPrettyString(user):actor} is trying to {prefix}place the item {ToPrettyString(held):item} in {ToPrettyString(target):target}'s hands");
|
|
|
|
var doAfterArgs = new DoAfterArgs(EntityManager, user, time, new StrippableDoAfterEvent(true, false, handName), user, target, held)
|
|
{
|
|
Hidden = stealth,
|
|
AttemptFrequency = AttemptFrequency.EveryTick,
|
|
BreakOnDamage = true,
|
|
BreakOnMove = true,
|
|
NeedHand = true,
|
|
DuplicateCondition = DuplicateConditions.SameTool
|
|
};
|
|
|
|
_doAfterSystem.TryStartDoAfter(doAfterArgs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Places the item in the user's active hand into one of the target's hands.
|
|
/// </summary>
|
|
private void StripInsertHand(
|
|
Entity<HandsComponent?> user,
|
|
Entity<HandsComponent?> target,
|
|
EntityUid held,
|
|
string handName,
|
|
bool stealth)
|
|
{
|
|
if (!Resolve(user, ref user.Comp) ||
|
|
!Resolve(target, ref target.Comp))
|
|
return;
|
|
|
|
if (!CanStripInsertHand(user, target, held, handName))
|
|
return;
|
|
|
|
_handsSystem.TryDrop(user, checkActionBlocker: false, handsComp: user.Comp);
|
|
_handsSystem.TryPickup(target, held, handName, checkActionBlocker: false, animateUser: stealth, animate: stealth, handsComp: target.Comp);
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Medium, $"{ToPrettyString(user):actor} has placed the item {ToPrettyString(held):item} in {ToPrettyString(target):target}'s hands");
|
|
|
|
// Hand update will trigger strippable update.
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks whether the item is in the target's hand and whether it can be dropped.
|
|
/// </summary>
|
|
private bool CanStripRemoveHand(
|
|
EntityUid user,
|
|
Entity<HandsComponent?> target,
|
|
EntityUid item,
|
|
string handName)
|
|
{
|
|
if (!Resolve(target, ref target.Comp))
|
|
return false;
|
|
|
|
if (!_handsSystem.TryGetHand(target, handName, out var handSlot, target.Comp))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-item-slot-free-message", ("owner", Identity.Name(target, EntityManager, user))), user);
|
|
return false;
|
|
}
|
|
|
|
if (HasComp<VirtualItemComponent>(handSlot.HeldEntity))
|
|
return false;
|
|
|
|
if (handSlot.HeldEntity == null)
|
|
return false;
|
|
|
|
if (handSlot.HeldEntity != item)
|
|
return false;
|
|
|
|
if (!_handsSystem.CanDropHeld(target, handSlot, false))
|
|
{
|
|
_popupSystem.PopupCursor(Loc.GetString("strippable-component-cannot-drop-message", ("owner", Identity.Name(target, EntityManager, user))), user);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins a DoAfter to remove the item from the target's hand and insert it in the user's active hand.
|
|
/// </summary>
|
|
private void StartStripRemoveHand(
|
|
Entity<HandsComponent?> user,
|
|
Entity<HandsComponent?> target,
|
|
EntityUid item,
|
|
string handName,
|
|
StrippableComponent? targetStrippable = null)
|
|
{
|
|
if (!Resolve(user, ref user.Comp) ||
|
|
!Resolve(target, ref target.Comp) ||
|
|
!Resolve(target, ref targetStrippable))
|
|
return;
|
|
|
|
if (!CanStripRemoveHand(user, target, item, handName))
|
|
return;
|
|
|
|
var (time, stealth) = GetStripTimeModifiers(user, target, targetStrippable.HandStripDelay);
|
|
|
|
if (!stealth)
|
|
_popupSystem.PopupEntity(Loc.GetString("strippable-component-alert-owner", ("user", Identity.Entity(user, EntityManager)), ("item", item)), target, target);
|
|
|
|
var prefix = stealth ? "stealthily " : "";
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Low, $"{ToPrettyString(user):actor} is trying to {prefix}strip the item {ToPrettyString(item):item} from {ToPrettyString(target):target}'s hands");
|
|
|
|
var doAfterArgs = new DoAfterArgs(EntityManager, user, time, new StrippableDoAfterEvent(false, false, handName), user, target, item)
|
|
{
|
|
Hidden = stealth,
|
|
AttemptFrequency = AttemptFrequency.EveryTick,
|
|
BreakOnDamage = true,
|
|
BreakOnMove = true,
|
|
NeedHand = true,
|
|
BreakOnHandChange = false, // Allow simultaneously removing multiple items.
|
|
DuplicateCondition = DuplicateConditions.SameTool
|
|
};
|
|
|
|
_doAfterSystem.TryStartDoAfter(doAfterArgs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Takes the item from the target's hand and inserts it in the user's active hand.
|
|
/// </summary>
|
|
private void StripRemoveHand(
|
|
Entity<HandsComponent?> user,
|
|
Entity<HandsComponent?> target,
|
|
EntityUid item,
|
|
string handName,
|
|
bool stealth)
|
|
{
|
|
if (!Resolve(user, ref user.Comp) ||
|
|
!Resolve(target, ref target.Comp))
|
|
return;
|
|
|
|
if (!CanStripRemoveHand(user, target, item, handName))
|
|
return;
|
|
|
|
_handsSystem.TryDrop(target, item, checkActionBlocker: false, handsComp: target.Comp);
|
|
_handsSystem.PickupOrDrop(user, item, animateUser: stealth, animate: stealth, handsComp: user.Comp);
|
|
_adminLogger.Add(LogType.Stripping, LogImpact.Medium, $"{ToPrettyString(user):actor} has stripped the item {ToPrettyString(item):item} from {ToPrettyString(target):target}'s hands");
|
|
|
|
// Hand update will trigger strippable update.
|
|
}
|
|
|
|
private void OnStrippableDoAfterRunning(Entity<HandsComponent> entity, ref DoAfterAttemptEvent<StrippableDoAfterEvent> ev)
|
|
{
|
|
var args = ev.DoAfter.Args;
|
|
|
|
DebugTools.Assert(entity.Owner == args.User);
|
|
DebugTools.Assert(args.Target != null);
|
|
DebugTools.Assert(args.Used != null);
|
|
DebugTools.Assert(ev.Event.SlotOrHandName != null);
|
|
|
|
if (ev.Event.InventoryOrHand)
|
|
{
|
|
if (ev.Event.InsertOrRemove && !CanStripInsertInventory((entity.Owner, entity.Comp), args.Target.Value, args.Used.Value, ev.Event.SlotOrHandName) ||
|
|
!ev.Event.InsertOrRemove && !CanStripRemoveInventory(entity.Owner, args.Target.Value, args.Used.Value, ev.Event.SlotOrHandName))
|
|
{
|
|
ev.Cancel();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (ev.Event.InsertOrRemove && !CanStripInsertHand((entity.Owner, entity.Comp), args.Target.Value, args.Used.Value, ev.Event.SlotOrHandName) ||
|
|
!ev.Event.InsertOrRemove && !CanStripRemoveHand(entity.Owner, args.Target.Value, args.Used.Value, ev.Event.SlotOrHandName))
|
|
{
|
|
ev.Cancel();
|
|
}
|
|
}
|
|
}
|
|
|
|
private void OnStrippableDoAfterFinished(Entity<HandsComponent> entity, ref StrippableDoAfterEvent ev)
|
|
{
|
|
if (ev.Cancelled)
|
|
return;
|
|
|
|
DebugTools.Assert(entity.Owner == ev.User);
|
|
DebugTools.Assert(ev.Target != null);
|
|
DebugTools.Assert(ev.Used != null);
|
|
DebugTools.Assert(ev.SlotOrHandName != null);
|
|
|
|
if (ev.InventoryOrHand)
|
|
{
|
|
if (ev.InsertOrRemove)
|
|
StripInsertInventory((entity.Owner, entity.Comp), ev.Target.Value, ev.Used.Value, ev.SlotOrHandName);
|
|
else StripRemoveInventory(entity.Owner, ev.Target.Value, ev.Used.Value, ev.SlotOrHandName, ev.Args.Hidden);
|
|
}
|
|
else
|
|
{
|
|
if (ev.InsertOrRemove)
|
|
StripInsertHand((entity.Owner, entity.Comp), ev.Target.Value, ev.Used.Value, ev.SlotOrHandName, ev.Args.Hidden);
|
|
else StripRemoveHand((entity.Owner, entity.Comp), ev.Target.Value, ev.Used.Value, ev.SlotOrHandName, ev.Args.Hidden);
|
|
}
|
|
}
|
|
}
|
|
}
|