osu/osu.Game/Storyboards/Drawables/DrawableStoryboardAnimation.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

155 lines
4.9 KiB
C#
Raw Normal View History

// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 09:19:50 +00:00
2022-06-17 07:37:17 +00:00
#nullable disable
2019-05-14 04:04:49 +00:00
using System;
using System.IO;
2017-09-07 21:55:05 +00:00
using osu.Framework.Allocation;
2017-09-08 18:39:17 +00:00
using osu.Framework.Graphics;
2017-09-07 21:55:05 +00:00
using osu.Framework.Graphics.Animations;
using osu.Framework.Graphics.Textures;
2020-01-09 04:43:44 +00:00
using osu.Framework.Utils;
using osu.Game.Beatmaps;
using osu.Game.Skinning;
using osuTK;
2018-04-13 09:19:50 +00:00
2017-09-07 21:55:05 +00:00
namespace osu.Game.Storyboards.Drawables
{
public partial class DrawableStoryboardAnimation : TextureAnimation, IFlippable, IVectorScalable
2017-09-07 21:55:05 +00:00
{
public StoryboardAnimation Animation { get; }
2018-04-13 09:19:50 +00:00
private bool flipH;
public bool FlipH
{
get => flipH;
set
{
if (flipH == value)
return;
flipH = value;
Invalidate(Invalidation.MiscGeometry);
}
}
private bool flipV;
public bool FlipV
{
get => flipV;
set
{
if (flipV == value)
return;
flipV = value;
Invalidate(Invalidation.MiscGeometry);
}
}
2018-04-13 09:19:50 +00:00
private Vector2 vectorScale = Vector2.One;
public Vector2 VectorScale
{
get => vectorScale;
set
{
if (vectorScale == value)
return;
if (!Validation.IsFinite(value)) throw new ArgumentException($@"{nameof(VectorScale)} must be finite, but is {value}.");
vectorScale = value;
Invalidate(Invalidation.MiscGeometry);
}
}
2018-03-03 16:48:00 +00:00
public override bool RemoveWhenNotAlive => false;
2018-04-13 09:19:50 +00:00
protected override Vector2 DrawScale
=> new Vector2(FlipH ? -base.DrawScale.X : base.DrawScale.X, FlipV ? -base.DrawScale.Y : base.DrawScale.Y) * VectorScale;
2018-04-13 09:19:50 +00:00
public override Anchor Origin => StoryboardExtensions.AdjustOrigin(base.Origin, VectorScale, FlipH, FlipV);
2018-04-13 09:19:50 +00:00
public override bool IsPresent
=> !float.IsNaN(DrawPosition.X) && !float.IsNaN(DrawPosition.Y) && base.IsPresent;
2018-04-13 09:19:50 +00:00
2017-09-13 09:22:24 +00:00
public DrawableStoryboardAnimation(StoryboardAnimation animation)
2017-09-07 21:55:05 +00:00
{
2017-09-13 09:22:24 +00:00
Animation = animation;
Origin = animation.Origin;
Position = animation.InitialPosition;
Loop = animation.LoopType == AnimationLoopType.LoopForever;
2018-04-13 09:19:50 +00:00
LifetimeStart = animation.StartTime;
LifetimeEnd = animation.EndTimeForDisplay;
2017-09-07 21:55:05 +00:00
}
2018-04-13 09:19:50 +00:00
[Resolved]
private ISkinSource skin { get; set; }
[Resolved]
private IBeatSyncProvider beatSyncProvider { get; set; }
2017-09-07 21:55:05 +00:00
[BackgroundDependencyLoader]
private void load(TextureStore textureStore, Storyboard storyboard)
2017-09-07 21:55:05 +00:00
{
int frameIndex = 0;
Texture frameTexture = storyboard.GetTextureFromPath(getFramePath(frameIndex), textureStore);
if (frameTexture != null)
2017-09-07 21:55:05 +00:00
{
// sourcing from storyboard.
for (frameIndex = 0; frameIndex < Animation.FrameCount; frameIndex++)
{
AddFrame(storyboard.GetTextureFromPath(getFramePath(frameIndex), textureStore), Animation.FrameDelay);
}
}
else if (storyboard.UseSkinSprites)
{
// fallback to skin if required.
skin.SourceChanged += skinSourceChanged;
skinSourceChanged();
2017-09-07 21:55:05 +00:00
}
2019-02-28 04:31:40 +00:00
2017-09-13 09:22:24 +00:00
Animation.ApplyTransforms(this);
2017-09-07 21:55:05 +00:00
}
protected override void LoadComplete()
{
base.LoadComplete();
// Framework animation class tries its best to synchronise the animation at LoadComplete,
// but in some cases (such as fast forward) this results in an incorrect start offset.
//
// In the case of storyboard animations, we want to synchronise with game time perfectly
// so let's get a correct time based on gameplay clock and earliest transform.
2022-09-23 14:06:55 +00:00
PlaybackPosition = (beatSyncProvider.Clock?.CurrentTime ?? Clock.CurrentTime) - Animation.EarliestTransformTime;
}
private void skinSourceChanged()
{
ClearFrames();
// When reading from a skin, we match stables weird behaviour where `FrameCount` is ignored
// and resources are retrieved until the end of the animation.
2022-12-16 11:18:02 +00:00
foreach (var texture in skin.GetTextures(Path.GetFileNameWithoutExtension(Animation.Path)!, default, default, true, string.Empty, out _))
AddFrame(texture, Animation.FrameDelay);
}
private string getFramePath(int i) => Animation.Path.Replace(".", $"{i}.");
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
if (skin != null)
skin.SourceChanged -= skinSourceChanged;
}
2017-09-07 21:55:05 +00:00
}
}