mirror of https://github.com/ppy/osu
232 lines
7.7 KiB
C#
232 lines
7.7 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 System.Collections.Generic;
|
|
using ManagedBass.Fx;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Audio;
|
|
using osu.Framework.Audio.Track;
|
|
using osu.Framework.Bindables;
|
|
using osu.Framework.Extensions.Color4Extensions;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Containers;
|
|
using osu.Framework.Graphics.Shapes;
|
|
using osu.Framework.Utils;
|
|
using osu.Game.Audio;
|
|
using osu.Game.Audio.Effects;
|
|
using osu.Game.Beatmaps;
|
|
using osu.Game.Configuration;
|
|
using osu.Game.Graphics;
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
|
using osu.Game.Rulesets.UI;
|
|
using osu.Game.Skinning;
|
|
using osuTK;
|
|
using osuTK.Graphics;
|
|
|
|
namespace osu.Game.Screens.Play
|
|
{
|
|
/// <summary>
|
|
/// Manage the animation to be applied when a player fails. Applies the animation to children.
|
|
/// Single use and automatically disposed after use.
|
|
/// </summary>
|
|
public partial class FailAnimationContainer : Container
|
|
{
|
|
public Action? OnComplete;
|
|
|
|
private readonly DrawableRuleset drawableRuleset;
|
|
private readonly BindableDouble trackFreq = new BindableDouble(1);
|
|
private readonly BindableDouble volumeAdjustment = new BindableDouble(0.5);
|
|
|
|
private Container filters = null!;
|
|
|
|
private Box redFlashLayer = null!;
|
|
|
|
private Track track = null!;
|
|
|
|
private AudioFilter failLowPassFilter = null!;
|
|
private AudioFilter failHighPassFilter = null!;
|
|
|
|
private const float duration = 2500;
|
|
|
|
private SkinnableSound failSample = null!;
|
|
|
|
[Resolved]
|
|
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>
|
|
public BackgroundScreen? Background { private get; set; }
|
|
|
|
public FailAnimationContainer(DrawableRuleset drawableRuleset)
|
|
{
|
|
this.drawableRuleset = drawableRuleset;
|
|
|
|
RelativeSizeAxes = Axes.Both;
|
|
}
|
|
|
|
[BackgroundDependencyLoader]
|
|
private void load(AudioManager audio, IBindable<WorkingBeatmap> beatmap)
|
|
{
|
|
track = beatmap.Value.Track;
|
|
AddInternal(failSample = new SkinnableSound(new SampleInfo("Gameplay/failsound")));
|
|
|
|
AddRangeInternal(new Drawable[]
|
|
{
|
|
filters = new Container
|
|
{
|
|
Children = new Drawable[]
|
|
{
|
|
failLowPassFilter = new AudioFilter(audio.TrackMixer),
|
|
failHighPassFilter = new AudioFilter(audio.TrackMixer, BQFType.HighPass),
|
|
},
|
|
},
|
|
Content,
|
|
redFlashLayer = new Box
|
|
{
|
|
Colour = Color4.Red.Opacity(0.6f),
|
|
RelativeSizeAxes = Axes.Both,
|
|
Blending = BlendingParameters.Additive,
|
|
Depth = float.MinValue,
|
|
Alpha = 0
|
|
},
|
|
});
|
|
}
|
|
|
|
private bool started;
|
|
private bool filtersRemoved;
|
|
|
|
/// <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.");
|
|
|
|
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);
|
|
OnComplete?.Invoke();
|
|
});
|
|
|
|
failHighPassFilter.CutoffTo(300);
|
|
failLowPassFilter.CutoffTo(300, duration, Easing.OutCubic);
|
|
failSample.Play();
|
|
|
|
track.AddAdjustment(AdjustableProperty.Frequency, trackFreq);
|
|
track.AddAdjustment(AdjustableProperty.Volume, volumeAdjustment);
|
|
|
|
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
|
|
});
|
|
|
|
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);
|
|
}
|
|
|
|
/// <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)
|
|
{
|
|
filtersRemoved = true;
|
|
|
|
if (!started)
|
|
return;
|
|
|
|
if (resetTrackFrequency)
|
|
track.RemoveAdjustment(AdjustableProperty.Frequency, trackFreq);
|
|
|
|
track.RemoveAdjustment(AdjustableProperty.Volume, volumeAdjustment);
|
|
|
|
if (filters.Parent == null)
|
|
return;
|
|
|
|
RemoveInternal(filters, true);
|
|
}
|
|
|
|
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;
|
|
|
|
foreach (var nested in playfield.NestedPlayfields)
|
|
applyToPlayfield(nested);
|
|
|
|
foreach (var entry in playfield.HitObjectContainer.AliveEntries)
|
|
{
|
|
var obj = entry.Value;
|
|
|
|
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
|
|
obj.ApplyCustomUpdateState += (_, _) => dropOffScreen(obj, failTime, rotation, originalScale, originalPosition);
|
|
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|