2019-01-24 08:43:03 +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.
2018-11-28 07:47:10 +00:00
2018-11-28 08:19:58 +00:00
using System ;
using System.Collections.Generic ;
2019-06-21 15:32:47 +00:00
using System.IO ;
2018-11-28 08:19:58 +00:00
using System.Linq ;
using System.Linq.Expressions ;
2020-09-09 08:37:11 +00:00
using System.Threading ;
2021-06-27 04:06:20 +00:00
using System.Threading.Tasks ;
2020-08-28 10:16:46 +00:00
using JetBrains.Annotations ;
2020-08-28 12:34:34 +00:00
using osu.Framework.Bindables ;
2018-11-28 07:47:10 +00:00
using osu.Framework.Platform ;
2021-09-01 11:56:23 +00:00
using osu.Framework.Threading ;
2018-11-28 07:47:10 +00:00
using osu.Game.Beatmaps ;
2020-08-28 12:34:34 +00:00
using osu.Game.Configuration ;
2018-11-28 07:47:10 +00:00
using osu.Game.Database ;
2021-09-30 09:21:16 +00:00
using osu.Game.IO ;
2018-11-28 07:47:10 +00:00
using osu.Game.IO.Archives ;
2019-06-11 20:01:57 +00:00
using osu.Game.Online.API ;
2021-09-30 09:21:16 +00:00
using osu.Game.Overlays.Notifications ;
2018-11-28 07:47:10 +00:00
using osu.Game.Rulesets ;
2021-03-18 10:19:53 +00:00
using osu.Game.Rulesets.Judgements ;
2020-08-28 10:16:46 +00:00
using osu.Game.Rulesets.Scoring ;
2018-11-28 07:47:10 +00:00
namespace osu.Game.Scoring
{
2021-10-18 07:10:37 +00:00
public class ScoreManager : IModelManager < ScoreInfo > , IModelFileManager < ScoreInfo , ScoreFileInfo > , IModelDownloader < ScoreInfo >
2018-11-28 07:47:10 +00:00
{
2021-09-01 11:56:23 +00:00
private readonly Scheduler scheduler ;
2020-11-06 04:14:23 +00:00
private readonly Func < BeatmapDifficultyCache > difficulties ;
2020-08-28 12:34:34 +00:00
private readonly OsuConfigManager configManager ;
2021-09-30 09:21:16 +00:00
private readonly ScoreModelManager scoreModelManager ;
private readonly ScoreModelDownloader scoreModelDownloader ;
2020-08-28 12:34:34 +00:00
2021-09-01 11:56:23 +00:00
public ScoreManager ( RulesetStore rulesets , Func < BeatmapManager > beatmaps , Storage storage , IAPIProvider api , IDatabaseContextFactory contextFactory , Scheduler scheduler ,
IIpcHost importHost = null , Func < BeatmapDifficultyCache > difficulties = null , OsuConfigManager configManager = null )
2018-11-28 07:47:10 +00:00
{
2021-09-01 11:56:23 +00:00
this . scheduler = scheduler ;
2020-08-28 10:16:46 +00:00
this . difficulties = difficulties ;
2020-08-28 12:34:34 +00:00
this . configManager = configManager ;
2018-11-28 08:19:58 +00:00
2021-09-30 09:21:16 +00:00
scoreModelManager = new ScoreModelManager ( rulesets , beatmaps , storage , contextFactory , importHost ) ;
scoreModelDownloader = new ScoreModelDownloader ( scoreModelManager , api , importHost ) ;
2018-11-28 07:47:10 +00:00
}
2018-11-28 08:19:58 +00:00
2021-09-30 09:21:16 +00:00
public Score GetScore ( ScoreInfo score ) = > scoreModelManager . GetScore ( score ) ;
2021-06-27 04:06:20 +00:00
2021-09-30 09:21:16 +00:00
public List < ScoreInfo > GetAllUsableScores ( ) = > scoreModelManager . GetAllUsableScores ( ) ;
2018-11-29 09:30:43 +00:00
2021-09-30 09:21:16 +00:00
public IEnumerable < ScoreInfo > QueryScores ( Expression < Func < ScoreInfo , bool > > query ) = > scoreModelManager . QueryScores ( query ) ;
2019-06-11 20:01:57 +00:00
2021-09-30 09:21:16 +00:00
public ScoreInfo Query ( Expression < Func < ScoreInfo , bool > > query ) = > scoreModelManager . Query ( query ) ;
2020-08-28 10:16:46 +00:00
2021-09-01 06:41:52 +00:00
/// <summary>
/// Orders an array of <see cref="ScoreInfo"/>s by total score.
/// </summary>
/// <param name="scores">The array of <see cref="ScoreInfo"/>s to reorder.</param>
/// <param name="cancellationToken">A <see cref="CancellationToken"/> to cancel the process.</param>
/// <returns>The given <paramref name="scores"/> ordered by decreasing total score.</returns>
public async Task < ScoreInfo [ ] > OrderByTotalScoreAsync ( ScoreInfo [ ] scores , CancellationToken cancellationToken = default )
2021-08-31 12:36:31 +00:00
{
var difficultyCache = difficulties ? . Invoke ( ) ;
2021-09-07 09:52:25 +00:00
if ( difficultyCache ! = null )
2021-08-31 12:36:31 +00:00
{
2021-09-07 09:52:25 +00:00
// Compute difficulties asynchronously first to prevent blocking via the GetTotalScore() call below.
foreach ( var s in scores )
{
2021-10-04 08:35:53 +00:00
await difficultyCache . GetDifficultyAsync ( s . BeatmapInfo , s . Ruleset , s . Mods , cancellationToken ) . ConfigureAwait ( false ) ;
2021-09-07 09:52:25 +00:00
cancellationToken . ThrowIfCancellationRequested ( ) ;
}
2021-08-31 12:36:31 +00:00
}
2021-10-08 05:23:53 +00:00
var totalScores = await Task . WhenAll ( scores . Select ( s = > GetTotalScoreAsync ( s , cancellationToken : cancellationToken ) ) ) . ConfigureAwait ( false ) ;
2021-10-10 07:50:55 +00:00
return scores . Select ( ( score , index ) = > ( score , totalScore : totalScores [ index ] ) )
2021-10-10 06:43:24 +00:00
. OrderByDescending ( g = > g . totalScore )
2021-10-10 06:47:39 +00:00
. ThenBy ( g = > g . score . OnlineScoreID )
2021-10-10 06:43:24 +00:00
. Select ( g = > g . score )
2021-09-07 09:52:25 +00:00
. ToArray ( ) ;
2021-08-31 12:36:31 +00:00
}
2020-09-09 08:04:02 +00:00
/// <summary>
/// Retrieves a bindable that represents the total score of a <see cref="ScoreInfo"/>.
/// </summary>
/// <remarks>
/// Responds to changes in the currently-selected <see cref="ScoringMode"/>.
/// </remarks>
/// <param name="score">The <see cref="ScoreInfo"/> to retrieve the bindable for.</param>
/// <returns>The bindable containing the total score.</returns>
2021-09-01 11:56:23 +00:00
public Bindable < long > GetBindableTotalScore ( [ NotNull ] ScoreInfo score )
2020-08-28 10:16:46 +00:00
{
2021-08-30 10:33:09 +00:00
var bindable = new TotalScoreBindable ( score , this ) ;
2020-08-28 12:34:34 +00:00
configManager ? . BindWith ( OsuSetting . ScoreDisplayMode , bindable . ScoringMode ) ;
return bindable ;
}
2020-08-28 10:16:46 +00:00
2020-09-09 08:04:02 +00:00
/// <summary>
/// Retrieves a bindable that represents the formatted total score string of a <see cref="ScoreInfo"/>.
/// </summary>
/// <remarks>
/// Responds to changes in the currently-selected <see cref="ScoringMode"/>.
/// </remarks>
/// <param name="score">The <see cref="ScoreInfo"/> to retrieve the bindable for.</param>
/// <returns>The bindable containing the formatted total score string.</returns>
2021-09-01 11:56:23 +00:00
public Bindable < string > GetBindableTotalScoreString ( [ NotNull ] ScoreInfo score ) = > new TotalScoreStringBindable ( GetBindableTotalScore ( score ) ) ;
2020-09-09 08:04:02 +00:00
2021-09-01 06:41:52 +00:00
/// <summary>
/// Retrieves the total score of a <see cref="ScoreInfo"/> in the given <see cref="ScoringMode"/>.
2021-09-01 11:56:23 +00:00
/// The score is returned in a callback that is run on the update thread.
2021-09-01 06:41:52 +00:00
/// </summary>
/// <param name="score">The <see cref="ScoreInfo"/> to calculate the total score of.</param>
2021-09-01 11:56:23 +00:00
/// <param name="callback">The callback to be invoked with the total score.</param>
2021-09-01 06:41:52 +00:00
/// <param name="mode">The <see cref="ScoringMode"/> to return the total score as.</param>
2021-09-01 11:56:23 +00:00
/// <param name="cancellationToken">A <see cref="CancellationToken"/> to cancel the process.</param>
public void GetTotalScore ( [ NotNull ] ScoreInfo score , [ NotNull ] Action < long > callback , ScoringMode mode = ScoringMode . Standardised , CancellationToken cancellationToken = default )
{
GetTotalScoreAsync ( score , mode , cancellationToken )
. ContinueWith ( s = > scheduler . Add ( ( ) = > callback ( s . Result ) ) , TaskContinuationOptions . OnlyOnRanToCompletion ) ;
}
2021-08-30 10:33:09 +00:00
2021-09-01 06:41:52 +00:00
/// <summary>
/// Retrieves the total score of a <see cref="ScoreInfo"/> in the given <see cref="ScoringMode"/>.
/// </summary>
/// <param name="score">The <see cref="ScoreInfo"/> to calculate the total score of.</param>
/// <param name="mode">The <see cref="ScoringMode"/> to return the total score as.</param>
/// <param name="cancellationToken">A <see cref="CancellationToken"/> to cancel the process.</param>
/// <returns>The total score.</returns>
2021-09-01 11:56:23 +00:00
public async Task < long > GetTotalScoreAsync ( [ NotNull ] ScoreInfo score , ScoringMode mode = ScoringMode . Standardised , CancellationToken cancellationToken = default )
2021-08-30 10:33:09 +00:00
{
2021-10-04 08:35:53 +00:00
if ( score . BeatmapInfo = = null )
2021-08-30 10:33:09 +00:00
return score . TotalScore ;
int beatmapMaxCombo ;
double accuracy = score . Accuracy ;
if ( score . IsLegacyScore )
{
if ( score . RulesetID = = 3 )
{
// In osu!stable, a full-GREAT score has 100% accuracy in mania. Along with a full combo, the score becomes indistinguishable from a full-PERFECT score.
// To get around this, recalculate accuracy based on the hit statistics.
// Note: This cannot be applied universally to all legacy scores, as some rulesets (e.g. catch) group multiple judgements together.
double maxBaseScore = score . Statistics . Select ( kvp = > kvp . Value ) . Sum ( ) * Judgement . ToNumericResult ( HitResult . Perfect ) ;
double baseScore = score . Statistics . Select ( kvp = > Judgement . ToNumericResult ( kvp . Key ) * kvp . Value ) . Sum ( ) ;
if ( maxBaseScore > 0 )
accuracy = baseScore / maxBaseScore ;
}
// This score is guaranteed to be an osu!stable score.
// The combo must be determined through either the beatmap's max combo value or the difficulty calculator, as lazer's scoring has changed and the score statistics cannot be used.
2021-10-04 08:35:53 +00:00
if ( score . BeatmapInfo . MaxCombo ! = null )
beatmapMaxCombo = score . BeatmapInfo . MaxCombo . Value ;
2021-08-30 10:33:09 +00:00
else
{
2021-10-04 08:35:53 +00:00
if ( score . BeatmapInfo . ID = = 0 | | difficulties = = null )
2021-08-30 10:33:09 +00:00
{
// We don't have enough information (max combo) to compute the score, so use the provided score.
return score . TotalScore ;
}
// We can compute the max combo locally after the async beatmap difficulty computation.
2021-10-04 08:35:53 +00:00
var difficulty = await difficulties ( ) . GetDifficultyAsync ( score . BeatmapInfo , score . Ruleset , score . Mods , cancellationToken ) . ConfigureAwait ( false ) ;
2021-08-30 10:33:09 +00:00
beatmapMaxCombo = difficulty . MaxCombo ;
}
}
else
{
// This is guaranteed to be a non-legacy score.
// The combo must be determined through the score's statistics, as both the beatmap's max combo and the difficulty calculator will provide osu!stable combo values.
beatmapMaxCombo = Enum . GetValues ( typeof ( HitResult ) ) . OfType < HitResult > ( ) . Where ( r = > r . AffectsCombo ( ) ) . Select ( r = > score . Statistics . GetValueOrDefault ( r ) ) . Sum ( ) ;
}
if ( beatmapMaxCombo = = 0 )
return 0 ;
var ruleset = score . Ruleset . CreateInstance ( ) ;
var scoreProcessor = ruleset . CreateScoreProcessor ( ) ;
scoreProcessor . Mods . Value = score . Mods ;
2021-09-01 06:41:52 +00:00
return ( long ) Math . Round ( scoreProcessor . GetScore ( mode , beatmapMaxCombo , accuracy , ( double ) score . MaxCombo / beatmapMaxCombo , score . Statistics ) ) ;
2021-08-30 10:33:09 +00:00
}
2020-09-09 08:04:02 +00:00
/// <summary>
/// Provides the total score of a <see cref="ScoreInfo"/>. Responds to changes in the currently-selected <see cref="ScoringMode"/>.
/// </summary>
2020-08-28 13:51:19 +00:00
private class TotalScoreBindable : Bindable < long >
2020-08-28 12:34:34 +00:00
{
public readonly Bindable < ScoringMode > ScoringMode = new Bindable < ScoringMode > ( ) ;
2021-09-01 11:56:23 +00:00
private readonly ScoreInfo score ;
private readonly ScoreManager scoreManager ;
private CancellationTokenSource difficultyCalculationCancellationSource ;
2020-09-09 08:04:02 +00:00
/// <summary>
/// Creates a new <see cref="TotalScoreBindable"/>.
/// </summary>
/// <param name="score">The <see cref="ScoreInfo"/> to provide the total score of.</param>
2021-08-30 10:33:09 +00:00
/// <param name="scoreManager">The <see cref="ScoreManager"/>.</param>
public TotalScoreBindable ( ScoreInfo score , ScoreManager scoreManager )
2020-08-28 10:16:46 +00:00
{
2021-09-01 11:56:23 +00:00
this . score = score ;
this . scoreManager = scoreManager ;
ScoringMode . BindValueChanged ( onScoringModeChanged , true ) ;
}
private void onScoringModeChanged ( ValueChangedEvent < ScoringMode > mode )
{
difficultyCalculationCancellationSource ? . Cancel ( ) ;
difficultyCalculationCancellationSource = new CancellationTokenSource ( ) ;
scoreManager . GetTotalScore ( score , s = > Value = s , mode . NewValue , difficultyCalculationCancellationSource . Token ) ;
2020-08-28 13:51:19 +00:00
}
}
2020-09-09 08:04:02 +00:00
/// <summary>
/// Provides the total score of a <see cref="ScoreInfo"/> as a formatted string. Responds to changes in the currently-selected <see cref="ScoringMode"/>.
/// </summary>
2020-08-28 13:51:19 +00:00
private class TotalScoreStringBindable : Bindable < string >
{
// ReSharper disable once PrivateFieldCanBeConvertedToLocalVariable (need to hold a reference)
private readonly IBindable < long > totalScore ;
public TotalScoreStringBindable ( IBindable < long > totalScore )
{
this . totalScore = totalScore ;
this . totalScore . BindValueChanged ( v = > Value = v . NewValue . ToString ( "N0" ) , true ) ;
2020-08-28 12:34:34 +00:00
}
2020-08-28 10:16:46 +00:00
}
2021-09-30 09:21:16 +00:00
#region Implementation of IPostNotifications
public Action < Notification > PostNotification
{
set
{
scoreModelManager . PostNotification = value ;
scoreModelDownloader . PostNotification = value ;
}
}
#endregion
#region Implementation of IModelManager < ScoreInfo >
public IBindable < WeakReference < ScoreInfo > > ItemUpdated = > scoreModelManager . ItemUpdated ;
public IBindable < WeakReference < ScoreInfo > > ItemRemoved = > scoreModelManager . ItemRemoved ;
public Task ImportFromStableAsync ( StableStorage stableStorage )
{
return scoreModelManager . ImportFromStableAsync ( stableStorage ) ;
}
public void Export ( ScoreInfo item )
{
scoreModelManager . Export ( item ) ;
}
public void ExportModelTo ( ScoreInfo model , Stream outputStream )
{
scoreModelManager . ExportModelTo ( model , outputStream ) ;
}
public void Update ( ScoreInfo item )
{
scoreModelManager . Update ( item ) ;
}
public bool Delete ( ScoreInfo item )
{
return scoreModelManager . Delete ( item ) ;
}
public void Delete ( List < ScoreInfo > items , bool silent = false )
{
scoreModelManager . Delete ( items , silent ) ;
}
public void Undelete ( List < ScoreInfo > items , bool silent = false )
{
scoreModelManager . Undelete ( items , silent ) ;
}
public void Undelete ( ScoreInfo item )
{
scoreModelManager . Undelete ( item ) ;
}
public Task Import ( params string [ ] paths )
{
return scoreModelManager . Import ( paths ) ;
}
public Task Import ( params ImportTask [ ] tasks )
{
return scoreModelManager . Import ( tasks ) ;
}
public IEnumerable < string > HandledExtensions = > scoreModelManager . HandledExtensions ;
2021-09-30 10:33:12 +00:00
public Task < IEnumerable < ILive < ScoreInfo > > > Import ( ProgressNotification notification , params ImportTask [ ] tasks )
2021-09-30 09:21:16 +00:00
{
return scoreModelManager . Import ( notification , tasks ) ;
}
2021-09-30 10:33:12 +00:00
public Task < ILive < ScoreInfo > > Import ( ImportTask task , bool lowPriority = false , CancellationToken cancellationToken = default )
2021-09-30 09:21:16 +00:00
{
return scoreModelManager . Import ( task , lowPriority , cancellationToken ) ;
}
2021-09-30 10:33:12 +00:00
public Task < ILive < ScoreInfo > > Import ( ArchiveReader archive , bool lowPriority = false , CancellationToken cancellationToken = default )
2021-09-30 09:21:16 +00:00
{
return scoreModelManager . Import ( archive , lowPriority , cancellationToken ) ;
}
2021-09-30 10:33:12 +00:00
public Task < ILive < ScoreInfo > > Import ( ScoreInfo item , ArchiveReader archive = null , bool lowPriority = false , CancellationToken cancellationToken = default )
2021-09-30 09:21:16 +00:00
{
return scoreModelManager . Import ( item , archive , lowPriority , cancellationToken ) ;
}
public bool IsAvailableLocally ( ScoreInfo model )
{
return scoreModelManager . IsAvailableLocally ( model ) ;
}
#endregion
#region Implementation of IModelFileManager < in ScoreInfo , in ScoreFileInfo >
public void ReplaceFile ( ScoreInfo model , ScoreFileInfo file , Stream contents , string filename = null )
{
scoreModelManager . ReplaceFile ( model , file , contents , filename ) ;
}
public void DeleteFile ( ScoreInfo model , ScoreFileInfo file )
{
scoreModelManager . DeleteFile ( model , file ) ;
}
public void AddFile ( ScoreInfo model , Stream contents , string filename )
{
scoreModelManager . AddFile ( model , contents , filename ) ;
}
#endregion
#region Implementation of IModelDownloader < ScoreInfo >
public IBindable < WeakReference < ArchiveDownloadRequest < ScoreInfo > > > DownloadBegan = > scoreModelDownloader . DownloadBegan ;
public IBindable < WeakReference < ArchiveDownloadRequest < ScoreInfo > > > DownloadFailed = > scoreModelDownloader . DownloadFailed ;
public bool Download ( ScoreInfo model , bool minimiseDownloadSize )
{
return scoreModelDownloader . Download ( model , minimiseDownloadSize ) ;
}
public ArchiveDownloadRequest < ScoreInfo > GetExistingDownload ( ScoreInfo model )
{
return scoreModelDownloader . GetExistingDownload ( model ) ;
}
#endregion
#region Implementation of IPresentImports < ScoreInfo >
2021-10-04 07:35:55 +00:00
public Action < IEnumerable < ILive < ScoreInfo > > > PostImport
2021-09-30 09:21:16 +00:00
{
2021-10-04 07:35:55 +00:00
set = > scoreModelManager . PostImport = value ;
2021-09-30 09:21:16 +00:00
}
#endregion
2018-11-28 07:47:10 +00:00
}
}