osu/osu.Game/Screens/Play/FailAnimation.cs

230 lines
7.6 KiB
C#
Raw Normal View History

2019-06-04 07:13:16 +00:00
// 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 System.Collections.Generic;
using ManagedBass.Fx;
2019-06-04 07:13:16 +00:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
2019-06-04 07:13:16 +00:00
using osu.Framework.Audio.Track;
using osu.Framework.Bindables;
2021-11-01 04:06:26 +00:00
using osu.Framework.Extensions.Color4Extensions;
2019-06-04 07:13:16 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
2020-01-09 04:43:44 +00:00
using osu.Framework.Utils;
using osu.Game.Audio;
using osu.Game.Audio.Effects;
2019-06-04 07:13:16 +00:00
using osu.Game.Beatmaps;
using osu.Game.Configuration;
using osu.Game.Graphics;
2019-06-04 07:13:16 +00:00
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.UI;
using osu.Game.Skinning;
2019-06-04 07:13:16 +00:00
using osuTK;
using osuTK.Graphics;
namespace osu.Game.Screens.Play
{
/// <summary>
/// Manage the animation to be applied when a player fails.
/// Single use and automatically disposed after use.
2019-06-04 07:13:16 +00:00
/// </summary>
2022-11-24 05:32:20 +00:00
public partial class FailAnimation : Container
2019-06-04 07:13:16 +00:00
{
2022-08-25 05:35:42 +00:00
public Action? OnComplete;
2019-06-04 07:13:16 +00:00
private readonly DrawableRuleset drawableRuleset;
private readonly BindableDouble trackFreq = new BindableDouble(1);
private readonly BindableDouble volumeAdjustment = new BindableDouble(0.5);
2019-06-04 07:13:16 +00:00
2022-08-25 05:35:42 +00:00
private Container filters = null!;
2021-10-15 12:22:38 +00:00
2022-08-25 05:35:42 +00:00
private Box redFlashLayer = null!;
private Track track = null!;
2019-06-04 07:13:16 +00:00
2022-08-25 05:35:42 +00:00
private AudioFilter failLowPassFilter = null!;
private AudioFilter failHighPassFilter = null!;
2019-06-04 07:13:16 +00:00
private const float duration = 2500;
private SkinnableSound failSample = null!;
2019-06-04 07:13:16 +00:00
[Resolved]
2022-08-25 05:35:42 +00:00
private OsuConfigManager config { get; set; } = null!;
protected override Container<Drawable> Content { get; } = new Container
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Both,
};
/// <summary>
/// The player screen background, used to adjust appearance on failing.
/// </summary>
2022-08-25 05:35:42 +00:00
public BackgroundScreen? Background { private get; set; }
2019-06-04 07:13:16 +00:00
public FailAnimation(DrawableRuleset drawableRuleset)
{
this.drawableRuleset = drawableRuleset;
RelativeSizeAxes = Axes.Both;
2019-06-04 07:13:16 +00:00
}
[BackgroundDependencyLoader]
private void load(AudioManager audio, IBindable<WorkingBeatmap> beatmap)
2019-06-04 07:13:16 +00:00
{
track = beatmap.Value.Track;
AddInternal(failSample = new SkinnableSound(new SampleInfo("Gameplay/failsound")));
AddRangeInternal(new Drawable[]
{
2021-10-15 12:22:38 +00:00
filters = new Container
{
Children = new Drawable[]
{
failLowPassFilter = new AudioFilter(audio.TrackMixer),
failHighPassFilter = new AudioFilter(audio.TrackMixer, BQFType.HighPass),
},
},
Content,
redFlashLayer = new Box
{
2021-11-01 04:06:26 +00:00
Colour = Color4.Red.Opacity(0.6f),
RelativeSizeAxes = Axes.Both,
Blending = BlendingParameters.Additive,
Depth = float.MinValue,
Alpha = 0
},
});
2019-06-04 07:13:16 +00:00
}
private bool started;
private bool filtersRemoved;
2019-06-04 07:13:16 +00:00
/// <summary>
/// Start the fail animation playing.
/// </summary>
/// <exception cref="InvalidOperationException">Thrown if started more than once.</exception>
public void Start()
{
if (started) throw new InvalidOperationException("Animation cannot be started more than once.");
if (filtersRemoved) throw new InvalidOperationException("Animation cannot be started after filters have been removed.");
2019-06-04 07:13:16 +00:00
started = true;
this.TransformBindableTo(trackFreq, 0, duration).OnComplete(_ =>
{
// Don't reset frequency as the pause screen may appear post transform, causing a second frequency sweep.
removeFilters(false);
2019-06-04 07:13:16 +00:00
OnComplete?.Invoke();
});
failHighPassFilter.CutoffTo(300);
failLowPassFilter.CutoffTo(300, duration, Easing.OutCubic);
failSample.Play();
2019-06-04 07:13:16 +00:00
track.AddAdjustment(AdjustableProperty.Frequency, trackFreq);
track.AddAdjustment(AdjustableProperty.Volume, volumeAdjustment);
2019-06-04 07:13:16 +00:00
applyToPlayfield(drawableRuleset.Playfield);
drawableRuleset.Playfield.HitObjectContainer.FadeOut(duration / 2);
if (config.Get<bool>(OsuSetting.FadePlayfieldWhenHealthLow))
redFlashLayer.FadeOutFromOne(1000);
Content.Masking = true;
Content.Add(new Box
{
Colour = Color4.Black,
RelativeSizeAxes = Axes.Both,
Depth = float.MaxValue
});
2021-10-15 10:37:22 +00:00
Content.ScaleTo(0.85f, duration, Easing.OutQuart);
Content.RotateTo(1, duration, Easing.OutQuart);
Content.FadeColour(Color4.Gray, duration);
// Will be restored by `ApplyToBackground` logic in `SongSelect`.
Background?.FadeColour(OsuColour.Gray(0.3f), 60);
2019-06-04 07:13:16 +00:00
}
/// <summary>
/// Stops any and all persistent effects added by the ongoing fail animation.
/// </summary>
public void Stop()
{
failSample.Stop();
removeFilters();
}
private void removeFilters(bool resetTrackFrequency = true)
2021-10-15 12:22:38 +00:00
{
filtersRemoved = true;
if (!started)
return;
if (resetTrackFrequency)
track.RemoveAdjustment(AdjustableProperty.Frequency, trackFreq);
track.RemoveAdjustment(AdjustableProperty.Volume, volumeAdjustment);
if (filters.Parent == null)
return;
2022-08-29 06:49:28 +00:00
RemoveInternal(filters, true);
2021-10-15 12:22:38 +00:00
}
2019-06-04 07:13:16 +00:00
protected override void Update()
{
base.Update();
if (!started)
return;
applyToPlayfield(drawableRuleset.Playfield);
}
private readonly List<DrawableHitObject> appliedObjects = new List<DrawableHitObject>();
private void applyToPlayfield(Playfield playfield)
{
double failTime = playfield.Time.Current;
2019-06-04 07:13:16 +00:00
foreach (var nested in playfield.NestedPlayfields)
applyToPlayfield(nested);
foreach (DrawableHitObject obj in playfield.HitObjectContainer.AliveObjects)
{
if (appliedObjects.Contains(obj))
continue;
float rotation = RNG.NextSingle(-90, 90);
Vector2 originalPosition = obj.Position;
Vector2 originalScale = obj.Scale;
dropOffScreen(obj, failTime, rotation, originalScale, originalPosition);
// need to reapply the fail drop after judgement state changes
2022-06-24 12:25:23 +00:00
obj.ApplyCustomUpdateState += (_, _) => dropOffScreen(obj, failTime, rotation, originalScale, originalPosition);
2019-06-04 07:13:16 +00:00
appliedObjects.Add(obj);
}
}
private void dropOffScreen(DrawableHitObject obj, double failTime, float randomRotation, Vector2 originalScale, Vector2 originalPosition)
{
using (obj.BeginAbsoluteSequence(failTime))
{
obj.RotateTo(randomRotation, duration);
obj.ScaleTo(originalScale * 0.5f, duration);
obj.MoveTo(originalPosition + new Vector2(0, 400), duration);
}
}
2019-06-04 07:13:16 +00:00
}
}