1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-22 17:52:57 +08:00
osu-lazer/osu.Game.Rulesets.Catch/Scoring/CatchScoreProcessor.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

138 lines
4.7 KiB
C#
Raw Normal View History

// 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.
2018-04-13 17:19:50 +08:00
2023-05-09 18:33:33 +08:00
using System;
2024-01-03 04:58:44 +08:00
using System.Collections.Generic;
2023-05-09 18:33:33 +08:00
using osu.Game.Rulesets.Judgements;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Scoring;
2023-11-29 18:05:24 +08:00
using osu.Game.Scoring;
2018-04-13 17:19:50 +08:00
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Catch.Scoring
{
2019-12-11 16:25:06 +08:00
public partial class CatchScoreProcessor : ScoreProcessor
{
2023-11-29 18:05:24 +08: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 18:33:33 +08:00
private const int combo_cap = 200;
private const double combo_base = 4;
2024-01-03 04:58:44 +08:00
private double fruitTinyScale;
public CatchScoreProcessor()
: base(new CatchRuleset())
{
}
2024-01-03 04:58:44 +08: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 16:27:02 +08:00
protected override double ComputeTotalScore(double comboProgress, double accuracyProgress, double bonusPortion)
2023-05-09 18:33:33 +08:00
{
2024-01-03 04:58:44 +08: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 13:37:26 +08:00
+ bonusPortion;
2023-05-09 18:33:33 +08:00
}
2024-01-03 04:58:44 +08: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 13:37:26 +08:00
protected override double GetComboScoreChange(JudgementResult result)
2024-01-03 04:58:44 +08: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 18:05:24 +08:00
public override ScoreRank RankFromScore(double accuracy, IReadOnlyDictionary<HitResult, int> results)
2023-11-29 18:05:24 +08: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);
}
}
}
}