mirror of
https://github.com/osukey/osukey.git
synced 2025-04-29 02:37:25 +09:00
290 lines
11 KiB
C#
290 lines
11 KiB
C#
// 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.
|
|
|
|
#nullable disable
|
|
|
|
using System.Diagnostics;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Bindables;
|
|
using osu.Framework.Extensions;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Containers;
|
|
using osu.Framework.Graphics.Shapes;
|
|
using osu.Game.Beatmaps;
|
|
using osu.Game.Graphics.UserInterface;
|
|
using osu.Game.Online.API;
|
|
using osu.Game.Online.API.Requests;
|
|
using osu.Game.Online.API.Requests.Responses;
|
|
using osu.Game.Rulesets;
|
|
using osu.Game.Scoring;
|
|
using osu.Game.Screens.Select.Leaderboards;
|
|
using osuTK;
|
|
using APIUser = osu.Game.Online.API.Requests.Responses.APIUser;
|
|
|
|
namespace osu.Game.Overlays.BeatmapSet.Scores
|
|
{
|
|
public class ScoresContainer : BeatmapSetLayoutSection
|
|
{
|
|
private const int spacing = 15;
|
|
|
|
public readonly Bindable<APIBeatmap> Beatmap = new Bindable<APIBeatmap>();
|
|
private readonly Bindable<IRulesetInfo> ruleset = new Bindable<IRulesetInfo>();
|
|
private readonly Bindable<BeatmapLeaderboardScope> scope = new Bindable<BeatmapLeaderboardScope>(BeatmapLeaderboardScope.Global);
|
|
private readonly IBindable<APIUser> user = new Bindable<APIUser>();
|
|
|
|
private readonly Box background;
|
|
private readonly ScoreTable scoreTable;
|
|
private readonly FillFlowContainer topScoresContainer;
|
|
private readonly LoadingLayer loading;
|
|
private readonly LeaderboardModSelector modSelector;
|
|
private readonly NoScoresPlaceholder noScoresPlaceholder;
|
|
private readonly NotSupporterPlaceholder notSupporterPlaceholder;
|
|
|
|
[Resolved]
|
|
private IAPIProvider api { get; set; }
|
|
|
|
[Resolved]
|
|
private RulesetStore rulesets { get; set; }
|
|
|
|
[Resolved]
|
|
private ScoreManager scoreManager { get; set; }
|
|
|
|
private GetScoresRequest getScoresRequest;
|
|
|
|
private CancellationTokenSource loadCancellationSource;
|
|
|
|
protected APIScoresCollection Scores
|
|
{
|
|
set => Schedule(() =>
|
|
{
|
|
loadCancellationSource?.Cancel();
|
|
loadCancellationSource = new CancellationTokenSource();
|
|
|
|
topScoresContainer.Clear();
|
|
scoreTable.ClearScores();
|
|
scoreTable.Hide();
|
|
|
|
loading.Hide();
|
|
loading.FinishTransforms();
|
|
|
|
if (value?.Scores.Any() != true)
|
|
return;
|
|
|
|
var apiBeatmap = Beatmap.Value;
|
|
|
|
Debug.Assert(apiBeatmap != null);
|
|
|
|
// TODO: temporary. should be removed once `OrderByTotalScore` can accept `IScoreInfo`.
|
|
var beatmapInfo = new BeatmapInfo
|
|
{
|
|
#pragma warning disable 618
|
|
MaxCombo = apiBeatmap.MaxCombo,
|
|
#pragma warning restore 618
|
|
Status = apiBeatmap.Status,
|
|
MD5Hash = apiBeatmap.MD5Hash
|
|
};
|
|
|
|
scoreManager.OrderByTotalScoreAsync(value.Scores.Select(s => s.CreateScoreInfo(rulesets, beatmapInfo)).ToArray(), loadCancellationSource.Token)
|
|
.ContinueWith(task => Schedule(() =>
|
|
{
|
|
if (loadCancellationSource.IsCancellationRequested)
|
|
return;
|
|
|
|
var scores = task.GetResultSafely();
|
|
|
|
var topScore = scores.First();
|
|
|
|
scoreTable.DisplayScores(scores, apiBeatmap.Status.GrantsPerformancePoints());
|
|
scoreTable.Show();
|
|
|
|
var userScore = value.UserScore;
|
|
var userScoreInfo = userScore?.Score.CreateScoreInfo(rulesets, beatmapInfo);
|
|
|
|
topScoresContainer.Add(new DrawableTopScore(topScore));
|
|
|
|
if (userScoreInfo != null && userScoreInfo.OnlineID != topScore.OnlineID)
|
|
topScoresContainer.Add(new DrawableTopScore(userScoreInfo, userScore.Position));
|
|
}), TaskContinuationOptions.OnlyOnRanToCompletion);
|
|
});
|
|
}
|
|
|
|
public ScoresContainer()
|
|
{
|
|
AddRange(new Drawable[]
|
|
{
|
|
background = new Box
|
|
{
|
|
RelativeSizeAxes = Axes.Both,
|
|
},
|
|
new FillFlowContainer
|
|
{
|
|
Anchor = Anchor.TopCentre,
|
|
Origin = Anchor.TopCentre,
|
|
RelativeSizeAxes = Axes.X,
|
|
AutoSizeAxes = Axes.Y,
|
|
Direction = FillDirection.Vertical,
|
|
Padding = new MarginPadding { Horizontal = 50 },
|
|
Margin = new MarginPadding { Vertical = 20 },
|
|
Children = new Drawable[]
|
|
{
|
|
new FillFlowContainer
|
|
{
|
|
RelativeSizeAxes = Axes.X,
|
|
AutoSizeAxes = Axes.Y,
|
|
Direction = FillDirection.Vertical,
|
|
Spacing = new Vector2(0, spacing),
|
|
Children = new Drawable[]
|
|
{
|
|
new LeaderboardScopeSelector
|
|
{
|
|
Anchor = Anchor.TopCentre,
|
|
Origin = Anchor.TopCentre,
|
|
Current = { BindTarget = scope }
|
|
},
|
|
modSelector = new LeaderboardModSelector
|
|
{
|
|
Anchor = Anchor.TopCentre,
|
|
Origin = Anchor.TopCentre,
|
|
Ruleset = { BindTarget = ruleset }
|
|
}
|
|
}
|
|
},
|
|
new Container
|
|
{
|
|
AutoSizeAxes = Axes.Y,
|
|
RelativeSizeAxes = Axes.X,
|
|
Margin = new MarginPadding { Top = spacing },
|
|
Children = new Drawable[]
|
|
{
|
|
noScoresPlaceholder = new NoScoresPlaceholder
|
|
{
|
|
Anchor = Anchor.TopCentre,
|
|
Origin = Anchor.TopCentre,
|
|
Alpha = 0,
|
|
AlwaysPresent = true,
|
|
Margin = new MarginPadding { Vertical = 10 }
|
|
},
|
|
notSupporterPlaceholder = new NotSupporterPlaceholder
|
|
{
|
|
Anchor = Anchor.TopCentre,
|
|
Origin = Anchor.TopCentre,
|
|
Alpha = 0,
|
|
},
|
|
new FillFlowContainer
|
|
{
|
|
RelativeSizeAxes = Axes.X,
|
|
AutoSizeAxes = Axes.Y,
|
|
Direction = FillDirection.Vertical,
|
|
Spacing = new Vector2(0, spacing),
|
|
Children = new Drawable[]
|
|
{
|
|
topScoresContainer = new FillFlowContainer
|
|
{
|
|
RelativeSizeAxes = Axes.X,
|
|
AutoSizeAxes = Axes.Y,
|
|
Direction = FillDirection.Vertical,
|
|
Spacing = new Vector2(0, 5),
|
|
},
|
|
scoreTable = new ScoreTable
|
|
{
|
|
Anchor = Anchor.TopCentre,
|
|
Origin = Anchor.TopCentre,
|
|
}
|
|
}
|
|
},
|
|
}
|
|
}
|
|
},
|
|
},
|
|
loading = new LoadingLayer()
|
|
});
|
|
}
|
|
|
|
[BackgroundDependencyLoader]
|
|
private void load(OverlayColourProvider colourProvider)
|
|
{
|
|
background.Colour = colourProvider.Background5;
|
|
|
|
user.BindTo(api.LocalUser);
|
|
}
|
|
|
|
protected override void LoadComplete()
|
|
{
|
|
base.LoadComplete();
|
|
scope.BindValueChanged(_ => getScores());
|
|
ruleset.BindValueChanged(_ => getScores());
|
|
|
|
modSelector.SelectedMods.CollectionChanged += (_, _) => getScores();
|
|
|
|
Beatmap.BindValueChanged(onBeatmapChanged);
|
|
user.BindValueChanged(onUserChanged, true);
|
|
}
|
|
|
|
private void onBeatmapChanged(ValueChangedEvent<APIBeatmap> beatmap)
|
|
{
|
|
var beatmapRuleset = beatmap.NewValue?.Ruleset;
|
|
|
|
if (ruleset.Value?.OnlineID == beatmapRuleset?.OnlineID)
|
|
{
|
|
modSelector.DeselectAll();
|
|
ruleset.TriggerChange();
|
|
}
|
|
else
|
|
ruleset.Value = beatmapRuleset;
|
|
|
|
scope.Value = BeatmapLeaderboardScope.Global;
|
|
}
|
|
|
|
private void onUserChanged(ValueChangedEvent<APIUser> user)
|
|
{
|
|
if (modSelector.SelectedMods.Any())
|
|
modSelector.DeselectAll();
|
|
else
|
|
getScores();
|
|
}
|
|
|
|
private void getScores()
|
|
{
|
|
getScoresRequest?.Cancel();
|
|
getScoresRequest = null;
|
|
|
|
noScoresPlaceholder.Hide();
|
|
|
|
if (Beatmap.Value == null || Beatmap.Value.OnlineID <= 0 || (Beatmap.Value.Status <= BeatmapOnlineStatus.Pending))
|
|
{
|
|
Scores = null;
|
|
Hide();
|
|
return;
|
|
}
|
|
|
|
if ((scope.Value != BeatmapLeaderboardScope.Global || modSelector.SelectedMods.Count > 0) && !userIsSupporter)
|
|
{
|
|
Scores = null;
|
|
notSupporterPlaceholder.Show();
|
|
return;
|
|
}
|
|
|
|
notSupporterPlaceholder.Hide();
|
|
|
|
Show();
|
|
loading.Show();
|
|
|
|
getScoresRequest = new GetScoresRequest(Beatmap.Value, Beatmap.Value.Ruleset, scope.Value, modSelector.SelectedMods);
|
|
getScoresRequest.Success += scores =>
|
|
{
|
|
Scores = scores;
|
|
|
|
if (!scores.Scores.Any())
|
|
noScoresPlaceholder.ShowWithScope(scope.Value);
|
|
};
|
|
|
|
api.Queue(getScoresRequest);
|
|
}
|
|
|
|
private bool userIsSupporter => api.IsLoggedIn && api.LocalUser.Value.IsSupporter;
|
|
}
|
|
}
|