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/OsuRuleset.cs

387 lines
14 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-04-13 17:19:50 +08:00
using System.Collections.Generic;
using System.Linq;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Sprites;
2018-04-13 17:19:50 +08:00
using osu.Framework.Input.Bindings;
using osu.Framework.Localisation;
using osu.Game.Beatmaps;
2018-04-13 21:41:35 +08:00
using osu.Game.Beatmaps.Legacy;
2019-01-23 18:46:53 +08:00
using osu.Game.Configuration;
using osu.Game.Graphics;
using osu.Game.Overlays.Settings;
2019-01-23 18:46:53 +08:00
using osu.Game.Rulesets.Configuration;
using osu.Game.Rulesets.Difficulty;
using osu.Game.Rulesets.Edit;
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Osu.Beatmaps;
2019-01-23 18:46:53 +08:00
using osu.Game.Rulesets.Osu.Configuration;
2018-05-15 16:36:29 +08:00
using osu.Game.Rulesets.Osu.Difficulty;
using osu.Game.Rulesets.Osu.Edit;
2021-08-22 22:49:27 +08:00
using osu.Game.Rulesets.Osu.Edit.Setup;
using osu.Game.Rulesets.Osu.Mods;
using osu.Game.Rulesets.Osu.Objects;
using osu.Game.Rulesets.Osu.Replays;
using osu.Game.Rulesets.Osu.Scoring;
using osu.Game.Rulesets.Osu.Skinning.Argon;
2024-01-25 18:14:04 +08:00
using osu.Game.Rulesets.Osu.Skinning.Default;
using osu.Game.Rulesets.Osu.Skinning.Legacy;
using osu.Game.Rulesets.Osu.Statistics;
using osu.Game.Rulesets.Osu.UI;
using osu.Game.Rulesets.Replays.Types;
using osu.Game.Rulesets.Scoring;
using osu.Game.Rulesets.Scoring.Legacy;
using osu.Game.Rulesets.UI;
using osu.Game.Scoring;
2021-08-22 22:49:27 +08:00
using osu.Game.Screens.Edit.Setup;
using osu.Game.Screens.Ranking.Statistics;
using osu.Game.Skinning;
using osuTK;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Osu
{
2019-12-24 12:48:27 +08:00
public class OsuRuleset : Ruleset, ILegacyRuleset
2018-04-13 17:19:50 +08:00
{
public override DrawableRuleset CreateDrawableRulesetWith(IBeatmap beatmap, IReadOnlyList<Mod>? mods = null) => new DrawableOsuRuleset(this, beatmap, mods);
2019-12-17 19:08:13 +08:00
public override ScoreProcessor CreateScoreProcessor() => new OsuScoreProcessor();
public override HealthProcessor CreateHealthProcessor(double drainStartTime) => new OsuHealthProcessor(drainStartTime);
public override IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap) => new OsuBeatmapConverter(beatmap, this);
2019-12-17 19:08:13 +08:00
public override IBeatmapProcessor CreateBeatmapProcessor(IBeatmap beatmap) => new OsuBeatmapProcessor(beatmap);
2018-04-13 17:19:50 +08:00
public const string SHORT_NAME = "osu";
public override string RulesetAPIVersionSupported => CURRENT_RULESET_API_VERSION;
2022-08-22 15:10:55 +08:00
2018-04-13 17:19:50 +08:00
public override IEnumerable<KeyBinding> GetDefaultKeyBindings(int variant = 0) => new[]
{
new KeyBinding(InputKey.Z, OsuAction.LeftButton),
new KeyBinding(InputKey.X, OsuAction.RightButton),
2022-09-19 03:08:34 +08:00
new KeyBinding(InputKey.C, OsuAction.Smoke),
2018-04-13 17:19:50 +08:00
new KeyBinding(InputKey.MouseLeft, OsuAction.LeftButton),
new KeyBinding(InputKey.MouseRight, OsuAction.RightButton),
};
public override IEnumerable<Mod> ConvertFromLegacyMods(LegacyMods mods)
2018-04-13 21:41:35 +08:00
{
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Nightcore))
2018-04-13 21:41:35 +08:00
yield return new OsuModNightcore();
2024-07-02 23:19:04 +08:00
else if (mods.HasFlag(LegacyMods.DoubleTime))
2018-04-13 21:41:35 +08:00
yield return new OsuModDoubleTime();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Perfect))
yield return new OsuModPerfect();
2024-07-02 23:19:04 +08:00
else if (mods.HasFlag(LegacyMods.SuddenDeath))
yield return new OsuModSuddenDeath();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Autopilot))
2018-04-13 21:41:35 +08:00
yield return new OsuModAutopilot();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Cinema))
2019-11-24 01:32:16 +08:00
yield return new OsuModCinema();
2024-07-02 23:19:04 +08:00
else if (mods.HasFlag(LegacyMods.Autoplay))
2018-04-13 21:41:35 +08:00
yield return new OsuModAutoplay();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Easy))
2018-04-13 21:41:35 +08:00
yield return new OsuModEasy();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Flashlight))
2018-04-13 21:41:35 +08:00
yield return new OsuModFlashlight();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.HalfTime))
2018-04-13 21:41:35 +08:00
yield return new OsuModHalfTime();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.HardRock))
2018-04-13 21:41:35 +08:00
yield return new OsuModHardRock();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Hidden))
2018-04-13 21:41:35 +08:00
yield return new OsuModHidden();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.NoFail))
2018-04-13 21:41:35 +08:00
yield return new OsuModNoFail();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Relax))
2018-04-13 21:41:35 +08:00
yield return new OsuModRelax();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.SpunOut))
2018-04-13 21:41:35 +08:00
yield return new OsuModSpunOut();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Target))
2022-11-01 18:47:20 +08:00
yield return new OsuModTargetPractice();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.TouchDevice))
yield return new OsuModTouchDevice();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.ScoreV2))
yield return new ModScoreV2();
2018-04-13 21:41:35 +08:00
}
2020-11-15 22:40:31 +08:00
public override LegacyMods ConvertToLegacyMods(Mod[] mods)
{
var value = base.ConvertToLegacyMods(mods);
foreach (var mod in mods)
{
switch (mod)
{
2022-06-24 20:25:23 +08:00
case OsuModAutopilot:
2020-11-15 22:40:31 +08:00
value |= LegacyMods.Autopilot;
break;
2022-06-24 20:25:23 +08:00
case OsuModSpunOut:
2020-11-15 22:40:31 +08:00
value |= LegacyMods.SpunOut;
break;
2022-11-01 18:47:20 +08:00
case OsuModTargetPractice:
2020-11-15 22:40:31 +08:00
value |= LegacyMods.Target;
break;
2022-06-24 20:25:23 +08:00
case OsuModTouchDevice:
2020-11-15 22:40:31 +08:00
value |= LegacyMods.TouchDevice;
break;
}
}
return value;
}
2018-04-13 17:19:50 +08:00
public override IEnumerable<Mod> GetModsFor(ModType type)
{
switch (type)
{
case ModType.DifficultyReduction:
return new Mod[]
{
new OsuModEasy(),
new OsuModNoFail(),
2018-06-06 13:07:50 +08:00
new MultiMod(new OsuModHalfTime(), new OsuModDaycore()),
2018-04-13 17:19:50 +08:00
};
2019-04-01 11:44:46 +08:00
2018-04-13 17:19:50 +08:00
case ModType.DifficultyIncrease:
return new Mod[]
{
new OsuModHardRock(),
2018-06-06 13:07:50 +08:00
new MultiMod(new OsuModSuddenDeath(), new OsuModPerfect()),
new MultiMod(new OsuModDoubleTime(), new OsuModNightcore()),
new OsuModHidden(),
2018-08-05 20:20:56 +08:00
new MultiMod(new OsuModFlashlight(), new OsuModBlinds()),
2022-05-24 22:23:44 +08:00
new OsuModStrictTracking(),
2023-01-17 23:22:58 +08:00
new OsuModAccuracyChallenge(),
2018-04-13 17:19:50 +08:00
};
2019-04-01 11:44:46 +08:00
case ModType.Conversion:
return new Mod[]
{
2022-11-01 18:47:20 +08:00
new OsuModTargetPractice(),
2019-12-20 18:30:23 +08:00
new OsuModDifficultyAdjust(),
2021-04-25 07:19:06 +08:00
new OsuModClassic(),
new OsuModRandom(),
2021-07-26 08:40:50 +08:00
new OsuModMirror(),
2022-07-13 06:07:26 +08:00
new MultiMod(new OsuModAlternate(), new OsuModSingleTap())
};
2019-04-01 11:44:46 +08:00
case ModType.Automation:
2018-04-13 17:19:50 +08:00
return new Mod[]
{
2019-11-24 01:32:16 +08:00
new MultiMod(new OsuModAutoplay(), new OsuModCinema()),
2018-04-13 17:19:50 +08:00
new OsuModRelax(),
new OsuModAutopilot(),
2020-02-09 13:34:35 +08:00
new OsuModSpunOut(),
2018-04-13 17:19:50 +08:00
};
2019-04-01 11:44:46 +08:00
2018-08-04 06:30:46 +08:00
case ModType.Fun:
2019-02-28 12:31:40 +08:00
return new Mod[]
{
2018-08-23 03:19:28 +08:00
new OsuModTransform(),
new OsuModWiggle(),
2018-09-21 07:06:37 +08:00
new OsuModSpinIn(),
2019-07-01 19:53:40 +08:00
new MultiMod(new OsuModGrow(), new OsuModDeflate()),
new MultiMod(new ModWindUp(), new ModWindDown()),
new OsuModTraceable(),
2021-04-14 15:52:29 +08:00
new OsuModBarrelRoll(),
2021-06-16 14:09:42 +08:00
new OsuModApproachDifferent(),
2021-07-28 18:21:08 +08:00
new OsuModMuted(),
2021-10-15 12:22:57 +08:00
new OsuModNoScope(),
2022-05-26 12:08:00 +08:00
new MultiMod(new OsuModMagnetised(), new OsuModRepel()),
2022-09-08 06:05:48 +08:00
new ModAdaptiveSpeed(),
2022-12-26 04:32:47 +08:00
new OsuModFreezeFrame(),
new OsuModBubbles(),
2023-12-03 09:58:17 +08:00
new OsuModSynesthesia(),
new OsuModDepth(),
new OsuModBloom()
};
2019-04-01 11:44:46 +08:00
case ModType.System:
return new Mod[]
{
new OsuModTouchDevice(),
new ModScoreV2(),
};
2018-04-13 17:19:50 +08:00
default:
2019-11-28 21:41:29 +08:00
return Array.Empty<Mod>();
2018-04-13 17:19:50 +08:00
}
}
public override Drawable CreateIcon() => new SpriteIcon { Icon = OsuIcon.RulesetOsu };
2018-04-13 17:19:50 +08:00
public override DifficultyCalculator CreateDifficultyCalculator(IWorkingBeatmap beatmap) => new OsuDifficultyCalculator(RulesetInfo, beatmap);
2018-04-13 17:19:50 +08:00
2022-03-15 11:37:39 +08:00
public override PerformanceCalculator CreatePerformanceCalculator() => new OsuPerformanceCalculator();
2018-04-13 17:19:50 +08:00
public override HitObjectComposer CreateHitObjectComposer() => new OsuHitObjectComposer(this);
2021-04-13 16:40:56 +08:00
public override IBeatmapVerifier CreateBeatmapVerifier() => new OsuBeatmapVerifier();
2021-04-07 20:36:43 +08:00
2018-04-13 17:19:50 +08:00
public override string Description => "osu!";
public override string ShortName => SHORT_NAME;
2018-04-13 17:19:50 +08:00
public override string PlayingVerb => "Clicking circles";
2019-01-23 18:46:53 +08:00
public override RulesetSettingsSubsection CreateSettings() => new OsuSettingsSubsection(this);
2018-04-13 17:19:50 +08:00
public override ISkin? CreateSkinTransformer(ISkin skin, IBeatmap beatmap)
{
switch (skin)
{
case LegacySkin:
return new OsuLegacySkinTransformer(skin);
case ArgonSkin:
return new OsuArgonSkinTransformer(skin);
2024-01-25 18:14:04 +08:00
case TrianglesSkin:
return new OsuTrianglesSkinTransformer(skin);
}
return null;
}
2019-12-24 12:48:27 +08:00
public int LegacyID => 0;
2018-04-13 17:19:50 +08:00
public ILegacyScoreSimulator CreateLegacyScoreSimulator() => new OsuLegacyScoreSimulator();
2018-04-13 17:19:50 +08:00
public override IConvertibleReplayFrame CreateConvertibleReplayFrame() => new OsuReplayFrame();
public override IRulesetConfigManager CreateConfig(SettingsStore? settings) => new OsuRulesetConfigManager(settings, RulesetInfo);
protected override IEnumerable<HitResult> GetValidHitResults()
{
return new[]
{
HitResult.Great,
HitResult.Ok,
HitResult.Meh,
HitResult.LargeTickHit,
HitResult.SmallTickHit,
HitResult.SliderTailHit,
HitResult.SmallBonus,
HitResult.LargeBonus,
};
}
2022-08-15 02:54:02 +08:00
public override LocalisableString GetDisplayNameForHitResult(HitResult result)
{
switch (result)
{
case HitResult.LargeTickHit:
return "slider tick";
case HitResult.SliderTailHit:
case HitResult.SmallTickHit:
return "slider end";
case HitResult.SmallBonus:
return "spinner spin";
case HitResult.LargeBonus:
return "spinner bonus";
}
return base.GetDisplayNameForHitResult(result);
}
public override StatisticItem[] CreateStatisticsForScore(ScoreInfo score, IBeatmap playableBeatmap)
{
var timedHitEvents = score.HitEvents.Where(e => e.HitObject is HitCircle && !(e.HitObject is SliderTailCircle)).ToList();
2020-08-28 02:07:30 +08:00
return new[]
{
new StatisticItem("Performance Breakdown", () => new PerformanceBreakdownChart(score, playableBeatmap)
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y
}),
new StatisticItem("Timing Distribution", () => new HitEventTimingDistributionGraph(timedHitEvents)
{
RelativeSizeAxes = Axes.X,
Height = 250
}, true),
new StatisticItem("Accuracy Heatmap", () => new AccuracyHeatmap(score, playableBeatmap)
{
RelativeSizeAxes = Axes.X,
Height = 250
}, true),
new StatisticItem("Statistics", () => new SimpleStatisticTable(2, new SimpleStatisticItem[]
2020-08-28 02:46:49 +08:00
{
new AverageHitError(timedHitEvents),
new UnstableRate(timedHitEvents)
}), true)
};
}
2021-08-22 22:49:27 +08:00
public override IEnumerable<Drawable> CreateEditorSetupSections() =>
[
new MetadataSection(),
new OsuDifficultySection(),
new FillFlowContainer
{
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Spacing = new Vector2(SetupScreen.SPACING),
Children = new Drawable[]
{
new ResourcesSection
{
RelativeSizeAxes = Axes.X,
},
new ColoursSection
{
RelativeSizeAxes = Axes.X,
}
}
},
new DesignSection(),
];
/// <seealso cref="OsuHitObject.ApplyDefaultsToSelf"/>
/// <seealso cref="OsuHitWindows"/>
public override BeatmapDifficulty GetRateAdjustedDisplayDifficulty(IBeatmapDifficultyInfo difficulty, double rate)
{
BeatmapDifficulty adjustedDifficulty = new BeatmapDifficulty(difficulty);
double preempt = IBeatmapDifficultyInfo.DifficultyRange(adjustedDifficulty.ApproachRate, OsuHitObject.PREEMPT_MAX, OsuHitObject.PREEMPT_MID, OsuHitObject.PREEMPT_MIN);
2023-11-10 00:23:53 +08:00
preempt /= rate;
adjustedDifficulty.ApproachRate = (float)IBeatmapDifficultyInfo.InverseDifficultyRange(preempt, OsuHitObject.PREEMPT_MAX, OsuHitObject.PREEMPT_MID, OsuHitObject.PREEMPT_MIN);
var greatHitWindowRange = OsuHitWindows.OSU_RANGES.Single(range => range.Result == HitResult.Great);
double greatHitWindow = IBeatmapDifficultyInfo.DifficultyRange(adjustedDifficulty.OverallDifficulty, greatHitWindowRange.Min, greatHitWindowRange.Average, greatHitWindowRange.Max);
greatHitWindow /= rate;
adjustedDifficulty.OverallDifficulty = (float)IBeatmapDifficultyInfo.InverseDifficultyRange(greatHitWindow, greatHitWindowRange.Min, greatHitWindowRange.Average, greatHitWindowRange.Max);
2023-11-12 15:20:13 +08:00
return adjustedDifficulty;
}
public override bool EditorShowScrollSpeed => false;
2018-04-13 17:19:50 +08:00
}
}