// Copyright (c) ppy Pty Ltd <[email protected]>. Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Collections.Generic; using System.Threading; using osu.Framework.Allocation; using osu.Framework.Bindables; using osu.Framework.Extensions.Color4Extensions; using osu.Framework.Graphics; using osu.Framework.Graphics.Colour; using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Shapes; using osu.Framework.Graphics.Sprites; using osu.Framework.Graphics.UserInterface; using osu.Game.Beatmaps; using osu.Game.Beatmaps.Drawables; using osu.Game.Graphics; using osu.Game.Graphics.Backgrounds; using osu.Game.Graphics.Carousel; using osu.Game.Graphics.Containers; using osu.Game.Graphics.Sprites; using osu.Game.Graphics.UserInterface; using osu.Game.Overlays; using osu.Game.Resources.Localisation.Web; using osu.Game.Rulesets; using osu.Game.Rulesets.Mods; using osuTK; namespace osu.Game.Screens.SelectV2 { public partial class PanelBeatmap : Panel { public const float HEIGHT = CarouselItem.DEFAULT_HEIGHT; private StarCounter starCounter = null!; private ConstrainedIconContainer difficultyIcon = null!; private OsuSpriteText keyCountText = null!; private StarRatingDisplay starRatingDisplay = null!; private PanelLocalRankDisplay localRank = null!; private OsuSpriteText difficultyText = null!; private OsuSpriteText authorText = null!; private FillFlowContainer mainFill = null!; private IBindable<StarDifficulty>? starDifficultyBindable; private CancellationTokenSource? starDifficultyCancellationSource; private Box backgroundBorder = null!; private Box backgroundDifficultyTint = null!; private TrianglesV2 triangles = null!; [Resolved] private IRulesetStore rulesets { get; set; } = null!; [Resolved] private OverlayColourProvider colourProvider { get; set; } = null!; [Resolved] private OsuColour colours { get; set; } = null!; [Resolved] private BeatmapDifficultyCache difficultyCache { get; set; } = null!; [Resolved] private IBindable<RulesetInfo> ruleset { get; set; } = null!; [Resolved] private IBindable<IReadOnlyList<Mod>> mods { get; set; } = null!; [Resolved] private ISongSelect? songSelect { get; set; } private BeatmapInfo beatmap => ((GroupedBeatmap)Item!.Model).Beatmap; public PanelBeatmap() { PanelXOffset = 60; } [BackgroundDependencyLoader] private void load(OverlayColourProvider colourProvider) { Height = HEIGHT; Icon = difficultyIcon = new ConstrainedIconContainer { Size = new Vector2(9f), Margin = new MarginPadding { Left = 2.5f, Right = 1.5f }, Colour = colourProvider.Background5, }; Background = backgroundBorder = new Box { RelativeSizeAxes = Axes.Both, }; Content.Children = new Drawable[] { new Box { RelativeSizeAxes = Axes.Both, Colour = ColourInfo.GradientHorizontal(colourProvider.Background3, colourProvider.Background4), }, backgroundDifficultyTint = new Box { RelativeSizeAxes = Axes.Both, }, triangles = new TrianglesV2 { ScaleAdjust = 1.2f, Thickness = 0.01f, Velocity = 0.3f, RelativeSizeAxes = Axes.Both, }, new FillFlowContainer { AutoSizeAxes = Axes.Both, Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, Spacing = new Vector2(5), Margin = new MarginPadding { Left = 6.5f }, Direction = FillDirection.Horizontal, Children = new Drawable[] { localRank = new PanelLocalRankDisplay { Scale = new Vector2(0.8f), Origin = Anchor.CentreLeft, Anchor = Anchor.CentreLeft, }, mainFill = new FillFlowContainer { Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, Direction = FillDirection.Vertical, AutoSizeAxes = Axes.Both, Padding = new MarginPadding { Bottom = 3.5f }, Children = new Drawable[] { new FillFlowContainer { Direction = FillDirection.Horizontal, AutoSizeAxes = Axes.Both, Padding = new MarginPadding { Bottom = 4 }, Children = new Drawable[] { keyCountText = new OsuSpriteText { Font = OsuFont.Style.Body.With(weight: FontWeight.SemiBold), Anchor = Anchor.BottomLeft, Origin = Anchor.BottomLeft, Alpha = 0, }, difficultyText = new OsuSpriteText { Font = OsuFont.Style.Body.With(weight: FontWeight.SemiBold), Anchor = Anchor.BottomLeft, Origin = Anchor.BottomLeft, Margin = new MarginPadding { Right = 3f }, }, authorText = new OsuSpriteText { Colour = colourProvider.Content2, Font = OsuFont.Style.Caption1.With(weight: FontWeight.SemiBold), Anchor = Anchor.BottomLeft, Origin = Anchor.BottomLeft } } }, new FillFlowContainer { Direction = FillDirection.Horizontal, Spacing = new Vector2(3), AutoSizeAxes = Axes.Both, Children = new Drawable[] { starRatingDisplay = new StarRatingDisplay(default, StarRatingDisplaySize.Small, animated: true) { Origin = Anchor.CentreLeft, Anchor = Anchor.CentreLeft, Scale = new Vector2(0.875f), }, starCounter = new StarCounter { Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, Scale = new Vector2(0.4f) } }, } } } } } }; } protected override void LoadComplete() { base.LoadComplete(); ruleset.BindValueChanged(_ => updateKeyCount()); mods.BindValueChanged(_ => updateKeyCount(), true); } protected override void PrepareForUse() { base.PrepareForUse(); difficultyIcon.Icon = getRulesetIcon(beatmap.Ruleset); localRank.Beatmap = beatmap; difficultyText.Text = beatmap.DifficultyName; authorText.Text = BeatmapsetsStrings.ShowDetailsMappedBy(beatmap.Metadata.Author.Username); computeStarRating(); updateKeyCount(); } private Drawable getRulesetIcon(RulesetInfo rulesetInfo) { var rulesetInstance = rulesets.GetRuleset(rulesetInfo.ShortName)?.CreateInstance(); if (rulesetInstance is null) return new SpriteIcon { Icon = FontAwesome.Regular.QuestionCircle }; return rulesetInstance.CreateIcon(); } protected override void FreeAfterUse() { base.FreeAfterUse(); localRank.Beatmap = null; starDifficultyBindable = null; starDifficultyCancellationSource?.Cancel(); } private void computeStarRating() { starDifficultyCancellationSource?.Cancel(); starDifficultyCancellationSource = new CancellationTokenSource(); if (Item == null) return; starDifficultyBindable = difficultyCache.GetBindableDifficulty(beatmap, starDifficultyCancellationSource.Token, SongSelect.DIFFICULTY_CALCULATION_DEBOUNCE); starDifficultyBindable.BindValueChanged(starDifficulty => { starRatingDisplay.Current.Value = starDifficulty.NewValue; starCounter.Current = (float)starDifficulty.NewValue.Stars; }, true); } protected override void Update() { base.Update(); if (Item?.IsVisible != true) { starDifficultyCancellationSource?.Cancel(); starDifficultyCancellationSource = null; } // Dirty hack to make sure we don't take up spacing in parent fill flow when not displaying a rank. // I can't find a better way to do this. mainFill.Margin = new MarginPadding { Left = 1 / starRatingDisplay.Scale.X * (localRank.HasRank ? 0 : -3) }; var diffColour = starRatingDisplay.DisplayedDifficultyColour; if (AccentColour != diffColour) { AccentColour = diffColour; starCounter.Colour = diffColour; backgroundBorder.Colour = diffColour; backgroundDifficultyTint.Colour = ColourInfo.GradientHorizontal(diffColour.Opacity(0.25f), diffColour.Opacity(0f)); difficultyIcon.Colour = starRatingDisplay.DisplayedStars.Value > OsuColour.STAR_DIFFICULTY_DEFINED_COLOUR_CUTOFF ? colours.Orange1 : colourProvider.Background5; triangles.Colour = ColourInfo.GradientVertical(diffColour.Opacity(0.25f), diffColour.Opacity(0f)); } } private void updateKeyCount() { if (Item == null) return; if (ruleset.Value.OnlineID == 3) { // Account for mania differences locally for now. // Eventually this should be handled in a more modular way, allowing rulesets to add more information to the panel. ILegacyRuleset legacyRuleset = (ILegacyRuleset)ruleset.Value.CreateInstance(); int keyCount = legacyRuleset.GetKeyCount(beatmap, mods.Value); keyCountText.Alpha = 1; keyCountText.Text = $"[{keyCount}K] "; } else keyCountText.Alpha = 0; } public override MenuItem[] ContextMenuItems { get { if (Item == null) return Array.Empty<MenuItem>(); List<MenuItem> items = new List<MenuItem>(); if (songSelect != null) items.AddRange(songSelect.GetForwardActions(beatmap)); return items.ToArray(); } } } }