ECS Buckle and Rotation visualizers (#13024)
Co-authored-by: DrSmugleaf <DrSmugleaf@users.noreply.github.com>
This commit is contained in:
@@ -1,68 +0,0 @@
|
|||||||
using System;
|
|
||||||
using Content.Shared.Buckle.Components;
|
|
||||||
using JetBrains.Annotations;
|
|
||||||
using Robust.Client.Animations;
|
|
||||||
using Robust.Client.GameObjects;
|
|
||||||
using Robust.Shared.Animations;
|
|
||||||
using Robust.Shared.GameObjects;
|
|
||||||
using Robust.Shared.IoC;
|
|
||||||
using Robust.Shared.Maths;
|
|
||||||
|
|
||||||
namespace Content.Client.Buckle
|
|
||||||
{
|
|
||||||
[UsedImplicitly]
|
|
||||||
public sealed class BuckleVisualizer : AppearanceVisualizer
|
|
||||||
{
|
|
||||||
[Obsolete("Subscribe to AppearanceChangeEvent instead.")]
|
|
||||||
public override void OnChangeData(AppearanceComponent component)
|
|
||||||
{
|
|
||||||
if (!component.TryGetData<bool>(BuckleVisuals.Buckled, out var buckled) ||
|
|
||||||
!buckled)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!component.TryGetData<int>(StrapVisuals.RotationAngle, out var angle))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
SetRotation(component, Angle.FromDegrees(angle));
|
|
||||||
}
|
|
||||||
|
|
||||||
private void SetRotation(AppearanceComponent component, Angle rotation)
|
|
||||||
{
|
|
||||||
var sprite = IoCManager.Resolve<IEntityManager>().GetComponent<ISpriteComponent>(component.Owner);
|
|
||||||
|
|
||||||
if (!IoCManager.Resolve<IEntityManager>().TryGetComponent(sprite.Owner, out AnimationPlayerComponent? animation))
|
|
||||||
{
|
|
||||||
sprite.Rotation = rotation;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (animation.HasRunningAnimation("rotate"))
|
|
||||||
{
|
|
||||||
animation.Stop("rotate");
|
|
||||||
}
|
|
||||||
|
|
||||||
animation.Play(new Animation
|
|
||||||
{
|
|
||||||
Length = TimeSpan.FromSeconds(0.125),
|
|
||||||
AnimationTracks =
|
|
||||||
{
|
|
||||||
new AnimationTrackComponentProperty
|
|
||||||
{
|
|
||||||
ComponentType = typeof(ISpriteComponent),
|
|
||||||
Property = nameof(ISpriteComponent.Rotation),
|
|
||||||
InterpolationMode = AnimationInterpolationMode.Linear,
|
|
||||||
KeyFrames =
|
|
||||||
{
|
|
||||||
new AnimationTrackProperty.KeyFrame(sprite.Rotation, 0),
|
|
||||||
new AnimationTrackProperty.KeyFrame(rotation, 0.125f)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, "rotate");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
23
Content.Client/Buckle/BuckleVisualizerSystem.cs
Normal file
23
Content.Client/Buckle/BuckleVisualizerSystem.cs
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
using Content.Shared.Buckle.Components;
|
||||||
|
using Content.Client.Rotation;
|
||||||
|
using Robust.Client.GameObjects;
|
||||||
|
|
||||||
|
namespace Content.Client.Buckle;
|
||||||
|
|
||||||
|
public sealed class BuckleVisualizer : VisualizerSystem<BuckleComponent>
|
||||||
|
{
|
||||||
|
protected override void OnAppearanceChange(EntityUid uid, BuckleComponent component, ref AppearanceChangeEvent args)
|
||||||
|
{
|
||||||
|
if (!args.Component.TryGetData<int>(StrapVisuals.RotationAngle, out var angle) ||
|
||||||
|
!args.Component.TryGetData<bool>(BuckleVisuals.Buckled, out var buckled) ||
|
||||||
|
!buckled ||
|
||||||
|
args.Sprite == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
EntityManager.System<RotationVisualizerSystem>()
|
||||||
|
.AnimateSpriteRotation(args.Sprite, Angle.FromDegrees(angle), 0.125f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@@ -1,76 +0,0 @@
|
|||||||
using System;
|
|
||||||
using Content.Shared.Rotation;
|
|
||||||
using JetBrains.Annotations;
|
|
||||||
using Robust.Client.Animations;
|
|
||||||
using Robust.Client.GameObjects;
|
|
||||||
using Robust.Shared.Animations;
|
|
||||||
using Robust.Shared.GameObjects;
|
|
||||||
using Robust.Shared.IoC;
|
|
||||||
using Robust.Shared.Maths;
|
|
||||||
|
|
||||||
namespace Content.Client.Rotation
|
|
||||||
{
|
|
||||||
[UsedImplicitly]
|
|
||||||
public sealed class RotationVisualizer : AppearanceVisualizer
|
|
||||||
{
|
|
||||||
[Obsolete("Subscribe to AppearanceChangeEvent instead.")]
|
|
||||||
public override void OnChangeData(AppearanceComponent component)
|
|
||||||
{
|
|
||||||
base.OnChangeData(component);
|
|
||||||
|
|
||||||
// if TryGet fails, state defaults to RotationState.Vertical.
|
|
||||||
component.TryGetData<RotationState>(RotationVisuals.RotationState, out var state);
|
|
||||||
|
|
||||||
switch (state)
|
|
||||||
{
|
|
||||||
case RotationState.Vertical:
|
|
||||||
SetRotation(component, 0);
|
|
||||||
break;
|
|
||||||
case RotationState.Horizontal:
|
|
||||||
SetRotation(component, Angle.FromDegrees(90));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void SetRotation(AppearanceComponent component, Angle rotation)
|
|
||||||
{
|
|
||||||
var entMan = IoCManager.Resolve<IEntityManager>();
|
|
||||||
var sprite = entMan.GetComponent<ISpriteComponent>(component.Owner);
|
|
||||||
|
|
||||||
if (sprite.Rotation.Equals(rotation))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!entMan.TryGetComponent(sprite.Owner, out AnimationPlayerComponent? animation))
|
|
||||||
{
|
|
||||||
sprite.Rotation = rotation;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (animation.HasRunningAnimation("rotate"))
|
|
||||||
{
|
|
||||||
animation.Stop("rotate");
|
|
||||||
}
|
|
||||||
|
|
||||||
animation.Play(new Animation
|
|
||||||
{
|
|
||||||
Length = TimeSpan.FromSeconds(0.125),
|
|
||||||
AnimationTracks =
|
|
||||||
{
|
|
||||||
new AnimationTrackComponentProperty
|
|
||||||
{
|
|
||||||
ComponentType = typeof(ISpriteComponent),
|
|
||||||
Property = nameof(ISpriteComponent.Rotation),
|
|
||||||
InterpolationMode = AnimationInterpolationMode.Linear,
|
|
||||||
KeyFrames =
|
|
||||||
{
|
|
||||||
new AnimationTrackProperty.KeyFrame(sprite.Rotation, 0),
|
|
||||||
new AnimationTrackProperty.KeyFrame(rotation, 0.125f)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, "rotate");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
73
Content.Client/Rotation/RotationVisualizerSystem.cs
Normal file
73
Content.Client/Rotation/RotationVisualizerSystem.cs
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
using Robust.Client.GameObjects;
|
||||||
|
using Robust.Client.Animations;
|
||||||
|
using Content.Shared.Rotation;
|
||||||
|
using Robust.Shared.Animations;
|
||||||
|
|
||||||
|
namespace Content.Client.Rotation;
|
||||||
|
|
||||||
|
public sealed class RotationVisualizerSystem : VisualizerSystem<RotationVisualsComponent>
|
||||||
|
{
|
||||||
|
protected override void OnAppearanceChange(EntityUid uid, RotationVisualsComponent component, ref AppearanceChangeEvent args)
|
||||||
|
{
|
||||||
|
base.OnAppearanceChange(uid, component, ref args);
|
||||||
|
|
||||||
|
if (args.Sprite != null)
|
||||||
|
{
|
||||||
|
// if TryGetData fails, state defaults to RotationState.Vertical.
|
||||||
|
args.Component.TryGetData<RotationState>(RotationVisuals.RotationState, out var state);
|
||||||
|
|
||||||
|
switch (state)
|
||||||
|
{
|
||||||
|
case RotationState.Vertical:
|
||||||
|
AnimateSpriteRotation(args.Sprite, component.VerticalRotation, component.AnimationTime);
|
||||||
|
break;
|
||||||
|
case RotationState.Horizontal:
|
||||||
|
AnimateSpriteRotation(args.Sprite, component.HorizontalRotation, component.AnimationTime);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Rotates a sprite between two animated keyframes given a certain time.
|
||||||
|
/// </summary>
|
||||||
|
public void AnimateSpriteRotation(SpriteComponent sprite, Angle rotation, float animationTime)
|
||||||
|
{
|
||||||
|
var entMan = IoCManager.Resolve<IEntityManager>();
|
||||||
|
|
||||||
|
if (sprite.Rotation.Equals(rotation))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!entMan.TryGetComponent(sprite.Owner, out AnimationPlayerComponent? animation))
|
||||||
|
{
|
||||||
|
sprite.Rotation = rotation;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (animation.HasRunningAnimation("rotate"))
|
||||||
|
{
|
||||||
|
animation.Stop("rotate");
|
||||||
|
}
|
||||||
|
|
||||||
|
animation.Play(new Animation
|
||||||
|
{
|
||||||
|
Length = TimeSpan.FromSeconds(animationTime),
|
||||||
|
AnimationTracks =
|
||||||
|
{
|
||||||
|
new AnimationTrackComponentProperty
|
||||||
|
{
|
||||||
|
ComponentType = typeof(ISpriteComponent),
|
||||||
|
Property = nameof(ISpriteComponent.Rotation),
|
||||||
|
InterpolationMode = AnimationInterpolationMode.Linear,
|
||||||
|
KeyFrames =
|
||||||
|
{
|
||||||
|
new AnimationTrackProperty.KeyFrame(sprite.Rotation, 0),
|
||||||
|
new AnimationTrackProperty.KeyFrame(rotation, animationTime)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, "rotate");
|
||||||
|
}
|
||||||
|
}
|
||||||
14
Content.Client/Rotation/RotationVisualsComponent.cs
Normal file
14
Content.Client/Rotation/RotationVisualsComponent.cs
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
namespace Content.Client.Rotation;
|
||||||
|
|
||||||
|
[RegisterComponent]
|
||||||
|
public sealed class RotationVisualsComponent : Component
|
||||||
|
{
|
||||||
|
[ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public Angle VerticalRotation = 0;
|
||||||
|
|
||||||
|
[ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public Angle HorizontalRotation = Angle.FromDegrees(90);
|
||||||
|
|
||||||
|
[ViewVariables(VVAccess.ReadWrite)]
|
||||||
|
public float AnimationTime = 0.125f;
|
||||||
|
}
|
||||||
@@ -127,8 +127,7 @@
|
|||||||
groups:
|
groups:
|
||||||
Brute: 5
|
Brute: 5
|
||||||
- type: Appearance
|
- type: Appearance
|
||||||
visuals:
|
- type: RotationVisuals
|
||||||
- type: BuckleVisualizer
|
|
||||||
- type: FireVisuals
|
- type: FireVisuals
|
||||||
sprite: Mobs/Effects/onfire.rsi
|
sprite: Mobs/Effects/onfire.rsi
|
||||||
normalState: Generic_mob_burning
|
normalState: Generic_mob_burning
|
||||||
|
|||||||
@@ -225,10 +225,9 @@
|
|||||||
sprite: Mobs/Effects/burn_damage.rsi
|
sprite: Mobs/Effects/burn_damage.rsi
|
||||||
- type: Appearance
|
- type: Appearance
|
||||||
visuals:
|
visuals:
|
||||||
- type: RotationVisualizer
|
|
||||||
- type: BuckleVisualizer
|
|
||||||
- type: CreamPiedVisualizer
|
- type: CreamPiedVisualizer
|
||||||
state: creampie_human
|
state: creampie_human
|
||||||
|
- type: RotationVisuals
|
||||||
- type: FireVisuals
|
- type: FireVisuals
|
||||||
sprite: Mobs/Effects/onfire.rsi
|
sprite: Mobs/Effects/onfire.rsi
|
||||||
normalState: Generic_mob_burning
|
normalState: Generic_mob_burning
|
||||||
@@ -368,6 +367,7 @@
|
|||||||
- type: Body
|
- type: Body
|
||||||
prototype: Human
|
prototype: Human
|
||||||
requiredLegs: 2
|
requiredLegs: 2
|
||||||
|
- type: Appearance
|
||||||
- type: Damageable
|
- type: Damageable
|
||||||
damageContainer: Biological
|
damageContainer: Biological
|
||||||
- type: MobState
|
- type: MobState
|
||||||
@@ -375,9 +375,6 @@
|
|||||||
0: Alive
|
0: Alive
|
||||||
100: Critical
|
100: Critical
|
||||||
200: Dead
|
200: Dead
|
||||||
- type: Appearance
|
|
||||||
visuals:
|
|
||||||
- type: RotationVisualizer
|
|
||||||
- type: UserInterface
|
- type: UserInterface
|
||||||
interfaces:
|
interfaces:
|
||||||
- key: enum.HumanoidMarkingModifierKey.Key # sure, this can go here too
|
- key: enum.HumanoidMarkingModifierKey.Key # sure, this can go here too
|
||||||
|
|||||||
Reference in New Issue
Block a user