using System; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; using Content.Server.Body.Behavior; using Content.Server.Hands.Components; using Content.Server.Items; using Content.Shared.Body.Components; using Content.Shared.Chemistry.EntitySystems; using Content.Shared.Chemistry.Reagent; using Content.Shared.Interaction; using Content.Shared.Interaction.Helpers; using Content.Shared.Popups; using Content.Shared.Sound; using Robust.Shared.Audio; using Robust.Shared.GameObjects; using Robust.Shared.Localization; using Robust.Shared.Player; using Robust.Shared.Prototypes; using Robust.Shared.Serialization.Manager.Attributes; using Robust.Shared.Serialization.TypeSerializers.Implementations.Custom.Prototype; using Robust.Shared.ViewVariables; namespace Content.Server.Nutrition.Components { [RegisterComponent] [ComponentReference(typeof(IAfterInteract))] public class FoodComponent : Component, IUse, IAfterInteract { public override string Name => "Food"; [DataField("solution")] public string SolutionName { get; set; } = "food"; [ViewVariables] [DataField("useSound")] private SoundSpecifier UseSound { get; set; } = new SoundPathSpecifier("/Audio/Items/eatfood.ogg"); [ViewVariables] [DataField("trash", customTypeSerializer: typeof(PrototypeIdSerializer))] private string? TrashPrototype { get; set; } [ViewVariables] [DataField("transferAmount")] private ReagentUnit? TransferAmount { get; set; } = ReagentUnit.New(5); [DataField("utensilsNeeded")] private UtensilType _utensilsNeeded = UtensilType.None; [DataField("eatMessage")] private string _eatMessage = "food-nom"; [ViewVariables] public int UsesRemaining { get { if (!EntitySystem.Get().TryGetSolution(Owner, SolutionName, out var solution)) { return 0; } if (TransferAmount == null) return solution.CurrentVolume == 0 ? 0 : 1; return solution.CurrentVolume == 0 ? 0 : Math.Max(1, (int) Math.Ceiling((solution.CurrentVolume / (ReagentUnit)TransferAmount).Float())); } } protected override void Initialize() { base.Initialize(); // Owner.EnsureComponentWarn(); } bool IUse.UseEntity(UseEntityEventArgs eventArgs) { if (_utensilsNeeded != UtensilType.None) { eventArgs.User.PopupMessage(Loc.GetString("food-you-need-utensil", ("utensil", _utensilsNeeded))); return false; } return TryUseFood(eventArgs.User, null); } // Feeding someone else async Task IAfterInteract.AfterInteract(AfterInteractEventArgs eventArgs) { if (eventArgs.Target == null) { return false; } TryUseFood(eventArgs.User, eventArgs.Target); return true; } public bool TryUseFood(IEntity? user, IEntity? target, UtensilComponent? utensilUsed = null) { var solutionContainerSys = EntitySystem.Get(); if (!solutionContainerSys.TryGetSolution(Owner, SolutionName, out var solution)) { return false; } if (user == null) { return false; } if (UsesRemaining <= 0) { user.PopupMessage(Loc.GetString("food-component-try-use-food-is-empty", ("entity", Owner))); DeleteAndSpawnTrash(user); return false; } var trueTarget = target ?? user; if (!trueTarget.TryGetComponent(out SharedBodyComponent? body) || !body.TryGetMechanismBehaviors(out var stomachs)) { return false; } var utensils = utensilUsed != null ? new List { utensilUsed } : null; if (_utensilsNeeded != UtensilType.None) { utensils = new List(); var types = UtensilType.None; if (user.TryGetComponent(out HandsComponent? hands)) { foreach (var item in hands.GetAllHeldItems()) { if (!item.Owner.TryGetComponent(out UtensilComponent? utensil)) { continue; } utensils.Add(utensil); types |= utensil.Types; } } if (!types.HasFlag(_utensilsNeeded)) { trueTarget.PopupMessage(user, Loc.GetString("food-you-need-to-hold-utensil", ("utensil", _utensilsNeeded))); return false; } } if (!user.InRangeUnobstructed(trueTarget, popup: true)) { return false; } var transferAmount = TransferAmount != null ? ReagentUnit.Min((ReagentUnit)TransferAmount, solution.CurrentVolume) : solution.CurrentVolume; var split = solutionContainerSys.SplitSolution(Owner.Uid, solution, transferAmount); var firstStomach = stomachs.FirstOrDefault(stomach => stomach.CanTransferSolution(split)); if (firstStomach == null) { solutionContainerSys.TryAddSolution(Owner.Uid, solution, split); trueTarget.PopupMessage(user, Loc.GetString("food-you-cannot-eat-any-more")); return false; } // TODO: Account for partial transfer. split.DoEntityReaction(trueTarget, ReactionMethod.Ingestion); firstStomach.TryTransferSolution(split); SoundSystem.Play(Filter.Pvs(trueTarget), UseSound.GetSound(), trueTarget, AudioParams.Default.WithVolume(-1f)); trueTarget.PopupMessage(user, Loc.GetString(_eatMessage, ("food", Owner))); // If utensils were used if (utensils != null) { foreach (var utensil in utensils) { utensil.TryBreak(user); } } if (UsesRemaining > 0) { return true; } if (string.IsNullOrEmpty(TrashPrototype)) { Owner.Delete(); return true; } DeleteAndSpawnTrash(user); return true; } private void DeleteAndSpawnTrash(IEntity user) { //We're empty. Become trash. var position = Owner.Transform.Coordinates; var finisher = Owner.EntityManager.SpawnEntity(TrashPrototype, position); // If the user is holding the item if (user.TryGetComponent(out HandsComponent? handsComponent) && handsComponent.IsHolding(Owner)) { Owner.Delete(); // Put the trash in the user's hand if (finisher.TryGetComponent(out ItemComponent? item) && handsComponent.CanPutInHand(item)) { handsComponent.PutInHand(item); } } else { Owner.Delete(); } } } }