osu/osu.Game/Screens/Select/Carousel/TopLocalRank.cs

108 lines
3.6 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.
using System;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
2020-04-07 06:30:06 +00:00
using osu.Framework.Graphics;
using osu.Framework.Threading;
using osu.Game.Beatmaps;
2021-12-13 10:01:20 +00:00
using osu.Game.Database;
using osu.Game.Online.API;
2020-04-07 06:31:22 +00:00
using osu.Game.Online.Leaderboards;
using osu.Game.Rulesets;
using osu.Game.Scoring;
using Realms;
2020-04-07 06:31:22 +00:00
namespace osu.Game.Screens.Select.Carousel
{
public class TopLocalRank : UpdateableRank
{
2021-10-02 15:55:29 +00:00
private readonly BeatmapInfo beatmapInfo;
2020-04-07 05:49:24 +00:00
[Resolved]
private IBindable<RulesetInfo> ruleset { get; set; }
2021-12-13 10:01:20 +00:00
[Resolved]
private RealmContextFactory realmFactory { get; set; }
2020-04-07 05:49:24 +00:00
[Resolved]
private IAPIProvider api { get; set; }
2021-10-02 15:55:29 +00:00
public TopLocalRank(BeatmapInfo beatmapInfo)
2020-04-04 19:42:13 +00:00
: base(null)
{
2021-10-02 15:55:29 +00:00
this.beatmapInfo = beatmapInfo;
}
[BackgroundDependencyLoader]
2020-04-07 05:49:24 +00:00
private void load()
{
ruleset.ValueChanged += _ => fetchAndLoadTopScore();
fetchAndLoadTopScore();
}
protected override void LoadComplete()
{
base.LoadComplete();
scoreSubscription = realmFactory.Context.All<ScoreInfo>()
.Filter($"{nameof(ScoreInfo.BeatmapInfo)}.{nameof(BeatmapInfo.ID)} = $0", beatmapInfo.ID)
.QueryAsyncWithNotifications((_, changes, ___) =>
{
if (changes == null)
return;
fetchTopScoreRank();
});
}
private IDisposable scoreSubscription;
2020-04-07 06:30:06 +00:00
private ScheduledDelegate scheduledRankUpdate;
private void fetchAndLoadTopScore()
{
// TODO: this lookup likely isn't required, we can use the results of the subscription directly.
2021-12-13 10:01:20 +00:00
var rank = fetchTopScoreRank();
scheduledRankUpdate = Scheduler.Add(() =>
2020-04-07 06:30:06 +00:00
{
Rank = rank;
2020-04-07 06:30:06 +00:00
// Required since presence is changed via IsPresent override
Invalidate(Invalidation.Presence);
});
}
2020-04-07 06:30:06 +00:00
// We're present if a rank is set, or if there is a pending rank update (IsPresent = true is required for the scheduler to run).
public override bool IsPresent => base.IsPresent && (Rank != null || scheduledRankUpdate?.Completed == false);
2021-12-13 10:01:20 +00:00
private ScoreRank? fetchTopScoreRank()
{
2021-12-13 10:01:20 +00:00
if (realmFactory == null || beatmapInfo == null || ruleset?.Value == null || api?.LocalUser.Value == null)
return null;
2021-12-13 10:01:20 +00:00
using (var realm = realmFactory.CreateContext())
{
return realm.All<ScoreInfo>()
.AsEnumerable()
// TODO: update to use a realm filter directly (or at least figure out the beatmap part to reduce scope).
.Where(s => s.UserID == api.LocalUser.Value.Id && s.BeatmapInfoID == beatmapInfo.ID && s.RulesetID == ruleset.Value.ID && !s.DeletePending)
2021-12-13 10:01:20 +00:00
.OrderByDescending(s => s.TotalScore)
.FirstOrDefault()
?.Rank;
}
}
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
scoreSubscription?.Dispose();
}
}
}