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