1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-15 16:25:32 +08:00
osu-lazer/osu.Game.Rulesets.Taiko/TaikoRuleset.cs

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 17:19:50 +08:00
2022-09-10 10:46:10 +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.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;
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 19:08:13 +08:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Rulesets.Taiko.Beatmaps;
using osu.Game.Rulesets.Taiko.Difficulty;
2020-05-29 15:40:10 +08:00
using osu.Game.Rulesets.Taiko.Edit;
using osu.Game.Rulesets.Taiko.Mods;
2020-06-19 19:31:52 +08: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 11:30:25 +08: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 19:31:52 +08: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 17:19:50 +08:00
namespace osu.Game.Rulesets.Taiko
{
2019-12-24 12:48:27 +08:00
public class TaikoRuleset : Ruleset, ILegacyRuleset
2018-04-13 17:19:50 +08:00
{
public override DrawableRuleset CreateDrawableRulesetWith(IBeatmap beatmap, IReadOnlyList<Mod>? mods = null) => new DrawableTaikoRuleset(this, beatmap, mods);
2019-12-17 19:08:13 +08:00
public override ScoreProcessor CreateScoreProcessor() => new TaikoScoreProcessor();
2019-12-17 19:08:13 +08:00
public override HealthProcessor CreateHealthProcessor(double drainStartTime) => new TaikoHealthProcessor();
public override IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap) => new TaikoBeatmapConverter(beatmap, this);
2018-04-13 17:19:50 +08: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 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.MouseRight, TaikoAction.LeftRim),
2018-04-13 17:19:50 +08:00
new KeyBinding(InputKey.D, TaikoAction.LeftRim),
new KeyBinding(InputKey.MouseLeft, TaikoAction.LeftCentre),
new KeyBinding(InputKey.F, TaikoAction.LeftCentre),
2018-04-13 17:19:50 +08:00
new KeyBinding(InputKey.J, TaikoAction.RightCentre),
new KeyBinding(InputKey.K, TaikoAction.RightRim),
};
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 TaikoModNightcore();
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 TaikoModDoubleTime();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Perfect))
yield return new TaikoModPerfect();
2024-07-02 23:19:04 +08:00
else if (mods.HasFlag(LegacyMods.SuddenDeath))
yield return new TaikoModSuddenDeath();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Cinema))
2019-11-24 01:32:16 +08:00
yield return new TaikoModCinema();
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 TaikoModAutoplay();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Easy))
2018-04-13 21:41:35 +08:00
yield return new TaikoModEasy();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Flashlight))
2018-04-13 21:41:35 +08:00
yield return new TaikoModFlashlight();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.HalfTime))
2018-04-13 21:41:35 +08:00
yield return new TaikoModHalfTime();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.HardRock))
2018-04-13 21:41:35 +08:00
yield return new TaikoModHardRock();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Hidden))
2018-04-13 21:41:35 +08:00
yield return new TaikoModHidden();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.NoFail))
2018-04-13 21:41:35 +08:00
yield return new TaikoModNoFail();
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.Relax))
2018-04-13 21:41:35 +08:00
yield return new TaikoModRelax();
2020-11-15 22:41:58 +08:00
2024-07-02 23:19:04 +08:00
if (mods.HasFlag(LegacyMods.ScoreV2))
yield return new ModScoreV2();
2020-11-15 22:41:58 +08:00
}
2018-04-13 17:19:50 +08:00
public override IEnumerable<Mod> GetModsFor(ModType type)
{
switch (type)
{
case ModType.DifficultyReduction:
return new Mod[]
{
new TaikoModEasy(),
new TaikoModNoFail(),
2018-06-06 13:07:50 +08:00
new MultiMod(new TaikoModHalfTime(), new TaikoModDaycore()),
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 TaikoModHardRock(),
2018-06-06 13:07:50 +08:00
new MultiMod(new TaikoModSuddenDeath(), new TaikoModPerfect()),
2018-06-13 11:21:37 +08:00
new MultiMod(new TaikoModDoubleTime(), new TaikoModNightcore()),
2018-04-13 17:19:50 +08:00
new TaikoModHidden(),
new TaikoModFlashlight(),
2022-05-24 22:56:31 +08:00
new ModAccuracyChallenge(),
2018-04-13 17:19:50 +08:00
};
2019-04-01 11:44:46 +08:00
2019-12-11 19:43:16 +08:00
case ModType.Conversion:
return new Mod[]
{
2021-05-16 12:03:03 +08:00
new TaikoModRandom(),
2019-12-20 18:30:23 +08:00
new TaikoModDifficultyAdjust(),
new TaikoModClassic(),
2021-05-17 11:04:01 +08:00
new TaikoModSwap(),
new TaikoModSingleTap(),
2024-02-23 18:27:12 +08:00
new TaikoModConstantSpeed(),
2019-12-11 19:43:16 +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 TaikoModAutoplay(), new TaikoModCinema()),
new TaikoModRelax(),
2018-04-13 17:19:50 +08:00
};
2019-04-01 11:44:46 +08:00
2019-01-26 12:15:45 +08:00
case ModType.Fun:
return new Mod[]
{
2021-07-28 18:21:08 +08:00
new MultiMod(new ModWindUp(), new ModWindDown()),
new TaikoModMuted(),
2022-03-01 21:12:06 +08:00
new ModAdaptiveSpeed()
2019-01-26 12:15:45 +08:00
};
2019-04-01 11:44:46 +08:00
case ModType.System:
return new Mod[]
{
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 string Description => "osu!taiko";
public override string ShortName => SHORT_NAME;
2018-04-13 17:19:50 +08:00
public override string PlayingVerb => "Bashing drums";
public override Drawable CreateIcon() => new SpriteIcon { Icon = OsuIcon.RulesetTaiko };
2018-04-13 17:19:50 +08:00
public override HitObjectComposer CreateHitObjectComposer() => new TaikoHitObjectComposer(this);
public override IEnumerable<SetupSection> CreateEditorSetupSections() =>
[
new TaikoDifficultySection(),
];
2024-03-23 01:48:22 +08:00
public override IBeatmapVerifier CreateBeatmapVerifier() => new TaikoBeatmapVerifier();
public override DifficultyCalculator CreateDifficultyCalculator(IWorkingBeatmap beatmap) => new TaikoDifficultyCalculator(RulesetInfo, beatmap);
2018-04-13 17:19:50 +08:00
2022-03-15 11:37:39 +08:00
public override PerformanceCalculator CreatePerformanceCalculator() => new TaikoPerformanceCalculator();
2019-12-24 12:48:27 +08:00
public int LegacyID => 1;
2018-04-13 17:19:50 +08:00
public ILegacyScoreSimulator CreateLegacyScoreSimulator() => new TaikoLegacyScoreSimulator();
public override IConvertibleReplayFrame CreateConvertibleReplayFrame() => new TaikoReplayFrame();
2020-06-19 19:31:52 +08: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-15 02:54:02 +08: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 19:31:52 +08:00
{
var timedHitEvents = score.HitEvents.Where(e => e.HitObject is Hit).ToList();
2020-08-28 02:07:30 +08:00
return new[]
2020-06-19 19:31:52 +08: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 21:30:35 +08:00
{
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)
};
}
/// <seealso cref="TaikoHitWindows"/>
public override BeatmapDifficulty GetRateAdjustedDisplayDifficulty(IBeatmapDifficultyInfo difficulty, double rate)
2023-11-04 23:25:09 +08: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 15:20:13 +08:00
return adjustedDifficulty;
}
2018-04-13 17:19:50 +08:00
}
}