1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-24 18:47:32 +08:00
osu-lazer/osu.Game/Screens/Select/Leaderboards/Leaderboard.cs

335 lines
11 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;
using System.Net;
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;
private FillFlowContainer<LeaderboardScore> scrollFlow;
private Container noResultsPlaceholder;
private Container retryPlaceholder;
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;
getScoresRequest?.Cancel();
getScoresRequest = null;
2017-03-04 16:05:31 +08:00
noResultsPlaceholder.FadeOut(fade_duration);
scrollFlow?.FadeOut(fade_duration).Expire();
scrollContainer.Clear(true); // scores stick around in scrollFlow in VisualTests without this for some reason
scrollFlow = null;
loading.Hide();
if (scores == null)
2017-03-15 13:38:38 +08:00
return;
if (scores.Count() == 0)
{
noResultsPlaceholder.FadeIn(fade_duration);
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(),
noResultsPlaceholder = new Container
{
Alpha = 0,
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
new FillFlowContainer
{
Direction = FillDirection.Horizontal,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
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 = @"No records yet!",
TextSize = 22,
},
}
},
},
},
retryPlaceholder = new Container
{
Alpha = 0,
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
new FillFlowContainer
{
Direction = FillDirection.Horizontal,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
AutoSizeAxes = Axes.Both,
Children = new Drawable[]
{
new RetryButton
{
Action = updateScores,
Margin = new MarginPadding { Right = 10 },
},
new OsuSpriteText
{
Anchor = Anchor.TopLeft,
Text = @"An error occurred!",
TextSize = 22,
},
}
},
},
},
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)
osuGame.Ruleset.ValueChanged += r => updateScores();
}
private GetScoresRequest getScoresRequest;
private void updateScores()
{
if (!IsLoaded) return;
retryPlaceholder.FadeOut(fade_duration);
Scores = null;
if (api == null || Beatmap?.OnlineBeatmapID == null) return;
if (Scope == LeaderboardScope.Local)
{
// TODO: get local scores from wherever here.
Scores = Enumerable.Empty<Score>();
return;
}
loading.Show();
getScoresRequest = new GetScoresRequest(Beatmap, osuGame?.Ruleset.Value, Scope);
getScoresRequest.Success += r =>
{
Scores = r.Scores;
};
getScoresRequest.Failure += e =>
{
// TODO: check why failure is repeatedly invoked even on successful requests
if (e is WebException)
{
Scores = null;
retryPlaceholder.FadeIn(fade_duration);
Logger.Error(e, @"Couldn't fetch beatmap scores!");
}
};
api.Queue(getScoresRequest);
}
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 RetryButton : ClickableContainer
{
private SpriteIcon icon;
public RetryButton()
{
Height = 26;
Width = 26;
Children = new Drawable[]
{
icon = new SpriteIcon
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Icon = FontAwesome.fa_refresh,
Size = new Vector2(26),
}
};
}
protected override bool OnHover(Framework.Input.InputState state)
{
icon.ScaleTo(1.4f, 400, Easing.OutQuint);
return base.OnHover(state);
}
protected override void OnHoverLost(Framework.Input.InputState state)
{
icon.ScaleTo(1f, 400, Easing.OutQuint);
base.OnHoverLost(state);
}
protected override bool OnMouseDown(Framework.Input.InputState state, Framework.Input.MouseDownEventArgs args)
{
icon.ScaleTo(0.8f, 200, Easing.InElastic);
return base.OnMouseDown(state, args);
}
protected override bool OnMouseUp(Framework.Input.InputState state, Framework.Input.MouseUpEventArgs args)
{
icon.ScaleTo(1.2f, 400, Easing.OutElastic).Then().ScaleTo(1f, 400, Easing.OutElastic);
return base.OnMouseUp(state, args);
}
}
2017-03-04 15:37:34 +08:00
}
public enum LeaderboardScope
{
Local,
Country,
Global,
Friends,
}
2017-03-04 15:37:34 +08:00
}