mirror of
https://github.com/ppy/osu.git
synced 2025-01-23 06:52:58 +08:00
223 lines
8.2 KiB
C#
223 lines
8.2 KiB
C#
// 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.
|
|
|
|
using System;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Bindables;
|
|
using osu.Framework.Extensions.ObjectExtensions;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Animations;
|
|
using osu.Framework.Graphics.Textures;
|
|
using osu.Game.Rulesets.Mania.Objects.Drawables;
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
|
using osu.Game.Rulesets.UI.Scrolling;
|
|
using osu.Game.Skinning;
|
|
using osuTK;
|
|
|
|
namespace osu.Game.Rulesets.Mania.Skinning.Legacy
|
|
{
|
|
public partial class LegacyBodyPiece : LegacyManiaColumnElement
|
|
{
|
|
private DrawableHoldNote holdNote = null!;
|
|
|
|
private readonly IBindable<ScrollingDirection> direction = new Bindable<ScrollingDirection>();
|
|
private readonly IBindable<bool> isHitting = new Bindable<bool>();
|
|
|
|
/// <summary>
|
|
/// Stores the start time of the fade animation that plays when any of the nested
|
|
/// hitobjects of the hold note are missed.
|
|
/// </summary>
|
|
private readonly Bindable<double?> missFadeTime = new Bindable<double?>();
|
|
|
|
private Drawable? bodySprite;
|
|
|
|
private Drawable? lightContainer;
|
|
|
|
private Drawable? light;
|
|
|
|
public LegacyBodyPiece()
|
|
{
|
|
RelativeSizeAxes = Axes.Both;
|
|
}
|
|
|
|
[BackgroundDependencyLoader]
|
|
private void load(ISkinSource skin, IScrollingInfo scrollingInfo, DrawableHitObject drawableObject)
|
|
{
|
|
holdNote = (DrawableHoldNote)drawableObject;
|
|
|
|
string imageName = GetColumnSkinConfig<string>(skin, LegacyManiaSkinConfigurationLookups.HoldNoteBodyImage)?.Value
|
|
?? $"mania-note{FallbackColumnIndex}L";
|
|
|
|
string lightImage = GetColumnSkinConfig<string>(skin, LegacyManiaSkinConfigurationLookups.HoldNoteLightImage)?.Value
|
|
?? "lightingL";
|
|
|
|
float lightScale = GetColumnSkinConfig<float>(skin, LegacyManiaSkinConfigurationLookups.HoldNoteLightScale)?.Value
|
|
?? 1;
|
|
|
|
float minimumColumnWidth = GetColumnSkinConfig<float>(skin, LegacyManiaSkinConfigurationLookups.MinimumColumnWidth)?.Value
|
|
?? 1;
|
|
|
|
// Create a temporary animation to retrieve the number of frames, in an effort to calculate the intended frame length.
|
|
// This animation is discarded and re-queried with the appropriate frame length afterwards.
|
|
var tmp = skin.GetAnimation(lightImage, true, false);
|
|
double frameLength = 0;
|
|
if (tmp is IFramedAnimation tmpAnimation && tmpAnimation.FrameCount > 0)
|
|
frameLength = Math.Max(1000 / 60.0, 170.0 / tmpAnimation.FrameCount);
|
|
|
|
light = skin.GetAnimation(lightImage, true, true, frameLength: frameLength).With(d =>
|
|
{
|
|
if (d == null)
|
|
return;
|
|
|
|
d.Origin = Anchor.Centre;
|
|
d.Blending = BlendingParameters.Additive;
|
|
d.Scale = new Vector2(lightScale);
|
|
});
|
|
|
|
if (light != null)
|
|
{
|
|
lightContainer = new HitTargetInsetContainer
|
|
{
|
|
Alpha = 0,
|
|
Child = light
|
|
};
|
|
}
|
|
|
|
bodySprite = skin.GetAnimation(imageName, WrapMode.ClampToEdge, WrapMode.ClampToEdge, true, true).With(d =>
|
|
{
|
|
if (d == null)
|
|
return;
|
|
|
|
if (d is TextureAnimation animation)
|
|
animation.IsPlaying = false;
|
|
|
|
d.Anchor = Anchor.TopCentre;
|
|
d.RelativeSizeAxes = Axes.Both;
|
|
d.Size = Vector2.One;
|
|
d.FillMode = FillMode.Stretch;
|
|
d.Height = minimumColumnWidth / d.DrawWidth * 1.6f; // constant matching stable.
|
|
// Todo: Wrap?
|
|
});
|
|
|
|
if (bodySprite != null)
|
|
InternalChild = bodySprite;
|
|
|
|
direction.BindTo(scrollingInfo.Direction);
|
|
isHitting.BindTo(holdNote.IsHitting);
|
|
}
|
|
|
|
protected override void LoadComplete()
|
|
{
|
|
base.LoadComplete();
|
|
|
|
direction.BindValueChanged(onDirectionChanged, true);
|
|
isHitting.BindValueChanged(onIsHittingChanged, true);
|
|
missFadeTime.BindValueChanged(onMissFadeTimeChanged, true);
|
|
|
|
holdNote.ApplyCustomUpdateState += applyCustomUpdateState;
|
|
applyCustomUpdateState(holdNote, holdNote.State.Value);
|
|
}
|
|
|
|
private void applyCustomUpdateState(DrawableHitObject hitObject, ArmedState state)
|
|
{
|
|
// ensure that the hold note is also faded out when the head/tail/any tick is missed.
|
|
if (state == ArmedState.Miss)
|
|
missFadeTime.Value ??= hitObject.HitStateUpdateTime;
|
|
}
|
|
|
|
private void onIsHittingChanged(ValueChangedEvent<bool> isHitting)
|
|
{
|
|
if (bodySprite is TextureAnimation bodyAnimation)
|
|
{
|
|
bodyAnimation.GotoFrame(0);
|
|
bodyAnimation.IsPlaying = isHitting.NewValue;
|
|
}
|
|
|
|
if (lightContainer == null)
|
|
return;
|
|
|
|
if (isHitting.NewValue)
|
|
{
|
|
// Clear the fade out and, more importantly, the removal.
|
|
lightContainer.ClearTransforms();
|
|
|
|
// Only add the container if the removal has taken place.
|
|
if (lightContainer.Parent == null)
|
|
Column.TopLevelContainer.Add(lightContainer);
|
|
|
|
// The light must be seeked only after being loaded, otherwise a nullref occurs (https://github.com/ppy/osu-framework/issues/3847).
|
|
if (light is TextureAnimation lightAnimation)
|
|
lightAnimation.GotoFrame(0);
|
|
|
|
lightContainer.FadeIn(80);
|
|
}
|
|
else
|
|
{
|
|
lightContainer.FadeOut(120)
|
|
.OnComplete(d => Column.TopLevelContainer.Remove(d, false));
|
|
}
|
|
}
|
|
|
|
private void onDirectionChanged(ValueChangedEvent<ScrollingDirection> direction)
|
|
{
|
|
if (direction.NewValue == ScrollingDirection.Up)
|
|
{
|
|
if (bodySprite != null)
|
|
{
|
|
bodySprite.Origin = Anchor.BottomCentre;
|
|
bodySprite.Scale = new Vector2(1, -1);
|
|
}
|
|
|
|
if (light != null)
|
|
light.Anchor = Anchor.TopCentre;
|
|
}
|
|
else
|
|
{
|
|
if (bodySprite != null)
|
|
{
|
|
bodySprite.Origin = Anchor.TopCentre;
|
|
bodySprite.Scale = Vector2.One;
|
|
}
|
|
|
|
if (light != null)
|
|
light.Anchor = Anchor.BottomCentre;
|
|
}
|
|
}
|
|
|
|
private void onMissFadeTimeChanged(ValueChangedEvent<double?> missFadeTimeChange)
|
|
{
|
|
if (missFadeTimeChange.NewValue == null)
|
|
return;
|
|
|
|
// this update could come from any nested object of the hold note (or even from an input).
|
|
// make sure the transforms are consistent across all affected parts.
|
|
using (BeginAbsoluteSequence(missFadeTimeChange.NewValue.Value))
|
|
{
|
|
// colour and duration matches stable
|
|
// transforms not applied to entire hold note in order to not affect hit lighting
|
|
const double fade_duration = 60;
|
|
|
|
holdNote.Head.FadeColour(Colour4.DarkGray, fade_duration);
|
|
holdNote.Tail.FadeColour(Colour4.DarkGray, fade_duration);
|
|
bodySprite?.FadeColour(Colour4.DarkGray, fade_duration);
|
|
}
|
|
}
|
|
|
|
protected override void Update()
|
|
{
|
|
base.Update();
|
|
missFadeTime.Value ??= holdNote.HoldBrokenTime;
|
|
}
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
{
|
|
base.Dispose(isDisposing);
|
|
|
|
if (holdNote.IsNotNull())
|
|
holdNote.ApplyCustomUpdateState -= applyCustomUpdateState;
|
|
|
|
lightContainer?.Expire();
|
|
}
|
|
}
|
|
}
|