Powered light ECS (#5028)
* Brrrr ECS * Create lit on powered system * Light bulb ECS * Finishing porting to ECS * Minor touches * Removed events * Removed old comments * Fixed test * To popup system
This commit is contained in:
42
Content.Client/Light/Visualizers/LightBulbVisualizer.cs
Normal file
42
Content.Client/Light/Visualizers/LightBulbVisualizer.cs
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
using Content.Shared.Light;
|
||||||
|
using JetBrains.Annotations;
|
||||||
|
using Robust.Client.GameObjects;
|
||||||
|
using Robust.Shared.Maths;
|
||||||
|
|
||||||
|
namespace Content.Client.Light.Visualizers
|
||||||
|
{
|
||||||
|
[UsedImplicitly]
|
||||||
|
public class LightBulbVisualizer : AppearanceVisualizer
|
||||||
|
{
|
||||||
|
public override void OnChangeData(AppearanceComponent component)
|
||||||
|
{
|
||||||
|
base.OnChangeData(component);
|
||||||
|
|
||||||
|
if (!component.Owner.TryGetComponent<SpriteComponent>(out var sprite))
|
||||||
|
return;
|
||||||
|
|
||||||
|
// update sprite state
|
||||||
|
if (component.TryGetData(LightBulbVisuals.State, out LightBulbState state))
|
||||||
|
{
|
||||||
|
switch (state)
|
||||||
|
{
|
||||||
|
case LightBulbState.Normal:
|
||||||
|
sprite.LayerSetState(0, "normal");
|
||||||
|
break;
|
||||||
|
case LightBulbState.Broken:
|
||||||
|
sprite.LayerSetState(0, "broken");
|
||||||
|
break;
|
||||||
|
case LightBulbState.Burned:
|
||||||
|
sprite.LayerSetState(0, "burned");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// also update sprites color
|
||||||
|
if (component.TryGetData(LightBulbVisuals.Color, out Color color))
|
||||||
|
{
|
||||||
|
sprite.Color = color;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,131 +1,45 @@
|
|||||||
using System;
|
using Content.Server.Light.EntitySystems;
|
||||||
using Content.Shared.Acts;
|
using Content.Shared.Acts;
|
||||||
using Content.Shared.Audio;
|
using Content.Shared.Light;
|
||||||
using Content.Shared.Sound;
|
using Content.Shared.Sound;
|
||||||
using Content.Shared.Throwing;
|
using Robust.Shared.Analyzers;
|
||||||
using Robust.Server.GameObjects;
|
|
||||||
using Robust.Shared.Audio;
|
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.IoC;
|
|
||||||
using Robust.Shared.Maths;
|
using Robust.Shared.Maths;
|
||||||
using Robust.Shared.Player;
|
|
||||||
using Robust.Shared.Prototypes;
|
|
||||||
using Robust.Shared.Random;
|
|
||||||
using Robust.Shared.Serialization.Manager.Attributes;
|
using Robust.Shared.Serialization.Manager.Attributes;
|
||||||
using Robust.Shared.ViewVariables;
|
|
||||||
|
|
||||||
namespace Content.Server.Light.Components
|
namespace Content.Server.Light.Components
|
||||||
{
|
{
|
||||||
public enum LightBulbState
|
|
||||||
{
|
|
||||||
Normal,
|
|
||||||
Broken,
|
|
||||||
Burned,
|
|
||||||
}
|
|
||||||
|
|
||||||
public enum LightBulbType
|
|
||||||
{
|
|
||||||
Bulb,
|
|
||||||
Tube,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Component that represents a light bulb. Can be broken, or burned, which turns them mostly useless.
|
/// Component that represents a light bulb. Can be broken, or burned, which turns them mostly useless.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[RegisterComponent]
|
[RegisterComponent, Friend(typeof(LightBulbSystem))]
|
||||||
public class LightBulbComponent : Component, IBreakAct
|
public class LightBulbComponent : Component, IBreakAct
|
||||||
{
|
{
|
||||||
/// <summary>
|
public override string Name => "LightBulb";
|
||||||
/// Invoked whenever the state of the light bulb changes.
|
|
||||||
/// </summary>
|
|
||||||
public event EventHandler<EventArgs>? OnLightBulbStateChange;
|
|
||||||
public event EventHandler<EventArgs?>? OnLightColorChange;
|
|
||||||
|
|
||||||
[DataField("color")]
|
[DataField("color")]
|
||||||
private Color _color = Color.White;
|
public Color Color = Color.White;
|
||||||
|
|
||||||
[ViewVariables(VVAccess.ReadWrite)]
|
|
||||||
public Color Color
|
|
||||||
{
|
|
||||||
get { return _color; }
|
|
||||||
set
|
|
||||||
{
|
|
||||||
_color = value;
|
|
||||||
OnLightColorChange?.Invoke(this, null);
|
|
||||||
UpdateColor();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string Name => "LightBulb";
|
|
||||||
|
|
||||||
[DataField("bulb")]
|
[DataField("bulb")]
|
||||||
public LightBulbType Type = LightBulbType.Tube;
|
public LightBulbType Type = LightBulbType.Tube;
|
||||||
|
|
||||||
|
[DataField("startingState")]
|
||||||
|
public LightBulbState State = LightBulbState.Normal;
|
||||||
|
|
||||||
[DataField("BurningTemperature")]
|
[DataField("BurningTemperature")]
|
||||||
private int _burningTemperature = 1400;
|
public int BurningTemperature = 1400;
|
||||||
public int BurningTemperature => _burningTemperature;
|
|
||||||
|
|
||||||
[DataField("PowerUse")]
|
[DataField("PowerUse")]
|
||||||
private int _powerUse = 40;
|
public int PowerUse = 40;
|
||||||
public int PowerUse => _powerUse;
|
|
||||||
|
|
||||||
[DataField("breakSound")]
|
[DataField("breakSound")]
|
||||||
private SoundSpecifier _breakSound = new SoundCollectionSpecifier("GlassBreak");
|
public SoundSpecifier BreakSound = new SoundCollectionSpecifier("GlassBreak");
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The current state of the light bulb. Invokes the OnLightBulbStateChange event when set.
|
|
||||||
/// It also updates the bulb's sprite accordingly.
|
|
||||||
/// </summary>
|
|
||||||
[ViewVariables(VVAccess.ReadWrite)]
|
|
||||||
public LightBulbState State
|
|
||||||
{
|
|
||||||
get { return _state; }
|
|
||||||
set
|
|
||||||
{
|
|
||||||
var sprite = Owner.GetComponent<SpriteComponent>();
|
|
||||||
OnLightBulbStateChange?.Invoke(this, EventArgs.Empty);
|
|
||||||
_state = value;
|
|
||||||
switch (value)
|
|
||||||
{
|
|
||||||
case LightBulbState.Normal:
|
|
||||||
sprite.LayerSetState(0, "normal");
|
|
||||||
break;
|
|
||||||
case LightBulbState.Broken:
|
|
||||||
sprite.LayerSetState(0, "broken");
|
|
||||||
break;
|
|
||||||
case LightBulbState.Burned:
|
|
||||||
sprite.LayerSetState(0, "burned");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private LightBulbState _state = LightBulbState.Normal;
|
|
||||||
|
|
||||||
public void UpdateColor()
|
|
||||||
{
|
|
||||||
if (!Owner.TryGetComponent(out SpriteComponent? sprite))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
sprite.Color = Color;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void Initialize()
|
|
||||||
{
|
|
||||||
base.Initialize();
|
|
||||||
UpdateColor();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// TODO: move me to ECS
|
||||||
public void OnBreak(BreakageEventArgs eventArgs)
|
public void OnBreak(BreakageEventArgs eventArgs)
|
||||||
{
|
{
|
||||||
State = LightBulbState.Broken;
|
EntitySystem.Get<LightBulbSystem>()
|
||||||
}
|
.SetState(Owner.Uid, LightBulbState.Broken, this);
|
||||||
|
|
||||||
public void PlayBreakSound()
|
|
||||||
{
|
|
||||||
SoundSystem.Play(Filter.Pvs(Owner), _breakSound.GetSound(), Owner);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using Content.Server.Light.EntitySystems;
|
||||||
using Content.Server.Storage.Components;
|
using Content.Server.Storage.Components;
|
||||||
|
using Content.Shared.Light;
|
||||||
using Content.Shared.Popups;
|
using Content.Shared.Popups;
|
||||||
using Content.Shared.Sound;
|
using Content.Shared.Sound;
|
||||||
using Robust.Shared.Audio;
|
using Robust.Shared.Audio;
|
||||||
@@ -41,7 +43,14 @@ namespace Content.Server.Light.Components
|
|||||||
public bool TryReplaceBulb(PoweredLightComponent fixture, IEntity? user = null)
|
public bool TryReplaceBulb(PoweredLightComponent fixture, IEntity? user = null)
|
||||||
{
|
{
|
||||||
// check if light bulb is broken or missing
|
// check if light bulb is broken or missing
|
||||||
if (fixture.LightBulb != null && fixture.LightBulb.State == LightBulbState.Normal) return false;
|
var lightSystem = EntitySystem.Get<PoweredLightSystem>();
|
||||||
|
var fixtureBulbUid = lightSystem.GetBulb(fixture.Owner.Uid, fixture);
|
||||||
|
if (fixtureBulbUid == null)
|
||||||
|
return false;
|
||||||
|
if (!Owner.EntityManager.TryGetComponent(fixtureBulbUid.Value, out LightBulbComponent? fixtureBulb))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (fixtureBulb.State == LightBulbState.Normal) return false;
|
||||||
|
|
||||||
// try get first inserted bulb of the same type as targeted light fixtutre
|
// try get first inserted bulb of the same type as targeted light fixtutre
|
||||||
var bulb = _insertedBulbs.ContainedEntities.FirstOrDefault(
|
var bulb = _insertedBulbs.ContainedEntities.FirstOrDefault(
|
||||||
@@ -79,7 +88,7 @@ namespace Content.Server.Light.Components
|
|||||||
}
|
}
|
||||||
|
|
||||||
// insert it into fixture
|
// insert it into fixture
|
||||||
var wasReplaced = fixture.ReplaceBulb(bulb);
|
var wasReplaced = lightSystem.ReplaceBulb(fixture.Owner.Uid, bulb.Uid, fixture);
|
||||||
if (wasReplaced)
|
if (wasReplaced)
|
||||||
{
|
{
|
||||||
SoundSystem.Play(Filter.Pvs(Owner), _sound.GetSound(), Owner,
|
SoundSystem.Play(Filter.Pvs(Owner), _sound.GetSound(), Owner,
|
||||||
|
|||||||
@@ -1,25 +1,12 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Threading.Tasks;
|
using Content.Server.Light.EntitySystems;
|
||||||
using Content.Server.Hands.Components;
|
|
||||||
using Content.Server.Items;
|
|
||||||
using Content.Server.Power.Components;
|
|
||||||
using Content.Server.Temperature.Components;
|
|
||||||
using Content.Shared.Audio;
|
|
||||||
using Content.Shared.Damage;
|
using Content.Shared.Damage;
|
||||||
using Content.Shared.Interaction;
|
|
||||||
using Content.Shared.Light;
|
using Content.Shared.Light;
|
||||||
using Content.Shared.Popups;
|
|
||||||
using Content.Shared.Sound;
|
using Content.Shared.Sound;
|
||||||
using Robust.Server.GameObjects;
|
using Robust.Shared.Analyzers;
|
||||||
using Robust.Shared.Audio;
|
|
||||||
using Robust.Shared.Containers;
|
using Robust.Shared.Containers;
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.IoC;
|
|
||||||
using Robust.Shared.Localization;
|
|
||||||
using Robust.Shared.Maths;
|
|
||||||
using Robust.Shared.Player;
|
|
||||||
using Robust.Shared.Serialization.Manager.Attributes;
|
using Robust.Shared.Serialization.Manager.Attributes;
|
||||||
using Robust.Shared.Timing;
|
|
||||||
using Robust.Shared.ViewVariables;
|
using Robust.Shared.ViewVariables;
|
||||||
|
|
||||||
namespace Content.Server.Light.Components
|
namespace Content.Server.Light.Components
|
||||||
@@ -27,39 +14,30 @@ namespace Content.Server.Light.Components
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Component that represents a wall light. It has a light bulb that can be replaced when broken.
|
/// Component that represents a wall light. It has a light bulb that can be replaced when broken.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[RegisterComponent]
|
[RegisterComponent, Friend(typeof(PoweredLightSystem))]
|
||||||
public class PoweredLightComponent : Component, IInteractHand, IInteractUsing, IMapInit
|
public class PoweredLightComponent : Component
|
||||||
{
|
{
|
||||||
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
|
||||||
|
|
||||||
public override string Name => "PoweredLight";
|
public override string Name => "PoweredLight";
|
||||||
|
|
||||||
private static readonly TimeSpan _thunkDelay = TimeSpan.FromSeconds(2);
|
|
||||||
|
|
||||||
[ComponentDependency]
|
|
||||||
private readonly AppearanceComponent? _appearance;
|
|
||||||
|
|
||||||
private TimeSpan _lastThunk;
|
|
||||||
public TimeSpan? LastGhostBlink;
|
|
||||||
|
|
||||||
[DataField("burnHandSound")]
|
[DataField("burnHandSound")]
|
||||||
private SoundSpecifier _burnHandSound = new SoundPathSpecifier("/Audio/Effects/lightburn.ogg");
|
public SoundSpecifier BurnHandSound = new SoundPathSpecifier("/Audio/Effects/lightburn.ogg");
|
||||||
|
|
||||||
[DataField("turnOnSound")]
|
[DataField("turnOnSound")]
|
||||||
private SoundSpecifier _turnOnSound = new SoundPathSpecifier("/Audio/Machines/light_tube_on.ogg");
|
public SoundSpecifier TurnOnSound = new SoundPathSpecifier("/Audio/Machines/light_tube_on.ogg");
|
||||||
|
|
||||||
[DataField("hasLampOnSpawn")]
|
[DataField("hasLampOnSpawn")]
|
||||||
private bool _hasLampOnSpawn = true;
|
public bool HasLampOnSpawn = true;
|
||||||
|
|
||||||
|
[DataField("bulb")]
|
||||||
|
public LightBulbType BulbType;
|
||||||
|
|
||||||
[ViewVariables]
|
[ViewVariables]
|
||||||
[DataField("on")]
|
[DataField("on")]
|
||||||
private bool _on = true;
|
public bool On = true;
|
||||||
|
|
||||||
[ViewVariables]
|
[DataField("damage", required: true)]
|
||||||
private bool _currentLit;
|
[ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public DamageSpecifier Damage = default!;
|
||||||
[ViewVariables]
|
|
||||||
public bool IsBlinking;
|
|
||||||
|
|
||||||
[ViewVariables]
|
[ViewVariables]
|
||||||
[DataField("ignoreGhostsBoo")]
|
[DataField("ignoreGhostsBoo")]
|
||||||
@@ -73,254 +51,15 @@ namespace Content.Server.Light.Components
|
|||||||
[DataField("ghostBlinkingCooldown")]
|
[DataField("ghostBlinkingCooldown")]
|
||||||
public TimeSpan GhostBlinkingCooldown = TimeSpan.FromSeconds(60);
|
public TimeSpan GhostBlinkingCooldown = TimeSpan.FromSeconds(60);
|
||||||
|
|
||||||
|
|
||||||
[DataField("bulb")] private LightBulbType _bulbType = LightBulbType.Tube;
|
|
||||||
public LightBulbType BulbType => _bulbType;
|
|
||||||
|
|
||||||
[ViewVariables] private ContainerSlot _lightBulbContainer = default!;
|
|
||||||
|
|
||||||
[DataField("damage", required: true)]
|
|
||||||
[ViewVariables(VVAccess.ReadWrite)]
|
|
||||||
public DamageSpecifier Damage = default!;
|
|
||||||
|
|
||||||
protected override void Initialize()
|
|
||||||
{
|
|
||||||
base.Initialize();
|
|
||||||
_lightBulbContainer = Owner.EnsureContainer<ContainerSlot>("light_bulb");
|
|
||||||
}
|
|
||||||
|
|
||||||
[ViewVariables]
|
[ViewVariables]
|
||||||
public LightBulbComponent? LightBulb
|
public ContainerSlot LightBulbContainer = default!;
|
||||||
{
|
[ViewVariables]
|
||||||
get
|
public bool CurrentLit;
|
||||||
{
|
[ViewVariables]
|
||||||
if (_lightBulbContainer.ContainedEntity == null) return null;
|
public bool IsBlinking;
|
||||||
|
[ViewVariables]
|
||||||
_lightBulbContainer.ContainedEntity.TryGetComponent(out LightBulbComponent? bulb);
|
public TimeSpan LastThunk;
|
||||||
|
[ViewVariables]
|
||||||
return bulb;
|
public TimeSpan? LastGhostBlink;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO CONSTRUCTION make this use a construction graph
|
|
||||||
|
|
||||||
async Task<bool> IInteractUsing.InteractUsing(InteractUsingEventArgs eventArgs)
|
|
||||||
{
|
|
||||||
return InsertBulb(eventArgs.Using);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IInteractHand.InteractHand(InteractHandEventArgs eventArgs)
|
|
||||||
{
|
|
||||||
if (!eventArgs.User.HasComponent<DamageableComponent>())
|
|
||||||
{
|
|
||||||
Eject();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (eventArgs.User.TryGetComponent(out HeatResistanceComponent? heatResistanceComponent))
|
|
||||||
{
|
|
||||||
if (CanBurn(heatResistanceComponent.GetHeatResistance()))
|
|
||||||
{
|
|
||||||
Burn();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Eject();
|
|
||||||
return true;
|
|
||||||
|
|
||||||
bool CanBurn(int heatResistance)
|
|
||||||
{
|
|
||||||
if (LightBulb == null)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return _currentLit && heatResistance < LightBulb.BurningTemperature;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Burn()
|
|
||||||
{
|
|
||||||
Owner.PopupMessage(eventArgs.User, Loc.GetString("powered-light-component-burn-hand"));
|
|
||||||
EntitySystem.Get<DamageableSystem>().TryChangeDamage(eventArgs.User.Uid, Damage);
|
|
||||||
SoundSystem.Play(Filter.Pvs(Owner), _burnHandSound.GetSound(), Owner);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Eject()
|
|
||||||
{
|
|
||||||
EjectBulb(eventArgs.User);
|
|
||||||
UpdateLight();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Try to replace current bulb with a new one
|
|
||||||
/// </summary>
|
|
||||||
public bool ReplaceBulb(IEntity bulb)
|
|
||||||
{
|
|
||||||
EjectBulb();
|
|
||||||
return InsertBulb(bulb);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Inserts the bulb if possible.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>True if it could insert it, false if it couldn't.</returns>
|
|
||||||
private bool InsertBulb(IEntity bulb)
|
|
||||||
{
|
|
||||||
if (LightBulb != null) return false;
|
|
||||||
if (!bulb.TryGetComponent(out LightBulbComponent? lightBulb)) return false;
|
|
||||||
if (lightBulb.Type != _bulbType) return false;
|
|
||||||
|
|
||||||
var inserted = _lightBulbContainer.Insert(bulb);
|
|
||||||
|
|
||||||
lightBulb.OnLightBulbStateChange += UpdateLight;
|
|
||||||
lightBulb.OnLightColorChange += UpdateLight;
|
|
||||||
|
|
||||||
UpdateLight();
|
|
||||||
|
|
||||||
return inserted;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Ejects the bulb to a mob's hand if possible.
|
|
||||||
/// </summary>
|
|
||||||
private void EjectBulb(IEntity? user = null)
|
|
||||||
{
|
|
||||||
if (LightBulb == null) return;
|
|
||||||
|
|
||||||
var bulb = LightBulb;
|
|
||||||
|
|
||||||
bulb.OnLightBulbStateChange -= UpdateLight;
|
|
||||||
bulb.OnLightColorChange -= UpdateLight;
|
|
||||||
|
|
||||||
if (!_lightBulbContainer.Remove(bulb.Owner)) return;
|
|
||||||
|
|
||||||
if (user != null)
|
|
||||||
{
|
|
||||||
if (!user.TryGetComponent(out HandsComponent? hands)
|
|
||||||
|| !hands.PutInHand(bulb.Owner.GetComponent<ItemComponent>()))
|
|
||||||
bulb.Owner.Transform.Coordinates = user.Transform.Coordinates;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bulb.Owner.Transform.Coordinates = Owner.Transform.Coordinates;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// For attaching UpdateLight() to events.
|
|
||||||
/// </summary>
|
|
||||||
public void UpdateLight(object? sender, EventArgs? e)
|
|
||||||
{
|
|
||||||
UpdateLight();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Updates the light's power drain, sprite and actual light state.
|
|
||||||
/// </summary>
|
|
||||||
public void UpdateLight()
|
|
||||||
{
|
|
||||||
var powerReceiver = Owner.GetComponent<ApcPowerReceiverComponent>();
|
|
||||||
powerReceiver.Load = (LightBulb != null && _on && LightBulb.State == LightBulbState.Normal) ? LightBulb.PowerUse : 0;
|
|
||||||
|
|
||||||
if (LightBulb == null) // No light bulb.
|
|
||||||
{
|
|
||||||
SetLight(false);
|
|
||||||
_appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Empty);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (LightBulb.State)
|
|
||||||
{
|
|
||||||
case LightBulbState.Normal:
|
|
||||||
if (powerReceiver.Powered && _on)
|
|
||||||
{
|
|
||||||
SetLight(true, LightBulb.Color);
|
|
||||||
_appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.On);
|
|
||||||
var time = _gameTiming.CurTime;
|
|
||||||
if (time > _lastThunk + _thunkDelay)
|
|
||||||
{
|
|
||||||
_lastThunk = time;
|
|
||||||
SoundSystem.Play(Filter.Pvs(Owner), _turnOnSound.GetSound(), Owner, AudioParams.Default.WithVolume(-10f));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
SetLight(false);
|
|
||||||
_appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Off);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case LightBulbState.Broken:
|
|
||||||
SetLight(false);
|
|
||||||
_appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Broken);
|
|
||||||
break;
|
|
||||||
case LightBulbState.Burned:
|
|
||||||
SetLight(false);
|
|
||||||
_appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Burned);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void SetLight(bool value, Color? color = null)
|
|
||||||
{
|
|
||||||
_currentLit = value;
|
|
||||||
EntitySystem.Get<SharedAmbientSoundSystem>().SetAmbience(Owner.Uid, value);
|
|
||||||
|
|
||||||
if (!Owner.TryGetComponent(out PointLightComponent? pointLight)) return;
|
|
||||||
pointLight.Enabled = value;
|
|
||||||
|
|
||||||
if (color != null)
|
|
||||||
pointLight.Color = color.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void HandleMessage(ComponentMessage message, IComponent? component)
|
|
||||||
{
|
|
||||||
base.HandleMessage(message, component);
|
|
||||||
switch (message)
|
|
||||||
{
|
|
||||||
case PowerChangedMessage:
|
|
||||||
UpdateLight();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void TryDestroyBulb()
|
|
||||||
{
|
|
||||||
if (LightBulb == null || LightBulb.State == LightBulbState.Broken)
|
|
||||||
return;
|
|
||||||
|
|
||||||
LightBulb.State = LightBulbState.Broken;
|
|
||||||
LightBulb.PlayBreakSound();
|
|
||||||
UpdateLight();
|
|
||||||
}
|
|
||||||
|
|
||||||
void IMapInit.MapInit()
|
|
||||||
{
|
|
||||||
if (_hasLampOnSpawn)
|
|
||||||
{
|
|
||||||
var prototype = _bulbType switch
|
|
||||||
{
|
|
||||||
LightBulbType.Bulb => "LightBulb",
|
|
||||||
LightBulbType.Tube => "LightTube",
|
|
||||||
_ => throw new ArgumentOutOfRangeException()
|
|
||||||
};
|
|
||||||
|
|
||||||
var entity = Owner.EntityManager.SpawnEntity(prototype, Owner.Transform.Coordinates);
|
|
||||||
_lightBulbContainer.Insert(entity);
|
|
||||||
}
|
|
||||||
|
|
||||||
// need this to update visualizers
|
|
||||||
UpdateLight();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void ToggleLight()
|
|
||||||
{
|
|
||||||
_on = !_on;
|
|
||||||
UpdateLight();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SetState(bool state)
|
|
||||||
{
|
|
||||||
_on = state;
|
|
||||||
UpdateLight();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
using Content.Server.Light.Components;
|
using Content.Server.Light.Components;
|
||||||
|
using Content.Server.Light.Events;
|
||||||
|
using Content.Shared.Light;
|
||||||
using Content.Shared.Throwing;
|
using Content.Shared.Throwing;
|
||||||
|
using Robust.Server.GameObjects;
|
||||||
|
using Robust.Shared.Audio;
|
||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
|
using Robust.Shared.Maths;
|
||||||
|
using Robust.Shared.Player;
|
||||||
|
|
||||||
namespace Content.Server.Light.EntitySystems
|
namespace Content.Server.Light.EntitySystems
|
||||||
{
|
{
|
||||||
@@ -9,13 +15,65 @@ namespace Content.Server.Light.EntitySystems
|
|||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
base.Initialize();
|
base.Initialize();
|
||||||
|
|
||||||
|
SubscribeLocalEvent<LightBulbComponent, ComponentInit>(OnInit);
|
||||||
SubscribeLocalEvent<LightBulbComponent, LandEvent>(HandleLand);
|
SubscribeLocalEvent<LightBulbComponent, LandEvent>(HandleLand);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void HandleLand(EntityUid uid, LightBulbComponent component, LandEvent args)
|
private void OnInit(EntityUid uid, LightBulbComponent bulb, ComponentInit args)
|
||||||
{
|
{
|
||||||
component.PlayBreakSound();
|
// update default state of bulbs
|
||||||
component.State = LightBulbState.Broken;
|
SetColor(uid, bulb.Color, bulb);
|
||||||
|
SetState(uid, bulb.State, bulb);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void HandleLand(EntityUid uid, LightBulbComponent bulb, LandEvent args)
|
||||||
|
{
|
||||||
|
PlayBreakSound(uid, bulb);
|
||||||
|
SetState(uid, LightBulbState.Broken, bulb);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Set a new color for a light bulb and raise event about change
|
||||||
|
/// </summary>
|
||||||
|
public void SetColor(EntityUid uid, Color color, LightBulbComponent? bulb = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref bulb))
|
||||||
|
return;
|
||||||
|
|
||||||
|
bulb.Color = color;
|
||||||
|
UpdateAppearance(uid, bulb);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Set a new state for a light bulb (broken, burned) and raise event about change
|
||||||
|
/// </summary>
|
||||||
|
public void SetState(EntityUid uid, LightBulbState state, LightBulbComponent? bulb = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref bulb))
|
||||||
|
return;
|
||||||
|
|
||||||
|
bulb.State = state;
|
||||||
|
UpdateAppearance(uid, bulb);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PlayBreakSound(EntityUid uid, LightBulbComponent? bulb = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref bulb))
|
||||||
|
return;
|
||||||
|
|
||||||
|
SoundSystem.Play(Filter.Pvs(uid), bulb.BreakSound.GetSound(), uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UpdateAppearance(EntityUid uid, LightBulbComponent? bulb = null,
|
||||||
|
AppearanceComponent? appearance = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref bulb, ref appearance, logMissing: false))
|
||||||
|
return;
|
||||||
|
|
||||||
|
// try to update appearance and color
|
||||||
|
appearance.SetData(LightBulbVisuals.State, bulb.State);
|
||||||
|
appearance.SetData(LightBulbVisuals.Color, bulb.Color);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
34
Content.Server/Light/EntitySystems/LitOnPoweredSystem.cs
Normal file
34
Content.Server/Light/EntitySystems/LitOnPoweredSystem.cs
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
using Content.Server.Light.Components;
|
||||||
|
using Content.Server.Power.Components;
|
||||||
|
using Content.Server.Power.EntitySystems;
|
||||||
|
using Robust.Server.GameObjects;
|
||||||
|
using Robust.Shared.GameObjects;
|
||||||
|
|
||||||
|
namespace Content.Server.Light.EntitySystems
|
||||||
|
{
|
||||||
|
public class LitOnPoweredSystem : EntitySystem
|
||||||
|
{
|
||||||
|
public override void Initialize()
|
||||||
|
{
|
||||||
|
base.Initialize();
|
||||||
|
SubscribeLocalEvent<LitOnPoweredComponent, PowerChangedEvent>(OnPowerChanged);
|
||||||
|
SubscribeLocalEvent<LitOnPoweredComponent, PowerNetBatterySupplyEvent>(OnPowerSupply);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnPowerChanged(EntityUid uid, LitOnPoweredComponent component, PowerChangedEvent args)
|
||||||
|
{
|
||||||
|
if (EntityManager.TryGetComponent<PointLightComponent>(uid, out var light))
|
||||||
|
{
|
||||||
|
light.Enabled = args.Powered;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnPowerSupply(EntityUid uid, LitOnPoweredComponent component, PowerNetBatterySupplyEvent args)
|
||||||
|
{
|
||||||
|
if (EntityManager.TryGetComponent<PointLightComponent>(uid, out var light))
|
||||||
|
{
|
||||||
|
light.Enabled = args.Supply;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -12,27 +12,270 @@ using Robust.Server.GameObjects;
|
|||||||
using Robust.Shared.GameObjects;
|
using Robust.Shared.GameObjects;
|
||||||
using Robust.Shared.IoC;
|
using Robust.Shared.IoC;
|
||||||
using Robust.Shared.Timing;
|
using Robust.Shared.Timing;
|
||||||
|
using Robust.Shared.Containers;
|
||||||
|
using Content.Shared.Interaction;
|
||||||
|
using Content.Shared.Hands.Components;
|
||||||
|
using Content.Server.Temperature.Components;
|
||||||
|
using Content.Shared.Popups;
|
||||||
|
using Robust.Shared.Localization;
|
||||||
|
using Robust.Shared.Audio;
|
||||||
|
using Robust.Shared.Player;
|
||||||
|
using Robust.Shared.Maths;
|
||||||
|
using Content.Shared.Audio;
|
||||||
|
|
||||||
namespace Content.Server.Light.EntitySystems
|
namespace Content.Server.Light.EntitySystems
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// System for the PoweredLightComponent. Currently bare-bones, to handle events from the DamageableSystem
|
/// System for the PoweredLightComponens
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class PoweredLightSystem : EntitySystem
|
public class PoweredLightSystem : EntitySystem
|
||||||
{
|
{
|
||||||
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
[Dependency] private readonly IGameTiming _gameTiming = default!;
|
||||||
|
[Dependency] private readonly DamageableSystem _damageableSystem = default!;
|
||||||
|
[Dependency] private readonly SharedAmbientSoundSystem _ambientSystem = default!;
|
||||||
|
[Dependency] private readonly LightBulbSystem _bulbSystem = default!;
|
||||||
|
[Dependency] private readonly SharedPopupSystem _popupSystem = default!;
|
||||||
|
|
||||||
|
private static readonly TimeSpan ThunkDelay = TimeSpan.FromSeconds(2);
|
||||||
|
|
||||||
public override void Initialize()
|
public override void Initialize()
|
||||||
{
|
{
|
||||||
base.Initialize();
|
base.Initialize();
|
||||||
|
SubscribeLocalEvent<PoweredLightComponent, ComponentInit>(OnInit);
|
||||||
|
SubscribeLocalEvent<PoweredLightComponent, MapInitEvent>(OnMapInit);
|
||||||
|
SubscribeLocalEvent<PoweredLightComponent, InteractUsingEvent>(OnInteractUsing);
|
||||||
|
SubscribeLocalEvent<PoweredLightComponent, InteractHandEvent>(OnInteractHand);
|
||||||
|
|
||||||
SubscribeLocalEvent<PoweredLightComponent, GhostBooEvent>(OnGhostBoo);
|
SubscribeLocalEvent<PoweredLightComponent, GhostBooEvent>(OnGhostBoo);
|
||||||
SubscribeLocalEvent<PoweredLightComponent, DamageChangedEvent>(HandleLightDamaged);
|
SubscribeLocalEvent<PoweredLightComponent, DamageChangedEvent>(HandleLightDamaged);
|
||||||
|
|
||||||
SubscribeLocalEvent<PoweredLightComponent, SignalReceivedEvent>(OnSignalReceived);
|
SubscribeLocalEvent<PoweredLightComponent, SignalReceivedEvent>(OnSignalReceived);
|
||||||
SubscribeLocalEvent<PoweredLightComponent, PacketSentEvent>(OnPacketReceived);
|
SubscribeLocalEvent<PoweredLightComponent, PacketSentEvent>(OnPacketReceived);
|
||||||
|
|
||||||
SubscribeLocalEvent<LitOnPoweredComponent, PowerChangedEvent>(OnPowerChanged);
|
SubscribeLocalEvent<PoweredLightComponent, PowerChangedEvent>(OnPowerChanged);
|
||||||
SubscribeLocalEvent<LitOnPoweredComponent, PowerNetBatterySupplyEvent>(OnPowerSupply);
|
}
|
||||||
|
|
||||||
|
private void OnInit(EntityUid uid, PoweredLightComponent light, ComponentInit args)
|
||||||
|
{
|
||||||
|
light.LightBulbContainer = light.Owner.EnsureContainer<ContainerSlot>("light_bulb");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnMapInit(EntityUid uid, PoweredLightComponent light, MapInitEvent args)
|
||||||
|
{
|
||||||
|
if (light.HasLampOnSpawn)
|
||||||
|
{
|
||||||
|
var prototype = light.BulbType switch
|
||||||
|
{
|
||||||
|
LightBulbType.Bulb => "LightBulb",
|
||||||
|
LightBulbType.Tube => "LightTube",
|
||||||
|
_ => throw new ArgumentOutOfRangeException()
|
||||||
|
};
|
||||||
|
|
||||||
|
var entity = EntityManager.SpawnEntity(prototype, light.Owner.Transform.Coordinates);
|
||||||
|
light.LightBulbContainer.Insert(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
// need this to update visualizers
|
||||||
|
UpdateLight(uid, light);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnInteractUsing(EntityUid uid, PoweredLightComponent component, InteractUsingEvent args)
|
||||||
|
{
|
||||||
|
if (args.Handled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
args.Handled = InsertBulb(uid, args.Used.Uid, component);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnInteractHand(EntityUid uid, PoweredLightComponent light, InteractHandEvent args)
|
||||||
|
{
|
||||||
|
if (args.Handled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// check if light has bulb to eject
|
||||||
|
var bulbUid = GetBulb(uid, light);
|
||||||
|
if (bulbUid == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// check if it's possible to apply burn damage to user
|
||||||
|
var userUid = args.User.Uid;
|
||||||
|
if (EntityManager.TryGetComponent(userUid, out HeatResistanceComponent? heatResist) &&
|
||||||
|
EntityManager.TryGetComponent(bulbUid.Value, out LightBulbComponent? lightBulb))
|
||||||
|
{
|
||||||
|
// get users heat resistance
|
||||||
|
var res = heatResist.GetHeatResistance();
|
||||||
|
|
||||||
|
// check heat resistance against user
|
||||||
|
var burnedHand = light.CurrentLit && res < lightBulb.BurningTemperature;
|
||||||
|
if (burnedHand)
|
||||||
|
{
|
||||||
|
// apply damage to users hands and show message with sound
|
||||||
|
var burnMsg = Loc.GetString("powered-light-component-burn-hand");
|
||||||
|
_popupSystem.PopupEntity(burnMsg, uid, Filter.Entities(userUid));
|
||||||
|
_damageableSystem.TryChangeDamage(userUid, light.Damage);
|
||||||
|
SoundSystem.Play(Filter.Pvs(uid), light.BurnHandSound.GetSound(), uid);
|
||||||
|
|
||||||
|
args.Handled = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// all checks passed
|
||||||
|
// just try to eject bulb
|
||||||
|
args.Handled = EjectBulb(uid, userUid, light) != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
#region Bulb Logic API
|
||||||
|
/// <summary>
|
||||||
|
/// Inserts the bulb if possible.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>True if it could insert it, false if it couldn't.</returns>
|
||||||
|
public bool InsertBulb(EntityUid uid, EntityUid bulbUid, PoweredLightComponent? light = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref light))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// check if light already has bulb
|
||||||
|
if (GetBulb(uid, light) != null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// check if bulb fits
|
||||||
|
if (!EntityManager.TryGetComponent(bulbUid, out LightBulbComponent? lightBulb))
|
||||||
|
return false;
|
||||||
|
if (lightBulb.Type != light.BulbType)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// try to insert bulb in container
|
||||||
|
if (!light.LightBulbContainer.Insert(EntityManager.GetEntity(bulbUid)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
UpdateLight(uid, light);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Ejects the bulb to a mob's hand if possible.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>Bulb uid if it was successfully ejected, null otherwise</returns>
|
||||||
|
public EntityUid? EjectBulb(EntityUid uid, EntityUid? userUid = null, PoweredLightComponent? light = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref light))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
// check if light has bulb
|
||||||
|
var bulbUid = GetBulb(uid, light);
|
||||||
|
if (bulbUid == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
// try to remove bulb from container
|
||||||
|
var bulbEnt = EntityManager.GetEntity(bulbUid.Value);
|
||||||
|
if (!light.LightBulbContainer.Remove(bulbEnt))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
// try to place bulb in hands
|
||||||
|
if (userUid != null)
|
||||||
|
{
|
||||||
|
if (EntityManager.TryGetComponent(userUid.Value, out SharedHandsComponent? hands))
|
||||||
|
hands.TryPutInActiveHandOrAny(bulbEnt);
|
||||||
|
}
|
||||||
|
|
||||||
|
UpdateLight(uid, light);
|
||||||
|
return bulbUid;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Try to replace current bulb with a new one
|
||||||
|
/// If succeed old bulb just drops on floor
|
||||||
|
/// </summary>
|
||||||
|
public bool ReplaceBulb(EntityUid uid, EntityUid bulb, PoweredLightComponent? light = null)
|
||||||
|
{
|
||||||
|
EjectBulb(uid, null, light);
|
||||||
|
return InsertBulb(uid, bulb, light);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Try to get light bulb inserted in powered light
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>Bulb uid if it exist, null otherwise</returns>
|
||||||
|
public EntityUid? GetBulb(EntityUid uid, PoweredLightComponent? light = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref light))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return light.LightBulbContainer.ContainedEntity?.Uid;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Try to break bulb inside light fixture
|
||||||
|
/// </summary>
|
||||||
|
public void TryDestroyBulb(EntityUid uid, PoweredLightComponent? light = null)
|
||||||
|
{
|
||||||
|
// check bulb state
|
||||||
|
var bulbUid = GetBulb(uid, light);
|
||||||
|
if (bulbUid == null || !EntityManager.TryGetComponent(bulbUid.Value, out LightBulbComponent? lightBulb))
|
||||||
|
return;
|
||||||
|
if (lightBulb.State == LightBulbState.Broken)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// break it
|
||||||
|
_bulbSystem.SetState(bulbUid.Value, LightBulbState.Broken, lightBulb);
|
||||||
|
_bulbSystem.PlayBreakSound(bulbUid.Value, lightBulb);
|
||||||
|
UpdateLight(uid, light);
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
private void UpdateLight(EntityUid uid,
|
||||||
|
PoweredLightComponent? light = null,
|
||||||
|
ApcPowerReceiverComponent? powerReceiver = null,
|
||||||
|
AppearanceComponent? appearance = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref light, ref powerReceiver, ref appearance))
|
||||||
|
return;
|
||||||
|
|
||||||
|
// check if light has bulb
|
||||||
|
var bulbUid = GetBulb(uid, light);
|
||||||
|
if (bulbUid == null || !EntityManager.TryGetComponent(bulbUid.Value, out LightBulbComponent? lightBulb))
|
||||||
|
{
|
||||||
|
SetLight(uid, false, light: light);
|
||||||
|
powerReceiver.Load = 0;
|
||||||
|
appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Empty);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
powerReceiver.Load = (light.On && lightBulb.State == LightBulbState.Normal) ? lightBulb.PowerUse : 0;
|
||||||
|
|
||||||
|
switch (lightBulb.State)
|
||||||
|
{
|
||||||
|
case LightBulbState.Normal:
|
||||||
|
if (powerReceiver.Powered && light.On)
|
||||||
|
{
|
||||||
|
SetLight(uid, true, lightBulb.Color, light);
|
||||||
|
appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.On);
|
||||||
|
var time = _gameTiming.CurTime;
|
||||||
|
if (time > light.LastThunk + ThunkDelay)
|
||||||
|
{
|
||||||
|
light.LastThunk = time;
|
||||||
|
SoundSystem.Play(Filter.Pvs(uid), light.TurnOnSound.GetSound(), uid, AudioParams.Default.WithVolume(-10f));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
SetLight(uid, false, light: light);
|
||||||
|
appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Off);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case LightBulbState.Broken:
|
||||||
|
SetLight(uid, false, light: light);
|
||||||
|
appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Broken);
|
||||||
|
break;
|
||||||
|
case LightBulbState.Burned:
|
||||||
|
SetLight(uid, false, light: light);
|
||||||
|
appearance?.SetData(PoweredLightVisuals.BulbState, PoweredLightState.Burned);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -44,7 +287,7 @@ namespace Content.Server.Light.EntitySystems
|
|||||||
if (args.DamageIncreased)
|
if (args.DamageIncreased)
|
||||||
{
|
{
|
||||||
// Eventually, this logic should all be done by this (or some other) system, not a component.
|
// Eventually, this logic should all be done by this (or some other) system, not a component.
|
||||||
component.TryDestroyBulb();
|
TryDestroyBulb(uid, component);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -72,6 +315,11 @@ namespace Content.Server.Light.EntitySystems
|
|||||||
args.Handled = true;
|
args.Handled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void OnPowerChanged(EntityUid uid, PoweredLightComponent component, PowerChangedEvent args)
|
||||||
|
{
|
||||||
|
UpdateLight(uid, component);
|
||||||
|
}
|
||||||
|
|
||||||
public void ToggleBlinkingLight(PoweredLightComponent light, bool isNowBlinking)
|
public void ToggleBlinkingLight(PoweredLightComponent light, bool isNowBlinking)
|
||||||
{
|
{
|
||||||
if (light.IsBlinking == isNowBlinking)
|
if (light.IsBlinking == isNowBlinking)
|
||||||
@@ -89,7 +337,7 @@ namespace Content.Server.Light.EntitySystems
|
|||||||
switch (args.Port)
|
switch (args.Port)
|
||||||
{
|
{
|
||||||
case "state":
|
case "state":
|
||||||
component.ToggleLight();
|
ToggleLight(uid, component);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -103,23 +351,42 @@ namespace Content.Server.Light.EntitySystems
|
|||||||
if (!args.Data.TryGetValue(DeviceNetworkConstants.Command, out string? command) || command != DeviceNetworkConstants.CmdSetState) return;
|
if (!args.Data.TryGetValue(DeviceNetworkConstants.Command, out string? command) || command != DeviceNetworkConstants.CmdSetState) return;
|
||||||
if (!args.Data.TryGetValue(DeviceNetworkConstants.StateEnabled, out bool enabled)) return;
|
if (!args.Data.TryGetValue(DeviceNetworkConstants.StateEnabled, out bool enabled)) return;
|
||||||
|
|
||||||
component.SetState(enabled);
|
SetState(uid, enabled, component);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnPowerChanged(EntityUid uid, LitOnPoweredComponent component, PowerChangedEvent args)
|
private void SetLight(EntityUid uid, bool value, Color? color = null, PoweredLightComponent? light = null)
|
||||||
{
|
{
|
||||||
if (EntityManager.TryGetComponent<PointLightComponent>(uid, out var light))
|
if (!Resolve(uid, ref light))
|
||||||
|
return;
|
||||||
|
|
||||||
|
light.CurrentLit = value;
|
||||||
|
_ambientSystem.SetAmbience(uid, value);
|
||||||
|
|
||||||
|
if (EntityManager.TryGetComponent(uid, out PointLightComponent? pointLight))
|
||||||
{
|
{
|
||||||
light.Enabled = args.Powered;
|
pointLight.Enabled = value;
|
||||||
|
|
||||||
|
if (color != null)
|
||||||
|
pointLight.Color = color.Value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnPowerSupply(EntityUid uid, LitOnPoweredComponent component, PowerNetBatterySupplyEvent args)
|
public void ToggleLight(EntityUid uid, PoweredLightComponent? light = null)
|
||||||
{
|
{
|
||||||
if (EntityManager.TryGetComponent<PointLightComponent>(uid, out var light))
|
if (!Resolve(uid, ref light))
|
||||||
{
|
return;
|
||||||
light.Enabled = args.Supply;
|
|
||||||
}
|
light.On = !light.On;
|
||||||
|
UpdateLight(uid, light);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetState(EntityUid uid, bool state, PoweredLightComponent? light = null)
|
||||||
|
{
|
||||||
|
if (!Resolve(uid, ref light))
|
||||||
|
return;
|
||||||
|
|
||||||
|
light.On = state;
|
||||||
|
UpdateLight(uid, light);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
27
Content.Shared/Light/SharedLightBulb.cs
Normal file
27
Content.Shared/Light/SharedLightBulb.cs
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
using Robust.Shared.Serialization;
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Content.Shared.Light
|
||||||
|
{
|
||||||
|
[Serializable, NetSerializable]
|
||||||
|
public enum LightBulbState : byte
|
||||||
|
{
|
||||||
|
Normal,
|
||||||
|
Broken,
|
||||||
|
Burned,
|
||||||
|
}
|
||||||
|
|
||||||
|
[Serializable, NetSerializable]
|
||||||
|
public enum LightBulbVisuals : byte
|
||||||
|
{
|
||||||
|
State,
|
||||||
|
Color
|
||||||
|
}
|
||||||
|
|
||||||
|
[Serializable, NetSerializable]
|
||||||
|
public enum LightBulbType : byte
|
||||||
|
{
|
||||||
|
Bulb,
|
||||||
|
Tube,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,13 @@
|
|||||||
# TODO: Add description (1)
|
# TODO: Add description (1)
|
||||||
- type: entity
|
- type: entity
|
||||||
parent: BaseItem
|
parent: BaseItem
|
||||||
id: BaseLightbulb
|
id: BaseLightbulb
|
||||||
abstract: true
|
abstract: true
|
||||||
components:
|
components:
|
||||||
|
- type: Sprite
|
||||||
|
netsync: false
|
||||||
|
sprite: Objects/Power/light_bulb.rsi
|
||||||
|
state: normal
|
||||||
- type: LightBulb
|
- type: LightBulb
|
||||||
- type: Damageable
|
- type: Damageable
|
||||||
damageContainer: Inorganic
|
damageContainer: Inorganic
|
||||||
@@ -40,6 +44,9 @@
|
|||||||
max: 1
|
max: 1
|
||||||
- !type:DoActsBehavior
|
- !type:DoActsBehavior
|
||||||
acts: [ "Destruction" ]
|
acts: [ "Destruction" ]
|
||||||
|
- type: Appearance
|
||||||
|
visuals:
|
||||||
|
- type: LightBulbVisualizer
|
||||||
|
|
||||||
# Lighting color values gathered from
|
# Lighting color values gathered from
|
||||||
# https://andi-siess.de/rgb-to-color-temperature/
|
# https://andi-siess.de/rgb-to-color-temperature/
|
||||||
|
|||||||
Reference in New Issue
Block a user