mirror of
https://github.com/WWhiteDreamProject/wwdpublic.git
synced 2026-04-19 06:28:40 +03:00
# Description "Walking by default" is now implemented as a client-side toggle in the settings. Also, fixed walking alerts being shitcode and not displaying the correct walking/running state. One issue still persists: if you press the "toggle walk" button while typing in the chat, it still resets the walking state to default. I don't know what causes it, probably the input field leaking button-up input events instead of marking them as handled. <details><summary><h1>Media</h1></summary> <p> https://github.com/user-attachments/assets/97d5b495-ea9b-4854-a6b0-84815ed5a332 </p> </details> # Changelog 🆑 - add: You can now choose whether you want to walk or run by default in the settings. --------- Signed-off-by: Mnemotechnican <69920617+Mnemotechnician@users.noreply.github.com> Co-authored-by: VMSolidus <evilexecutive@gmail.com> # Conflicts: # Content.Client/Options/UI/Tabs/KeyRebindTab.xaml.cs # Content.Shared/CCVar/CCVars.cs
146 lines
5.8 KiB
C#
146 lines
5.8 KiB
C#
using Content.Shared.CCVar;
|
|
using Content.Shared.Movement.Components;
|
|
using Content.Shared.Movement.Events;
|
|
using Content.Shared.Movement.Pulling.Components;
|
|
using Content.Shared.Movement.Systems;
|
|
using Robust.Client.GameObjects;
|
|
using Robust.Client.Physics;
|
|
using Robust.Client.Player;
|
|
using Robust.Shared.Configuration;
|
|
using Robust.Shared.Physics.Components;
|
|
using Robust.Shared.Player;
|
|
using Robust.Shared.Timing;
|
|
|
|
namespace Content.Client.Physics.Controllers
|
|
{
|
|
public sealed class MoverController : SharedMoverController
|
|
{
|
|
[Dependency] private readonly IConfigurationManager _config = default!;
|
|
[Dependency] private readonly IGameTiming _timing = default!;
|
|
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
|
|
|
public override void Initialize()
|
|
{
|
|
base.Initialize();
|
|
SubscribeLocalEvent<RelayInputMoverComponent, LocalPlayerAttachedEvent>(OnRelayPlayerAttached);
|
|
SubscribeLocalEvent<RelayInputMoverComponent, LocalPlayerDetachedEvent>(OnRelayPlayerDetached);
|
|
SubscribeLocalEvent<InputMoverComponent, LocalPlayerAttachedEvent>(OnPlayerAttached);
|
|
SubscribeLocalEvent<InputMoverComponent, LocalPlayerDetachedEvent>(OnPlayerDetached);
|
|
|
|
SubscribeLocalEvent<InputMoverComponent, UpdateIsPredictedEvent>(OnUpdatePredicted);
|
|
SubscribeLocalEvent<MovementRelayTargetComponent, UpdateIsPredictedEvent>(OnUpdateRelayTargetPredicted);
|
|
SubscribeLocalEvent<PullableComponent, UpdateIsPredictedEvent>(OnUpdatePullablePredicted);
|
|
|
|
Subs.CVar(_config, CCVars.DefaultWalk, _ => RaiseNetworkEvent(new UpdateInputCVarsMessage()));
|
|
}
|
|
|
|
private void OnUpdatePredicted(EntityUid uid, InputMoverComponent component, ref UpdateIsPredictedEvent args)
|
|
{
|
|
// Enable prediction if an entity is controlled by the player
|
|
if (uid == _playerManager.LocalEntity)
|
|
args.IsPredicted = true;
|
|
}
|
|
|
|
private void OnUpdateRelayTargetPredicted(EntityUid uid, MovementRelayTargetComponent component, ref UpdateIsPredictedEvent args)
|
|
{
|
|
if (component.Source == _playerManager.LocalEntity)
|
|
args.IsPredicted = true;
|
|
}
|
|
|
|
private void OnUpdatePullablePredicted(EntityUid uid, PullableComponent component, ref UpdateIsPredictedEvent args)
|
|
{
|
|
// Enable prediction if an entity is being pulled by the player.
|
|
// Disable prediction if an entity is being pulled by some non-player entity.
|
|
|
|
if (component.Puller == _playerManager.LocalEntity)
|
|
args.IsPredicted = true;
|
|
else if (component.Puller != null)
|
|
args.BlockPrediction = true;
|
|
|
|
// TODO recursive pulling checks?
|
|
// What if the entity is being pulled by a vehicle controlled by the player?
|
|
}
|
|
|
|
private void OnRelayPlayerAttached(EntityUid uid, RelayInputMoverComponent component, LocalPlayerAttachedEvent args)
|
|
{
|
|
Physics.UpdateIsPredicted(uid);
|
|
Physics.UpdateIsPredicted(component.RelayEntity);
|
|
if (MoverQuery.TryGetComponent(component.RelayEntity, out var inputMover))
|
|
SetMoveInput(inputMover, MoveButtons.None);
|
|
}
|
|
|
|
private void OnRelayPlayerDetached(EntityUid uid, RelayInputMoverComponent component, LocalPlayerDetachedEvent args)
|
|
{
|
|
Physics.UpdateIsPredicted(uid);
|
|
Physics.UpdateIsPredicted(component.RelayEntity);
|
|
if (MoverQuery.TryGetComponent(component.RelayEntity, out var inputMover))
|
|
SetMoveInput(inputMover, MoveButtons.None);
|
|
}
|
|
|
|
private void OnPlayerAttached(EntityUid uid, InputMoverComponent component, LocalPlayerAttachedEvent args)
|
|
{
|
|
SetMoveInput(component, MoveButtons.None);
|
|
}
|
|
|
|
private void OnPlayerDetached(EntityUid uid, InputMoverComponent component, LocalPlayerDetachedEvent args)
|
|
{
|
|
SetMoveInput(component, MoveButtons.None);
|
|
}
|
|
|
|
public override void UpdateBeforeSolve(bool prediction, float frameTime)
|
|
{
|
|
base.UpdateBeforeSolve(prediction, frameTime);
|
|
|
|
if (_playerManager.LocalEntity is not {Valid: true} player)
|
|
return;
|
|
|
|
if (RelayQuery.TryGetComponent(player, out var relayMover))
|
|
HandleClientsideMovement(relayMover.RelayEntity, frameTime);
|
|
|
|
HandleClientsideMovement(player, frameTime);
|
|
}
|
|
|
|
private void HandleClientsideMovement(EntityUid player, float frameTime)
|
|
{
|
|
if (!MoverQuery.TryGetComponent(player, out var mover) ||
|
|
!XformQuery.TryGetComponent(player, out var xform))
|
|
{
|
|
return;
|
|
}
|
|
|
|
var physicsUid = player;
|
|
PhysicsComponent? body;
|
|
var xformMover = xform;
|
|
|
|
if (mover.ToParent && RelayQuery.HasComponent(xform.ParentUid))
|
|
{
|
|
if (!PhysicsQuery.TryGetComponent(xform.ParentUid, out body) ||
|
|
!XformQuery.TryGetComponent(xform.ParentUid, out xformMover))
|
|
{
|
|
return;
|
|
}
|
|
|
|
physicsUid = xform.ParentUid;
|
|
}
|
|
else if (!PhysicsQuery.TryGetComponent(player, out body))
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Server-side should just be handled on its own so we'll just do this shizznit
|
|
HandleMobMovement(
|
|
player,
|
|
mover,
|
|
physicsUid,
|
|
body,
|
|
xformMover,
|
|
frameTime);
|
|
}
|
|
|
|
protected override bool CanSound()
|
|
{
|
|
return _timing is { IsFirstTimePredicted: true, InSimulation: true };
|
|
}
|
|
}
|
|
}
|