1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-10 20:32:58 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Mods/OsuModRelax.cs

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

208 lines
7.0 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;
2018-08-03 20:03:11 +08:00
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
2022-08-11 04:09:11 +08:00
using osu.Framework.Localisation;
using osu.Game.Rulesets.Mods;
2018-08-03 20:03:11 +08:00
using osu.Game.Rulesets.Objects.Types;
2018-08-16 17:18:15 +08:00
using osu.Game.Rulesets.Osu.Objects;
2018-08-03 20:03:11 +08:00
using osu.Game.Rulesets.Osu.Objects.Drawables;
using osu.Game.Rulesets.Osu.UI;
2020-02-14 16:13:50 +08:00
using osu.Game.Rulesets.Replays;
2018-08-03 20:03:11 +08:00
using osu.Game.Rulesets.UI;
using osu.Game.Screens.Play;
2018-08-03 20:03:11 +08:00
using static osu.Game.Input.Handlers.ReplayInputHandler;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Osu.Mods
{
public class OsuModRelax : ModRelax, IUpdatableByPlayfield, IApplicableToDrawableRuleset<OsuHitObject>, IApplicableToPlayer, IHasNoTimedInputs
{
2022-08-11 04:09:11 +08:00
public override LocalisableString Description => @"You don't need to click. Give your clicking/tapping fingers a break from the heat of things.";
2022-10-10 14:42:08 +08:00
public override Type[] IncompatibleMods =>
base.IncompatibleMods.Concat(new[] { typeof(OsuModAutopilot), typeof(OsuModMagnetised), typeof(OsuModAlternate), typeof(OsuModSingleTap) }).ToArray();
2018-08-03 20:03:11 +08:00
/// <summary>
/// How early before a hitobject's start time to trigger a hit.
/// </summary>
private const float relax_leniency = 3;
2020-02-14 16:13:50 +08:00
private bool isDownState;
2020-02-14 16:03:23 +08:00
private bool wasLeft;
2022-07-31 21:43:16 +08:00
private OsuInputManager osuInputManager = null!;
2020-02-14 16:03:23 +08:00
2022-07-31 21:43:16 +08:00
private ReplayState<OsuAction> state = null!;
2020-02-14 16:13:50 +08:00
private double lastStateChangeTime;
private DrawableOsuRuleset ruleset = null!;
private IPressHandler pressHandler = null!;
private bool hasReplay;
private bool legacyReplay;
2020-02-14 16:03:23 +08:00
public void ApplyToDrawableRuleset(DrawableRuleset<OsuHitObject> drawableRuleset)
{
ruleset = (DrawableOsuRuleset)drawableRuleset;
2020-02-14 16:03:23 +08:00
// grab the input manager for future use.
osuInputManager = ruleset.KeyBindingInputManager;
}
public void ApplyToPlayer(Player player)
{
if (osuInputManager.ReplayInputHandler != null)
{
hasReplay = true;
Debug.Assert(ruleset.ReplayScore != null);
legacyReplay = ruleset.ReplayScore.ScoreInfo.IsLegacyScore;
pressHandler = legacyReplay ? new LegacyReplayPressHandler(this) : new PressHandler(this);
return;
}
2020-04-21 15:06:40 +08:00
pressHandler = new PressHandler(this);
2022-10-10 14:42:08 +08:00
osuInputManager.AllowGameplayInputs = false;
2020-02-14 16:03:23 +08:00
}
public void Update(Playfield playfield)
2018-08-03 20:03:11 +08:00
{
if (hasReplay && !legacyReplay)
return;
2018-08-16 17:18:15 +08:00
bool requiresHold = false;
bool requiresHit = false;
double time = playfield.Clock.CurrentTime;
2018-08-03 20:03:11 +08:00
foreach (var h in playfield.HitObjectContainer.AliveObjects.OfType<DrawableOsuHitObject>())
{
// we are not yet close enough to the object.
if (time < h.HitObject.StartTime - relax_leniency)
break;
// already hit or beyond the hittable end time.
2020-05-27 11:38:39 +08:00
if (h.IsHit || (h.HitObject is IHasDuration hasEnd && time > hasEnd.EndTime))
continue;
2018-08-03 20:03:11 +08:00
switch (h)
{
case DrawableHitCircle circle:
handleHitCircle(circle);
break;
2018-08-03 20:03:11 +08:00
case DrawableSlider slider:
// Handles cases like "2B" beatmaps, where sliders may be overlapping and simply holding is not enough.
if (!slider.HeadCircle.IsHit)
handleHitCircle(slider.HeadCircle);
requiresHold |= slider.SliderInputManager.IsMouseInFollowArea(slider.Tracking.Value);
break;
2018-08-03 20:03:11 +08:00
2022-02-12 08:51:09 +08:00
case DrawableSpinner spinner:
requiresHold |= spinner.HitObject.SpinsRequired > 0;
break;
}
2018-08-03 20:03:11 +08:00
}
2018-08-16 17:18:15 +08:00
if (requiresHit)
2018-08-03 20:03:11 +08:00
{
2020-02-14 16:13:50 +08:00
changeState(false);
changeState(true);
2018-08-03 20:03:11 +08:00
}
2018-08-16 17:18:15 +08:00
2020-02-14 16:13:50 +08:00
if (requiresHold)
changeState(true);
else if (isDownState && time - lastStateChangeTime > AutoGenerator.KEY_UP_DELAY)
changeState(false);
void handleHitCircle(DrawableHitCircle circle)
{
2020-03-31 14:17:27 +08:00
if (!circle.HitArea.IsHovered)
return;
Debug.Assert(circle.HitObject.HitWindows != null);
requiresHit |= circle.HitObject.HitWindows.CanBeHit(time - circle.HitObject.StartTime);
}
2018-08-03 20:03:11 +08:00
2020-02-14 16:13:50 +08:00
void changeState(bool down)
{
if (isDownState == down)
return;
2018-08-16 17:18:15 +08:00
2020-02-14 16:13:50 +08:00
isDownState = down;
lastStateChangeTime = time;
2018-08-03 20:03:11 +08:00
2020-02-14 16:13:50 +08:00
state = new ReplayState<OsuAction>
{
PressedActions = new List<OsuAction>()
};
2018-08-16 17:18:15 +08:00
2020-02-14 16:13:50 +08:00
if (down)
{
pressHandler.HandlePress(wasLeft);
2020-02-14 16:13:50 +08:00
wasLeft = !wasLeft;
}
else
{
pressHandler.HandleRelease(wasLeft);
}
}
}
2018-08-16 17:18:15 +08:00
private interface IPressHandler
{
void HandlePress(bool wasLeft);
void HandleRelease(bool wasLeft);
}
private class PressHandler : IPressHandler
{
private readonly OsuModRelax mod;
public PressHandler(OsuModRelax mod)
{
this.mod = mod;
}
public void HandlePress(bool wasLeft)
{
mod.state.PressedActions.Add(wasLeft ? OsuAction.LeftButton : OsuAction.RightButton);
mod.state.Apply(mod.osuInputManager.CurrentState, mod.osuInputManager);
}
public void HandleRelease(bool wasLeft)
{
mod.state.Apply(mod.osuInputManager.CurrentState, mod.osuInputManager);
}
}
// legacy replays do not contain key-presses with Relax mod, so they need to be triggered by themselves.
private class LegacyReplayPressHandler : IPressHandler
{
private readonly OsuModRelax mod;
public LegacyReplayPressHandler(OsuModRelax mod)
{
this.mod = mod;
}
public void HandlePress(bool wasLeft)
{
mod.osuInputManager.KeyBindingContainer.TriggerPressed(wasLeft ? OsuAction.LeftButton : OsuAction.RightButton);
}
public void HandleRelease(bool wasLeft)
{
// this intentionally releases right when `wasLeft` is true because `wasLeft` is set at point of press and not at point of release
mod.osuInputManager.KeyBindingContainer.TriggerReleased(wasLeft ? OsuAction.RightButton : OsuAction.LeftButton);
2020-02-14 16:13:50 +08:00
}
2018-08-16 17:18:15 +08:00
}
}
}