feat: properly perform predicted porta pottys (fix toilet prediction) (#39394)
This commit is contained in:
@@ -1,8 +0,0 @@
|
||||
using Content.Shared.Toilet.Systems;
|
||||
|
||||
namespace Content.Server.Toilet;
|
||||
|
||||
public sealed class ToiletSystem : SharedToiletSystem
|
||||
{
|
||||
|
||||
}
|
||||
@@ -2,10 +2,10 @@ using Robust.Shared.Audio;
|
||||
using Robust.Shared.GameStates;
|
||||
using Robust.Shared.Serialization;
|
||||
|
||||
namespace Content.Shared.Toilet.Components
|
||||
{
|
||||
namespace Content.Shared.Toilet.Components;
|
||||
|
||||
/// <summary>
|
||||
/// Toilets that can be flushed, seats toggled up and down, items hidden in cistern.
|
||||
/// Seats that can toggled up and down with visuals to match.
|
||||
/// </summary>
|
||||
[RegisterComponent, NetworkedComponent, AutoGenerateComponentState]
|
||||
public sealed partial class ToiletComponent : Component
|
||||
@@ -16,7 +16,6 @@ namespace Content.Shared.Toilet.Components
|
||||
[DataField, AutoNetworkedField]
|
||||
public bool ToggleSeat;
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Sound to play when toggling toilet seat.
|
||||
/// </summary>
|
||||
@@ -34,7 +33,5 @@ namespace Content.Shared.Toilet.Components
|
||||
public enum SeatVisualState : byte
|
||||
{
|
||||
SeatUp,
|
||||
SeatDown
|
||||
SeatDown,
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,109 +0,0 @@
|
||||
using Content.Shared.Buckle.Components;
|
||||
using Content.Shared.Interaction;
|
||||
using Content.Shared.Verbs;
|
||||
using Content.Shared.Plunger.Components;
|
||||
using Robust.Shared.Audio.Systems;
|
||||
using Robust.Shared.Random;
|
||||
using Robust.Shared.Utility;
|
||||
using Content.Shared.Toilet.Components;
|
||||
|
||||
namespace Content.Shared.Toilet.Systems
|
||||
{
|
||||
/// <summary>
|
||||
/// Handles sprite changes for both toilet seat up and down as well as for lid open and closed. Handles interactions with hidden stash
|
||||
/// </summary>
|
||||
|
||||
public abstract class SharedToiletSystem : EntitySystem
|
||||
{
|
||||
[Dependency] private readonly IRobustRandom _random = default!;
|
||||
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
||||
[Dependency] private readonly SharedAppearanceSystem _appearance = default!;
|
||||
|
||||
public override void Initialize()
|
||||
{
|
||||
base.Initialize();
|
||||
|
||||
SubscribeLocalEvent<ToiletComponent, MapInitEvent>(OnMapInit);
|
||||
SubscribeLocalEvent<ToiletComponent, GetVerbsEvent<AlternativeVerb>>(OnToggleSeatVerb);
|
||||
SubscribeLocalEvent<ToiletComponent, ActivateInWorldEvent>(OnActivateInWorld);
|
||||
}
|
||||
|
||||
private void OnMapInit(EntityUid uid, ToiletComponent component, MapInitEvent args)
|
||||
{
|
||||
if (_random.Prob(0.5f))
|
||||
component.ToggleSeat = true;
|
||||
|
||||
if (_random.Prob(0.3f))
|
||||
{
|
||||
TryComp<PlungerUseComponent>(uid, out var plunger);
|
||||
|
||||
if (plunger == null)
|
||||
return;
|
||||
|
||||
plunger.NeedsPlunger = true;
|
||||
}
|
||||
|
||||
UpdateAppearance(uid);
|
||||
Dirty(uid, component);
|
||||
}
|
||||
|
||||
public bool CanToggle(EntityUid uid)
|
||||
{
|
||||
return TryComp<StrapComponent>(uid, out var strap) && strap.BuckledEntities.Count == 0;
|
||||
}
|
||||
|
||||
private void OnToggleSeatVerb(EntityUid uid, ToiletComponent component, GetVerbsEvent<AlternativeVerb> args)
|
||||
{
|
||||
if (!args.CanInteract || !args.CanAccess || !CanToggle(uid) || args.Hands == null)
|
||||
return;
|
||||
|
||||
AlternativeVerb toggleVerb = new()
|
||||
{
|
||||
Act = () => ToggleToiletSeat(uid, args.User, component)
|
||||
};
|
||||
|
||||
if (component.ToggleSeat)
|
||||
{
|
||||
toggleVerb.Text = Loc.GetString("toilet-seat-close");
|
||||
toggleVerb.Icon =
|
||||
new SpriteSpecifier.Texture(new ResPath("/Textures/Interface/VerbIcons/close.svg.192dpi.png"));
|
||||
}
|
||||
else
|
||||
{
|
||||
toggleVerb.Text = Loc.GetString("toilet-seat-open");
|
||||
toggleVerb.Icon =
|
||||
new SpriteSpecifier.Texture(new ResPath("/Textures/Interface/VerbIcons/open.svg.192dpi.png"));
|
||||
}
|
||||
args.Verbs.Add(toggleVerb);
|
||||
}
|
||||
|
||||
private void OnActivateInWorld(EntityUid uid, ToiletComponent comp, ActivateInWorldEvent args)
|
||||
{
|
||||
if (args.Handled || !args.Complex)
|
||||
return;
|
||||
|
||||
args.Handled = true;
|
||||
ToggleToiletSeat(uid, args.User, comp);
|
||||
}
|
||||
|
||||
public void ToggleToiletSeat(EntityUid uid, EntityUid? user = null, ToiletComponent? component = null, MetaDataComponent? meta = null)
|
||||
{
|
||||
if (!Resolve(uid, ref component))
|
||||
return;
|
||||
|
||||
component.ToggleSeat = !component.ToggleSeat;
|
||||
|
||||
_audio.PlayPredicted(component.SeatSound, uid, uid);
|
||||
UpdateAppearance(uid, component);
|
||||
Dirty(uid, component, meta);
|
||||
}
|
||||
|
||||
private void UpdateAppearance(EntityUid uid, ToiletComponent? component = null)
|
||||
{
|
||||
if (!Resolve(uid, ref component))
|
||||
return;
|
||||
|
||||
_appearance.SetData(uid, ToiletVisuals.SeatVisualState, component.ToggleSeat ? SeatVisualState.SeatUp : SeatVisualState.SeatDown);
|
||||
}
|
||||
}
|
||||
}
|
||||
116
Content.Shared/Toilet/Systems/ToiletSystem.cs
Normal file
116
Content.Shared/Toilet/Systems/ToiletSystem.cs
Normal file
@@ -0,0 +1,116 @@
|
||||
using Content.Shared.Buckle.Components;
|
||||
using Content.Shared.Interaction;
|
||||
using Content.Shared.Verbs;
|
||||
using Content.Shared.Plunger.Components;
|
||||
using Robust.Shared.Audio.Systems;
|
||||
using Robust.Shared.Random;
|
||||
using Robust.Shared.Utility;
|
||||
using Content.Shared.Toilet.Components;
|
||||
|
||||
namespace Content.Shared.Toilet.Systems;
|
||||
|
||||
/// <summary>
|
||||
/// Handles sprite changes for both toilet seat up and down as well as for lid
|
||||
/// open and closed.
|
||||
/// </summary>
|
||||
public sealed class ToiletSystem : EntitySystem
|
||||
{
|
||||
[Dependency] private readonly IRobustRandom _random = default!;
|
||||
[Dependency] private readonly SharedAudioSystem _audio = default!;
|
||||
[Dependency] private readonly SharedAppearanceSystem _appearance = default!;
|
||||
|
||||
public override void Initialize()
|
||||
{
|
||||
base.Initialize();
|
||||
|
||||
SubscribeLocalEvent<ToiletComponent, MapInitEvent>(OnMapInit);
|
||||
SubscribeLocalEvent<ToiletComponent, GetVerbsEvent<AlternativeVerb>>(OnToggleSeatVerb);
|
||||
SubscribeLocalEvent<ToiletComponent, ActivateInWorldEvent>(OnActivateInWorld);
|
||||
}
|
||||
|
||||
private void OnMapInit(Entity<ToiletComponent> ent, ref MapInitEvent args)
|
||||
{
|
||||
if (_random.Prob(0.5f))
|
||||
{
|
||||
ent.Comp.ToggleSeat = true;
|
||||
Dirty(ent);
|
||||
}
|
||||
|
||||
if (_random.Prob(0.3f)
|
||||
&& TryComp<PlungerUseComponent>(ent, out var plunger))
|
||||
{
|
||||
plunger.NeedsPlunger = true;
|
||||
Dirty(ent, plunger);
|
||||
}
|
||||
|
||||
UpdateAppearance(ent);
|
||||
}
|
||||
|
||||
private void OnToggleSeatVerb(Entity<ToiletComponent> ent, ref GetVerbsEvent<AlternativeVerb> args)
|
||||
{
|
||||
if (!args.CanInteract || !args.CanAccess || args.Hands == null || !CanToggle(ent))
|
||||
return;
|
||||
|
||||
var user = args.User;
|
||||
AlternativeVerb toggleVerb = new() { Act = () => ToggleToiletSeat(ent.AsNullable(), user) };
|
||||
|
||||
if (ent.Comp.ToggleSeat)
|
||||
{
|
||||
toggleVerb.Text = Loc.GetString("toilet-seat-close");
|
||||
toggleVerb.Icon =
|
||||
new SpriteSpecifier.Texture(new ResPath("/Textures/Interface/VerbIcons/close.svg.192dpi.png"));
|
||||
}
|
||||
else
|
||||
{
|
||||
toggleVerb.Text = Loc.GetString("toilet-seat-open");
|
||||
toggleVerb.Icon =
|
||||
new SpriteSpecifier.Texture(new ResPath("/Textures/Interface/VerbIcons/open.svg.192dpi.png"));
|
||||
}
|
||||
args.Verbs.Add(toggleVerb);
|
||||
}
|
||||
|
||||
private void OnActivateInWorld(Entity<ToiletComponent> ent, ref ActivateInWorldEvent args)
|
||||
{
|
||||
if (args.Handled || !args.Complex)
|
||||
return;
|
||||
|
||||
args.Handled = true;
|
||||
ToggleToiletSeat(ent.AsNullable(), args.User);
|
||||
}
|
||||
|
||||
private void UpdateAppearance(Entity<ToiletComponent> ent)
|
||||
{
|
||||
_appearance.SetData(ent,
|
||||
ToiletVisuals.SeatVisualState,
|
||||
ent.Comp.ToggleSeat ? SeatVisualState.SeatUp : SeatVisualState.SeatDown);
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Toggles a toilet's seat. Yup. Doesn't check if anyone is on the seat.
|
||||
/// </summary>
|
||||
/// <param name="ent">The toilet being seat-toggled.</param>
|
||||
/// <param name="user">The user doing the toggling; used for predicted audio.</param>
|
||||
/// <seealso cref="CanToggle" />
|
||||
public void ToggleToiletSeat(Entity<ToiletComponent?> ent, EntityUid? user = null)
|
||||
{
|
||||
if (!Resolve(ent, ref ent.Comp))
|
||||
return;
|
||||
|
||||
ent.Comp.ToggleSeat = !ent.Comp.ToggleSeat;
|
||||
|
||||
_audio.PlayPredicted(ent.Comp.SeatSound, ent, user);
|
||||
UpdateAppearance((ent, ent.Comp));
|
||||
Dirty(ent);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Whether or not a toilet seat can be toggled without phasing through
|
||||
/// someone's back. (That is, no one is seated on it.)
|
||||
/// </summary>
|
||||
/// <seealso cref="ToggleToiletSeat" />
|
||||
public bool CanToggle(EntityUid uid)
|
||||
{
|
||||
return TryComp<StrapComponent>(uid, out var strap) && strap.BuckledEntities.Count == 0;
|
||||
}
|
||||
}
|
||||
@@ -75,6 +75,9 @@
|
||||
autoDrain: false
|
||||
- type: StaticPrice
|
||||
price: 100
|
||||
- type: ActivatableUI
|
||||
key: enum.DisposalUnitUiKey.Key
|
||||
verbOnly: true # Not strictly needed, but we want E to toggle lid
|
||||
- type: UserInterface
|
||||
interfaces:
|
||||
enum.DisposalUnitUiKey.Key:
|
||||
|
||||
Reference in New Issue
Block a user