osu/osu.Game/Overlays/BeatmapListingOverlay.cs

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

399 lines
14 KiB
C#
Raw Normal View History

2020-02-19 14:40:54 +00:00
// 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.
2022-06-17 07:37:17 +00:00
#nullable disable
using System.Collections.Generic;
2020-02-19 14:40:54 +00:00
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
2020-02-19 14:40:54 +00:00
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions.IEnumerableExtensions;
2021-06-22 05:53:21 +00:00
using osu.Framework.Localisation;
2020-02-19 14:40:54 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
using osu.Framework.Graphics.Textures;
using osu.Framework.Input.Events;
2020-02-19 14:40:54 +00:00
using osu.Game.Audio;
using osu.Game.Beatmaps.Drawables.Cards;
2021-06-19 12:54:24 +00:00
using osu.Game.Graphics;
2020-02-19 14:40:54 +00:00
using osu.Game.Graphics.Sprites;
2021-06-19 12:54:24 +00:00
using osu.Game.Graphics.Containers;
using osu.Game.Online.API;
using osu.Game.Online.API.Requests.Responses;
2020-02-19 14:40:54 +00:00
using osu.Game.Overlays.BeatmapListing;
using osu.Game.Resources.Localisation.Web;
2020-02-19 14:40:54 +00:00
using osuTK;
using osuTK.Graphics;
namespace osu.Game.Overlays
{
2021-01-18 08:13:38 +00:00
public partial class BeatmapListingOverlay : OnlineOverlay<BeatmapListingHeader>
2020-02-19 14:40:54 +00:00
{
[Resolved]
private PreviewTrackManager previewTrackManager { get; set; }
[Resolved]
private IAPIProvider api { get; set; }
private IBindable<APIUser> apiUser;
private Container panelTarget;
private FillFlowContainer<BeatmapCard> foundContent;
private BeatmapListingFilterControl filterControl => Header.FilterControl;
2020-02-19 14:40:54 +00:00
public BeatmapListingOverlay()
: base(OverlayColourScheme.Blue)
{
}
[BackgroundDependencyLoader]
private void load()
{
2021-01-18 07:48:12 +00:00
Child = new FillFlowContainer
2020-02-19 14:40:54 +00:00
{
2021-01-18 07:48:12 +00:00
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Children = new Drawable[]
2020-02-19 14:40:54 +00:00
{
2021-01-18 07:48:12 +00:00
new Container
2020-02-19 14:40:54 +00:00
{
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
Children = new Drawable[]
{
2021-01-18 07:48:12 +00:00
new Box
2020-02-19 14:40:54 +00:00
{
2021-01-18 07:48:12 +00:00
RelativeSizeAxes = Axes.Both,
Colour = ColourProvider.Background5,
2020-02-19 14:40:54 +00:00
},
2021-01-18 07:48:12 +00:00
panelTarget = new Container
2020-02-19 14:40:54 +00:00
{
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
Masking = true,
2021-01-18 07:48:12 +00:00
Padding = new MarginPadding { Horizontal = 20 },
2020-02-19 14:40:54 +00:00
}
2021-01-18 07:48:12 +00:00
},
},
2020-02-19 14:40:54 +00:00
}
};
filterControl.TypingStarted = onTypingStarted;
filterControl.SearchStarted = onSearchStarted;
filterControl.SearchFinished = onSearchFinished;
2020-02-19 14:40:54 +00:00
}
protected override void LoadComplete()
{
base.LoadComplete();
filterControl.CardSize.BindValueChanged(_ => onCardSizeChanged());
apiUser = api.LocalUser.GetBoundCopy();
apiUser.BindValueChanged(_ => Schedule(() =>
{
if (api.IsLoggedIn)
replaceResultsAreaContent(Drawable.Empty());
}));
}
2021-07-01 10:45:17 +00:00
public void ShowWithSearch(string query)
{
filterControl.Search(query);
Show();
ScrollFlow.ScrollToStart();
2021-07-01 10:45:17 +00:00
}
2020-02-19 14:40:54 +00:00
public void ShowWithGenreFilter(SearchGenre genre)
{
ShowWithSearch(string.Empty);
filterControl.FilterGenre(genre);
}
public void ShowWithLanguageFilter(SearchLanguage language)
{
ShowWithSearch(string.Empty);
filterControl.FilterLanguage(language);
}
2021-01-18 07:48:12 +00:00
protected override BeatmapListingHeader CreateHeader() => new BeatmapListingHeader();
2020-02-19 14:40:54 +00:00
protected override Color4 BackgroundColour => ColourProvider.Background6;
2020-02-19 14:40:54 +00:00
private void onTypingStarted()
{
// temporary until the textbox/header is updated to always stay on screen.
2021-01-18 07:48:12 +00:00
ScrollFlow.ScrollToStart();
2020-02-19 23:54:35 +00:00
}
protected override void OnFocus(FocusEvent e)
2020-02-19 23:54:35 +00:00
{
base.OnFocus(e);
2020-02-19 23:54:35 +00:00
filterControl.TakeFocus();
2020-02-19 14:40:54 +00:00
}
private CancellationTokenSource cancellationToken;
2020-02-19 14:40:54 +00:00
2021-12-24 10:00:09 +00:00
private Task panelLoadTask;
private void onSearchStarted()
2020-02-19 23:54:35 +00:00
{
cancellationToken?.Cancel();
2020-02-19 14:40:54 +00:00
previewTrackManager.StopAnyPlaying(this);
if (panelTarget.Any())
2021-01-18 07:48:12 +00:00
Loading.Show();
2020-02-19 14:40:54 +00:00
}
2021-06-22 05:53:21 +00:00
private void onSearchFinished(BeatmapListingFilterControl.SearchResult searchResult)
2020-02-19 14:40:54 +00:00
{
2021-12-24 10:00:09 +00:00
cancellationToken?.Cancel();
if (searchResult.Type == BeatmapListingFilterControl.SearchResultType.SupporterOnlyFilters)
2020-02-19 14:40:54 +00:00
{
var supporterOnly = new SupporterRequiredDrawable(searchResult.SupporterOnlyFiltersUsed);
replaceResultsAreaContent(supporterOnly);
2020-02-19 14:40:54 +00:00
return;
}
var newCards = createCardsFor(searchResult.Results);
if (filterControl.CurrentPage == 0)
2020-02-19 14:40:54 +00:00
{
//No matches case
if (!newCards.Any())
2020-02-19 14:40:54 +00:00
{
replaceResultsAreaContent(new NotFoundDrawable());
return;
}
2020-02-19 14:40:54 +00:00
var content = createCardContainerFor(newCards);
panelLoadTask = LoadComponentAsync(foundContent = content, replaceResultsAreaContent, (cancellationToken = new CancellationTokenSource()).Token);
}
else
2020-02-19 14:40:54 +00:00
{
2022-09-13 21:49:02 +00:00
// new results may contain beatmaps from a previous page,
// this is dodgy but matches web behaviour for now.
// see: https://github.com/ppy/osu-web/issues/9270
// todo: replace custom equality compraer with ExceptBy in net6.0
// newCards = newCards.ExceptBy(foundContent.Select(c => c.BeatmapSet.OnlineID), c => c.BeatmapSet.OnlineID);
newCards = newCards.Except(foundContent, BeatmapCardEqualityComparer.Default);
panelLoadTask = LoadComponentsAsync(newCards, loaded =>
{
lastFetchDisplayedTime = Time.Current;
foundContent.AddRange(loaded);
loaded.ForEach(p => p.FadeIn(200, Easing.OutQuint));
}, (cancellationToken = new CancellationTokenSource()).Token);
}
}
private IEnumerable<BeatmapCard> createCardsFor(IEnumerable<APIBeatmapSet> beatmapSets) => beatmapSets.Select(set => BeatmapCard.Create(set, filterControl.CardSize.Value).With(c =>
{
c.Anchor = Anchor.TopCentre;
c.Origin = Anchor.TopCentre;
})).ToArray();
private static ReverseChildIDFillFlowContainer<BeatmapCard> createCardContainerFor(IEnumerable<BeatmapCard> newCards)
{
// spawn new children with the contained so we only clear old content at the last moment.
// reverse ID flow is required for correct Z-ordering of the cards' expandable content (last card should be front-most).
var content = new ReverseChildIDFillFlowContainer<BeatmapCard>
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Spacing = new Vector2(10),
Alpha = 0,
Margin = new MarginPadding
{
Top = 15,
// the + 20 adjustment is roughly eyeballed in order to fit all of the expanded content height after it's scaled
// as well as provide visual balance to the top margin.
Bottom = ExpandedContentScrollContainer.HEIGHT + 20
},
ChildrenEnumerable = newCards
};
return content;
}
private void replaceResultsAreaContent(Drawable content)
{
2021-01-18 07:48:12 +00:00
Loading.Hide();
lastFetchDisplayedTime = Time.Current;
panelTarget.Child = content;
content.FadeInFromZero();
}
private void onCardSizeChanged()
{
if (foundContent?.IsAlive != true || !foundContent.Any())
return;
Loading.Show();
var newCards = createCardsFor(foundContent.Reverse().Select(card => card.BeatmapSet));
2021-12-24 10:00:09 +00:00
cancellationToken?.Cancel();
panelLoadTask = LoadComponentsAsync(newCards, cards =>
{
foundContent.Clear();
foundContent.AddRange(cards);
Loading.Hide();
}, (cancellationToken = new CancellationTokenSource()).Token);
}
protected override void Dispose(bool isDisposing)
{
cancellationToken?.Cancel();
base.Dispose(isDisposing);
}
public partial class NotFoundDrawable : CompositeDrawable
{
public NotFoundDrawable()
{
RelativeSizeAxes = Axes.X;
Height = 250;
2020-02-20 02:08:42 +00:00
Alpha = 0;
Margin = new MarginPadding { Top = 15 };
}
[BackgroundDependencyLoader]
private void load(LargeTextureStore textures)
{
AddInternal(new FillFlowContainer
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Direction = FillDirection.Horizontal,
Spacing = new Vector2(10, 0),
Children = new Drawable[]
{
new Sprite
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Both,
FillMode = FillMode.Fit,
Texture = textures.Get(@"Online/not-found")
},
new OsuSpriteText
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Text = BeatmapsStrings.ListingSearchNotFoundQuote,
}
}
});
}
2020-02-19 14:40:54 +00:00
}
// TODO: localisation requires Text/LinkFlowContainer support for localising strings with links inside
// (https://github.com/ppy/osu-framework/issues/4530)
2021-06-19 12:54:24 +00:00
public partial class SupporterRequiredDrawable : CompositeDrawable
{
2021-06-24 05:45:38 +00:00
private LinkFlowContainer supporterRequiredText;
private readonly List<LocalisableString> filtersUsed;
public SupporterRequiredDrawable(List<LocalisableString> filtersUsed)
2021-06-19 12:54:24 +00:00
{
RelativeSizeAxes = Axes.X;
2021-06-20 09:17:07 +00:00
Height = 225;
2021-06-19 12:54:24 +00:00
Alpha = 0;
this.filtersUsed = filtersUsed;
2021-06-19 12:54:24 +00:00
}
[BackgroundDependencyLoader]
private void load(LargeTextureStore textures)
2021-06-19 12:54:24 +00:00
{
AddInternal(new FillFlowContainer
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Direction = FillDirection.Horizontal,
2021-06-24 05:45:38 +00:00
Children = new Drawable[]
2021-06-19 12:54:24 +00:00
{
new Sprite
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Both,
FillMode = FillMode.Fit,
Texture = textures.Get(@"Online/supporter-required"),
},
2021-06-24 05:45:38 +00:00
supporterRequiredText = new LinkFlowContainer
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
AutoSizeAxes = Axes.Both,
Margin = new MarginPadding { Bottom = 10 },
},
2021-06-19 12:54:24 +00:00
}
});
2021-06-24 05:45:38 +00:00
supporterRequiredText.AddText(
BeatmapsStrings.ListingSearchSupporterFilterQuoteDefault(string.Join(" and ", filtersUsed), "").ToString(),
2021-06-22 05:53:21 +00:00
t =>
{
t.Font = OsuFont.GetFont(size: 16);
t.Colour = Colour4.White;
}
2021-06-24 05:45:38 +00:00
);
2021-06-22 05:53:21 +00:00
supporterRequiredText.AddLink(BeatmapsStrings.ListingSearchSupporterFilterQuoteLinkText.ToString(), @"/store/products/supporter-tag");
2021-06-19 12:54:24 +00:00
}
}
private const double time_between_fetches = 500;
private double lastFetchDisplayedTime;
protected override void Update()
{
base.Update();
const int pagination_scroll_distance = 500;
bool shouldShowMore = panelLoadTask?.IsCompleted != false
&& Time.Current - lastFetchDisplayedTime > time_between_fetches
2021-01-18 07:48:12 +00:00
&& (ScrollFlow.ScrollableExtent > 0 && ScrollFlow.IsScrolledToEnd(pagination_scroll_distance));
if (shouldShowMore)
filterControl.FetchNextPage();
}
private class BeatmapCardEqualityComparer : IEqualityComparer<BeatmapCard>
{
public static BeatmapCardEqualityComparer Default { get; } = new BeatmapCardEqualityComparer();
public bool Equals(BeatmapCard x, BeatmapCard y)
{
if (ReferenceEquals(x, y)) return true;
if (ReferenceEquals(x, null)) return false;
if (ReferenceEquals(y, null)) return false;
return x.BeatmapSet.Equals(y.BeatmapSet);
}
public int GetHashCode(BeatmapCard obj) => obj.BeatmapSet.GetHashCode();
}
2020-02-19 14:40:54 +00:00
}
}