Blindness refactor (#15705)
Co-authored-by: metalgearsloth <comedian_vs_clown@hotmail.com>
This commit is contained in:
@@ -15,6 +15,7 @@ using Robust.Shared.Map;
|
|||||||
using Robust.Shared.Utility;
|
using Robust.Shared.Utility;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
using static Content.Shared.Interaction.SharedInteractionSystem;
|
using static Content.Shared.Interaction.SharedInteractionSystem;
|
||||||
using static Robust.Client.UserInterface.Controls.BoxContainer;
|
using static Robust.Client.UserInterface.Controls.BoxContainer;
|
||||||
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ using Robust.Client.Player;
|
|||||||
using Robust.Shared.Enums;
|
using Robust.Shared.Enums;
|
||||||
using Robust.Shared.Prototypes;
|
using Robust.Shared.Prototypes;
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
namespace Content.Client.Eye.Blinding
|
namespace Content.Client.Eye.Blinding
|
||||||
{
|
{
|
||||||
@@ -46,7 +47,7 @@ namespace Content.Client.Eye.Blinding
|
|||||||
|
|
||||||
_blindableComponent = blindComp;
|
_blindableComponent = blindComp;
|
||||||
|
|
||||||
var blind = _blindableComponent.Sources > 0;
|
var blind = _blindableComponent.IsBlind;
|
||||||
|
|
||||||
if (!blind && _blindableComponent.LightSetup) // Turn FOV back on if we can see again
|
if (!blind && _blindableComponent.LightSetup) // Turn FOV back on if we can see again
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ using System.Collections.Generic;
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
using Content.Shared.Administration;
|
using Content.Shared.Administration;
|
||||||
using Content.Shared.Administration.Events;
|
using Content.Shared.Administration.Events;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
using Content.Shared.GameTicking;
|
using Content.Shared.GameTicking;
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.Network;
|
using Robust.Shared.Network;
|
||||||
|
|||||||
@@ -4,24 +4,21 @@ using Robust.Client.Player;
|
|||||||
using Robust.Shared.Enums;
|
using Robust.Shared.Enums;
|
||||||
using Robust.Shared.Prototypes;
|
using Robust.Shared.Prototypes;
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
namespace Content.Client.Eye.Blinding
|
namespace Content.Client.Eye.Blinding
|
||||||
{
|
{
|
||||||
public sealed class BlurryVisionOverlay : Overlay
|
public sealed class BlurryVisionOverlay : Overlay
|
||||||
{
|
{
|
||||||
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
|
|
||||||
[Dependency] private readonly IEntityManager _entityManager = default!;
|
[Dependency] private readonly IEntityManager _entityManager = default!;
|
||||||
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
[Dependency] private readonly IPlayerManager _playerManager = default!;
|
||||||
|
|
||||||
public override bool RequestScreenTexture => true;
|
|
||||||
public override OverlaySpace Space => OverlaySpace.WorldSpace;
|
public override OverlaySpace Space => OverlaySpace.WorldSpace;
|
||||||
private readonly ShaderInstance _dim;
|
private float _magnitude;
|
||||||
private BlurryVisionComponent _blurryVisionComponent = default!;
|
|
||||||
|
|
||||||
public BlurryVisionOverlay()
|
public BlurryVisionOverlay()
|
||||||
{
|
{
|
||||||
IoCManager.InjectDependencies(this);
|
IoCManager.InjectDependencies(this);
|
||||||
_dim = _prototypeManager.Index<ShaderPrototype>("Dim").InstanceUnique();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool BeforeDraw(in OverlayDrawArgs args)
|
protected override bool BeforeDraw(in OverlayDrawArgs args)
|
||||||
@@ -40,33 +37,29 @@ namespace Content.Client.Eye.Blinding
|
|||||||
if (!_entityManager.TryGetComponent<BlurryVisionComponent>(playerEntity, out var blurComp))
|
if (!_entityManager.TryGetComponent<BlurryVisionComponent>(playerEntity, out var blurComp))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!blurComp.Active)
|
if (blurComp.Magnitude <= 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (_entityManager.TryGetComponent<BlindableComponent>(playerEntity, out var blindComp)
|
if (_entityManager.TryGetComponent<BlindableComponent>(playerEntity, out var blindComp)
|
||||||
&& blindComp.Sources > 0)
|
&& blindComp.IsBlind)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
_blurryVisionComponent = blurComp;
|
_magnitude = blurComp.Magnitude;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void Draw(in OverlayDrawArgs args)
|
protected override void Draw(in OverlayDrawArgs args)
|
||||||
{
|
{
|
||||||
if (ScreenTexture == null)
|
// TODO make this better.
|
||||||
return;
|
// This is a really shitty effect.
|
||||||
|
// Maybe gradually shrink the view-size?
|
||||||
var opacity = -(_blurryVisionComponent.Magnitude / 15) + 0.9f;
|
// Make the effect only apply to the edge of the viewport?
|
||||||
|
// Actually make it blurry??
|
||||||
_dim.SetParameter("DAMAGE_AMOUNT", opacity);
|
var opacity = 0.5f * _magnitude / BlurryVisionComponent.MaxMagnitude;
|
||||||
|
|
||||||
var worldHandle = args.WorldHandle;
|
var worldHandle = args.WorldHandle;
|
||||||
var viewport = args.WorldBounds;
|
var viewport = args.WorldBounds;
|
||||||
|
|
||||||
worldHandle.UseShader(_dim);
|
|
||||||
worldHandle.SetTransform(Matrix3.Identity);
|
worldHandle.SetTransform(Matrix3.Identity);
|
||||||
worldHandle.DrawRect(viewport, Color.Black);
|
worldHandle.DrawRect(viewport, Color.White.WithAlpha(opacity));
|
||||||
worldHandle.UseShader(null);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
using Robust.Client.GameObjects;
|
using Robust.Client.GameObjects;
|
||||||
using Robust.Client.Graphics;
|
using Robust.Client.Graphics;
|
||||||
using Robust.Client.Player;
|
using Robust.Client.Player;
|
||||||
@@ -22,8 +23,6 @@ public sealed class BlurryVisionSystem : EntitySystem
|
|||||||
SubscribeLocalEvent<BlurryVisionComponent, PlayerAttachedEvent>(OnPlayerAttached);
|
SubscribeLocalEvent<BlurryVisionComponent, PlayerAttachedEvent>(OnPlayerAttached);
|
||||||
SubscribeLocalEvent<BlurryVisionComponent, PlayerDetachedEvent>(OnPlayerDetached);
|
SubscribeLocalEvent<BlurryVisionComponent, PlayerDetachedEvent>(OnPlayerDetached);
|
||||||
|
|
||||||
SubscribeLocalEvent<BlurryVisionComponent, ComponentHandleState>(OnHandleState);
|
|
||||||
|
|
||||||
_overlay = new();
|
_overlay = new();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,12 +49,4 @@ public sealed class BlurryVisionSystem : EntitySystem
|
|||||||
_overlayMan.RemoveOverlay(_overlay);
|
_overlayMan.RemoveOverlay(_overlay);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnHandleState(EntityUid uid, BlurryVisionComponent component, ref ComponentHandleState args)
|
|
||||||
{
|
|
||||||
if (args.Current is not BlurryVisionComponentState state)
|
|
||||||
return;
|
|
||||||
|
|
||||||
component.Magnitude = state.Magnitude;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,28 @@
|
|||||||
using Content.Shared.Chemistry.Reagent;
|
using Content.Shared.Chemistry.Reagent;
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
using JetBrains.Annotations;
|
using JetBrains.Annotations;
|
||||||
|
|
||||||
namespace Content.Server.Chemistry.ReagentEffects
|
namespace Content.Server.Chemistry.ReagentEffects
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Heal eye damage (or deal)
|
/// Heal or apply eye damage
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[UsedImplicitly]
|
[UsedImplicitly]
|
||||||
public sealed class ChemHealEyeDamage : ReagentEffect
|
public sealed class ChemHealEyeDamage : ReagentEffect
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// How much eye damage to remove.
|
/// How much eye damage to add.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[DataField("amount")]
|
[DataField("amount")]
|
||||||
public int Amount = -1;
|
public int Amount = -1;
|
||||||
|
|
||||||
public override void Effect(ReagentEffectArgs args)
|
public override void Effect(ReagentEffectArgs args)
|
||||||
{
|
{
|
||||||
if (args.Scale != 1f)
|
if (args.Scale != 1f) // huh?
|
||||||
return;
|
return;
|
||||||
|
|
||||||
args.EntityManager.EntitySysManager.GetEntitySystem<SharedBlindingSystem>().AdjustEyeDamage(args.SolutionEntity, Amount);
|
args.EntityManager.EntitySysManager.GetEntitySystem<BlindableSystem>().AdjustEyeDamage(args.SolutionEntity, Amount);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
using Content.Server.UserInterface;
|
using Content.Server.UserInterface;
|
||||||
using Content.Server.Popups;
|
using Content.Server.Popups;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
using Robust.Shared.Player;
|
using Robust.Shared.Player;
|
||||||
using Robust.Server.GameObjects;
|
using Robust.Server.GameObjects;
|
||||||
|
|
||||||
@@ -23,14 +25,14 @@ public sealed class ActivatableUIRequiresVisionSystem : EntitySystem
|
|||||||
if (args.Cancelled)
|
if (args.Cancelled)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (TryComp<BlindableComponent>(args.User, out var blindable) && blindable.Sources > 0)
|
if (TryComp<BlindableComponent>(args.User, out var blindable) && blindable.IsBlind)
|
||||||
{
|
{
|
||||||
_popupSystem.PopupCursor(Loc.GetString("blindness-fail-attempt"), args.User, Shared.Popups.PopupType.MediumCaution);
|
_popupSystem.PopupCursor(Loc.GetString("blindness-fail-attempt"), args.User, Shared.Popups.PopupType.MediumCaution);
|
||||||
args.Cancel();
|
args.Cancel();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnBlindnessChanged(EntityUid uid, BlindableComponent component, BlindnessChangedEvent args)
|
private void OnBlindnessChanged(EntityUid uid, BlindableComponent component, ref BlindnessChangedEvent args)
|
||||||
{
|
{
|
||||||
if (!args.Blind)
|
if (!args.Blind)
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
using Content.Shared.Eye.Blinding.EyeProtection; // why aren't tools predicted 🙂
|
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
using Content.Shared.StatusEffect;
|
using Content.Shared.StatusEffect;
|
||||||
using Content.Shared.Clothing.Components;
|
|
||||||
using Content.Shared.Inventory;
|
using Content.Shared.Inventory;
|
||||||
using Content.Shared.Inventory.Events;
|
|
||||||
using Content.Server.Tools;
|
using Content.Server.Tools;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
using Content.Shared.Tools.Components;
|
using Content.Shared.Tools.Components;
|
||||||
|
|
||||||
namespace Content.Server.Eye.Blinding.EyeProtection
|
namespace Content.Server.Eye.Blinding.EyeProtection
|
||||||
@@ -12,15 +11,26 @@ namespace Content.Server.Eye.Blinding.EyeProtection
|
|||||||
public sealed class EyeProtectionSystem : EntitySystem
|
public sealed class EyeProtectionSystem : EntitySystem
|
||||||
{
|
{
|
||||||
[Dependency] private readonly StatusEffectsSystem _statusEffectsSystem = default!;
|
[Dependency] private readonly StatusEffectsSystem _statusEffectsSystem = default!;
|
||||||
[Dependency] private readonly SharedBlindingSystem _blindingSystem = default!;
|
[Dependency] private readonly BlindableSystem _blindingSystem = default!;
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
base.Initialize();
|
base.Initialize();
|
||||||
SubscribeLocalEvent<RequiresEyeProtectionComponent, ToolUseAttemptEvent>(OnUseAttempt);
|
SubscribeLocalEvent<RequiresEyeProtectionComponent, ToolUseAttemptEvent>(OnUseAttempt);
|
||||||
SubscribeLocalEvent<RequiresEyeProtectionComponent, WelderToggledEvent>(OnWelderToggled);
|
SubscribeLocalEvent<RequiresEyeProtectionComponent, WelderToggledEvent>(OnWelderToggled);
|
||||||
|
|
||||||
SubscribeLocalEvent<EyeProtectionComponent, GotEquippedEvent>(OnEquipped);
|
SubscribeLocalEvent<EyeProtectionComponent, GetEyeProtectionEvent>(OnGetProtection);
|
||||||
SubscribeLocalEvent<EyeProtectionComponent, GotUnequippedEvent>(OnUnequipped);
|
SubscribeLocalEvent<EyeProtectionComponent, InventoryRelayedEvent<GetEyeProtectionEvent>>(OnGetRelayedProtection);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnGetRelayedProtection(EntityUid uid, EyeProtectionComponent component,
|
||||||
|
InventoryRelayedEvent<GetEyeProtectionEvent> args)
|
||||||
|
{
|
||||||
|
OnGetProtection(uid, component, args.Args);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnGetProtection(EntityUid uid, EyeProtectionComponent component, GetEyeProtectionEvent args)
|
||||||
|
{
|
||||||
|
args.Protection += component.ProtectionTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnUseAttempt(EntityUid uid, RequiresEyeProtectionComponent component, ToolUseAttemptEvent args)
|
private void OnUseAttempt(EntityUid uid, RequiresEyeProtectionComponent component, ToolUseAttemptEvent args)
|
||||||
@@ -28,51 +38,26 @@ namespace Content.Server.Eye.Blinding.EyeProtection
|
|||||||
if (!component.Toggled)
|
if (!component.Toggled)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (!HasComp<StatusEffectsComponent>(args.User) || !TryComp<BlindableComponent>(args.User, out var blindable))
|
if (!TryComp<BlindableComponent>(args.User, out var blindable) || blindable.IsBlind)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (blindable.Sources > 0)
|
var ev = new GetEyeProtectionEvent();
|
||||||
|
RaiseLocalEvent(args.User, ev);
|
||||||
|
|
||||||
|
var time = (float) (component.StatusEffectTime- ev.Protection).TotalSeconds;
|
||||||
|
if (time <= 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var statusTime = (float) component.StatusEffectTime.TotalSeconds - blindable.BlindResistance;
|
// Add permanent eye damage if they had zero protection, also somewhat scale their temporary blindness by
|
||||||
|
// how much damage they already accumulated.
|
||||||
if (statusTime <= 0)
|
_blindingSystem.AdjustEyeDamage(args.User, 1, blindable);
|
||||||
return;
|
var statusTimeSpan = TimeSpan.FromSeconds(time * MathF.Sqrt(blindable.EyeDamage));
|
||||||
|
_statusEffectsSystem.TryAddStatusEffect(args.User, TemporaryBlindnessSystem.BlindingStatusEffect,
|
||||||
var statusTimeSpan = TimeSpan.FromSeconds(statusTime * (blindable.EyeDamage + 1));
|
statusTimeSpan, false, TemporaryBlindnessSystem.BlindingStatusEffect);
|
||||||
// Add permanent eye damage if they had zero protection, also scale their temporary blindness by how much they already accumulated.
|
|
||||||
if (_statusEffectsSystem.TryAddStatusEffect(args.User, SharedBlindingSystem.BlindingStatusEffect, statusTimeSpan, false, "TemporaryBlindness") && blindable.BlindResistance <= 0)
|
|
||||||
_blindingSystem.AdjustEyeDamage(args.User, 1, blindable);
|
|
||||||
}
|
}
|
||||||
private void OnWelderToggled(EntityUid uid, RequiresEyeProtectionComponent component, WelderToggledEvent args)
|
private void OnWelderToggled(EntityUid uid, RequiresEyeProtectionComponent component, WelderToggledEvent args)
|
||||||
{
|
{
|
||||||
component.Toggled = args.WelderOn;
|
component.Toggled = args.WelderOn;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnEquipped(EntityUid uid, EyeProtectionComponent component, GotEquippedEvent args)
|
|
||||||
{
|
|
||||||
if (!TryComp<ClothingComponent>(uid, out var clothing) || clothing.Slots == SlotFlags.PREVENTEQUIP)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (!clothing.Slots.HasFlag(args.SlotFlags))
|
|
||||||
return;
|
|
||||||
|
|
||||||
component.IsActive = true;
|
|
||||||
if (!TryComp<BlindableComponent>(args.Equipee, out var blindComp))
|
|
||||||
return;
|
|
||||||
|
|
||||||
blindComp.BlindResistance += (float) component.ProtectionTime.TotalSeconds;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnUnequipped(EntityUid uid, EyeProtectionComponent component, GotUnequippedEvent args)
|
|
||||||
{
|
|
||||||
if (!component.IsActive)
|
|
||||||
return;
|
|
||||||
component.IsActive = false;
|
|
||||||
if (!TryComp<BlindableComponent>(args.Equipee, out var blindComp))
|
|
||||||
return;
|
|
||||||
|
|
||||||
blindComp.BlindResistance -= (float) component.ProtectionTime.TotalSeconds;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ using Content.Server.Light.EntitySystems;
|
|||||||
using Content.Server.Stunnable;
|
using Content.Server.Stunnable;
|
||||||
using Content.Shared.Examine;
|
using Content.Shared.Examine;
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
using Content.Shared.Flash;
|
using Content.Shared.Flash;
|
||||||
using Content.Shared.IdentityManagement;
|
using Content.Shared.IdentityManagement;
|
||||||
using Content.Shared.Interaction;
|
using Content.Shared.Interaction;
|
||||||
|
|||||||
@@ -1,20 +1,21 @@
|
|||||||
using Content.Shared.Eye.Blinding;
|
|
||||||
using Content.Shared.Speech;
|
using Content.Shared.Speech;
|
||||||
using Content.Shared.Actions;
|
using Content.Shared.Actions;
|
||||||
using Content.Shared.Bed.Sleep;
|
using Content.Shared.Bed.Sleep;
|
||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
|
|
||||||
namespace Content.Server.Bed.Sleep
|
namespace Content.Server.Bed.Sleep
|
||||||
{
|
{
|
||||||
public abstract class SharedSleepingSystem : EntitySystem
|
public abstract class SharedSleepingSystem : EntitySystem
|
||||||
{
|
{
|
||||||
[Dependency] private readonly SharedBlindingSystem _blindingSystem = default!;
|
[Dependency] private readonly BlindableSystem _blindableSystem = default!;
|
||||||
|
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
base.Initialize();
|
base.Initialize();
|
||||||
SubscribeLocalEvent<SleepingComponent, ComponentInit>(OnInit);
|
SubscribeLocalEvent<SleepingComponent, ComponentStartup>(OnStartup);
|
||||||
SubscribeLocalEvent<SleepingComponent, ComponentShutdown>(OnShutdown);
|
SubscribeLocalEvent<SleepingComponent, ComponentShutdown>(OnShutdown);
|
||||||
SubscribeLocalEvent<SleepingComponent, SpeakAttemptEvent>(OnSpeakAttempt);
|
SubscribeLocalEvent<SleepingComponent, SpeakAttemptEvent>(OnSpeakAttempt);
|
||||||
|
SubscribeLocalEvent<SleepingComponent, CanSeeAttemptEvent>(OnSeeAttempt);
|
||||||
SubscribeLocalEvent<SleepingComponent, EntityUnpausedEvent>(OnSleepUnpaused);
|
SubscribeLocalEvent<SleepingComponent, EntityUnpausedEvent>(OnSleepUnpaused);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -24,24 +25,30 @@ namespace Content.Server.Bed.Sleep
|
|||||||
Dirty(component);
|
Dirty(component);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnInit(EntityUid uid, SleepingComponent component, ComponentInit args)
|
private void OnStartup(EntityUid uid, SleepingComponent component, ComponentStartup args)
|
||||||
{
|
{
|
||||||
var ev = new SleepStateChangedEvent(true);
|
var ev = new SleepStateChangedEvent(true);
|
||||||
RaiseLocalEvent(uid, ev);
|
RaiseLocalEvent(uid, ev);
|
||||||
_blindingSystem.AdjustBlindSources(uid, 1);
|
_blindableSystem.UpdateIsBlind(uid);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnShutdown(EntityUid uid, SleepingComponent component, ComponentShutdown args)
|
private void OnShutdown(EntityUid uid, SleepingComponent component, ComponentShutdown args)
|
||||||
{
|
{
|
||||||
var ev = new SleepStateChangedEvent(false);
|
var ev = new SleepStateChangedEvent(false);
|
||||||
RaiseLocalEvent(uid, ev);
|
RaiseLocalEvent(uid, ev);
|
||||||
_blindingSystem.AdjustBlindSources(uid, -1);
|
_blindableSystem.UpdateIsBlind(uid);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnSpeakAttempt(EntityUid uid, SleepingComponent component, SpeakAttemptEvent args)
|
private void OnSpeakAttempt(EntityUid uid, SleepingComponent component, SpeakAttemptEvent args)
|
||||||
{
|
{
|
||||||
args.Cancel();
|
args.Cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void OnSeeAttempt(EntityUid uid, SleepingComponent component, CanSeeAttemptEvent args)
|
||||||
|
{
|
||||||
|
if (component.LifeStage <= ComponentLifeStage.Running)
|
||||||
|
args.Cancel();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ using System.Linq;
|
|||||||
using Content.Shared.DragDrop;
|
using Content.Shared.DragDrop;
|
||||||
using Content.Shared.Interaction;
|
using Content.Shared.Interaction;
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
using Content.Shared.Mobs.Components;
|
using Content.Shared.Mobs.Components;
|
||||||
using Content.Shared.Mobs.Systems;
|
using Content.Shared.Mobs.Systems;
|
||||||
using JetBrains.Annotations;
|
using JetBrains.Annotations;
|
||||||
@@ -122,14 +123,12 @@ namespace Content.Shared.Examine
|
|||||||
{
|
{
|
||||||
if (MobStateSystem.IsDead(examiner, mobState))
|
if (MobStateSystem.IsDead(examiner, mobState))
|
||||||
return DeadExamineRange;
|
return DeadExamineRange;
|
||||||
else if (MobStateSystem.IsCritical(examiner, mobState) || (TryComp<BlindableComponent>(examiner, out var blind) && blind.Sources > 0))
|
|
||||||
|
if (MobStateSystem.IsCritical(examiner, mobState) || TryComp<BlindableComponent>(examiner, out var blind) && blind.IsBlind)
|
||||||
return CritExamineRange;
|
return CritExamineRange;
|
||||||
|
|
||||||
else if (TryComp<BlurryVisionComponent>(examiner, out var blurry) && blurry.Magnitude != 0)
|
if (TryComp<BlurryVisionComponent>(examiner, out var blurry))
|
||||||
{
|
return Math.Clamp(ExamineRange - blurry.Magnitude, 2, ExamineRange);
|
||||||
float range = ExamineRange - (2 * (8 - blurry.Magnitude));
|
|
||||||
return Math.Clamp(range, 2, 16);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return ExamineRange;
|
return ExamineRange;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,56 +0,0 @@
|
|||||||
using Robust.Shared.GameStates;
|
|
||||||
using Robust.Shared.Serialization;
|
|
||||||
|
|
||||||
namespace Content.Shared.Eye.Blinding
|
|
||||||
{
|
|
||||||
[RegisterComponent]
|
|
||||||
[NetworkedComponent]
|
|
||||||
public sealed class BlindableComponent : Component
|
|
||||||
{
|
|
||||||
/// <description>
|
|
||||||
/// How many sources of blindness are affecting us?
|
|
||||||
/// </description>
|
|
||||||
[DataField("sources")]
|
|
||||||
public int Sources = 0;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// How many seconds will be subtracted from each attempt to add blindness to us?
|
|
||||||
/// </summary>
|
|
||||||
[DataField("blindResistance")]
|
|
||||||
public float BlindResistance = 0;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Replace with actual eye damage after bobby I guess
|
|
||||||
/// </summary>
|
|
||||||
[ViewVariables]
|
|
||||||
public int EyeDamage = 0;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Whether eye damage has accumulated enough to blind them.
|
|
||||||
/// </summary>
|
|
||||||
[ViewVariables]
|
|
||||||
public bool EyeTooDamaged = false;
|
|
||||||
|
|
||||||
/// <description>
|
|
||||||
/// Used to ensure that this doesn't break with sandbox or admin tools.
|
|
||||||
/// This is not "enabled/disabled".
|
|
||||||
/// </description>
|
|
||||||
public bool LightSetup = false;
|
|
||||||
|
|
||||||
/// <description>
|
|
||||||
/// Gives an extra frame of blindness to reenable light manager during
|
|
||||||
/// </description>
|
|
||||||
public bool GraceFrame = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Serializable, NetSerializable]
|
|
||||||
public sealed class BlindableComponentState : ComponentState
|
|
||||||
{
|
|
||||||
public readonly int Sources;
|
|
||||||
|
|
||||||
public BlindableComponentState(int sources)
|
|
||||||
{
|
|
||||||
Sources = sources;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
using Robust.Shared.GameStates;
|
|
||||||
|
|
||||||
namespace Content.Shared.Eye.Blinding
|
|
||||||
{
|
|
||||||
[RegisterComponent]
|
|
||||||
[NetworkedComponent]
|
|
||||||
public sealed class BlindfoldComponent : Component
|
|
||||||
{
|
|
||||||
[ViewVariables]
|
|
||||||
public bool IsActive = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
using Robust.Shared.GameStates;
|
|
||||||
|
|
||||||
namespace Content.Shared.Eye.Blinding
|
|
||||||
{
|
|
||||||
[RegisterComponent]
|
|
||||||
[NetworkedComponent]
|
|
||||||
public sealed class BlurryVisionComponent : Component
|
|
||||||
{
|
|
||||||
[DataField("mangitude")]
|
|
||||||
public float Magnitude = 1f;
|
|
||||||
|
|
||||||
public bool Active => Magnitude < 10f;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
41
Content.Shared/Eye/Blinding/Components/BlindableComponent.cs
Normal file
41
Content.Shared/Eye/Blinding/Components/BlindableComponent.cs
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
|
using Robust.Shared.GameStates;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
[RegisterComponent]
|
||||||
|
[NetworkedComponent, AutoGenerateComponentState]
|
||||||
|
[Access(typeof(BlindableSystem))]
|
||||||
|
public sealed partial class BlindableComponent : Component
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// How many seconds will be subtracted from each attempt to add blindness to us?
|
||||||
|
/// </summary>
|
||||||
|
[ViewVariables(VVAccess.ReadWrite), DataField("isBlind"), AutoNetworkedField]
|
||||||
|
public bool IsBlind;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Eye damage due to things like staring directly at welders. Causes blurry vision or outright
|
||||||
|
/// blindness if greater than or equal to <see cref="MaxDamage"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Should eventually be replaced with a proper eye health system when we have bobby.
|
||||||
|
/// </remarks>
|
||||||
|
[ViewVariables(VVAccess.ReadWrite), DataField("EyeDamage"), AutoNetworkedField]
|
||||||
|
public int EyeDamage = 0;
|
||||||
|
|
||||||
|
public const int MaxDamage = 8;
|
||||||
|
|
||||||
|
/// <description>
|
||||||
|
/// Used to ensure that this doesn't break with sandbox or admin tools.
|
||||||
|
/// This is not "enabled/disabled".
|
||||||
|
/// </description>
|
||||||
|
[Access(Other = AccessPermissions.ReadWriteExecute)]
|
||||||
|
public bool LightSetup = false;
|
||||||
|
|
||||||
|
/// <description>
|
||||||
|
/// Gives an extra frame of blindness to reenable light manager during
|
||||||
|
/// </description>
|
||||||
|
[Access(Other = AccessPermissions.ReadWriteExecute)]
|
||||||
|
public bool GraceFrame = false;
|
||||||
|
}
|
||||||
12
Content.Shared/Eye/Blinding/Components/BlindfoldComponent.cs
Normal file
12
Content.Shared/Eye/Blinding/Components/BlindfoldComponent.cs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
using Robust.Shared.GameStates;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Blinds a person when an item with this component is equipped to the eye, head, or mask slot.
|
||||||
|
/// </summary>
|
||||||
|
[RegisterComponent]
|
||||||
|
[NetworkedComponent]
|
||||||
|
public sealed class BlindfoldComponent : Component
|
||||||
|
{
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
|
using Robust.Shared.GameStates;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This component adds a white overlay to the viewport. It does not actually cause blurring.
|
||||||
|
/// </summary>
|
||||||
|
[RegisterComponent]
|
||||||
|
[NetworkedComponent, AutoGenerateComponentState]
|
||||||
|
[Access(typeof(BlurryVisionSystem))]
|
||||||
|
public sealed partial class BlurryVisionComponent : Component
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Amount of "blurring". Also modifies examine ranges.
|
||||||
|
/// </summary>
|
||||||
|
[ViewVariables(VVAccess.ReadWrite), DataField("magnitude"), AutoNetworkedField]
|
||||||
|
public float Magnitude;
|
||||||
|
|
||||||
|
public const float MaxMagnitude = 10;
|
||||||
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// For welding masks, sunglasses, etc.
|
||||||
|
/// </summary>
|
||||||
|
[RegisterComponent]
|
||||||
|
public sealed class EyeProtectionComponent : Component
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// How many seconds to subtract from the status effect. If it's greater than the source
|
||||||
|
/// of blindness, do not blind.
|
||||||
|
/// </summary>
|
||||||
|
[DataField("protectionTime")]
|
||||||
|
public readonly TimeSpan ProtectionTime = TimeSpan.FromSeconds(10);
|
||||||
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
using Robust.Shared.GameStates;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// For tools like welders that will damage your eyes when you use them.
|
||||||
|
/// </summary>
|
||||||
|
[RegisterComponent, NetworkedComponent, AutoGenerateComponentState]
|
||||||
|
public sealed partial class RequiresEyeProtectionComponent : Component
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// How long to apply temporary blindness to the user.
|
||||||
|
/// </summary>
|
||||||
|
[ViewVariables(VVAccess.ReadWrite), DataField("statusEffectTime"), AutoNetworkedField]
|
||||||
|
public TimeSpan StatusEffectTime = TimeSpan.FromSeconds(10);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// You probably want to turn this on in yaml if it's something always on and not a welder.
|
||||||
|
/// </summary>
|
||||||
|
[ViewVariables(VVAccess.ReadWrite), DataField("toggled"), AutoNetworkedField]
|
||||||
|
public bool Toggled;
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
using Robust.Shared.GameStates;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Component used for the blind status effect.
|
||||||
|
/// </summary>
|
||||||
|
[NetworkedComponent, RegisterComponent]
|
||||||
|
public sealed class TemporaryBlindnessComponent : Component
|
||||||
|
{
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
using Robust.Shared.GameStates;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This component allows equipment to offset blurry vision.
|
||||||
|
/// </summary>
|
||||||
|
[RegisterComponent]
|
||||||
|
[NetworkedComponent, AutoGenerateComponentState]
|
||||||
|
public sealed partial class VisionCorrectionComponent : Component
|
||||||
|
{
|
||||||
|
[ViewVariables(VVAccess.ReadWrite), DataField("visionBonus"), AutoNetworkedField]
|
||||||
|
public float VisionBonus = 3f;
|
||||||
|
}
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
namespace Content.Shared.Eye.Blinding.EyeProtection
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// For welding masks, sunglasses, etc.
|
|
||||||
/// </summary>
|
|
||||||
[RegisterComponent]
|
|
||||||
public sealed class EyeProtectionComponent : Component
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// How many seconds to subtract from the status effect. If it's greater than the source
|
|
||||||
/// of blindness, do not blind.
|
|
||||||
/// </summary>
|
|
||||||
[DataField("protectionTime")]
|
|
||||||
public TimeSpan ProtectionTime = TimeSpan.FromSeconds(10);
|
|
||||||
|
|
||||||
public bool IsActive = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
namespace Content.Shared.Eye.Blinding.EyeProtection
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// For tools like welders that will damage your eyes when you use them.
|
|
||||||
/// </summary>
|
|
||||||
[RegisterComponent]
|
|
||||||
public sealed class RequiresEyeProtectionComponent : Component
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// How long to apply temporary blindness to the user.
|
|
||||||
/// </summary>
|
|
||||||
[DataField("statusEffectTime")]
|
|
||||||
public TimeSpan StatusEffectTime = TimeSpan.FromSeconds(10);
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// You probably want to turn this on in yaml if it's something always on and not a welder.
|
|
||||||
/// </summary>
|
|
||||||
[DataField("toggled")]
|
|
||||||
public bool Toggled = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,198 +0,0 @@
|
|||||||
using Content.Shared.Clothing.Components;
|
|
||||||
using Content.Shared.Inventory.Events;
|
|
||||||
using Content.Shared.Inventory;
|
|
||||||
using Content.Shared.Rejuvenate;
|
|
||||||
using Robust.Shared.GameStates;
|
|
||||||
using Robust.Shared.Serialization;
|
|
||||||
using JetBrains.Annotations;
|
|
||||||
|
|
||||||
namespace Content.Shared.Eye.Blinding
|
|
||||||
{
|
|
||||||
public sealed class SharedBlindingSystem : EntitySystem
|
|
||||||
{
|
|
||||||
public const string BlindingStatusEffect = "TemporaryBlindness";
|
|
||||||
public override void Initialize()
|
|
||||||
{
|
|
||||||
base.Initialize();
|
|
||||||
|
|
||||||
SubscribeLocalEvent<BlindableComponent, ComponentGetState>(OnGetBlindableState);
|
|
||||||
SubscribeLocalEvent<BlindableComponent, ComponentHandleState>(OnHandleBlindableState);
|
|
||||||
|
|
||||||
SubscribeLocalEvent<BlindfoldComponent, GotEquippedEvent>(OnEquipped);
|
|
||||||
SubscribeLocalEvent<BlindfoldComponent, GotUnequippedEvent>(OnUnequipped);
|
|
||||||
|
|
||||||
SubscribeLocalEvent<VisionCorrectionComponent, GotEquippedEvent>(OnGlassesEquipped);
|
|
||||||
SubscribeLocalEvent<VisionCorrectionComponent, GotUnequippedEvent>(OnGlassesUnequipped);
|
|
||||||
|
|
||||||
SubscribeLocalEvent<BlurryVisionComponent, ComponentGetState>(OnGetState);
|
|
||||||
|
|
||||||
SubscribeLocalEvent<TemporaryBlindnessComponent, ComponentInit>(OnInit);
|
|
||||||
SubscribeLocalEvent<TemporaryBlindnessComponent, ComponentShutdown>(OnShutdown);
|
|
||||||
|
|
||||||
SubscribeLocalEvent<BlindableComponent, RejuvenateEvent>(OnRejuvenate);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnGetBlindableState(EntityUid uid, BlindableComponent component, ref ComponentGetState args)
|
|
||||||
{
|
|
||||||
args.State = new BlindableComponentState(component.Sources);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnHandleBlindableState(EntityUid uid, BlindableComponent component, ref ComponentHandleState args)
|
|
||||||
{
|
|
||||||
if (args.Current is not BlindableComponentState cast)
|
|
||||||
return;
|
|
||||||
|
|
||||||
component.Sources = cast.Sources;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnEquipped(EntityUid uid, BlindfoldComponent component, GotEquippedEvent args)
|
|
||||||
{
|
|
||||||
if (!TryComp<ClothingComponent>(uid, out var clothing) || clothing.Slots == SlotFlags.PREVENTEQUIP) // we live in a society
|
|
||||||
return;
|
|
||||||
// Is the clothing in its actual slot?
|
|
||||||
if (!clothing.Slots.HasFlag(args.SlotFlags))
|
|
||||||
return;
|
|
||||||
|
|
||||||
component.IsActive = true;
|
|
||||||
if (!TryComp<BlindableComponent>(args.Equipee, out var blindComp))
|
|
||||||
return;
|
|
||||||
AdjustBlindSources(args.Equipee, 1, blindComp);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnUnequipped(EntityUid uid, BlindfoldComponent component, GotUnequippedEvent args)
|
|
||||||
{
|
|
||||||
if (!component.IsActive)
|
|
||||||
return;
|
|
||||||
component.IsActive = false;
|
|
||||||
if (!TryComp<BlindableComponent>(args.Equipee, out var blindComp))
|
|
||||||
return;
|
|
||||||
AdjustBlindSources(args.Equipee, -1, blindComp);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnGlassesEquipped(EntityUid uid, VisionCorrectionComponent component, GotEquippedEvent args)
|
|
||||||
{
|
|
||||||
if (!TryComp<ClothingComponent>(uid, out var clothing) || clothing.Slots == SlotFlags.PREVENTEQUIP) // we live in a society
|
|
||||||
return;
|
|
||||||
// Is the clothing in its actual slot?
|
|
||||||
if (!clothing.Slots.HasFlag(args.SlotFlags))
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (!TryComp<BlurryVisionComponent>(args.Equipee, out var blur))
|
|
||||||
return;
|
|
||||||
|
|
||||||
component.IsActive = true;
|
|
||||||
blur.Magnitude += component.VisionBonus;
|
|
||||||
Dirty(blur);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnGlassesUnequipped(EntityUid uid, VisionCorrectionComponent component, GotUnequippedEvent args)
|
|
||||||
{
|
|
||||||
if (!component.IsActive || !TryComp<BlurryVisionComponent>(args.Equipee, out var blur))
|
|
||||||
return;
|
|
||||||
component.IsActive = false;
|
|
||||||
blur.Magnitude -= component.VisionBonus;
|
|
||||||
Dirty(blur);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnGetState(EntityUid uid, BlurryVisionComponent component, ref ComponentGetState args)
|
|
||||||
{
|
|
||||||
args.State = new BlurryVisionComponentState(component.Magnitude);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnInit(EntityUid uid, TemporaryBlindnessComponent component, ComponentInit args)
|
|
||||||
{
|
|
||||||
AdjustBlindSources(uid, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnShutdown(EntityUid uid, TemporaryBlindnessComponent component, ComponentShutdown args)
|
|
||||||
{
|
|
||||||
AdjustBlindSources(uid, -1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void OnRejuvenate(EntityUid uid, BlindableComponent component, RejuvenateEvent args)
|
|
||||||
{
|
|
||||||
AdjustEyeDamage(uid, -component.EyeDamage, component);
|
|
||||||
}
|
|
||||||
|
|
||||||
[PublicAPI]
|
|
||||||
public void AdjustBlindSources(EntityUid uid, int amount, BlindableComponent? blindable = null)
|
|
||||||
{
|
|
||||||
if (!Resolve(uid, ref blindable, false))
|
|
||||||
return;
|
|
||||||
|
|
||||||
var oldSources = blindable.Sources;
|
|
||||||
|
|
||||||
blindable.Sources += amount;
|
|
||||||
blindable.Sources = Math.Max(blindable.Sources, 0);
|
|
||||||
|
|
||||||
if (oldSources == 0 && blindable.Sources > 0)
|
|
||||||
{
|
|
||||||
var ev = new BlindnessChangedEvent(true);
|
|
||||||
RaiseLocalEvent(uid, ev, false);
|
|
||||||
}
|
|
||||||
else if (blindable.Sources == 0 && oldSources > 0)
|
|
||||||
{
|
|
||||||
var ev = new BlindnessChangedEvent(false);
|
|
||||||
RaiseLocalEvent(uid, ev, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
Dirty(blindable);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void AdjustEyeDamage(EntityUid uid, int amount, BlindableComponent? blindable = null)
|
|
||||||
{
|
|
||||||
if (!Resolve(uid, ref blindable, false))
|
|
||||||
return;
|
|
||||||
|
|
||||||
blindable.EyeDamage += amount;
|
|
||||||
|
|
||||||
if (blindable.EyeDamage > 0)
|
|
||||||
{
|
|
||||||
var blurry = EnsureComp<BlurryVisionComponent>(uid);
|
|
||||||
blurry.Magnitude = (9 - blindable.EyeDamage);
|
|
||||||
Dirty(blurry);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
RemComp<BlurryVisionComponent>(uid);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!blindable.EyeTooDamaged && blindable.EyeDamage >= 8)
|
|
||||||
{
|
|
||||||
blindable.EyeTooDamaged = true;
|
|
||||||
AdjustBlindSources(uid, 1, blindable);
|
|
||||||
}
|
|
||||||
if (blindable.EyeTooDamaged && blindable.EyeDamage < 8)
|
|
||||||
{
|
|
||||||
blindable.EyeTooDamaged = false;
|
|
||||||
AdjustBlindSources(uid, -1, blindable);
|
|
||||||
}
|
|
||||||
|
|
||||||
blindable.EyeDamage = Math.Clamp(blindable.EyeDamage, 0, 8);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// I have no idea why blurry vision needs this but blindness doesn't
|
|
||||||
[Serializable, NetSerializable]
|
|
||||||
public sealed class BlurryVisionComponentState : ComponentState
|
|
||||||
{
|
|
||||||
public float Magnitude;
|
|
||||||
public BlurryVisionComponentState(float magnitude)
|
|
||||||
{
|
|
||||||
Magnitude = magnitude;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// You became blind or lost blindess, not just changed # of sources.
|
|
||||||
/// </summary>
|
|
||||||
public sealed class BlindnessChangedEvent : EntityEventArgs
|
|
||||||
{
|
|
||||||
public bool Blind;
|
|
||||||
|
|
||||||
public BlindnessChangedEvent(bool blind)
|
|
||||||
{
|
|
||||||
Blind = blind;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
93
Content.Shared/Eye/Blinding/Systems/BlindableSystem.cs
Normal file
93
Content.Shared/Eye/Blinding/Systems/BlindableSystem.cs
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
using Content.Shared.Inventory;
|
||||||
|
using Content.Shared.Rejuvenate;
|
||||||
|
using JetBrains.Annotations;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Systems;
|
||||||
|
|
||||||
|
public sealed class BlindableSystem : EntitySystem
|
||||||
|
{
|
||||||
|
public override void Initialize()
|
||||||
|
{
|
||||||
|
base.Initialize();
|
||||||
|
SubscribeLocalEvent<BlindableComponent, RejuvenateEvent>(OnRejuvenate);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnRejuvenate(EntityUid uid, BlindableComponent component, RejuvenateEvent args)
|
||||||
|
{
|
||||||
|
AdjustEyeDamage(uid, -component.EyeDamage, component);
|
||||||
|
}
|
||||||
|
|
||||||
|
[PublicAPI]
|
||||||
|
public void UpdateIsBlind(EntityUid uid, BlindableComponent? blindable = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref blindable, false))
|
||||||
|
return;
|
||||||
|
|
||||||
|
var old = blindable.IsBlind;
|
||||||
|
|
||||||
|
// Don't bother raising an event if the eye is too damaged.
|
||||||
|
if (blindable.EyeDamage >= BlindableComponent.MaxDamage)
|
||||||
|
{
|
||||||
|
blindable.IsBlind = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var ev = new CanSeeAttemptEvent();
|
||||||
|
RaiseLocalEvent(uid, ev);
|
||||||
|
blindable.IsBlind = ev.Blind;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (old == blindable.IsBlind)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var changeEv = new BlindnessChangedEvent(blindable.IsBlind);
|
||||||
|
RaiseLocalEvent(uid, ref changeEv);
|
||||||
|
Dirty(blindable);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AdjustEyeDamage(EntityUid uid, int amount, BlindableComponent? blindable = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref blindable, false) || amount == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
blindable.EyeDamage += amount;
|
||||||
|
blindable.EyeDamage = Math.Clamp(blindable.EyeDamage, 0, BlindableComponent.MaxDamage);
|
||||||
|
Dirty(blindable);
|
||||||
|
UpdateIsBlind(uid, blindable);
|
||||||
|
|
||||||
|
var ev = new EyeDamageChangedEvent(blindable.EyeDamage);
|
||||||
|
RaiseLocalEvent(uid, ref ev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This event is raised when an entity's blindness changes
|
||||||
|
/// </summary>
|
||||||
|
[ByRefEvent]
|
||||||
|
public record struct BlindnessChangedEvent(bool Blind);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This event is raised when an entity's eye damage changes
|
||||||
|
/// </summary>
|
||||||
|
[ByRefEvent]
|
||||||
|
public record struct EyeDamageChangedEvent(int Damage);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Raised directed at an entity to see whether the entity is currently blind or not.
|
||||||
|
/// </summary>
|
||||||
|
public sealed class CanSeeAttemptEvent : CancellableEntityEventArgs, IInventoryRelayEvent
|
||||||
|
{
|
||||||
|
public bool Blind => Cancelled;
|
||||||
|
public SlotFlags TargetSlots => SlotFlags.EYES | SlotFlags.MASK | SlotFlags.HEAD;
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class GetEyeProtectionEvent : EntityEventArgs, IInventoryRelayEvent
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Time to subtract from any temporary blindness sources.
|
||||||
|
/// </summary>
|
||||||
|
public TimeSpan Protection;
|
||||||
|
|
||||||
|
public SlotFlags TargetSlots => SlotFlags.EYES | SlotFlags.MASK | SlotFlags.HEAD;
|
||||||
|
}
|
||||||
34
Content.Shared/Eye/Blinding/Systems/BlindfoldSystem.cs
Normal file
34
Content.Shared/Eye/Blinding/Systems/BlindfoldSystem.cs
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
using Content.Shared.Inventory.Events;
|
||||||
|
using Content.Shared.Inventory;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Systems;
|
||||||
|
|
||||||
|
public sealed class BlindfoldSystem : EntitySystem
|
||||||
|
{
|
||||||
|
[Dependency] private readonly BlindableSystem _blindableSystem = default!;
|
||||||
|
|
||||||
|
public override void Initialize()
|
||||||
|
{
|
||||||
|
base.Initialize();
|
||||||
|
|
||||||
|
SubscribeLocalEvent<BlindfoldComponent, GotEquippedEvent>(OnEquipped);
|
||||||
|
SubscribeLocalEvent<BlindfoldComponent, GotUnequippedEvent>(OnUnequipped);
|
||||||
|
SubscribeLocalEvent<BlindfoldComponent, InventoryRelayedEvent<CanSeeAttemptEvent>>(OnBlindfoldTrySee);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnBlindfoldTrySee(EntityUid uid, BlindfoldComponent component, InventoryRelayedEvent<CanSeeAttemptEvent> args)
|
||||||
|
{
|
||||||
|
args.Args.Cancel();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnEquipped(EntityUid uid, BlindfoldComponent component, GotEquippedEvent args)
|
||||||
|
{
|
||||||
|
_blindableSystem.UpdateIsBlind(args.Equipee);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnUnequipped(EntityUid uid, BlindfoldComponent component, GotUnequippedEvent args)
|
||||||
|
{
|
||||||
|
_blindableSystem.UpdateIsBlind(args.Equipee);
|
||||||
|
}
|
||||||
|
}
|
||||||
72
Content.Shared/Eye/Blinding/Systems/BlurryVisionSystem.cs
Normal file
72
Content.Shared/Eye/Blinding/Systems/BlurryVisionSystem.cs
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
using Content.Shared.Inventory.Events;
|
||||||
|
using Content.Shared.Inventory;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Systems;
|
||||||
|
|
||||||
|
public sealed class BlurryVisionSystem : EntitySystem
|
||||||
|
{
|
||||||
|
public override void Initialize()
|
||||||
|
{
|
||||||
|
base.Initialize();
|
||||||
|
|
||||||
|
SubscribeLocalEvent<BlindableComponent, EyeDamageChangedEvent>(OnDamageChanged);
|
||||||
|
SubscribeLocalEvent<VisionCorrectionComponent, GotEquippedEvent>(OnGlassesEquipped);
|
||||||
|
SubscribeLocalEvent<VisionCorrectionComponent, GotUnequippedEvent>(OnGlassesUnequipped);
|
||||||
|
SubscribeLocalEvent<VisionCorrectionComponent, InventoryRelayedEvent<GetBlurEvent>>(OnGetBlur);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnGetBlur(EntityUid uid, VisionCorrectionComponent component, InventoryRelayedEvent<GetBlurEvent> args)
|
||||||
|
{
|
||||||
|
args.Args.Blur += component.VisionBonus;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnDamageChanged(EntityUid uid, BlindableComponent component, ref EyeDamageChangedEvent args)
|
||||||
|
{
|
||||||
|
UpdateBlurMagnitude(uid, component);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UpdateBlurMagnitude(EntityUid uid, BlindableComponent? component = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref component, false))
|
||||||
|
return;
|
||||||
|
|
||||||
|
var ev = new GetBlurEvent(component.EyeDamage);
|
||||||
|
RaiseLocalEvent(uid, ev);
|
||||||
|
|
||||||
|
var blur = Math.Clamp(0, ev.Blur, BlurryVisionComponent.MaxMagnitude);
|
||||||
|
if (blur <= 0)
|
||||||
|
{
|
||||||
|
RemCompDeferred<BlurryVisionComponent>(uid);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var blurry = EnsureComp<BlurryVisionComponent>(uid);
|
||||||
|
blurry.Magnitude = blur;
|
||||||
|
Dirty(blurry);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnGlassesEquipped(EntityUid uid, VisionCorrectionComponent component, GotEquippedEvent args)
|
||||||
|
{
|
||||||
|
UpdateBlurMagnitude(uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnGlassesUnequipped(EntityUid uid, VisionCorrectionComponent component, GotUnequippedEvent args)
|
||||||
|
{
|
||||||
|
UpdateBlurMagnitude(uid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class GetBlurEvent : EntityEventArgs, IInventoryRelayEvent
|
||||||
|
{
|
||||||
|
public readonly float BaseBlur;
|
||||||
|
public float Blur;
|
||||||
|
|
||||||
|
public GetBlurEvent(float blur)
|
||||||
|
{
|
||||||
|
Blur = blur;
|
||||||
|
BaseBlur = blur;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SlotFlags TargetSlots => SlotFlags.HEAD | SlotFlags.MASK | SlotFlags.EYES;
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
using Content.Shared.Eye.Blinding.Components;
|
||||||
|
|
||||||
|
namespace Content.Shared.Eye.Blinding.Systems;
|
||||||
|
|
||||||
|
public sealed class TemporaryBlindnessSystem : EntitySystem
|
||||||
|
{
|
||||||
|
public const string BlindingStatusEffect = "TemporaryBlindness";
|
||||||
|
|
||||||
|
[Dependency] private readonly BlindableSystem _blindableSystem = default!;
|
||||||
|
|
||||||
|
public override void Initialize()
|
||||||
|
{
|
||||||
|
base.Initialize();
|
||||||
|
|
||||||
|
SubscribeLocalEvent<TemporaryBlindnessComponent, ComponentStartup>(OnStartup);
|
||||||
|
SubscribeLocalEvent<TemporaryBlindnessComponent, ComponentShutdown>(OnShutdown);
|
||||||
|
SubscribeLocalEvent<TemporaryBlindnessComponent, CanSeeAttemptEvent>(OnBlindTrySee);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnStartup(EntityUid uid, TemporaryBlindnessComponent component, ComponentStartup args)
|
||||||
|
{
|
||||||
|
_blindableSystem.UpdateIsBlind(uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnShutdown(EntityUid uid, TemporaryBlindnessComponent component, ComponentShutdown args)
|
||||||
|
{
|
||||||
|
_blindableSystem.UpdateIsBlind(uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnBlindTrySee(EntityUid uid, TemporaryBlindnessComponent component, CanSeeAttemptEvent args)
|
||||||
|
{
|
||||||
|
if (component.LifeStage <= ComponentLifeStage.Running)
|
||||||
|
args.Cancel();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
using Robust.Shared.GameStates;
|
|
||||||
|
|
||||||
namespace Content.Shared.Eye.Blinding
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Blind status effect.
|
|
||||||
/// </summary>
|
|
||||||
[NetworkedComponent, RegisterComponent]
|
|
||||||
public sealed class TemporaryBlindnessComponent : Component
|
|
||||||
{}
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
using Robust.Shared.GameStates;
|
|
||||||
|
|
||||||
namespace Content.Shared.Eye.Blinding
|
|
||||||
{
|
|
||||||
[RegisterComponent]
|
|
||||||
[NetworkedComponent]
|
|
||||||
public sealed class VisionCorrectionComponent : Component
|
|
||||||
{
|
|
||||||
[ViewVariables]
|
|
||||||
public bool IsActive = false;
|
|
||||||
|
|
||||||
[DataField("visionBonus")]
|
|
||||||
public float VisionBonus = 3f;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
using Content.Shared.Damage;
|
using Content.Shared.Damage;
|
||||||
using Content.Shared.Electrocution;
|
using Content.Shared.Electrocution;
|
||||||
using Content.Shared.Explosion;
|
using Content.Shared.Explosion;
|
||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
using Content.Shared.IdentityManagement.Components;
|
using Content.Shared.IdentityManagement.Components;
|
||||||
using Content.Shared.Movement.Systems;
|
using Content.Shared.Movement.Systems;
|
||||||
using Content.Shared.Radio;
|
using Content.Shared.Radio;
|
||||||
@@ -26,6 +27,11 @@ public partial class InventorySystem
|
|||||||
SubscribeLocalEvent<InventoryComponent, ModifyChangedTemperatureEvent>(RelayInventoryEvent);
|
SubscribeLocalEvent<InventoryComponent, ModifyChangedTemperatureEvent>(RelayInventoryEvent);
|
||||||
SubscribeLocalEvent<InventoryComponent, GetDefaultRadioChannelEvent>(RelayInventoryEvent);
|
SubscribeLocalEvent<InventoryComponent, GetDefaultRadioChannelEvent>(RelayInventoryEvent);
|
||||||
|
|
||||||
|
// Eye/vision events
|
||||||
|
SubscribeLocalEvent<InventoryComponent, CanSeeAttemptEvent>(RelayInventoryEvent);
|
||||||
|
SubscribeLocalEvent<InventoryComponent, GetEyeProtectionEvent>(RelayInventoryEvent);
|
||||||
|
SubscribeLocalEvent<InventoryComponent, GetBlurEvent>(RelayInventoryEvent);
|
||||||
|
|
||||||
SubscribeLocalEvent<InventoryComponent, GetVerbsEvent<EquipmentVerb>>(OnGetStrippingVerbs);
|
SubscribeLocalEvent<InventoryComponent, GetVerbsEvent<EquipmentVerb>>(OnGetStrippingVerbs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using Content.Shared.Examine;
|
using Content.Shared.Examine;
|
||||||
using Content.Shared.Eye.Blinding;
|
using Content.Shared.Eye.Blinding;
|
||||||
|
using Content.Shared.Eye.Blinding.Systems;
|
||||||
using Content.Shared.IdentityManagement;
|
using Content.Shared.IdentityManagement;
|
||||||
using Robust.Shared.Network;
|
using Robust.Shared.Network;
|
||||||
|
|
||||||
@@ -11,13 +12,14 @@ namespace Content.Shared.Traits.Assorted;
|
|||||||
public sealed class PermanentBlindnessSystem : EntitySystem
|
public sealed class PermanentBlindnessSystem : EntitySystem
|
||||||
{
|
{
|
||||||
[Dependency] private readonly INetManager _net = default!;
|
[Dependency] private readonly INetManager _net = default!;
|
||||||
[Dependency] private readonly SharedBlindingSystem _blinding = default!;
|
[Dependency] private readonly BlindableSystem _blinding = default!;
|
||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
SubscribeLocalEvent<PermanentBlindnessComponent, ComponentStartup>(OnStartup);
|
SubscribeLocalEvent<PermanentBlindnessComponent, ComponentStartup>(OnStartup);
|
||||||
SubscribeLocalEvent<PermanentBlindnessComponent, ComponentShutdown>(OnShutdown);
|
SubscribeLocalEvent<PermanentBlindnessComponent, ComponentShutdown>(OnShutdown);
|
||||||
|
SubscribeLocalEvent<PermanentBlindnessComponent, CanSeeAttemptEvent>(OnTrySee);
|
||||||
SubscribeLocalEvent<PermanentBlindnessComponent, ExaminedEvent>(OnExamined);
|
SubscribeLocalEvent<PermanentBlindnessComponent, ExaminedEvent>(OnExamined);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -31,11 +33,17 @@ public sealed class PermanentBlindnessSystem : EntitySystem
|
|||||||
|
|
||||||
private void OnShutdown(EntityUid uid, PermanentBlindnessComponent component, ComponentShutdown args)
|
private void OnShutdown(EntityUid uid, PermanentBlindnessComponent component, ComponentShutdown args)
|
||||||
{
|
{
|
||||||
_blinding.AdjustBlindSources(uid, -1);
|
_blinding.UpdateIsBlind(uid);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnStartup(EntityUid uid, PermanentBlindnessComponent component, ComponentStartup args)
|
private void OnStartup(EntityUid uid, PermanentBlindnessComponent component, ComponentStartup args)
|
||||||
{
|
{
|
||||||
_blinding.AdjustBlindSources(uid, 1);
|
_blinding.UpdateIsBlind(uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnTrySee(EntityUid uid, PermanentBlindnessComponent component, CanSeeAttemptEvent args)
|
||||||
|
{
|
||||||
|
if (component.LifeStage <= ComponentLifeStage.Running)
|
||||||
|
args.Cancel();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -66,11 +66,6 @@
|
|||||||
kind: source
|
kind: source
|
||||||
path: "/Textures/Shaders/blurryy.swsl"
|
path: "/Textures/Shaders/blurryy.swsl"
|
||||||
|
|
||||||
- type: shader
|
|
||||||
id: Dim
|
|
||||||
kind: source
|
|
||||||
path: "/Textures/Shaders/dim.swsl"
|
|
||||||
|
|
||||||
# cloaking distortion effect
|
# cloaking distortion effect
|
||||||
- type: shader
|
- type: shader
|
||||||
id: Stealth
|
id: Stealth
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
uniform highp float DAMAGE_AMOUNT;
|
|
||||||
|
|
||||||
void fragment() {
|
|
||||||
|
|
||||||
// Higher exponent -> stronger blinding effect
|
|
||||||
|
|
||||||
// Gradually mixes between the texture mix and a full-white texture, causing the "blinding" effect
|
|
||||||
highp vec4 mixed = mix(vec4(1.0, 1.0, 1.0, 1.0), vec4(1.0, 1.0, 1.0, 1.0), DAMAGE_AMOUNT);
|
|
||||||
|
|
||||||
COLOR = vec4(mixed.rgb, DAMAGE_AMOUNT);
|
|
||||||
}
|
|
||||||
|
|
||||||
Reference in New Issue
Block a user