1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-07 16:52:54 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs

372 lines
13 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 17:19:50 +08:00
2022-06-17 15:37:17 +08:00
#nullable disable
using System;
using System.Collections.Generic;
2018-04-13 17:19:50 +08:00
using System.Linq;
using JetBrains.Annotations;
2018-04-13 17:19:50 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2020-11-19 19:40:30 +08:00
using osu.Game.Audio;
using osu.Game.Rulesets.Judgements;
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.Osu.Judgements;
using osu.Game.Rulesets.Osu.Scoring;
using osu.Game.Rulesets.Osu.Skinning;
2020-12-04 19:21:53 +08:00
using osu.Game.Rulesets.Osu.Skinning.Default;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Scoring;
2019-09-06 14:24:00 +08:00
using osu.Game.Screens.Ranking;
using osu.Game.Skinning;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
{
2022-11-24 13:32:20 +08:00
public partial class DrawableSpinner : DrawableOsuHitObject
2018-04-13 17:19:50 +08:00
{
2020-11-05 12:51:46 +08:00
public new Spinner HitObject => (Spinner)base.HitObject;
2018-04-13 17:19:50 +08:00
public new OsuSpinnerJudgementResult Result => (OsuSpinnerJudgementResult)base.Result;
public SkinnableDrawable Body { get; private set; }
2020-11-05 12:51:46 +08:00
public SpinnerRotationTracker RotationTracker { get; private set; }
private SpinnerSpmCalculator spmCalculator;
2020-11-05 12:51:46 +08:00
private Container<DrawableSpinnerTick> ticks;
2020-11-19 19:40:30 +08:00
private PausableSkinnableSound spinningSample;
2018-11-09 12:58:46 +08:00
2020-11-05 12:51:46 +08:00
private Bindable<bool> isSpinning;
private bool spinnerFrequencyModulate;
private const float spinning_sample_initial_frequency = 1.0f;
private const float spinning_sample_modulated_base_frequency = 0.5f;
private PausableSkinnableSound maxBonusSample;
/// <summary>
/// The amount of bonus score gained from spinning after the required number of spins, for display purposes.
/// </summary>
2023-10-20 14:57:13 +08:00
public double CurrentBonusScore => score_per_tick * Math.Clamp(completedFullSpins.Value - HitObject.SpinsRequiredForBonus, 0, HitObject.MaximumBonusSpins);
/// <summary>
/// The maximum amount of bonus score which can be achieved from extra spins.
/// </summary>
public double MaximumBonusScore => score_per_tick * HitObject.MaximumBonusSpins;
public IBindable<int> CompletedFullSpins => completedFullSpins;
private readonly Bindable<int> completedFullSpins = new Bindable<int>();
/// <summary>
/// The number of spins per minute this spinner is spinning at, for display purposes.
/// </summary>
public readonly IBindable<double> SpinsPerMinute = new BindableDouble();
private const double fade_out_duration = 240;
2020-11-10 23:22:06 +08:00
public DrawableSpinner()
: this(null)
{
}
public DrawableSpinner([CanBeNull] Spinner s = null)
2019-02-28 12:31:40 +08:00
: base(s)
2018-04-13 17:19:50 +08:00
{
2020-11-05 12:51:46 +08:00
}
2018-04-13 17:19:50 +08:00
2020-11-05 12:51:46 +08:00
[BackgroundDependencyLoader]
2022-01-15 08:06:39 +08:00
private void load()
2020-11-05 12:51:46 +08:00
{
Origin = Anchor.Centre;
2018-04-13 17:19:50 +08:00
RelativeSizeAxes = Axes.Both;
AddRangeInternal(new Drawable[]
2018-04-13 17:19:50 +08:00
{
spmCalculator = new SpinnerSpmCalculator
{
Result = { BindTarget = SpinsPerMinute },
},
ticks = new Container<DrawableSpinnerTick>
{
RelativeSizeAxes = Axes.Both,
},
new AspectContainer
2018-04-13 17:19:50 +08:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Y,
Children = new Drawable[]
2018-04-13 17:19:50 +08:00
{
Body = new SkinnableDrawable(new OsuSkinComponentLookup(OsuSkinComponents.SpinnerBody), _ => new DefaultSpinner()),
RotationTracker = new SpinnerRotationTracker(this)
2018-04-13 17:19:50 +08:00
}
},
2020-11-19 19:40:30 +08:00
spinningSample = new PausableSkinnableSound
{
Volume = { Value = 0 },
MinimumSampleVolume = MINIMUM_SAMPLE_VOLUME,
2020-11-19 19:40:30 +08:00
Looping = true,
Frequency = { Value = spinning_sample_initial_frequency }
},
maxBonusSample = new PausableSkinnableSound
{
MinimumSampleVolume = MINIMUM_SAMPLE_VOLUME,
2020-11-19 19:40:30 +08:00
}
});
PositionBindable.BindValueChanged(pos => Position = pos.NewValue);
2020-11-05 12:51:46 +08:00
}
2020-07-30 18:34:59 +08:00
protected override void LoadComplete()
{
base.LoadComplete();
isSpinning = RotationTracker.IsSpinning.GetBoundCopy();
isSpinning.BindValueChanged(updateSpinningSample);
}
2020-11-30 18:24:38 +08:00
protected override void OnFree()
2020-11-19 19:40:30 +08:00
{
2020-11-30 18:24:38 +08:00
base.OnFree();
2020-11-19 19:40:30 +08:00
spinningSample.ClearSamples();
maxBonusSample.ClearSamples();
2020-11-19 19:40:30 +08:00
}
2020-07-30 18:34:59 +08:00
protected override void LoadSamples()
{
base.LoadSamples();
spinningSample.Samples = HitObject.CreateSpinningSamples().Cast<ISampleInfo>().ToArray();
spinningSample.Frequency.Value = spinning_sample_initial_frequency;
maxBonusSample.Samples = new ISampleInfo[] { new SpinnerBonusMaxSampleInfo(HitObject.CreateHitSampleInfo()) };
2020-07-30 18:34:59 +08:00
}
private void updateSpinningSample(ValueChangedEvent<bool> tracking)
{
2020-09-29 11:45:20 +08:00
if (tracking.NewValue)
2020-07-30 18:34:59 +08:00
{
2021-01-19 16:11:40 +08:00
if (!spinningSample.IsPlaying)
spinningSample.Play();
spinningSample.VolumeTo(1, 300);
2020-07-30 18:34:59 +08:00
}
else
{
spinningSample.VolumeTo(0, fade_out_duration);
2020-07-30 18:34:59 +08:00
}
}
public override void StopAllSamples()
{
base.StopAllSamples();
spinningSample?.Stop();
maxBonusSample?.Stop();
}
protected override void AddNestedHitObject(DrawableHitObject hitObject)
{
base.AddNestedHitObject(hitObject);
switch (hitObject)
{
case DrawableSpinnerTick tick:
ticks.Add(tick);
break;
}
}
2020-11-04 15:19:07 +08:00
protected override void UpdateHitStateTransforms(ArmedState state)
{
2020-11-04 15:19:07 +08:00
base.UpdateHitStateTransforms(state);
this.FadeOut(fade_out_duration).OnComplete(_ =>
{
// looping sample should be stopped here as it is safer than running in the OnComplete
// of the volume transition above.
spinningSample.Stop();
});
Expire();
2020-07-30 18:34:59 +08: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();
2020-11-12 14:59:48 +08:00
ticks.Clear(false);
}
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 12:58:46 +08:00
}
protected override void ApplySkin(ISkinSource skin, bool allowFallback)
{
2020-08-16 02:34:17 +08:00
base.ApplySkin(skin, allowFallback);
spinnerFrequencyModulate = skin.GetConfig<OsuSkinConfiguration, bool>(OsuSkinConfiguration.SpinnerFrequencyModulate)?.Value ?? true;
2018-04-13 17:19:50 +08:00
}
2020-07-30 11:55:34 +08:00
/// <summary>
/// The completion progress of this spinner from 0..1 (clamped).
/// </summary>
public float Progress
{
get
{
2020-11-05 12:51:46 +08:00
if (HitObject.SpinsRequired == 0)
// some spinners are so short they can't require an integer spin count.
// these become implicitly hit.
return 1;
return Math.Clamp(Result.TotalRotation / 360 / HitObject.SpinsRequired, 0, 1);
}
}
2018-04-13 17:19:50 +08:00
protected override JudgementResult CreateResult(Judgement judgement) => new OsuSpinnerJudgementResult(HitObject, judgement);
protected override void CheckForResult(bool userTriggered, double timeOffset)
2018-04-13 17:19:50 +08:00
{
if (Time.Current < HitObject.StartTime) return;
if (Progress >= 1)
Result.TimeCompleted ??= Time.Current;
2018-04-13 17:19:50 +08:00
2020-11-05 12:51:46 +08:00
if (userTriggered || Time.Current < HitObject.EndTime)
return;
// Trigger a miss result for remaining ticks to avoid infinite gameplay.
foreach (var tick in ticks.Where(t => !t.Result.HasResult))
2020-07-21 18:48:44 +08:00
tick.TriggerResult(false);
ApplyResult(r =>
2018-04-13 17:19:50 +08:00
{
if (Progress >= 1)
r.Type = HitResult.Great;
2018-04-13 17:19:50 +08:00
else if (Progress > .9)
2020-09-29 16:16:55 +08:00
r.Type = HitResult.Ok;
2018-04-13 17:19:50 +08:00
else if (Progress > .75)
r.Type = HitResult.Meh;
2020-11-05 12:51:46 +08:00
else if (Time.Current >= HitObject.EndTime)
r.Type = r.Judgement.MinResult;
});
2018-04-13 17:19:50 +08:00
}
protected override void Update()
{
base.Update();
2020-07-30 18:34:59 +08:00
if (HandleUserInput)
{
bool isValidSpinningTime = Time.Current >= HitObject.StartTime && Time.Current <= HitObject.EndTime;
bool correctButtonPressed = (OsuActionInputManager?.PressedActions.Any(x => x == OsuAction.LeftButton || x == OsuAction.RightButton) ?? false);
RotationTracker.Tracking = !Result.HasResult
&& correctButtonPressed
&& isValidSpinningTime;
}
2020-07-30 18:34:59 +08:00
if (spinningSample != null && spinnerFrequencyModulate)
2020-08-16 02:44:02 +08:00
spinningSample.Frequency.Value = spinning_sample_modulated_base_frequency + Progress;
// Ticks can theoretically be judged at any point in the spinner's duration.
2023-11-02 15:18:37 +08:00
// A tick must be alive to correctly play back samples,
// but for performance reasons, we only want to keep the next tick alive.
var next = NestedHitObjects.FirstOrDefault(h => !h.Judged);
// See default `LifetimeStart` as set in `DrawableSpinnerTick`.
if (next?.LifetimeStart == double.MaxValue)
next.LifetimeStart = HitObject.StartTime;
2018-04-13 17:19:50 +08:00
}
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
if (Result.TimeStarted == null && RotationTracker.Tracking)
Result.TimeStarted = Time.Current;
// don't update after end time to avoid the rate display dropping during fade out.
// this shouldn't be limited to StartTime as it causes weirdness with the underlying calculation, which is expecting updates during that period.
if (Time.Current <= HitObject.EndTime)
spmCalculator.SetRotation(Result.TotalRotation);
updateBonusScore();
2018-04-13 17:19:50 +08:00
}
2023-12-21 13:58:23 +08:00
private static readonly int score_per_tick = new OsuScoreProcessor().GetBaseScoreForResult(new SpinnerBonusTick.OsuSpinnerBonusTickJudgement().MaxResult);
2021-03-13 10:22:20 +08:00
private void updateBonusScore()
{
if (ticks.Count == 0)
return;
int spins = (int)(Result.TotalRotation / 360);
if (spins < completedFullSpins.Value)
{
2020-07-21 18:21:30 +08:00
// rewinding, silently handle
completedFullSpins.Value = spins;
2020-07-21 18:21:30 +08:00
return;
}
while (completedFullSpins.Value != spins)
2020-07-21 18:21:30 +08:00
{
2020-10-04 00:08:24 +08:00
var tick = ticks.FirstOrDefault(t => !t.Result.HasResult);
2020-07-21 18:21:30 +08:00
// tick may be null if we've hit the spin limit.
if (tick == null)
{
// we still want to play a sound. this will probably be a new sound in the future, but for now let's continue playing the bonus sound.
// TODO: this doesn't concurrency. i can't figure out how to make it concurrency. samples are bad and need a refactor.
maxBonusSample.Play();
}
else
tick.TriggerResult(true);
2020-07-21 18:21:30 +08:00
completedFullSpins.Value++;
}
}
public class SpinnerBonusMaxSampleInfo : HitSampleInfo
{
public override IEnumerable<string> LookupNames
{
get
{
foreach (string name in base.LookupNames)
yield return name;
foreach (string name in base.LookupNames)
yield return name.Replace("-max", string.Empty);
}
}
public SpinnerBonusMaxSampleInfo(HitSampleInfo sampleInfo)
: base("spinnerbonus-max", sampleInfo.Bank, sampleInfo.Suffix, sampleInfo.Volume)
{
}
}
2018-04-13 17:19:50 +08:00
}
}