1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-13 03:13:21 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Mods/OsuModSpunOut.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

58 lines
2.7 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
using System;
2020-02-10 14:14:04 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Sprites;
2022-08-11 04:09:11 +08:00
using osu.Framework.Localisation;
using osu.Framework.Utils;
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
namespace osu.Game.Rulesets.Osu.Mods
{
public class OsuModSpunOut : Mod, IApplicableToDrawableHitObject
{
public override string Name => "Spun Out";
public override string Acronym => "SO";
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.";
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
public void ApplyToDrawableHitObject(DrawableHitObject hitObject)
2020-02-08 08:59:35 +08:00
{
if (hitObject is DrawableSpinner spinner)
2020-02-08 08:59:35 +08:00
{
spinner.HandleUserInput = false;
spinner.OnUpdate += onSpinnerUpdate;
2020-02-08 08:59:35 +08:00
}
}
2020-02-11 04:42:34 +08:00
private void onSpinnerUpdate(Drawable drawable)
2020-02-11 04:42:34 +08:00
{
var spinner = (DrawableSpinner)drawable;
spinner.RotationTracker.Tracking = true;
// early-return if we were paused to avoid division-by-zero in the subsequent calculations.
if (Precision.AlmostEquals(spinner.Clock.Rate, 0))
return;
// 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.
double rateIndependentElapsedTime = spinner.Clock.ElapsedFrameTime / spinner.Clock.Rate;
// 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
}
}
}