2023-09-13 18:06:59 +00: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.
|
|
|
|
|
|
|
|
using System;
|
2023-09-26 06:20:09 +00:00
|
|
|
using System.Collections.Generic;
|
2023-09-26 06:44:32 +00:00
|
|
|
using System.Linq;
|
2023-09-13 18:06:59 +00:00
|
|
|
using NUnit.Framework;
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
using osu.Game.Rulesets.Judgements;
|
|
|
|
using osu.Game.Rulesets.Mania.Beatmaps;
|
|
|
|
using osu.Game.Rulesets.Mania.Judgements;
|
|
|
|
using osu.Game.Rulesets.Mania.Objects;
|
|
|
|
using osu.Game.Rulesets.Mania.Scoring;
|
2023-09-26 06:20:09 +00:00
|
|
|
using osu.Game.Rulesets.Mods;
|
2023-09-13 18:06:59 +00:00
|
|
|
using osu.Game.Rulesets.Scoring;
|
2023-09-26 06:44:32 +00:00
|
|
|
using osu.Game.Rulesets.Scoring.Legacy;
|
2023-09-13 18:06:59 +00:00
|
|
|
using osu.Game.Tests.Visual.Gameplay;
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Mania.Tests
|
|
|
|
{
|
|
|
|
[TestFixture]
|
|
|
|
public partial class TestSceneScoring : ScoringTestScene
|
|
|
|
{
|
|
|
|
protected override IBeatmap CreateBeatmap(int maxCombo)
|
|
|
|
{
|
|
|
|
var beatmap = new ManiaBeatmap(new StageDefinition(5));
|
|
|
|
for (int i = 0; i < maxCombo; ++i)
|
|
|
|
beatmap.HitObjects.Add(new Note());
|
|
|
|
return beatmap;
|
|
|
|
}
|
|
|
|
|
2023-09-26 06:44:32 +00:00
|
|
|
protected override IScoringAlgorithm CreateScoreV1(IReadOnlyList<Mod> selectedMods) => new ScoreV1(MaxCombo.Value, selectedMods);
|
|
|
|
protected override IScoringAlgorithm CreateScoreV2(int maxCombo, IReadOnlyList<Mod> selectedMods) => new ScoreV2(maxCombo, selectedMods);
|
2023-09-26 06:20:09 +00:00
|
|
|
|
|
|
|
protected override ProcessorBasedScoringAlgorithm CreateScoreAlgorithm(IBeatmap beatmap, ScoringMode mode, IReadOnlyList<Mod> selectedMods)
|
|
|
|
=> new ManiaProcessorBasedScoringAlgorithm(beatmap, mode, selectedMods);
|
2023-09-13 18:06:59 +00:00
|
|
|
|
|
|
|
[Test]
|
|
|
|
public void TestBasicScenarios()
|
|
|
|
{
|
|
|
|
AddStep("set max combo to 100", () => MaxCombo.Value = 100);
|
|
|
|
AddStep("set perfect score", () =>
|
|
|
|
{
|
|
|
|
NonPerfectLocations.Clear();
|
|
|
|
MissLocations.Clear();
|
|
|
|
});
|
|
|
|
AddStep("set score with misses", () =>
|
|
|
|
{
|
|
|
|
NonPerfectLocations.Clear();
|
|
|
|
MissLocations.Clear();
|
|
|
|
MissLocations.AddRange(new[] { 24d, 49 });
|
|
|
|
});
|
|
|
|
AddStep("set score with misses and OKs", () =>
|
|
|
|
{
|
|
|
|
NonPerfectLocations.Clear();
|
|
|
|
MissLocations.Clear();
|
|
|
|
|
|
|
|
NonPerfectLocations.AddRange(new[] { 9d, 19, 29, 39, 59, 69, 79, 89, 99 });
|
|
|
|
MissLocations.AddRange(new[] { 24d, 49 });
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private class ScoreV1 : IScoringAlgorithm
|
|
|
|
{
|
|
|
|
private int currentCombo;
|
|
|
|
private double comboAddition = 100;
|
|
|
|
private double totalScoreDouble;
|
2023-09-26 06:44:32 +00:00
|
|
|
|
2023-09-13 18:06:59 +00:00
|
|
|
private readonly double scoreMultiplier;
|
|
|
|
|
2023-09-26 06:44:32 +00:00
|
|
|
public ScoreV1(int maxCombo, IReadOnlyList<Mod> selectedMods)
|
2023-09-13 18:06:59 +00:00
|
|
|
{
|
2023-09-26 06:44:32 +00:00
|
|
|
var ruleset = new ManiaRuleset();
|
|
|
|
|
|
|
|
scoreMultiplier = 500000d / maxCombo * ruleset.CreateLegacyScoreSimulator().GetLegacyScoreMultiplier(selectedMods, new LegacyBeatmapConversionDifficultyInfo
|
|
|
|
{
|
|
|
|
SourceRuleset = ruleset.RulesetInfo
|
|
|
|
});
|
2023-09-13 18:06:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void ApplyHit() => applyHitV1(320, add => add + 2, 32);
|
|
|
|
public void ApplyNonPerfect() => applyHitV1(100, add => add - 24, 8);
|
|
|
|
public void ApplyMiss() => applyHitV1(0, _ => -56, 0);
|
|
|
|
|
|
|
|
private void applyHitV1(int scoreIncrease, Func<double, double> comboAdditionFunc, int delta)
|
|
|
|
{
|
|
|
|
comboAddition = comboAdditionFunc(comboAddition);
|
|
|
|
if (currentCombo != 0 && currentCombo % 384 == 0)
|
|
|
|
comboAddition = 100;
|
|
|
|
comboAddition = Math.Max(0, Math.Min(comboAddition, 100));
|
|
|
|
double scoreIncreaseD = Math.Sqrt(comboAddition) * delta * scoreMultiplier / 320;
|
|
|
|
|
|
|
|
TotalScore = (long)totalScoreDouble;
|
|
|
|
|
|
|
|
scoreIncreaseD += scoreIncrease * scoreMultiplier / 320;
|
|
|
|
scoreIncrease = (int)scoreIncreaseD;
|
|
|
|
|
|
|
|
TotalScore += scoreIncrease;
|
|
|
|
totalScoreDouble += scoreIncreaseD;
|
|
|
|
|
|
|
|
if (scoreIncrease > 0)
|
|
|
|
currentCombo++;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long TotalScore { get; private set; }
|
|
|
|
}
|
|
|
|
|
|
|
|
private class ScoreV2 : IScoringAlgorithm
|
|
|
|
{
|
|
|
|
private int currentCombo;
|
|
|
|
private double comboPortion;
|
|
|
|
private double currentBaseScore;
|
|
|
|
private double maxBaseScore;
|
|
|
|
private int currentHits;
|
|
|
|
|
|
|
|
private readonly double comboPortionMax;
|
|
|
|
private readonly int maxCombo;
|
2023-09-26 06:44:32 +00:00
|
|
|
private readonly double modMultiplier;
|
2023-09-13 18:06:59 +00:00
|
|
|
|
|
|
|
private const double combo_base = 4;
|
|
|
|
|
2023-09-26 06:44:32 +00:00
|
|
|
public ScoreV2(int maxCombo, IReadOnlyList<Mod> selectedMods)
|
2023-09-13 18:06:59 +00:00
|
|
|
{
|
|
|
|
this.maxCombo = maxCombo;
|
|
|
|
|
2023-09-26 06:44:32 +00:00
|
|
|
var ruleset = new ManiaRuleset();
|
|
|
|
modMultiplier = new ManiaRuleset().CreateLegacyScoreSimulator().GetLegacyScoreMultiplier(
|
|
|
|
selectedMods.Append(new ModScoreV2()).ToArray(),
|
|
|
|
new LegacyBeatmapConversionDifficultyInfo
|
|
|
|
{
|
|
|
|
SourceRuleset = ruleset.RulesetInfo
|
|
|
|
});
|
|
|
|
|
2023-09-13 18:06:59 +00:00
|
|
|
for (int i = 0; i < this.maxCombo; i++)
|
|
|
|
ApplyHit();
|
|
|
|
|
|
|
|
comboPortionMax = comboPortion;
|
|
|
|
|
|
|
|
currentCombo = 0;
|
|
|
|
comboPortion = 0;
|
|
|
|
currentBaseScore = 0;
|
|
|
|
maxBaseScore = 0;
|
|
|
|
currentHits = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ApplyHit() => applyHitV2(305, 300);
|
|
|
|
public void ApplyNonPerfect() => applyHitV2(100, 100);
|
|
|
|
|
|
|
|
private void applyHitV2(int hitValue, int baseHitValue)
|
|
|
|
{
|
|
|
|
maxBaseScore += 305;
|
|
|
|
currentBaseScore += hitValue;
|
|
|
|
comboPortion += baseHitValue * Math.Min(Math.Max(0.5, Math.Log(++currentCombo, combo_base)), Math.Log(400, combo_base));
|
|
|
|
|
|
|
|
currentHits++;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ApplyMiss()
|
|
|
|
{
|
|
|
|
currentHits++;
|
|
|
|
maxBaseScore += 305;
|
|
|
|
currentCombo = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long TotalScore
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
float accuracy = (float)(currentBaseScore / maxBaseScore);
|
|
|
|
|
|
|
|
return (int)Math.Round
|
2023-09-26 06:44:32 +00:00
|
|
|
((
|
2023-09-13 18:06:59 +00:00
|
|
|
200000 * comboPortion / comboPortionMax +
|
|
|
|
800000 * Math.Pow(accuracy, 2 + 2 * accuracy) * ((double)currentHits / maxCombo)
|
2023-09-26 06:44:32 +00:00
|
|
|
) * modMultiplier);
|
2023-09-13 18:06:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private class ManiaProcessorBasedScoringAlgorithm : ProcessorBasedScoringAlgorithm
|
|
|
|
{
|
2023-09-26 06:20:09 +00:00
|
|
|
public ManiaProcessorBasedScoringAlgorithm(IBeatmap beatmap, ScoringMode mode, IReadOnlyList<Mod> selectedMods)
|
|
|
|
: base(beatmap, mode, selectedMods)
|
2023-09-13 18:06:59 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override ScoreProcessor CreateScoreProcessor() => new ManiaScoreProcessor();
|
|
|
|
|
|
|
|
protected override JudgementResult CreatePerfectJudgementResult() => new JudgementResult(new Note(), new ManiaJudgement()) { Type = HitResult.Perfect };
|
|
|
|
|
|
|
|
protected override JudgementResult CreateNonPerfectJudgementResult() => new JudgementResult(new Note(), new ManiaJudgement()) { Type = HitResult.Ok };
|
|
|
|
|
|
|
|
protected override JudgementResult CreateMissJudgementResult() => new JudgementResult(new Note(), new ManiaJudgement()) { Type = HitResult.Miss };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|