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-02-11 18:03:01 +08:00
|
|
|
|
using System;
|
2020-12-03 13:28:37 +08:00
|
|
|
|
using System.Diagnostics;
|
2020-11-12 04:46:58 +08:00
|
|
|
|
using System.Linq;
|
2018-01-09 12:41:57 +08:00
|
|
|
|
using osu.Framework.Graphics;
|
2021-10-25 12:34:40 +08:00
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
|
using osu.Game.Configuration;
|
2020-12-03 13:28:37 +08:00
|
|
|
|
using osu.Game.Beatmaps;
|
2018-01-09 12:41:57 +08:00
|
|
|
|
using osu.Game.Rulesets.Mods;
|
2019-11-25 18:01:24 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
2018-01-09 12:41:57 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
|
|
|
|
using osu.Game.Rulesets.Osu.Objects.Drawables;
|
2018-06-06 12:51:51 +08:00
|
|
|
|
using osu.Game.Rulesets.Osu.Objects;
|
2021-06-20 01:06:28 +08:00
|
|
|
|
using osu.Game.Rulesets.Osu.Skinning;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-01-09 12:41:57 +08:00
|
|
|
|
namespace osu.Game.Rulesets.Osu.Mods
|
|
|
|
|
{
|
2021-06-28 23:54:21 +08:00
|
|
|
|
public class OsuModHidden : ModHidden, IHidesApproachCircles
|
2018-01-09 12:41:57 +08:00
|
|
|
|
{
|
2021-10-25 13:52:10 +08:00
|
|
|
|
[SettingSource("Only fade approach circles", "The main object body will not fade when enabled.")]
|
2021-10-26 18:56:54 +08:00
|
|
|
|
public Bindable<bool> OnlyFadeApproachCircles { get; } = new BindableBool();
|
2021-10-25 12:34:40 +08:00
|
|
|
|
|
2018-03-14 11:07:03 +08:00
|
|
|
|
public override string Description => @"Play with no approach circles and fading circles/sliders.";
|
2022-07-18 12:22:25 +08:00
|
|
|
|
public override double ScoreMultiplier => UsesDefaultConfiguration ? 1.06 : 1;
|
2019-09-18 18:36:07 +08:00
|
|
|
|
|
2021-06-28 23:55:24 +08:00
|
|
|
|
public override Type[] IncompatibleMods => new[] { typeof(IRequiresApproachCircles), typeof(OsuModSpinIn) };
|
2019-09-18 18:36:07 +08:00
|
|
|
|
|
2022-05-12 16:19:07 +08:00
|
|
|
|
public const double FADE_IN_DURATION_MULTIPLIER = 0.4;
|
|
|
|
|
public const double FADE_OUT_DURATION_MULTIPLIER = 0.3;
|
2018-04-25 16:15:53 +08:00
|
|
|
|
|
2020-12-04 05:40:30 +08:00
|
|
|
|
protected override bool IsFirstAdjustableObject(HitObject hitObject) => !(hitObject is Spinner || hitObject is SpinnerTick);
|
2020-04-16 13:11:38 +08:00
|
|
|
|
|
2020-12-03 13:28:37 +08:00
|
|
|
|
public override void ApplyToBeatmap(IBeatmap beatmap)
|
2018-06-06 12:51:51 +08:00
|
|
|
|
{
|
2020-12-03 13:28:37 +08:00
|
|
|
|
base.ApplyToBeatmap(beatmap);
|
2018-06-06 12:51:51 +08:00
|
|
|
|
|
2020-12-03 13:28:37 +08:00
|
|
|
|
foreach (var obj in beatmap.HitObjects.OfType<OsuHitObject>())
|
|
|
|
|
applyFadeInAdjustment(obj);
|
2020-11-11 15:35:48 +08:00
|
|
|
|
|
2020-12-03 13:28:37 +08:00
|
|
|
|
static void applyFadeInAdjustment(OsuHitObject osuObject)
|
|
|
|
|
{
|
2022-05-12 16:19:07 +08:00
|
|
|
|
osuObject.TimeFadeIn = osuObject.TimePreempt * FADE_IN_DURATION_MULTIPLIER;
|
2020-12-03 13:28:37 +08:00
|
|
|
|
foreach (var nested in osuObject.NestedHitObjects.OfType<OsuHitObject>())
|
|
|
|
|
applyFadeInAdjustment(nested);
|
|
|
|
|
}
|
2020-11-05 15:12:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-11-05 14:36:44 +08:00
|
|
|
|
protected override void ApplyIncreasedVisibilityState(DrawableHitObject hitObject, ArmedState state)
|
|
|
|
|
{
|
2021-10-26 10:42:15 +08:00
|
|
|
|
applyHiddenState(hitObject, true);
|
2020-11-05 14:36:44 +08:00
|
|
|
|
}
|
2020-10-09 02:07:01 +08:00
|
|
|
|
|
2020-11-05 14:36:44 +08:00
|
|
|
|
protected override void ApplyNormalVisibilityState(DrawableHitObject hitObject, ArmedState state)
|
|
|
|
|
{
|
2021-10-26 10:42:15 +08:00
|
|
|
|
applyHiddenState(hitObject, false);
|
2020-11-05 14:36:44 +08:00
|
|
|
|
}
|
2020-10-09 02:07:01 +08:00
|
|
|
|
|
2021-10-26 10:42:15 +08:00
|
|
|
|
private void applyHiddenState(DrawableHitObject drawableObject, bool increaseVisibility)
|
2018-01-09 12:41:57 +08:00
|
|
|
|
{
|
2020-12-03 13:28:37 +08:00
|
|
|
|
if (!(drawableObject is DrawableOsuHitObject drawableOsuObject))
|
2018-01-09 12:41:57 +08:00
|
|
|
|
return;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-12-03 13:28:37 +08:00
|
|
|
|
OsuHitObject hitObject = drawableOsuObject.HitObject;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-05-14 21:56:13 +08:00
|
|
|
|
(double fadeStartTime, double fadeDuration) = getFadeOutParameters(drawableOsuObject);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 10:42:15 +08:00
|
|
|
|
// process approach circle hiding first (to allow for early return below).
|
|
|
|
|
if (!increaseVisibility)
|
|
|
|
|
{
|
|
|
|
|
if (drawableObject is DrawableHitCircle circle)
|
|
|
|
|
{
|
|
|
|
|
using (circle.BeginAbsoluteSequence(hitObject.StartTime - hitObject.TimePreempt))
|
|
|
|
|
circle.ApproachCircle.Hide();
|
|
|
|
|
}
|
2021-10-26 11:16:45 +08:00
|
|
|
|
else if (drawableObject is DrawableSpinner spinner)
|
|
|
|
|
{
|
|
|
|
|
spinner.Body.OnSkinChanged += () => hideSpinnerApproachCircle(spinner);
|
|
|
|
|
hideSpinnerApproachCircle(spinner);
|
|
|
|
|
}
|
2021-10-26 10:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (OnlyFadeApproachCircles.Value)
|
|
|
|
|
return;
|
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
switch (drawableObject)
|
2018-01-09 12:41:57 +08:00
|
|
|
|
{
|
2022-06-24 20:25:23 +08:00
|
|
|
|
case DrawableSliderTail:
|
2021-10-26 05:09:22 +08:00
|
|
|
|
using (drawableObject.BeginAbsoluteSequence(fadeStartTime))
|
|
|
|
|
drawableObject.FadeOut(fadeDuration);
|
2020-10-02 17:41:28 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
break;
|
2020-10-02 17:41:28 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
case DrawableSliderRepeat sliderRepeat:
|
|
|
|
|
using (drawableObject.BeginAbsoluteSequence(fadeStartTime))
|
|
|
|
|
// only apply to circle piece – reverse arrow is not affected by hidden.
|
|
|
|
|
sliderRepeat.CirclePiece.FadeOut(fadeDuration);
|
2020-10-02 17:41:28 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
break;
|
2020-10-02 17:41:28 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
case DrawableHitCircle circle:
|
|
|
|
|
Drawable fadeTarget = circle;
|
2020-10-09 17:43:16 +08:00
|
|
|
|
|
2021-10-26 10:42:15 +08:00
|
|
|
|
if (increaseVisibility)
|
2021-10-26 05:09:22 +08:00
|
|
|
|
{
|
|
|
|
|
// only fade the circle piece (not the approach circle) for the increased visibility object.
|
|
|
|
|
fadeTarget = circle.CirclePiece;
|
|
|
|
|
}
|
2021-10-25 15:25:32 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
using (drawableObject.BeginAbsoluteSequence(fadeStartTime))
|
|
|
|
|
fadeTarget.FadeOut(fadeDuration);
|
|
|
|
|
break;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
case DrawableSlider slider:
|
|
|
|
|
using (slider.BeginAbsoluteSequence(fadeStartTime))
|
|
|
|
|
slider.Body.FadeOut(fadeDuration, Easing.Out);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
break;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
case DrawableSliderTick sliderTick:
|
|
|
|
|
using (sliderTick.BeginAbsoluteSequence(fadeStartTime))
|
|
|
|
|
sliderTick.FadeOut(fadeDuration);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
break;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
case DrawableSpinner spinner:
|
|
|
|
|
// hide elements we don't care about.
|
|
|
|
|
// todo: hide background
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
using (spinner.BeginAbsoluteSequence(fadeStartTime))
|
|
|
|
|
spinner.FadeOut(fadeDuration);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 05:09:22 +08:00
|
|
|
|
break;
|
2018-01-09 12:41:57 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-11-12 04:46:58 +08:00
|
|
|
|
|
2021-05-15 11:26:16 +08:00
|
|
|
|
private (double fadeStartTime, double fadeDuration) getFadeOutParameters(DrawableOsuHitObject drawableObject)
|
2020-11-12 04:46:58 +08:00
|
|
|
|
{
|
2020-12-03 13:28:37 +08:00
|
|
|
|
switch (drawableObject)
|
2020-11-12 04:46:58 +08:00
|
|
|
|
{
|
2020-12-03 13:28:37 +08:00
|
|
|
|
case DrawableSliderTail tail:
|
|
|
|
|
// Use the same fade sequence as the slider head.
|
|
|
|
|
Debug.Assert(tail.Slider != null);
|
|
|
|
|
return getParameters(tail.Slider.HeadCircle);
|
|
|
|
|
|
|
|
|
|
case DrawableSliderRepeat repeat:
|
|
|
|
|
// Use the same fade sequence as the slider head.
|
|
|
|
|
Debug.Assert(repeat.Slider != null);
|
|
|
|
|
return getParameters(repeat.Slider.HeadCircle);
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return getParameters(drawableObject.HitObject);
|
2020-11-12 04:46:58 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-05-15 11:26:16 +08:00
|
|
|
|
static (double fadeStartTime, double fadeDuration) getParameters(OsuHitObject hitObject)
|
2020-12-03 13:28:37 +08:00
|
|
|
|
{
|
2021-10-27 12:04:41 +08:00
|
|
|
|
double fadeOutStartTime = hitObject.StartTime - hitObject.TimePreempt + hitObject.TimeFadeIn;
|
2022-05-12 16:19:07 +08:00
|
|
|
|
double fadeOutDuration = hitObject.TimePreempt * FADE_OUT_DURATION_MULTIPLIER;
|
2020-12-03 13:28:37 +08:00
|
|
|
|
|
|
|
|
|
// new duration from completed fade in to end (before fading out)
|
2021-10-27 12:04:41 +08:00
|
|
|
|
double longFadeDuration = hitObject.GetEndTime() - fadeOutStartTime;
|
2020-12-03 13:28:37 +08:00
|
|
|
|
|
|
|
|
|
switch (hitObject)
|
|
|
|
|
{
|
2022-06-24 20:25:23 +08:00
|
|
|
|
case Slider:
|
2020-12-03 13:28:37 +08:00
|
|
|
|
return (fadeOutStartTime, longFadeDuration);
|
|
|
|
|
|
2022-06-24 20:25:23 +08:00
|
|
|
|
case SliderTick:
|
2021-10-27 12:04:41 +08:00
|
|
|
|
double tickFadeOutDuration = Math.Min(hitObject.TimePreempt - DrawableSliderTick.ANIM_DURATION, 1000);
|
2020-12-03 13:28:37 +08:00
|
|
|
|
return (hitObject.StartTime - tickFadeOutDuration, tickFadeOutDuration);
|
|
|
|
|
|
2022-06-24 20:25:23 +08:00
|
|
|
|
case Spinner:
|
2020-12-03 13:28:37 +08:00
|
|
|
|
return (fadeOutStartTime + longFadeDuration, fadeOutDuration);
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return (fadeOutStartTime, fadeOutDuration);
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-11-12 04:46:58 +08:00
|
|
|
|
}
|
2021-06-20 01:06:28 +08:00
|
|
|
|
|
|
|
|
|
private static void hideSpinnerApproachCircle(DrawableSpinner spinner)
|
|
|
|
|
{
|
2021-06-21 08:43:11 +08:00
|
|
|
|
var approachCircle = (spinner.Body.Drawable as IHasApproachCircle)?.ApproachCircle;
|
2021-06-20 01:06:28 +08:00
|
|
|
|
if (approachCircle == null)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
using (spinner.BeginAbsoluteSequence(spinner.HitObject.StartTime - spinner.HitObject.TimePreempt))
|
|
|
|
|
approachCircle.Hide();
|
|
|
|
|
}
|
2018-01-09 12:41:57 +08:00
|
|
|
|
}
|
|
|
|
|
}
|