// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Linq; using osu.Framework.Bindables; using osu.Framework.Graphics; using osu.Game.Configuration; using osu.Game.Rulesets.Mods; using osu.Game.Rulesets.Objects; using osu.Game.Rulesets.Objects.Drawables; using osu.Game.Rulesets.Osu.Objects; using osu.Game.Rulesets.Osu.Objects.Drawables; using osu.Game.Rulesets.Osu.Scoring; using osu.Game.Rulesets.Osu.UI; using osu.Game.Rulesets.Scoring; using osu.Game.Rulesets.UI; namespace osu.Game.Rulesets.Osu.Mods { public class OsuModClassic : ModClassic, IApplicableToHitObject, IApplicableToDrawableHitObject, IApplicableToDrawableRuleset { public override Type[] IncompatibleMods => base.IncompatibleMods.Append(typeof(OsuModStrictTracking)).ToArray(); [SettingSource("No slider head accuracy requirement", "Scores sliders proportionally to the number of ticks hit.")] public Bindable NoSliderHeadAccuracy { get; } = new BindableBool(true); [SettingSource("No slider head movement", "Pins slider heads at their starting position, regardless of time.")] public Bindable NoSliderHeadMovement { get; } = new BindableBool(true); [SettingSource("Apply classic note lock", "Applies note lock to the full hit window.")] public Bindable ClassicNoteLock { get; } = new BindableBool(true); [SettingSource("Always play a slider's tail sample", "Always plays a slider's tail sample regardless of whether it was hit or not.")] public Bindable AlwaysPlayTailSample { get; } = new BindableBool(true); [SettingSource("Fade out hit circles earlier", "Make hit circles fade out into a miss, rather than after it.")] public Bindable FadeHitCircleEarly { get; } = new Bindable(true); private bool usingHiddenFading; public void ApplyToHitObject(HitObject hitObject) { switch (hitObject) { case Slider slider: slider.OnlyJudgeNestedObjects = !NoSliderHeadAccuracy.Value; foreach (var head in slider.NestedHitObjects.OfType()) head.JudgeAsNormalHitCircle = !NoSliderHeadAccuracy.Value; break; } } public void ApplyToDrawableRuleset(DrawableRuleset drawableRuleset) { var osuRuleset = (DrawableOsuRuleset)drawableRuleset; if (ClassicNoteLock.Value) { double hittableRange = OsuHitWindows.MISS_WINDOW - (drawableRuleset.Mods.OfType().Any() ? 200 : 0); osuRuleset.Playfield.HitPolicy = new LegacyHitPolicy(hittableRange); } usingHiddenFading = drawableRuleset.Mods.OfType().SingleOrDefault()?.OnlyFadeApproachCircles.Value == false; } public void ApplyToDrawableHitObject(DrawableHitObject obj) { switch (obj) { case DrawableSliderHead head: head.TrackFollowCircle = !NoSliderHeadMovement.Value; if (FadeHitCircleEarly.Value && !usingHiddenFading) applyEarlyFading(head); if (ClassicNoteLock.Value) blockInputToUnderlyingObjects(head); break; case DrawableSliderTail tail: tail.SamplePlaysOnlyOnHit = !AlwaysPlayTailSample.Value; break; case DrawableHitCircle circle: if (FadeHitCircleEarly.Value && !usingHiddenFading) applyEarlyFading(circle); if (ClassicNoteLock.Value) blockInputToUnderlyingObjects(circle); break; } } /// /// On stable, hitcircles that have already been hit block input from reaching objects that may be underneath them. /// The purpose of this method is to restore that behaviour. /// In order to avoid introducing yet another confusing config option, this behaviour is roped into the general notion of "note lock". /// private static void blockInputToUnderlyingObjects(DrawableHitCircle circle) { var oldHitAction = circle.HitArea.Hit; circle.HitArea.Hit = () => { oldHitAction?.Invoke(); return true; }; } private void applyEarlyFading(DrawableHitCircle circle) { circle.ApplyCustomUpdateState += (dho, state) => { using (dho.BeginAbsoluteSequence(dho.StateUpdateTime)) { if (state != ArmedState.Hit) { double okWindow = dho.HitObject.HitWindows.WindowFor(HitResult.Ok); double lateMissFadeTime = dho.HitObject.HitWindows.WindowFor(HitResult.Meh) - okWindow; dho.Delay(okWindow).FadeOut(lateMissFadeTime); } } }; } } }