1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-11 07:13:00 +08:00
osu-lazer/osu.Game/Screens/Select/Carousel/DrawableCarouselBeatmap.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

317 lines
12 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.Collections.Generic;
2020-09-02 20:08:31 +08:00
using System.Linq;
using System.Threading;
2017-12-12 16:48:38 +08:00
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Extensions.LocalisationExtensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Colour;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Cursor;
2017-12-12 16:48:38 +08:00
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
2017-12-12 16:48:38 +08:00
using osu.Framework.Graphics.UserInterface;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
2017-12-12 16:48:38 +08:00
using osu.Game.Beatmaps;
using osu.Game.Beatmaps.Drawables;
2020-09-02 20:08:31 +08:00
using osu.Game.Collections;
2022-07-27 14:59:36 +08:00
using osu.Game.Database;
using osu.Game.Graphics;
using osu.Game.Graphics.Backgrounds;
2017-12-12 16:48:38 +08:00
using osu.Game.Graphics.Sprites;
using osu.Game.Graphics.UserInterface;
using osu.Game.Online.API;
using osu.Game.Overlays;
using osu.Game.Resources.Localisation.Web;
using osu.Game.Rulesets;
using osu.Game.Rulesets.Mods;
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
using CommonStrings = osu.Game.Localisation.CommonStrings;
using WebCommonStrings = osu.Game.Resources.Localisation.Web.CommonStrings;
2018-04-13 17:19:50 +08:00
2017-12-12 16:48:38 +08:00
namespace osu.Game.Screens.Select.Carousel
{
2017-12-12 16:48:38 +08:00
public partial class DrawableCarouselBeatmap : DrawableCarouselItem, IHasContextMenu
{
2020-10-12 19:26:20 +08:00
public const float CAROUSEL_BEATMAP_SPACING = 5;
/// <summary>
/// The height of a carousel beatmap, including vertical spacing.
/// </summary>
public const float HEIGHT = height + CAROUSEL_BEATMAP_SPACING;
private const float height = MAX_HEIGHT * 0.6f;
2021-10-02 23:55:29 +08:00
private readonly BeatmapInfo beatmapInfo;
2018-04-13 17:19:50 +08:00
2023-01-09 02:02:48 +08:00
private Sprite background = null!;
2018-04-13 17:19:50 +08:00
private MenuItem[]? mainMenuItems;
2023-01-16 08:09:04 +08:00
private Action<BeatmapInfo>? selectRequested;
private Action<BeatmapInfo>? hideRequested;
2018-04-13 17:19:50 +08:00
2023-01-09 02:02:48 +08:00
private Triangles triangles = null!;
2023-01-09 02:02:48 +08:00
private StarCounter starCounter = null!;
private DifficultyIcon difficultyIcon = null!;
2018-04-13 17:19:50 +08:00
private OsuSpriteText keyCountText = null!;
2023-01-09 02:02:48 +08:00
[Resolved]
private BeatmapSetOverlay? beatmapOverlay { get; set; }
[Resolved]
2023-01-09 02:02:48 +08:00
private BeatmapDifficultyCache difficultyCache { get; set; } = null!;
2023-01-09 02:02:48 +08:00
[Resolved]
private ManageCollectionsDialog? manageCollectionsDialog { get; set; }
2020-09-05 02:52:07 +08:00
2022-07-27 14:59:36 +08:00
[Resolved]
2023-01-09 02:02:48 +08:00
private RealmAccess realm { get; set; } = null!;
2022-07-27 14:59:36 +08:00
[Resolved]
private IBindable<RulesetInfo> ruleset { get; set; } = null!;
[Resolved]
private IBindable<IReadOnlyList<Mod>> mods { get; set; } = null!;
[Resolved]
private IAPIProvider api { get; set; } = null!;
[Resolved]
private OsuGame? game { get; set; }
2023-01-09 02:02:48 +08:00
private IBindable<StarDifficulty?> starDifficultyBindable = null!;
private CancellationTokenSource? starDifficultyCancellationSource;
2020-07-21 22:13:04 +08:00
2019-02-28 12:31:40 +08:00
public DrawableCarouselBeatmap(CarouselBeatmap panel)
{
2021-10-02 23:55:29 +08:00
beatmapInfo = panel.BeatmapInfo;
2020-10-12 14:36:03 +08:00
Item = panel;
}
2018-04-13 17:19:50 +08:00
2023-01-09 02:02:48 +08:00
[BackgroundDependencyLoader]
2024-08-22 01:17:11 +08:00
private void load(BeatmapManager? manager, SongSelect? songSelect)
{
Header.Height = height;
2017-12-13 18:56:16 +08:00
if (songSelect != null)
{
mainMenuItems = songSelect.CreateForwardNavigationMenuItemsForBeatmap(() => beatmapInfo);
selectRequested = b => songSelect.FinaliseSelection(b);
2017-12-13 18:56:16 +08:00
}
2018-04-13 17:19:50 +08:00
2017-12-13 18:56:16 +08:00
if (manager != null)
2017-12-18 10:13:16 +08:00
hideRequested = manager.Hide;
2018-04-13 17:19:50 +08:00
Header.Children = new Drawable[]
{
background = new Box
{
RelativeSizeAxes = Axes.Both,
},
2017-01-30 15:53:12 +08:00
triangles = new Triangles
2016-11-26 16:10:13 +08:00
{
TriangleScale = 2,
2016-11-26 16:10:13 +08:00
RelativeSizeAxes = Axes.Both,
ColourLight = Color4Extensions.FromHex(@"3a7285"),
ColourDark = Color4Extensions.FromHex(@"123744")
},
2017-03-02 02:33:01 +08:00
new FillFlowContainer
{
Padding = new MarginPadding(5),
2017-03-04 18:00:17 +08:00
Direction = FillDirection.Horizontal,
AutoSizeAxes = Axes.Both,
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
Children = new Drawable[]
{
difficultyIcon = new DifficultyIcon(beatmapInfo)
{
TooltipType = DifficultyIconTooltipType.None,
Scale = new Vector2(1.8f),
},
2017-03-02 02:33:01 +08:00
new FillFlowContainer
{
Padding = new MarginPadding { Left = 5 },
2017-03-04 18:00:17 +08:00
Direction = FillDirection.Vertical,
AutoSizeAxes = Axes.Both,
Children = new Drawable[]
{
2017-03-02 02:33:01 +08:00
new FillFlowContainer
{
2017-03-04 18:00:17 +08:00
Direction = FillDirection.Horizontal,
2017-03-02 02:33:01 +08:00
Spacing = new Vector2(4, 0),
AutoSizeAxes = Axes.Both,
Children = new[]
{
keyCountText = new OsuSpriteText
{
Font = OsuFont.GetFont(size: 20),
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Alpha = 0,
},
new OsuSpriteText
{
Text = beatmapInfo.DifficultyName,
2019-02-20 15:52:36 +08:00
Font = OsuFont.GetFont(size: 20),
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft
},
new OsuSpriteText
{
Text = BeatmapsetsStrings.ShowDetailsMappedBy(beatmapInfo.Metadata.Author.Username),
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft
},
}
},
new FillFlowContainer
{
Direction = FillDirection.Horizontal,
Spacing = new Vector2(4, 0),
2022-07-18 17:06:11 +08:00
Scale = new Vector2(0.8f),
AutoSizeAxes = Axes.Both,
Children = new Drawable[]
{
2022-07-18 17:06:11 +08:00
new TopLocalRank(beatmapInfo),
starCounter = new StarCounter()
}
}
}
}
}
}
};
}
2018-04-13 17:19:50 +08:00
protected override void LoadComplete()
{
base.LoadComplete();
ruleset.BindValueChanged(_ => updateKeyCount());
mods.BindValueChanged(_ => updateKeyCount());
}
2017-12-12 16:48:38 +08:00
protected override void Selected()
{
base.Selected();
2018-04-13 17:19:50 +08:00
MovementContainer.MoveToX(-50, 500, Easing.OutExpo);
2017-12-12 16:48:38 +08:00
background.Colour = ColourInfo.GradientVertical(
new Color4(20, 43, 51, 255),
new Color4(40, 86, 102, 255));
2018-04-13 17:19:50 +08:00
2017-12-12 16:48:38 +08:00
triangles.Colour = Color4.White;
}
2018-04-13 17:19:50 +08:00
2017-12-12 16:48:38 +08:00
protected override void Deselected()
{
base.Deselected();
2018-04-13 17:19:50 +08:00
MovementContainer.MoveToX(0, 500, Easing.OutExpo);
2017-12-12 16:48:38 +08:00
background.Colour = new Color4(20, 43, 51, 255);
triangles.Colour = OsuColour.Gray(0.5f);
}
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnClick(ClickEvent e)
2017-12-12 16:48:38 +08:00
{
2023-01-16 08:09:04 +08:00
if (Item?.State.Value == CarouselItemState.Selected)
selectRequested?.Invoke(beatmapInfo);
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
return base.OnClick(e);
2017-12-12 16:48:38 +08:00
}
2018-04-13 17:19:50 +08:00
2017-12-12 16:48:38 +08:00
protected override void ApplyState()
{
if (Item?.State.Value != CarouselItemState.Collapsed && Alpha == 0)
2017-12-12 16:48:38 +08:00
starCounter.ReplayAnimation();
2018-04-13 17:19:50 +08:00
2020-07-23 22:18:43 +08:00
starDifficultyCancellationSource?.Cancel();
2020-07-21 22:13:04 +08:00
2020-07-23 22:18:43 +08:00
// Only compute difficulty when the item is visible.
if (Item?.State.Value != CarouselItemState.Collapsed)
2020-07-23 22:18:43 +08:00
{
2020-07-21 22:13:04 +08:00
// We've potentially cancelled the computation above so a new bindable is required.
2021-10-02 23:55:29 +08:00
starDifficultyBindable = difficultyCache.GetBindableDifficulty(beatmapInfo, (starDifficultyCancellationSource = new CancellationTokenSource()).Token);
starDifficultyBindable.BindValueChanged(d =>
{
starCounter.Current = (float)(d.NewValue?.Stars ?? 0);
if (d.NewValue != null)
difficultyIcon.Current.Value = d.NewValue.Value;
}, true);
updateKeyCount();
2020-07-21 22:13:04 +08:00
}
2017-12-12 16:48:38 +08:00
base.ApplyState();
}
private void updateKeyCount()
{
if (Item?.State.Value == CarouselItemState.Collapsed)
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();
keyCountText.Alpha = 1;
keyCountText.Text = $"[{legacyRuleset.GetKeyCount(beatmapInfo, mods.Value)}K]";
}
else
keyCountText.Alpha = 0;
}
public MenuItem[] ContextMenuItems
{
get
{
List<MenuItem> items = new List<MenuItem>();
2019-04-03 22:40:20 +08:00
if (mainMenuItems != null)
items.AddRange(mainMenuItems);
if (beatmapInfo.OnlineID > 0 && beatmapOverlay != null)
items.Add(new OsuMenuItem("Details...", MenuItemType.Standard, () => beatmapOverlay.FetchAndShowBeatmap(beatmapInfo.OnlineID)));
var collectionItems = realm.Realm.All<BeatmapCollection>()
.OrderBy(c => c.Name)
.AsEnumerable()
.Select(c => new CollectionToggleMenuItem(c.ToLive(realm), beatmapInfo)).Cast<OsuMenuItem>().ToList();
2022-07-27 14:59:36 +08:00
if (manageCollectionsDialog != null)
collectionItems.Add(new OsuMenuItem("Manage...", MenuItemType.Standard, manageCollectionsDialog.Show));
2020-09-05 02:52:07 +08:00
2022-07-27 14:59:36 +08:00
items.Add(new OsuMenuItem("Collections") { Items = collectionItems });
2020-09-02 20:08:31 +08:00
if (beatmapInfo.GetOnlineURL(api, ruleset.Value) is string url)
items.Add(new OsuMenuItem(CommonStrings.CopyLink, MenuItemType.Standard, () => game?.CopyUrlToClipboard(url)));
2020-09-08 11:04:35 +08:00
if (hideRequested != null)
items.Add(new OsuMenuItem(WebCommonStrings.ButtonsHide.ToSentence(), MenuItemType.Destructive, () => hideRequested(beatmapInfo)));
2020-09-08 11:04:35 +08:00
return items.ToArray();
}
}
2020-07-21 22:13:04 +08:00
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
starDifficultyCancellationSource?.Cancel();
}
}
}