2019-01-24 16:43:03 +08: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 17:19:50 +08:00
|
|
|
|
|
2018-01-09 12:41:57 +08:00
|
|
|
|
using System;
|
2020-02-10 14:14:04 +08:00
|
|
|
|
using osu.Framework.Graphics;
|
2019-03-27 18:29:27 +08:00
|
|
|
|
using osu.Framework.Graphics.Sprites;
|
2022-08-11 04:09:11 +08:00
|
|
|
|
using osu.Framework.Localisation;
|
2020-03-04 13:51:12 +08:00
|
|
|
|
using osu.Framework.Utils;
|
2018-01-09 12:41:57 +08:00
|
|
|
|
using osu.Game.Graphics;
|
|
|
|
|
using osu.Game.Rulesets.Mods;
|
2020-02-08 08:59:35 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
|
|
|
|
using osu.Game.Rulesets.Osu.Objects.Drawables;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-01-09 12:41:57 +08:00
|
|
|
|
namespace osu.Game.Rulesets.Osu.Mods
|
|
|
|
|
{
|
2021-06-16 17:52:01 +08:00
|
|
|
|
public class OsuModSpunOut : Mod, IApplicableToDrawableHitObject
|
2018-01-09 12:41:57 +08:00
|
|
|
|
{
|
|
|
|
|
public override string Name => "Spun Out";
|
2018-11-30 16:16:00 +08:00
|
|
|
|
public override string Acronym => "SO";
|
2021-12-10 13:15:00 +08:00
|
|
|
|
public override IconUsage? Icon => OsuIcon.ModSpunOut;
|
2020-02-09 13:33:43 +08:00
|
|
|
|
public override ModType Type => ModType.Automation;
|
2022-08-11 04:09:11 +08:00
|
|
|
|
public override LocalisableString Description => @"Spinners will be automatically completed.";
|
2018-01-09 12:41:57 +08:00
|
|
|
|
public override double ScoreMultiplier => 0.9;
|
2022-11-01 18:47:20 +08:00
|
|
|
|
public override Type[] IncompatibleMods => new[] { typeof(ModAutoplay), typeof(OsuModAutopilot), typeof(OsuModTargetPractice) };
|
2024-01-31 21:59:35 +08:00
|
|
|
|
public override bool Ranked => UsesDefaultConfiguration;
|
2020-02-08 08:59:35 +08:00
|
|
|
|
|
2021-06-16 17:52:01 +08:00
|
|
|
|
public void ApplyToDrawableHitObject(DrawableHitObject hitObject)
|
2020-02-08 08:59:35 +08:00
|
|
|
|
{
|
2021-06-16 17:52:01 +08:00
|
|
|
|
if (hitObject is DrawableSpinner spinner)
|
2020-02-08 08:59:35 +08:00
|
|
|
|
{
|
2021-06-16 17:52:01 +08:00
|
|
|
|
spinner.HandleUserInput = false;
|
|
|
|
|
spinner.OnUpdate += onSpinnerUpdate;
|
2020-02-08 08:59:35 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-02-11 04:42:34 +08:00
|
|
|
|
|
2020-03-29 13:31:03 +08:00
|
|
|
|
private void onSpinnerUpdate(Drawable drawable)
|
2020-02-11 04:42:34 +08:00
|
|
|
|
{
|
2020-03-29 13:31:03 +08:00
|
|
|
|
var spinner = (DrawableSpinner)drawable;
|
|
|
|
|
|
2020-07-29 19:01:01 +08:00
|
|
|
|
spinner.RotationTracker.Tracking = true;
|
2020-08-12 04:04:18 +08:00
|
|
|
|
|
2020-08-12 04:34:46 +08:00
|
|
|
|
// early-return if we were paused to avoid division-by-zero in the subsequent calculations.
|
|
|
|
|
if (Precision.AlmostEquals(spinner.Clock.Rate, 0))
|
|
|
|
|
return;
|
|
|
|
|
|
2020-08-12 04:04:18 +08:00
|
|
|
|
// because the spinner is under the gameplay clock, it is affected by rate adjustments on the track;
|
|
|
|
|
// for that reason using ElapsedFrameTime directly leads to fewer SPM with Half Time and more SPM with Double Time.
|
|
|
|
|
// for spinners we want the real (wall clock) elapsed time; to achieve that, unapply the clock rate locally here.
|
2021-10-27 12:04:41 +08:00
|
|
|
|
double rateIndependentElapsedTime = spinner.Clock.ElapsedFrameTime / spinner.Clock.Rate;
|
2022-02-14 03:28:40 +08:00
|
|
|
|
|
|
|
|
|
// multiply the SPM by 1.01 to ensure that the spinner is completed. if the calculation is left exact,
|
|
|
|
|
// some spinners may not complete due to very minor decimal loss during calculation
|
|
|
|
|
float rotationSpeed = (float)(1.01 * spinner.HitObject.SpinsRequired / spinner.HitObject.Duration);
|
2024-03-06 10:42:20 +08:00
|
|
|
|
spinner.RotationTracker.AddRotation(float.RadiansToDegrees((float)rateIndependentElapsedTime * rotationSpeed * MathF.PI * 2.0f));
|
2020-02-11 04:42:34 +08:00
|
|
|
|
}
|
2018-01-09 12:41:57 +08:00
|
|
|
|
}
|
|
|
|
|
}
|