osu/osu.Game.Rulesets.Taiko/TaikoRuleset.cs

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

278 lines
10 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 09:19:50 +00:00
2022-09-10 02:46:10 +00:00
using System;
2017-03-10 02:04:46 +00:00
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Sprites;
2017-08-20 12:56:37 +00:00
using osu.Framework.Input.Bindings;
using osu.Framework.Localisation;
using osu.Game.Beatmaps;
2018-04-13 13:41:35 +00:00
using osu.Game.Beatmaps.Legacy;
using osu.Game.Graphics;
using osu.Game.Rulesets.Difficulty;
using osu.Game.Rulesets.Edit;
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Replays.Types;
2019-12-17 11:08:13 +00:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Rulesets.Taiko.Beatmaps;
using osu.Game.Rulesets.Taiko.Difficulty;
2020-05-29 07:40:10 +00:00
using osu.Game.Rulesets.Taiko.Edit;
using osu.Game.Rulesets.Taiko.Mods;
2020-06-19 11:31:52 +00:00
using osu.Game.Rulesets.Taiko.Objects;
using osu.Game.Rulesets.Taiko.Replays;
using osu.Game.Rulesets.Taiko.Scoring;
using osu.Game.Rulesets.Taiko.Skinning.Argon;
2020-12-07 03:30:25 +00:00
using osu.Game.Rulesets.Taiko.Skinning.Legacy;
using osu.Game.Rulesets.Taiko.UI;
using osu.Game.Rulesets.UI;
using osu.Game.Overlays.Settings;
using osu.Game.Scoring;
2020-06-19 11:31:52 +00:00
using osu.Game.Screens.Ranking.Statistics;
using osu.Game.Skinning;
using osu.Game.Rulesets.Configuration;
using osu.Game.Configuration;
using osu.Game.Rulesets.Scoring.Legacy;
using osu.Game.Rulesets.Taiko.Configuration;
using osu.Game.Rulesets.Taiko.Edit.Setup;
using osu.Game.Screens.Edit.Setup;
2018-04-13 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
namespace osu.Game.Rulesets.Taiko
{
2019-12-24 04:48:27 +00:00
public class TaikoRuleset : Ruleset, ILegacyRuleset
{
public override DrawableRuleset CreateDrawableRulesetWith(IBeatmap beatmap, IReadOnlyList<Mod>? mods = null) => new DrawableTaikoRuleset(this, beatmap, mods);
2019-12-17 11:08:13 +00:00
public override ScoreProcessor CreateScoreProcessor() => new TaikoScoreProcessor();
2019-12-17 11:08:13 +00:00
public override HealthProcessor CreateHealthProcessor(double drainStartTime) => new TaikoHealthProcessor();
public override IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap) => new TaikoBeatmapConverter(beatmap, this);
2018-04-13 09:19:50 +00:00
public override ISkin? CreateSkinTransformer(ISkin skin, IBeatmap beatmap)
{
switch (skin)
{
case ArgonSkin:
return new TaikoArgonSkinTransformer(skin);
case LegacySkin:
return new TaikoLegacySkinTransformer(skin);
}
return null;
}
public const string SHORT_NAME = "taiko";
public override string RulesetAPIVersionSupported => CURRENT_RULESET_API_VERSION;
2022-08-22 07:10:55 +00:00
2017-08-20 12:56:37 +00:00
public override IEnumerable<KeyBinding> GetDefaultKeyBindings(int variant = 0) => new[]
{
new KeyBinding(InputKey.MouseRight, TaikoAction.LeftRim),
2017-08-20 12:56:37 +00:00
new KeyBinding(InputKey.D, TaikoAction.LeftRim),
new KeyBinding(InputKey.MouseLeft, TaikoAction.LeftCentre),
new KeyBinding(InputKey.F, TaikoAction.LeftCentre),
2017-08-20 12:56:37 +00:00
new KeyBinding(InputKey.J, TaikoAction.RightCentre),
new KeyBinding(InputKey.K, TaikoAction.RightRim),
};
2018-04-13 09:19:50 +00:00
public override IEnumerable<Mod> ConvertFromLegacyMods(LegacyMods mods)
2018-04-13 13:41:35 +00:00
{
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Nightcore))
2018-04-13 13:41:35 +00:00
yield return new TaikoModNightcore();
2024-07-02 15:19:04 +00:00
else if (mods.HasFlag(LegacyMods.DoubleTime))
2018-04-13 13:41:35 +00:00
yield return new TaikoModDoubleTime();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Perfect))
yield return new TaikoModPerfect();
2024-07-02 15:19:04 +00:00
else if (mods.HasFlag(LegacyMods.SuddenDeath))
yield return new TaikoModSuddenDeath();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Cinema))
2019-11-23 17:32:16 +00:00
yield return new TaikoModCinema();
2024-07-02 15:19:04 +00:00
else if (mods.HasFlag(LegacyMods.Autoplay))
2018-04-13 13:41:35 +00:00
yield return new TaikoModAutoplay();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Easy))
2018-04-13 13:41:35 +00:00
yield return new TaikoModEasy();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Flashlight))
2018-04-13 13:41:35 +00:00
yield return new TaikoModFlashlight();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.HalfTime))
2018-04-13 13:41:35 +00:00
yield return new TaikoModHalfTime();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.HardRock))
2018-04-13 13:41:35 +00:00
yield return new TaikoModHardRock();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Hidden))
2018-04-13 13:41:35 +00:00
yield return new TaikoModHidden();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.NoFail))
2018-04-13 13:41:35 +00:00
yield return new TaikoModNoFail();
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.Relax))
2018-04-13 13:41:35 +00:00
yield return new TaikoModRelax();
2020-11-15 14:41:58 +00:00
2024-07-02 15:19:04 +00:00
if (mods.HasFlag(LegacyMods.ScoreV2))
yield return new ModScoreV2();
2020-11-15 14:41:58 +00:00
}
public override IEnumerable<Mod> GetModsFor(ModType type)
{
switch (type)
{
case ModType.DifficultyReduction:
return new Mod[]
{
new TaikoModEasy(),
new TaikoModNoFail(),
2018-06-06 05:07:50 +00:00
new MultiMod(new TaikoModHalfTime(), new TaikoModDaycore()),
};
2019-04-01 03:44:46 +00:00
case ModType.DifficultyIncrease:
return new Mod[]
{
new TaikoModHardRock(),
2018-06-06 05:07:50 +00:00
new MultiMod(new TaikoModSuddenDeath(), new TaikoModPerfect()),
2018-06-13 03:21:37 +00:00
new MultiMod(new TaikoModDoubleTime(), new TaikoModNightcore()),
new TaikoModHidden(),
new TaikoModFlashlight(),
2022-05-24 14:56:31 +00:00
new ModAccuracyChallenge(),
};
2019-04-01 03:44:46 +00:00
2019-12-11 11:43:16 +00:00
case ModType.Conversion:
return new Mod[]
{
2021-05-16 04:03:03 +00:00
new TaikoModRandom(),
2019-12-20 10:30:23 +00:00
new TaikoModDifficultyAdjust(),
new TaikoModClassic(),
2021-05-17 03:04:01 +00:00
new TaikoModSwap(),
new TaikoModSingleTap(),
2024-02-23 10:27:12 +00:00
new TaikoModConstantSpeed(),
2019-12-11 11:43:16 +00:00
};
case ModType.Automation:
return new Mod[]
{
2019-11-23 17:32:16 +00:00
new MultiMod(new TaikoModAutoplay(), new TaikoModCinema()),
new TaikoModRelax(),
};
2019-04-01 03:44:46 +00:00
2019-01-26 04:15:45 +00:00
case ModType.Fun:
return new Mod[]
{
2021-07-28 10:21:08 +00:00
new MultiMod(new ModWindUp(), new ModWindDown()),
new TaikoModMuted(),
2022-03-01 13:12:06 +00:00
new ModAdaptiveSpeed()
2019-01-26 04:15:45 +00:00
};
2019-04-01 03:44:46 +00:00
case ModType.System:
return new Mod[]
{
new ModScoreV2(),
};
default:
2019-11-28 13:41:29 +00:00
return Array.Empty<Mod>();
}
}
2018-04-13 09:19:50 +00:00
public override string Description => "osu!taiko";
2018-04-13 09:19:50 +00:00
public override string ShortName => SHORT_NAME;
2018-04-13 09:19:50 +00:00
public override string PlayingVerb => "Bashing drums";
public override Drawable CreateIcon() => new SpriteIcon { Icon = OsuIcon.RulesetTaiko };
2018-04-13 09:19:50 +00:00
public override HitObjectComposer CreateHitObjectComposer() => new TaikoHitObjectComposer(this);
public override IEnumerable<SetupSection> CreateEditorSetupSections() =>
[
new TaikoDifficultySection(),
];
2024-03-22 17:48:22 +00:00
public override IBeatmapVerifier CreateBeatmapVerifier() => new TaikoBeatmapVerifier();
public override DifficultyCalculator CreateDifficultyCalculator(IWorkingBeatmap beatmap) => new TaikoDifficultyCalculator(RulesetInfo, beatmap);
2018-04-13 09:19:50 +00:00
2022-03-15 03:37:39 +00:00
public override PerformanceCalculator CreatePerformanceCalculator() => new TaikoPerformanceCalculator();
2019-12-24 04:48:27 +00:00
public int LegacyID => 1;
2018-04-13 09:19:50 +00:00
public ILegacyScoreSimulator CreateLegacyScoreSimulator() => new TaikoLegacyScoreSimulator();
public override IConvertibleReplayFrame CreateConvertibleReplayFrame() => new TaikoReplayFrame();
2020-06-19 11:31:52 +00:00
public override IRulesetConfigManager CreateConfig(SettingsStore? settings) => new TaikoRulesetConfigManager(settings, RulesetInfo);
public override RulesetSettingsSubsection CreateSettings() => new TaikoSettingsSubsection(this);
protected override IEnumerable<HitResult> GetValidHitResults()
{
return new[]
{
HitResult.Great,
HitResult.Ok,
HitResult.SmallBonus,
HitResult.LargeBonus,
};
}
2022-08-14 18:54:02 +00:00
public override LocalisableString GetDisplayNameForHitResult(HitResult result)
{
switch (result)
{
case HitResult.SmallBonus:
return "drum tick";
case HitResult.LargeBonus:
return "bonus";
}
return base.GetDisplayNameForHitResult(result);
}
public override StatisticItem[] CreateStatisticsForScore(ScoreInfo score, IBeatmap playableBeatmap)
2020-06-19 11:31:52 +00:00
{
var timedHitEvents = score.HitEvents.Where(e => e.HitObject is Hit).ToList();
2020-08-27 18:07:30 +00:00
return new[]
2020-06-19 11:31:52 +00:00
{
new StatisticItem("Performance Breakdown", () => new PerformanceBreakdownChart(score, playableBeatmap)
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y
}),
new StatisticItem("Timing Distribution", () => new HitEventTimingDistributionGraph(timedHitEvents)
2022-02-05 13:30:35 +00:00
{
RelativeSizeAxes = Axes.X,
Height = 250
}, true),
new StatisticItem("Statistics", () => new SimpleStatisticTable(2, new SimpleStatisticItem[]
2020-08-27 18:46:49 +00:00
{
new AverageHitError(timedHitEvents),
new UnstableRate(timedHitEvents)
}), true)
};
}
/// <seealso cref="TaikoHitWindows"/>
public override BeatmapDifficulty GetRateAdjustedDisplayDifficulty(IBeatmapDifficultyInfo difficulty, double rate)
2023-11-04 15:25:09 +00:00
{
BeatmapDifficulty adjustedDifficulty = new BeatmapDifficulty(difficulty);
var greatHitWindowRange = TaikoHitWindows.TAIKO_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 07:20:13 +00:00
return adjustedDifficulty;
}
}
}