Popup duplicate stacking (#27978)

This commit is contained in:
ShadowCommander
2024-05-28 21:05:40 -07:00
committed by GitHub
parent 7f9ed797af
commit 49982acc88
2 changed files with 77 additions and 38 deletions

View File

@@ -1,18 +1,20 @@
using System.Linq; using System.Linq;
using Content.Shared.Containers;
using Content.Shared.Examine; using Content.Shared.Examine;
using Content.Shared.GameTicking; using Content.Shared.GameTicking;
using Content.Shared.Popups; using Content.Shared.Popups;
using JetBrains.Annotations;
using Robust.Client.Graphics; using Robust.Client.Graphics;
using Robust.Client.Input; using Robust.Client.Input;
using Robust.Client.Player; using Robust.Client.Player;
using Robust.Client.UserInterface; using Robust.Client.UserInterface;
using Robust.Shared.Collections;
using Robust.Shared.Configuration; using Robust.Shared.Configuration;
using Robust.Shared.Map; using Robust.Shared.Map;
using Robust.Shared.Player; using Robust.Shared.Player;
using Robust.Shared.Prototypes; using Robust.Shared.Prototypes;
using Robust.Shared.Replays; using Robust.Shared.Replays;
using Robust.Shared.Timing; using Robust.Shared.Timing;
using Robust.Shared.Utility;
namespace Content.Client.Popups namespace Content.Client.Popups
{ {
@@ -29,11 +31,11 @@ namespace Content.Client.Popups
[Dependency] private readonly ExamineSystemShared _examine = default!; [Dependency] private readonly ExamineSystemShared _examine = default!;
[Dependency] private readonly SharedTransformSystem _transform = default!; [Dependency] private readonly SharedTransformSystem _transform = default!;
public IReadOnlyList<WorldPopupLabel> WorldLabels => _aliveWorldLabels; public IReadOnlyCollection<WorldPopupLabel> WorldLabels => _aliveWorldLabels.Values;
public IReadOnlyList<CursorPopupLabel> CursorLabels => _aliveCursorLabels; public IReadOnlyCollection<CursorPopupLabel> CursorLabels => _aliveCursorLabels.Values;
private readonly List<WorldPopupLabel> _aliveWorldLabels = new(); private readonly Dictionary<WorldPopupData, WorldPopupLabel> _aliveWorldLabels = new();
private readonly List<CursorPopupLabel> _aliveCursorLabels = new(); private readonly Dictionary<CursorPopupData, CursorPopupLabel> _aliveCursorLabels = new();
public const float MinimumPopupLifetime = 0.7f; public const float MinimumPopupLifetime = 0.7f;
public const float MaximumPopupLifetime = 5f; public const float MaximumPopupLifetime = 5f;
@@ -65,6 +67,15 @@ namespace Content.Client.Popups
.RemoveOverlay<PopupOverlay>(); .RemoveOverlay<PopupOverlay>();
} }
private void WrapAndRepeatPopup(PopupLabel existingLabel, string popupMessage)
{
existingLabel.TotalTime = 0;
existingLabel.Repeats += 1;
existingLabel.Text = Loc.GetString("popup-system-repeated-popup-stacking-wrap",
("popup-message", popupMessage),
("count", existingLabel.Repeats));
}
private void PopupMessage(string? message, PopupType type, EntityCoordinates coordinates, EntityUid? entity, bool recordReplay) private void PopupMessage(string? message, PopupType type, EntityCoordinates coordinates, EntityUid? entity, bool recordReplay)
{ {
if (message == null) if (message == null)
@@ -78,13 +89,20 @@ namespace Content.Client.Popups
_replayRecording.RecordClientMessage(new PopupCoordinatesEvent(message, type, GetNetCoordinates(coordinates))); _replayRecording.RecordClientMessage(new PopupCoordinatesEvent(message, type, GetNetCoordinates(coordinates)));
} }
var popupData = new WorldPopupData(message, type, coordinates, entity);
if (_aliveWorldLabels.TryGetValue(popupData, out var existingLabel))
{
WrapAndRepeatPopup(existingLabel, popupData.Message);
return;
}
var label = new WorldPopupLabel(coordinates) var label = new WorldPopupLabel(coordinates)
{ {
Text = message, Text = message,
Type = type, Type = type,
}; };
_aliveWorldLabels.Add(label); _aliveWorldLabels.Add(popupData, label);
} }
#region Abstract Method Implementations #region Abstract Method Implementations
@@ -113,13 +131,20 @@ namespace Content.Client.Popups
if (recordReplay && _replayRecording.IsRecording) if (recordReplay && _replayRecording.IsRecording)
_replayRecording.RecordClientMessage(new PopupCursorEvent(message, type)); _replayRecording.RecordClientMessage(new PopupCursorEvent(message, type));
var popupData = new CursorPopupData(message, type);
if (_aliveCursorLabels.TryGetValue(popupData, out var existingLabel))
{
WrapAndRepeatPopup(existingLabel, popupData.Message);
return;
}
var label = new CursorPopupLabel(_inputManager.MouseScreenPosition) var label = new CursorPopupLabel(_inputManager.MouseScreenPosition)
{ {
Text = message, Text = message,
Type = type, Type = type,
}; };
_aliveCursorLabels.Add(label); _aliveCursorLabels.Add(popupData, label);
} }
public override void PopupCursor(string? message, PopupType type = PopupType.Small) public override void PopupCursor(string? message, PopupType type = PopupType.Small)
@@ -249,27 +274,37 @@ namespace Content.Client.Popups
if (_aliveWorldLabels.Count == 0 && _aliveCursorLabels.Count == 0) if (_aliveWorldLabels.Count == 0 && _aliveCursorLabels.Count == 0)
return; return;
for (var i = 0; i < _aliveWorldLabels.Count; i++) if (_aliveWorldLabels.Count > 0)
{
var aliveWorldToRemove = new ValueList<WorldPopupData>();
foreach (var (data, label) in _aliveWorldLabels)
{ {
var label = _aliveWorldLabels[i];
label.TotalTime += frameTime; label.TotalTime += frameTime;
if (label.TotalTime > GetPopupLifetime(label) || Deleted(label.InitialPos.EntityId)) if (label.TotalTime > GetPopupLifetime(label) || Deleted(label.InitialPos.EntityId))
{ {
_aliveWorldLabels.RemoveSwap(i); aliveWorldToRemove.Add(data);
i--;
} }
} }
foreach (var data in aliveWorldToRemove)
for (var i = 0; i < _aliveCursorLabels.Count; i++)
{ {
var label = _aliveCursorLabels[i]; _aliveWorldLabels.Remove(data);
label.TotalTime += frameTime; }
}
if (_aliveCursorLabels.Count > 0)
{
var aliveCursorToRemove = new ValueList<CursorPopupData>();
foreach (var (data, label) in _aliveCursorLabels)
{
label.TotalTime += frameTime;
if (label.TotalTime > GetPopupLifetime(label)) if (label.TotalTime > GetPopupLifetime(label))
{ {
_aliveCursorLabels.RemoveSwap(i); aliveCursorToRemove.Add(data);
i--; }
}
foreach (var data in aliveCursorToRemove)
{
_aliveCursorLabels.Remove(data);
} }
} }
} }
@@ -279,29 +314,32 @@ namespace Content.Client.Popups
public PopupType Type = PopupType.Small; public PopupType Type = PopupType.Small;
public string Text { get; set; } = string.Empty; public string Text { get; set; } = string.Empty;
public float TotalTime { get; set; } public float TotalTime { get; set; }
public int Repeats = 1;
} }
public sealed class CursorPopupLabel : PopupLabel public sealed class WorldPopupLabel(EntityCoordinates coordinates) : PopupLabel
{
public ScreenCoordinates InitialPos;
public CursorPopupLabel(ScreenCoordinates screenCoords)
{
InitialPos = screenCoords;
}
}
public sealed class WorldPopupLabel : PopupLabel
{ {
/// <summary> /// <summary>
/// The original EntityCoordinates of the label. /// The original EntityCoordinates of the label.
/// </summary> /// </summary>
public EntityCoordinates InitialPos; public EntityCoordinates InitialPos = coordinates;
}
public WorldPopupLabel(EntityCoordinates coordinates) public sealed class CursorPopupLabel(ScreenCoordinates screenCoords) : PopupLabel
{ {
InitialPos = coordinates; public ScreenCoordinates InitialPos = screenCoords;
} }
}
[UsedImplicitly]
private record struct WorldPopupData(
string Message,
PopupType Type,
EntityCoordinates Coordinates,
EntityUid? Entity);
[UsedImplicitly]
private record struct CursorPopupData(
string Message,
PopupType Type);
} }
} }

View File

@@ -0,0 +1 @@
popup-system-repeated-popup-stacking-wrap = {$popup-message} x{$count}