forked from space-syndicate/space-station-14
Signed-off-by: Prole <172158352+Prole0@users.noreply.github.com> Co-authored-by: PJBot <pieterjan.briers+bot@gmail.com> Co-authored-by: ScarKy0 <106310278+ScarKy0@users.noreply.github.com> Co-authored-by: Samuka-C <47865393+Samuka-C@users.noreply.github.com> Co-authored-by: ArtisticRoomba <145879011+ArtisticRoomba@users.noreply.github.com> Co-authored-by: Partmedia <kevinz5000@gmail.com> Co-authored-by: slarticodefast <161409025+slarticodefast@users.noreply.github.com> Co-authored-by: themias <89101928+themias@users.noreply.github.com> Co-authored-by: Victor Shen <71985089+Vexerot@users.noreply.github.com> Co-authored-by: Ed <96445749+TheShuEd@users.noreply.github.com> Co-authored-by: Milon <milonpl.git@proton.me> Co-authored-by: Kirus59 <145689588+Kirus59@users.noreply.github.com> Co-authored-by: Nemanja <98561806+EmoGarbage404@users.noreply.github.com> Co-authored-by: Stomf <5dorkydorks@gmail.com> Co-authored-by: drakewill-CRL <46307022+drakewill-CRL@users.noreply.github.com> Co-authored-by: PraxisMapper <praxismapper@gmail.com> Co-authored-by: EmoGarbage404 <retron404@gmail.com> Co-authored-by: lzk <124214523+lzk228@users.noreply.github.com> Co-authored-by: Princess Cheeseballs <66055347+Pronana@users.noreply.github.com> Co-authored-by: IProduceWidgets <107586145+IProduceWidgets@users.noreply.github.com> Co-authored-by: TytosB <54259736+TytosB@users.noreply.github.com> Co-authored-by: abadaba695 <spacestation13thingy@gmail.com> Co-authored-by: kosticia <kosticia46@gmail.com> Co-authored-by: Thinbug <101073555+Thinbug0@users.noreply.github.com> Co-authored-by: pathetic meowmeow <uhhadd@gmail.com> Co-authored-by: SlamBamActionman <83650252+SlamBamActionman@users.noreply.github.com> Co-authored-by: Boaz1111 <149967078+Boaz1111@users.noreply.github.com> Co-authored-by: ActiveMammmoth <140334666+ActiveMammmoth@users.noreply.github.com> Co-authored-by: Myra <vasilis@pikachu.systems> Co-authored-by: Whatstone <166147148+whatston3@users.noreply.github.com> Co-authored-by: Pieter-Jan Briers <pieterjan.briers+git@gmail.com> Co-authored-by: K-Dynamic <20566341+K-Dynamic@users.noreply.github.com> Co-authored-by: Gentleman-Bird <dcgreen406@gmail.com> Co-authored-by: metalgearsloth <31366439+metalgearsloth@users.noreply.github.com> Co-authored-by: BIGZi0348 <svalker0348@gmail.com> Co-authored-by: LaCumbiaDelCoronavirus <90893484+LaCumbiaDelCoronavirus@users.noreply.github.com> Co-authored-by: imatsoup <93290208+imatsoup@users.noreply.github.com> Co-authored-by: Matthew Herber <32679887+happyrobot33@users.noreply.github.com> Co-authored-by: Ertanic <36124833+Ertanic@users.noreply.github.com> Co-authored-by: MissKay1994 <15877268+MissKay1994@users.noreply.github.com> Co-authored-by: Errant <35878406+Errant-4@users.noreply.github.com> Co-authored-by: eoineoineoin <helloworld@eoinrul.es> Co-authored-by: Tiniest Shark <head.rebel@yahoo.com> Co-authored-by: nikitosych <boriszyn@gmail.com> Co-authored-by: Tayrtahn <tayrtahn@gmail.com> Co-authored-by: Perry Fraser <perryprog@users.noreply.github.com> Co-authored-by: YoungThug <ramialanbagy@gmail.com> Co-authored-by: beck-thompson <107373427+beck-thompson@users.noreply.github.com> Co-authored-by: metalgearsloth <comedian_vs_clown@hotmail.com> Co-authored-by: Southbridge <7013162+southbridge-fur@users.noreply.github.com> Co-authored-by: Vladislav Suchkov <20380250+murolem@users.noreply.github.com> Co-authored-by: Prole <172158352+Prole0@users.noreply.github.com> Co-authored-by: Unkn0wn_Gh0st <shadowstalkermll@gmail.com> Co-authored-by: 3nderall <101940324+3nderall@users.noreply.github.com> Co-authored-by: Radezolid <snappednexus@gmail.com> Co-authored-by: J <billsmith116@gmail.com> Co-authored-by: Ghagliiarghii <68826635+Ghagliiarghii@users.noreply.github.com> Co-authored-by: chromiumboy <50505512+chromiumboy@users.noreply.github.com> Co-authored-by: youtissoum <51883137+youtissoum@users.noreply.github.com> Co-authored-by: Minemoder5000 <minemoder50000@gmail.com> Co-authored-by: Spanky <scott@wearejacob.com> Co-authored-by: Spessmann <156740760+Spessmann@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: brainfood1183 <113240905+brainfood1183@users.noreply.github.com> Co-authored-by: Deerstop <edainturner@gmail.com> Co-authored-by: B_Kirill <153602297+B-Kirill@users.noreply.github.com> Co-authored-by: archee1 <archee3@hotmail.co.uk> Co-authored-by: Cojoke <83733158+Cojoke-dot@users.noreply.github.com> Co-authored-by: Quantum-cross <7065792+Quantum-cross@users.noreply.github.com> Co-authored-by: poklj <compgeek223@gmail.com> Co-authored-by: Krunklehorn <42424291+Krunklehorn@users.noreply.github.com> Co-authored-by: OnyxTheBrave <131422822+OnyxTheBrave@users.noreply.github.com> Co-authored-by: UpAndLeaves <92269094+Alpha-Two@users.noreply.github.com> Co-authored-by: Flareguy <78941145+Flareguy@users.noreply.github.com> Co-authored-by: Zalycon <84675130+Zalycon@users.noreply.github.com> Co-authored-by: deltanedas <39013340+deltanedas@users.noreply.github.com> Co-authored-by: Verm <32827189+Vermidia@users.noreply.github.com> Co-authored-by: nikthechampiongr <32041239+nikthechampiongr@users.noreply.github.com> Co-authored-by: ScarKy0 <scarky0@onet.eu> Co-authored-by: Dmitry <57028746+dimm00n@users.noreply.github.com>
262 lines
12 KiB
C#
262 lines
12 KiB
C#
using System.Linq;
|
|
using System.Numerics;
|
|
using Content.Client.Administration.Systems;
|
|
using Content.Client.Stylesheets;
|
|
using Content.Shared.Administration;
|
|
using Content.Shared.CCVar;
|
|
using Content.Shared.Ghost;
|
|
using Content.Shared.Mind;
|
|
using Content.Shared.Roles;
|
|
using Robust.Client.Graphics;
|
|
using Robust.Client.ResourceManagement;
|
|
using Robust.Client.UserInterface;
|
|
using Robust.Shared.Configuration;
|
|
using Robust.Shared.Enums;
|
|
using Robust.Shared.Prototypes;
|
|
|
|
namespace Content.Client.Administration;
|
|
|
|
internal sealed class AdminNameOverlay : Overlay
|
|
{
|
|
private readonly AdminSystem _system;
|
|
private readonly IEntityManager _entityManager;
|
|
private readonly IEyeManager _eyeManager;
|
|
private readonly EntityLookupSystem _entityLookup;
|
|
private readonly IUserInterfaceManager _userInterfaceManager;
|
|
private readonly SharedRoleSystem _roles;
|
|
private readonly Font _font;
|
|
private readonly Font _fontBold;
|
|
private AdminOverlayAntagFormat _overlayFormat;
|
|
private AdminOverlayAntagSymbolStyle _overlaySymbolStyle;
|
|
private bool _overlayPlaytime;
|
|
private bool _overlayStartingJob;
|
|
private float _ghostFadeDistance;
|
|
private float _ghostHideDistance;
|
|
private int _overlayStackMax;
|
|
private float _overlayMergeDistance;
|
|
|
|
//TODO make this adjustable via GUI?
|
|
private readonly ProtoId<RoleTypePrototype>[] _filter =
|
|
["SoloAntagonist", "TeamAntagonist", "SiliconAntagonist", "FreeAgent"];
|
|
|
|
private readonly string _antagLabelClassic = Loc.GetString("admin-overlay-antag-classic");
|
|
|
|
public AdminNameOverlay(
|
|
AdminSystem system,
|
|
IEntityManager entityManager,
|
|
IEyeManager eyeManager,
|
|
IResourceCache resourceCache,
|
|
EntityLookupSystem entityLookup,
|
|
IUserInterfaceManager userInterfaceManager,
|
|
IConfigurationManager config,
|
|
SharedRoleSystem roles)
|
|
{
|
|
_system = system;
|
|
_entityManager = entityManager;
|
|
_eyeManager = eyeManager;
|
|
_entityLookup = entityLookup;
|
|
_userInterfaceManager = userInterfaceManager;
|
|
_roles = roles;
|
|
ZIndex = 200;
|
|
// Setting these to a specific ttf would break the antag symbols
|
|
_font = resourceCache.NotoStack();
|
|
_fontBold = resourceCache.NotoStack(variation: "Bold");
|
|
|
|
config.OnValueChanged(CCVars.AdminOverlayAntagFormat, (show) => { _overlayFormat = UpdateOverlayFormat(show); }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlaySymbolStyle, (show) => { _overlaySymbolStyle = UpdateOverlaySymbolStyle(show); }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlayPlaytime, (show) => { _overlayPlaytime = show; }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlayStartingJob, (show) => { _overlayStartingJob = show; }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlayGhostHideDistance, (f) => { _ghostHideDistance = f; }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlayGhostFadeDistance, (f) => { _ghostFadeDistance = f; }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlayStackMax, (i) => { _overlayStackMax = i; }, true);
|
|
config.OnValueChanged(CCVars.AdminOverlayMergeDistance, (f) => { _overlayMergeDistance = f; }, true);
|
|
}
|
|
|
|
private AdminOverlayAntagFormat UpdateOverlayFormat(string formatString)
|
|
{
|
|
if (!Enum.TryParse<AdminOverlayAntagFormat>(formatString, out var format))
|
|
format = AdminOverlayAntagFormat.Binary;
|
|
|
|
return format;
|
|
}
|
|
|
|
private AdminOverlayAntagSymbolStyle UpdateOverlaySymbolStyle(string symbolString)
|
|
{
|
|
if (!Enum.TryParse<AdminOverlayAntagSymbolStyle>(symbolString, out var symbolStyle))
|
|
symbolStyle = AdminOverlayAntagSymbolStyle.Off;
|
|
|
|
return symbolStyle;
|
|
}
|
|
|
|
public override OverlaySpace Space => OverlaySpace.ScreenSpace;
|
|
|
|
protected override void Draw(in OverlayDrawArgs args)
|
|
{
|
|
var viewport = args.WorldAABB;
|
|
var colorDisconnected = Color.White;
|
|
var uiScale = _userInterfaceManager.RootControl.UIScale;
|
|
var lineoffset = new Vector2(0f, 14f) * uiScale;
|
|
var drawnOverlays = new List<(Vector2,Vector2)>() ; // A saved list of the overlays already drawn
|
|
|
|
// Get all player positions before drawing overlays, so they can be sorted before iteration
|
|
var sortable = new List<(PlayerInfo, Box2, EntityUid, Vector2)>();
|
|
foreach (var info in _system.PlayerList)
|
|
{
|
|
var entity = _entityManager.GetEntity(info.NetEntity);
|
|
|
|
// If entity does not exist or is on a different map, skip
|
|
if (entity == null
|
|
|| !_entityManager.EntityExists(entity)
|
|
|| _entityManager.GetComponent<TransformComponent>(entity.Value).MapID != args.MapId)
|
|
continue;
|
|
|
|
var aabb = _entityLookup.GetWorldAABB(entity.Value);
|
|
// if not on screen, skip
|
|
if (!aabb.Intersects(in viewport))
|
|
continue;
|
|
|
|
// Get on-screen coordinates of player
|
|
var screenCoordinates = _eyeManager.WorldToScreen(aabb.Center).Rounded();
|
|
|
|
sortable.Add((info, aabb, entity.Value, screenCoordinates));
|
|
}
|
|
|
|
// Draw overlays for visible players, starting from the top of the screen
|
|
foreach (var info in sortable.OrderBy(s => s.Item4.Y).ToList())
|
|
{
|
|
var playerInfo = info.Item1;
|
|
var aabb = info.Item2;
|
|
var entity = info.Item3;
|
|
var screenCoordinatesCenter = info.Item4;
|
|
//the center position is kept separately, for simpler position comparison later
|
|
var centerOffset = new Vector2(28f, -18f) * uiScale;
|
|
var screenCoordinates = screenCoordinatesCenter + centerOffset;
|
|
var alpha = 1f;
|
|
|
|
//TODO make a smarter system where the starting offset can be modified by the predicted position and size of already-drawn overlays/stacks?
|
|
var currentOffset = Vector2.Zero;
|
|
|
|
// Ghosts near the cursor are made transparent/invisible
|
|
// TODO would be "cheaper" if playerinfo already contained a ghost bool, this gets called every frame for every onscreen player!
|
|
if (_entityManager.HasComponent<GhostComponent>(entity))
|
|
{
|
|
// We want the map positions here, so we don't have to worry about resolution and such shenanigans
|
|
var mobPosition = aabb.Center;
|
|
var mousePosition = _eyeManager
|
|
.ScreenToMap(_userInterfaceManager.MousePositionScaled.Position * uiScale)
|
|
.Position;
|
|
var dist = Vector2.Distance(mobPosition, mousePosition);
|
|
if (dist < _ghostHideDistance)
|
|
continue;
|
|
|
|
alpha = Math.Clamp((dist - _ghostHideDistance) / (_ghostFadeDistance - _ghostHideDistance), 0f, 1f);
|
|
colorDisconnected.A = alpha;
|
|
}
|
|
|
|
// If the new overlay text block is within merge distance of any previous ones
|
|
// merge them into a stack so they don't hide each other
|
|
var stack = drawnOverlays.FindAll(x =>
|
|
Vector2.Distance(_eyeManager.ScreenToMap(x.Item1).Position, aabb.Center) <= _overlayMergeDistance);
|
|
if (stack.Count > 0)
|
|
{
|
|
screenCoordinates = stack.First().Item1 + centerOffset;
|
|
// Replacing this overlay's coordinates for the later save with the stack root's coordinates
|
|
// so that other overlays don't try to stack to these coordinates
|
|
screenCoordinatesCenter = stack.First().Item1;
|
|
|
|
var i = 1;
|
|
foreach (var s in stack)
|
|
{
|
|
// additional entries after maximum stack size is reached will be drawn over the last entry
|
|
if (i <= _overlayStackMax - 1)
|
|
currentOffset = lineoffset + s.Item2 ;
|
|
i++;
|
|
}
|
|
}
|
|
|
|
// Character name
|
|
var color = Color.Aquamarine;
|
|
color.A = alpha;
|
|
args.ScreenHandle.DrawString(_font, screenCoordinates + currentOffset, playerInfo.CharacterName, uiScale, playerInfo.Connected ? color : colorDisconnected);
|
|
currentOffset += lineoffset;
|
|
|
|
// Username
|
|
color = Color.Yellow;
|
|
color.A = alpha;
|
|
args.ScreenHandle.DrawString(_font, screenCoordinates + currentOffset, playerInfo.Username, uiScale, playerInfo.Connected ? color : colorDisconnected);
|
|
currentOffset += lineoffset;
|
|
|
|
// Playtime
|
|
if (!string.IsNullOrEmpty(playerInfo.PlaytimeString) && _overlayPlaytime)
|
|
{
|
|
color = Color.Orange;
|
|
color.A = alpha;
|
|
args.ScreenHandle.DrawString(_font, screenCoordinates + currentOffset, playerInfo.PlaytimeString, uiScale, playerInfo.Connected ? color : colorDisconnected);
|
|
currentOffset += lineoffset;
|
|
}
|
|
|
|
// Job
|
|
if (!string.IsNullOrEmpty(playerInfo.StartingJob) && _overlayStartingJob)
|
|
{
|
|
color = Color.GreenYellow;
|
|
color.A = alpha;
|
|
args.ScreenHandle.DrawString(_font, screenCoordinates + currentOffset, Loc.GetString(playerInfo.StartingJob), uiScale, playerInfo.Connected ? color : colorDisconnected);
|
|
currentOffset += lineoffset;
|
|
}
|
|
|
|
// Determine antag symbol
|
|
string? symbol;
|
|
switch (_overlaySymbolStyle)
|
|
{
|
|
case AdminOverlayAntagSymbolStyle.Specific:
|
|
symbol = playerInfo.RoleProto.Symbol;
|
|
break;
|
|
case AdminOverlayAntagSymbolStyle.Basic:
|
|
symbol = Loc.GetString("player-tab-antag-prefix");
|
|
break;
|
|
default:
|
|
case AdminOverlayAntagSymbolStyle.Off:
|
|
symbol = string.Empty;
|
|
break;
|
|
}
|
|
|
|
// Determine antag/role type name
|
|
string? text;
|
|
switch (_overlayFormat)
|
|
{
|
|
case AdminOverlayAntagFormat.Roletype:
|
|
color = playerInfo.RoleProto.Color;
|
|
symbol = _filter.Contains(playerInfo.RoleProto) ? symbol : string.Empty;
|
|
text = _filter.Contains(playerInfo.RoleProto)
|
|
? Loc.GetString(playerInfo.RoleProto.Name).ToUpper()
|
|
: string.Empty;
|
|
break;
|
|
case AdminOverlayAntagFormat.Subtype:
|
|
color = playerInfo.RoleProto.Color;
|
|
symbol = _filter.Contains(playerInfo.RoleProto) ? symbol : string.Empty;
|
|
text = _filter.Contains(playerInfo.RoleProto)
|
|
? _roles.GetRoleSubtypeLabel(playerInfo.RoleProto.Name, playerInfo.Subtype).ToUpper()
|
|
: string.Empty;
|
|
break;
|
|
default:
|
|
case AdminOverlayAntagFormat.Binary:
|
|
color = Color.OrangeRed;
|
|
symbol = playerInfo.Antag ? symbol : string.Empty;
|
|
text = playerInfo.Antag ? _antagLabelClassic : string.Empty;
|
|
break;
|
|
}
|
|
|
|
// Draw antag label
|
|
color.A = alpha;
|
|
var label = !string.IsNullOrEmpty(symbol)
|
|
? Loc.GetString("player-tab-character-name-antag-symbol", ("symbol", symbol), ("name", text))
|
|
: text;
|
|
args.ScreenHandle.DrawString(_fontBold, screenCoordinates + currentOffset, label, uiScale, color);
|
|
currentOffset += lineoffset;
|
|
|
|
//Save the coordinates and size of the text block, for stack merge check
|
|
drawnOverlays.Add((screenCoordinatesCenter, currentOffset));
|
|
}
|
|
}
|
|
}
|