dynamic alert sprites (#25452)

* dynamic alert sprite

* fix fat cooldowns
This commit is contained in:
Nemanja
2024-03-28 02:32:56 -04:00
committed by GitHub
parent 6863a7cc26
commit d576f5cbbb
38 changed files with 154 additions and 84 deletions

View 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;
}
}

View File

@@ -1,3 +1,5 @@
using Content.Client.Alerts;
using Content.Shared.Alert;
using Content.Shared.Revenant; using Content.Shared.Revenant;
using Content.Shared.Revenant.Components; using Content.Shared.Revenant.Components;
using Robust.Client.GameObjects; using Robust.Client.GameObjects;
@@ -13,6 +15,7 @@ public sealed class RevenantSystem : EntitySystem
base.Initialize(); base.Initialize();
SubscribeLocalEvent<RevenantComponent, AppearanceChangeEvent>(OnAppearanceChange); SubscribeLocalEvent<RevenantComponent, AppearanceChangeEvent>(OnAppearanceChange);
SubscribeLocalEvent<RevenantComponent, UpdateAlertSpriteEvent>(OnUpdateAlert);
} }
private void OnAppearanceChange(EntityUid uid, RevenantComponent component, ref AppearanceChangeEvent args) 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); 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}");
}
} }

View File

@@ -3,6 +3,8 @@ using Content.Client.Gameplay;
using Content.Client.UserInterface.Systems.Alerts.Widgets; using Content.Client.UserInterface.Systems.Alerts.Widgets;
using Content.Client.UserInterface.Systems.Gameplay; using Content.Client.UserInterface.Systems.Gameplay;
using Content.Shared.Alert; using Content.Shared.Alert;
using Robust.Client.GameObjects;
using Robust.Client.Player;
using Robust.Client.UserInterface; using Robust.Client.UserInterface;
using Robust.Client.UserInterface.Controllers; using Robust.Client.UserInterface.Controllers;
@@ -10,6 +12,8 @@ namespace Content.Client.UserInterface.Systems.Alerts;
public sealed class AlertsUIController : UIController, IOnStateEntered<GameplayState>, IOnSystemChanged<ClientAlertsSystem> public sealed class AlertsUIController : UIController, IOnStateEntered<GameplayState>, IOnSystemChanged<ClientAlertsSystem>
{ {
[Dependency] private readonly IPlayerManager _player = default!;
[UISystemDependency] private readonly ClientAlertsSystem? _alertsSystem = default; [UISystemDependency] private readonly ClientAlertsSystem? _alertsSystem = default;
private AlertsUI? UI => UIManager.GetActiveUIWidgetOrNull<AlertsUI>(); private AlertsUI? UI => UIManager.GetActiveUIWidgetOrNull<AlertsUI>();
@@ -84,4 +88,16 @@ public sealed class AlertsUIController : UIController, IOnStateEntered<GameplayS
SystemOnSyncAlerts(_alertsSystem, alerts); 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);
}
} }

View File

