2023-05-18 11:52:40 +00:00
|
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
2019-01-24 08:43:03 +00:00
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2023-05-09 10:33:33 +00:00
|
|
|
|
using System;
|
2024-01-02 20:58:44 +00:00
|
|
|
|
using System.Collections.Generic;
|
2023-05-09 10:33:33 +00:00
|
|
|
|
using osu.Game.Rulesets.Judgements;
|
2017-04-18 07:05:58 +00:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2023-11-29 10:05:24 +00:00
|
|
|
|
using osu.Game.Scoring;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2017-04-18 07:05:58 +00:00
|
|
|
|
namespace osu.Game.Rulesets.Catch.Scoring
|
2017-03-16 03:40:35 +00:00
|
|
|
|
{
|
2019-12-11 08:25:06 +00:00
|
|
|
|
public partial class CatchScoreProcessor : ScoreProcessor
|
2017-03-16 03:40:35 +00:00
|
|
|
|
{
|
2023-11-29 10:05:24 +00:00
|
|
|
|
private const double accuracy_cutoff_x = 1;
|
|
|
|
|
private const double accuracy_cutoff_s = 0.98;
|
|
|
|
|
private const double accuracy_cutoff_a = 0.94;
|
|
|
|
|
private const double accuracy_cutoff_b = 0.9;
|
|
|
|
|
private const double accuracy_cutoff_c = 0.85;
|
|
|
|
|
private const double accuracy_cutoff_d = 0;
|
|
|
|
|
|
2023-05-09 10:33:33 +00:00
|
|
|
|
private const int combo_cap = 200;
|
|
|
|
|
private const double combo_base = 4;
|
|
|
|
|
|
2024-01-02 20:58:44 +00:00
|
|
|
|
private double fruitTinyScale;
|
|
|
|
|
|
2023-05-18 11:52:40 +00:00
|
|
|
|
public CatchScoreProcessor()
|
|
|
|
|
: base(new CatchRuleset())
|
2022-03-14 06:51:10 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-02 20:58:44 +00:00
|
|
|
|
protected override void Reset(bool storeResults)
|
|
|
|
|
{
|
|
|
|
|
base.Reset(storeResults);
|
|
|
|
|
|
|
|
|
|
// large ticks are *purposefully* not counted to match stable
|
|
|
|
|
int fruitTinyScaleDivisor = MaximumResultCounts.GetValueOrDefault(HitResult.SmallTickHit) + MaximumResultCounts.GetValueOrDefault(HitResult.Great);
|
|
|
|
|
fruitTinyScale = fruitTinyScaleDivisor == 0
|
|
|
|
|
? 0
|
|
|
|
|
: (double)MaximumResultCounts.GetValueOrDefault(HitResult.SmallTickHit) / fruitTinyScaleDivisor;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-19 08:27:02 +00:00
|
|
|
|
protected override double ComputeTotalScore(double comboProgress, double accuracyProgress, double bonusPortion)
|
2023-05-09 10:33:33 +00:00
|
|
|
|
{
|
2024-01-02 20:58:44 +00:00
|
|
|
|
const int max_tiny_droplets_portion = 400000;
|
|
|
|
|
|
|
|
|
|
double comboPortion = 1000000 - max_tiny_droplets_portion + max_tiny_droplets_portion * (1 - fruitTinyScale);
|
|
|
|
|
double dropletsPortion = max_tiny_droplets_portion * fruitTinyScale;
|
|
|
|
|
double dropletsHit = MaximumResultCounts.GetValueOrDefault(HitResult.SmallTickHit) == 0
|
|
|
|
|
? 0
|
|
|
|
|
: (double)ScoreResultCounts.GetValueOrDefault(HitResult.SmallTickHit) / MaximumResultCounts.GetValueOrDefault(HitResult.SmallTickHit);
|
|
|
|
|
|
|
|
|
|
return comboPortion * comboProgress
|
|
|
|
|
+ dropletsPortion * dropletsHit
|
2023-05-19 05:37:26 +00:00
|
|
|
|
+ bonusPortion;
|
2023-05-09 10:33:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-02 20:58:44 +00:00
|
|
|
|
public override int GetBaseScoreForResult(HitResult result)
|
|
|
|
|
{
|
|
|
|
|
switch (result)
|
|
|
|
|
{
|
|
|
|
|
// dirty hack to emulate accuracy on stable weighting every object equally in accuracy portion
|
|
|
|
|
case HitResult.Great:
|
|
|
|
|
case HitResult.LargeTickHit:
|
|
|
|
|
case HitResult.SmallTickHit:
|
|
|
|
|
return 300;
|
|
|
|
|
|
|
|
|
|
case HitResult.LargeBonus:
|
|
|
|
|
return 200;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return base.GetBaseScoreForResult(result);
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-19 05:37:26 +00:00
|
|
|
|
protected override double GetComboScoreChange(JudgementResult result)
|
2024-01-02 20:58:44 +00:00
|
|
|
|
{
|
|
|
|
|
double baseIncrease = 0;
|
|
|
|
|
|
|
|
|
|
switch (result.Type)
|
|
|
|
|
{
|
|
|
|
|
case HitResult.Great:
|
|
|
|
|
baseIncrease = 300;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case HitResult.LargeTickHit:
|
|
|
|
|
baseIncrease = 100;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return baseIncrease * Math.Min(Math.Max(0.5, Math.Log(result.ComboAfterJudgement, combo_base)), Math.Log(combo_cap, combo_base));
|
|
|
|
|
}
|
2023-11-29 10:05:24 +00:00
|
|
|
|
|
2024-01-22 18:56:30 +00:00
|
|
|
|
public override ScoreRank RankFromScore(double accuracy, IReadOnlyDictionary<HitResult, int> results)
|
2023-11-29 10:05:24 +00:00
|
|
|
|
{
|
|
|
|
|
if (accuracy == accuracy_cutoff_x)
|
|
|
|
|
return ScoreRank.X;
|
|
|
|
|
if (accuracy >= accuracy_cutoff_s)
|
|
|
|
|
return ScoreRank.S;
|
|
|
|
|
if (accuracy >= accuracy_cutoff_a)
|
|
|
|
|
return ScoreRank.A;
|
|
|
|
|
if (accuracy >= accuracy_cutoff_b)
|
|
|
|
|
return ScoreRank.B;
|
|
|
|
|
if (accuracy >= accuracy_cutoff_c)
|
|
|
|
|
return ScoreRank.C;
|
|
|
|
|
|
|
|
|
|
return ScoreRank.D;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override double AccuracyCutoffFromRank(ScoreRank rank)
|
|
|
|
|
{
|
|
|
|
|
switch (rank)
|
|
|
|
|
{
|
|
|
|
|
case ScoreRank.X:
|
|
|
|
|
case ScoreRank.XH:
|
|
|
|
|
return accuracy_cutoff_x;
|
|
|
|
|
|
|
|
|
|
case ScoreRank.S:
|
|
|
|
|
case ScoreRank.SH:
|
|
|
|
|
return accuracy_cutoff_s;
|
|
|
|
|
|
|
|
|
|
case ScoreRank.A:
|
|
|
|
|
return accuracy_cutoff_a;
|
|
|
|
|
|
|
|
|
|
case ScoreRank.B:
|
|
|
|
|
return accuracy_cutoff_b;
|
|
|
|
|
|
|
|
|
|
case ScoreRank.C:
|
|
|
|
|
return accuracy_cutoff_c;
|
|
|
|
|
|
|
|
|
|
case ScoreRank.D:
|
|
|
|
|
return accuracy_cutoff_d;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
throw new ArgumentOutOfRangeException(nameof(rank), rank, null);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-03-16 03:40:35 +00:00
|
|
|
|
}
|
|
|
|
|
}
|