// 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(); } } }