| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362 |
- using System.Diagnostics.CodeAnalysis;
- using Content.Server.Power.Components;
- using Robust.Shared.Map.Components;
- using Robust.Shared.Physics;
- using Robust.Shared.Physics.Components;
- namespace Content.Server.Power.EntitySystems
- {
- public sealed class ExtensionCableSystem : EntitySystem
- {
- public override void Initialize()
- {
- base.Initialize();
- //Lifecycle events
- SubscribeLocalEvent<ExtensionCableProviderComponent, ComponentStartup>(OnProviderStarted);
- SubscribeLocalEvent<ExtensionCableProviderComponent, ComponentShutdown>(OnProviderShutdown);
- SubscribeLocalEvent<ExtensionCableReceiverComponent, ComponentStartup>(OnReceiverStarted);
- SubscribeLocalEvent<ExtensionCableReceiverComponent, ComponentShutdown>(OnReceiverShutdown);
- //Anchoring
- SubscribeLocalEvent<ExtensionCableReceiverComponent, AnchorStateChangedEvent>(OnReceiverAnchorStateChanged);
- SubscribeLocalEvent<ExtensionCableReceiverComponent, ReAnchorEvent>(OnReceiverReAnchor);
- SubscribeLocalEvent<ExtensionCableProviderComponent, AnchorStateChangedEvent>(OnProviderAnchorStateChanged);
- SubscribeLocalEvent<ExtensionCableProviderComponent, ReAnchorEvent>(OnProviderReAnchor);
- }
- #region Provider
- public void SetProviderTransferRange(EntityUid uid, int range, ExtensionCableProviderComponent? provider = null)
- {
- if (!Resolve(uid, ref provider))
- return;
- provider.TransferRange = range;
- ResetReceivers(provider);
- }
- private void OnProviderStarted(EntityUid uid, ExtensionCableProviderComponent provider, ComponentStartup args)
- {
- Connect(uid, provider);
- }
- private void OnProviderShutdown(EntityUid uid, ExtensionCableProviderComponent provider, ComponentShutdown args)
- {
- var xform = Transform(uid);
- // If grid deleting no need to update power.
- if (HasComp<MapGridComponent>(xform.GridUid) &&
- MetaData(xform.GridUid.Value).EntityLifeStage > EntityLifeStage.MapInitialized)
- {
- return;
- }
- Disconnect(uid, provider);
- }
- private void OnProviderAnchorStateChanged(EntityUid uid, ExtensionCableProviderComponent provider, ref AnchorStateChangedEvent args)
- {
- if (args.Anchored)
- Connect(uid, provider);
- else
- Disconnect(uid, provider);
- }
- private void Connect(EntityUid uid, ExtensionCableProviderComponent provider)
- {
- provider.Connectable = true;
- foreach (var receiver in FindAvailableReceivers(uid, provider.TransferRange))
- {
- receiver.Comp.Provider?.LinkedReceivers.Remove(receiver);
- receiver.Comp.Provider = provider;
- provider.LinkedReceivers.Add(receiver);
- RaiseLocalEvent(receiver, new ProviderConnectedEvent(provider), broadcast: false);
- RaiseLocalEvent(uid, new ReceiverConnectedEvent(receiver), broadcast: false);
- }
- }
- private void Disconnect(EntityUid uid, ExtensionCableProviderComponent provider)
- {
- // same as OnProviderShutdown
- provider.Connectable = false;
- ResetReceivers(provider);
- }
- private void OnProviderReAnchor(EntityUid uid, ExtensionCableProviderComponent component, ref ReAnchorEvent args)
- {
- Disconnect(uid, component);
- Connect(uid, component);
- }
- private void ResetReceivers(ExtensionCableProviderComponent provider)
- {
- var providerId = provider.Owner;
- var receivers = provider.LinkedReceivers.ToArray();
- provider.LinkedReceivers.Clear();
- foreach (var receiver in receivers)
- {
- var receiverId = receiver.Owner;
- receiver.Provider = null;
- RaiseLocalEvent(receiverId, new ProviderDisconnectedEvent(provider), broadcast: false);
- RaiseLocalEvent(providerId, new ReceiverDisconnectedEvent((receiverId, receiver)), broadcast: false);
- }
- foreach (var receiver in receivers)
- {
- // No point resetting what the receiver is doing if it's deleting, plus significant perf savings
- // in not doing needless lookups
- var receiverId = receiver.Owner;
- if (!EntityManager.IsQueuedForDeletion(receiverId)
- && MetaData(receiverId).EntityLifeStage <= EntityLifeStage.MapInitialized)
- {
- TryFindAndSetProvider(receiver);
- }
- }
- }
- private IEnumerable<Entity<ExtensionCableReceiverComponent>> FindAvailableReceivers(EntityUid owner, float range)
- {
- var xform = Transform(owner);
- var coordinates = xform.Coordinates;
- if (!TryComp(xform.GridUid, out MapGridComponent? grid))
- yield break;
- var nearbyEntities = grid.GetCellsInSquareArea(coordinates, (int) Math.Ceiling(range / grid.TileSize));
- foreach (var entity in nearbyEntities)
- {
- if (entity == owner)
- continue;
- if (EntityManager.IsQueuedForDeletion(entity) || MetaData(entity).EntityLifeStage > EntityLifeStage.MapInitialized)
- continue;
- if (!TryComp(entity, out ExtensionCableReceiverComponent? receiver))
- continue;
- if (!receiver.Connectable || receiver.Provider != null)
- continue;
- if ((Transform(entity).LocalPosition - xform.LocalPosition).Length() <= Math.Min(range, receiver.ReceptionRange))
- yield return (entity, receiver);
- }
- }
- #endregion
- #region Receiver
- public void SetReceiverReceptionRange(EntityUid uid, int range, ExtensionCableReceiverComponent? receiver = null)
- {
- if (!Resolve(uid, ref receiver))
- return;
- var provider = receiver.Provider;
- receiver.Provider = null;
- RaiseLocalEvent(uid, new ProviderDisconnectedEvent(provider), broadcast: false);
- if (provider != null)
- {
- RaiseLocalEvent(provider.Owner, new ReceiverDisconnectedEvent((uid, receiver)), broadcast: false);
- provider.LinkedReceivers.Remove(receiver);
- }
- receiver.ReceptionRange = range;
- TryFindAndSetProvider(receiver);
- }
- private void OnReceiverStarted(EntityUid uid, ExtensionCableReceiverComponent receiver, ComponentStartup args)
- {
- if (EntityManager.TryGetComponent(receiver.Owner, out PhysicsComponent? physicsComponent))
- {
- receiver.Connectable = physicsComponent.BodyType == BodyType.Static;
- }
- if (receiver.Provider == null)
- {
- TryFindAndSetProvider(receiver);
- }
- }
- private void OnReceiverShutdown(EntityUid uid, ExtensionCableReceiverComponent receiver, ComponentShutdown args)
- {
- Disconnect(uid, receiver);
- }
- private void OnReceiverAnchorStateChanged(EntityUid uid, ExtensionCableReceiverComponent receiver, ref AnchorStateChangedEvent args)
- {
- if (args.Anchored)
- {
- Connect(uid, receiver);
- }
- else
- {
- Disconnect(uid, receiver);
- }
- }
- private void OnReceiverReAnchor(EntityUid uid, ExtensionCableReceiverComponent receiver, ref ReAnchorEvent args)
- {
- Disconnect(uid, receiver);
- Connect(uid, receiver);
- }
- private void Connect(EntityUid uid, ExtensionCableReceiverComponent receiver)
- {
- receiver.Connectable = true;
- if (receiver.Provider == null)
- {
- TryFindAndSetProvider(receiver);
- }
- }
- private void Disconnect(EntityUid uid, ExtensionCableReceiverComponent receiver)
- {
- receiver.Connectable = false;
- RaiseLocalEvent(uid, new ProviderDisconnectedEvent(receiver.Provider), broadcast: false);
- if (receiver.Provider != null)
- {
- RaiseLocalEvent(receiver.Provider.Owner, new ReceiverDisconnectedEvent((uid, receiver)), broadcast: false);
- receiver.Provider.LinkedReceivers.Remove(receiver);
- }
- receiver.Provider = null;
- }
- private void TryFindAndSetProvider(ExtensionCableReceiverComponent receiver, TransformComponent? xform = null)
- {
- var uid = receiver.Owner;
- if (!receiver.Connectable)
- return;
- if (!TryFindAvailableProvider(uid, receiver.ReceptionRange, out var provider, xform))
- return;
- receiver.Provider = provider;
- provider.LinkedReceivers.Add(receiver);
- RaiseLocalEvent(uid, new ProviderConnectedEvent(provider), broadcast: false);
- RaiseLocalEvent(provider.Owner, new ReceiverConnectedEvent((uid, receiver)), broadcast: false);
- }
- private bool TryFindAvailableProvider(EntityUid owner, float range, [NotNullWhen(true)] out ExtensionCableProviderComponent? foundProvider, TransformComponent? xform = null)
- {
- if (!Resolve(owner, ref xform) || !TryComp(xform.GridUid, out MapGridComponent? grid))
- {
- foundProvider = null;
- return false;
- }
- var coordinates = xform.Coordinates;
- var nearbyEntities = grid.GetCellsInSquareArea(coordinates, (int) Math.Ceiling(range / grid.TileSize));
- var cableQuery = GetEntityQuery<ExtensionCableProviderComponent>();
- var metaQuery = GetEntityQuery<MetaDataComponent>();
- var xformQuery = GetEntityQuery<TransformComponent>();
- ExtensionCableProviderComponent? closestCandidate = null;
- var closestDistanceFound = float.MaxValue;
- foreach (var entity in nearbyEntities)
- {
- if (entity == owner || !cableQuery.TryGetComponent(entity, out var provider) || !provider.Connectable)
- continue;
- if (EntityManager.IsQueuedForDeletion(entity))
- continue;
- if (!metaQuery.TryGetComponent(entity, out var meta) || meta.EntityLifeStage > EntityLifeStage.MapInitialized)
- continue;
- // Find the closest provider
- if (!xformQuery.TryGetComponent(entity, out var entityXform))
- continue;
- var distance = (entityXform.LocalPosition - xform.LocalPosition).Length();
- if (distance >= closestDistanceFound)
- continue;
- closestCandidate = provider;
- closestDistanceFound = distance;
- }
- // Make sure the provider is in range before claiming success
- if (closestCandidate != null && closestDistanceFound <= Math.Min(range, closestCandidate.TransferRange))
- {
- foundProvider = closestCandidate;
- return true;
- }
- foundProvider = null;
- return false;
- }
- #endregion
- #region Events
- /// <summary>
- /// Sent when a <see cref="ExtensionCableProviderComponent"/> connects to a <see cref="ExtensionCableReceiverComponent"/>
- /// </summary>
- public sealed class ProviderConnectedEvent : EntityEventArgs
- {
- /// <summary>
- /// The <see cref="ExtensionCableProviderComponent"/> that connected.
- /// </summary>
- public ExtensionCableProviderComponent Provider;
- public ProviderConnectedEvent(ExtensionCableProviderComponent provider)
- {
- Provider = provider;
- }
- }
- /// <summary>
- /// Sent when a <see cref="ExtensionCableProviderComponent"/> disconnects from a <see cref="ExtensionCableReceiverComponent"/>
- /// </summary>
- public sealed class ProviderDisconnectedEvent : EntityEventArgs
- {
- /// <summary>
- /// The <see cref="ExtensionCableProviderComponent"/> that disconnected.
- /// </summary>
- public ExtensionCableProviderComponent? Provider;
- public ProviderDisconnectedEvent(ExtensionCableProviderComponent? provider)
- {
- Provider = provider;
- }
- }
- /// <summary>
- /// Sent when a <see cref="ExtensionCableReceiverComponent"/> connects to a <see cref="ExtensionCableProviderComponent"/>
- /// </summary>
- public sealed class ReceiverConnectedEvent : EntityEventArgs
- {
- /// <summary>
- /// The <see cref="ExtensionCableReceiverComponent"/> that connected.
- /// </summary>
- public Entity<ExtensionCableReceiverComponent> Receiver;
- public ReceiverConnectedEvent(Entity<ExtensionCableReceiverComponent> receiver)
- {
- Receiver = receiver;
- }
- }
- /// <summary>
- /// Sent when a <see cref="ExtensionCableReceiverComponent"/> disconnects from a <see cref="ExtensionCableProviderComponent"/>
- /// </summary>
- public sealed class ReceiverDisconnectedEvent : EntityEventArgs
- {
- /// <summary>
- /// The <see cref="ExtensionCableReceiverComponent"/> that disconnected.
- /// </summary>
- public Entity<ExtensionCableReceiverComponent> Receiver;
- public ReceiverDisconnectedEvent(Entity<ExtensionCableReceiverComponent> receiver)
- {
- Receiver = receiver;
- }
- }
- #endregion
- }
- }
|