1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-15 18:33:22 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Mods/OsuModAutopilot.cs

80 lines
3.2 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;
2019-05-12 19:00:43 +08:00
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Graphics.Sprites;
2019-05-12 19:00:43 +08:00
using osu.Framework.Input.StateChanges;
2022-08-11 04:09:11 +08:00
using osu.Framework.Localisation;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics;
using osu.Game.Rulesets.Mods;
2019-05-12 19:00:43 +08:00
using osu.Game.Rulesets.Osu.Objects;
using osu.Game.Rulesets.Osu.Replays;
using osu.Game.Rulesets.Osu.UI;
2019-05-12 19:00:43 +08:00
using osu.Game.Rulesets.UI;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Osu.Mods
{
2019-05-12 19:00:43 +08:00
public class OsuModAutopilot : Mod, IApplicableFailOverride, IUpdatableByPlayfield, IApplicableToDrawableRuleset<OsuHitObject>
2018-04-13 17:19:50 +08:00
{
public override string Name => "Autopilot";
public override string Acronym => "AP";
2020-01-14 21:22:00 +08:00
public override IconUsage? Icon => OsuIcon.ModAutopilot;
public override ModType Type => ModType.Automation;
2022-08-11 04:09:11 +08:00
public override LocalisableString Description => @"Automatic cursor movement - just follow the rhythm.";
public override double ScoreMultiplier => 0.1;
public override Type[] IncompatibleMods => new[]
{
typeof(OsuModSpunOut),
typeof(ModRelax),
typeof(ModFailCondition),
typeof(ModNoFail),
typeof(ModAutoplay),
typeof(OsuModMagnetised),
typeof(OsuModRepel)
};
2019-05-12 19:00:43 +08:00
public bool PerformFail() => false;
2019-09-19 00:45:42 +08:00
public bool RestartOnFail => false;
2019-05-12 19:00:43 +08:00
2022-07-31 21:43:16 +08:00
private OsuInputManager inputManager = null!;
2019-05-12 19:00:43 +08:00
2022-07-31 21:43:16 +08:00
private List<OsuReplayFrame> replayFrames = null!;
private int currentFrame = -1;
2019-05-12 19:00:43 +08:00
public void Update(Playfield playfield)
{
if (currentFrame == replayFrames.Count - 1) return;
double time = playfield.Clock.CurrentTime;
2019-05-12 19:00:43 +08:00
// Very naive implementation of autopilot based on proximity to replay frames.
// Special case for the first frame is required to ensure the mouse is in a sane position until the actual time of the first frame is hit.
// TODO: this needs to be based on user interactions to better match stable (pausing until judgement is registered).
if (currentFrame < 0 || Math.Abs(replayFrames[currentFrame + 1].Time - time) <= Math.Abs(replayFrames[currentFrame].Time - time))
2019-05-12 19:00:43 +08:00
{
currentFrame++;
new MousePositionAbsoluteInput { Position = playfield.ToScreenSpace(replayFrames[currentFrame].Position) }.Apply(inputManager.CurrentState, inputManager);
2019-05-12 19:00:43 +08:00
}
// TODO: Implement the functionality to automatically spin spinners
}
public void ApplyToDrawableRuleset(DrawableRuleset<OsuHitObject> drawableRuleset)
{
// Grab the input manager to disable the user's cursor, and for future use
inputManager = ((DrawableOsuRuleset)drawableRuleset).KeyBindingInputManager;
2019-05-12 19:00:43 +08:00
inputManager.AllowUserCursorMovement = false;
// Generate the replay frames the cursor should follow
replayFrames = new OsuAutoGenerator(drawableRuleset.Beatmap, drawableRuleset.Mods).Generate().Frames.Cast<OsuReplayFrame>().ToList();
drawableRuleset.UseResumeOverlay = false;
2019-05-12 19:00:43 +08:00
}
2018-04-13 17:19:50 +08:00
}
}