| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362 |
- using Content.Server.Chemistry.Components;
- using Content.Server.Labels;
- using Content.Server.Popups;
- using Content.Server.Storage.EntitySystems;
- using Content.Shared.Administration.Logs;
- using Content.Shared.Chemistry;
- using Content.Shared.Chemistry.Components;
- using Content.Shared.Chemistry.EntitySystems;
- using Content.Shared.Chemistry.Reagent;
- using Content.Shared.Containers.ItemSlots;
- using Content.Shared.Database;
- using Content.Shared.FixedPoint;
- using Content.Shared.Storage;
- using JetBrains.Annotations;
- using Robust.Server.Audio;
- using Robust.Server.GameObjects;
- using Robust.Shared.Audio;
- using Robust.Shared.Containers;
- using Robust.Shared.Prototypes;
- using System.Diagnostics.CodeAnalysis;
- using System.Linq;
- namespace Content.Server.Chemistry.EntitySystems
- {
- /// <summary>
- /// Contains all the server-side logic for ChemMasters.
- /// <seealso cref="ChemMasterComponent"/>
- /// </summary>
- [UsedImplicitly]
- public sealed class ChemMasterSystem : EntitySystem
- {
- [Dependency] private readonly PopupSystem _popupSystem = default!;
- [Dependency] private readonly AudioSystem _audioSystem = default!;
- [Dependency] private readonly SharedSolutionContainerSystem _solutionContainerSystem = default!;
- [Dependency] private readonly ItemSlotsSystem _itemSlotsSystem = default!;
- [Dependency] private readonly UserInterfaceSystem _userInterfaceSystem = default!;
- [Dependency] private readonly StorageSystem _storageSystem = default!;
- [Dependency] private readonly LabelSystem _labelSystem = default!;
- [Dependency] private readonly ISharedAdminLogManager _adminLogger = default!;
- [ValidatePrototypeId<EntityPrototype>]
- private const string PillPrototypeId = "Pill";
- public override void Initialize()
- {
- base.Initialize();
- SubscribeLocalEvent<ChemMasterComponent, ComponentStartup>(SubscribeUpdateUiState);
- SubscribeLocalEvent<ChemMasterComponent, SolutionContainerChangedEvent>(SubscribeUpdateUiState);
- SubscribeLocalEvent<ChemMasterComponent, EntInsertedIntoContainerMessage>(SubscribeUpdateUiState);
- SubscribeLocalEvent<ChemMasterComponent, EntRemovedFromContainerMessage>(SubscribeUpdateUiState);
- SubscribeLocalEvent<ChemMasterComponent, BoundUIOpenedEvent>(SubscribeUpdateUiState);
- SubscribeLocalEvent<ChemMasterComponent, ChemMasterSetModeMessage>(OnSetModeMessage);
- SubscribeLocalEvent<ChemMasterComponent, ChemMasterSortingTypeCycleMessage>(OnCycleSortingTypeMessage);
- SubscribeLocalEvent<ChemMasterComponent, ChemMasterSetPillTypeMessage>(OnSetPillTypeMessage);
- SubscribeLocalEvent<ChemMasterComponent, ChemMasterReagentAmountButtonMessage>(OnReagentButtonMessage);
- SubscribeLocalEvent<ChemMasterComponent, ChemMasterCreatePillsMessage>(OnCreatePillsMessage);
- SubscribeLocalEvent<ChemMasterComponent, ChemMasterOutputToBottleMessage>(OnOutputToBottleMessage);
- }
- private void SubscribeUpdateUiState<T>(Entity<ChemMasterComponent> ent, ref T ev)
- {
- UpdateUiState(ent);
- }
- private void UpdateUiState(Entity<ChemMasterComponent> ent, bool updateLabel = false)
- {
- var (owner, chemMaster) = ent;
- if (!_solutionContainerSystem.TryGetSolution(owner, SharedChemMaster.BufferSolutionName, out _, out var bufferSolution))
- return;
- var inputContainer = _itemSlotsSystem.GetItemOrNull(owner, SharedChemMaster.InputSlotName);
- var outputContainer = _itemSlotsSystem.GetItemOrNull(owner, SharedChemMaster.OutputSlotName);
- var bufferReagents = bufferSolution.Contents;
- var bufferCurrentVolume = bufferSolution.Volume;
- var state = new ChemMasterBoundUserInterfaceState(
- chemMaster.Mode, chemMaster.SortingType, BuildInputContainerInfo(inputContainer), BuildOutputContainerInfo(outputContainer),
- bufferReagents, bufferCurrentVolume, chemMaster.PillType, chemMaster.PillDosageLimit, updateLabel);
- _userInterfaceSystem.SetUiState(owner, ChemMasterUiKey.Key, state);
- }
- private void OnSetModeMessage(Entity<ChemMasterComponent> chemMaster, ref ChemMasterSetModeMessage message)
- {
- // Ensure the mode is valid, either Transfer or Discard.
- if (!Enum.IsDefined(typeof(ChemMasterMode), message.ChemMasterMode))
- return;
- chemMaster.Comp.Mode = message.ChemMasterMode;
- UpdateUiState(chemMaster);
- ClickSound(chemMaster);
- }
- private void OnCycleSortingTypeMessage(Entity<ChemMasterComponent> chemMaster, ref ChemMasterSortingTypeCycleMessage message)
- {
- chemMaster.Comp.SortingType++;
- if (chemMaster.Comp.SortingType > ChemMasterSortingType.Latest)
- chemMaster.Comp.SortingType = ChemMasterSortingType.None;
- UpdateUiState(chemMaster);
- ClickSound(chemMaster);
- }
- private void OnSetPillTypeMessage(Entity<ChemMasterComponent> chemMaster, ref ChemMasterSetPillTypeMessage message)
- {
- // Ensure valid pill type. There are 20 pills selectable, 0-19.
- if (message.PillType > SharedChemMaster.PillTypes - 1)
- return;
- chemMaster.Comp.PillType = message.PillType;
- UpdateUiState(chemMaster);
- ClickSound(chemMaster);
- }
- private void OnReagentButtonMessage(Entity<ChemMasterComponent> chemMaster, ref ChemMasterReagentAmountButtonMessage message)
- {
- // Ensure the amount corresponds to one of the reagent amount buttons.
- if (!Enum.IsDefined(typeof(ChemMasterReagentAmount), message.Amount))
- return;
- switch (chemMaster.Comp.Mode)
- {
- case ChemMasterMode.Transfer:
- TransferReagents(chemMaster, message.ReagentId, message.Amount.GetFixedPoint(), message.FromBuffer);
- break;
- case ChemMasterMode.Discard:
- DiscardReagents(chemMaster, message.ReagentId, message.Amount.GetFixedPoint(), message.FromBuffer);
- break;
- default:
- // Invalid mode.
- return;
- }
- ClickSound(chemMaster);
- }
- private void TransferReagents(Entity<ChemMasterComponent> chemMaster, ReagentId id, FixedPoint2 amount, bool fromBuffer)
- {
- var container = _itemSlotsSystem.GetItemOrNull(chemMaster, SharedChemMaster.InputSlotName);
- if (container is null ||
- !_solutionContainerSystem.TryGetFitsInDispenser(container.Value, out var containerSoln, out var containerSolution) ||
- !_solutionContainerSystem.TryGetSolution(chemMaster.Owner, SharedChemMaster.BufferSolutionName, out _, out var bufferSolution))
- {
- return;
- }
- if (fromBuffer) // Buffer to container
- {
- amount = FixedPoint2.Min(amount, containerSolution.AvailableVolume);
- amount = bufferSolution.RemoveReagent(id, amount, preserveOrder: true);
- _solutionContainerSystem.TryAddReagent(containerSoln.Value, id, amount, out var _);
- }
- else // Container to buffer
- {
- amount = FixedPoint2.Min(amount, containerSolution.GetReagentQuantity(id));
- _solutionContainerSystem.RemoveReagent(containerSoln.Value, id, amount);
- bufferSolution.AddReagent(id, amount);
- }
- UpdateUiState(chemMaster, updateLabel: true);
- }
- private void DiscardReagents(Entity<ChemMasterComponent> chemMaster, ReagentId id, FixedPoint2 amount, bool fromBuffer)
- {
- if (fromBuffer)
- {
- if (_solutionContainerSystem.TryGetSolution(chemMaster.Owner, SharedChemMaster.BufferSolutionName, out _, out var bufferSolution))
- bufferSolution.RemoveReagent(id, amount, preserveOrder: true);
- else
- return;
- }
- else
- {
- var container = _itemSlotsSystem.GetItemOrNull(chemMaster, SharedChemMaster.InputSlotName);
- if (container is not null &&
- _solutionContainerSystem.TryGetFitsInDispenser(container.Value, out var containerSolution, out _))
- {
- _solutionContainerSystem.RemoveReagent(containerSolution.Value, id, amount);
- }
- else
- return;
- }
- UpdateUiState(chemMaster, updateLabel: fromBuffer);
- }
- private void OnCreatePillsMessage(Entity<ChemMasterComponent> chemMaster, ref ChemMasterCreatePillsMessage message)
- {
- var user = message.Actor;
- var maybeContainer = _itemSlotsSystem.GetItemOrNull(chemMaster, SharedChemMaster.OutputSlotName);
- if (maybeContainer is not { Valid: true } container
- || !TryComp(container, out StorageComponent? storage))
- {
- return; // output can't fit pills
- }
- // Ensure the number is valid.
- if (message.Number == 0 || !_storageSystem.HasSpace((container, storage)))
- return;
- // Ensure the amount is valid.
- if (message.Dosage == 0 || message.Dosage > chemMaster.Comp.PillDosageLimit)
- return;
- // Ensure label length is within the character limit.
- if (message.Label.Length > SharedChemMaster.LabelMaxLength)
- return;
- var needed = message.Dosage * message.Number;
- if (!WithdrawFromBuffer(chemMaster, needed, user, out var withdrawal))
- return;
- _labelSystem.Label(container, message.Label);
- for (var i = 0; i < message.Number; i++)
- {
- var item = Spawn(PillPrototypeId, Transform(container).Coordinates);
- _storageSystem.Insert(container, item, out _, user: user, storage);
- _labelSystem.Label(item, message.Label);
- _solutionContainerSystem.EnsureSolutionEntity(item, SharedChemMaster.PillSolutionName,out var itemSolution ,message.Dosage);
- if (!itemSolution.HasValue)
- return;
- _solutionContainerSystem.TryAddSolution(itemSolution.Value, withdrawal.SplitSolution(message.Dosage));
- var pill = EnsureComp<PillComponent>(item);
- pill.PillType = chemMaster.Comp.PillType;
- Dirty(item, pill);
- // Log pill creation by a user
- _adminLogger.Add(LogType.Action, LogImpact.Low,
- $"{ToPrettyString(user):user} printed {ToPrettyString(item):pill} {SharedSolutionContainerSystem.ToPrettyString(itemSolution.Value.Comp.Solution)}");
- }
- UpdateUiState(chemMaster);
- ClickSound(chemMaster);
- }
- private void OnOutputToBottleMessage(Entity<ChemMasterComponent> chemMaster, ref ChemMasterOutputToBottleMessage message)
- {
- var user = message.Actor;
- var maybeContainer = _itemSlotsSystem.GetItemOrNull(chemMaster, SharedChemMaster.OutputSlotName);
- if (maybeContainer is not { Valid: true } container
- || !_solutionContainerSystem.TryGetSolution(container, SharedChemMaster.BottleSolutionName, out var soln, out var solution))
- {
- return; // output can't fit reagents
- }
- // Ensure the amount is valid.
- if (message.Dosage == 0 || message.Dosage > solution.AvailableVolume)
- return;
- // Ensure label length is within the character limit.
- if (message.Label.Length > SharedChemMaster.LabelMaxLength)
- return;
- if (!WithdrawFromBuffer(chemMaster, message.Dosage, user, out var withdrawal))
- return;
- _labelSystem.Label(container, message.Label);
- _solutionContainerSystem.TryAddSolution(soln.Value, withdrawal);
- // Log bottle creation by a user
- _adminLogger.Add(LogType.Action, LogImpact.Low,
- $"{ToPrettyString(user):user} bottled {ToPrettyString(container):bottle} {SharedSolutionContainerSystem.ToPrettyString(solution)}");
- UpdateUiState(chemMaster);
- ClickSound(chemMaster);
- }
- private bool WithdrawFromBuffer(
- Entity<ChemMasterComponent> chemMaster,
- FixedPoint2 neededVolume, EntityUid? user,
- [NotNullWhen(returnValue: true)] out Solution? outputSolution)
- {
- outputSolution = null;
- if (!_solutionContainerSystem.TryGetSolution(chemMaster.Owner, SharedChemMaster.BufferSolutionName, out _, out var solution))
- {
- return false;
- }
- if (solution.Volume == 0)
- {
- if (user.HasValue)
- _popupSystem.PopupCursor(Loc.GetString("chem-master-window-buffer-empty-text"), user.Value);
- return false;
- }
- // ReSharper disable once InvertIf
- if (neededVolume > solution.Volume)
- {
- if (user.HasValue)
- _popupSystem.PopupCursor(Loc.GetString("chem-master-window-buffer-low-text"), user.Value);
- return false;
- }
- outputSolution = solution.SplitSolution(neededVolume);
- return true;
- }
- private void ClickSound(Entity<ChemMasterComponent> chemMaster)
- {
- _audioSystem.PlayPvs(chemMaster.Comp.ClickSound, chemMaster, AudioParams.Default.WithVolume(-2f));
- }
- private ContainerInfo? BuildInputContainerInfo(EntityUid? container)
- {
- if (container is not { Valid: true })
- return null;
- if (!TryComp(container, out FitsInDispenserComponent? fits)
- || !_solutionContainerSystem.TryGetSolution(container.Value, fits.Solution, out _, out var solution))
- {
- return null;
- }
- return BuildContainerInfo(Name(container.Value), solution);
- }
- private ContainerInfo? BuildOutputContainerInfo(EntityUid? container)
- {
- if (container is not { Valid: true })
- return null;
- var name = Name(container.Value);
- {
- if (_solutionContainerSystem.TryGetSolution(
- container.Value, SharedChemMaster.BottleSolutionName, out _, out var solution))
- {
- return BuildContainerInfo(name, solution);
- }
- }
- if (!TryComp(container, out StorageComponent? storage))
- return null;
- var pills = storage.Container.ContainedEntities.Select((Func<EntityUid, (string, FixedPoint2 quantity)>) (pill =>
- {
- _solutionContainerSystem.TryGetSolution(pill, SharedChemMaster.PillSolutionName, out _, out var solution);
- var quantity = solution?.Volume ?? FixedPoint2.Zero;
- return (Name(pill), quantity);
- })).ToList();
- return new ContainerInfo(name, _storageSystem.GetCumulativeItemAreas((container.Value, storage)), storage.Grid.GetArea())
- {
- Entities = pills
- };
- }
- private static ContainerInfo BuildContainerInfo(string name, Solution solution)
- {
- return new ContainerInfo(name, solution.Volume, solution.MaxVolume)
- {
- Reagents = solution.Contents
- };
- }
- }
- }
|