mirror of
https://github.com/WWhiteDreamProject/wwdpublic.git
synced 2026-04-24 17:18:14 +03:00
## Mirror of PR #25978: [Fix SCRAM implant not working while cuffed. Incidentally fix freedom implant working while dead/crit](https://github.com/space-wizards/space-station-14/pull/25978) from <img src="https://avatars.githubusercontent.com/u/10567778?v=4" alt="space-wizards" width="22"/> [space-wizards](https://github.com/space-wizards)/[space-station-14](https://github.com/space-wizards/space-station-14) ###### `22e9d6562f21bdd4f0962d6e3b6fcdd81bb4c253` PR opened by <img src="https://avatars.githubusercontent.com/u/32041239?v=4" width="16"/><a href="https://github.com/nikthechampiongr"> nikthechampiongr</a> at 2024-03-10 19:33:05 UTC --- PR changed 13 files with 142 additions and 3 deletions. The PR had the following labels: - Status: Needs Review --- <details open="true"><summary><h1>Original Body</h1></summary> > fixes #25662 > > <!-- Please read these guidelines before opening your PR: https://docs.spacestation14.io/en/getting-started/pr-guideline --> > <!-- The text between the arrows are comments - they will not be visible on your PR. --> > > ## About the PR > <!-- What did you change in this PR? --> > This pr makes it so you can use the SCRAM implant while cuffed. Also fixes the freedom implant working while you are in crit, or even dead. > > ## Why / Balance > <!-- Why was it changed? Link any discussions or issues here. Please discuss how this would affect game balance. --> > The implant is made so you can escape horrible situations. It should as such work while in crit. > > ## Technical details > <!-- If this is a code change, summarize at high level how your new code works. This makes it easier to review. --> > ActionBlockers now has a specific check to see if someone is conscious. It is also used in the Interaction check now. Actions can now use that check if the bool is set. The bool is now set for the freedom and SCRAM! implant actions. Additionally the SCRAM! action now ignores whether the user can interact in order to be able to use it while cuffed. > > ## Media > <!-- > PRs which make ingame changes (adding clothing, items, new features, etc) are required to have media attached that showcase the changes. > Small fixes/refactors are exempt. > Any media may be used in SS14 progress reports, with clear credit given. > > If you're unsure whether your PR will require media, ask a maintainer. > > Check the box below to confirm that you have in fact seen this (put an X in the brackets, like [X]): > --> > > - [x] I have added screenshots/videos to this PR showcasing its changes ingame, **or** this PR does not require an ingame showcase > > ## Breaking changes > <!-- > List any breaking changes, including namespace, public class/method/field changes, prototype renames; and provide instructions for fixing them. This will be pasted in #codebase-changes. > --> > > **Changelog** > <!-- > Make players aware of new features and changes that could affect how they play the game by adding a Changelog entry. Please read the Changelog guidelines located at: https://docs.spacestation14.io/en/getting-started/pr-guideline#changelog > --> > > <!-- > Make sure to take this Changelog template out of the comment block in order for it to show up. > 🆑 > - add: Added fun! > - remove: Removed fun! > - tweak: Changed fun! > - fix: Fixed fun! > --> > 🆑 > fix: You can now use the SCRAM! implant while cuffed. > fix: The freedom implant can no longer be used while in crit, or dead. </details> Co-authored-by: SimpleStation14 <Unknown>
180 lines
6.8 KiB
C#
180 lines
6.8 KiB
C#
using Content.Shared.Bed.Sleep;
|
|
using Content.Shared.CombatMode.Pacification;
|
|
using Content.Shared.Damage.ForceSay;
|
|
using Content.Shared.Emoting;
|
|
using Content.Shared.Hands;
|
|
using Content.Shared.Interaction;
|
|
using Content.Shared.Interaction.Events;
|
|
using Content.Shared.Inventory.Events;
|
|
using Content.Shared.Item;
|
|
using Content.Shared.Mobs.Components;
|
|
using Content.Shared.Movement.Events;
|
|
using Content.Shared.Pointing;
|
|
using Content.Shared.Pulling.Events;
|
|
using Content.Shared.Speech;
|
|
using Content.Shared.Standing;
|
|
using Content.Shared.Strip.Components;
|
|
using Content.Shared.Throwing;
|
|
using Robust.Shared.Physics.Components;
|
|
|
|
namespace Content.Shared.Mobs.Systems;
|
|
|
|
public partial class MobStateSystem
|
|
{
|
|
//General purpose event subscriptions. If you can avoid it register these events inside their own systems
|
|
private void SubscribeEvents()
|
|
{
|
|
SubscribeLocalEvent<MobStateComponent, BeforeGettingStrippedEvent>(OnGettingStripped);
|
|
SubscribeLocalEvent<MobStateComponent, ChangeDirectionAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, UseAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, AttackAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, ConsciousAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, ThrowAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, SpeakAttemptEvent>(OnSpeakAttempt);
|
|
SubscribeLocalEvent<MobStateComponent, IsEquippingAttemptEvent>(OnEquipAttempt);
|
|
SubscribeLocalEvent<MobStateComponent, EmoteAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, IsUnequippingAttemptEvent>(OnUnequipAttempt);
|
|
SubscribeLocalEvent<MobStateComponent, DropAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, PickupAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, StartPullAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, UpdateCanMoveEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, StandAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, PointAttemptEvent>(CheckAct);
|
|
SubscribeLocalEvent<MobStateComponent, TryingToSleepEvent>(OnSleepAttempt);
|
|
SubscribeLocalEvent<MobStateComponent, CombatModeShouldHandInteractEvent>(OnCombatModeShouldHandInteract);
|
|
SubscribeLocalEvent<MobStateComponent, AttemptPacifiedAttackEvent>(OnAttemptPacifiedAttack);
|
|
}
|
|
|
|
private void OnStateExitSubscribers(EntityUid target, MobStateComponent component, MobState state)
|
|
{
|
|
switch (state)
|
|
{
|
|
case MobState.Alive:
|
|
//unused
|
|
break;
|
|
case MobState.Critical:
|
|
_standing.Stand(target);
|
|
break;
|
|
case MobState.Dead:
|
|
RemComp<CollisionWakeComponent>(target);
|
|
_standing.Stand(target);
|
|
if (!_standing.IsDown(target) && TryComp<PhysicsComponent>(target, out var physics))
|
|
{
|
|
_physics.SetCanCollide(target, true, body: physics);
|
|
}
|
|
|
|
break;
|
|
case MobState.Invalid:
|
|
//unused
|
|
break;
|
|
default:
|
|
throw new NotImplementedException();
|
|
}
|
|
}
|
|
|
|
private void OnStateEnteredSubscribers(EntityUid target, MobStateComponent component, MobState state)
|
|
{
|
|
// All of the state changes here should already be networked, so we do nothing if we are currently applying a
|
|
// server state.
|
|
if (_timing.ApplyingState)
|
|
return;
|
|
|
|
_blocker.UpdateCanMove(target); //update movement anytime a state changes
|
|
switch (state)
|
|
{
|
|
case MobState.Alive:
|
|
_standing.Stand(target);
|
|
_appearance.SetData(target, MobStateVisuals.State, MobState.Alive);
|
|
break;
|
|
case MobState.Critical:
|
|
_standing.Down(target);
|
|
_appearance.SetData(target, MobStateVisuals.State, MobState.Critical);
|
|
break;
|
|
case MobState.Dead:
|
|
EnsureComp<CollisionWakeComponent>(target);
|
|
_standing.Down(target);
|
|
|
|
if (_standing.IsDown(target) && TryComp<PhysicsComponent>(target, out var physics))
|
|
{
|
|
_physics.SetCanCollide(target, false, body: physics);
|
|
}
|
|
|
|
_appearance.SetData(target, MobStateVisuals.State, MobState.Dead);
|
|
break;
|
|
case MobState.Invalid:
|
|
//unused;
|
|
break;
|
|
default:
|
|
throw new NotImplementedException();
|
|
}
|
|
}
|
|
|
|
#region Event Subscribers
|
|
|
|
private void OnSleepAttempt(EntityUid target, MobStateComponent component, ref TryingToSleepEvent args)
|
|
{
|
|
if (IsDead(target, component))
|
|
args.Cancelled = true;
|
|
}
|
|
|
|
private void OnGettingStripped(EntityUid target, MobStateComponent component, BeforeGettingStrippedEvent args)
|
|
{
|
|
// Incapacitated or dead targets get stripped two or three times as fast. Makes stripping corpses less tedious.
|
|
if (IsDead(target, component))
|
|
args.Multiplier /= 3;
|
|
else if (IsCritical(target, component))
|
|
args.Multiplier /= 2;
|
|
}
|
|
|
|
private void OnSpeakAttempt(EntityUid uid, MobStateComponent component, SpeakAttemptEvent args)
|
|
{
|
|
if (HasComp<AllowNextCritSpeechComponent>(uid))
|
|
{
|
|
RemCompDeferred<AllowNextCritSpeechComponent>(uid);
|
|
return;
|
|
}
|
|
|
|
CheckAct(uid, component, args);
|
|
}
|
|
|
|
private void CheckAct(EntityUid target, MobStateComponent component, CancellableEntityEventArgs args)
|
|
{
|
|
switch (component.CurrentState)
|
|
{
|
|
case MobState.Dead:
|
|
case MobState.Critical:
|
|
args.Cancel();
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void OnEquipAttempt(EntityUid target, MobStateComponent component, IsEquippingAttemptEvent args)
|
|
{
|
|
// is this a self-equip, or are they being stripped?
|
|
if (args.Equipee == target)
|
|
CheckAct(target, component, args);
|
|
}
|
|
|
|
private void OnUnequipAttempt(EntityUid target, MobStateComponent component, IsUnequippingAttemptEvent args)
|
|
{
|
|
// is this a self-equip, or are they being stripped?
|
|
if (args.Unequipee == target)
|
|
CheckAct(target, component, args);
|
|
}
|
|
|
|
private void OnCombatModeShouldHandInteract(EntityUid uid, MobStateComponent component, ref CombatModeShouldHandInteractEvent args)
|
|
{
|
|
// Disallow empty-hand-interacting in combat mode
|
|
// for non-dead mobs
|
|
if (!IsDead(uid, component))
|
|
args.Cancelled = true;
|
|
}
|
|
|
|
private void OnAttemptPacifiedAttack(Entity<MobStateComponent> ent, ref AttemptPacifiedAttackEvent args)
|
|
{
|
|
args.Cancelled = true;
|
|
}
|
|
|
|
#endregion
|
|
}
|