// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Collections.Generic; using System.Linq; using osu.Game.Beatmaps; using osu.Game.Models; using osu.Game.Online.API.Requests.Responses; using osu.Game.Rulesets.Scoring; using osu.Game.Screens.Select.Leaderboards; using Realms; namespace osu.Game.Scoring { public static class ScoreInfoExtensions { /// /// A user-presentable display title representing this score. /// public static string GetDisplayTitle(this IScoreInfo scoreInfo) => $"{scoreInfo.User.Username} playing {scoreInfo.Beatmap?.GetDisplayTitle() ?? "unknown"}"; /// /// Orders an array of s by total score. /// /// The array of s to reorder. /// The given ordered by decreasing total score. public static IEnumerable OrderByTotalScore(this IEnumerable scores) => scores.OrderByDescending(s => s.TotalScore) .ThenBy(s => s.OnlineID) // Local scores may not have an online ID. Fall back to date in these cases. .ThenBy(s => s.Date); /// /// Orders an array of s by the selected . /// /// The array of s to reorder. /// The attribute to sort the scores by. /// The given ordered by the selected mode. public static IEnumerable OrderByCriteria(this IEnumerable scores, LeaderboardSortMode leaderboardSortMode) { switch (leaderboardSortMode) { case LeaderboardSortMode.Score: return scores.OrderByDescending(s => s.TotalScore); case LeaderboardSortMode.Accuracy: return scores.OrderByDescending(s => s.Accuracy).ThenByDescending(s => s.TotalScore); case LeaderboardSortMode.MaxCombo: return scores.OrderByDescending(s => s.MaxCombo).ThenByDescending(s => s.TotalScore); case LeaderboardSortMode.Misses: return scores.OrderBy(s => s.Statistics.GetValueOrDefault(HitResult.Miss, 0)).ThenByDescending(s => s.TotalScore); case LeaderboardSortMode.Date: return scores.OrderByDescending(s => s.Date); default: throw new ArgumentOutOfRangeException(nameof(leaderboardSortMode), leaderboardSortMode, null); } } /// /// Retrieves the maximum achievable combo for the provided score. /// /// The to compute the maximum achievable combo for. /// The maximum achievable combo. public static int GetMaximumAchievableCombo(this ScoreInfo score) => score.MaximumStatistics.Where(kvp => kvp.Key.AffectsCombo()).Sum(kvp => kvp.Value); /// /// Performs a realm filter that returns all scores that belong to the user with the given . /// (for guests) is supported. /// /// /// All guest scores (with user ID of ), /// as well as scores of unknown provenance (with default user ID of 1, see ), /// will be treated as if they belong to the local user. /// This may not be necessarily considered fully correct in some circumstances, but in most cases it is the desired effect. /// public static IQueryable GetAllLocalScoresForUser(this Realm realm, int? userId) { return realm.All() .Filter($@"({nameof(ScoreInfo.User)}.{nameof(RealmUser.OnlineID)} == $0 || {nameof(ScoreInfo.User)}.{nameof(RealmUser.OnlineID)} <= 1)" + $@" && {nameof(ScoreInfo.BeatmapInfo)}.{nameof(BeatmapInfo.Hash)} == {nameof(ScoreInfo.BeatmapHash)}" + $@" && {nameof(ScoreInfo.DeletePending)} == false", userId); } } }