| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 |
- using System.Numerics;
- using System.Text;
- using Content.Client.Resources;
- using Robust.Client.Graphics;
- using Robust.Client.Input;
- using Robust.Client.ResourceManagement;
- using Robust.Shared.Enums;
- using Robust.Shared.Map;
- using Robust.Shared.Map.Components;
- using Robust.Shared.Timing;
- using Robust.Shared.Utility;
- using static Content.Shared.NodeContainer.NodeVis;
- namespace Content.Client.NodeContainer
- {
- public sealed class NodeVisualizationOverlay : Overlay
- {
- private readonly NodeGroupSystem _system;
- private readonly EntityLookupSystem _lookup;
- private readonly IMapManager _mapManager;
- private readonly IInputManager _inputManager;
- private readonly IEntityManager _entityManager;
- private readonly SharedTransformSystem _transformSystem;
- private readonly Dictionary<(int, int), NodeRenderData> _nodeIndex = new();
- private readonly Dictionary<EntityUid, Dictionary<Vector2i, List<(GroupData, NodeDatum)>>> _gridIndex = new ();
- private List<Entity<MapGridComponent>> _grids = new();
- private readonly Font _font;
- private Vector2 _mouseWorldPos = default;
- private (int group, int node)? _hovered;
- private float _time;
- public override OverlaySpace Space => OverlaySpace.ScreenSpace | OverlaySpace.WorldSpace;
- public NodeVisualizationOverlay(
- NodeGroupSystem system,
- EntityLookupSystem lookup,
- IMapManager mapManager,
- IInputManager inputManager,
- IResourceCache cache,
- IEntityManager entityManager)
- {
- _system = system;
- _lookup = lookup;
- _mapManager = mapManager;
- _inputManager = inputManager;
- _entityManager = entityManager;
- _transformSystem = _entityManager.System<SharedTransformSystem>();
- _font = cache.GetFont("/Fonts/NotoSans/NotoSans-Regular.ttf", 12);
- }
- protected override void Draw(in OverlayDrawArgs args)
- {
- if ((args.Space & OverlaySpace.WorldSpace) != 0)
- {
- DrawWorld(args);
- }
- else if ((args.Space & OverlaySpace.ScreenSpace) != 0)
- {
- DrawScreen(args);
- }
- }
- private void DrawScreen(in OverlayDrawArgs args)
- {
- var mousePos = _inputManager.MouseScreenPosition.Position;
- _mouseWorldPos = args
- .ViewportControl!
- .PixelToMap(mousePos)
- .Position;
- if (_hovered == null)
- return;
- var (groupId, nodeId) = _hovered.Value;
- var group = _system.Groups[groupId];
- var node = _system.NodeLookup[(groupId, nodeId)];
- var xform = _entityManager.GetComponent<TransformComponent>(_entityManager.GetEntity(node.Entity));
- if (!_entityManager.TryGetComponent<MapGridComponent>(xform.GridUid, out var grid))
- return;
- var gridTile = grid.TileIndicesFor(xform.Coordinates);
- var sb = new StringBuilder();
- sb.Append($"entity: {node.Entity}\n");
- sb.Append($"group id: {group.GroupId}\n");
- sb.Append($"node: {node.Name}\n");
- sb.Append($"type: {node.Type}\n");
- sb.Append($"grid pos: {gridTile}\n");
- sb.Append(group.DebugData);
- args.ScreenHandle.DrawString(_font, mousePos + new Vector2(20, -20), sb.ToString());
- }
- private void DrawWorld(in OverlayDrawArgs overlayDrawArgs)
- {
- const float nodeSize = 8f / 32;
- const float nodeOffset = 6f / 32;
- var handle = overlayDrawArgs.WorldHandle;
- var map = overlayDrawArgs.Viewport.Eye?.Position.MapId ?? default;
- if (map == MapId.Nullspace)
- return;
- _hovered = default;
- var cursorBox = Box2.CenteredAround(_mouseWorldPos, new Vector2(nodeSize, nodeSize));
- // Group visible nodes by grid tiles.
- var worldAABB = overlayDrawArgs.WorldAABB;
- var xformQuery = _entityManager.GetEntityQuery<TransformComponent>();
- _grids.Clear();
- _mapManager.FindGridsIntersecting(map, worldAABB, ref _grids);
- foreach (var grid in _grids)
- {
- foreach (var entity in _lookup.GetEntitiesIntersecting(grid, worldAABB))
- {
- if (!_system.Entities.TryGetValue(entity, out var nodeData))
- continue;
- var gridDict = _gridIndex.GetOrNew(grid);
- var coords = xformQuery.GetComponent(entity).Coordinates;
- // TODO: This probably shouldn't be capable of returning NaN...
- if (float.IsNaN(coords.Position.X) || float.IsNaN(coords.Position.Y))
- continue;
- var tile = gridDict.GetOrNew(grid.Comp.TileIndicesFor(coords));
- foreach (var (group, nodeDatum) in nodeData)
- {
- if (!_system.Filtered.Contains(group.GroupId))
- {
- tile.Add((group, nodeDatum));
- }
- }
- }
- }
- foreach (var (gridId, gridDict) in _gridIndex)
- {
- var grid = _entityManager.GetComponent<MapGridComponent>(gridId);
- var (_, _, worldMatrix, invMatrix) = _transformSystem.GetWorldPositionRotationMatrixWithInv(gridId);
- var lCursorBox = invMatrix.TransformBox(cursorBox);
- foreach (var (pos, list) in gridDict)
- {
- var centerPos = (Vector2) pos + grid.TileSizeHalfVector;
- list.Sort(NodeDisplayComparer.Instance);
- var offset = -(list.Count - 1) * nodeOffset / 2;
- foreach (var (group, node) in list)
- {
- var nodePos = centerPos + new Vector2(offset, offset);
- if (lCursorBox.Contains(nodePos))
- _hovered = (group.NetId, node.NetId);
- _nodeIndex[(group.NetId, node.NetId)] = new NodeRenderData(group, node, nodePos);
- offset += nodeOffset;
- }
- }
- handle.SetTransform(worldMatrix);
- foreach (var nodeRenderData in _nodeIndex.Values)
- {
- var pos = nodeRenderData.NodePos;
- var bounds = Box2.CenteredAround(pos, new Vector2(nodeSize, nodeSize));
- var groupData = nodeRenderData.GroupData;
- var color = groupData.Color;
- if (!_hovered.HasValue)
- color.A = 0.5f;
- else if (_hovered.Value.group != groupData.NetId)
- color.A = 0.2f;
- else
- color.A = 0.75f + MathF.Sin(_time * 4) * 0.25f;
- handle.DrawRect(bounds, color);
- foreach (var reachable in nodeRenderData.NodeDatum.Reachable)
- {
- if (_nodeIndex.TryGetValue((groupData.NetId, reachable), out var reachDat))
- {
- handle.DrawLine(pos, reachDat.NodePos, color);
- }
- }
- }
- _nodeIndex.Clear();
- }
- handle.SetTransform(Matrix3x2.Identity);
- _gridIndex.Clear();
- }
- protected override void FrameUpdate(FrameEventArgs args)
- {
- base.FrameUpdate(args);
- _time += args.DeltaSeconds;
- }
- private sealed class NodeDisplayComparer : IComparer<(GroupData, NodeDatum)>
- {
- public static readonly NodeDisplayComparer Instance = new();
- public int Compare((GroupData, NodeDatum) x, (GroupData, NodeDatum) y)
- {
- var (groupX, nodeX) = x;
- var (groupY, nodeY) = y;
- var cmp = groupX.NetId.CompareTo(groupY.NetId);
- if (cmp != 0)
- return cmp;
- return nodeX.NetId.CompareTo(nodeY.NetId);
- }
- }
- private sealed class NodeRenderData
- {
- public GroupData GroupData;
- public NodeDatum NodeDatum;
- public Vector2 NodePos;
- public NodeRenderData(GroupData groupData, NodeDatum nodeDatum, Vector2 nodePos)
- {
- GroupData = groupData;
- NodeDatum = nodeDatum;
- NodePos = nodePos;
- }
- }
- }
- }
|