@@ -2,6 +2,8 @@
using Content.Client.Actions.UI; using Content.Client.Actions.UI;
using Content.Client.Cooldown; using Content.Client.Cooldown;
using Content.Shared.Alert; using Content.Shared.Alert;
using Robust.Client.GameObjects;
using Robust.Client.Graphics;
using Robust.Client.UserInterface; using Robust.Client.UserInterface;
using Robust.Client.UserInterface.Controls; using Robust.Client.UserInterface.Controls;
using Robust.Shared.Timing; using Robust.Shared.Timing;
@@ -33,9 +35,12 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
private short? _severity; private short? _severity;
private readonly IGameTiming _gameTiming; private readonly IGameTiming _gameTiming;
private readonly AnimatedTextureRect _icon; private readonly IEntityManager _entityManager;
private readonly SpriteView _icon;
private readonly CooldownGraphic _cooldownGraphic; private readonly CooldownGraphic _cooldownGraphic;
private EntityUid _spriteViewEntity;
/// <summary> /// <summary>
/// Creates an alert control reflecting the indicated alert + state /// Creates an alert control reflecting the indicated alert + state
/// </summary> /// </summary>
@@ -44,19 +49,30 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
public AlertControl(AlertPrototype alert, short? severity) public AlertControl(AlertPrototype alert, short? severity)
{ {
_gameTiming = IoCManager.Resolve<IGameTiming>(); _gameTiming = IoCManager.Resolve<IGameTiming>();
_entityManager = IoCManager.Resolve<IEntityManager>();
TooltipSupplier = SupplyTooltip; TooltipSupplier = SupplyTooltip;
Alert = alert; Alert = alert;
_severity = severity; _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); Children.Add(_icon);
_cooldownGraphic = new CooldownGraphic(); _cooldownGraphic = new CooldownGraphic
{
MaxSize = new Vector2(64, 64)
};
Children.Add(_cooldownGraphic); Children.Add(_cooldownGraphic);
} }
@@ -72,16 +88,22 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
/// </summary> /// </summary>
public void SetSeverity(short? severity) public void SetSeverity(short? severity)
{ {
if (_severity != severity) if (_severity == severity)
{ return;
_severity = severity; _severity = severity;
_icon.SetFromSpriteSpecifier(Alert.GetIcon(_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) protected override void FrameUpdate(FrameEventArgs args)
{ {
base.FrameUpdate(args); base.FrameUpdate(args);
UserInterfaceManager.GetUIController<AlertsUIController>().UpdateAlertSpriteEntity(_spriteViewEntity, Alert);
if (!Cooldown.HasValue) if (!Cooldown.HasValue)
{ {
_cooldownGraphic.Visible = false; _cooldownGraphic.Visible = false;
@@ -91,5 +113,17 @@ namespace Content.Client.UserInterface.Systems.Alerts.Controls
_cooldownGraphic.FromTime(Cooldown.Value.Start, Cooldown.Value.End); _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
} }
} }

View File

@@ -133,6 +133,7 @@ public sealed partial class RevenantSystem : EntitySystem
return false; return false;
component.Essence += amount; component.Essence += amount;
Dirty(uid, component);
if (regenCap) if (regenCap)
FixedPoint2.Min(component.Essence, component.EssenceRegenCap); FixedPoint2.Min(component.Essence, component.EssenceRegenCap);
@@ -140,7 +141,7 @@ public sealed partial class RevenantSystem : EntitySystem
if (TryComp<StoreComponent>(uid, out var store)) if (TryComp<StoreComponent>(uid, out var store))
_store.UpdateUserInterface(uid, uid, 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) if (component.Essence <= 0)
{ {

View File

@@ -25,6 +25,12 @@ namespace Content.Shared.Alert
[DataField("icons", required: true)] [DataField("icons", required: true)]
public List<SpriteSpecifier> Icons = new(); 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> /// <summary>
/// Name to show in tooltip window. Accepts formatting. /// Name to show in tooltip window. Accepts formatting.
/// </summary> /// </summary>
@@ -83,13 +89,9 @@ namespace Content.Shared.Alert
/// <returns>the icon path to the texture for the provided severity level</returns> /// <returns>the icon path to the texture for the provided severity level</returns>
public SpriteSpecifier GetIcon(short? severity = null) public SpriteSpecifier GetIcon(short? severity = null)
{ {
if (!SupportsSeverity && severity != null)
{
throw new InvalidOperationException($"This alert ({AlertKey}) does not support severity");
}
var minIcons = SupportsSeverity var minIcons = SupportsSeverity
? MaxSeverity - MinSeverity : 1; ? MaxSeverity - MinSeverity
: 1;
if (Icons.Count < minIcons) if (Icons.Count < minIcons)
throw new InvalidOperationException($"Insufficient number of icons given for alert {AlertType}"); throw new InvalidOperationException($"Insufficient number of icons given for alert {AlertType}");

View File

@@ -9,13 +9,15 @@ using Robust.Shared.Serialization.TypeSerializers.Implementations.Custom.Prototy
namespace Content.Shared.Revenant.Components; namespace Content.Shared.Revenant.Components;
[RegisterComponent, NetworkedComponent] [RegisterComponent, NetworkedComponent]
[AutoGenerateComponentState]
public sealed partial class RevenantComponent : Component public sealed partial class RevenantComponent : Component
{ {
/// <summary> /// <summary>
/// The total amount of Essence the revenant has. Functions /// The total amount of Essence the revenant has. Functions
/// as health and is regenerated. /// as health and is regenerated.
/// </summary> /// </summary>
[ViewVariables(VVAccess.ReadWrite)] [DataField, ViewVariables(VVAccess.ReadWrite)]
[AutoNetworkedField]
public FixedPoint2 Essence = 75; public FixedPoint2 Essence = 75;
[DataField("stolenEssenceCurrencyPrototype", customTypeSerializer: typeof(PrototypeIdSerializer<CurrencyPrototype>))] [DataField("stolenEssenceCurrencyPrototype", customTypeSerializer: typeof(PrototypeIdSerializer<CurrencyPrototype>))]

View File

@@ -70,3 +70,11 @@ public enum RevenantVisuals : byte
Stunned, Stunned,
Harvesting, Harvesting,
} }
[NetSerializable, Serializable]
public enum RevenantVisualLayers : byte
{
Digit1,
Digit2,
Digit3
}

View File

@@ -25,6 +25,14 @@
- alertType: Magboots - alertType: Magboots
- alertType: Pacified - alertType: Pacified
- type: entity
id: AlertSpriteView
categories: [ hideSpawnMenu ]
components:
- type: Sprite
layers:
- map: [ "enum.AlertVisualLayers.Base" ]
- type: alert - type: alert
id: LowOxygen id: LowOxygen
category: Breathing category: Breathing

View File

@@ -4,45 +4,26 @@
icons: icons:
- sprite: /Textures/Interface/Alerts/essence_counter.rsi - sprite: /Textures/Interface/Alerts/essence_counter.rsi
state: essence0 state: essence0
- sprite: /Textures/Interface/Alerts/essence_counter.rsi alertViewEntity: AlertEssenceSpriteView
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
name: alerts-revenant-essence-name name: alerts-revenant-essence-name
description: alerts-revenant-essence-desc description: alerts-revenant-essence-desc
minSeverity: 0
maxSeverity: 16
- type: alert - type: alert
id: Corporeal id: Corporeal
icons: [ /Textures/Mobs/Ghosts/revenant.rsi/icon.png ] icons: [ /Textures/Mobs/Ghosts/revenant.rsi/icon.png ]
name: alerts-revenant-corporeal-name 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

Binary file not shown.

After

Width:  |  Height:  |  Size: 155 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 150 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 164 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 156 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 156 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 168 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 167 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 154 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 156 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 164 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 319 B

After

Width:  |  Height:  |  Size: 255 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 298 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 308 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 304 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 311 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 311 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 321 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 307 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 313 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 297 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 299 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 321 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 302 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 297 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 317 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 295 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 292 B

View File

@@ -11,52 +11,34 @@
"name": "essence0" "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": "9"
},
{
"name": "essence11"
},
{
"name": "essence12"
},
{
"name": "essence13"
},
{
"name": "essence14"
},
{
"name": "essence15"
},
{
"name": "essence16"
} }
] ]
} }