dynamic alert sprites (#25452)
* dynamic alert sprite * fix fat cooldowns
21
Content.Client/Alerts/UpdateAlertSpriteEvent.cs
Normal file
@@ -0,0 +1,21 @@
|
||||
using Content.Shared.Alert;
|
||||
using Robust.Client.GameObjects;
|
||||
|
||||
namespace Content.Client.Alerts;
|
||||
|
||||
/// <summary>
|
||||
/// Event raised on an entity with alerts in order to allow it to update visuals for the alert sprite entity.
|
||||
/// </summary>
|
||||
[ByRefEvent]
|
||||
public record struct UpdateAlertSpriteEvent
|
||||
{
|
||||
public Entity<SpriteComponent> SpriteViewEnt;
|
||||
|
||||
public AlertPrototype Alert;
|
||||
|
||||
public UpdateAlertSpriteEvent(Entity<SpriteComponent> spriteViewEnt, AlertPrototype alert)
|
||||
{
|
||||
SpriteViewEnt = spriteViewEnt;
|
||||
Alert = alert;
|
||||
}
|
||||
}
|
||||
@@ -1,3 +1,5 @@
|
||||
using Content.Client.Alerts;
|
||||
using Content.Shared.Alert;
|
||||
using Content.Shared.Revenant;
|
||||
using Content.Shared.Revenant.Components;
|
||||
using Robust.Client.GameObjects;
|
||||
@@ -13,6 +15,7 @@ public sealed class RevenantSystem : EntitySystem
|
||||
base.Initialize();
|
||||
|
||||
SubscribeLocalEvent<RevenantComponent, AppearanceChangeEvent>(OnAppearanceChange);
|
||||
SubscribeLocalEvent<RevenantComponent, UpdateAlertSpriteEvent>(OnUpdateAlert);
|
||||
}
|
||||
|
||||
private void OnAppearanceChange(EntityUid uid, RevenantComponent component, ref AppearanceChangeEvent args)
|
||||
@@ -36,4 +39,16 @@ public sealed class RevenantSystem : EntitySystem
|
||||
args.Sprite.LayerSetState(0, component.State);
|
||||
}
|
||||
}
|
||||
|
||||
private void OnUpdateAlert(Entity<RevenantComponent> ent, ref UpdateAlertSpriteEvent args)
|
||||
{
|
||||
if (args.Alert.AlertType != AlertType.Essence)
|
||||
return;
|
||||
|
||||
var sprite = args.SpriteViewEnt.Comp;
|
||||
var essence = Math.Clamp(ent.Comp.Essence.Int(), 0, 999);
|
||||
sprite.LayerSetState(RevenantVisualLayers.Digit1, $"{(essence / 100) % 10}");
|
||||
sprite.LayerSetState(RevenantVisualLayers.Digit2, $"{(essence / 10) % 10}");
|
||||
sprite.LayerSetState(RevenantVisualLayers.Digit3, $"{essence % 10}");
|
||||
}
|
||||
}
|
||||
|
||||
@@ -3,6 +3,8 @@ using Content.Client.Gameplay;
|
||||
using Content.Client.UserInterface.Systems.Alerts.Widgets;
|
||||
using Content.Client.UserInterface.Systems.Gameplay;
|
||||
using Content.Shared.Alert;
|
||||
using Robust.Client.GameObjects;
|
||||
using Robust.Client.Player;
|
||||
using Robust.Client.UserInterface;
|
||||
using Robust.Client.UserInterface.Controllers;
|
||||
|
||||
@@ -10,6 +12,8 @@ namespace Content.Client.UserInterface.Systems.Alerts;
|
||||
|
||||
public sealed class AlertsUIController : UIController, IOnStateEntered<GameplayState>, IOnSystemChanged<ClientAlertsSystem>
|
||||
{
|
||||
[Dependency] private readonly IPlayerManager _player = default!;
|
||||
|
||||
[UISystemDependency] private readonly ClientAlertsSystem? _alertsSystem = default;
|
||||
|
||||
private AlertsUI? UI => UIManager.GetActiveUIWidgetOrNull<AlertsUI>();
|
||||
@@ -84,4 +88,16 @@ public sealed class AlertsUIController : UIController, IOnStateEntered<GameplayS
|
||||
SystemOnSyncAlerts(_alertsSystem, alerts);
|
||||
}
|
||||
}
|
||||
|
||||
public void UpdateAlertSpriteEntity(EntityUid spriteViewEnt, AlertPrototype alert)
|
||||
{
|
||||
if (_player.LocalEntity is not { } player)
|
||||
return;
|
||||
|
||||
if (!EntityManager.TryGetComponent<SpriteComponent>(spriteViewEnt, out var sprite))
|
||||
return;
|
||||
|
||||
var ev = new UpdateAlertSpriteEvent((spriteViewEnt, sprite), alert);
|
||||
EntityManager.EventBus.RaiseLocalEvent(player, ref ev);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2,6 +2,8 @@
|
||||
using Content.Client.Actions.UI;
|
||||
using Content.Client.Cooldown;
|
||||
using Content.Shared.Alert;
|
||||
using Robust.Client.GameObjects;
|
||||
using Robust.Client.Graphics;
|
||||
using Robust.Client.UserInterface;
|
||||
using Robust.Client.UserInterface.Controls;
|
||||
using Robust.Shared.Timing;
|
||||
@@ -33,9 +35,12 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
|
||||
|
||||
private short? _severity;
|
||||
private readonly IGameTiming _gameTiming;
|
||||
private readonly AnimatedTextureRect _icon;
|
||||
private readonly IEntityManager _entityManager;
|
||||
private readonly SpriteView _icon;
|
||||
private readonly CooldownGraphic _cooldownGraphic;
|
||||
|
||||
private EntityUid _spriteViewEntity;
|
||||
|
||||
/// <summary>
|
||||
/// Creates an alert control reflecting the indicated alert + state
|
||||
/// </summary>
|
||||
@@ -44,19 +49,30 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
|
||||
public AlertControl(AlertPrototype alert, short? severity)
|
||||
{
|
||||
_gameTiming = IoCManager.Resolve<IGameTiming>();
|
||||
_entityManager = IoCManager.Resolve<IEntityManager>();
|
||||
TooltipSupplier = SupplyTooltip;
|
||||
Alert = alert;
|
||||
_severity = severity;
|
||||
var specifier = alert.GetIcon(_severity);
|
||||
_icon = new AnimatedTextureRect
|
||||
{
|
||||
DisplayRect = {TextureScale = new Vector2(2, 2)}
|
||||
};
|
||||
|
||||
_icon.SetFromSpriteSpecifier(specifier);
|
||||
_spriteViewEntity = _entityManager.Spawn(Alert.AlertViewEntity);
|
||||
if (_entityManager.TryGetComponent<SpriteComponent>(_spriteViewEntity, out var sprite))
|
||||
{
|
||||
var icon = Alert.GetIcon(_severity);
|
||||
if (sprite.LayerMapTryGet(AlertVisualLayers.Base, out var layer))
|
||||
sprite.LayerSetSprite(layer, icon);
|
||||
}
|
||||
|
||||
_icon = new SpriteView
|
||||
{
|
||||
Scale = new Vector2(2, 2)
|
||||
};
|
||||
_icon.SetEntity(_spriteViewEntity);
|
||||
|
||||
Children.Add(_icon);
|
||||
_cooldownGraphic = new CooldownGraphic();
|
||||
_cooldownGraphic = new CooldownGraphic
|
||||
{
|
||||
MaxSize = new Vector2(64, 64)
|
||||
};
|
||||
Children.Add(_cooldownGraphic);
|
||||
}
|
||||
|
||||
@@ -72,16 +88,22 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
|
||||
/// </summary>
|
||||
public void SetSeverity(short? severity)
|
||||
{
|
||||
if (_severity != severity)
|
||||
{
|
||||
_severity = severity;
|
||||
_icon.SetFromSpriteSpecifier(Alert.GetIcon(_severity));
|
||||
}
|
||||
if (_severity == severity)
|
||||
return;
|
||||
_severity = severity;
|
||||
|
||||
if (!_entityManager.TryGetComponent<SpriteComponent>(_spriteViewEntity, out var sprite))
|
||||
return;
|
||||
var icon = Alert.GetIcon(_severity);
|
||||
if (sprite.LayerMapTryGet(AlertVisualLayers.Base, out var layer))
|
||||
sprite.LayerSetSprite(layer, icon);
|
||||
}
|
||||
|
||||
protected override void FrameUpdate(FrameEventArgs args)
|
||||
{
|
||||
base.FrameUpdate(args);
|
||||
UserInterfaceManager.GetUIController<AlertsUIController>().UpdateAlertSpriteEntity(_spriteViewEntity, Alert);
|
||||
|
||||
if (!Cooldown.HasValue)
|
||||
{
|
||||
_cooldownGraphic.Visible = false;
|
||||
@@ -91,5 +113,17 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
|
||||
|
||||
_cooldownGraphic.FromTime(Cooldown.Value.Start, Cooldown.Value.End);
|
||||
}
|
||||
|
||||
protected override void Dispose(bool disposing)
|
||||
{
|
||||
base.Dispose(disposing);
|
||||
|
||||
_entityManager.DeleteEntity(_spriteViewEntity);
|
||||
}
|
||||
}
|
||||
|
||||
public enum AlertVisualLayers : byte
|
||||
{
|
||||
Base
|
||||
}
|
||||
}
|
||||
|
||||
@@ -133,6 +133,7 @@ public sealed partial class RevenantSystem : EntitySystem
|
||||
return false;
|
||||
|
||||
component.Essence += amount;
|
||||
Dirty(uid, component);
|
||||
|
||||
if (regenCap)
|
||||
FixedPoint2.Min(component.Essence, component.EssenceRegenCap);
|
||||
@@ -140,7 +141,7 @@ public sealed partial class RevenantSystem : EntitySystem
|
||||
if (TryComp<StoreComponent>(uid, out var store))
|
||||
_store.UpdateUserInterface(uid, uid, store);
|
||||
|
||||
_alerts.ShowAlert(uid, AlertType.Essence, (short) Math.Clamp(Math.Round(component.Essence.Float() / 10f), 0, 16));
|
||||
_alerts.ShowAlert(uid, AlertType.Essence);
|
||||
|
||||
if (component.Essence <= 0)
|
||||
{
|
||||
|
||||
@@ -25,6 +25,12 @@ namespace Content.Shared.Alert
|
||||
[DataField("icons", required: true)]
|
||||
public List<SpriteSpecifier> Icons = new();
|
||||
|
||||
/// <summary>
|
||||
/// An entity used for displaying the <see cref="Icons"/> in the UI control.
|
||||
/// </summary>
|
||||
[DataField]
|
||||
public EntProtoId AlertViewEntity = "AlertSpriteView";
|
||||
|
||||
/// <summary>
|
||||
/// Name to show in tooltip window. Accepts formatting.
|
||||
/// </summary>
|
||||
@@ -83,13 +89,9 @@ namespace Content.Shared.Alert
|
||||
/// <returns>the icon path to the texture for the provided severity level</returns>
|
||||
public SpriteSpecifier GetIcon(short? severity = null)
|
||||
{
|
||||
if (!SupportsSeverity && severity != null)
|
||||
{
|
||||
throw new InvalidOperationException($"This alert ({AlertKey}) does not support severity");
|
||||
}
|
||||
|
||||
var minIcons = SupportsSeverity
|
||||
? MaxSeverity - MinSeverity : 1;
|
||||
? MaxSeverity - MinSeverity
|
||||
: 1;
|
||||
|
||||
if (Icons.Count < minIcons)
|
||||
throw new InvalidOperationException($"Insufficient number of icons given for alert {AlertType}");
|
||||
|
||||
@@ -9,13 +9,15 @@ using Robust.Shared.Serialization.TypeSerializers.Implementations.Custom.Prototy
|
||||
namespace Content.Shared.Revenant.Components;
|
||||
|
||||
[RegisterComponent, NetworkedComponent]
|
||||
[AutoGenerateComponentState]
|
||||
public sealed partial class RevenantComponent : Component
|
||||
{
|
||||
/// <summary>
|
||||
/// The total amount of Essence the revenant has. Functions
|
||||
/// as health and is regenerated.
|
||||
/// </summary>
|
||||
[ViewVariables(VVAccess.ReadWrite)]
|
||||
[DataField, ViewVariables(VVAccess.ReadWrite)]
|
||||
[AutoNetworkedField]
|
||||
public FixedPoint2 Essence = 75;
|
||||
|
||||
[DataField("stolenEssenceCurrencyPrototype", customTypeSerializer: typeof(PrototypeIdSerializer<CurrencyPrototype>))]
|
||||
|
||||
@@ -70,3 +70,11 @@ public enum RevenantVisuals : byte
|
||||
Stunned,
|
||||
Harvesting,
|
||||
}
|
||||
|
||||
[NetSerializable, Serializable]
|
||||
public enum RevenantVisualLayers : byte
|
||||
{
|
||||
Digit1,
|
||||
Digit2,
|
||||
Digit3
|
||||
}
|
||||
|
||||
@@ -25,6 +25,14 @@
|
||||
- alertType: Magboots
|
||||
- alertType: Pacified
|
||||
|
||||
- type: entity
|
||||
id: AlertSpriteView
|
||||
categories: [ hideSpawnMenu ]
|
||||
components:
|
||||
- type: Sprite
|
||||
layers:
|
||||
- map: [ "enum.AlertVisualLayers.Base" ]
|
||||
|
||||
- type: alert
|
||||
id: LowOxygen
|
||||
category: Breathing
|
||||
|
||||
@@ -4,45 +4,26 @@
|
||||
icons:
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence0
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence1
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence2
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence3
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence4
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence5
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence6
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence7
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence8
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence9
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence10
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence11
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence12
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence13
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence14
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence15
|
||||
- sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
state: essence16
|
||||
alertViewEntity: AlertEssenceSpriteView
|
||||
name: alerts-revenant-essence-name
|
||||
description: alerts-revenant-essence-desc
|
||||
minSeverity: 0
|
||||
maxSeverity: 16
|
||||
|
||||
- type: alert
|
||||
id: Corporeal
|
||||
icons: [ /Textures/Mobs/Ghosts/revenant.rsi/icon.png ]
|
||||
name: alerts-revenant-corporeal-name
|
||||
description: alerts-revenant-corporeal-desc
|
||||
description: alerts-revenant-corporeal-desc
|
||||
|
||||
- type: entity
|
||||
id: AlertEssenceSpriteView
|
||||
categories: [ hideSpawnMenu ]
|
||||
components:
|
||||
- type: Sprite
|
||||
sprite: /Textures/Interface/Alerts/essence_counter.rsi
|
||||
layers:
|
||||
- map: [ "enum.AlertVisualLayers.Base" ]
|
||||
- map: [ "enum.RevenantVisualLayers.Digit1" ]
|
||||
- map: [ "enum.RevenantVisualLayers.Digit2" ]
|
||||
offset: 0.125, 0
|
||||
- map: [ "enum.RevenantVisualLayers.Digit3" ]
|
||||
offset: 0.25, 0
|
||||
|
||||
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/0.png
Normal file
|
After Width: | Height: | Size: 155 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/1.png
Normal file
|
After Width: | Height: | Size: 150 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/2.png
Normal file
|
After Width: | Height: | Size: 164 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/3.png
Normal file
|
After Width: | Height: | Size: 156 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/4.png
Normal file
|
After Width: | Height: | Size: 156 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/5.png
Normal file
|
After Width: | Height: | Size: 168 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/6.png
Normal file
|
After Width: | Height: | Size: 167 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/7.png
Normal file
|
After Width: | Height: | Size: 154 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/8.png
Normal file
|
After Width: | Height: | Size: 156 B |
BIN
Resources/Textures/Interface/Alerts/essence_counter.rsi/9.png
Normal file
|
After Width: | Height: | Size: 164 B |
|
Before Width: | Height: | Size: 319 B After Width: | Height: | Size: 255 B |
|
Before Width: | Height: | Size: 298 B |
|
Before Width: | Height: | Size: 308 B |
|
Before Width: | Height: | Size: 304 B |
|
Before Width: | Height: | Size: 311 B |
|
Before Width: | Height: | Size: 311 B |
|
Before Width: | Height: | Size: 321 B |
|
Before Width: | Height: | Size: 307 B |
|
Before Width: | Height: | Size: 313 B |
|
Before Width: | Height: | Size: 297 B |
|
Before Width: | Height: | Size: 299 B |
|
Before Width: | Height: | Size: 321 B |
|
Before Width: | Height: | Size: 302 B |
|
Before Width: | Height: | Size: 297 B |
|
Before Width: | Height: | Size: 317 B |
|
Before Width: | Height: | Size: 295 B |
|
Before Width: | Height: | Size: 292 B |
@@ -11,52 +11,34 @@
|
||||
"name": "essence0"
|
||||
},
|
||||
{
|
||||
"name": "essence1"
|
||||
"name": "0"
|
||||
},
|
||||
{
|
||||
"name": "essence2"
|
||||
"name": "1"
|
||||
},
|
||||
{
|
||||
"name": "essence3"
|
||||
"name": "2"
|
||||
},
|
||||
{
|
||||
"name": "essence4"
|
||||
"name": "3"
|
||||
},
|
||||
{
|
||||
"name": "essence5"
|
||||
"name": "4"
|
||||
},
|
||||
{
|
||||
"name": "essence6"
|
||||
"name": "5"
|
||||
},
|
||||
{
|
||||
"name": "essence7"
|
||||
"name": "6"
|
||||
},
|
||||
{
|
||||
"name": "essence8"
|
||||
"name": "7"
|
||||
},
|
||||
{
|
||||
"name": "essence9"
|
||||
"name": "8"
|
||||
},
|
||||
{
|
||||
"name": "essence10"
|
||||
},
|
||||
{
|
||||
"name": "essence11"
|
||||
},
|
||||
{
|
||||
"name": "essence12"
|
||||
},
|
||||
{
|
||||
"name": "essence13"
|
||||
},
|
||||
{
|
||||
"name": "essence14"
|
||||
},
|
||||
{
|
||||
"name": "essence15"
|
||||
},
|
||||
{
|
||||
"name": "essence16"
|
||||
"name": "9"
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||