Multi-viewport popups (#11591)
This commit is contained in:
119
Content.Client/Popups/PopupOverlay.cs
Normal file
119
Content.Client/Popups/PopupOverlay.cs
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
using Content.Shared.Popups;
|
||||||
|
using Robust.Client.Graphics;
|
||||||
|
using Robust.Client.ResourceManagement;
|
||||||
|
using Robust.Shared.Enums;
|
||||||
|
using Robust.Shared.Prototypes;
|
||||||
|
|
||||||
|
namespace Content.Client.Popups;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Draws popup text, either in world or on screen.
|
||||||
|
/// </summary>
|
||||||
|
public sealed class PopupOverlay : Overlay
|
||||||
|
{
|
||||||
|
private readonly IEntityManager _entManager;
|
||||||
|
private readonly PopupSystem _popup;
|
||||||
|
|
||||||
|
private readonly ShaderInstance _shader;
|
||||||
|
private readonly Font _smallFont;
|
||||||
|
private readonly Font _mediumFont;
|
||||||
|
private readonly Font _largeFont;
|
||||||
|
|
||||||
|
public override OverlaySpace Space => OverlaySpace.ScreenSpace;
|
||||||
|
|
||||||
|
public PopupOverlay(IEntityManager entManager, IPrototypeManager protoManager, IResourceCache cache, PopupSystem popup)
|
||||||
|
{
|
||||||
|
_entManager = entManager;
|
||||||
|
_popup = popup;
|
||||||
|
|
||||||
|
_shader = protoManager.Index<ShaderPrototype>("unshaded").Instance();
|
||||||
|
_smallFont = new VectorFont(cache.GetResource<FontResource>("/Fonts/NotoSans/NotoSans-Italic.ttf"), 10);
|
||||||
|
_mediumFont = new VectorFont(cache.GetResource<FontResource>("/Fonts/NotoSans/NotoSans-Italic.ttf"), 12);
|
||||||
|
_largeFont = new VectorFont(cache.GetResource<FontResource>("/Fonts/NotoSans/NotoSans-BoldItalic.ttf"), 14);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Draw(in OverlayDrawArgs args)
|
||||||
|
{
|
||||||
|
if (args.ViewportControl == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
args.DrawingHandle.SetTransform(Matrix3.Identity);
|
||||||
|
args.DrawingHandle.UseShader(_shader);
|
||||||
|
|
||||||
|
DrawWorld(args.ScreenHandle, args);
|
||||||
|
DrawScreen(args.ScreenHandle, args);
|
||||||
|
|
||||||
|
args.DrawingHandle.UseShader(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DrawWorld(DrawingHandleScreen worldHandle, OverlayDrawArgs args)
|
||||||
|
{
|
||||||
|
if (_popup.WorldLabels.Count == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var matrix = args.ViewportControl!.GetWorldToScreenMatrix();
|
||||||
|
|
||||||
|
foreach (var popup in _popup.WorldLabels)
|
||||||
|
{
|
||||||
|
var mapPos = popup.InitialPos.ToMap(_entManager);
|
||||||
|
|
||||||
|
if (mapPos.MapId != args.MapId)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!args.WorldAABB.Contains(mapPos.Position))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
var pos = matrix.Transform(mapPos.Position);
|
||||||
|
DrawPopup(popup, worldHandle, pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DrawScreen(DrawingHandleScreen screenHandle, OverlayDrawArgs args)
|
||||||
|
{
|
||||||
|
foreach (var popup in _popup.CursorLabels)
|
||||||
|
{
|
||||||
|
// Different window
|
||||||
|
if (popup.InitialPos.Window != args.ViewportControl?.Window?.Id)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
DrawPopup(popup, screenHandle, popup.InitialPos.Position);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DrawPopup(PopupSystem.PopupLabel popup, DrawingHandleScreen handle, Vector2 position)
|
||||||
|
{
|
||||||
|
const float alphaMinimum = 0.5f;
|
||||||
|
|
||||||
|
var alpha = MathF.Min(1f, 1f - (popup.TotalTime - alphaMinimum) / (PopupSystem.PopupLifetime - alphaMinimum));
|
||||||
|
var updatedPosition = position - new Vector2(0f, 20f * (popup.TotalTime * popup.TotalTime + popup.TotalTime));
|
||||||
|
var font = _smallFont;
|
||||||
|
var dimensions = Vector2.Zero;
|
||||||
|
var color = Color.White.WithAlpha(alpha);
|
||||||
|
|
||||||
|
switch (popup.Type)
|
||||||
|
{
|
||||||
|
case PopupType.SmallCaution:
|
||||||
|
color = Color.Red;
|
||||||
|
break;
|
||||||
|
case PopupType.Medium:
|
||||||
|
font = _mediumFont;
|
||||||
|
color = Color.LightGray;
|
||||||
|
break;
|
||||||
|
case PopupType.MediumCaution:
|
||||||
|
font = _mediumFont;
|
||||||
|
color = Color.Red;
|
||||||
|
break;
|
||||||
|
case PopupType.Large:
|
||||||
|
font = _largeFont;
|
||||||
|
color = Color.LightGray;
|
||||||
|
break;
|
||||||
|
case PopupType.LargeCaution:
|
||||||
|
font = _largeFont;
|
||||||
|
color = Color.Red;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
dimensions = handle.GetDimensions(font, popup.Text, 1f);
|
||||||
|
handle.DrawString(font, updatedPosition - dimensions / 2f, popup.Text, color.WithAlpha(alpha));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,16 +1,13 @@
|
|||||||
using Content.Client.Stylesheets;
|
|
||||||
using Content.Shared.Examine;
|
|
||||||
using Content.Shared.GameTicking;
|
using Content.Shared.GameTicking;
|
||||||
using Content.Shared.Popups;
|
using Content.Shared.Popups;
|
||||||
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.ResourceManagement;
|
||||||
using Robust.Client.UserInterface.Controls;
|
|
||||||
using Robust.Shared.Map;
|
using Robust.Shared.Map;
|
||||||
using Robust.Shared.Player;
|
using Robust.Shared.Player;
|
||||||
using Robust.Shared.Players;
|
using Robust.Shared.Players;
|
||||||
using Robust.Shared.Timing;
|
using Robust.Shared.Prototypes;
|
||||||
using Robust.Shared.Utility;
|
using Robust.Shared.Utility;
|
||||||
|
|
||||||
namespace Content.Client.Popups
|
namespace Content.Client.Popups
|
||||||
@@ -18,10 +15,13 @@ namespace Content.Client.Popups
|
|||||||
public sealed class PopupSystem : SharedPopupSystem
|
public sealed class PopupSystem : SharedPopupSystem
|
||||||
{
|
{
|
||||||
[Dependency] private readonly IInputManager _inputManager = default!;
|
[Dependency] private readonly IInputManager _inputManager = default!;
|
||||||
[Dependency] private readonly IUserInterfaceManager _userInterfaceManager = default!;
|
[Dependency] private readonly IOverlayManager _overlay = default!;
|
||||||
[Dependency] private readonly IEyeManager _eyeManager = default!;
|
|
||||||
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
||||||
[Dependency] private readonly ExamineSystemShared _examineSystem = default!;
|
[Dependency] private readonly IPrototypeManager _prototype = default!;
|
||||||
|
[Dependency] private readonly IResourceCache _resource = default!;
|
||||||
|
|
||||||
|
public IReadOnlyList<WorldPopupLabel> WorldLabels => _aliveWorldLabels;
|
||||||
|
public IReadOnlyList<CursorPopupLabel> CursorLabels => _aliveCursorLabels;
|
||||||
|
|
||||||
private readonly List<WorldPopupLabel> _aliveWorldLabels = new();
|
private readonly List<WorldPopupLabel> _aliveWorldLabels = new();
|
||||||
private readonly List<CursorPopupLabel> _aliveCursorLabels = new();
|
private readonly List<CursorPopupLabel> _aliveCursorLabels = new();
|
||||||
@@ -34,21 +34,25 @@ namespace Content.Client.Popups
|
|||||||
SubscribeNetworkEvent<PopupCoordinatesEvent>(OnPopupCoordinatesEvent);
|
SubscribeNetworkEvent<PopupCoordinatesEvent>(OnPopupCoordinatesEvent);
|
||||||
SubscribeNetworkEvent<PopupEntityEvent>(OnPopupEntityEvent);
|
SubscribeNetworkEvent<PopupEntityEvent>(OnPopupEntityEvent);
|
||||||
SubscribeNetworkEvent<RoundRestartCleanupEvent>(OnRoundRestart);
|
SubscribeNetworkEvent<RoundRestartCleanupEvent>(OnRoundRestart);
|
||||||
|
_overlay
|
||||||
|
.AddOverlay(new PopupOverlay(EntityManager, _prototype, _resource, this));
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void Shutdown()
|
||||||
|
{
|
||||||
|
base.Shutdown();
|
||||||
|
_overlay
|
||||||
|
.RemoveOverlay<PopupOverlay>();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void PopupMessage(string message, PopupType type, EntityCoordinates coordinates, EntityUid? entity = null)
|
private void PopupMessage(string message, PopupType type, EntityCoordinates coordinates, EntityUid? entity = null)
|
||||||
{
|
{
|
||||||
var label = new WorldPopupLabel(_eyeManager, EntityManager)
|
var label = new WorldPopupLabel(coordinates)
|
||||||
{
|
{
|
||||||
Entity = entity,
|
|
||||||
Text = message,
|
Text = message,
|
||||||
StyleClasses = { GetStyleClass(type) },
|
Type = type,
|
||||||
};
|
};
|
||||||
|
|
||||||
_userInterfaceManager.PopupRoot.AddChild(label);
|
|
||||||
label.Measure(Vector2.Infinity);
|
|
||||||
|
|
||||||
label.InitialPos = coordinates;
|
|
||||||
_aliveWorldLabels.Add(label);
|
_aliveWorldLabels.Add(label);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,9 +79,9 @@ namespace Content.Client.Popups
|
|||||||
var label = new CursorPopupLabel(_inputManager.MouseScreenPosition)
|
var label = new CursorPopupLabel(_inputManager.MouseScreenPosition)
|
||||||
{
|
{
|
||||||
Text = message,
|
Text = message,
|
||||||
StyleClasses = { GetStyleClass(type) },
|
Type = type,
|
||||||
};
|
};
|
||||||
_userInterfaceManager.PopupRoot.AddChild(label);
|
|
||||||
_aliveCursorLabels.Add(label);
|
_aliveCursorLabels.Add(label);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -146,159 +150,69 @@ namespace Content.Client.Popups
|
|||||||
|
|
||||||
private void OnRoundRestart(RoundRestartCleanupEvent ev)
|
private void OnRoundRestart(RoundRestartCleanupEvent ev)
|
||||||
{
|
{
|
||||||
foreach (var label in _aliveWorldLabels)
|
_aliveCursorLabels.Clear();
|
||||||
{
|
|
||||||
label.Dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
_aliveWorldLabels.Clear();
|
_aliveWorldLabels.Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
private static string GetStyleClass(PopupType type) =>
|
|
||||||
type switch
|
|
||||||
{
|
|
||||||
PopupType.Small => StyleNano.StyleClassPopupMessageSmall,
|
|
||||||
PopupType.SmallCaution => StyleNano.StyleClassPopupMessageSmallCaution,
|
|
||||||
PopupType.Medium => StyleNano.StyleClassPopupMessageMedium,
|
|
||||||
PopupType.MediumCaution => StyleNano.StyleClassPopupMessageMediumCaution,
|
|
||||||
PopupType.Large => StyleNano.StyleClassPopupMessageLarge,
|
|
||||||
PopupType.LargeCaution => StyleNano.StyleClassPopupMessageLargeCaution,
|
|
||||||
_ => StyleNano.StyleClassPopupMessageSmall
|
|
||||||
};
|
|
||||||
|
|
||||||
public override void FrameUpdate(float frameTime)
|
public override void FrameUpdate(float frameTime)
|
||||||
{
|
{
|
||||||
if (_aliveWorldLabels.Count == 0 && _aliveCursorLabels.Count == 0)
|
if (_aliveWorldLabels.Count == 0 && _aliveCursorLabels.Count == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var player = _playerManager.LocalPlayer?.ControlledEntity;
|
for (var i = 0; i < _aliveWorldLabels.Count; i++)
|
||||||
var playerPos = player != null ? Transform(player.Value).MapPosition : MapCoordinates.Nullspace;
|
|
||||||
|
|
||||||
// ReSharper disable once ConvertToLocalFunction
|
|
||||||
var predicate = static (EntityUid uid, (EntityUid? compOwner, EntityUid? attachedEntity) data)
|
|
||||||
=> uid == data.compOwner || uid == data.attachedEntity;
|
|
||||||
var occluded = player != null && _examineSystem.IsOccluded(player.Value);
|
|
||||||
|
|
||||||
for (var i = _aliveWorldLabels.Count - 1; i >= 0; i--)
|
|
||||||
{
|
{
|
||||||
var label = _aliveWorldLabels[i];
|
var label = _aliveWorldLabels[i];
|
||||||
if (label.TotalTime > PopupLifetime ||
|
label.TotalTime += frameTime;
|
||||||
label.Entity != null && Deleted(label.Entity))
|
|
||||||
|
if (label.TotalTime > PopupLifetime || Deleted(label.InitialPos.EntityId))
|
||||||
{
|
{
|
||||||
label.Dispose();
|
|
||||||
_aliveWorldLabels.RemoveSwap(i);
|
_aliveWorldLabels.RemoveSwap(i);
|
||||||
continue;
|
i--;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (label.Entity == player)
|
for (var i = 0; i < _aliveCursorLabels.Count; i++)
|
||||||
{
|
|
||||||
label.Visible = true;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
var otherPos = label.Entity != null ? Transform(label.Entity.Value).MapPosition : label.InitialPos.ToMap(EntityManager);
|
|
||||||
|
|
||||||
if (occluded && !ExamineSystemShared.InRangeUnOccluded(
|
|
||||||
playerPos,
|
|
||||||
otherPos, 0f,
|
|
||||||
(label.Entity, player), predicate))
|
|
||||||
{
|
|
||||||
label.Visible = false;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
label.Visible = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (var i = _aliveCursorLabels.Count - 1; i >= 0; i--)
|
|
||||||
{
|
{
|
||||||
var label = _aliveCursorLabels[i];
|
var label = _aliveCursorLabels[i];
|
||||||
|
label.TotalTime += frameTime;
|
||||||
|
|
||||||
if (label.TotalTime > PopupLifetime)
|
if (label.TotalTime > PopupLifetime)
|
||||||
{
|
{
|
||||||
label.Dispose();
|
|
||||||
_aliveCursorLabels.RemoveSwap(i);
|
_aliveCursorLabels.RemoveSwap(i);
|
||||||
|
i--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private abstract class PopupLabel : Label
|
public abstract class PopupLabel
|
||||||
{
|
{
|
||||||
public float TotalTime { get; protected set; }
|
public PopupType Type = PopupType.Small;
|
||||||
|
public string Text { get; set; } = string.Empty;
|
||||||
public PopupLabel()
|
public float TotalTime { get; set; }
|
||||||
{
|
|
||||||
ShadowOffsetXOverride = ShadowOffsetYOverride = 1;
|
|
||||||
FontColorShadowOverride = Color.Black;
|
|
||||||
Measure(Vector2.Infinity);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void FrameUpdate(FrameEventArgs eventArgs)
|
public sealed class CursorPopupLabel : PopupLabel
|
||||||
{
|
{
|
||||||
TotalTime += eventArgs.DeltaSeconds;
|
public ScreenCoordinates InitialPos;
|
||||||
if (TotalTime > 0.5f)
|
|
||||||
Modulate = Color.White.WithAlpha(1f - 0.2f * MathF.Pow(TotalTime - 0.5f, 3f));
|
public CursorPopupLabel(ScreenCoordinates screenCoords)
|
||||||
|
{
|
||||||
|
InitialPos = screenCoords;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private sealed class CursorPopupLabel : PopupLabel
|
public sealed class WorldPopupLabel : PopupLabel
|
||||||
{
|
{
|
||||||
public Vector2 InitialPos { get; set; }
|
|
||||||
|
|
||||||
public CursorPopupLabel(ScreenCoordinates screenCoords) : base()
|
|
||||||
{
|
|
||||||
InitialPos = screenCoords.Position - DesiredSize / 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void FrameUpdate(FrameEventArgs eventArgs)
|
|
||||||
{
|
|
||||||
base.FrameUpdate(eventArgs);
|
|
||||||
LayoutContainer.SetPosition(this, InitialPos / UIScale - (0, 20 * (TotalTime * TotalTime + TotalTime)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private sealed class WorldPopupLabel : PopupLabel
|
|
||||||
{
|
|
||||||
private readonly IEyeManager _eyeManager;
|
|
||||||
private readonly IEntityManager _entityManager;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The original Mapid and ScreenPosition of the label.
|
/// The original EntityCoordinates of the label.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <remarks>
|
public EntityCoordinates InitialPos;
|
||||||
/// Yes that's right it's not technically MapCoordinates.
|
|
||||||
/// </remarks>
|
|
||||||
public EntityCoordinates InitialPos { get; set; }
|
|
||||||
public EntityUid? Entity { get; set; }
|
|
||||||
|
|
||||||
public WorldPopupLabel(IEyeManager eyeManager, IEntityManager entityManager) : base()
|
public WorldPopupLabel(EntityCoordinates coordinates)
|
||||||
{
|
{
|
||||||
_eyeManager = eyeManager;
|
InitialPos = coordinates;
|
||||||
_entityManager = entityManager;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void FrameUpdate(FrameEventArgs eventArgs)
|
|
||||||
{
|
|
||||||
base.FrameUpdate(eventArgs);
|
|
||||||
ScreenCoordinates screenCoords;
|
|
||||||
|
|
||||||
if (Entity == null)
|
|
||||||
screenCoords = _eyeManager.CoordinatesToScreen(InitialPos);
|
|
||||||
else if (_entityManager.TryGetComponent(Entity.Value, out TransformComponent? xform)
|
|
||||||
&& xform.MapID == _eyeManager.CurrentMap)
|
|
||||||
screenCoords = _eyeManager.CoordinatesToScreen(xform.Coordinates);
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Visible = false;
|
|
||||||
if (Entity != null && _entityManager.Deleted(Entity))
|
|
||||||
TotalTime += PopupLifetime;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Visible = true;
|
|
||||||
var position = screenCoords.Position / UIScale - DesiredSize / 2;
|
|
||||||
LayoutContainer.SetPosition(this, position - (0, 20 * (TotalTime * TotalTime + TotalTime)));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1031,49 +1031,6 @@ namespace Content.Client.Stylesheets
|
|||||||
new StyleProperty("font", notoSans16)
|
new StyleProperty("font", notoSans16)
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Popup messages
|
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPopupMessageSmall}, null, null),
|
|
||||||
new[]
|
|
||||||
{
|
|
||||||
new StyleProperty("font", notoSansItalic10),
|
|
||||||
new StyleProperty("font-color", Color.White),
|
|
||||||
}),
|
|
||||||
|
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPopupMessageSmallCaution}, null, null),
|
|
||||||
new[]
|
|
||||||
{
|
|
||||||
new StyleProperty("font", notoSansItalic10),
|
|
||||||
new StyleProperty("font-color", Color.Red),
|
|
||||||
}),
|
|
||||||
|
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPopupMessageMedium}, null, null),
|
|
||||||
new[]
|
|
||||||
{
|
|
||||||
new StyleProperty("font", notoSansItalic12),
|
|
||||||
new StyleProperty("font-color", Color.LightGray),
|
|
||||||
}),
|
|
||||||
|
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPopupMessageMediumCaution}, null, null),
|
|
||||||
new[]
|
|
||||||
{
|
|
||||||
new StyleProperty("font", notoSansItalic12),
|
|
||||||
new StyleProperty("font-color", Color.Red),
|
|
||||||
}),
|
|
||||||
|
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPopupMessageLarge}, null, null),
|
|
||||||
new[]
|
|
||||||
{
|
|
||||||
new StyleProperty("font", notoSansBoldItalic14),
|
|
||||||
new StyleProperty("font-color", Color.LightGray),
|
|
||||||
}),
|
|
||||||
|
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPopupMessageLargeCaution}, null, null),
|
|
||||||
new[]
|
|
||||||
{
|
|
||||||
new StyleProperty("font", notoSansBoldItalic14),
|
|
||||||
new StyleProperty("font-color", Color.Red),
|
|
||||||
}),
|
|
||||||
|
|
||||||
//APC and SMES power state label colors
|
//APC and SMES power state label colors
|
||||||
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPowerStateNone}, null, null), new[]
|
new StyleRule(new SelectorElement(typeof(Label), new[] {StyleClassPowerStateNone}, null, null), new[]
|
||||||
{
|
{
|
||||||
|
|||||||
Reference in New Issue
Block a user