2019-02-13 14:49:30 +08:00
|
|
|
|
// 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.
|
|
|
|
|
|
2020-10-09 20:50:11 +08:00
|
|
|
|
using System;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using osu.Game.Beatmaps;
|
2021-11-15 17:19:23 +08:00
|
|
|
|
using osu.Game.Extensions;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
using osu.Game.Rulesets.Difficulty;
|
|
|
|
|
using osu.Game.Rulesets.Difficulty.Preprocessing;
|
|
|
|
|
using osu.Game.Rulesets.Difficulty.Skills;
|
|
|
|
|
using osu.Game.Rulesets.Mania.Beatmaps;
|
|
|
|
|
using osu.Game.Rulesets.Mania.Difficulty.Preprocessing;
|
|
|
|
|
using osu.Game.Rulesets.Mania.Difficulty.Skills;
|
2020-10-09 20:50:11 +08:00
|
|
|
|
using osu.Game.Rulesets.Mania.MathUtils;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
using osu.Game.Rulesets.Mania.Mods;
|
2020-08-28 18:16:24 +08:00
|
|
|
|
using osu.Game.Rulesets.Mania.Objects;
|
2019-09-06 14:24:00 +08:00
|
|
|
|
using osu.Game.Rulesets.Mania.Scoring;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
using osu.Game.Rulesets.Mods;
|
2020-10-09 20:50:11 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
2019-09-06 14:24:00 +08:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Mania.Difficulty
|
|
|
|
|
{
|
|
|
|
|
public class ManiaDifficultyCalculator : DifficultyCalculator
|
|
|
|
|
{
|
|
|
|
|
private const double star_scaling_factor = 0.018;
|
|
|
|
|
|
|
|
|
|
private readonly bool isForCurrentRuleset;
|
2020-10-14 19:40:29 +08:00
|
|
|
|
private readonly double originalOverallDifficulty;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
|
2022-09-02 15:27:25 +08:00
|
|
|
|
public override int Version => 20220902;
|
2022-07-21 01:05:18 +08:00
|
|
|
|
|
2023-06-29 16:28:06 +08:00
|
|
|
|
private readonly IWorkingBeatmap workingBeatmap;
|
2023-06-26 21:19:01 +08:00
|
|
|
|
|
2021-11-15 17:23:03 +08:00
|
|
|
|
public ManiaDifficultyCalculator(IRulesetInfo ruleset, IWorkingBeatmap beatmap)
|
2019-02-13 14:49:30 +08:00
|
|
|
|
: base(ruleset, beatmap)
|
|
|
|
|
{
|
2023-06-26 21:19:01 +08:00
|
|
|
|
workingBeatmap = beatmap;
|
|
|
|
|
|
2021-11-15 17:19:23 +08:00
|
|
|
|
isForCurrentRuleset = beatmap.BeatmapInfo.Ruleset.MatchesOnlineID(ruleset);
|
|
|
|
|
originalOverallDifficulty = beatmap.BeatmapInfo.Difficulty.OverallDifficulty;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-19 16:48:00 +08:00
|
|
|
|
protected override DifficultyAttributes CreateDifficultyAttributes(IBeatmap beatmap, Mod[] mods, Skill[] skills, double clockRate)
|
2019-02-13 14:49:30 +08:00
|
|
|
|
{
|
2019-02-19 16:48:00 +08:00
|
|
|
|
if (beatmap.HitObjects.Count == 0)
|
2021-11-21 11:12:24 +08:00
|
|
|
|
return new ManiaDifficultyAttributes { Mods = mods };
|
2019-02-13 14:49:30 +08:00
|
|
|
|
|
2019-09-02 16:38:52 +08:00
|
|
|
|
HitWindows hitWindows = new ManiaHitWindows();
|
2021-10-02 11:34:29 +08:00
|
|
|
|
hitWindows.SetDifficulty(beatmap.Difficulty.OverallDifficulty);
|
2019-09-02 16:38:52 +08:00
|
|
|
|
|
2023-06-24 00:03:18 +08:00
|
|
|
|
ManiaDifficultyAttributes attributes = new ManiaDifficultyAttributes
|
2019-02-19 16:48:00 +08:00
|
|
|
|
{
|
2020-10-09 20:47:34 +08:00
|
|
|
|
StarRating = skills[0].DifficultyValue() * star_scaling_factor,
|
2019-02-19 16:48:00 +08:00
|
|
|
|
Mods = mods,
|
2022-02-16 19:07:26 +08:00
|
|
|
|
// In osu-stable mania, rate-adjustment mods don't affect the hit window.
|
|
|
|
|
// This is done the way it is to introduce fractional differences in order to match osu-stable for the time being.
|
2022-02-16 18:50:27 +08:00
|
|
|
|
GreatHitWindow = Math.Ceiling((int)(getHitWindow300(mods) * clockRate) / clockRate),
|
2023-06-13 22:22:27 +08:00
|
|
|
|
MaxCombo = beatmap.HitObjects.Sum(maxComboForObject),
|
2019-02-19 16:48:00 +08:00
|
|
|
|
};
|
2023-06-24 00:03:18 +08:00
|
|
|
|
|
|
|
|
|
if (ComputeLegacyScoringValues)
|
|
|
|
|
{
|
2023-06-28 14:04:13 +08:00
|
|
|
|
ManiaLegacyScoreProcessor sv1Processor = new ManiaLegacyScoreProcessor();
|
2023-06-26 21:19:01 +08:00
|
|
|
|
sv1Processor.Simulate(workingBeatmap, beatmap, mods);
|
|
|
|
|
attributes.LegacyAccuracyScore = sv1Processor.AccuracyScore;
|
|
|
|
|
attributes.LegacyComboScore = sv1Processor.ComboScore;
|
|
|
|
|
attributes.LegacyBonusScoreRatio = sv1Processor.BonusScoreRatio;
|
2023-06-24 00:03:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return attributes;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-02 18:16:31 +08:00
|
|
|
|
private static int maxComboForObject(HitObject hitObject)
|
|
|
|
|
{
|
|
|
|
|
if (hitObject is HoldNote hold)
|
|
|
|
|
return 1 + (int)((hold.EndTime - hold.StartTime) / 100);
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-19 16:48:00 +08:00
|
|
|
|
protected override IEnumerable<DifficultyHitObject> CreateDifficultyHitObjects(IBeatmap beatmap, double clockRate)
|
2019-02-13 14:49:30 +08:00
|
|
|
|
{
|
2020-10-09 20:50:11 +08:00
|
|
|
|
var sortedObjects = beatmap.HitObjects.ToArray();
|
|
|
|
|
|
|
|
|
|
LegacySortHelper<HitObject>.Sort(sortedObjects, Comparer<HitObject>.Create((a, b) => (int)Math.Round(a.StartTime) - (int)Math.Round(b.StartTime)));
|
|
|
|
|
|
2022-05-22 23:26:22 +08:00
|
|
|
|
List<DifficultyHitObject> objects = new List<DifficultyHitObject>();
|
|
|
|
|
|
2020-10-09 20:50:11 +08:00
|
|
|
|
for (int i = 1; i < sortedObjects.Length; i++)
|
2022-05-27 02:26:14 +08:00
|
|
|
|
objects.Add(new ManiaDifficultyHitObject(sortedObjects[i], sortedObjects[i - 1], clockRate, objects, objects.Count));
|
2022-05-22 23:26:22 +08:00
|
|
|
|
|
|
|
|
|
return objects;
|
2019-02-13 14:49:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-10-09 20:50:11 +08:00
|
|
|
|
// Sorting is done in CreateDifficultyHitObjects, since the full list of hitobjects is required.
|
|
|
|
|
protected override IEnumerable<DifficultyHitObject> SortObjects(IEnumerable<DifficultyHitObject> input) => input;
|
|
|
|
|
|
2021-06-03 14:09:37 +08:00
|
|
|
|
protected override Skill[] CreateSkills(IBeatmap beatmap, Mod[] mods, double clockRate) => new Skill[]
|
2019-02-13 14:49:30 +08:00
|
|
|
|
{
|
2021-10-01 18:57:45 +08:00
|
|
|
|
new Strain(mods, ((ManiaBeatmap)Beatmap).TotalColumns)
|
2020-10-09 20:47:34 +08:00
|
|
|
|
};
|
2019-02-13 14:49:30 +08:00
|
|
|
|
|
|
|
|
|
protected override Mod[] DifficultyAdjustmentMods
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
var mods = new Mod[]
|
|
|
|
|
{
|
|
|
|
|
new ManiaModDoubleTime(),
|
|
|
|
|
new ManiaModHalfTime(),
|
|
|
|
|
new ManiaModEasy(),
|
|
|
|
|
new ManiaModHardRock(),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (isForCurrentRuleset)
|
|
|
|
|
return mods;
|
|
|
|
|
|
|
|
|
|
// if we are a convert, we can be played in any key mod.
|
|
|
|
|
return mods.Concat(new Mod[]
|
|
|
|
|
{
|
|
|
|
|
new ManiaModKey1(),
|
|
|
|
|
new ManiaModKey2(),
|
|
|
|
|
new ManiaModKey3(),
|
|
|
|
|
new ManiaModKey4(),
|
|
|
|
|
new ManiaModKey5(),
|
2020-10-14 16:53:28 +08:00
|
|
|
|
new MultiMod(new ManiaModKey5(), new ManiaModDualStages()),
|
2019-02-13 14:49:30 +08:00
|
|
|
|
new ManiaModKey6(),
|
2020-10-14 16:53:28 +08:00
|
|
|
|
new MultiMod(new ManiaModKey6(), new ManiaModDualStages()),
|
2019-02-13 14:49:30 +08:00
|
|
|
|
new ManiaModKey7(),
|
2020-10-14 16:53:28 +08:00
|
|
|
|
new MultiMod(new ManiaModKey7(), new ManiaModDualStages()),
|
2019-02-13 14:49:30 +08:00
|
|
|
|
new ManiaModKey8(),
|
2020-10-14 16:53:28 +08:00
|
|
|
|
new MultiMod(new ManiaModKey8(), new ManiaModDualStages()),
|
2019-02-13 14:49:30 +08:00
|
|
|
|
new ManiaModKey9(),
|
2020-10-14 16:53:28 +08:00
|
|
|
|
new MultiMod(new ManiaModKey9(), new ManiaModDualStages()),
|
2019-02-13 14:49:30 +08:00
|
|
|
|
}).ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-10-14 16:53:28 +08:00
|
|
|
|
|
2022-02-16 18:50:27 +08:00
|
|
|
|
private double getHitWindow300(Mod[] mods)
|
2020-10-14 19:40:29 +08:00
|
|
|
|
{
|
|
|
|
|
if (isForCurrentRuleset)
|
|
|
|
|
{
|
|
|
|
|
double od = Math.Min(10.0, Math.Max(0, 10.0 - originalOverallDifficulty));
|
|
|
|
|
return applyModAdjustments(34 + 3 * od, mods);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Math.Round(originalOverallDifficulty) > 4)
|
|
|
|
|
return applyModAdjustments(34, mods);
|
|
|
|
|
|
|
|
|
|
return applyModAdjustments(47, mods);
|
|
|
|
|
|
2022-02-16 18:50:27 +08:00
|
|
|
|
static double applyModAdjustments(double value, Mod[] mods)
|
2020-10-14 19:40:29 +08:00
|
|
|
|
{
|
|
|
|
|
if (mods.Any(m => m is ManiaModHardRock))
|
|
|
|
|
value /= 1.4;
|
|
|
|
|
else if (mods.Any(m => m is ManiaModEasy))
|
|
|
|
|
value *= 1.4;
|
|
|
|
|
|
2022-02-16 18:50:27 +08:00
|
|
|
|
return value;
|
2020-10-14 19:40:29 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-02-13 14:49:30 +08:00
|
|
|
|
}
|
|
|
|
|
}
|