1
0

SharedImplanterSystem.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. using System.Diagnostics.CodeAnalysis;
  2. using System.Linq;
  3. using Content.Shared.Containers.ItemSlots;
  4. using Content.Shared.Damage;
  5. using Content.Shared.DoAfter;
  6. using Content.Shared.Examine;
  7. using Content.Shared.Forensics;
  8. using Content.Shared.IdentityManagement;
  9. using Content.Shared.Implants.Components;
  10. using Content.Shared.Interaction.Events;
  11. using Content.Shared.Popups;
  12. using Content.Shared.Verbs;
  13. using Content.Shared.Whitelist;
  14. using Robust.Shared.Containers;
  15. using Robust.Shared.Prototypes;
  16. using Robust.Shared.Serialization;
  17. using Robust.Shared.Utility;
  18. namespace Content.Shared.Implants;
  19. public abstract class SharedImplanterSystem : EntitySystem
  20. {
  21. [Dependency] private readonly SharedContainerSystem _container = default!;
  22. [Dependency] private readonly ItemSlotsSystem _itemSlots = default!;
  23. [Dependency] private readonly SharedAppearanceSystem _appearance = default!;
  24. [Dependency] private readonly SharedPopupSystem _popup = default!;
  25. [Dependency] private readonly EntityWhitelistSystem _whitelistSystem = default!;
  26. [Dependency] private readonly DamageableSystem _damageableSystem = default!;
  27. [Dependency] private readonly SharedUserInterfaceSystem _uiSystem = default!;
  28. [Dependency] private readonly IPrototypeManager _proto = default!;
  29. public override void Initialize()
  30. {
  31. base.Initialize();
  32. SubscribeLocalEvent<ImplanterComponent, ComponentInit>(OnImplanterInit);
  33. SubscribeLocalEvent<ImplanterComponent, EntInsertedIntoContainerMessage>(OnEntInserted);
  34. SubscribeLocalEvent<ImplanterComponent, ExaminedEvent>(OnExamine);
  35. SubscribeLocalEvent<ImplanterComponent, UseInHandEvent>(OnUseInHand);
  36. SubscribeLocalEvent<ImplanterComponent, GetVerbsEvent<InteractionVerb>>(OnVerb);
  37. SubscribeLocalEvent<ImplanterComponent, DeimplantChangeVerbMessage>(OnSelected);
  38. }
  39. private void OnImplanterInit(EntityUid uid, ImplanterComponent component, ComponentInit args)
  40. {
  41. if (component.Implant != null)
  42. component.ImplanterSlot.StartingItem = component.Implant;
  43. _itemSlots.AddItemSlot(uid, ImplanterComponent.ImplanterSlotId, component.ImplanterSlot);
  44. component.DeimplantChosen ??= component.DeimplantWhitelist.FirstOrNull();
  45. Dirty(uid, component);
  46. }
  47. private void OnEntInserted(EntityUid uid, ImplanterComponent component, EntInsertedIntoContainerMessage args)
  48. {
  49. var implantData = EntityManager.GetComponent<MetaDataComponent>(args.Entity);
  50. component.ImplantData = (implantData.EntityName, implantData.EntityDescription);
  51. }
  52. private void OnExamine(EntityUid uid, ImplanterComponent component, ExaminedEvent args)
  53. {
  54. if (!component.ImplanterSlot.HasItem || !args.IsInDetailsRange)
  55. return;
  56. args.PushMarkup(Loc.GetString("implanter-contained-implant-text", ("desc", component.ImplantData.Item2)));
  57. }
  58. public bool CheckSameImplant(EntityUid target, EntityUid implant)
  59. {
  60. if (!TryComp<ImplantedComponent>(target, out var implanted))
  61. return false;
  62. var implantPrototype = Prototype(implant);
  63. return implanted.ImplantContainer.ContainedEntities.Any(entity => Prototype(entity) == implantPrototype);
  64. }
  65. private void OnVerb(EntityUid uid, ImplanterComponent component, GetVerbsEvent<InteractionVerb> args)
  66. {
  67. if (!args.CanAccess || !args.CanInteract)
  68. return;
  69. if (component.CurrentMode == ImplanterToggleMode.Draw)
  70. {
  71. args.Verbs.Add(new InteractionVerb()
  72. {
  73. Text = Loc.GetString("implanter-set-draw-verb"),
  74. Act = () => TryOpenUi(uid, args.User, component)
  75. });
  76. }
  77. }
  78. private void OnUseInHand(EntityUid uid, ImplanterComponent? component, UseInHandEvent args)
  79. {
  80. if (!Resolve(uid, ref component))
  81. return;
  82. if (component.CurrentMode == ImplanterToggleMode.Draw)
  83. TryOpenUi(uid, args.User, component);
  84. }
  85. private void OnSelected(EntityUid uid, ImplanterComponent component, DeimplantChangeVerbMessage args)
  86. {
  87. component.DeimplantChosen = args.Implant;
  88. SetSelectedDeimplant(uid, args.Implant, component: component);
  89. }
  90. private void TryOpenUi(EntityUid uid, EntityUid user, ImplanterComponent? component = null)
  91. {
  92. if (!Resolve(uid, ref component))
  93. return;
  94. _uiSystem.TryToggleUi(uid, DeimplantUiKey.Key, user);
  95. component.DeimplantChosen ??= component.DeimplantWhitelist.FirstOrNull();
  96. Dirty(uid, component);
  97. }
  98. //Instantly implant something and add all necessary components and containers.
  99. //Set to draw mode if not implant only
  100. public void Implant(EntityUid user, EntityUid target, EntityUid implanter, ImplanterComponent component)
  101. {
  102. if (!CanImplant(user, target, implanter, component, out var implant, out var implantComp))
  103. return;
  104. // Check if we are trying to implant a implant which is already implanted
  105. // Check AFTER the doafter to prevent "is it a fake?" metagaming against deceptive implants
  106. if (!component.AllowMultipleImplants && CheckSameImplant(target, implant.Value))
  107. {
  108. var name = Identity.Name(target, EntityManager, user);
  109. var msg = Loc.GetString("implanter-component-implant-already", ("implant", implant), ("target", name));
  110. _popup.PopupEntity(msg, target, user);
  111. return;
  112. }
  113. //If the target doesn't have the implanted component, add it.
  114. var implantedComp = EnsureComp<ImplantedComponent>(target);
  115. var implantContainer = implantedComp.ImplantContainer;
  116. if (component.ImplanterSlot.ContainerSlot != null)
  117. _container.Remove(implant.Value, component.ImplanterSlot.ContainerSlot);
  118. implantComp.ImplantedEntity = target;
  119. implantContainer.OccludesLight = false;
  120. _container.Insert(implant.Value, implantContainer);
  121. if (component.CurrentMode == ImplanterToggleMode.Inject && !component.ImplantOnly)
  122. DrawMode(implanter, component);
  123. else
  124. ImplantMode(implanter, component);
  125. var ev = new TransferDnaEvent { Donor = target, Recipient = implanter };
  126. RaiseLocalEvent(target, ref ev);
  127. Dirty(implanter, component);
  128. }
  129. public bool CanImplant(
  130. EntityUid user,
  131. EntityUid target,
  132. EntityUid implanter,
  133. ImplanterComponent component,
  134. [NotNullWhen(true)] out EntityUid? implant,
  135. [NotNullWhen(true)] out SubdermalImplantComponent? implantComp)
  136. {
  137. implant = component.ImplanterSlot.ContainerSlot?.ContainedEntities.FirstOrNull();
  138. if (!TryComp(implant, out implantComp))
  139. return false;
  140. if (!CheckTarget(target, component.Whitelist, component.Blacklist) ||
  141. !CheckTarget(target, implantComp.Whitelist, implantComp.Blacklist))
  142. {
  143. return false;
  144. }
  145. var ev = new AddImplantAttemptEvent(user, target, implant.Value, implanter);
  146. RaiseLocalEvent(target, ev);
  147. return !ev.Cancelled;
  148. }
  149. protected bool CheckTarget(EntityUid target, EntityWhitelist? whitelist, EntityWhitelist? blacklist)
  150. {
  151. return _whitelistSystem.IsWhitelistPassOrNull(whitelist, target) &&
  152. _whitelistSystem.IsBlacklistFailOrNull(blacklist, target);
  153. }
  154. //Draw the implant out of the target
  155. //TODO: Rework when surgery is in so implant cases can be a thing
  156. public void Draw(EntityUid implanter, EntityUid user, EntityUid target, ImplanterComponent component)
  157. {
  158. var implanterContainer = component.ImplanterSlot.ContainerSlot;
  159. if (implanterContainer is null)
  160. return;
  161. var permanentFound = false;
  162. if (_container.TryGetContainer(target, ImplanterComponent.ImplantSlotId, out var implantContainer))
  163. {
  164. var implantCompQuery = GetEntityQuery<SubdermalImplantComponent>();
  165. if (component.AllowDeimplantAll)
  166. {
  167. foreach (var implant in implantContainer.ContainedEntities)
  168. {
  169. if (!implantCompQuery.TryGetComponent(implant, out var implantComp))
  170. continue;
  171. //Don't remove a permanent implant and look for the next that can be drawn
  172. if (!_container.CanRemove(implant, implantContainer))
  173. {
  174. DrawPermanentFailurePopup(implant, target, user);
  175. permanentFound = implantComp.Permanent;
  176. continue;
  177. }
  178. DrawImplantIntoImplanter(implanter, target, implant, implantContainer, implanterContainer, implantComp);
  179. permanentFound = implantComp.Permanent;
  180. //Break so only one implant is drawn
  181. break;
  182. }
  183. if (component.CurrentMode == ImplanterToggleMode.Draw && !component.ImplantOnly && !permanentFound)
  184. ImplantMode(implanter, component);
  185. }
  186. else
  187. {
  188. EntityUid? implant = null;
  189. var implants = implantContainer.ContainedEntities;
  190. foreach (var implantEntity in implants)
  191. {
  192. if (TryComp<SubdermalImplantComponent>(implantEntity, out var subdermalComp))
  193. {
  194. if (component.DeimplantChosen == subdermalComp.DrawableProtoIdOverride ||
  195. (Prototype(implantEntity) != null && component.DeimplantChosen == Prototype(implantEntity)!))
  196. implant = implantEntity;
  197. }
  198. }
  199. if (implant != null && implantCompQuery.TryGetComponent(implant, out var implantComp))
  200. {
  201. //Don't remove a permanent implant
  202. if (!_container.CanRemove(implant.Value, implantContainer))
  203. {
  204. DrawPermanentFailurePopup(implant.Value, target, user);
  205. permanentFound = implantComp.Permanent;
  206. }
  207. else
  208. {
  209. DrawImplantIntoImplanter(implanter, target, implant.Value, implantContainer, implanterContainer, implantComp);
  210. permanentFound = implantComp.Permanent;
  211. }
  212. if (component.CurrentMode == ImplanterToggleMode.Draw && !component.ImplantOnly && !permanentFound)
  213. ImplantMode(implanter, component);
  214. }
  215. else
  216. {
  217. DrawCatastrophicFailure(implanter, component, user);
  218. }
  219. }
  220. Dirty(implanter, component);
  221. }
  222. else
  223. {
  224. DrawCatastrophicFailure(implanter, component, user);
  225. }
  226. }
  227. private void DrawPermanentFailurePopup(EntityUid implant, EntityUid target, EntityUid user)
  228. {
  229. var implantName = Identity.Entity(implant, EntityManager);
  230. var targetName = Identity.Entity(target, EntityManager);
  231. var failedPermanentMessage = Loc.GetString("implanter-draw-failed-permanent",
  232. ("implant", implantName), ("target", targetName));
  233. _popup.PopupEntity(failedPermanentMessage, target, user);
  234. }
  235. private void DrawImplantIntoImplanter(EntityUid implanter, EntityUid target, EntityUid implant, BaseContainer implantContainer, ContainerSlot implanterContainer, SubdermalImplantComponent implantComp)
  236. {
  237. _container.Remove(implant, implantContainer);
  238. implantComp.ImplantedEntity = null;
  239. _container.Insert(implant, implanterContainer);
  240. var ev = new TransferDnaEvent { Donor = target, Recipient = implanter };
  241. RaiseLocalEvent(target, ref ev);
  242. }
  243. private void DrawCatastrophicFailure(EntityUid implanter, ImplanterComponent component, EntityUid user)
  244. {
  245. _damageableSystem.TryChangeDamage(user, component.DeimplantFailureDamage, ignoreResistances: true, origin: implanter);
  246. var userName = Identity.Entity(user, EntityManager);
  247. var failedCatastrophicallyMessage = Loc.GetString("implanter-draw-failed-catastrophically", ("user", userName));
  248. _popup.PopupEntity(failedCatastrophicallyMessage, user, PopupType.MediumCaution);
  249. }
  250. private void ImplantMode(EntityUid uid, ImplanterComponent component)
  251. {
  252. component.CurrentMode = ImplanterToggleMode.Inject;
  253. ChangeOnImplantVisualizer(uid, component);
  254. }
  255. private void DrawMode(EntityUid uid, ImplanterComponent component)
  256. {
  257. component.CurrentMode = ImplanterToggleMode.Draw;
  258. ChangeOnImplantVisualizer(uid, component);
  259. }
  260. private void ChangeOnImplantVisualizer(EntityUid uid, ImplanterComponent component)
  261. {
  262. if (!TryComp<AppearanceComponent>(uid, out var appearance))
  263. return;
  264. bool implantFound;
  265. if (component.ImplanterSlot.HasItem)
  266. implantFound = true;
  267. else
  268. implantFound = false;
  269. if (component.CurrentMode == ImplanterToggleMode.Inject && !component.ImplantOnly)
  270. _appearance.SetData(uid, ImplanterVisuals.Full, implantFound, appearance);
  271. else if (component.CurrentMode == ImplanterToggleMode.Inject && component.ImplantOnly)
  272. {
  273. _appearance.SetData(uid, ImplanterVisuals.Full, implantFound, appearance);
  274. _appearance.SetData(uid, ImplanterImplantOnlyVisuals.ImplantOnly, component.ImplantOnly,
  275. appearance);
  276. }
  277. else
  278. _appearance.SetData(uid, ImplanterVisuals.Full, implantFound, appearance);
  279. }
  280. public void SetSelectedDeimplant(EntityUid uid, string? implant, ImplanterComponent? component = null)
  281. {
  282. if (!Resolve(uid, ref component, false))
  283. return;
  284. if (implant != null && _proto.TryIndex(implant, out EntityPrototype? proto))
  285. component.DeimplantChosen = proto;
  286. Dirty(uid, component);
  287. }
  288. }
  289. [Serializable, NetSerializable]
  290. public sealed partial class ImplantEvent : SimpleDoAfterEvent
  291. {
  292. }
  293. [Serializable, NetSerializable]
  294. public sealed partial class DrawEvent : SimpleDoAfterEvent
  295. {
  296. }
  297. public sealed class AddImplantAttemptEvent : CancellableEntityEventArgs
  298. {
  299. public readonly EntityUid User;
  300. public readonly EntityUid Target;
  301. public readonly EntityUid Implant;
  302. public readonly EntityUid Implanter;
  303. public AddImplantAttemptEvent(EntityUid user, EntityUid target, EntityUid implant, EntityUid implanter)
  304. {
  305. User = user;
  306. Target = target;
  307. Implant = implant;
  308. Implanter = implanter;
  309. }
  310. }
  311. /// <summary>
  312. /// Change the chosen implanter in the UI.
  313. /// </summary>
  314. [Serializable, NetSerializable]
  315. public sealed class DeimplantChangeVerbMessage : BoundUserInterfaceMessage
  316. {
  317. public readonly string? Implant;
  318. public DeimplantChangeVerbMessage(string? implant)
  319. {
  320. Implant = implant;
  321. }
  322. }
  323. [Serializable, NetSerializable]
  324. public enum DeimplantUiKey : byte
  325. {
  326. Key
  327. }