osu/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs

286 lines
9.0 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
using System;
2018-04-13 09:19:50 +00:00
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Audio;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Graphics;
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.Osu.Objects.Drawables.Pieces;
using osu.Game.Rulesets.Osu.Skinning;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets.Scoring;
2019-09-06 06:24:00 +00:00
using osu.Game.Screens.Ranking;
using osu.Game.Skinning;
using osuTK;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
{
public class DrawableSpinner : DrawableOsuHitObject
{
protected readonly Spinner Spinner;
private readonly Container<DrawableSpinnerTick> ticks;
public readonly SpinnerRotationTracker RotationTracker;
2019-12-18 00:04:37 +00:00
public readonly SpinnerSpmCounter SpmCounter;
private readonly SpinnerBonusDisplay bonusDisplay;
2018-04-13 09:19:50 +00:00
2018-11-09 04:58:46 +00:00
private readonly IBindable<Vector2> positionBindable = new Bindable<Vector2>();
private bool spinnerFrequencyModulate;
2019-02-28 04:31:40 +00:00
public DrawableSpinner(Spinner s)
: base(s)
2018-04-13 09:19:50 +00:00
{
Origin = Anchor.Centre;
Position = s.Position;
RelativeSizeAxes = Axes.Both;
Spinner = s;
InternalChildren = new Drawable[]
{
ticks = new Container<DrawableSpinnerTick>(),
new AspectContainer
2018-04-13 09:19:50 +00:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Y,
Children = new Drawable[]
2018-04-13 09:19:50 +00:00
{
2020-07-29 13:31:18 +00:00
new SkinnableDrawable(new OsuSkinComponent(OsuSkinComponents.SpinnerBody), _ => new DefaultSpinnerDisc()),
RotationTracker = new SpinnerRotationTracker(Spinner)
2018-04-13 09:19:50 +00:00
}
},
2019-12-18 00:04:37 +00:00
SpmCounter = new SpinnerSpmCounter
2018-04-13 09:19:50 +00:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Y = 120,
Alpha = 0
},
bonusDisplay = new SpinnerBonusDisplay
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Y = -120,
2018-04-13 09:19:50 +00:00
}
};
}
2020-07-30 10:34:59 +00:00
private Bindable<bool> isSpinning;
protected override void LoadComplete()
{
base.LoadComplete();
isSpinning = RotationTracker.IsSpinning.GetBoundCopy();
isSpinning.BindValueChanged(updateSpinningSample);
}
private PausableSkinnableSound spinningSample;
2020-08-15 18:44:02 +00:00
private const float spinning_sample_initial_frequency = 1.0f;
private const float spinning_sample_modulated_base_frequency = 0.5f;
2020-07-30 10:34:59 +00:00
protected override void LoadSamples()
{
base.LoadSamples();
spinningSample?.Expire();
spinningSample = null;
var firstSample = HitObject.Samples.FirstOrDefault();
if (firstSample != null)
{
var clone = HitObject.SampleControlPoint.ApplyTo(firstSample);
clone.Name = "spinnerspin";
AddInternal(spinningSample = new PausableSkinnableSound(clone)
2020-07-30 10:34:59 +00:00
{
Volume = { Value = 0 },
2020-07-30 10:34:59 +00:00
Looping = true,
2020-08-15 18:44:02 +00:00
Frequency = { Value = spinning_sample_initial_frequency }
2020-07-30 10:34:59 +00:00
});
}
}
private void updateSpinningSample(ValueChangedEvent<bool> tracking)
{
2020-09-29 03:45:20 +00:00
if (tracking.NewValue)
2020-07-30 10:34:59 +00:00
{
spinningSample?.Play();
spinningSample?.VolumeTo(1, 200);
}
else
{
spinningSample?.VolumeTo(0, 200).Finally(_ => spinningSample.Stop());
2020-07-30 10:34:59 +00:00
}
}
public override void StopAllSamples()
{
base.StopAllSamples();
spinningSample?.Stop();
}
protected override void AddNestedHitObject(DrawableHitObject hitObject)
{
base.AddNestedHitObject(hitObject);
switch (hitObject)
{
case DrawableSpinnerTick tick:
ticks.Add(tick);
break;
}
}
protected override void UpdateStateTransforms(ArmedState state)
{
base.UpdateStateTransforms(state);
using (BeginDelayedSequence(Spinner.Duration, true))
this.FadeOut(160);
2020-07-30 10:34:59 +00:00
// skin change does a rewind of transforms, which will stop the spinning sound from playing if it's currently in playback.
isSpinning?.TriggerChange();
}
protected override void ClearNestedHitObjects()
{
base.ClearNestedHitObjects();
ticks.Clear();
}
protected override DrawableHitObject CreateNestedHitObject(HitObject hitObject)
{
switch (hitObject)
{
case SpinnerBonusTick bonusTick:
return new DrawableSpinnerBonusTick(bonusTick);
case SpinnerTick tick:
return new DrawableSpinnerTick(tick);
}
return base.CreateNestedHitObject(hitObject);
2018-11-09 04:58:46 +00:00
}
[BackgroundDependencyLoader]
2020-08-15 18:34:17 +00:00
private void load(OsuColour colours)
2018-11-09 04:58:46 +00:00
{
positionBindable.BindValueChanged(pos => Position = pos.NewValue);
2018-11-09 04:58:46 +00:00
positionBindable.BindTo(HitObject.PositionBindable);
}
protected override void ApplySkin(ISkinSource skin, bool allowFallback)
{
2020-08-15 18:34:17 +00:00
base.ApplySkin(skin, allowFallback);
spinnerFrequencyModulate = skin.GetConfig<OsuSkinConfiguration, bool>(OsuSkinConfiguration.SpinnerFrequencyModulate)?.Value ?? true;
2018-04-13 09:19:50 +00:00
}
2020-07-30 03:55:34 +00:00
/// <summary>
/// The completion progress of this spinner from 0..1 (clamped).
/// </summary>
public float Progress
{
get
{
if (Spinner.SpinsRequired == 0)
// some spinners are so short they can't require an integer spin count.
// these become implicitly hit.
return 1;
return Math.Clamp(RotationTracker.RateAdjustedRotation / 360 / Spinner.SpinsRequired, 0, 1);
}
}
2018-04-13 09:19:50 +00:00
protected override void CheckForResult(bool userTriggered, double timeOffset)
2018-04-13 09:19:50 +00:00
{
if (Time.Current < HitObject.StartTime) return;
RotationTracker.Complete.Value = Progress >= 1;
2018-04-13 09:19:50 +00:00
if (userTriggered || Time.Current < Spinner.EndTime)
return;
// Trigger a miss result for remaining ticks to avoid infinite gameplay.
foreach (var tick in ticks.Where(t => !t.IsHit))
2020-07-21 10:48:44 +00:00
tick.TriggerResult(false);
ApplyResult(r =>
2018-04-13 09:19:50 +00:00
{
if (Progress >= 1)
r.Type = HitResult.Great;
2018-04-13 09:19:50 +00:00
else if (Progress > .9)
2020-09-29 08:16:55 +00:00
r.Type = HitResult.Ok;
2018-04-13 09:19:50 +00:00
else if (Progress > .75)
r.Type = HitResult.Meh;
2018-04-13 09:19:50 +00:00
else if (Time.Current >= Spinner.EndTime)
r.Type = HitResult.Miss;
});
2018-04-13 09:19:50 +00:00
}
protected override void Update()
{
base.Update();
2020-07-30 10:34:59 +00:00
if (HandleUserInput)
2020-07-30 10:34:59 +00:00
RotationTracker.Tracking = !Result.HasResult && (OsuActionInputManager?.PressedActions.Any(x => x == OsuAction.LeftButton || x == OsuAction.RightButton) ?? false);
if (spinningSample != null && spinnerFrequencyModulate)
2020-08-15 18:44:02 +00:00
spinningSample.Frequency.Value = spinning_sample_modulated_base_frequency + Progress;
2018-04-13 09:19:50 +00:00
}
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
if (!SpmCounter.IsPresent && RotationTracker.Tracking)
SpmCounter.FadeIn(HitObject.TimeFadeIn);
SpmCounter.SetRotation(RotationTracker.RateAdjustedRotation);
updateBonusScore();
2018-04-13 09:19:50 +00:00
}
private int wholeSpins;
private void updateBonusScore()
{
if (ticks.Count == 0)
return;
int spins = (int)(RotationTracker.RateAdjustedRotation / 360);
2020-07-21 10:21:30 +00:00
if (spins < wholeSpins)
{
2020-07-21 10:21:30 +00:00
// rewinding, silently handle
wholeSpins = spins;
return;
}
2020-07-21 10:21:30 +00:00
while (wholeSpins != spins)
{
var tick = ticks.FirstOrDefault(t => !t.IsHit);
2020-07-21 10:21:30 +00:00
// tick may be null if we've hit the spin limit.
if (tick != null)
{
2020-07-21 10:48:44 +00:00
tick.TriggerResult(true);
2020-07-21 10:21:30 +00:00
if (tick is DrawableSpinnerBonusTick)
bonusDisplay.SetBonusCount(spins - Spinner.SpinsRequired);
}
2020-07-21 10:21:30 +00:00
wholeSpins++;
}
}
2018-04-13 09:19:50 +00:00
}
}