mirror of
https://github.com/ppy/osu.git
synced 2024-11-15 15:17:44 +08:00
c8a64c5950
Before I go with a hammer to redesign these, I want to remove stuff that does nothing first. Hard-breaks API to allow rulesets to specify an enumerable of custom sections rather than two specific weird ones. For specific rulesets: - osu!: - Stack leniency slider merged into difficulty section. - osu!taiko: - Approach rate and circle size sliders removed. - Colours section removed. - osu!catch: - No functional changes. - osu!mania: - Special style toggle merged into difficulty section. - Colours section removed.
278 lines
10 KiB
C#
278 lines
10 KiB
C#
// 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.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Sprites;
|
|
using osu.Framework.Input.Bindings;
|
|
using osu.Framework.Localisation;
|
|
using osu.Game.Beatmaps;
|
|
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;
|
|
using osu.Game.Rulesets.Scoring;
|
|
using osu.Game.Rulesets.Taiko.Beatmaps;
|
|
using osu.Game.Rulesets.Taiko.Difficulty;
|
|
using osu.Game.Rulesets.Taiko.Edit;
|
|
using osu.Game.Rulesets.Taiko.Mods;
|
|
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;
|
|
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;
|
|
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;
|
|
|
|
namespace osu.Game.Rulesets.Taiko
|
|
{
|
|
public class TaikoRuleset : Ruleset, ILegacyRuleset
|
|
{
|
|
public override DrawableRuleset CreateDrawableRulesetWith(IBeatmap beatmap, IReadOnlyList<Mod>? mods = null) => new DrawableTaikoRuleset(this, beatmap, mods);
|
|
|
|
public override ScoreProcessor CreateScoreProcessor() => new TaikoScoreProcessor();
|
|
|
|
public override HealthProcessor CreateHealthProcessor(double drainStartTime) => new TaikoHealthProcessor();
|
|
|
|
public override IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap) => new TaikoBeatmapConverter(beatmap, this);
|
|
|
|
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;
|
|
|
|
public override IEnumerable<KeyBinding> GetDefaultKeyBindings(int variant = 0) => new[]
|
|
{
|
|
new KeyBinding(InputKey.MouseRight, TaikoAction.LeftRim),
|
|
new KeyBinding(InputKey.D, TaikoAction.LeftRim),
|
|
new KeyBinding(InputKey.MouseLeft, TaikoAction.LeftCentre),
|
|
new KeyBinding(InputKey.F, TaikoAction.LeftCentre),
|
|
new KeyBinding(InputKey.J, TaikoAction.RightCentre),
|
|
new KeyBinding(InputKey.K, TaikoAction.RightRim),
|
|
};
|
|
|
|
public override IEnumerable<Mod> ConvertFromLegacyMods(LegacyMods mods)
|
|
{
|
|
if (mods.HasFlag(LegacyMods.Nightcore))
|
|
yield return new TaikoModNightcore();
|
|
else if (mods.HasFlag(LegacyMods.DoubleTime))
|
|
yield return new TaikoModDoubleTime();
|
|
|
|
if (mods.HasFlag(LegacyMods.Perfect))
|
|
yield return new TaikoModPerfect();
|
|
else if (mods.HasFlag(LegacyMods.SuddenDeath))
|
|
yield return new TaikoModSuddenDeath();
|
|
|
|
if (mods.HasFlag(LegacyMods.Cinema))
|
|
yield return new TaikoModCinema();
|
|
else if (mods.HasFlag(LegacyMods.Autoplay))
|
|
yield return new TaikoModAutoplay();
|
|
|
|
if (mods.HasFlag(LegacyMods.Easy))
|
|
yield return new TaikoModEasy();
|
|
|
|
if (mods.HasFlag(LegacyMods.Flashlight))
|
|
yield return new TaikoModFlashlight();
|
|
|
|
if (mods.HasFlag(LegacyMods.HalfTime))
|
|
yield return new TaikoModHalfTime();
|
|
|
|
if (mods.HasFlag(LegacyMods.HardRock))
|
|
yield return new TaikoModHardRock();
|
|
|
|
if (mods.HasFlag(LegacyMods.Hidden))
|
|
yield return new TaikoModHidden();
|
|
|
|
if (mods.HasFlag(LegacyMods.NoFail))
|
|
yield return new TaikoModNoFail();
|
|
|
|
if (mods.HasFlag(LegacyMods.Relax))
|
|
yield return new TaikoModRelax();
|
|
|
|
if (mods.HasFlag(LegacyMods.ScoreV2))
|
|
yield return new ModScoreV2();
|
|
}
|
|
|
|
public override IEnumerable<Mod> GetModsFor(ModType type)
|
|
{
|
|
switch (type)
|
|
{
|
|
case ModType.DifficultyReduction:
|
|
return new Mod[]
|
|
{
|
|
new TaikoModEasy(),
|
|
new TaikoModNoFail(),
|
|
new MultiMod(new TaikoModHalfTime(), new TaikoModDaycore()),
|
|
};
|
|
|
|
case ModType.DifficultyIncrease:
|
|
return new Mod[]
|
|
{
|
|
new TaikoModHardRock(),
|
|
new MultiMod(new TaikoModSuddenDeath(), new TaikoModPerfect()),
|
|
new MultiMod(new TaikoModDoubleTime(), new TaikoModNightcore()),
|
|
new TaikoModHidden(),
|
|
new TaikoModFlashlight(),
|
|
new ModAccuracyChallenge(),
|
|
};
|
|
|
|
case ModType.Conversion:
|
|
return new Mod[]
|
|
{
|
|
new TaikoModRandom(),
|
|
new TaikoModDifficultyAdjust(),
|
|
new TaikoModClassic(),
|
|
new TaikoModSwap(),
|
|
new TaikoModSingleTap(),
|
|
new TaikoModConstantSpeed(),
|
|
};
|
|
|
|
case ModType.Automation:
|
|
return new Mod[]
|
|
{
|
|
new MultiMod(new TaikoModAutoplay(), new TaikoModCinema()),
|
|
new TaikoModRelax(),
|
|
};
|
|
|
|
case ModType.Fun:
|
|
return new Mod[]
|
|
{
|
|
new MultiMod(new ModWindUp(), new ModWindDown()),
|
|
new TaikoModMuted(),
|
|
new ModAdaptiveSpeed()
|
|
};
|
|
|
|
case ModType.System:
|
|
return new Mod[]
|
|
{
|
|
new ModScoreV2(),
|
|
};
|
|
|
|
default:
|
|
return Array.Empty<Mod>();
|
|
}
|
|
}
|
|
|
|
public override string Description => "osu!taiko";
|
|
|
|
public override string ShortName => SHORT_NAME;
|
|
|
|
public override string PlayingVerb => "Bashing drums";
|
|
|
|
public override Drawable CreateIcon() => new SpriteIcon { Icon = OsuIcon.RulesetTaiko };
|
|
|
|
public override HitObjectComposer CreateHitObjectComposer() => new TaikoHitObjectComposer(this);
|
|
|
|
public override IEnumerable<SetupSection> CreateEditorSetupSections() =>
|
|
[
|
|
new TaikoDifficultySection(),
|
|
];
|
|
|
|
public override IBeatmapVerifier CreateBeatmapVerifier() => new TaikoBeatmapVerifier();
|
|
|
|
public override DifficultyCalculator CreateDifficultyCalculator(IWorkingBeatmap beatmap) => new TaikoDifficultyCalculator(RulesetInfo, beatmap);
|
|
|
|
public override PerformanceCalculator CreatePerformanceCalculator() => new TaikoPerformanceCalculator();
|
|
|
|
public int LegacyID => 1;
|
|
|
|
public ILegacyScoreSimulator CreateLegacyScoreSimulator() => new TaikoLegacyScoreSimulator();
|
|
|
|
public override IConvertibleReplayFrame CreateConvertibleReplayFrame() => new TaikoReplayFrame();
|
|
|
|
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,
|
|
};
|
|
}
|
|
|
|
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)
|
|
{
|
|
var timedHitEvents = score.HitEvents.Where(e => e.HitObject is Hit).ToList();
|
|
|
|
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("Statistics", () => new SimpleStatisticTable(2, new SimpleStatisticItem[]
|
|
{
|
|
new AverageHitError(timedHitEvents),
|
|
new UnstableRate(timedHitEvents)
|
|
}), true)
|
|
};
|
|
}
|
|
|
|
/// <seealso cref="TaikoHitWindows"/>
|
|
public override BeatmapDifficulty GetRateAdjustedDisplayDifficulty(IBeatmapDifficultyInfo difficulty, double rate)
|
|
{
|
|
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);
|
|
|
|
return adjustedDifficulty;
|
|
}
|
|
}
|
|
}
|