2019-03-27 07:55:46 +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.
|
|
|
|
|
|
2023-05-18 09:53:43 +00:00
|
|
|
|
using System;
|
2022-09-06 15:11:51 +00:00
|
|
|
|
using System.Collections.Generic;
|
2023-05-18 09:53:43 +00:00
|
|
|
|
using System.Linq;
|
2023-09-28 10:02:57 +00:00
|
|
|
|
using osu.Game.Online.API.Requests.Responses;
|
2019-03-27 07:55:46 +00:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Scoring.Legacy
|
|
|
|
|
{
|
2019-12-03 06:38:57 +00:00
|
|
|
|
public static class ScoreInfoExtensions
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2023-05-18 09:53:43 +00:00
|
|
|
|
public static long GetDisplayScore(this ScoreProcessor scoreProcessor, ScoringMode mode)
|
|
|
|
|
=> getDisplayScore(scoreProcessor.Ruleset.RulesetInfo.OnlineID, scoreProcessor.TotalScore.Value, mode, scoreProcessor.MaximumStatistics);
|
|
|
|
|
|
|
|
|
|
public static long GetDisplayScore(this ScoreInfo scoreInfo, ScoringMode mode)
|
2023-06-28 07:30:50 +00:00
|
|
|
|
=> getDisplayScore(scoreInfo.Ruleset.OnlineID, scoreInfo.TotalScore, mode, scoreInfo.MaximumStatistics);
|
2023-05-18 09:53:43 +00:00
|
|
|
|
|
2023-09-28 10:02:57 +00:00
|
|
|
|
public static long GetDisplayScore(this SoloScoreInfo soloScoreInfo, ScoringMode mode)
|
|
|
|
|
=> getDisplayScore(soloScoreInfo.RulesetID, soloScoreInfo.TotalScore, mode, soloScoreInfo.MaximumStatistics);
|
|
|
|
|
|
2023-05-18 09:53:43 +00:00
|
|
|
|
private static long getDisplayScore(int rulesetId, long score, ScoringMode mode, IReadOnlyDictionary<HitResult, int> maximumStatistics)
|
|
|
|
|
{
|
|
|
|
|
if (mode == ScoringMode.Standardised)
|
|
|
|
|
return score;
|
|
|
|
|
|
2023-05-30 05:20:26 +00:00
|
|
|
|
int maxBasicJudgements = maximumStatistics
|
|
|
|
|
.Where(k => k.Key.IsBasic())
|
|
|
|
|
.Select(k => k.Value)
|
|
|
|
|
.DefaultIfEmpty(0)
|
|
|
|
|
.Sum();
|
|
|
|
|
|
2023-09-15 12:11:14 +00:00
|
|
|
|
return convertStandardisedToClassic(rulesetId, score, maxBasicJudgements);
|
2023-05-30 05:20:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2023-09-15 12:11:14 +00:00
|
|
|
|
/// Returns a ballpark "classic" score which gives a similar "feel" to stable.
|
2023-05-30 05:20:26 +00:00
|
|
|
|
/// This is different per ruleset to match the different algorithms used in the scoring implementation.
|
|
|
|
|
/// </summary>
|
2023-09-15 12:11:14 +00:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// The coefficients chosen here were determined by a least-squares fit performed over all beatmaps
|
|
|
|
|
/// with the goal of minimising the relative error of maximum possible base score (without bonus).
|
|
|
|
|
/// The constant coefficients (100000, 1 / 10d) - while being detrimental to the least-squares fit - are forced,
|
|
|
|
|
/// so that every 10 points in standardised mode converts to at least 1 point in classic mode.
|
|
|
|
|
/// This is done to account for bonus judgements in a way that does not reorder scores.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
private static long convertStandardisedToClassic(int rulesetId, long standardisedTotalScore, int objectCount)
|
2023-05-30 05:20:26 +00:00
|
|
|
|
{
|
2023-05-18 09:53:43 +00:00
|
|
|
|
switch (rulesetId)
|
|
|
|
|
{
|
|
|
|
|
case 0:
|
2023-11-22 23:15:46 +00:00
|
|
|
|
return (long)Math.Round((Math.Pow(objectCount, 2) * 32.57 + 100000) * standardisedTotalScore / ScoreProcessor.MAX_SCORE);
|
2023-05-18 09:53:43 +00:00
|
|
|
|
|
|
|
|
|
case 1:
|
2023-09-15 12:11:14 +00:00
|
|
|
|
return (long)Math.Round((objectCount * 1109 + 100000) * standardisedTotalScore / ScoreProcessor.MAX_SCORE);
|
2023-05-18 09:53:43 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
2023-09-15 12:11:14 +00:00
|
|
|
|
return (long)Math.Round(Math.Pow(standardisedTotalScore / ScoreProcessor.MAX_SCORE * objectCount, 2) * 21.62 + standardisedTotalScore / 10d);
|
2023-05-18 09:53:43 +00:00
|
|
|
|
|
|
|
|
|
case 3:
|
2023-09-15 12:11:14 +00:00
|
|
|
|
default:
|
|
|
|
|
return standardisedTotalScore;
|
2023-05-18 09:53:43 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static int? GetCountGeki(this ScoreInfo scoreInfo)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2022-09-06 15:11:51 +00:00
|
|
|
|
case 1:
|
|
|
|
|
return getCount(scoreInfo, HitResult.LargeBonus);
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
case 3:
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return getCount(scoreInfo, HitResult.Perfect);
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
|
|
|
|
return null;
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static void SetCountGeki(this ScoreInfo scoreInfo, int value)
|
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-12-03 06:28:10 +00:00
|
|
|
|
{
|
2022-09-06 15:11:51 +00:00
|
|
|
|
// For legacy scores, Geki indicates hit300 + perfect strong note hit.
|
|
|
|
|
// Lazer only has one result for a perfect strong note hit (LargeBonus).
|
|
|
|
|
case 1:
|
|
|
|
|
scoreInfo.Statistics[HitResult.LargeBonus] = scoreInfo.Statistics.GetValueOrDefault(HitResult.LargeBonus) + value;
|
|
|
|
|
break;
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
case 3:
|
|
|
|
|
scoreInfo.Statistics[HitResult.Perfect] = value;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-03-27 07:55:46 +00:00
|
|
|
|
|
2020-09-29 08:16:55 +00:00
|
|
|
|
public static int? GetCount300(this ScoreInfo scoreInfo) => getCount(scoreInfo, HitResult.Great);
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
2020-09-29 08:16:55 +00:00
|
|
|
|
public static void SetCount300(this ScoreInfo scoreInfo, int value) => scoreInfo.Statistics[HitResult.Great] = value;
|
2019-03-27 07:55:46 +00:00
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static int? GetCountKatu(this ScoreInfo scoreInfo)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2022-09-06 15:11:51 +00:00
|
|
|
|
// For taiko, Katu is bundled into Geki.
|
|
|
|
|
case 1:
|
|
|
|
|
break;
|
2020-04-16 09:16:08 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return getCount(scoreInfo, HitResult.SmallTickMiss);
|
2022-09-06 15:11:51 +00:00
|
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
|
return getCount(scoreInfo, HitResult.Good);
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
|
|
|
|
return null;
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static void SetCountKatu(this ScoreInfo scoreInfo, int value)
|
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-12-03 06:28:10 +00:00
|
|
|
|
{
|
2022-09-06 15:11:51 +00:00
|
|
|
|
// For legacy scores, Katu indicates hit100 + perfect strong note hit.
|
|
|
|
|
// Lazer only has one result for a perfect strong note hit (LargeBonus).
|
|
|
|
|
case 1:
|
|
|
|
|
scoreInfo.Statistics[HitResult.LargeBonus] = scoreInfo.Statistics.GetValueOrDefault(HitResult.LargeBonus) + value;
|
2019-12-03 06:28:10 +00:00
|
|
|
|
break;
|
2020-04-16 09:16:08 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
|
scoreInfo.Statistics[HitResult.SmallTickMiss] = value;
|
|
|
|
|
break;
|
2022-09-06 15:11:51 +00:00
|
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
|
scoreInfo.Statistics[HitResult.Good] = value;
|
|
|
|
|
break;
|
2019-12-03 06:28:10 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-03-27 07:55:46 +00:00
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static int? GetCount100(this ScoreInfo scoreInfo)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2019-12-03 06:28:10 +00:00
|
|
|
|
case 0:
|
|
|
|
|
case 1:
|
|
|
|
|
case 3:
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return getCount(scoreInfo, HitResult.Ok);
|
2020-04-16 09:16:08 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return getCount(scoreInfo, HitResult.LargeTickHit);
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
|
|
|
|
return null;
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static void SetCount100(this ScoreInfo scoreInfo, int value)
|
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-12-03 06:28:10 +00:00
|
|
|
|
{
|
|
|
|
|
case 0:
|
|
|
|
|
case 1:
|
|
|
|
|
case 3:
|
|
|
|
|
scoreInfo.Statistics[HitResult.Ok] = value;
|
|
|
|
|
break;
|
2020-04-16 09:16:08 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
|
scoreInfo.Statistics[HitResult.LargeTickHit] = value;
|
|
|
|
|
break;
|
2019-12-03 06:28:10 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-03-27 07:55:46 +00:00
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static int? GetCount50(this ScoreInfo scoreInfo)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2019-12-03 06:28:10 +00:00
|
|
|
|
case 0:
|
|
|
|
|
case 3:
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return getCount(scoreInfo, HitResult.Meh);
|
2020-04-16 09:16:08 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return getCount(scoreInfo, HitResult.SmallTickHit);
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
|
|
|
|
return null;
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-03 06:28:10 +00:00
|
|
|
|
public static void SetCount50(this ScoreInfo scoreInfo, int value)
|
2019-03-27 07:55:46 +00:00
|
|
|
|
{
|
2021-12-13 08:09:13 +00:00
|
|
|
|
switch (scoreInfo.Ruleset.OnlineID)
|
2019-12-03 06:28:10 +00:00
|
|
|
|
{
|
|
|
|
|
case 0:
|
|
|
|
|
case 3:
|
|
|
|
|
scoreInfo.Statistics[HitResult.Meh] = value;
|
|
|
|
|
break;
|
2020-04-16 09:16:08 +00:00
|
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
|
scoreInfo.Statistics[HitResult.SmallTickHit] = value;
|
|
|
|
|
break;
|
2019-12-03 06:28:10 +00:00
|
|
|
|
}
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
|
|
|
|
public static int? GetCountMiss(this ScoreInfo scoreInfo) =>
|
2020-05-07 03:22:07 +00:00
|
|
|
|
getCount(scoreInfo, HitResult.Miss);
|
2019-12-03 06:28:10 +00:00
|
|
|
|
|
|
|
|
|
public static void SetCountMiss(this ScoreInfo scoreInfo, int value) =>
|
|
|
|
|
scoreInfo.Statistics[HitResult.Miss] = value;
|
2020-05-07 03:22:07 +00:00
|
|
|
|
|
|
|
|
|
private static int? getCount(ScoreInfo scoreInfo, HitResult result)
|
|
|
|
|
{
|
2021-10-27 04:04:41 +00:00
|
|
|
|
if (scoreInfo.Statistics.TryGetValue(result, out int existing))
|
2020-05-07 03:22:07 +00:00
|
|
|
|
return existing;
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2019-03-27 07:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
}
|