1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-16 06:52:55 +08:00
osu-lazer/osu.Game/Overlays/BeatmapSet/BeatmapPicker.cs

398 lines
15 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.
2018-04-13 17:19:50 +08:00
using System;
using System.Linq;
using osu.Framework;
using osu.Framework.Allocation;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
using osu.Framework.Extensions.LocalisationExtensions;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
2018-04-13 17:19:50 +08:00
using osu.Game.Beatmaps;
using osu.Game.Beatmaps.Drawables;
using osu.Game.Extensions;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
using osu.Game.Graphics.Sprites;
using osu.Game.Online.API.Requests.Responses;
2021-08-07 04:50:57 +08:00
using osu.Game.Resources.Localisation.Web;
using osu.Game.Rulesets;
2018-11-20 15:51:59 +08:00
using osuTK;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Overlays.BeatmapSet
{
2022-11-24 13:32:20 +08:00
public partial class BeatmapPicker : Container
2018-04-13 17:19:50 +08:00
{
private const float tile_icon_padding = 7;
private const float tile_spacing = 2;
2021-08-07 04:50:57 +08:00
private readonly OsuSpriteText version, starRating, starRatingText;
2023-04-03 19:09:49 +08:00
private readonly LinkFlowContainer guestMapperContainer;
2021-08-07 04:50:57 +08:00
private readonly FillFlowContainer starRatingContainer;
2018-04-13 17:19:50 +08:00
private readonly Statistic plays, favourites;
public readonly DifficultiesContainer Difficulties;
2023-01-16 06:15:36 +08:00
public readonly Bindable<APIBeatmap?> Beatmap = new Bindable<APIBeatmap?>();
private APIBeatmapSet? beatmapSet;
2019-02-28 12:31:40 +08:00
2023-01-16 06:15:36 +08:00
public APIBeatmapSet? BeatmapSet
2018-04-13 17:19:50 +08:00
{
get => beatmapSet;
2018-04-13 17:19:50 +08:00
set
{
if (value == beatmapSet) return;
2019-02-28 12:31:40 +08:00
2018-04-13 17:19:50 +08:00
beatmapSet = value;
updateDisplay();
}
}
2018-04-13 17:19:50 +08:00
public BeatmapPicker()
{
RelativeSizeAxes = Axes.X;
AutoSizeAxes = Axes.Y;
Children = new Drawable[]
{
new FillFlowContainer
{
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
Direction = FillDirection.Vertical,
Children = new Drawable[]
{
Difficulties = new DifficultiesContainer
2018-04-13 17:19:50 +08:00
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Margin = new MarginPadding { Left = -(tile_icon_padding + tile_spacing / 2), Bottom = 10 },
2018-04-13 17:19:50 +08:00
OnLostHover = () =>
{
showBeatmap(Beatmap.Value);
2021-08-07 04:50:57 +08:00
starRatingContainer.FadeOut(100);
2018-04-13 17:19:50 +08:00
},
},
new FillFlowContainer
{
AutoSizeAxes = Axes.Both,
Spacing = new Vector2(5f),
2021-08-07 04:50:57 +08:00
Children = new Drawable[]
2018-04-13 17:19:50 +08:00
{
version = new OsuSpriteText
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Font = OsuFont.GetFont(size: 17, weight: FontWeight.Bold)
2018-04-13 17:19:50 +08:00
},
2023-04-03 19:09:49 +08:00
guestMapperContainer = new LinkFlowContainer(s =>
s.Font = OsuFont.GetFont(weight: FontWeight.Bold, size: 11))
{
AutoSizeAxes = Axes.Both,
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Margin = new MarginPadding { Bottom = 1 },
},
2021-08-07 04:50:57 +08:00
starRatingContainer = new FillFlowContainer
2018-04-13 17:19:50 +08:00
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Alpha = 0,
2021-08-07 04:50:57 +08:00
Direction = FillDirection.Horizontal,
Spacing = new Vector2(2f, 0),
2018-04-13 17:19:50 +08:00
Margin = new MarginPadding { Bottom = 1 },
2021-08-07 04:50:57 +08:00
Children = new[]
{
starRatingText = new OsuSpriteText
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Font = OsuFont.GetFont(size: 11, weight: FontWeight.Bold),
Text = BeatmapsetsStrings.ShowStatsStars,
},
starRating = new OsuSpriteText
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Font = OsuFont.GetFont(size: 11, weight: FontWeight.Bold),
Text = string.Empty,
},
}
2018-04-13 17:19:50 +08:00
},
},
},
new FillFlowContainer
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Spacing = new Vector2(10f),
Margin = new MarginPadding { Top = 5 },
Children = new[]
{
2019-04-02 18:55:24 +08:00
plays = new Statistic(FontAwesome.Solid.PlayCircle),
favourites = new Statistic(FontAwesome.Solid.Heart),
2018-04-13 17:19:50 +08:00
},
},
},
},
};
Beatmap.ValueChanged += b =>
{
showBeatmap(b.NewValue);
2018-04-13 17:19:50 +08:00
updateDifficultyButtons();
};
}
[Resolved]
2023-01-16 06:15:36 +08:00
private IBindable<RulesetInfo> ruleset { get; set; } = null!;
2018-04-13 17:19:50 +08:00
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
starRating.Colour = colours.Yellow;
2021-08-07 04:50:57 +08:00
starRatingText.Colour = colours.Yellow;
updateDisplay();
2018-04-13 17:19:50 +08:00
}
protected override void LoadComplete()
{
base.LoadComplete();
2022-06-24 20:25:23 +08:00
ruleset.ValueChanged += _ => updateDisplay();
2018-04-13 17:19:50 +08:00
// done here so everything can bind in intialization and get the first trigger
Beatmap.TriggerChange();
}
2019-10-31 16:13:00 +08:00
private void updateDisplay()
{
Difficulties.Clear();
if (BeatmapSet != null)
{
2023-01-16 04:46:41 +08:00
Difficulties.ChildrenEnumerable = BeatmapSet.Beatmaps.Concat(BeatmapSet.Converts ?? Array.Empty<APIBeatmap>())
.Where(b => b.Ruleset.MatchesOnlineID(ruleset.Value))
.OrderBy(b => !b.Convert)
.ThenBy(b => b.StarRating)
.Select(b => new DifficultySelectorButton(b, b.Convert ? new RulesetInfo { OnlineID = 0 } : null)
{
State = DifficultySelectorState.NotSelected,
OnHovered = beatmap =>
{
showBeatmap(beatmap);
starRating.Text = beatmap.StarRating.ToLocalisableString(@"0.##");
starRatingContainer.FadeIn(100);
},
OnClicked = beatmap => { Beatmap.Value = beatmap; },
});
2019-10-31 16:13:00 +08:00
}
2021-08-07 04:50:57 +08:00
starRatingContainer.FadeOut(100);
// If a selection is already made, try and maintain it.
if (Beatmap.Value != null)
Beatmap.Value = Difficulties.FirstOrDefault(b => b.Beatmap.OnlineID == Beatmap.Value.OnlineID)?.Beatmap;
// Else just choose the first available difficulty for now.
Beatmap.Value ??= Difficulties.FirstOrDefault()?.Beatmap;
plays.Value = BeatmapSet?.PlayCount ?? 0;
favourites.Value = BeatmapSet?.FavouriteCount ?? 0;
2019-10-31 16:13:00 +08:00
updateDifficultyButtons();
}
private void showBeatmap(APIBeatmap? beatmapInfo)
{
guestMapperContainer.Clear();
if (beatmapInfo?.AuthorID != BeatmapSet?.AuthorID)
{
2023-04-03 19:24:38 +08:00
APIUser? user = BeatmapSet?.RelatedUsers?.SingleOrDefault(u => u.OnlineID == beatmapInfo?.AuthorID);
2023-04-03 19:07:21 +08:00
if (user != null)
{
guestMapperContainer.AddText("mapped by ");
guestMapperContainer.AddUserLink(user);
}
}
2023-01-16 06:15:36 +08:00
version.Text = beatmapInfo?.DifficultyName ?? string.Empty;
}
2018-04-13 17:19:50 +08:00
private void updateDifficultyButtons()
{
Difficulties.Children.ToList().ForEach(diff => diff.State = diff.Beatmap == Beatmap.Value ? DifficultySelectorState.Selected : DifficultySelectorState.NotSelected);
2018-04-13 17:19:50 +08:00
}
2022-11-24 13:32:20 +08:00
public partial class DifficultiesContainer : FillFlowContainer<DifficultySelectorButton>
2018-04-13 17:19:50 +08:00
{
2023-01-16 06:15:36 +08:00
public Action? OnLostHover;
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override void OnHoverLost(HoverLostEvent e)
2018-04-13 17:19:50 +08:00
{
2018-10-02 11:02:47 +08:00
base.OnHoverLost(e);
2018-04-13 17:19:50 +08:00
OnLostHover?.Invoke();
}
}
2022-11-24 13:32:20 +08:00
public partial class DifficultySelectorButton : OsuClickableContainer, IStateful<DifficultySelectorState>
2018-04-13 17:19:50 +08:00
{
private const float transition_duration = 100;
private const float size = 54;
private const float background_size = size - 2;
2018-04-13 17:19:50 +08:00
private readonly Container background;
private readonly Box backgroundBox;
2018-04-13 17:19:50 +08:00
private readonly DifficultyIcon icon;
public readonly APIBeatmap Beatmap;
2018-04-13 17:19:50 +08:00
2023-01-16 06:15:36 +08:00
public Action<APIBeatmap>? OnHovered;
public Action<APIBeatmap>? OnClicked;
public event Action<DifficultySelectorState>? StateChanged;
2018-04-13 17:19:50 +08:00
private DifficultySelectorState state;
2019-02-28 12:31:40 +08:00
2018-04-13 17:19:50 +08:00
public DifficultySelectorState State
{
get => state;
2018-04-13 17:19:50 +08:00
set
{
if (value == state) return;
2019-02-28 12:31:40 +08:00
2018-04-13 17:19:50 +08:00
state = value;
StateChanged?.Invoke(State);
if (value == DifficultySelectorState.Selected)
fadeIn();
else
fadeOut();
}
}
public DifficultySelectorButton(APIBeatmap beatmapInfo, IRulesetInfo? ruleset)
2018-04-13 17:19:50 +08:00
{
Beatmap = beatmapInfo;
2018-04-13 17:19:50 +08:00
Size = new Vector2(size);
Margin = new MarginPadding { Horizontal = tile_spacing / 2 };
Children = new Drawable[]
{
background = new Container
2018-04-13 17:19:50 +08:00
{
Size = new Vector2(background_size),
2018-04-13 17:19:50 +08:00
Masking = true,
CornerRadius = 4,
Child = backgroundBox = new Box
2018-04-13 17:19:50 +08:00
{
RelativeSizeAxes = Axes.Both,
Alpha = 0.5f
}
2018-04-13 17:19:50 +08:00
},
icon = new DifficultyIcon(beatmapInfo, ruleset)
2018-04-13 17:19:50 +08:00
{
ShowTooltip = false,
Current = { Value = new StarDifficulty(beatmapInfo.StarRating, 0) },
2018-04-13 17:19:50 +08:00
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Size = new Vector2(size - tile_icon_padding * 2),
Margin = new MarginPadding { Bottom = 1 },
},
};
}
2018-10-02 11:02:47 +08:00
protected override bool OnHover(HoverEvent e)
2018-04-13 17:19:50 +08:00
{
fadeIn();
OnHovered?.Invoke(Beatmap);
2018-10-02 11:02:47 +08:00
return base.OnHover(e);
2018-04-13 17:19:50 +08:00
}
2018-10-02 11:02:47 +08:00
protected override void OnHoverLost(HoverLostEvent e)
2018-04-13 17:19:50 +08:00
{
if (State == DifficultySelectorState.NotSelected)
fadeOut();
2018-10-02 11:02:47 +08:00
base.OnHoverLost(e);
2018-04-13 17:19:50 +08:00
}
2018-10-02 11:02:47 +08:00
protected override bool OnClick(ClickEvent e)
2018-04-13 17:19:50 +08:00
{
OnClicked?.Invoke(Beatmap);
2018-10-02 11:02:47 +08:00
return base.OnClick(e);
2018-04-13 17:19:50 +08:00
}
private void fadeIn()
{
background.FadeIn(transition_duration);
2018-04-13 17:19:50 +08:00
icon.FadeIn(transition_duration);
}
private void fadeOut()
{
background.FadeOut();
2018-04-13 17:19:50 +08:00
icon.FadeTo(0.7f, transition_duration);
}
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colourProvider)
{
backgroundBox.Colour = colourProvider.Background6;
}
2018-04-13 17:19:50 +08:00
}
2022-11-24 13:32:20 +08:00
private partial class Statistic : FillFlowContainer
2018-04-13 17:19:50 +08:00
{
private readonly OsuSpriteText text;
private int value;
2019-02-28 12:31:40 +08:00
2018-04-13 17:19:50 +08:00
public int Value
{
get => value;
2018-04-13 17:19:50 +08:00
set
{
this.value = value;
2021-08-07 04:50:57 +08:00
text.Text = Value.ToLocalisableString(@"N0");
2018-04-13 17:19:50 +08:00
}
}
public Statistic(IconUsage icon)
2018-04-13 17:19:50 +08:00
{
AutoSizeAxes = Axes.Both;
Direction = FillDirection.Horizontal;
Spacing = new Vector2(2f);
Children = new Drawable[]
{
new SpriteIcon
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
Icon = icon,
Shadow = true,
Size = new Vector2(12),
2018-04-13 17:19:50 +08:00
},
text = new OsuSpriteText
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
Font = OsuFont.GetFont(size: 12, weight: FontWeight.SemiBold, italics: true),
2018-04-13 17:19:50 +08:00
},
};
}
}
public enum DifficultySelectorState
2018-04-13 17:19:50 +08:00
{
Selected,
NotSelected,
}
}
}