using System.Linq; using System.Numerics; using System.Threading; using Content.Server.Access.Systems; using Content.Server.Administration.Logs; using Content.Server.Administration.Managers; using Content.Server.Chat.Systems; using Content.Server.Communications; using Content.Server.DeviceNetwork.Systems; using Content.Server.GameTicking.Events; using Content.Server.Pinpointer; using Content.Server.Popups; using Content.Server.RoundEnd; using Content.Server.Screens.Components; using Content.Server.Shuttles.Components; using Content.Server.Shuttles.Events; using Content.Server.Station.Components; using Content.Server.Station.Events; using Content.Server.Station.Systems; using Content.Shared.Access.Systems; using Content.Shared.CCVar; using Content.Shared.Database; using Content.Shared.DeviceNetwork; using Content.Shared.DeviceNetwork.Components; using Content.Shared.GameTicking; using Content.Shared.Localizations; using Content.Shared.Shuttles.Components; using Content.Shared.Shuttles.Events; using Content.Shared.Tag; using Content.Shared.Tiles; using Robust.Server.GameObjects; using Robust.Shared.Audio.Systems; using Robust.Shared.Configuration; using Robust.Shared.EntitySerialization.Systems; using Robust.Shared.Map.Components; using Robust.Shared.Player; using Robust.Shared.Prototypes; using Robust.Shared.Random; using Robust.Shared.Timing; using Robust.Shared.Utility; namespace Content.Server.Shuttles.Systems; public sealed partial class EmergencyShuttleSystem : EntitySystem { /* * Handles the escape shuttle + CentCom. */ [Dependency] private readonly IAdminLogManager _logger = default!; [Dependency] private readonly IAdminManager _admin = default!; [Dependency] private readonly IConfigurationManager _configManager = default!; [Dependency] private readonly IGameTiming _timing = default!; [Dependency] private readonly IRobustRandom _random = default!; [Dependency] private readonly SharedMapSystem _mapSystem = default!; [Dependency] private readonly AccessReaderSystem _reader = default!; [Dependency] private readonly ChatSystem _chatSystem = default!; [Dependency] private readonly CommunicationsConsoleSystem _commsConsole = default!; [Dependency] private readonly DeviceNetworkSystem _deviceNetworkSystem = default!; [Dependency] private readonly DockingSystem _dock = default!; [Dependency] private readonly IdCardSystem _idSystem = default!; [Dependency] private readonly NavMapSystem _navMap = default!; [Dependency] private readonly MapLoaderSystem _loader = default!; [Dependency] private readonly MetaDataSystem _metaData = default!; [Dependency] private readonly PopupSystem _popup = default!; [Dependency] private readonly RoundEndSystem _roundEnd = default!; [Dependency] private readonly SharedAudioSystem _audio = default!; [Dependency] private readonly ShuttleSystem _shuttle = default!; [Dependency] private readonly StationSystem _station = default!; [Dependency] private readonly TransformSystem _transformSystem = default!; [Dependency] private readonly UserInterfaceSystem _uiSystem = default!; private const float ShuttleSpawnBuffer = 1f; private bool _emergencyShuttleEnabled; private static readonly ProtoId DockTag = "DockEmergency"; public override void Initialize() { _emergencyShuttleEnabled = _configManager.GetCVar(CCVars.EmergencyShuttleEnabled); // Don't immediately invoke as roundstart will just handle it. Subs.CVar(_configManager, CCVars.EmergencyShuttleEnabled, SetEmergencyShuttleEnabled); SubscribeLocalEvent(OnRoundStart); SubscribeLocalEvent(OnRoundCleanup); SubscribeLocalEvent(OnStationStartup); SubscribeLocalEvent(OnCentcommShutdown); SubscribeLocalEvent(OnStationInit); SubscribeLocalEvent(OnEmergencyFTL); SubscribeLocalEvent(OnEmergencyFTLComplete); SubscribeNetworkEvent(OnShuttleRequestPosition); InitializeEmergencyConsole(); } private void OnRoundStart(RoundStartingEvent ev) { CleanupEmergencyConsole(); _roundEndCancelToken = new CancellationTokenSource(); } private void OnRoundCleanup(RoundRestartCleanupEvent ev) { _roundEndCancelToken?.Cancel(); _roundEndCancelToken = null; } private void OnCentcommShutdown(EntityUid uid, StationCentcommComponent component, ComponentShutdown args) { ClearCentcomm(component); } private void ClearCentcomm(StationCentcommComponent component) { QueueDel(component.Entity); QueueDel(component.MapEntity); component.Entity = null; component.MapEntity = null; } /// /// Attempts to get the EntityUid of the emergency shuttle /// public EntityUid? GetShuttle() { AllEntityQuery().MoveNext(out var shuttle, out _); return shuttle; } private void SetEmergencyShuttleEnabled(bool value) { if (_emergencyShuttleEnabled == value) return; _emergencyShuttleEnabled = value; if (value) { SetupEmergencyShuttle(); } else { CleanupEmergencyShuttle(); } } private void CleanupEmergencyShuttle() { var query = AllEntityQuery(); while (query.MoveNext(out var uid, out _)) { RemCompDeferred(uid); } } public override void Update(float frameTime) { base.Update(frameTime); UpdateEmergencyConsole(frameTime); } /// /// If the client is requesting debug info on where an emergency shuttle would dock. /// private void OnShuttleRequestPosition(EmergencyShuttleRequestPositionMessage msg, EntitySessionEventArgs args) { if (!_admin.IsAdmin(args.SenderSession)) return; var player = args.SenderSession.AttachedEntity; if (player is null) return; var station = _station.GetOwningStation(player.Value); if (!TryComp(station, out var stationShuttle) || !HasComp(stationShuttle.EmergencyShuttle)) { return; } var targetGrid = _station.GetLargestGrid(Comp(station.Value)); if (targetGrid == null) return; var config = _dock.GetDockingConfig(stationShuttle.EmergencyShuttle.Value, targetGrid.Value, DockTag); if (config == null) return; RaiseNetworkEvent(new EmergencyShuttlePositionMessage() { StationUid = GetNetEntity(targetGrid), Position = config.Area, }); } /// /// Escape shuttle FTL event handler. The only escape shuttle FTL transit should be from station to centcomm at round end /// private void OnEmergencyFTL(EntityUid uid, EmergencyShuttleComponent component, ref FTLStartedEvent args) { var ftlTime = TimeSpan.FromSeconds ( TryComp(uid, out var ftlComp) ? ftlComp.TravelTime : _shuttle.DefaultTravelTime ); if (TryComp(uid, out var netComp)) { var payload = new NetworkPayload { [ShuttleTimerMasks.ShuttleMap] = uid, [ShuttleTimerMasks.SourceMap] = args.FromMapUid, [ShuttleTimerMasks.DestMap] = _transformSystem.GetMap(args.TargetCoordinates), [ShuttleTimerMasks.ShuttleTime] = ftlTime, [ShuttleTimerMasks.SourceTime] = ftlTime, [ShuttleTimerMasks.DestTime] = ftlTime }; _deviceNetworkSystem.QueuePacket(uid, null, payload, netComp.TransmitFrequency); } } /// /// When the escape shuttle finishes FTL (docks at centcomm), have the timers display the round end countdown /// private void OnEmergencyFTLComplete(EntityUid uid, EmergencyShuttleComponent component, ref FTLCompletedEvent args) { var countdownTime = TimeSpan.FromSeconds(_configManager.GetCVar(CCVars.RoundRestartTime)); var shuttle = args.Entity; if (TryComp(shuttle, out var net)) { var payload = new NetworkPayload { [ShuttleTimerMasks.ShuttleMap] = shuttle, [ShuttleTimerMasks.SourceMap] = _roundEnd.GetCentcomm(), [ShuttleTimerMasks.DestMap] = _roundEnd.GetStation(), [ShuttleTimerMasks.ShuttleTime] = countdownTime, [ShuttleTimerMasks.SourceTime] = countdownTime, [ShuttleTimerMasks.DestTime] = countdownTime, }; // by popular request // https://discord.com/channels/310555209753690112/770682801607278632/1189989482234126356 if (_random.Next(1000) == 0) { payload.Add(ScreenMasks.Text, ShuttleTimerMasks.Kill); payload.Add(ScreenMasks.Color, Color.Red); } else payload.Add(ScreenMasks.Text, ShuttleTimerMasks.Bye); _deviceNetworkSystem.QueuePacket(shuttle, null, payload, net.TransmitFrequency); } } /// /// Attempts to dock a station's emergency shuttle. /// /// public ShuttleDockResult? DockSingleEmergencyShuttle(EntityUid stationUid, StationEmergencyShuttleComponent? stationShuttle = null) { if (!Resolve(stationUid, ref stationShuttle)) return null; if (!TryComp(stationShuttle.EmergencyShuttle, out TransformComponent? xform) || !TryComp(stationShuttle.EmergencyShuttle, out var shuttle)) { Log.Error($"Attempted to call an emergency shuttle for an uninitialized station? Station: {ToPrettyString(stationUid)}. Shuttle: {ToPrettyString(stationShuttle.EmergencyShuttle)}"); return null; } var targetGrid = _station.GetLargestGrid(Comp(stationUid)); // UHH GOOD LUCK if (targetGrid == null) { _logger.Add( LogType.EmergencyShuttle, LogImpact.High, $"Emergency shuttle {ToPrettyString(stationUid)} unable to dock with station {ToPrettyString(stationUid)}"); return new ShuttleDockResult { Station = (stationUid, stationShuttle), ResultType = ShuttleDockResultType.GoodLuck, }; } ShuttleDockResultType resultType; if (_shuttle.TryFTLDock(stationShuttle.EmergencyShuttle.Value, shuttle, targetGrid.Value, out var config, DockTag)) { _logger.Add( LogType.EmergencyShuttle, LogImpact.High, $"Emergency shuttle {ToPrettyString(stationUid)} docked with stations"); resultType = _dock.IsConfigPriority(config, DockTag) ? ShuttleDockResultType.PriorityDock : ShuttleDockResultType.OtherDock; } else { _logger.Add( LogType.EmergencyShuttle, LogImpact.High, $"Emergency shuttle {ToPrettyString(stationUid)} unable to find a valid docking port for {ToPrettyString(stationUid)}"); resultType = ShuttleDockResultType.NoDock; } return new ShuttleDockResult { Station = (stationUid, stationShuttle), DockingConfig = config, ResultType = resultType, TargetGrid = targetGrid, }; } /// /// Do post-shuttle-dock setup. Announce to the crew and set up shuttle timers. /// public void AnnounceShuttleDock(ShuttleDockResult result, bool extended) { var stationShuttleComp = result.Station.Comp; var shuttle = result.Station.Comp.EmergencyShuttle; DebugTools.Assert(shuttle != null); if (result.ResultType == ShuttleDockResultType.GoodLuck) { _chatSystem.DispatchStationAnnouncement( result.Station, Loc.GetString(stationShuttleComp.FailureAnnouncement), playDefaultSound: false); // TODO: Need filter extensions or something don't blame me. _audio.PlayGlobal(stationShuttleComp.FailureAudio, Filter.Broadcast(), true); return; } DebugTools.Assert(result.TargetGrid != null); // Send station announcement. var targetXform = Transform(result.TargetGrid.Value); var angle = _dock.GetAngle( shuttle.Value, Transform(shuttle.Value), result.TargetGrid.Value, targetXform); var direction = ContentLocalizationManager.FormatDirection(angle.GetDir()); var location = FormattedMessage.RemoveMarkupPermissive( _navMap.GetNearestBeaconString((shuttle.Value, Transform(shuttle.Value)))); var extendedText = extended ? Loc.GetString(stationShuttleComp.LaunchExtendedMessage) : ""; var locKey = result.ResultType == ShuttleDockResultType.NoDock ? stationShuttleComp.NearbyAnnouncement : stationShuttleComp.DockedAnnouncement; _chatSystem.DispatchStationAnnouncement( result.Station, Loc.GetString( locKey, ("time", $"{_consoleAccumulator:0}"), ("direction", direction), ("location", location), ("extended", extendedText)), playDefaultSound: false); // Trigger shuttle timers on the shuttle. var time = TimeSpan.FromSeconds(_consoleAccumulator); if (TryComp(shuttle, out var netComp)) { var payload = new NetworkPayload { [ShuttleTimerMasks.ShuttleMap] = shuttle, [ShuttleTimerMasks.SourceMap] = targetXform.MapUid, [ShuttleTimerMasks.DestMap] = _roundEnd.GetCentcomm(), [ShuttleTimerMasks.ShuttleTime] = time, [ShuttleTimerMasks.SourceTime] = time, [ShuttleTimerMasks.DestTime] = time + TimeSpan.FromSeconds(TransitTime), [ShuttleTimerMasks.Docked] = true, }; _deviceNetworkSystem.QueuePacket(shuttle.Value, null, payload, netComp.TransmitFrequency); } // Play announcement audio. var audioFile = result.ResultType == ShuttleDockResultType.NoDock ? stationShuttleComp.NearbyAudio : stationShuttleComp.DockedAudio; // TODO: Need filter extensions or something don't blame me. _audio.PlayGlobal(audioFile, Filter.Broadcast(), true); } private void OnStationInit(EntityUid uid, StationCentcommComponent component, MapInitEvent args) { // This is handled on map-init, so that centcomm has finished initializing by the time the StationPostInitEvent // gets raised if (!_emergencyShuttleEnabled) return; // Post mapinit? fancy if (TryComp(component.Entity, out TransformComponent? xform)) { component.MapEntity = xform.MapUid; return; } AddCentcomm(uid, component); } private void OnStationStartup(Entity ent, ref StationPostInitEvent args) { AddEmergencyShuttle((ent, ent)); } /// /// Teleports the emergency shuttle to its station and starts the countdown until it launches. /// /// /// If the emergency shuttle is disabled, this immediately ends the round. /// public void DockEmergencyShuttle() { if (EmergencyShuttleArrived) return; if (!_emergencyShuttleEnabled) { _roundEnd.EndRound(); return; } _consoleAccumulator = _configManager.GetCVar(CCVars.EmergencyShuttleDockTime); EmergencyShuttleArrived = true; var query = AllEntityQuery(); var dockResults = new List(); while (query.MoveNext(out var uid, out var comp)) { if (DockSingleEmergencyShuttle(uid, comp) is { } dockResult) dockResults.Add(dockResult); } // Make the shuttle wait longer if it couldn't dock in the normal spot. // We have to handle the possibility of there being multiple stations, so since the shuttle timer is global, // use the WORST value we have. var worstResult = dockResults.Max(x => x.ResultType); var multiplier = worstResult switch { ShuttleDockResultType.OtherDock => _configManager.GetCVar( CCVars.EmergencyShuttleDockTimeMultiplierOtherDock), ShuttleDockResultType.NoDock => _configManager.GetCVar( CCVars.EmergencyShuttleDockTimeMultiplierNoDock), // GoodLuck doesn't get a multiplier. // Quite frankly at that point the round is probably so fucked that you'd rather it be over ASAP. _ => 1, }; _consoleAccumulator *= multiplier; foreach (var shuttleDockResult in dockResults) { AnnounceShuttleDock(shuttleDockResult, multiplier > 1); } _commsConsole.UpdateCommsConsoleInterface(); } private void SetupEmergencyShuttle() { if (!_emergencyShuttleEnabled) return; var centcommQuery = AllEntityQuery(); while (centcommQuery.MoveNext(out var uid, out var centcomm)) { AddCentcomm(uid, centcomm); } var query = AllEntityQuery(); while (query.MoveNext(out var uid, out var comp)) { AddEmergencyShuttle((uid, comp)); } } private void AddCentcomm(EntityUid station, StationCentcommComponent component) { DebugTools.Assert(LifeStage(station) >= EntityLifeStage.MapInitialized); if (component.MapEntity != null || component.Entity != null) { Log.Warning("Attempted to re-add an existing centcomm map."); return; } // Check for existing centcomms and just point to that var query = AllEntityQuery(); while (query.MoveNext(out var otherComp)) { if (otherComp == component) continue; if (!Exists(otherComp.MapEntity) || !Exists(otherComp.Entity)) { Log.Error($"Discovered invalid centcomm component?"); ClearCentcomm(otherComp); continue; } component.MapEntity = otherComp.MapEntity; component.Entity = otherComp.Entity; component.ShuttleIndex = otherComp.ShuttleIndex; return; } if (string.IsNullOrEmpty(component.Map.ToString())) { Log.Warning("No CentComm map found, skipping setup."); return; } var map = _mapSystem.CreateMap(out var mapId); if (!_loader.TryLoadGrid(mapId, component.Map, out var grid)) { Log.Error($"Failed to set up centcomm grid!"); return; } if (!Exists(map)) { Log.Error($"Failed to set up centcomm map!"); QueueDel(grid); return; } if (!Exists(grid)) { Log.Error($"Failed to set up centcomm grid!"); QueueDel(map); return; } var xform = Transform(grid.Value); if (xform.ParentUid != map || xform.MapUid != map) { Log.Error($"Centcomm grid is not parented to its own map?"); QueueDel(map); QueueDel(grid); return; } component.MapEntity = map; _metaData.SetEntityName(map, Loc.GetString("map-name-centcomm")); component.Entity = grid; _shuttle.TryAddFTLDestination(mapId, true, out _); Log.Info($"Created centcomm grid {ToPrettyString(grid)} on map {ToPrettyString(map)} for station {ToPrettyString(station)}"); } public HashSet GetCentcommMaps() { var query = AllEntityQuery(); var maps = new HashSet(Count()); while (query.MoveNext(out var comp)) { if (comp.MapEntity != null) maps.Add(comp.MapEntity.Value); } return maps; } private void AddEmergencyShuttle(Entity ent) { if (!Resolve(ent.Owner, ref ent.Comp1, ref ent.Comp2)) return; if (!_emergencyShuttleEnabled) return; if (ent.Comp1.EmergencyShuttle != null) { if (Exists(ent.Comp1.EmergencyShuttle)) { Log.Error($"Attempted to add an emergency shuttle to {ToPrettyString(ent)}, despite a shuttle already existing?"); return; } Log.Error($"Encountered deleted emergency shuttle during initialization of {ToPrettyString(ent)}"); ent.Comp1.EmergencyShuttle = null; } if (!TryComp(ent.Comp2.MapEntity, out MapComponent? map)) { Log.Error($"Failed to add emergency shuttle - centcomm has not been initialized? {ToPrettyString(ent)}"); return; } // Load escape shuttle var shuttlePath = ent.Comp1.EmergencyShuttlePath; if (!_loader.TryLoadGrid(map.MapId, shuttlePath, out var shuttle, // Should be far enough... right? I'm too lazy to bounds check CentCom rn. offset: new Vector2(500f + ent.Comp2.ShuttleIndex, 0f))) { Log.Error($"Unable to spawn emergency shuttle {shuttlePath} for {ToPrettyString(ent)}"); return; } ent.Comp2.ShuttleIndex += Comp(shuttle.Value).LocalAABB.Width + ShuttleSpawnBuffer; // Update indices for all centcomm comps pointing to same map var query = AllEntityQuery(); while (query.MoveNext(out var comp)) { if (comp == ent.Comp2 || comp.MapEntity != ent.Comp2.MapEntity) continue; comp.ShuttleIndex = ent.Comp2.ShuttleIndex; } ent.Comp1.EmergencyShuttle = shuttle; EnsureComp(shuttle.Value); EnsureComp(shuttle.Value); EnsureComp(shuttle.Value); Log.Info($"Added emergency shuttle {ToPrettyString(shuttle)} for station {ToPrettyString(ent)} and centcomm {ToPrettyString(ent.Comp2.Entity)}"); } /// /// Returns whether a target is escaping on the emergency shuttle, but only if evac has arrived. /// public bool IsTargetEscaping(EntityUid target) { // if evac isn't here then sitting in a pod doesn't return true if (!EmergencyShuttleArrived) return false; // check if target is on an emergency shuttle var xform = Transform(target); if (HasComp(xform.GridUid)) return true; return false; } private bool IsOnGrid(TransformComponent xform, EntityUid shuttle, MapGridComponent? grid = null, TransformComponent? shuttleXform = null) { if (!Resolve(shuttle, ref grid, ref shuttleXform)) return false; return _transformSystem.GetWorldMatrix(shuttleXform).TransformBox(grid.LocalAABB).Contains(_transformSystem.GetWorldPosition(xform)); } /// /// A result of a shuttle dock operation done by . /// /// public sealed class ShuttleDockResult { /// /// The station for which the emergency shuttle got docked. /// public Entity Station; /// /// The target grid of the station that the shuttle tried to dock to. /// /// /// Not present if is . /// public EntityUid? TargetGrid; /// /// Enum code describing the dock result. /// public ShuttleDockResultType ResultType; /// /// The docking config used to actually dock to the station. /// /// /// Only present if is /// or . /// public DockingConfig? DockingConfig; } /// /// Emergency shuttle dock result codes used by . /// public enum ShuttleDockResultType : byte { // This enum is ordered from "best" to "worst". This is used to sort the results. /// /// The shuttle was docked at a priority dock, which is the intended destination. /// PriorityDock, /// /// The shuttle docked at another dock on the station then the intended priority dock. /// OtherDock, /// /// The shuttle couldn't find any suitable dock on the station at all, it did not dock. /// NoDock, /// /// No station grid was found at all, shuttle did not get moved. /// GoodLuck, } }