using Content.Client.SubFloor; using Content.Shared.Atmos; using Content.Shared.Atmos.Components; using Content.Shared.Atmos.Piping; using JetBrains.Annotations; using Robust.Client.GameObjects; namespace Content.Client.Atmos.EntitySystems; [UsedImplicitly] public sealed class AtmosPipeAppearanceSystem : EntitySystem { [Dependency] private readonly SharedAppearanceSystem _appearance = default!; [Dependency] private readonly SpriteSystem _sprite = default!; public override void Initialize() { base.Initialize(); SubscribeLocalEvent(OnInit); SubscribeLocalEvent(OnAppearanceChanged, after: [typeof(SubFloorHideSystem)]); } private void OnInit(EntityUid uid, PipeAppearanceComponent component, ComponentInit args) { if (!TryComp(uid, out SpriteComponent? sprite)) return; foreach (var layerKey in Enum.GetValues()) { var layer = _sprite.LayerMapReserve((uid, sprite), layerKey); _sprite.LayerSetRsi((uid, sprite), layer, component.Sprite.RsiPath); _sprite.LayerSetRsiState((uid, sprite), layer, component.Sprite.RsiState); _sprite.LayerSetDirOffset((uid, sprite), layer, ToOffset(layerKey)); } } private void HideAllPipeConnection(Entity entity) { var sprite = entity.Comp; foreach (var layerKey in Enum.GetValues()) { if (!_sprite.LayerMapTryGet(entity.AsNullable(), layerKey, out var key, false)) continue; var layer = sprite[key]; layer.Visible = false; } } private void OnAppearanceChanged(EntityUid uid, PipeAppearanceComponent component, ref AppearanceChangeEvent args) { if (args.Sprite == null) return; if (!args.Sprite.Visible) { // This entity is probably below a floor and is not even visible to the user -> don't bother updating sprite data. // Note that if the subfloor visuals change, then another AppearanceChangeEvent will get triggered. return; } if (!_appearance.TryGetData(uid, PipeVisuals.VisualState, out var worldConnectedDirections, args.Component)) { HideAllPipeConnection((uid, args.Sprite)); return; } if (!_appearance.TryGetData(uid, PipeColorVisuals.Color, out var color, args.Component)) color = Color.White; // transform connected directions to local-coordinates var connectedDirections = worldConnectedDirections.RotatePipeDirection(-Transform(uid).LocalRotation); foreach (var layerKey in Enum.GetValues()) { if (!_sprite.LayerMapTryGet((uid, args.Sprite), layerKey, out var key, false)) continue; var layer = args.Sprite[key]; var dir = (PipeDirection)layerKey; var visible = connectedDirections.HasDirection(dir); layer.Visible &= visible; if (!visible) continue; layer.Color = color; } } private SpriteComponent.DirectionOffset ToOffset(PipeConnectionLayer layer) { return layer switch { PipeConnectionLayer.NorthConnection => SpriteComponent.DirectionOffset.Flip, PipeConnectionLayer.EastConnection => SpriteComponent.DirectionOffset.CounterClockwise, PipeConnectionLayer.WestConnection => SpriteComponent.DirectionOffset.Clockwise, _ => SpriteComponent.DirectionOffset.None, }; } private enum PipeConnectionLayer : byte { NorthConnection = PipeDirection.North, SouthConnection = PipeDirection.South, EastConnection = PipeDirection.East, WestConnection = PipeDirection.West, } }