Fix clicking a locker while inside of said locker (#3965)
* Fix clicking a locker while inside of said locker Actually fixes #1535 * Address reviews * Add test * Clean up test
This commit is contained in:
@@ -0,0 +1,133 @@
|
||||
using Content.Server.GameObjects.Components.GUI;
|
||||
using Content.Server.GameObjects.Components.Items.Storage;
|
||||
using Content.Server.GameObjects.EntitySystems.Click;
|
||||
using Content.Shared.Interfaces.GameObjects.Components;
|
||||
using NUnit.Framework;
|
||||
using Robust.Shared.Containers;
|
||||
using Robust.Shared.GameObjects;
|
||||
using Robust.Shared.IoC;
|
||||
using Robust.Shared.Map;
|
||||
using Robust.Shared.Maths;
|
||||
using Robust.Shared.Reflection;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Content.IntegrationTests.Tests.Interaction.Click
|
||||
{
|
||||
[TestFixture]
|
||||
[TestOf(typeof(InteractionSystem))]
|
||||
public class InteractionSystemTests : ContentIntegrationTest
|
||||
{
|
||||
[Reflect(false)]
|
||||
private class TestAttackEntitySystem : EntitySystem
|
||||
{
|
||||
public EntityEventHandler<AttackEventArgs> AttackEvent;
|
||||
public EntityEventHandler<InteractUsingMessage> InteractUsingEvent;
|
||||
public EntityEventHandler<AttackHandMessage> InteractHandEvent;
|
||||
|
||||
public override void Initialize()
|
||||
{
|
||||
base.Initialize();
|
||||
SubscribeLocalEvent<AttackEventArgs>((e) => AttackEvent?.Invoke(e));
|
||||
SubscribeLocalEvent<InteractUsingMessage>((e) => InteractUsingEvent?.Invoke(e));
|
||||
SubscribeLocalEvent<AttackHandMessage>((e) => InteractHandEvent?.Invoke(e));
|
||||
}
|
||||
|
||||
public override void Shutdown()
|
||||
{
|
||||
base.Shutdown();
|
||||
UnsubscribeLocalEvent<AttackEventArgs>();
|
||||
UnsubscribeLocalEvent<InteractUsingMessage>();
|
||||
UnsubscribeLocalEvent<AttackHandMessage>();
|
||||
}
|
||||
}
|
||||
|
||||
[Test]
|
||||
public async Task InsideContainerInteractionBlockTest()
|
||||
{
|
||||
var server = StartServerDummyTicker(new ServerContentIntegrationOption
|
||||
{
|
||||
ContentBeforeIoC = () =>
|
||||
{
|
||||
IoCManager.Resolve<IEntitySystemManager>().LoadExtraSystemType<TestAttackEntitySystem>();
|
||||
}
|
||||
});
|
||||
|
||||
await server.WaitIdleAsync();
|
||||
|
||||
var entityManager = server.ResolveDependency<IEntityManager>();
|
||||
var mapManager = server.ResolveDependency<IMapManager>();
|
||||
|
||||
IEntity origin = null;
|
||||
IEntity other = null;
|
||||
IEntity containerEntity = null;
|
||||
IContainer container = null;
|
||||
|
||||
server.Assert(() =>
|
||||
{
|
||||
var mapId = mapManager.CreateMap();
|
||||
var coordinates = new MapCoordinates(Vector2.Zero, mapId);
|
||||
|
||||
origin = entityManager.SpawnEntity(null, coordinates);
|
||||
origin.EnsureComponent<HandsComponent>();
|
||||
other = entityManager.SpawnEntity(null, coordinates);
|
||||
containerEntity = entityManager.SpawnEntity(null, coordinates);
|
||||
container = ContainerHelpers.EnsureContainer<Container>(containerEntity, "InteractionTestContainer");
|
||||
});
|
||||
|
||||
await server.WaitIdleAsync();
|
||||
|
||||
var attack = false;
|
||||
var interactUsing = false;
|
||||
var interactHand = false;
|
||||
server.Assert(() =>
|
||||
{
|
||||
Assert.That(container.Insert(origin));
|
||||
Assert.That(origin.Transform.Parent!.Owner, Is.EqualTo(containerEntity));
|
||||
|
||||
var entitySystemManager = IoCManager.Resolve<IEntitySystemManager>();
|
||||
Assert.That(entitySystemManager.TryGetEntitySystem<InteractionSystem>(out var interactionSystem));
|
||||
|
||||
Assert.That(entitySystemManager.TryGetEntitySystem<TestAttackEntitySystem>(out var testAttackEntitySystem));
|
||||
testAttackEntitySystem.AttackEvent = (ev) =>
|
||||
{
|
||||
Assert.That(ev.Target, Is.EqualTo(containerEntity.Uid));
|
||||
attack = true;
|
||||
};
|
||||
testAttackEntitySystem.InteractUsingEvent = (ev) =>
|
||||
{
|
||||
Assert.That(ev.Attacked, Is.EqualTo(containerEntity));
|
||||
interactUsing = true;
|
||||
};
|
||||
testAttackEntitySystem.InteractHandEvent = (ev) =>
|
||||
{
|
||||
Assert.That(ev.Attacked, Is.EqualTo(containerEntity));
|
||||
interactHand = true;
|
||||
};
|
||||
|
||||
interactionSystem.DoAttack(origin, other.Transform.Coordinates, false, other.Uid);
|
||||
interactionSystem.UserInteraction(origin, other.Transform.Coordinates, other.Uid);
|
||||
Assert.That(attack, Is.False);
|
||||
Assert.That(interactUsing, Is.False);
|
||||
Assert.That(interactHand, Is.False);
|
||||
|
||||
interactionSystem.DoAttack(origin, containerEntity.Transform.Coordinates, false, containerEntity.Uid);
|
||||
interactionSystem.UserInteraction(origin, containerEntity.Transform.Coordinates, containerEntity.Uid);
|
||||
Assert.That(attack);
|
||||
Assert.That(interactUsing, Is.False);
|
||||
Assert.That(interactHand);
|
||||
|
||||
var itemEntity = entityManager.SpawnEntity(null, origin.Transform.Coordinates);
|
||||
var item = itemEntity.EnsureComponent<ItemComponent>();
|
||||
|
||||
Assert.That(origin.TryGetComponent<HandsComponent>(out var hands));
|
||||
hands.PutInHand(item);
|
||||
|
||||
interactionSystem.UserInteraction(origin, other.Transform.Coordinates, other.Uid);
|
||||
Assert.That(interactUsing, Is.False);
|
||||
|
||||
interactionSystem.UserInteraction(origin, containerEntity.Transform.Coordinates, containerEntity.Uid);
|
||||
Assert.That(interactUsing);
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -287,7 +287,7 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
||||
return pull.TogglePull(player);
|
||||
}
|
||||
|
||||
private async void UserInteraction(IEntity player, EntityCoordinates coordinates, EntityUid clickedUid)
|
||||
public async void UserInteraction(IEntity player, EntityCoordinates coordinates, EntityUid clickedUid)
|
||||
{
|
||||
// Get entity clicked upon from UID if valid UID, if not assume no entity clicked upon and null
|
||||
if (!EntityManager.TryGetEntity(clickedUid, out var attacked))
|
||||
@@ -324,13 +324,6 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
||||
return;
|
||||
}
|
||||
|
||||
// If in a container
|
||||
if (player.IsInContainer())
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
// In a container where the attacked entity is not the container's owner
|
||||
if (player.TryGetContainer(out var playerContainer) &&
|
||||
attacked != playerContainer.Owner)
|
||||
@@ -789,7 +782,7 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
||||
}
|
||||
}
|
||||
|
||||
private void DoAttack(IEntity player, EntityCoordinates coordinates, bool wideAttack, EntityUid target = default)
|
||||
public void DoAttack(IEntity player, EntityCoordinates coordinates, bool wideAttack, EntityUid targetUid = default)
|
||||
{
|
||||
// Verify player is on the same map as the entity he clicked on
|
||||
if (coordinates.GetMapId(EntityManager) != player.Transform.MapID)
|
||||
@@ -807,7 +800,22 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
||||
return;
|
||||
}
|
||||
|
||||
var eventArgs = new AttackEventArgs(player, coordinates, wideAttack, target);
|
||||
|
||||
// In a container where the target entity is not the container's owner
|
||||
if (player.TryGetContainer(out var playerContainer) &&
|
||||
(!EntityManager.TryGetEntity(targetUid, out var target) ||
|
||||
target != playerContainer.Owner))
|
||||
{
|
||||
// Either the target entity is null, not contained or in a different container
|
||||
if (target == null ||
|
||||
!target.TryGetContainer(out var attackedContainer) ||
|
||||
attackedContainer != playerContainer)
|
||||
{
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
var eventArgs = new AttackEventArgs(player, coordinates, wideAttack, targetUid);
|
||||
|
||||
// Verify player has a hand, and find what object he is currently holding in his active hand
|
||||
if (player.TryGetComponent<IHandsComponent>(out var hands))
|
||||
@@ -826,7 +834,7 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
||||
else
|
||||
{
|
||||
// We pick up items if our hand is empty, even if we're in combat mode.
|
||||
if(EntityManager.TryGetEntity(target, out var targetEnt))
|
||||
if (EntityManager.TryGetEntity(targetUid, out var targetEnt))
|
||||
{
|
||||
if (targetEnt.HasComponent<ItemComponent>())
|
||||
{
|
||||
|
||||
Reference in New Issue
Block a user