Files
space-station-14/Content.Shared/Metabolism/MetabolizerComponent.cs
pathetic meowmeow ff1af35afb Replace metabolism groups with metabolism stages (#42172)
* Replace metabolism groups with metabolism stages

* return to return

* killed

* yaml linted

* yaml linted again

* margarine

* bugfixes and balancing tweaks
2026-01-23 01:50:33 +00:00

149 lines
4.9 KiB
C#

using Content.Shared.Body.Components;
using Content.Shared.FixedPoint;
using Robust.Shared.Prototypes;
using Robust.Shared.Serialization.TypeSerializers.Implementations.Custom;
namespace Content.Shared.Metabolism;
/// <summary>
/// Handles metabolizing various reagents with given effects.
/// </summary>
[RegisterComponent, AutoGenerateComponentPause, Access(typeof(MetabolizerSystem))]
public sealed partial class MetabolizerComponent : Component
{
/// <summary>
/// The next time that reagents will be metabolized.
/// </summary>
[DataField, AutoPausedField]
public TimeSpan NextUpdate;
/// <summary>
/// How often to metabolize reagents.
/// </summary>
/// <returns></returns>
[DataField]
public TimeSpan UpdateInterval = TimeSpan.FromSeconds(1);
/// <summary>
/// Multiplier applied to <see cref="UpdateInterval"/> for adjusting based on metabolic rate multiplier.
/// </summary>
[DataField]
public float UpdateIntervalMultiplier = 1f;
/// <summary>
/// Adjusted update interval based off of the multiplier value.
/// </summary>
[ViewVariables]
public TimeSpan AdjustedUpdateInterval => UpdateInterval * UpdateIntervalMultiplier;
/// <summary>
/// From which solution will this metabolizer attempt to metabolize chemicals for a given stage
/// </summary>
[DataField]
public Dictionary<ProtoId<MetabolismStagePrototype>, MetabolismSolutionEntry> Solutions = new()
{
["Respiration"] = new()
{
SolutionName = "Lung",
SolutionOnBody = false,
TransferSolutionName = BloodstreamComponent.DefaultBloodSolutionName,
MetabolizeAll = true
},
["Digestion"] = new()
{
SolutionName = "stomach",
SolutionOnBody = false,
TransferSolutionName = BloodstreamComponent.DefaultBloodSolutionName,
TransferEfficacy = 0.5
},
["Bloodstream"] = new()
{
SolutionName = BloodstreamComponent.DefaultBloodSolutionName,
TransferSolutionName = BloodstreamComponent.DefaultMetabolitesSolutionName,
},
["Metabolites"] = new()
{
SolutionName = BloodstreamComponent.DefaultMetabolitesSolutionName
}
};
/// <summary>
/// Does this component use a solution on it's parent entity (the body) or itself
/// </summary>
/// <remarks>
/// Most things will use the parent entity (bloodstream).
/// </remarks>
[DataField]
public bool SolutionOnBody = true;
/// <summary>
/// List of metabolizer types that this organ is. ex. Human, Slime, Felinid, w/e.
/// </summary>
[DataField]
[Access(typeof(MetabolizerSystem), Other = AccessPermissions.ReadExecute)] // FIXME Friends
public HashSet<ProtoId<MetabolizerTypePrototype>>? MetabolizerTypes;
/// <summary>
/// How many reagents can this metabolizer process at once?
/// Used to nerf 'stacked poisons' where having 5+ different poisons in a syringe, even at low
/// quantity, would be muuuuch better than just one poison acting.
/// </summary>
[DataField("maxReagents")]
public int MaxReagentsProcessable = 3;
/// <summary>
/// A list of metabolism stages that this metabolizer will act on, in order of precedence.
/// </summary>
[DataField]
public List<ProtoId<MetabolismStagePrototype>> Stages = new();
}
[DataDefinition]
public sealed partial class MetabolismSolutionEntry
{
/// <summary>
/// From which solution will this metabolizer attempt to metabolize chemicals
/// </summary>
[DataField(required: true)]
public string SolutionName;
/// <summary>
/// Does this metabolizer use a solution on it's parent entity (the body) or itself
/// </summary>
/// <remarks>
/// Most things will use the parent entity (bloodstream).
/// </remarks>
[DataField]
public bool SolutionOnBody = true;
/// <summary>
/// When true, this solution will be metabolized entirely instead of at a certain rate
/// </summary>
[DataField]
public bool MetabolizeAll = false;
/// <summary>
/// Reagents without a metabolism for the current stage will be transferred to this solution
/// </summary>
[DataField]
public string? TransferSolutionName;
/// <summary>
/// Reagents transferred by this metabolizer will transfer at this rate if they don't have a metabolism
/// </summary>
[DataField]
public FixedPoint2 TransferRate = 0.25;
/// <summary>
/// The percentage of transferred reagents that actually make it to the next step in metabolism if they don't have explicit metabolites
/// </summary>
[DataField]
public FixedPoint2 TransferEfficacy = 1;
/// <summary>
/// Does this metabolizer transfer to a solution on the body or on the entity itself
/// </summary>
[DataField]
public bool TransferSolutionOnBody = true;
}