mirror of
https://github.com/ppy/osu.git
synced 2025-03-28 09:37:23 +08:00
Rank graph improvements
This commit is contained in:
parent
be526d68e8
commit
f3c93c894b
@ -27,6 +27,7 @@ namespace osu.Game.Overlays.Profile
|
|||||||
private readonly OsuTextFlowContainer infoTextLeft;
|
private readonly OsuTextFlowContainer infoTextLeft;
|
||||||
private readonly LinkFlowContainer infoTextRight;
|
private readonly LinkFlowContainer infoTextRight;
|
||||||
private readonly FillFlowContainer<SpriteText> scoreText, scoreNumberText;
|
private readonly FillFlowContainer<SpriteText> scoreText, scoreNumberText;
|
||||||
|
protected readonly RankChart rankGraph;
|
||||||
|
|
||||||
private readonly Container coverContainer, chartContainer, supporterTag;
|
private readonly Container coverContainer, chartContainer, supporterTag;
|
||||||
private readonly Sprite levelBadge;
|
private readonly Sprite levelBadge;
|
||||||
@ -285,6 +286,10 @@ namespace osu.Game.Overlays.Profile
|
|||||||
{
|
{
|
||||||
Colour = Color4.Black.Opacity(0.25f),
|
Colour = Color4.Black.Opacity(0.25f),
|
||||||
RelativeSizeAxes = Axes.Both
|
RelativeSizeAxes = Axes.Both
|
||||||
|
},
|
||||||
|
rankGraph = new RankChart
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -303,11 +308,7 @@ namespace osu.Game.Overlays.Profile
|
|||||||
|
|
||||||
public User User
|
public User User
|
||||||
{
|
{
|
||||||
get
|
get { return user; }
|
||||||
{
|
|
||||||
return user;
|
|
||||||
}
|
|
||||||
|
|
||||||
set
|
set
|
||||||
{
|
{
|
||||||
user = value;
|
user = value;
|
||||||
@ -420,7 +421,7 @@ namespace osu.Game.Overlays.Profile
|
|||||||
gradeSPlus.DisplayCount = 0;
|
gradeSPlus.DisplayCount = 0;
|
||||||
gradeSSPlus.DisplayCount = 0;
|
gradeSSPlus.DisplayCount = 0;
|
||||||
|
|
||||||
chartContainer.Add(new RankChart(user) { RelativeSizeAxes = Axes.Both });
|
rankGraph.Redraw(user);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,30 +14,40 @@ using osu.Game.Graphics;
|
|||||||
using osu.Game.Graphics.Sprites;
|
using osu.Game.Graphics.Sprites;
|
||||||
using osu.Game.Graphics.UserInterface;
|
using osu.Game.Graphics.UserInterface;
|
||||||
using osu.Game.Users;
|
using osu.Game.Users;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace osu.Game.Overlays.Profile
|
namespace osu.Game.Overlays.Profile
|
||||||
{
|
{
|
||||||
public class RankChart : Container
|
public class RankChart : Container
|
||||||
{
|
{
|
||||||
|
private const float primary_textsize = 25;
|
||||||
|
private const float secondary_textsize = 13;
|
||||||
|
private const float padding = 10;
|
||||||
|
private const float fade_duration = 150;
|
||||||
|
private const int rankedDays = 88;
|
||||||
|
|
||||||
private readonly SpriteText rankText, performanceText, relativeText;
|
private readonly SpriteText rankText, performanceText, relativeText;
|
||||||
private readonly RankChartLineGraph graph;
|
private readonly RankChartLineGraph graph;
|
||||||
|
private readonly OsuSpriteText placeholder;
|
||||||
|
|
||||||
private readonly int[] ranks;
|
private KeyValuePair<int, int>[] ranks;
|
||||||
|
private User user;
|
||||||
|
|
||||||
private const float primary_textsize = 25, secondary_textsize = 13, padding = 10;
|
public RankChart()
|
||||||
|
|
||||||
private readonly User user;
|
|
||||||
|
|
||||||
public RankChart(User user)
|
|
||||||
{
|
{
|
||||||
this.user = user;
|
|
||||||
|
|
||||||
int[] userRanks = user.RankHistory?.Data ?? new[] { user.Statistics.Rank };
|
|
||||||
ranks = userRanks.SkipWhile(x => x == 0).ToArray();
|
|
||||||
|
|
||||||
Padding = new MarginPadding { Vertical = padding };
|
Padding = new MarginPadding { Vertical = padding };
|
||||||
Children = new Drawable[]
|
Children = new Drawable[]
|
||||||
{
|
{
|
||||||
|
placeholder = new OsuSpriteText
|
||||||
|
{
|
||||||
|
Anchor = Anchor.BottomCentre,
|
||||||
|
Origin = Anchor.BottomCentre,
|
||||||
|
Text = "No recent plays",
|
||||||
|
TextSize = 14,
|
||||||
|
Font = @"Exo2.0-RegularItalic",
|
||||||
|
Alpha = 0,
|
||||||
|
Padding = new MarginPadding { Bottom = padding }
|
||||||
|
},
|
||||||
rankText = new OsuSpriteText
|
rankText = new OsuSpriteText
|
||||||
{
|
{
|
||||||
Anchor = Anchor.TopCentre,
|
Anchor = Anchor.TopCentre,
|
||||||
@ -60,50 +70,73 @@ namespace osu.Game.Overlays.Profile
|
|||||||
Font = @"Exo2.0-RegularItalic",
|
Font = @"Exo2.0-RegularItalic",
|
||||||
TextSize = secondary_textsize
|
TextSize = secondary_textsize
|
||||||
},
|
},
|
||||||
};
|
graph = new RankChartLineGraph
|
||||||
|
|
||||||
if (ranks.Length > 0)
|
|
||||||
{
|
|
||||||
Add(graph = new RankChartLineGraph
|
|
||||||
{
|
{
|
||||||
Anchor = Anchor.BottomCentre,
|
Anchor = Anchor.BottomCentre,
|
||||||
Origin = Anchor.BottomCentre,
|
Origin = Anchor.BottomCentre,
|
||||||
RelativeSizeAxes = Axes.X,
|
RelativeSizeAxes = Axes.X,
|
||||||
Y = -secondary_textsize,
|
Y = -secondary_textsize,
|
||||||
DefaultValueCount = ranks.Length,
|
Alpha = 0,
|
||||||
});
|
}
|
||||||
|
};
|
||||||
|
|
||||||
graph.OnBallMove += showHistoryRankTexts;
|
graph.OnBallMove += showHistoryRankTexts;
|
||||||
|
}
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load(OsuColour colours)
|
||||||
|
{
|
||||||
|
graph.Colour = colours.Yellow;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Redraw(User user)
|
||||||
|
{
|
||||||
|
if (this.user == null && user != null)
|
||||||
|
placeholder.FadeOut(fade_duration, Easing.Out);
|
||||||
|
|
||||||
|
this.user = user;
|
||||||
|
|
||||||
|
if (user == null)
|
||||||
|
{
|
||||||
|
rankText.Text = string.Empty;
|
||||||
|
performanceText.Text = string.Empty;
|
||||||
|
relativeText.Text = string.Empty;
|
||||||
|
graph.FadeOut(fade_duration, Easing.Out);
|
||||||
|
placeholder.FadeIn(fade_duration, Easing.Out);
|
||||||
|
ranks = null;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int[] userRanks = user.RankHistory?.Data ?? new[] { user.Statistics.Rank };
|
||||||
|
var tempRanks = userRanks.Select((x, index) => new KeyValuePair<int, int>(index, x)).SkipWhile(x => x.Value == 0).ToArray();
|
||||||
|
ranks = tempRanks.Where(x => x.Value != 0).ToArray();
|
||||||
|
|
||||||
|
if (ranks.Length > 1)
|
||||||
|
{
|
||||||
|
graph.DefaultValueCount = ranks.Length;
|
||||||
|
graph.Values = ranks.Select(x => -(float)Math.Log(x.Value));
|
||||||
|
graph.SetStaticBallPosition();
|
||||||
|
graph.FadeIn(fade_duration, Easing.Out);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
graph.FadeOut(fade_duration, Easing.Out);
|
||||||
|
}
|
||||||
|
|
||||||
|
updateRankTexts();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void updateRankTexts()
|
private void updateRankTexts()
|
||||||
{
|
{
|
||||||
rankText.Text = user.Statistics.Rank > 0 ? $"#{user.Statistics.Rank:#,0}" : "no rank";
|
rankText.Text = user.Statistics.Rank > 0 ? $"#{user.Statistics.Rank:#,0}" : "no rank";
|
||||||
performanceText.Text = user.Statistics.PP != null ? $"{user.Statistics.PP:#,0}pp" : string.Empty;
|
performanceText.Text = user.Statistics.PP != null ? $"{user.Statistics.PP:#,0}pp" : string.Empty;
|
||||||
relativeText.Text = user.CountryRank > 0 ? $"{user.Country?.FullName} #{user.CountryRank:#,0}" : $"{user.Country?.FullName}";
|
relativeText.Text = $"{user.Country?.FullName} #{user.CountryRank:#,0}";
|
||||||
}
|
}
|
||||||
|
|
||||||
private void showHistoryRankTexts(int dayIndex)
|
private void showHistoryRankTexts(int dayIndex)
|
||||||
{
|
{
|
||||||
rankText.Text = ranks[dayIndex] > 0 ? $"#{ranks[dayIndex]:#,0}" : "no rank";
|
rankText.Text = $"#{ranks[dayIndex].Value:#,0}";
|
||||||
dayIndex++;
|
relativeText.Text = dayIndex + 1 == ranks.Length ? "Now" : $"{rankedDays - ranks[dayIndex].Key} days ago";
|
||||||
relativeText.Text = dayIndex == ranks.Length ? "Now" : $"{ranks.Length - dayIndex} days ago";
|
|
||||||
//plural should be handled in a general way
|
|
||||||
}
|
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
|
||||||
private void load(OsuColour colours)
|
|
||||||
{
|
|
||||||
if (graph != null)
|
|
||||||
{
|
|
||||||
graph.Colour = colours.Yellow;
|
|
||||||
// use logarithmic coordinates
|
|
||||||
graph.Values = ranks.Select(x => x == 0 ? float.MinValue : -(float)Math.Log(x));
|
|
||||||
graph.SetStaticBallPosition();
|
|
||||||
}
|
|
||||||
|
|
||||||
updateRankTexts();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public override bool Invalidate(Invalidation invalidation = Invalidation.All, Drawable source = null, bool shallPropagate = true)
|
public override bool Invalidate(Invalidation invalidation = Invalidation.All, Drawable source = null, bool shallPropagate = true)
|
||||||
@ -118,31 +151,35 @@ namespace osu.Game.Overlays.Profile
|
|||||||
|
|
||||||
protected override bool OnHover(InputState state)
|
protected override bool OnHover(InputState state)
|
||||||
{
|
{
|
||||||
graph?.UpdateBallPosition(state.Mouse.Position.X);
|
if (ranks != null && ranks.Length > 1)
|
||||||
graph?.ShowBall();
|
{
|
||||||
|
graph.UpdateBallPosition(state.Mouse.Position.X);
|
||||||
|
graph.ShowBall();
|
||||||
|
}
|
||||||
return base.OnHover(state);
|
return base.OnHover(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool OnMouseMove(InputState state)
|
protected override bool OnMouseMove(InputState state)
|
||||||
{
|
{
|
||||||
graph?.UpdateBallPosition(state.Mouse.Position.X);
|
if (ranks != null && ranks.Length > 1)
|
||||||
|
graph.UpdateBallPosition(state.Mouse.Position.X);
|
||||||
|
|
||||||
return base.OnMouseMove(state);
|
return base.OnMouseMove(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnHoverLost(InputState state)
|
protected override void OnHoverLost(InputState state)
|
||||||
{
|
{
|
||||||
if (graph != null)
|
if (ranks != null && ranks.Length > 1)
|
||||||
{
|
{
|
||||||
graph.HideBall();
|
graph.HideBall();
|
||||||
updateRankTexts();
|
updateRankTexts();
|
||||||
}
|
}
|
||||||
|
|
||||||
base.OnHoverLost(state);
|
base.OnHoverLost(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
private class RankChartLineGraph : LineGraph
|
private class RankChartLineGraph : LineGraph
|
||||||
{
|
{
|
||||||
private const double fade_duration = 200;
|
|
||||||
|
|
||||||
private readonly CircularContainer staticBall;
|
private readonly CircularContainer staticBall;
|
||||||
private readonly CircularContainer movingBall;
|
private readonly CircularContainer movingBall;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user