Add client pulling prediction (#2041)
* WIP changes * Merge conflict fixes * Bring pull controlelr to current year * Sync and predict PullController on the client * Clean imports * Slow down pullers and make pulling tighter * Stop pulls on pullable or puller component removals * Make pulling not occur when moving towards the pulled entity
This commit is contained in:
@@ -92,6 +92,23 @@ namespace Content.Client.GameObjects.Components.Mobs
|
|||||||
_cooldown.Clear();
|
_cooldown.Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public override void ChangeStatusEffectIcon(StatusEffect effect, string icon)
|
||||||
|
{
|
||||||
|
if (_status.TryGetValue(effect, out var value) &&
|
||||||
|
value.Icon == icon)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_status[effect] = new StatusEffectStatus
|
||||||
|
{
|
||||||
|
Icon = icon,
|
||||||
|
Cooldown = value.Cooldown
|
||||||
|
};
|
||||||
|
|
||||||
|
Dirty();
|
||||||
|
}
|
||||||
|
|
||||||
public void UpdateStatusEffects()
|
public void UpdateStatusEffects()
|
||||||
{
|
{
|
||||||
if (!CurrentlyControlled || _ui == null)
|
if (!CurrentlyControlled || _ui == null)
|
||||||
@@ -132,10 +149,15 @@ namespace Content.Client.GameObjects.Components.Mobs
|
|||||||
SendNetworkMessage(new ClickStatusMessage(status.Effect));
|
SendNetworkMessage(new ClickStatusMessage(status.Effect));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void RemoveStatusEffect(StatusEffect name)
|
public override void RemoveStatusEffect(StatusEffect effect)
|
||||||
{
|
{
|
||||||
_status.Remove(name);
|
if (!_status.Remove(effect))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
UpdateStatusEffects();
|
UpdateStatusEffects();
|
||||||
|
Dirty();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void FrameUpdate(float frameTime)
|
public void FrameUpdate(float frameTime)
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
using Content.Shared.GameObjects.Components.Pulling;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
|
|
||||||
|
namespace Content.Client.GameObjects.Components.Pulling
|
||||||
|
{
|
||||||
|
[RegisterComponent]
|
||||||
|
[ComponentReference(typeof(SharedPullableComponent))]
|
||||||
|
public class PullableComponent : SharedPullableComponent
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -124,7 +124,6 @@
|
|||||||
"RCD",
|
"RCD",
|
||||||
"RCDDeconstructWhitelist",
|
"RCDDeconstructWhitelist",
|
||||||
"RCDAmmo",
|
"RCDAmmo",
|
||||||
"Pullable",
|
|
||||||
"CursedEntityStorage",
|
"CursedEntityStorage",
|
||||||
"Listening",
|
"Listening",
|
||||||
"Radio",
|
"Radio",
|
||||||
|
|||||||
@@ -5,10 +5,8 @@ using System.Diagnostics.CodeAnalysis;
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
using Content.Server.GameObjects.Components.Items.Storage;
|
using Content.Server.GameObjects.Components.Items.Storage;
|
||||||
using Content.Server.GameObjects.Components.Mobs;
|
using Content.Server.GameObjects.Components.Mobs;
|
||||||
using Content.Server.GameObjects.Components.Movement;
|
|
||||||
using Content.Server.GameObjects.EntitySystems.Click;
|
using Content.Server.GameObjects.EntitySystems.Click;
|
||||||
using Content.Server.Interfaces.GameObjects.Components.Items;
|
using Content.Server.Interfaces.GameObjects.Components.Items;
|
||||||
using Content.Shared.GameObjects.Components.Body;
|
|
||||||
using Content.Shared.GameObjects.Components.Body.Part;
|
using Content.Shared.GameObjects.Components.Body.Part;
|
||||||
using Content.Shared.GameObjects.Components.Items;
|
using Content.Shared.GameObjects.Components.Items;
|
||||||
using Content.Shared.GameObjects.Components.Mobs;
|
using Content.Shared.GameObjects.Components.Mobs;
|
||||||
@@ -25,10 +23,11 @@ using Robust.Shared.Interfaces.GameObjects;
|
|||||||
using Robust.Shared.Interfaces.Network;
|
using Robust.Shared.Interfaces.Network;
|
||||||
using Robust.Shared.IoC;
|
using Robust.Shared.IoC;
|
||||||
using Robust.Shared.Log;
|
using Robust.Shared.Log;
|
||||||
using Robust.Shared.Map;
|
|
||||||
using Robust.Shared.Maths;
|
using Robust.Shared.Maths;
|
||||||
using Robust.Shared.Players;
|
using Robust.Shared.Players;
|
||||||
using Robust.Shared.ViewVariables;
|
using Robust.Shared.ViewVariables;
|
||||||
|
using Content.Server.GameObjects.Components.Pulling;
|
||||||
|
using Robust.Shared.Map;
|
||||||
|
|
||||||
namespace Content.Server.GameObjects.Components.GUI
|
namespace Content.Server.GameObjects.Components.GUI
|
||||||
{
|
{
|
||||||
@@ -516,77 +515,6 @@ namespace Content.Server.GameObjects.Components.GUI
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void StartPull(PullableComponent pullable)
|
|
||||||
{
|
|
||||||
if (Owner == pullable.Owner)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!Owner.IsInSameOrNoContainer(pullable.Owner))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (IsPulling)
|
|
||||||
{
|
|
||||||
StopPull();
|
|
||||||
}
|
|
||||||
|
|
||||||
PulledObject = pullable.Owner.GetComponent<IPhysicsComponent>();
|
|
||||||
var controller = PulledObject.EnsureController<PullController>();
|
|
||||||
controller.StartPull(Owner.GetComponent<IPhysicsComponent>());
|
|
||||||
}
|
|
||||||
|
|
||||||
public void MovePulledObject(EntityCoordinates puller, EntityCoordinates to)
|
|
||||||
{
|
|
||||||
if (PulledObject != null &&
|
|
||||||
PulledObject.TryGetController(out PullController controller))
|
|
||||||
{
|
|
||||||
controller.TryMoveTo(puller, to);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void MoveEvent(MoveEvent moveEvent)
|
|
||||||
{
|
|
||||||
if (moveEvent.Sender != Owner)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!IsPulling)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
PulledObject!.WakeBody();
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void HandleMessage(ComponentMessage message, IComponent? component)
|
|
||||||
{
|
|
||||||
base.HandleMessage(message, component);
|
|
||||||
|
|
||||||
if (!(message is PullMessage pullMessage) ||
|
|
||||||
pullMessage.Puller.Owner != Owner)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (message)
|
|
||||||
{
|
|
||||||
case PullStartedMessage msg:
|
|
||||||
Owner.EntityManager.EventBus.SubscribeEvent<MoveEvent>(EventSource.Local, this, MoveEvent);
|
|
||||||
|
|
||||||
AddPullingStatuses(msg.Pulled.Owner);
|
|
||||||
break;
|
|
||||||
case PullStoppedMessage msg:
|
|
||||||
Owner.EntityManager.EventBus.UnsubscribeEvent<MoveEvent>(EventSource.Local, this);
|
|
||||||
|
|
||||||
RemovePullingStatuses(msg.Pulled.Owner);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void HandleNetworkMessage(ComponentMessage message, INetChannel channel, ICommonSession? session = null)
|
public override void HandleNetworkMessage(ComponentMessage message, INetChannel channel, ICommonSession? session = null)
|
||||||
{
|
{
|
||||||
base.HandleNetworkMessage(message, channel, session);
|
base.HandleNetworkMessage(message, channel, session);
|
||||||
|
|||||||
@@ -1,38 +1,15 @@
|
|||||||
#nullable enable
|
#nullable enable
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Content.Server.GameObjects.Components.Body;
|
|
||||||
using Content.Server.GameObjects.Components.Chemistry;
|
|
||||||
using Content.Server.GameObjects.Components.GUI;
|
|
||||||
using Content.Server.GameObjects.Components.Items.Storage;
|
|
||||||
using Content.Server.GameObjects.Components.Power.ApcNetComponents;
|
|
||||||
using Content.Server.GameObjects.EntitySystems;
|
|
||||||
using Content.Server.Interfaces.Chat;
|
using Content.Server.Interfaces.Chat;
|
||||||
using Content.Server.Interfaces.GameObjects;
|
using Content.Server.Interfaces.GameObjects;
|
||||||
using Content.Server.Utility;
|
using Content.Server.Utility;
|
||||||
using Content.Shared.Chemistry;
|
using Content.Shared.GameObjects.EntitySystems;
|
||||||
using Content.Shared.GameObjects.Components.Body;
|
|
||||||
using Content.Shared.GameObjects.Components.Power;
|
|
||||||
using Content.Shared.Interfaces;
|
using Content.Shared.Interfaces;
|
||||||
using Content.Shared.Interfaces.GameObjects.Components;
|
using Content.Shared.Interfaces.GameObjects.Components;
|
||||||
using Content.Shared.Kitchen;
|
|
||||||
using Content.Shared.Prototypes.Kitchen;
|
|
||||||
using Robust.Server.GameObjects;
|
using Robust.Server.GameObjects;
|
||||||
using Robust.Server.GameObjects.Components.Container;
|
|
||||||
using Robust.Server.GameObjects.Components.UserInterface;
|
|
||||||
using Robust.Server.GameObjects.EntitySystems;
|
|
||||||
using Robust.Server.Interfaces.GameObjects;
|
|
||||||
using Robust.Shared.Audio;
|
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.GameObjects.Systems;
|
using Robust.Shared.GameObjects.Systems;
|
||||||
using Robust.Shared.Interfaces.GameObjects;
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
using Robust.Shared.IoC;
|
|
||||||
using Robust.Shared.Localization;
|
using Robust.Shared.Localization;
|
||||||
using Robust.Shared.Serialization;
|
|
||||||
using Robust.Shared.Timers;
|
|
||||||
using Robust.Shared.ViewVariables;
|
|
||||||
|
|
||||||
namespace Content.Server.GameObjects.Components.Kitchen
|
namespace Content.Server.GameObjects.Components.Kitchen
|
||||||
{
|
{
|
||||||
@@ -49,11 +26,9 @@ namespace Content.Server.GameObjects.Components.Kitchen
|
|||||||
|
|
||||||
void IActivate.Activate(ActivateEventArgs eventArgs)
|
void IActivate.Activate(ActivateEventArgs eventArgs)
|
||||||
{
|
{
|
||||||
var victim = eventArgs.User.GetComponent<HandsComponent>().PulledObject?.Owner;
|
SpriteComponent? sprite;
|
||||||
|
|
||||||
var sprite = Owner.GetComponent<SpriteComponent>();
|
if (!EntitySystem.Get<SharedPullingSystem>().TryGetPulled(eventArgs.User, out var victim))
|
||||||
|
|
||||||
if (victim == null)
|
|
||||||
{
|
{
|
||||||
if (_meatParts == 0)
|
if (_meatParts == 0)
|
||||||
{
|
{
|
||||||
@@ -72,12 +47,18 @@ namespace Content.Server.GameObjects.Components.Kitchen
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
sprite.LayerSetState(0, "spike");
|
if (Owner.TryGetComponent(out sprite))
|
||||||
|
{
|
||||||
|
sprite.LayerSetState(0, "spike");
|
||||||
|
}
|
||||||
|
|
||||||
eventArgs.User.PopupMessage(_meatSource0);
|
eventArgs.User.PopupMessage(_meatSource0);
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
else if (_meatParts > 0)
|
|
||||||
|
if (_meatParts > 0)
|
||||||
{
|
{
|
||||||
Owner.PopupMessage(eventArgs.User, Loc.GetString("The spike already has something on it, finish collecting its meat first!"));
|
Owner.PopupMessage(eventArgs.User, Loc.GetString("The spike already has something on it, finish collecting its meat first!"));
|
||||||
return;
|
return;
|
||||||
@@ -94,7 +75,10 @@ namespace Content.Server.GameObjects.Components.Kitchen
|
|||||||
_meatSource1p = Loc.GetString("You remove some meat from {0:theName}.", victim);
|
_meatSource1p = Loc.GetString("You remove some meat from {0:theName}.", victim);
|
||||||
_meatSource0 = Loc.GetString("You remove the last piece of meat from {0:theName}!", victim);
|
_meatSource0 = Loc.GetString("You remove the last piece of meat from {0:theName}!", victim);
|
||||||
|
|
||||||
sprite.LayerSetState(0, "spikebloody");
|
if (Owner.TryGetComponent(out sprite))
|
||||||
|
{
|
||||||
|
sprite.LayerSetState(0, "spikebloody");
|
||||||
|
}
|
||||||
|
|
||||||
Owner.PopupMessageEveryone(Loc.GetString("{0:theName} has forced {1:theName} onto the spike, killing them instantly!", eventArgs.User, victim));
|
Owner.PopupMessageEveryone(Loc.GetString("{0:theName} has forced {1:theName} onto the spike, killing them instantly!", eventArgs.User, victim));
|
||||||
victim.Delete();
|
victim.Delete();
|
||||||
|
|||||||
@@ -5,8 +5,11 @@ using Content.Server.GameObjects.Components.Buckle;
|
|||||||
using Content.Server.GameObjects.Components.GUI;
|
using Content.Server.GameObjects.Components.GUI;
|
||||||
using Content.Server.GameObjects.Components.Movement;
|
using Content.Server.GameObjects.Components.Movement;
|
||||||
using Content.Shared.GameObjects.Components.Mobs;
|
using Content.Shared.GameObjects.Components.Mobs;
|
||||||
|
using Content.Shared.GameObjects.Components.Pulling;
|
||||||
|
using Content.Shared.GameObjects.EntitySystems;
|
||||||
using Content.Shared.Interfaces;
|
using Content.Shared.Interfaces;
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
|
using Robust.Shared.GameObjects.Systems;
|
||||||
using Robust.Shared.Interfaces.Network;
|
using Robust.Shared.Interfaces.Network;
|
||||||
using Robust.Shared.Players;
|
using Robust.Shared.Players;
|
||||||
using Robust.Shared.ViewVariables;
|
using Robust.Shared.ViewVariables;
|
||||||
@@ -25,7 +28,7 @@ namespace Content.Server.GameObjects.Components.Mobs
|
|||||||
return new StatusEffectComponentState(_statusEffects);
|
return new StatusEffectComponentState(_statusEffects);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void ChangeStatusEffectIcon(StatusEffect effect, string icon)
|
public override void ChangeStatusEffectIcon(StatusEffect effect, string icon)
|
||||||
{
|
{
|
||||||
if (_statusEffects.TryGetValue(effect, out var value) && value.Icon == icon)
|
if (_statusEffects.TryGetValue(effect, out var value) && value.Icon == icon)
|
||||||
{
|
{
|
||||||
@@ -60,7 +63,7 @@ namespace Content.Server.GameObjects.Components.Mobs
|
|||||||
Dirty();
|
Dirty();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void RemoveStatusEffect(StatusEffect effect)
|
public override void RemoveStatusEffect(StatusEffect effect)
|
||||||
{
|
{
|
||||||
if (!_statusEffects.Remove(effect))
|
if (!_statusEffects.Remove(effect))
|
||||||
{
|
{
|
||||||
@@ -106,19 +109,19 @@ namespace Content.Server.GameObjects.Components.Mobs
|
|||||||
controller.RemoveController();
|
controller.RemoveController();
|
||||||
break;
|
break;
|
||||||
case StatusEffect.Pulling:
|
case StatusEffect.Pulling:
|
||||||
if (!player.TryGetComponent(out HandsComponent hands))
|
EntitySystem
|
||||||
break;
|
.Get<SharedPullingSystem>()
|
||||||
|
.GetPulled(player)?
|
||||||
|
.GetComponentOrNull<SharedPullableComponent>()?
|
||||||
|
.TryStopPull();
|
||||||
|
|
||||||
hands.StopPull();
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case StatusEffect.Fire:
|
case StatusEffect.Fire:
|
||||||
if (!player.TryGetComponent(out FlammableComponent flammable))
|
if (!player.TryGetComponent(out FlammableComponent flammable))
|
||||||
break;
|
break;
|
||||||
|
|
||||||
flammable.Resist();
|
flammable.Resist();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
player.PopupMessage(msg.Effect.ToString());
|
player.PopupMessage(msg.Effect.ToString());
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -1,10 +0,0 @@
|
|||||||
using Robust.Shared.GameObjects;
|
|
||||||
|
|
||||||
namespace Content.Server.GameObjects.Components.Movement
|
|
||||||
{
|
|
||||||
[RegisterComponent]
|
|
||||||
public class PullableComponent: Component
|
|
||||||
{
|
|
||||||
public override string Name => "Pullable";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Content.Shared.GameObjects.Components.Items;
|
||||||
|
using Content.Shared.GameObjects.Components.Pulling;
|
||||||
|
using Content.Shared.GameObjects.EntitySystems;
|
||||||
|
using Content.Shared.GameObjects.Verbs;
|
||||||
|
using Content.Shared.Physics.Pull;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
|
using Robust.Shared.GameObjects.Components;
|
||||||
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
|
using Robust.Shared.Localization;
|
||||||
|
|
||||||
|
namespace Content.Server.GameObjects.Components.Pulling
|
||||||
|
{
|
||||||
|
[RegisterComponent]
|
||||||
|
[ComponentReference(typeof(SharedPullableComponent))]
|
||||||
|
public class PullableComponent : SharedPullableComponent
|
||||||
|
{
|
||||||
|
[Verb]
|
||||||
|
public class PullingVerb : Verb<PullableComponent>
|
||||||
|
{
|
||||||
|
protected override void GetData(IEntity user, PullableComponent component, VerbData data)
|
||||||
|
{
|
||||||
|
data.Visibility = VerbVisibility.Invisible;
|
||||||
|
|
||||||
|
if (user == component.Owner)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!user.Transform.Coordinates.TryDistance(user.EntityManager, component.Owner.Transform.Coordinates, out var distance) ||
|
||||||
|
distance > SharedInteractionSystem.InteractionRange)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!user.HasComponent<ISharedHandsComponent>() ||
|
||||||
|
!user.TryGetComponent(out IPhysicsComponent? userPhysics) ||
|
||||||
|
!component.Owner.TryGetComponent(out IPhysicsComponent? targetPhysics) ||
|
||||||
|
targetPhysics.Anchored)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var controller = targetPhysics.EnsureController<PullController>();
|
||||||
|
|
||||||
|
data.Visibility = VerbVisibility.Visible;
|
||||||
|
data.Text = controller.Puller == userPhysics
|
||||||
|
? Loc.GetString("Stop pulling")
|
||||||
|
: Loc.GetString("Pull");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Activate(IEntity user, PullableComponent component)
|
||||||
|
{
|
||||||
|
if (!user.TryGetComponent(out IPhysicsComponent? userCollidable) ||
|
||||||
|
!component.Owner.TryGetComponent(out IPhysicsComponent? targetCollidable) ||
|
||||||
|
targetCollidable.Anchored)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var controller = targetCollidable.EnsureController<PullController>();
|
||||||
|
|
||||||
|
if (controller.Puller == userCollidable)
|
||||||
|
{
|
||||||
|
component.TryStopPull();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
component.TryStartPull(user);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,6 +5,7 @@ using Content.Server.GameObjects.Components.GUI;
|
|||||||
using Content.Server.GameObjects.Components.Items.Storage;
|
using Content.Server.GameObjects.Components.Items.Storage;
|
||||||
using Content.Server.GameObjects.Components.Mobs;
|
using Content.Server.GameObjects.Components.Mobs;
|
||||||
using Content.Server.GameObjects.Components.Movement;
|
using Content.Server.GameObjects.Components.Movement;
|
||||||
|
using Content.Server.GameObjects.Components.Pulling;
|
||||||
using Content.Server.GameObjects.Components.Timing;
|
using Content.Server.GameObjects.Components.Timing;
|
||||||
using Content.Server.Interfaces.GameObjects.Components.Items;
|
using Content.Server.Interfaces.GameObjects.Components.Items;
|
||||||
using Content.Shared.GameObjects.Components.Inventory;
|
using Content.Shared.GameObjects.Components.Inventory;
|
||||||
@@ -273,12 +274,7 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pulledObject.TryGetComponent<PullableComponent>(out var pull))
|
if (!pulledObject.TryGetComponent(out PullableComponent pull))
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!player.TryGetComponent<HandsComponent>(out var hands))
|
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -289,24 +285,7 @@ namespace Content.Server.GameObjects.EntitySystems.Click
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pull.Owner.TryGetComponent(out IPhysicsComponent physics) ||
|
return pull.TogglePull(player);
|
||||||
physics.Anchored)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
var controller = physics.EnsureController<PullController>();
|
|
||||||
|
|
||||||
if (controller.GettingPulled)
|
|
||||||
{
|
|
||||||
hands.StopPull();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
hands.StartPull(pull);
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void UserInteraction(IEntity player, EntityCoordinates coordinates, EntityUid clickedUid)
|
private void UserInteraction(IEntity player, EntityCoordinates coordinates, EntityUid clickedUid)
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ using Robust.Shared.GameObjects;
|
|||||||
using Robust.Shared.GameObjects.Systems;
|
using Robust.Shared.GameObjects.Systems;
|
||||||
using Robust.Shared.Input.Binding;
|
using Robust.Shared.Input.Binding;
|
||||||
using Robust.Shared.Interfaces.GameObjects;
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
using Robust.Shared.Interfaces.Map;
|
|
||||||
using Robust.Shared.IoC;
|
using Robust.Shared.IoC;
|
||||||
using Robust.Shared.Localization;
|
using Robust.Shared.Localization;
|
||||||
using Robust.Shared.Map;
|
using Robust.Shared.Map;
|
||||||
@@ -47,8 +46,6 @@ namespace Content.Server.GameObjects.EntitySystems
|
|||||||
.Bind(ContentKeyFunctions.ThrowItemInHand, new PointerInputCmdHandler(HandleThrowItem))
|
.Bind(ContentKeyFunctions.ThrowItemInHand, new PointerInputCmdHandler(HandleThrowItem))
|
||||||
.Bind(ContentKeyFunctions.SmartEquipBackpack, InputCmdHandler.FromDelegate(HandleSmartEquipBackpack))
|
.Bind(ContentKeyFunctions.SmartEquipBackpack, InputCmdHandler.FromDelegate(HandleSmartEquipBackpack))
|
||||||
.Bind(ContentKeyFunctions.SmartEquipBelt, InputCmdHandler.FromDelegate(HandleSmartEquipBelt))
|
.Bind(ContentKeyFunctions.SmartEquipBelt, InputCmdHandler.FromDelegate(HandleSmartEquipBelt))
|
||||||
.Bind(ContentKeyFunctions.MovePulledObject, new PointerInputCmdHandler(HandleMovePulledObject))
|
|
||||||
.Bind(ContentKeyFunctions.ReleasePulledObject, InputCmdHandler.FromDelegate(HandleReleasePulledObject))
|
|
||||||
.Register<HandsSystem>();
|
.Register<HandsSystem>();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -229,29 +226,5 @@ namespace Content.Server.GameObjects.EntitySystems
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool HandleMovePulledObject(ICommonSession session, EntityCoordinates coords, EntityUid uid)
|
|
||||||
{
|
|
||||||
var playerEntity = session.AttachedEntity;
|
|
||||||
|
|
||||||
if (playerEntity == null ||
|
|
||||||
!playerEntity.TryGetComponent<HandsComponent>(out var hands))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
hands.MovePulledObject(playerEntity.Transform.Coordinates, coords);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void HandleReleasePulledObject(ICommonSession session)
|
|
||||||
{
|
|
||||||
if (!TryGetAttachedComponent(session as IPlayerSession, out HandsComponent handsComp))
|
|
||||||
return;
|
|
||||||
|
|
||||||
handsComp.StopPull();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,78 +0,0 @@
|
|||||||
using Content.Server.GameObjects.Components.GUI;
|
|
||||||
using Content.Server.GameObjects.Components.Movement;
|
|
||||||
using Content.Shared.GameObjects.Components.Items;
|
|
||||||
using Content.Shared.GameObjects.EntitySystems;
|
|
||||||
using Content.Shared.GameObjects.Verbs;
|
|
||||||
using Content.Shared.Physics.Pull;
|
|
||||||
using Robust.Server.Interfaces.GameObjects;
|
|
||||||
using Robust.Shared.GameObjects.Components;
|
|
||||||
using Robust.Shared.Interfaces.GameObjects;
|
|
||||||
using Robust.Shared.Localization;
|
|
||||||
|
|
||||||
namespace Content.Server.GlobalVerbs
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Global verb that pulls an entity.
|
|
||||||
/// </summary>
|
|
||||||
[GlobalVerb]
|
|
||||||
public class PullingVerb : GlobalVerb
|
|
||||||
{
|
|
||||||
public override bool RequireInteractionRange => false;
|
|
||||||
|
|
||||||
public override void GetData(IEntity user, IEntity target, VerbData data)
|
|
||||||
{
|
|
||||||
data.Visibility = VerbVisibility.Invisible;
|
|
||||||
|
|
||||||
if (user == target ||
|
|
||||||
!user.HasComponent<IActorComponent>() ||
|
|
||||||
!target.HasComponent<PullableComponent>())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var dist = user.Transform.Coordinates.Position - target.Transform.Coordinates.Position;
|
|
||||||
if (dist.LengthSquared > SharedInteractionSystem.InteractionRangeSquared)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!user.HasComponent<ISharedHandsComponent>() ||
|
|
||||||
!user.TryGetComponent(out IPhysicsComponent userPhysics) ||
|
|
||||||
!target.TryGetComponent(out IPhysicsComponent targetPhysics) ||
|
|
||||||
targetPhysics.Anchored)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var controller = targetPhysics.EnsureController<PullController>();
|
|
||||||
|
|
||||||
data.Visibility = VerbVisibility.Visible;
|
|
||||||
data.Text = controller.Puller == userPhysics
|
|
||||||
? Loc.GetString("Stop pulling")
|
|
||||||
: Loc.GetString("Pull");
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void Activate(IEntity user, IEntity target)
|
|
||||||
{
|
|
||||||
if (!user.TryGetComponent(out IPhysicsComponent userPhysics) ||
|
|
||||||
!target.TryGetComponent(out IPhysicsComponent targetPhysics) ||
|
|
||||||
targetPhysics.Anchored ||
|
|
||||||
!target.TryGetComponent(out PullableComponent pullable) ||
|
|
||||||
!user.TryGetComponent(out HandsComponent hands))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var controller = targetPhysics.EnsureController<PullController>();
|
|
||||||
|
|
||||||
if (controller.Puller == userPhysics)
|
|
||||||
{
|
|
||||||
hands.StopPull();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
hands.StartPull(pullable);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -4,6 +4,5 @@ namespace Content.Shared.GameObjects.Components.Items
|
|||||||
{
|
{
|
||||||
public interface ISharedHandsComponent : IComponent
|
public interface ISharedHandsComponent : IComponent
|
||||||
{
|
{
|
||||||
void StopPull();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
#nullable enable
|
#nullable enable
|
||||||
using System;
|
using System;
|
||||||
|
using Content.Shared.GameObjects.Components.Pulling;
|
||||||
using Content.Shared.Physics.Pull;
|
using Content.Shared.Physics.Pull;
|
||||||
|
using Robust.Shared.Containers;
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.GameObjects.Components;
|
using Robust.Shared.GameObjects.Components;
|
||||||
using Robust.Shared.Interfaces.GameObjects;
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
@@ -13,42 +15,6 @@ namespace Content.Shared.GameObjects.Components.Items
|
|||||||
{
|
{
|
||||||
public sealed override string Name => "Hands";
|
public sealed override string Name => "Hands";
|
||||||
public sealed override uint? NetID => ContentNetIDs.HANDS;
|
public sealed override uint? NetID => ContentNetIDs.HANDS;
|
||||||
|
|
||||||
[ViewVariables]
|
|
||||||
public IPhysicsComponent? PulledObject { get; protected set; }
|
|
||||||
|
|
||||||
[ViewVariables]
|
|
||||||
protected bool IsPulling => PulledObject != null;
|
|
||||||
|
|
||||||
public virtual void StopPull()
|
|
||||||
{
|
|
||||||
if (PulledObject != null &&
|
|
||||||
PulledObject.TryGetController(out PullController controller))
|
|
||||||
{
|
|
||||||
controller.StopPull();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void HandleMessage(ComponentMessage message, IComponent? component)
|
|
||||||
{
|
|
||||||
base.HandleMessage(message, component);
|
|
||||||
|
|
||||||
if (!(message is PullMessage pullMessage) ||
|
|
||||||
pullMessage.Puller.Owner != Owner)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (message)
|
|
||||||
{
|
|
||||||
case PullStartedMessage msg:
|
|
||||||
PulledObject = msg.Pulled;
|
|
||||||
break;
|
|
||||||
case PullStoppedMessage _:
|
|
||||||
PulledObject = null;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[Serializable, NetSerializable]
|
[Serializable, NetSerializable]
|
||||||
|
|||||||
@@ -14,7 +14,11 @@ namespace Content.Shared.GameObjects.Components.Mobs
|
|||||||
public override string Name => "StatusEffectsUI";
|
public override string Name => "StatusEffectsUI";
|
||||||
public override uint? NetID => ContentNetIDs.STATUSEFFECTS;
|
public override uint? NetID => ContentNetIDs.STATUSEFFECTS;
|
||||||
|
|
||||||
|
public abstract void ChangeStatusEffectIcon(StatusEffect effect, string icon);
|
||||||
|
|
||||||
public abstract void ChangeStatusEffect(StatusEffect effect, string icon, ValueTuple<TimeSpan, TimeSpan>? cooldown);
|
public abstract void ChangeStatusEffect(StatusEffect effect, string icon, ValueTuple<TimeSpan, TimeSpan>? cooldown);
|
||||||
|
|
||||||
|
public abstract void RemoveStatusEffect(StatusEffect effect);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Serializable, NetSerializable]
|
[Serializable, NetSerializable]
|
||||||
|
|||||||
@@ -0,0 +1,263 @@
|
|||||||
|
#nullable enable
|
||||||
|
using System;
|
||||||
|
using Content.Shared.GameObjects.Components.Mobs;
|
||||||
|
using Content.Shared.Physics.Pull;
|
||||||
|
using Robust.Shared.Containers;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
|
using Robust.Shared.GameObjects.Components;
|
||||||
|
using Robust.Shared.GameObjects.Components.Transform;
|
||||||
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
|
using Robust.Shared.Map;
|
||||||
|
using Robust.Shared.Serialization;
|
||||||
|
|
||||||
|
namespace Content.Shared.GameObjects.Components.Pulling
|
||||||
|
{
|
||||||
|
public abstract class SharedPullableComponent : Component
|
||||||
|
{
|
||||||
|
public override string Name => "Pullable";
|
||||||
|
public override uint? NetID => ContentNetIDs.PULLABLE;
|
||||||
|
|
||||||
|
private IEntity? _puller;
|
||||||
|
|
||||||
|
public virtual IEntity? Puller
|
||||||
|
{
|
||||||
|
get => _puller;
|
||||||
|
private set
|
||||||
|
{
|
||||||
|
if (_puller == value)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_puller = value;
|
||||||
|
Dirty();
|
||||||
|
|
||||||
|
if (!Owner.TryGetComponent(out IPhysicsComponent? physics))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
PullController controller;
|
||||||
|
|
||||||
|
if (value == null)
|
||||||
|
{
|
||||||
|
if (physics.TryGetController(out controller))
|
||||||
|
{
|
||||||
|
controller.StopPull();
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
controller = physics.EnsureController<PullController>();
|
||||||
|
controller.StartPull(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool BeingPulled => Puller != null;
|
||||||
|
|
||||||
|
public bool CanStartPull(IEntity puller)
|
||||||
|
{
|
||||||
|
if (!puller.HasComponent<SharedPullerComponent>())
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!puller.TryGetComponent(out IPhysicsComponent? physics))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (physics.Anchored)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (puller == Owner)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!puller.IsInSameOrNoContainer(Owner))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool TryStartPull(IEntity puller)
|
||||||
|
{
|
||||||
|
if (!CanStartPull(puller))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
TryStopPull();
|
||||||
|
|
||||||
|
Puller = puller;
|
||||||
|
|
||||||
|
if (Puller != puller)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool TryStopPull()
|
||||||
|
{
|
||||||
|
if (!BeingPulled)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Puller = null;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool TogglePull(IEntity puller)
|
||||||
|
{
|
||||||
|
if (Puller == null)
|
||||||
|
{
|
||||||
|
return TryStartPull(puller);
|
||||||
|
}
|
||||||
|
|
||||||
|
return TryStopPull();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool TryMoveTo(EntityCoordinates to)
|
||||||
|
{
|
||||||
|
if (Puller == null)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Owner.TryGetComponent(out IPhysicsComponent? physics))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!physics.TryGetController(out PullController controller))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return controller.TryMoveTo(Puller.Transform.Coordinates, to);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void PullerMoved(MoveEvent moveEvent)
|
||||||
|
{
|
||||||
|
if (Puller == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (moveEvent.Sender != Puller)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Owner.TryGetComponent(out IPhysicsComponent? physics))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
physics.WakeBody();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override ComponentState GetComponentState()
|
||||||
|
{
|
||||||
|
return new PullableComponentState(Puller?.Uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void HandleComponentState(ComponentState? curState, ComponentState? nextState)
|
||||||
|
{
|
||||||
|
base.HandleComponentState(curState, nextState);
|
||||||
|
|
||||||
|
if (!(curState is PullableComponentState state))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state.Puller == null)
|
||||||
|
{
|
||||||
|
Puller = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Puller = Owner.EntityManager.GetEntity(state.Puller.Value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void HandleMessage(ComponentMessage message, IComponent? component)
|
||||||
|
{
|
||||||
|
base.HandleMessage(message, component);
|
||||||
|
|
||||||
|
if (!(message is PullMessage pullMessage) ||
|
||||||
|
pullMessage.Pulled.Owner != Owner)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (message)
|
||||||
|
{
|
||||||
|
case PullStartedMessage msg:
|
||||||
|
Owner.EntityManager.EventBus.SubscribeEvent<MoveEvent>(EventSource.Local, this, PullerMoved);
|
||||||
|
|
||||||
|
AddPullingStatuses(msg.Puller.Owner);
|
||||||
|
break;
|
||||||
|
case PullStoppedMessage msg:
|
||||||
|
Owner.EntityManager.EventBus.UnsubscribeEvent<MoveEvent>(EventSource.Local, this);
|
||||||
|
|
||||||
|
RemovePullingStatuses(msg.Puller.Owner);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void AddPullingStatuses(IEntity puller)
|
||||||
|
{
|
||||||
|
if (Owner.TryGetComponent(out SharedStatusEffectsComponent? pulledStatus))
|
||||||
|
{
|
||||||
|
pulledStatus.ChangeStatusEffectIcon(StatusEffect.Pulled,
|
||||||
|
"/Textures/Interface/StatusEffects/Pull/pulled.png");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (puller.TryGetComponent(out SharedStatusEffectsComponent? ownerStatus))
|
||||||
|
{
|
||||||
|
ownerStatus.ChangeStatusEffectIcon(StatusEffect.Pulling,
|
||||||
|
"/Textures/Interface/StatusEffects/Pull/pulling.png");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RemovePullingStatuses(IEntity puller)
|
||||||
|
{
|
||||||
|
if (Owner.TryGetComponent(out SharedStatusEffectsComponent? pulledStatus))
|
||||||
|
{
|
||||||
|
pulledStatus.RemoveStatusEffect(StatusEffect.Pulled);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (puller.TryGetComponent(out SharedStatusEffectsComponent? ownerStatus))
|
||||||
|
{
|
||||||
|
ownerStatus.RemoveStatusEffect(StatusEffect.Pulling);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void OnRemove()
|
||||||
|
{
|
||||||
|
TryStopPull();
|
||||||
|
|
||||||
|
base.OnRemove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Serializable, NetSerializable]
|
||||||
|
public class PullableComponentState : ComponentState
|
||||||
|
{
|
||||||
|
public readonly EntityUid? Puller;
|
||||||
|
|
||||||
|
public PullableComponentState(EntityUid? puller) : base(ContentNetIDs.PULLABLE)
|
||||||
|
{
|
||||||
|
Puller = puller;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Content.Shared.GameObjects.Components.Movement;
|
||||||
|
using Content.Shared.Physics.Pull;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
|
using Component = Robust.Shared.GameObjects.Component;
|
||||||
|
|
||||||
|
namespace Content.Shared.GameObjects.Components.Pulling
|
||||||
|
{
|
||||||
|
[RegisterComponent]
|
||||||
|
public class SharedPullerComponent : Component, IMoveSpeedModifier
|
||||||
|
{
|
||||||
|
public override string Name => "Puller";
|
||||||
|
|
||||||
|
private IEntity? _pulling;
|
||||||
|
|
||||||
|
public float WalkSpeedModifier => Pulling == null ? 1.0f : 0.75f;
|
||||||
|
|
||||||
|
public float SprintSpeedModifier => Pulling == null ? 1.0f : 0.75f;
|
||||||
|
|
||||||
|
public IEntity? Pulling
|
||||||
|
{
|
||||||
|
get => _pulling;
|
||||||
|
private set
|
||||||
|
{
|
||||||
|
if (_pulling == value)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_pulling = value;
|
||||||
|
|
||||||
|
if (Owner.TryGetComponent(out MovementSpeedModifierComponent? speed))
|
||||||
|
{
|
||||||
|
speed.RefreshMovementSpeedModifiers();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void OnRemove()
|
||||||
|
{
|
||||||
|
if (Pulling != null &&
|
||||||
|
Pulling.TryGetComponent(out SharedPullableComponent? pullable))
|
||||||
|
{
|
||||||
|
pullable.TryStopPull();
|
||||||
|
}
|
||||||
|
|
||||||
|
base.OnRemove();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void HandleMessage(ComponentMessage message, IComponent? component)
|
||||||
|
{
|
||||||
|
base.HandleMessage(message, component);
|
||||||
|
|
||||||
|
if (!(message is PullMessage pullMessage) ||
|
||||||
|
pullMessage.Puller.Owner != Owner)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (message)
|
||||||
|
{
|
||||||
|
case PullStartedMessage msg:
|
||||||
|
Pulling = msg.Pulled.Owner;
|
||||||
|
break;
|
||||||
|
case PullStoppedMessage _:
|
||||||
|
Pulling = null;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -81,6 +81,7 @@
|
|||||||
public const uint CRAYONS = 1075;
|
public const uint CRAYONS = 1075;
|
||||||
public const uint PLACEABLE_SURFACE = 1076;
|
public const uint PLACEABLE_SURFACE = 1076;
|
||||||
public const uint STORABLE = 1077;
|
public const uint STORABLE = 1077;
|
||||||
|
public const uint PULLABLE = 1078;
|
||||||
|
|
||||||
// Net IDs for integration tests.
|
// Net IDs for integration tests.
|
||||||
public const uint PREDICTION_TEST = 10001;
|
public const uint PREDICTION_TEST = 10001;
|
||||||
|
|||||||
117
Content.Shared/GameObjects/EntitySystems/SharedPullingSystem.cs
Normal file
117
Content.Shared/GameObjects/EntitySystems/SharedPullingSystem.cs
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
#nullable enable
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using Content.Shared.GameObjects.Components.Pulling;
|
||||||
|
using Content.Shared.Input;
|
||||||
|
using Content.Shared.Physics.Pull;
|
||||||
|
using JetBrains.Annotations;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
|
using Robust.Shared.GameObjects.Systems;
|
||||||
|
using Robust.Shared.Input.Binding;
|
||||||
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
|
using Robust.Shared.Map;
|
||||||
|
using Robust.Shared.Players;
|
||||||
|
|
||||||
|
namespace Content.Shared.GameObjects.EntitySystems
|
||||||
|
{
|
||||||
|
[UsedImplicitly]
|
||||||
|
public class SharedPullingSystem : EntitySystem
|
||||||
|
{
|
||||||
|
private readonly Dictionary<IEntity, IEntity> _pullers =
|
||||||
|
new Dictionary<IEntity, IEntity>();
|
||||||
|
|
||||||
|
public override void Initialize()
|
||||||
|
{
|
||||||
|
base.Initialize();
|
||||||
|
|
||||||
|
SubscribeLocalEvent<PullStartedMessage>(OnPullStarted);
|
||||||
|
SubscribeLocalEvent<PullStoppedMessage>(OnPullStopped);
|
||||||
|
|
||||||
|
CommandBinds.Builder
|
||||||
|
.Bind(ContentKeyFunctions.MovePulledObject, new PointerInputCmdHandler(HandleMovePulledObject))
|
||||||
|
.Bind(ContentKeyFunctions.ReleasePulledObject, InputCmdHandler.FromDelegate(HandleReleasePulledObject))
|
||||||
|
.Register<SharedPullingSystem>();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnPullStarted(PullStartedMessage message)
|
||||||
|
{
|
||||||
|
SetPuller(message.Puller.Owner, message.Pulled.Owner);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnPullStopped(PullStoppedMessage message)
|
||||||
|
{
|
||||||
|
RemovePuller(message.Puller.Owner);
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool HandleMovePulledObject(ICommonSession? session, EntityCoordinates coords, EntityUid uid)
|
||||||
|
{
|
||||||
|
var player = session?.AttachedEntity;
|
||||||
|
|
||||||
|
if (player == null)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!TryGetPulled(player, out var pulled))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pulled.TryGetComponent(out SharedPullableComponent? pullable))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
pullable.TryMoveTo(coords);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void HandleReleasePulledObject(ICommonSession? session)
|
||||||
|
{
|
||||||
|
var player = session?.AttachedEntity;
|
||||||
|
|
||||||
|
if (player == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!TryGetPulled(player, out var pulled))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pulled.TryGetComponent(out SharedPullableComponent? pullable))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
pullable.TryStopPull();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SetPuller(IEntity puller, IEntity pulled)
|
||||||
|
{
|
||||||
|
_pullers[puller] = pulled;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool RemovePuller(IEntity puller)
|
||||||
|
{
|
||||||
|
return _pullers.Remove(puller);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IEntity? GetPulled(IEntity by)
|
||||||
|
{
|
||||||
|
return _pullers.GetValueOrDefault(by);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool TryGetPulled(IEntity by, [NotNullWhen(true)] out IEntity? pulled)
|
||||||
|
{
|
||||||
|
return (pulled = GetPulled(by)) != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsPulling(IEntity puller)
|
||||||
|
{
|
||||||
|
return _pullers.ContainsKey(puller);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,22 +1,28 @@
|
|||||||
#nullable enable
|
#nullable enable
|
||||||
using System;
|
using System;
|
||||||
using Content.Shared.GameObjects.EntitySystems;
|
using System.Linq;
|
||||||
using Robust.Shared.Containers;
|
using Robust.Shared.Containers;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.GameObjects.Components;
|
using Robust.Shared.GameObjects.Components;
|
||||||
using Robust.Shared.Interfaces.GameObjects;
|
using Robust.Shared.Interfaces.GameObjects;
|
||||||
|
using Robust.Shared.Interfaces.Physics;
|
||||||
using Robust.Shared.IoC;
|
using Robust.Shared.IoC;
|
||||||
using Robust.Shared.Map;
|
using Robust.Shared.Map;
|
||||||
using Robust.Shared.Maths;
|
using Robust.Shared.Maths;
|
||||||
using Robust.Shared.Physics;
|
using Robust.Shared.Physics;
|
||||||
using Robust.Shared.Utility;
|
using Robust.Shared.Utility;
|
||||||
|
using static Content.Shared.GameObjects.EntitySystems.SharedInteractionSystem;
|
||||||
|
|
||||||
namespace Content.Shared.Physics.Pull
|
namespace Content.Shared.Physics.Pull
|
||||||
{
|
{
|
||||||
public class PullController : VirtualController
|
public class PullController : VirtualController
|
||||||
{
|
{
|
||||||
private const float DistBeforePull = 1.0f;
|
[Dependency] private readonly IEntityManager _entityManager = default!;
|
||||||
|
[Dependency] private readonly IPhysicsManager _physicsManager = default!;
|
||||||
|
|
||||||
private const float DistBeforeStopPull = SharedInteractionSystem.InteractionRange;
|
private const float DistBeforeStopPull = InteractionRange;
|
||||||
|
|
||||||
|
private const float StopMoveThreshold = 0.25f;
|
||||||
|
|
||||||
private IPhysicsComponent? _puller;
|
private IPhysicsComponent? _puller;
|
||||||
|
|
||||||
@@ -26,81 +32,167 @@ namespace Content.Shared.Physics.Pull
|
|||||||
|
|
||||||
public IPhysicsComponent? Puller => _puller;
|
public IPhysicsComponent? Puller => _puller;
|
||||||
|
|
||||||
public void StartPull(IPhysicsComponent puller)
|
public EntityCoordinates? MovingTo
|
||||||
|
{
|
||||||
|
get => _movingTo;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if (_movingTo == value || ControlledComponent == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_movingTo = value;
|
||||||
|
ControlledComponent.WakeBody();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private float DistanceBeforeStopPull()
|
||||||
|
{
|
||||||
|
if (_puller == null)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
var aabbSize = _puller.AABB.Size;
|
||||||
|
|
||||||
|
return (aabbSize.X > aabbSize.Y ? aabbSize.X : aabbSize.Y) + 0.15f;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool PullerMovingTowardsPulled()
|
||||||
|
{
|
||||||
|
if (_puller == null)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ControlledComponent == null)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_puller.LinearVelocity.EqualsApprox(Vector2.Zero))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var pullerTransform = _puller.Owner.Transform;
|
||||||
|
var origin = pullerTransform.Coordinates.Position;
|
||||||
|
var velocity = _puller.LinearVelocity.Normalized;
|
||||||
|
var mapId = pullerTransform.MapPosition.MapId;
|
||||||
|
var ray = new CollisionRay(origin, velocity, (int) CollisionGroup.AllMask);
|
||||||
|
bool Predicate(IEntity e) => e != ControlledComponent.Owner;
|
||||||
|
var rayResults =
|
||||||
|
_physicsManager.IntersectRayWithPredicate(mapId, ray, DistanceBeforeStopPull() * 2, Predicate);
|
||||||
|
|
||||||
|
return rayResults.Any();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StartPull(IEntity entity)
|
||||||
|
{
|
||||||
|
DebugTools.AssertNotNull(entity);
|
||||||
|
|
||||||
|
if (!entity.TryGetComponent(out IPhysicsComponent? physics))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return StartPull(physics);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StartPull(IPhysicsComponent puller)
|
||||||
{
|
{
|
||||||
DebugTools.AssertNotNull(puller);
|
DebugTools.AssertNotNull(puller);
|
||||||
|
|
||||||
if (_puller == puller)
|
if (_puller == puller)
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
_puller = puller;
|
|
||||||
|
|
||||||
if (ControlledComponent == null)
|
if (ControlledComponent == null)
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
ControlledComponent.WakeBody();
|
_puller = puller;
|
||||||
|
|
||||||
var message = new PullStartedMessage(this, _puller, ControlledComponent);
|
var message = new PullStartedMessage(this, _puller, ControlledComponent);
|
||||||
|
|
||||||
_puller.Owner.SendMessage(null, message);
|
_puller.Owner.SendMessage(null, message);
|
||||||
ControlledComponent.Owner.SendMessage(null, message);
|
ControlledComponent.Owner.SendMessage(null, message);
|
||||||
|
|
||||||
|
_puller.Owner.EntityManager.EventBus.RaiseEvent(EventSource.Local, message);
|
||||||
|
|
||||||
|
ControlledComponent.WakeBody();
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void StopPull()
|
public bool StopPull()
|
||||||
{
|
{
|
||||||
var oldPuller = _puller;
|
var oldPuller = _puller;
|
||||||
|
|
||||||
if (oldPuller == null)
|
if (oldPuller == null)
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
_puller = null;
|
_puller = null;
|
||||||
|
|
||||||
if (ControlledComponent == null)
|
if (ControlledComponent == null)
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
ControlledComponent.WakeBody();
|
var message = new PullStoppedMessage(oldPuller, ControlledComponent);
|
||||||
|
|
||||||
var message = new PullStoppedMessage(this, oldPuller, ControlledComponent);
|
|
||||||
|
|
||||||
oldPuller.Owner.SendMessage(null, message);
|
oldPuller.Owner.SendMessage(null, message);
|
||||||
ControlledComponent.Owner.SendMessage(null, message);
|
ControlledComponent.Owner.SendMessage(null, message);
|
||||||
|
|
||||||
|
oldPuller.Owner.EntityManager.EventBus.RaiseEvent(EventSource.Local, message);
|
||||||
|
|
||||||
|
ControlledComponent.WakeBody();
|
||||||
ControlledComponent.TryRemoveController<PullController>();
|
ControlledComponent.TryRemoveController<PullController>();
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void TryMoveTo(EntityCoordinates from, EntityCoordinates to)
|
public bool TryMoveTo(EntityCoordinates from, EntityCoordinates to)
|
||||||
{
|
{
|
||||||
if (_puller == null || ControlledComponent == null)
|
if (_puller == null || ControlledComponent == null)
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
var entityManager = IoCManager.Resolve<IEntityManager>();
|
if (!_puller.Owner.Transform.Coordinates.InRange(_entityManager, from, InteractionRange))
|
||||||
|
|
||||||
if (!from.InRange(entityManager, to, SharedInteractionSystem.InteractionRange))
|
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
ControlledComponent.WakeBody();
|
if (!_puller.Owner.Transform.Coordinates.InRange(_entityManager, to, InteractionRange))
|
||||||
|
|
||||||
var dist = _puller.Owner.Transform.Coordinates.Position - to.Position;
|
|
||||||
|
|
||||||
if (Math.Sqrt(dist.LengthSquared) > DistBeforeStopPull ||
|
|
||||||
Math.Sqrt(dist.LengthSquared) < 0.25f)
|
|
||||||
{
|
{
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
_movingTo = to;
|
if (!from.InRange(_entityManager, to, InteractionRange))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (from.Position.EqualsApprox(to.Position))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!_puller.Owner.Transform.Coordinates.TryDistance(_entityManager, to, out var distance) ||
|
||||||
|
Math.Sqrt(distance) > DistBeforeStopPull ||
|
||||||
|
Math.Sqrt(distance) < StopMoveThreshold)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
MovingTo = to;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void UpdateBeforeProcessing()
|
public override void UpdateBeforeProcessing()
|
||||||
@@ -116,21 +208,20 @@ namespace Content.Shared.Physics.Pull
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Are we outside of pulling range?
|
var distance = _puller.Owner.Transform.WorldPosition - ControlledComponent.Owner.Transform.WorldPosition;
|
||||||
var dist = _puller.Owner.Transform.WorldPosition - ControlledComponent.Owner.Transform.WorldPosition;
|
|
||||||
|
|
||||||
if (dist.Length > DistBeforeStopPull)
|
if (distance.Length > DistBeforeStopPull)
|
||||||
{
|
{
|
||||||
StopPull();
|
StopPull();
|
||||||
}
|
}
|
||||||
else if (_movingTo.HasValue)
|
else if (MovingTo.HasValue)
|
||||||
{
|
{
|
||||||
var diff = _movingTo.Value.Position - ControlledComponent.Owner.Transform.Coordinates.Position;
|
var diff = MovingTo.Value.Position - ControlledComponent.Owner.Transform.Coordinates.Position;
|
||||||
LinearVelocity = diff.Normalized * 5;
|
LinearVelocity = diff.Normalized * 5;
|
||||||
}
|
}
|
||||||
else if (dist.Length > DistBeforePull)
|
else if (distance.Length > DistanceBeforeStopPull() && !PullerMovingTowardsPulled())
|
||||||
{
|
{
|
||||||
LinearVelocity = dist.Normalized * _puller.LinearVelocity.Length * 1.1f;
|
LinearVelocity = distance.Normalized * _puller.LinearVelocity.Length * 1.5f;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -144,18 +235,14 @@ namespace Content.Shared.Physics.Pull
|
|||||||
|
|
||||||
if (ControlledComponent == null)
|
if (ControlledComponent == null)
|
||||||
{
|
{
|
||||||
_movingTo = null;
|
MovingTo = null;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_movingTo == null)
|
if (MovingTo != null &&
|
||||||
|
ControlledComponent.Owner.Transform.Coordinates.Position.EqualsApprox(MovingTo.Value.Position, 0.01))
|
||||||
{
|
{
|
||||||
return;
|
MovingTo = null;
|
||||||
}
|
|
||||||
|
|
||||||
if (ControlledComponent.Owner.Transform.Coordinates.Position.EqualsApprox(_movingTo.Value.Position, 0.01))
|
|
||||||
{
|
|
||||||
_movingTo = null;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,13 +5,11 @@ namespace Content.Shared.Physics.Pull
|
|||||||
{
|
{
|
||||||
public class PullMessage : ComponentMessage
|
public class PullMessage : ComponentMessage
|
||||||
{
|
{
|
||||||
public readonly PullController Controller;
|
|
||||||
public readonly IPhysicsComponent Puller;
|
public readonly IPhysicsComponent Puller;
|
||||||
public readonly IPhysicsComponent Pulled;
|
public readonly IPhysicsComponent Pulled;
|
||||||
|
|
||||||
protected PullMessage(PullController controller, IPhysicsComponent puller, IPhysicsComponent pulled)
|
protected PullMessage(IPhysicsComponent puller, IPhysicsComponent pulled)
|
||||||
{
|
{
|
||||||
Controller = controller;
|
|
||||||
Puller = puller;
|
Puller = puller;
|
||||||
Pulled = pulled;
|
Pulled = pulled;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ namespace Content.Shared.Physics.Pull
|
|||||||
public class PullStartedMessage : PullMessage
|
public class PullStartedMessage : PullMessage
|
||||||
{
|
{
|
||||||
public PullStartedMessage(PullController controller, IPhysicsComponent puller, IPhysicsComponent pulled) :
|
public PullStartedMessage(PullController controller, IPhysicsComponent puller, IPhysicsComponent pulled) :
|
||||||
base(controller, puller, pulled)
|
base(puller, pulled)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,8 +4,7 @@ namespace Content.Shared.Physics.Pull
|
|||||||
{
|
{
|
||||||
public class PullStoppedMessage : PullMessage
|
public class PullStoppedMessage : PullMessage
|
||||||
{
|
{
|
||||||
public PullStoppedMessage(PullController controller, IPhysicsComponent puller, IPhysicsComponent pulled) :
|
public PullStoppedMessage(IPhysicsComponent puller, IPhysicsComponent pulled) : base(puller, pulled)
|
||||||
base(controller, puller, pulled)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -57,3 +57,4 @@
|
|||||||
normal: running
|
normal: running
|
||||||
crit: crit
|
crit: crit
|
||||||
dead: dead
|
dead: dead
|
||||||
|
- type: Puller
|
||||||
|
|||||||
@@ -195,6 +195,7 @@
|
|||||||
type: StrippableBoundUserInterface
|
type: StrippableBoundUserInterface
|
||||||
- key: enum.AcceptCloningUiKey.Key
|
- key: enum.AcceptCloningUiKey.Key
|
||||||
type: AcceptCloningBoundUserInterface
|
type: AcceptCloningBoundUserInterface
|
||||||
|
- type: Puller
|
||||||
|
|
||||||
- type: entity
|
- type: entity
|
||||||
save: false
|
save: false
|
||||||
|
|||||||
Reference in New Issue
Block a user