501 lines
22 KiB
C#
501 lines
22 KiB
C#
using System.Diagnostics.CodeAnalysis;
|
|
using System.Linq;
|
|
using Content.Server.GameTicking;
|
|
using Content.Server.Station.Components;
|
|
using Content.Shared.CCVar;
|
|
using Content.Shared.GameTicking;
|
|
using Content.Shared.Preferences;
|
|
using Content.Shared.Roles;
|
|
using JetBrains.Annotations;
|
|
using Robust.Server.Player;
|
|
using Robust.Shared.Configuration;
|
|
using Robust.Shared.Player;
|
|
using Robust.Shared.Random;
|
|
|
|
namespace Content.Server.Station.Systems;
|
|
|
|
/// <summary>
|
|
/// Manages job slots for stations.
|
|
/// </summary>
|
|
[PublicAPI]
|
|
public sealed partial class StationJobsSystem : EntitySystem
|
|
{
|
|
[Dependency] private readonly IConfigurationManager _configurationManager = default!;
|
|
[Dependency] private readonly IRobustRandom _random = default!;
|
|
[Dependency] private readonly GameTicker _gameTicker = default!;
|
|
[Dependency] private readonly StationSystem _stationSystem = default!;
|
|
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
|
|
|
/// <inheritdoc/>
|
|
public override void Initialize()
|
|
{
|
|
SubscribeLocalEvent<StationInitializedEvent>(OnStationInitialized);
|
|
SubscribeLocalEvent<StationJobsComponent, StationRenamedEvent>(OnStationRenamed);
|
|
SubscribeLocalEvent<StationJobsComponent, ComponentShutdown>(OnStationDeletion);
|
|
SubscribeLocalEvent<PlayerJoinedLobbyEvent>(OnPlayerJoinedLobby);
|
|
_configurationManager.OnValueChanged(CCVars.GameDisallowLateJoins, _ => UpdateJobsAvailable(), true);
|
|
}
|
|
|
|
public override void Update(float _)
|
|
{
|
|
if (_availableJobsDirty)
|
|
{
|
|
_cachedAvailableJobs = GenerateJobsAvailableEvent();
|
|
RaiseNetworkEvent(_cachedAvailableJobs, Filter.Empty().AddPlayers(_playerManager.ServerSessions));
|
|
_availableJobsDirty = false;
|
|
}
|
|
}
|
|
|
|
private void OnStationDeletion(EntityUid uid, StationJobsComponent component, ComponentShutdown args)
|
|
{
|
|
UpdateJobsAvailable(); // we no longer exist so the jobs list is changed.
|
|
}
|
|
|
|
private void OnStationInitialized(StationInitializedEvent msg)
|
|
{
|
|
if (!TryComp<StationJobsComponent>(msg.Station, out var stationJobs))
|
|
return;
|
|
|
|
var mapJobList = stationJobs.SetupAvailableJobs;
|
|
|
|
stationJobs.RoundStartTotalJobs = mapJobList.Values.Where(x => x[0] is not null && x[0] > 0).Sum(x => x[0]!.Value);
|
|
stationJobs.MidRoundTotalJobs = mapJobList.Values.Where(x => x[1] is not null && x[1] > 0).Sum(x => x[1]!.Value);
|
|
|
|
stationJobs.TotalJobs = stationJobs.MidRoundTotalJobs;
|
|
|
|
stationJobs.JobList = mapJobList.ToDictionary(x => x.Key, x =>
|
|
{
|
|
if (x.Value[1] <= -1)
|
|
return null;
|
|
return (uint?) x.Value[1];
|
|
});
|
|
|
|
stationJobs.RoundStartJobList = mapJobList.ToDictionary(x => x.Key, x =>
|
|
{
|
|
if (x.Value[0] <= -1)
|
|
return null;
|
|
return (uint?) x.Value[0];
|
|
});
|
|
|
|
stationJobs.OverflowJobs = stationJobs.OverflowJobs.ToHashSet();
|
|
|
|
UpdateJobsAvailable();
|
|
}
|
|
|
|
#region Public API
|
|
|
|
/// <inheritdoc cref="TryAssignJob(Robust.Shared.GameObjects.EntityUid,string,Content.Server.Station.Components.StationJobsComponent?)"/>
|
|
/// <param name="station">Station to assign a job on.</param>
|
|
/// <param name="job">Job to assign.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
public bool TryAssignJob(EntityUid station, JobPrototype job, StationJobsComponent? stationJobs = null)
|
|
{
|
|
return TryAssignJob(station, job.ID, stationJobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to assign the given job once. (essentially, it decrements the slot if possible).
|
|
/// </summary>
|
|
/// <param name="station">Station to assign a job on.</param>
|
|
/// <param name="jobPrototypeId">Job prototype ID to assign.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Whether or not assignment was a success.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public bool TryAssignJob(EntityUid station, string jobPrototypeId, StationJobsComponent? stationJobs = null)
|
|
{
|
|
return TryAdjustJobSlot(station, jobPrototypeId, -1, false, false, stationJobs);
|
|
}
|
|
|
|
/// <inheritdoc cref="TryAdjustJobSlot(Robust.Shared.GameObjects.EntityUid,string,int,bool,bool,Content.Server.Station.Components.StationJobsComponent?)"/>
|
|
/// <param name="station">Station to adjust the job slot on.</param>
|
|
/// <param name="job">Job to adjust.</param>
|
|
/// <param name="amount">Amount to adjust by.</param>
|
|
/// <param name="createSlot">Whether or not it should create the slot if it doesn't exist.</param>
|
|
/// <param name="clamp">Whether or not to clamp to zero if you'd remove more jobs than are available.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
public bool TryAdjustJobSlot(EntityUid station, JobPrototype job, int amount, bool createSlot = false, bool clamp = false,
|
|
StationJobsComponent? stationJobs = null)
|
|
{
|
|
return TryAdjustJobSlot(station, job.ID, amount, createSlot, clamp, stationJobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to adjust the given job slot by the amount provided.
|
|
/// </summary>
|
|
/// <param name="station">Station to adjust the job slot on.</param>
|
|
/// <param name="jobPrototypeId">Job prototype ID to adjust.</param>
|
|
/// <param name="amount">Amount to adjust by.</param>
|
|
/// <param name="createSlot">Whether or not it should create the slot if it doesn't exist.</param>
|
|
/// <param name="clamp">Whether or not to clamp to zero if you'd remove more jobs than are available.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Whether or not slot adjustment was a success.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public bool TryAdjustJobSlot(EntityUid station, string jobPrototypeId, int amount, bool createSlot = false, bool clamp = false,
|
|
StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
var jobList = stationJobs.JobList;
|
|
|
|
// This should:
|
|
// - Return true when zero slots are added/removed.
|
|
// - Return true when you add.
|
|
// - Return true when you remove and do not exceed the number of slot available.
|
|
// - Return false when you remove from a job that doesn't exist.
|
|
// - Return false when you remove and exceed the number of slots available.
|
|
// And additionally, if adding would add a job not previously on the manifest when createSlot is false, return false and do nothing.
|
|
switch (jobList.ContainsKey(jobPrototypeId))
|
|
{
|
|
case false when amount < 0:
|
|
return false;
|
|
case false:
|
|
if (!createSlot)
|
|
return false;
|
|
stationJobs.TotalJobs += amount;
|
|
jobList[jobPrototypeId] = (uint?)amount;
|
|
UpdateJobsAvailable();
|
|
return true;
|
|
case true:
|
|
// Job is unlimited so just say we adjusted it and do nothing.
|
|
if (jobList[jobPrototypeId] == null)
|
|
return true;
|
|
|
|
// Would remove more jobs than we have available.
|
|
if (amount < 0 && (jobList[jobPrototypeId] + amount < 0 && !clamp))
|
|
return false;
|
|
|
|
stationJobs.TotalJobs += amount;
|
|
|
|
//C# type handling moment
|
|
if (amount > 0)
|
|
jobList[jobPrototypeId] += (uint)amount;
|
|
else
|
|
{
|
|
if ((int)jobList[jobPrototypeId]!.Value - Math.Abs(amount) <= 0)
|
|
jobList[jobPrototypeId] = 0;
|
|
else
|
|
jobList[jobPrototypeId] -= (uint) Math.Abs(amount);
|
|
}
|
|
|
|
UpdateJobsAvailable();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc cref="TrySetJobSlot(Robust.Shared.GameObjects.EntityUid,string,int,bool,Content.Server.Station.Components.StationJobsComponent?)"/>
|
|
/// <param name="station">Station to adjust the job slot on.</param>
|
|
/// <param name="jobPrototype">Job prototype to adjust.</param>
|
|
/// <param name="amount">Amount to set to.</param>
|
|
/// <param name="createSlot">Whether or not it should create the slot if it doesn't exist.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns></returns>
|
|
public bool TrySetJobSlot(EntityUid station, JobPrototype jobPrototype, int amount, bool createSlot = false,
|
|
StationJobsComponent? stationJobs = null)
|
|
{
|
|
return TrySetJobSlot(station, jobPrototype.ID, amount, createSlot, stationJobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to set the given job slot to the amount provided.
|
|
/// </summary>
|
|
/// <param name="station">Station to adjust the job slot on.</param>
|
|
/// <param name="jobPrototypeId">Job prototype ID to adjust.</param>
|
|
/// <param name="amount">Amount to set to.</param>
|
|
/// <param name="createSlot">Whether or not it should create the slot if it doesn't exist.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Whether or not setting the value succeeded.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public bool TrySetJobSlot(EntityUid station, string jobPrototypeId, int amount, bool createSlot = false,
|
|
StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
if (amount < 0)
|
|
throw new ArgumentException("Tried to set a job to have a negative number of slots!", nameof(amount));
|
|
|
|
var jobList = stationJobs.JobList;
|
|
|
|
switch (jobList.ContainsKey(jobPrototypeId))
|
|
{
|
|
case false:
|
|
if (!createSlot)
|
|
return false;
|
|
stationJobs.TotalJobs += amount;
|
|
jobList[jobPrototypeId] = (uint?)amount;
|
|
UpdateJobsAvailable();
|
|
return true;
|
|
case true:
|
|
stationJobs.TotalJobs += amount - (int) (jobList[jobPrototypeId] ?? 0);
|
|
|
|
jobList[jobPrototypeId] = (uint)amount;
|
|
UpdateJobsAvailable();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc cref="MakeJobUnlimited(Robust.Shared.GameObjects.EntityUid,string,Content.Server.Station.Components.StationJobsComponent?)"/>
|
|
/// <param name="station">Station to make a job unlimited on.</param>
|
|
/// <param name="job">Job to make unlimited.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
public void MakeJobUnlimited(EntityUid station, JobPrototype job, StationJobsComponent? stationJobs = null)
|
|
{
|
|
MakeJobUnlimited(station, job.ID, stationJobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Makes the given job have unlimited slots.
|
|
/// </summary>
|
|
/// <param name="station">Station to make a job unlimited on.</param>
|
|
/// <param name="jobPrototypeId">Job prototype ID to make unlimited.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public void MakeJobUnlimited(EntityUid station, string jobPrototypeId, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
// Subtract out the job we're fixing to make have unlimited slots.
|
|
if (stationJobs.JobList.ContainsKey(jobPrototypeId) && stationJobs.JobList[jobPrototypeId] != null)
|
|
stationJobs.TotalJobs -= (int)stationJobs.JobList[jobPrototypeId]!.Value;
|
|
|
|
stationJobs.JobList[jobPrototypeId] = null;
|
|
|
|
UpdateJobsAvailable();
|
|
}
|
|
|
|
/// <inheritdoc cref="IsJobUnlimited(Robust.Shared.GameObjects.EntityUid,string,Content.Server.Station.Components.StationJobsComponent?)"/>
|
|
/// <param name="station">Station to check.</param>
|
|
/// <param name="job">Job to check.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
public bool IsJobUnlimited(EntityUid station, JobPrototype job, StationJobsComponent? stationJobs = null)
|
|
{
|
|
return IsJobUnlimited(station, job.ID, stationJobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks if the given job is unlimited.
|
|
/// </summary>
|
|
/// <param name="station">Station to check.</param>
|
|
/// <param name="jobPrototypeId">Job prototype ID to check.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Returns if the given slot is unlimited.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public bool IsJobUnlimited(EntityUid station, string jobPrototypeId, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
var res = stationJobs.JobList.TryGetValue(jobPrototypeId, out var job) && job == null;
|
|
return res;
|
|
}
|
|
|
|
/// <inheritdoc cref="TryGetJobSlot(Robust.Shared.GameObjects.EntityUid,string,out System.Nullable{uint},Content.Server.Station.Components.StationJobsComponent?)"/>
|
|
/// <param name="station">Station to get slot info from.</param>
|
|
/// <param name="job">Job to get slot info for.</param>
|
|
/// <param name="slots">The number of slots remaining. Null if infinite.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
public bool TryGetJobSlot(EntityUid station, JobPrototype job, out uint? slots, StationJobsComponent? stationJobs = null)
|
|
{
|
|
return TryGetJobSlot(station, job.ID, out slots, stationJobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns information about the given job slot.
|
|
/// </summary>
|
|
/// <param name="station">Station to get slot info from.</param>
|
|
/// <param name="jobPrototypeId">Job prototype ID to get slot info for.</param>
|
|
/// <param name="slots">The number of slots remaining. Null if infinite.</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Whether or not the slot exists.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
/// <remarks>slots will be null if the slot doesn't exist, as well, so make sure to check the return value.</remarks>
|
|
public bool TryGetJobSlot(EntityUid station, string jobPrototypeId, out uint? slots, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
if (stationJobs.JobList.TryGetValue(jobPrototypeId, out var job))
|
|
{
|
|
slots = job;
|
|
return true;
|
|
}
|
|
else // Else if slot isn't present return null.
|
|
{
|
|
slots = null;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns all jobs available on the station.
|
|
/// </summary>
|
|
/// <param name="station">Station to get jobs for</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Set containing all jobs available.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public IReadOnlySet<string> GetAvailableJobs(EntityUid station, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
return stationJobs.JobList.Where(x => x.Value != 0).Select(x => x.Key).ToHashSet();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns all overflow jobs available on the station.
|
|
/// </summary>
|
|
/// <param name="station">Station to get jobs for</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>Set containing all overflow jobs available.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public IReadOnlySet<string> GetOverflowJobs(EntityUid station, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
return stationJobs.OverflowJobs.ToHashSet();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a readonly dictionary of all jobs and their slot info.
|
|
/// </summary>
|
|
/// <param name="station">Station to get jobs for</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>List of all jobs on the station.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public IReadOnlyDictionary<string, uint?> GetJobs(EntityUid station, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
return stationJobs.JobList;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a readonly dictionary of all round-start jobs and their slot info.
|
|
/// </summary>
|
|
/// <param name="station">Station to get jobs for</param>
|
|
/// <param name="stationJobs">Resolve pattern, station jobs component of the station.</param>
|
|
/// <returns>List of all round-start jobs.</returns>
|
|
/// <exception cref="ArgumentException">Thrown when the given station is not a station.</exception>
|
|
public IReadOnlyDictionary<string, uint?> GetRoundStartJobs(EntityUid station, StationJobsComponent? stationJobs = null)
|
|
{
|
|
if (!Resolve(station, ref stationJobs))
|
|
throw new ArgumentException("Tried to use a non-station entity as a station!", nameof(station));
|
|
|
|
return stationJobs.RoundStartJobList;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Looks at the given priority list, and picks the best available job (optionally with the given exclusions)
|
|
/// </summary>
|
|
/// <param name="station">Station to pick from.</param>
|
|
/// <param name="jobPriorities">The priority list to use for selecting a job.</param>
|
|
/// <param name="pickOverflows">Whether or not to pick from the overflow list.</param>
|
|
/// <param name="disallowedJobs">A set of disallowed jobs, if any.</param>
|
|
/// <returns>The selected job, if any.</returns>
|
|
public string? PickBestAvailableJobWithPriority(EntityUid station, IReadOnlyDictionary<string, JobPriority> jobPriorities, bool pickOverflows, IReadOnlySet<string>? disallowedJobs = null)
|
|
{
|
|
if (station == EntityUid.Invalid)
|
|
return null;
|
|
|
|
var available = GetAvailableJobs(station);
|
|
bool TryPick(JobPriority priority, [NotNullWhen(true)] out string? jobId)
|
|
{
|
|
var filtered = jobPriorities
|
|
.Where(p =>
|
|
p.Value == priority
|
|
&& disallowedJobs != null
|
|
&& !disallowedJobs.Contains(p.Key)
|
|
&& available.Contains(p.Key))
|
|
.Select(p => p.Key)
|
|
.ToList();
|
|
|
|
if (filtered.Count != 0)
|
|
{
|
|
jobId = _random.Pick(filtered);
|
|
return true;
|
|
}
|
|
|
|
jobId = default;
|
|
return false;
|
|
}
|
|
|
|
if (TryPick(JobPriority.High, out var picked))
|
|
{
|
|
return picked;
|
|
}
|
|
|
|
if (TryPick(JobPriority.Medium, out picked))
|
|
{
|
|
return picked;
|
|
}
|
|
|
|
if (TryPick(JobPriority.Low, out picked))
|
|
{
|
|
return picked;
|
|
}
|
|
|
|
if (!pickOverflows)
|
|
return null;
|
|
|
|
var overflows = GetOverflowJobs(station);
|
|
return overflows.Count != 0 ? _random.Pick(overflows) : null;
|
|
}
|
|
|
|
#endregion Public API
|
|
|
|
#region Latejoin job management
|
|
|
|
private bool _availableJobsDirty;
|
|
|
|
private TickerJobsAvailableEvent _cachedAvailableJobs = new (new Dictionary<NetEntity, string>(), new Dictionary<NetEntity, Dictionary<string, uint?>>());
|
|
|
|
/// <summary>
|
|
/// Assembles an event from the current available-to-play jobs.
|
|
/// This is moderately expensive to construct.
|
|
/// </summary>
|
|
/// <returns>The event.</returns>
|
|
private TickerJobsAvailableEvent GenerateJobsAvailableEvent()
|
|
{
|
|
// If late join is disallowed, return no available jobs.
|
|
if (_gameTicker.DisallowLateJoin)
|
|
return new TickerJobsAvailableEvent(new Dictionary<NetEntity, string>(), new Dictionary<NetEntity, Dictionary<string, uint?>>());
|
|
|
|
var jobs = new Dictionary<NetEntity, Dictionary<string, uint?>>();
|
|
var stationNames = new Dictionary<NetEntity, string>();
|
|
|
|
var query = EntityQueryEnumerator<StationJobsComponent>();
|
|
|
|
while (query.MoveNext(out var station, out var comp))
|
|
{
|
|
var netStation = GetNetEntity(station);
|
|
var list = comp.JobList.ToDictionary(x => x.Key, x => x.Value);
|
|
jobs.Add(netStation, list);
|
|
stationNames.Add(netStation, Name(station));
|
|
}
|
|
return new TickerJobsAvailableEvent(stationNames, jobs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Updates the cached available jobs. Moderately expensive.
|
|
/// </summary>
|
|
private void UpdateJobsAvailable()
|
|
{
|
|
_availableJobsDirty = true;
|
|
}
|
|
|
|
private void OnPlayerJoinedLobby(PlayerJoinedLobbyEvent ev)
|
|
{
|
|
RaiseNetworkEvent(_cachedAvailableJobs, ev.PlayerSession.ConnectedClient);
|
|
}
|
|
|
|
private void OnStationRenamed(EntityUid uid, StationJobsComponent component, StationRenamedEvent args)
|
|
{
|
|
UpdateJobsAvailable();
|
|
}
|
|
|
|
#endregion
|
|
}
|