| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393 |
- using System.Diagnostics.CodeAnalysis;
- using System.Linq;
- using Content.Shared.Containers.ItemSlots;
- using Content.Shared.Damage;
- using Content.Shared.DoAfter;
- using Content.Shared.Examine;
- using Content.Shared.Forensics;
- using Content.Shared.IdentityManagement;
- using Content.Shared.Implants.Components;
- using Content.Shared.Interaction.Events;
- using Content.Shared.Popups;
- using Content.Shared.Verbs;
- using Content.Shared.Whitelist;
- using Robust.Shared.Containers;
- using Robust.Shared.Prototypes;
- using Robust.Shared.Serialization;
- using Robust.Shared.Utility;
- namespace Content.Shared.Implants;
- public abstract class SharedImplanterSystem : EntitySystem
- {
- [Dependency] private readonly SharedContainerSystem _container = default!;
- [Dependency] private readonly ItemSlotsSystem _itemSlots = default!;
- [Dependency] private readonly SharedAppearanceSystem _appearance = default!;
- [Dependency] private readonly SharedPopupSystem _popup = default!;
- [Dependency] private readonly EntityWhitelistSystem _whitelistSystem = default!;
- [Dependency] private readonly DamageableSystem _damageableSystem = default!;
- [Dependency] private readonly SharedUserInterfaceSystem _uiSystem = default!;
- [Dependency] private readonly IPrototypeManager _proto = default!;
- public override void Initialize()
- {
- base.Initialize();
- SubscribeLocalEvent<ImplanterComponent, ComponentInit>(OnImplanterInit);
- SubscribeLocalEvent<ImplanterComponent, EntInsertedIntoContainerMessage>(OnEntInserted);
- SubscribeLocalEvent<ImplanterComponent, ExaminedEvent>(OnExamine);
- SubscribeLocalEvent<ImplanterComponent, UseInHandEvent>(OnUseInHand);
- SubscribeLocalEvent<ImplanterComponent, GetVerbsEvent<InteractionVerb>>(OnVerb);
- SubscribeLocalEvent<ImplanterComponent, DeimplantChangeVerbMessage>(OnSelected);
- }
- private void OnImplanterInit(EntityUid uid, ImplanterComponent component, ComponentInit args)
- {
- if (component.Implant != null)
- component.ImplanterSlot.StartingItem = component.Implant;
- _itemSlots.AddItemSlot(uid, ImplanterComponent.ImplanterSlotId, component.ImplanterSlot);
- component.DeimplantChosen ??= component.DeimplantWhitelist.FirstOrNull();
- Dirty(uid, component);
- }
- private void OnEntInserted(EntityUid uid, ImplanterComponent component, EntInsertedIntoContainerMessage args)
- {
- var implantData = EntityManager.GetComponent<MetaDataComponent>(args.Entity);
- component.ImplantData = (implantData.EntityName, implantData.EntityDescription);
- }
- private void OnExamine(EntityUid uid, ImplanterComponent component, ExaminedEvent args)
- {
- if (!component.ImplanterSlot.HasItem || !args.IsInDetailsRange)
- return;
- args.PushMarkup(Loc.GetString("implanter-contained-implant-text", ("desc", component.ImplantData.Item2)));
- }
- public bool CheckSameImplant(EntityUid target, EntityUid implant)
- {
- if (!TryComp<ImplantedComponent>(target, out var implanted))
- return false;
- var implantPrototype = Prototype(implant);
- return implanted.ImplantContainer.ContainedEntities.Any(entity => Prototype(entity) == implantPrototype);
- }
- private void OnVerb(EntityUid uid, ImplanterComponent component, GetVerbsEvent<InteractionVerb> args)
- {
- if (!args.CanAccess || !args.CanInteract)
- return;
- if (component.CurrentMode == ImplanterToggleMode.Draw)
- {
- args.Verbs.Add(new InteractionVerb()
- {
- Text = Loc.GetString("implanter-set-draw-verb"),
- Act = () => TryOpenUi(uid, args.User, component)
- });
- }
- }
- private void OnUseInHand(EntityUid uid, ImplanterComponent? component, UseInHandEvent args)
- {
- if (!Resolve(uid, ref component))
- return;
- if (component.CurrentMode == ImplanterToggleMode.Draw)
- TryOpenUi(uid, args.User, component);
- }
- private void OnSelected(EntityUid uid, ImplanterComponent component, DeimplantChangeVerbMessage args)
- {
- component.DeimplantChosen = args.Implant;
- SetSelectedDeimplant(uid, args.Implant, component: component);
- }
- private void TryOpenUi(EntityUid uid, EntityUid user, ImplanterComponent? component = null)
- {
- if (!Resolve(uid, ref component))
- return;
- _uiSystem.TryToggleUi(uid, DeimplantUiKey.Key, user);
- component.DeimplantChosen ??= component.DeimplantWhitelist.FirstOrNull();
- Dirty(uid, component);
- }
- //Instantly implant something and add all necessary components and containers.
- //Set to draw mode if not implant only
- public void Implant(EntityUid user, EntityUid target, EntityUid implanter, ImplanterComponent component)
- {
- if (!CanImplant(user, target, implanter, component, out var implant, out var implantComp))
- return;
- // Check if we are trying to implant a implant which is already implanted
- // Check AFTER the doafter to prevent "is it a fake?" metagaming against deceptive implants
- if (!component.AllowMultipleImplants && CheckSameImplant(target, implant.Value))
- {
- var name = Identity.Name(target, EntityManager, user);
- var msg = Loc.GetString("implanter-component-implant-already", ("implant", implant), ("target", name));
- _popup.PopupEntity(msg, target, user);
- return;
- }
- //If the target doesn't have the implanted component, add it.
- var implantedComp = EnsureComp<ImplantedComponent>(target);
- var implantContainer = implantedComp.ImplantContainer;
- if (component.ImplanterSlot.ContainerSlot != null)
- _container.Remove(implant.Value, component.ImplanterSlot.ContainerSlot);
- implantComp.ImplantedEntity = target;
- implantContainer.OccludesLight = false;
- _container.Insert(implant.Value, implantContainer);
- if (component.CurrentMode == ImplanterToggleMode.Inject && !component.ImplantOnly)
- DrawMode(implanter, component);
- else
- ImplantMode(implanter, component);
- var ev = new TransferDnaEvent { Donor = target, Recipient = implanter };
- RaiseLocalEvent(target, ref ev);
- Dirty(implanter, component);
- }
- public bool CanImplant(
- EntityUid user,
- EntityUid target,
- EntityUid implanter,
- ImplanterComponent component,
- [NotNullWhen(true)] out EntityUid? implant,
- [NotNullWhen(true)] out SubdermalImplantComponent? implantComp)
- {
- implant = component.ImplanterSlot.ContainerSlot?.ContainedEntities.FirstOrNull();
- if (!TryComp(implant, out implantComp))
- return false;
- if (!CheckTarget(target, component.Whitelist, component.Blacklist) ||
- !CheckTarget(target, implantComp.Whitelist, implantComp.Blacklist))
- {
- return false;
- }
- var ev = new AddImplantAttemptEvent(user, target, implant.Value, implanter);
- RaiseLocalEvent(target, ev);
- return !ev.Cancelled;
- }
- protected bool CheckTarget(EntityUid target, EntityWhitelist? whitelist, EntityWhitelist? blacklist)
- {
- return _whitelistSystem.IsWhitelistPassOrNull(whitelist, target) &&
- _whitelistSystem.IsBlacklistFailOrNull(blacklist, target);
- }
- //Draw the implant out of the target
- //TODO: Rework when surgery is in so implant cases can be a thing
- public void Draw(EntityUid implanter, EntityUid user, EntityUid target, ImplanterComponent component)
- {
- var implanterContainer = component.ImplanterSlot.ContainerSlot;
- if (implanterContainer is null)
- return;
- var permanentFound = false;
- if (_container.TryGetContainer(target, ImplanterComponent.ImplantSlotId, out var implantContainer))
- {
- var implantCompQuery = GetEntityQuery<SubdermalImplantComponent>();
- if (component.AllowDeimplantAll)
- {
- foreach (var implant in implantContainer.ContainedEntities)
- {
- if (!implantCompQuery.TryGetComponent(implant, out var implantComp))
- continue;
- //Don't remove a permanent implant and look for the next that can be drawn
- if (!_container.CanRemove(implant, implantContainer))
- {
- DrawPermanentFailurePopup(implant, target, user);
- permanentFound = implantComp.Permanent;
- continue;
- }
- DrawImplantIntoImplanter(implanter, target, implant, implantContainer, implanterContainer, implantComp);
- permanentFound = implantComp.Permanent;
- //Break so only one implant is drawn
- break;
- }
- if (component.CurrentMode == ImplanterToggleMode.Draw && !component.ImplantOnly && !permanentFound)
- ImplantMode(implanter, component);
- }
- else
- {
- EntityUid? implant = null;
- var implants = implantContainer.ContainedEntities;
- foreach (var implantEntity in implants)
- {
- if (TryComp<SubdermalImplantComponent>(implantEntity, out var subdermalComp))
- {
- if (component.DeimplantChosen == subdermalComp.DrawableProtoIdOverride ||
- (Prototype(implantEntity) != null && component.DeimplantChosen == Prototype(implantEntity)!))
- implant = implantEntity;
- }
- }
- if (implant != null && implantCompQuery.TryGetComponent(implant, out var implantComp))
- {
- //Don't remove a permanent implant
- if (!_container.CanRemove(implant.Value, implantContainer))
- {
- DrawPermanentFailurePopup(implant.Value, target, user);
- permanentFound = implantComp.Permanent;
- }
- else
- {
- DrawImplantIntoImplanter(implanter, target, implant.Value, implantContainer, implanterContainer, implantComp);
- permanentFound = implantComp.Permanent;
- }
- if (component.CurrentMode == ImplanterToggleMode.Draw && !component.ImplantOnly && !permanentFound)
- ImplantMode(implanter, component);
- }
- else
- {
- DrawCatastrophicFailure(implanter, component, user);
- }
- }
- Dirty(implanter, component);
- }
- else
- {
- DrawCatastrophicFailure(implanter, component, user);
- }
- }
- private void DrawPermanentFailurePopup(EntityUid implant, EntityUid target, EntityUid user)
- {
- var implantName = Identity.Entity(implant, EntityManager);
- var targetName = Identity.Entity(target, EntityManager);
- var failedPermanentMessage = Loc.GetString("implanter-draw-failed-permanent",
- ("implant", implantName), ("target", targetName));
- _popup.PopupEntity(failedPermanentMessage, target, user);
- }
- private void DrawImplantIntoImplanter(EntityUid implanter, EntityUid target, EntityUid implant, BaseContainer implantContainer, ContainerSlot implanterContainer, SubdermalImplantComponent implantComp)
- {
- _container.Remove(implant, implantContainer);
- implantComp.ImplantedEntity = null;
- _container.Insert(implant, implanterContainer);
- var ev = new TransferDnaEvent { Donor = target, Recipient = implanter };
- RaiseLocalEvent(target, ref ev);
- }
- private void DrawCatastrophicFailure(EntityUid implanter, ImplanterComponent component, EntityUid user)
- {
- _damageableSystem.TryChangeDamage(user, component.DeimplantFailureDamage, ignoreResistances: true, origin: implanter);
- var userName = Identity.Entity(user, EntityManager);
- var failedCatastrophicallyMessage = Loc.GetString("implanter-draw-failed-catastrophically", ("user", userName));
- _popup.PopupEntity(failedCatastrophicallyMessage, user, PopupType.MediumCaution);
- }
- private void ImplantMode(EntityUid uid, ImplanterComponent component)
- {
- component.CurrentMode = ImplanterToggleMode.Inject;
- ChangeOnImplantVisualizer(uid, component);
- }
- private void DrawMode(EntityUid uid, ImplanterComponent component)
- {
- component.CurrentMode = ImplanterToggleMode.Draw;
- ChangeOnImplantVisualizer(uid, component);
- }
- private void ChangeOnImplantVisualizer(EntityUid uid, ImplanterComponent component)
- {
- if (!TryComp<AppearanceComponent>(uid, out var appearance))
- return;
- bool implantFound;
- if (component.ImplanterSlot.HasItem)
- implantFound = true;
- else
- implantFound = false;
- if (component.CurrentMode == ImplanterToggleMode.Inject && !component.ImplantOnly)
- _appearance.SetData(uid, ImplanterVisuals.Full, implantFound, appearance);
- else if (component.CurrentMode == ImplanterToggleMode.Inject && component.ImplantOnly)
- {
- _appearance.SetData(uid, ImplanterVisuals.Full, implantFound, appearance);
- _appearance.SetData(uid, ImplanterImplantOnlyVisuals.ImplantOnly, component.ImplantOnly,
- appearance);
- }
- else
- _appearance.SetData(uid, ImplanterVisuals.Full, implantFound, appearance);
- }
- public void SetSelectedDeimplant(EntityUid uid, string? implant, ImplanterComponent? component = null)
- {
- if (!Resolve(uid, ref component, false))
- return;
- if (implant != null && _proto.TryIndex(implant, out EntityPrototype? proto))
- component.DeimplantChosen = proto;
- Dirty(uid, component);
- }
- }
- [Serializable, NetSerializable]
- public sealed partial class ImplantEvent : SimpleDoAfterEvent
- {
- }
- [Serializable, NetSerializable]
- public sealed partial class DrawEvent : SimpleDoAfterEvent
- {
- }
- public sealed class AddImplantAttemptEvent : CancellableEntityEventArgs
- {
- public readonly EntityUid User;
- public readonly EntityUid Target;
- public readonly EntityUid Implant;
- public readonly EntityUid Implanter;
- public AddImplantAttemptEvent(EntityUid user, EntityUid target, EntityUid implant, EntityUid implanter)
- {
- User = user;
- Target = target;
- Implant = implant;
- Implanter = implanter;
- }
- }
- /// <summary>
- /// Change the chosen implanter in the UI.
- /// </summary>
- [Serializable, NetSerializable]
- public sealed class DeimplantChangeVerbMessage : BoundUserInterfaceMessage
- {
- public readonly string? Implant;
- public DeimplantChangeVerbMessage(string? implant)
- {
- Implant = implant;
- }
- }
- [Serializable, NetSerializable]
- public enum DeimplantUiKey : byte
- {
- Key
- }
|