1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-12 00:27:25 +08:00
osu-lazer/osu.Game/Screens/Select/Leaderboards/Leaderboard.cs

379 lines
12 KiB
C#
Raw Normal View History

2017-03-04 15:37:34 +08:00
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System;
using System.Collections.Generic;
2017-03-14 23:58:22 +08:00
using OpenTK;
2017-03-16 00:57:41 +08:00
using OpenTK.Graphics;
using osu.Framework.Allocation;
2017-03-16 00:57:41 +08:00
using osu.Framework.Extensions.Color4Extensions;
2017-03-04 15:37:34 +08:00
using osu.Framework.Graphics;
2017-03-16 00:57:41 +08:00
using osu.Framework.Graphics.Colour;
2017-03-04 15:37:34 +08:00
using osu.Framework.Graphics.Containers;
using osu.Framework.Threading;
2017-07-26 12:22:46 +08:00
using osu.Game.Beatmaps;
using osu.Game.Graphics.Containers;
using osu.Game.Graphics.UserInterface;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Online.API;
using osu.Game.Online.API.Requests;
using System.Linq;
using osu.Game.Graphics.Sprites;
using osu.Game.Graphics;
using osu.Framework.Logging;
2017-11-25 23:53:36 +08:00
using osu.Game.Rulesets;
using osu.Framework.Input;
2017-03-04 15:37:34 +08:00
namespace osu.Game.Screens.Select.Leaderboards
{
public class Leaderboard : Container
{
private const double fade_duration = 200;
private readonly ScrollContainer scrollContainer;
2017-11-26 17:33:49 +08:00
private readonly Container placeholderContainer;
private FillFlowContainer<LeaderboardScore> scrollFlow;
2017-03-04 15:37:34 +08:00
2017-04-11 13:01:47 +08:00
public Action<Score> ScoreSelected;
2017-06-13 14:54:26 +08:00
private readonly LoadingAnimation loading;
2017-03-15 13:38:38 +08:00
private IEnumerable<Score> scores;
2017-03-15 13:38:38 +08:00
public IEnumerable<Score> Scores
2017-03-04 15:37:34 +08:00
{
2017-03-04 16:05:31 +08:00
get { return scores; }
set
2017-03-04 15:37:34 +08:00
{
2017-03-04 16:05:31 +08:00
scores = value;
placeholderContainer.FadeOut(fade_duration);
2017-12-20 20:50:45 +08:00
scrollFlow?.FadeOut(fade_duration).Expire();
scrollFlow = null;
loading.Hide();
if (scores == null)
2017-03-15 13:38:38 +08:00
return;
2017-11-26 17:33:49 +08:00
if (!scores.Any())
{
replacePlaceholder(new MessagePlaceholder(@"No records yet!"));
return;
}
// schedule because we may not be loaded yet (LoadComponentAsync complains).
Schedule(() =>
2017-03-04 15:37:34 +08:00
{
LoadComponentAsync(new FillFlowContainer<LeaderboardScore>
2017-03-15 16:07:56 +08:00
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Spacing = new Vector2(0f, 5f),
Padding = new MarginPadding { Top = 10, Bottom = 5 },
ChildrenEnumerable = scores.Select((s, index) => new LeaderboardScore(s, index + 1) { Action = () => ScoreSelected?.Invoke(s) })
}, f =>
{
scrollContainer.Add(scrollFlow = f);
2017-03-04 16:05:31 +08:00
int i = 0;
foreach (var s in f.Children)
{
using (s.BeginDelayedSequence(i++ * 50, true))
s.Show();
}
scrollContainer.ScrollTo(0f, false);
});
});
2017-03-04 15:37:34 +08:00
}
}
private LeaderboardScope scope;
public LeaderboardScope Scope
{
get { return scope; }
set
{
if (value == scope) return;
scope = value;
UpdateScores();
}
}
2017-03-04 15:37:34 +08:00
public Leaderboard()
{
Children = new Drawable[]
{
scrollContainer = new OsuScrollContainer
2017-03-04 15:37:34 +08:00
{
RelativeSizeAxes = Axes.Both,
2017-05-30 15:33:26 +08:00
ScrollbarVisible = false,
2017-03-04 15:37:34 +08:00
},
loading = new LoadingAnimation(),
placeholderContainer = new Container
{
Alpha = 0,
AutoSizeAxes = Axes.Both,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
2017-03-04 15:37:34 +08:00
};
}
2017-03-16 12:15:06 +08:00
private APIAccess api;
private BeatmapInfo beatmap;
private OsuGame osuGame;
private ScheduledDelegate pendingBeatmapSwitch;
public BeatmapInfo Beatmap
{
get { return beatmap; }
set
{
if (beatmap == value) return;
beatmap = value;
Scores = null;
pendingBeatmapSwitch?.Cancel();
pendingBeatmapSwitch = Schedule(UpdateScores);
}
}
[BackgroundDependencyLoader(permitNulls: true)]
private void load(APIAccess api, OsuGame osuGame)
{
this.api = api;
this.osuGame = osuGame;
if (osuGame != null)
2017-11-25 23:53:36 +08:00
osuGame.Ruleset.ValueChanged += handleRulesetChange;
if (api != null)
api.OnStateChange += handleApiStateChange;
2017-11-25 23:53:36 +08:00
}
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
if (osuGame != null)
osuGame.Ruleset.ValueChanged -= handleRulesetChange;
}
private GetScoresRequest getScoresRequest;
private void handleRulesetChange(RulesetInfo ruleset) => UpdateScores();
2017-11-25 23:53:36 +08:00
private void handleApiStateChange(APIState oldState, APIState newState)
{
if (Scope == LeaderboardScope.Local)
// No need to respond to API state change while current scope is local
return;
if (newState == APIState.Online)
UpdateScores();
}
protected virtual void UpdateScores()
{
if (!IsLoaded) return;
getScoresRequest?.Cancel();
getScoresRequest = null;
Scores = null;
if (Scope == LeaderboardScope.Local)
{
// TODO: get local scores from wherever here.
Scores = Enumerable.Empty<Score>();
return;
}
if (api?.IsLoggedIn != true)
{
replacePlaceholder(new MessagePlaceholder(@"Please login to view online leaderboards!"));
return;
}
if (Beatmap?.OnlineBeatmapID == null)
{
replacePlaceholder(new RetrievalFailurePlaceholder
{
OnRetry = UpdateScores,
});
return;
}
loading.Show();
if (Scope != LeaderboardScope.Global && !api.LocalUser.Value.IsSupporter)
{
loading.Hide();
replacePlaceholder(new MessagePlaceholder(@"Please invest in a supporter tag to view this leaderboard!"));
return;
}
2017-12-20 20:06:33 +08:00
getScoresRequest = new GetScoresRequest(Beatmap, osuGame?.Ruleset.Value ?? Beatmap.Ruleset, Scope);
getScoresRequest.Success += r =>
{
Scores = r.Scores;
};
getScoresRequest.Failure += OnUpdateFailed;
api.Queue(getScoresRequest);
}
protected void OnUpdateFailed(Exception e)
{
if (e is OperationCanceledException) return;
Scores = null;
replacePlaceholder(new RetrievalFailurePlaceholder
{
OnRetry = UpdateScores,
});
Logger.Error(e, @"Couldn't fetch beatmap scores!");
}
private void replacePlaceholder(Drawable placeholder)
{
placeholderContainer.FadeOutFromOne(fade_duration, Easing.OutQuint);
placeholderContainer.Clear(true);
placeholderContainer.Child = placeholder;
placeholderContainer.FadeInFromZero(fade_duration, Easing.OutQuint);
}
2017-03-16 12:15:06 +08:00
protected override void Update()
{
base.Update();
2017-03-19 20:49:29 +08:00
var fadeStart = scrollContainer.Current + scrollContainer.DrawHeight;
2017-03-16 12:15:06 +08:00
2017-03-19 20:49:29 +08:00
if (!scrollContainer.IsScrolledToEnd())
fadeStart -= LeaderboardScore.HEIGHT;
if (scrollFlow == null) return;
2017-03-19 20:49:29 +08:00
foreach (var c in scrollFlow.Children)
2017-03-16 12:15:06 +08:00
{
2017-03-19 20:49:29 +08:00
var topY = c.ToSpaceOfOtherDrawable(Vector2.Zero, scrollFlow).Y;
2017-03-18 06:07:45 +08:00
var bottomY = topY + LeaderboardScore.HEIGHT;
2017-03-16 12:15:06 +08:00
2017-03-19 20:49:29 +08:00
if (bottomY < fadeStart)
c.Colour = Color4.White;
else if (topY > fadeStart + LeaderboardScore.HEIGHT)
c.Colour = Color4.Transparent;
else
{
2017-07-23 13:30:50 +08:00
c.Colour = ColourInfo.GradientVertical(
2017-03-19 20:49:29 +08:00
Color4.White.Opacity(Math.Min(1 - (topY - fadeStart) / LeaderboardScore.HEIGHT, 1)),
Color4.White.Opacity(Math.Min(1 - (bottomY - fadeStart) / LeaderboardScore.HEIGHT, 1)));
}
2017-03-16 12:15:06 +08:00
}
}
private class MessagePlaceholder : FillFlowContainer
{
public MessagePlaceholder(string message)
{
Direction = FillDirection.Horizontal;
AutoSizeAxes = Axes.Both;
Children = new Drawable[]
{
new SpriteIcon
{
Icon = FontAwesome.fa_exclamation_circle,
Size = new Vector2(26),
Margin = new MarginPadding { Right = 10 },
},
new OsuSpriteText
{
Text = message,
TextSize = 22,
},
};
}
}
private class RetrievalFailurePlaceholder : FillFlowContainer
{
public Action OnRetry;
public RetrievalFailurePlaceholder()
{
Direction = FillDirection.Horizontal;
AutoSizeAxes = Axes.Both;
Children = new Drawable[]
{
new RetryButton
{
Action = () => OnRetry?.Invoke(),
Margin = new MarginPadding { Right = 10 },
},
new OsuSpriteText
{
Anchor = Anchor.TopLeft,
Text = @"Couldn't retrieve scores!",
TextSize = 22,
},
};
}
}
2017-12-20 22:41:48 +08:00
private class RetryButton : OsuHoverContainer
{
2017-11-26 17:33:49 +08:00
private readonly SpriteIcon icon;
public Action Action;
public RetryButton()
{
Height = 26;
Width = 26;
Child = new OsuClickableContainer
{
AutoSizeAxes = Axes.Both,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Action = () => Action?.Invoke(),
Child = icon = new SpriteIcon
{
Icon = FontAwesome.fa_refresh,
Size = new Vector2(26),
2017-11-28 17:08:35 +08:00
Shadow = true,
},
};
}
protected override bool OnMouseDown(InputState state, MouseDownEventArgs args)
{
2017-12-20 22:41:48 +08:00
icon.ScaleTo(0.8f, 400, Easing.OutQuint);
return base.OnMouseDown(state, args);
}
protected override bool OnMouseUp(InputState state, MouseUpEventArgs args)
{
2017-12-20 22:41:48 +08:00
icon.ScaleTo(1, 400, Easing.OutElastic);
return base.OnMouseUp(state, args);
}
}
2017-03-04 15:37:34 +08:00
}
public enum LeaderboardScope
{
Local,
Country,
Global,
2017-12-20 19:33:16 +08:00
Friend,
}
2017-03-04 15:37:34 +08:00
}