2020-02-19 22:40:54 +08: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 15:37:17 +08:00
|
|
|
|
#nullable disable
|
|
|
|
|
|
2020-03-06 08:09:43 +08:00
|
|
|
|
using System.Collections.Generic;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using System.Linq;
|
2020-03-06 08:09:43 +08:00
|
|
|
|
using System.Threading;
|
2020-05-14 14:35:11 +08:00
|
|
|
|
using System.Threading.Tasks;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using osu.Framework.Allocation;
|
2022-02-22 13:56:04 +08:00
|
|
|
|
using osu.Framework.Bindables;
|
2020-05-14 14:35:11 +08:00
|
|
|
|
using osu.Framework.Extensions.IEnumerableExtensions;
|
2021-06-22 13:53:21 +08:00
|
|
|
|
using osu.Framework.Localisation;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Graphics.Shapes;
|
2020-02-19 23:17:02 +08:00
|
|
|
|
using osu.Framework.Graphics.Sprites;
|
|
|
|
|
using osu.Framework.Graphics.Textures;
|
2020-04-21 14:47:43 +08:00
|
|
|
|
using osu.Framework.Input.Events;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using osu.Game.Audio;
|
2021-10-23 22:55:05 +08:00
|
|
|
|
using osu.Game.Beatmaps.Drawables.Cards;
|
2021-06-19 20:54:24 +08:00
|
|
|
|
using osu.Game.Graphics;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using osu.Game.Graphics.Sprites;
|
2021-06-19 20:54:24 +08:00
|
|
|
|
using osu.Game.Graphics.Containers;
|
2022-02-22 13:56:04 +08:00
|
|
|
|
using osu.Game.Online.API;
|
2021-12-22 20:16:36 +08:00
|
|
|
|
using osu.Game.Online.API.Requests.Responses;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using osu.Game.Overlays.BeatmapListing;
|
2021-06-16 12:46:13 +08:00
|
|
|
|
using osu.Game.Resources.Localisation.Web;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
using osuTK;
|
|
|
|
|
using osuTK.Graphics;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Overlays
|
|
|
|
|
{
|
2021-01-18 16:13:38 +08:00
|
|
|
|
public class BeatmapListingOverlay : OnlineOverlay<BeatmapListingHeader>
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
|
|
|
|
[Resolved]
|
|
|
|
|
private PreviewTrackManager previewTrackManager { get; set; }
|
|
|
|
|
|
2022-02-22 13:56:04 +08:00
|
|
|
|
[Resolved]
|
|
|
|
|
private IAPIProvider api { get; set; }
|
|
|
|
|
|
|
|
|
|
private IBindable<APIUser> apiUser;
|
|
|
|
|
|
2020-02-20 10:08:42 +08:00
|
|
|
|
private Drawable currentContent;
|
2020-03-06 08:09:43 +08:00
|
|
|
|
private Container panelTarget;
|
2021-11-28 00:53:57 +08:00
|
|
|
|
private FillFlowContainer<BeatmapCard> foundContent;
|
2020-05-12 02:18:47 +08:00
|
|
|
|
private NotFoundDrawable notFoundContent;
|
2021-06-19 20:54:24 +08:00
|
|
|
|
private SupporterRequiredDrawable supporterRequiredContent;
|
2021-01-18 15:48:12 +08:00
|
|
|
|
private BeatmapListingFilterControl filterControl;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
|
|
|
|
public BeatmapListingOverlay()
|
|
|
|
|
: base(OverlayColourScheme.Blue)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load()
|
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
Child = new FillFlowContainer
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
Direction = FillDirection.Vertical,
|
|
|
|
|
Children = new Drawable[]
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
filterControl = new BeatmapListingFilterControl
|
|
|
|
|
{
|
|
|
|
|
TypingStarted = onTypingStarted,
|
|
|
|
|
SearchStarted = onSearchStarted,
|
|
|
|
|
SearchFinished = onSearchFinished,
|
|
|
|
|
},
|
|
|
|
|
new Container
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
new Box
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
2021-10-23 22:55:05 +08:00
|
|
|
|
Colour = ColourProvider.Background5,
|
2020-02-19 22:40:54 +08:00
|
|
|
|
},
|
2021-01-18 15:48:12 +08:00
|
|
|
|
panelTarget = new Container
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
2021-09-11 21:54:49 +08:00
|
|
|
|
Masking = true,
|
2021-01-18 15:48:12 +08:00
|
|
|
|
Padding = new MarginPadding { Horizontal = 20 },
|
2020-02-19 22:40:54 +08:00
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
notFoundContent = new NotFoundDrawable(),
|
2021-06-19 20:54:24 +08:00
|
|
|
|
supporterRequiredContent = new SupporterRequiredDrawable(),
|
2020-02-19 22:40:54 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-01-18 15:48:12 +08:00
|
|
|
|
},
|
2021-01-04 21:42:39 +08:00
|
|
|
|
},
|
2020-02-19 22:40:54 +08:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-22 20:16:36 +08:00
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
|
|
|
|
filterControl.CardSize.BindValueChanged(_ => onCardSizeChanged());
|
2022-02-22 13:56:04 +08:00
|
|
|
|
|
|
|
|
|
apiUser = api.LocalUser.GetBoundCopy();
|
2022-08-09 15:38:12 +08:00
|
|
|
|
apiUser.BindValueChanged(_ => Schedule(() =>
|
2022-02-22 13:56:04 +08:00
|
|
|
|
{
|
|
|
|
|
if (api.IsLoggedIn)
|
|
|
|
|
addContentToResultsArea(Drawable.Empty());
|
2022-08-09 15:38:12 +08:00
|
|
|
|
}));
|
2021-12-22 20:16:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-01 18:45:17 +08:00
|
|
|
|
public void ShowWithSearch(string query)
|
|
|
|
|
{
|
|
|
|
|
filterControl.Search(query);
|
|
|
|
|
Show();
|
2022-10-26 09:06:49 +08:00
|
|
|
|
ScrollFlow.ScrollToStart();
|
2021-07-01 18:45:17 +08:00
|
|
|
|
}
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
2021-01-18 15:48:12 +08:00
|
|
|
|
protected override BeatmapListingHeader CreateHeader() => new BeatmapListingHeader();
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
2021-02-09 17:32:44 +08:00
|
|
|
|
protected override Color4 BackgroundColour => ColourProvider.Background6;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
2020-11-10 15:26:30 +08:00
|
|
|
|
private void onTypingStarted()
|
|
|
|
|
{
|
|
|
|
|
// temporary until the textbox/header is updated to always stay on screen.
|
2021-01-18 15:48:12 +08:00
|
|
|
|
ScrollFlow.ScrollToStart();
|
2020-02-20 07:54:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-21 14:47:43 +08:00
|
|
|
|
protected override void OnFocus(FocusEvent e)
|
2020-02-20 07:54:35 +08:00
|
|
|
|
{
|
2020-04-21 14:47:43 +08:00
|
|
|
|
base.OnFocus(e);
|
2020-02-20 07:54:35 +08:00
|
|
|
|
|
2020-04-21 14:47:43 +08:00
|
|
|
|
filterControl.TakeFocus();
|
2020-02-19 22:40:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-06 08:09:43 +08:00
|
|
|
|
private CancellationTokenSource cancellationToken;
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
2021-12-24 18:00:09 +08:00
|
|
|
|
private Task panelLoadTask;
|
|
|
|
|
|
2020-03-06 08:09:43 +08:00
|
|
|
|
private void onSearchStarted()
|
2020-02-20 07:54:35 +08:00
|
|
|
|
{
|
2020-03-06 08:09:43 +08:00
|
|
|
|
cancellationToken?.Cancel();
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
|
|
|
|
previewTrackManager.StopAnyPlaying(this);
|
|
|
|
|
|
2020-03-06 08:09:43 +08:00
|
|
|
|
if (panelTarget.Any())
|
2021-01-18 15:48:12 +08:00
|
|
|
|
Loading.Show();
|
2020-02-19 22:40:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-22 13:53:21 +08:00
|
|
|
|
private void onSearchFinished(BeatmapListingFilterControl.SearchResult searchResult)
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2021-12-24 18:00:09 +08:00
|
|
|
|
cancellationToken?.Cancel();
|
|
|
|
|
|
2021-06-27 02:31:26 +08:00
|
|
|
|
if (searchResult.Type == BeatmapListingFilterControl.SearchResultType.SupporterOnlyFilters)
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2021-06-27 02:31:26 +08:00
|
|
|
|
supporterRequiredContent.UpdateText(searchResult.SupporterOnlyFiltersUsed);
|
2022-01-04 02:30:17 +08:00
|
|
|
|
addContentToResultsArea(supporterRequiredContent);
|
2020-02-19 22:40:54 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-22 20:16:36 +08:00
|
|
|
|
var newCards = createCardsFor(searchResult.Results);
|
2020-02-19 23:17:02 +08:00
|
|
|
|
|
2020-05-14 14:35:11 +08:00
|
|
|
|
if (filterControl.CurrentPage == 0)
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2020-05-14 14:35:11 +08:00
|
|
|
|
//No matches case
|
2021-12-22 20:16:36 +08:00
|
|
|
|
if (!newCards.Any())
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2022-01-04 02:30:17 +08:00
|
|
|
|
addContentToResultsArea(notFoundContent);
|
2020-05-14 14:35:11 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
2020-02-19 22:40:54 +08:00
|
|
|
|
|
2021-12-22 20:16:36 +08:00
|
|
|
|
var content = createCardContainerFor(newCards);
|
2020-05-12 02:18:47 +08:00
|
|
|
|
|
2022-01-04 02:30:17 +08:00
|
|
|
|
panelLoadTask = LoadComponentAsync(foundContent = content, addContentToResultsArea, (cancellationToken = new CancellationTokenSource()).Token);
|
2020-05-12 02:18:47 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
2020-02-19 22:40:54 +08:00
|
|
|
|
{
|
2022-09-14 05:49:02 +08: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
|
2022-11-05 00:35:00 +08:00
|
|
|
|
// todo: replace custom equality compraer with ExceptBy in net6.0
|
|
|
|
|
// newCards = newCards.ExceptBy(foundContent.Select(c => c.BeatmapSet.OnlineID), c => c.BeatmapSet.OnlineID);
|
2022-11-05 00:05:03 +08:00
|
|
|
|
newCards = newCards.Except(foundContent, BeatmapCardEqualityComparer.Default);
|
2022-09-13 06:38:52 +08:00
|
|
|
|
|
2021-12-24 17:58:31 +08:00
|
|
|
|
panelLoadTask = LoadComponentsAsync(newCards, loaded =>
|
2020-05-12 02:18:47 +08:00
|
|
|
|
{
|
2020-05-14 14:35:11 +08:00
|
|
|
|
lastFetchDisplayedTime = Time.Current;
|
|
|
|
|
foundContent.AddRange(loaded);
|
|
|
|
|
loaded.ForEach(p => p.FadeIn(200, Easing.OutQuint));
|
2021-12-24 17:58:31 +08:00
|
|
|
|
}, (cancellationToken = new CancellationTokenSource()).Token);
|
2020-05-12 02:18:47 +08:00
|
|
|
|
}
|
2020-02-19 23:17:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-09-13 06:38:52 +08:00
|
|
|
|
private IEnumerable<BeatmapCard> createCardsFor(IEnumerable<APIBeatmapSet> beatmapSets) => beatmapSets.Select(set => BeatmapCard.Create(set, filterControl.CardSize.Value).With(c =>
|
2021-12-22 20:16:36 +08:00
|
|
|
|
{
|
|
|
|
|
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,
|
2022-01-03 12:18:27 +08:00
|
|
|
|
Margin = new MarginPadding
|
|
|
|
|
{
|
2022-01-20 05:10:05 +08:00
|
|
|
|
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
|
2022-01-03 12:18:27 +08:00
|
|
|
|
},
|
2021-12-22 20:16:36 +08:00
|
|
|
|
ChildrenEnumerable = newCards
|
|
|
|
|
};
|
|
|
|
|
return content;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-04 02:30:17 +08:00
|
|
|
|
private void addContentToResultsArea(Drawable content)
|
2020-02-19 23:17:02 +08:00
|
|
|
|
{
|
2021-01-18 15:48:12 +08:00
|
|
|
|
Loading.Hide();
|
2020-05-14 14:35:11 +08:00
|
|
|
|
lastFetchDisplayedTime = Time.Current;
|
2020-02-21 15:13:24 +08:00
|
|
|
|
|
2021-01-26 03:11:50 +08:00
|
|
|
|
if (content == currentContent)
|
|
|
|
|
return;
|
2020-02-21 15:13:24 +08:00
|
|
|
|
|
2020-03-06 08:09:43 +08:00
|
|
|
|
var lastContent = currentContent;
|
2020-02-20 12:48:34 +08:00
|
|
|
|
|
|
|
|
|
if (lastContent != null)
|
|
|
|
|
{
|
2022-01-04 02:39:07 +08:00
|
|
|
|
lastContent.FadeOut();
|
2022-01-04 02:33:01 +08:00
|
|
|
|
if (!isPlaceholderContent(lastContent))
|
2022-01-04 02:39:07 +08:00
|
|
|
|
lastContent.Expire();
|
2020-02-20 12:48:34 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 02:18:47 +08:00
|
|
|
|
if (!content.IsAlive)
|
|
|
|
|
panelTarget.Add(content);
|
|
|
|
|
|
2022-01-04 02:39:07 +08:00
|
|
|
|
content.FadeInFromZero();
|
2020-05-12 02:18:47 +08:00
|
|
|
|
currentContent = content;
|
2020-02-19 23:17:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-04 02:33:01 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether <paramref name="drawable"/> is a static placeholder reused multiple times by this overlay.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private bool isPlaceholderContent(Drawable drawable)
|
|
|
|
|
=> drawable == notFoundContent || drawable == supporterRequiredContent;
|
|
|
|
|
|
2021-12-22 20:16:36 +08:00
|
|
|
|
private void onCardSizeChanged()
|
|
|
|
|
{
|
2022-01-04 03:02:46 +08:00
|
|
|
|
if (foundContent?.IsAlive != true || !foundContent.Any())
|
2021-12-22 20:16:36 +08:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Loading.Show();
|
|
|
|
|
|
|
|
|
|
var newCards = createCardsFor(foundContent.Reverse().Select(card => card.BeatmapSet));
|
|
|
|
|
|
2021-12-24 18:00:09 +08:00
|
|
|
|
cancellationToken?.Cancel();
|
|
|
|
|
|
2021-12-24 17:58:31 +08:00
|
|
|
|
panelLoadTask = LoadComponentsAsync(newCards, cards =>
|
2021-12-22 20:16:36 +08:00
|
|
|
|
{
|
|
|
|
|
foundContent.Clear();
|
|
|
|
|
foundContent.AddRange(cards);
|
|
|
|
|
Loading.Hide();
|
|
|
|
|
}, (cancellationToken = new CancellationTokenSource()).Token);
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-19 23:17:02 +08:00
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
|
{
|
2020-03-06 08:09:43 +08:00
|
|
|
|
cancellationToken?.Cancel();
|
2020-02-19 23:17:02 +08:00
|
|
|
|
base.Dispose(isDisposing);
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-26 04:41:05 +08:00
|
|
|
|
public class NotFoundDrawable : CompositeDrawable
|
2020-02-19 23:17:02 +08:00
|
|
|
|
{
|
2020-02-20 07:43:13 +08:00
|
|
|
|
public NotFoundDrawable()
|
2020-02-19 23:17:02 +08:00
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X;
|
|
|
|
|
Height = 250;
|
2020-02-20 10:08:42 +08:00
|
|
|
|
Alpha = 0;
|
2020-02-19 23:17:02 +08:00
|
|
|
|
Margin = new MarginPadding { Top = 15 };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(TextureStore 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,
|
2021-06-16 12:46:13 +08:00
|
|
|
|
Text = BeatmapsStrings.ListingSearchNotFoundQuote,
|
2020-02-19 23:17:02 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-02-19 22:40:54 +08:00
|
|
|
|
}
|
2020-05-12 02:18:47 +08:00
|
|
|
|
|
2021-06-27 02:40:54 +08:00
|
|
|
|
// TODO: localisation requires Text/LinkFlowContainer support for localising strings with links inside
|
|
|
|
|
// (https://github.com/ppy/osu-framework/issues/4530)
|
2021-06-19 20:54:24 +08:00
|
|
|
|
public class SupporterRequiredDrawable : CompositeDrawable
|
|
|
|
|
{
|
2021-06-24 13:45:38 +08:00
|
|
|
|
private LinkFlowContainer supporterRequiredText;
|
2021-06-20 21:23:54 +08:00
|
|
|
|
|
2021-06-19 20:54:24 +08:00
|
|
|
|
public SupporterRequiredDrawable()
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X;
|
2021-06-20 17:17:07 +08:00
|
|
|
|
Height = 225;
|
2021-06-19 20:54:24 +08:00
|
|
|
|
Alpha = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(TextureStore textures)
|
|
|
|
|
{
|
|
|
|
|
AddInternal(new FillFlowContainer
|
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre,
|
|
|
|
|
RelativeSizeAxes = Axes.Y,
|
|
|
|
|
AutoSizeAxes = Axes.X,
|
|
|
|
|
Direction = FillDirection.Horizontal,
|
2021-06-24 13:45:38 +08:00
|
|
|
|
Children = new Drawable[]
|
2021-06-19 20:54:24 +08:00
|
|
|
|
{
|
|
|
|
|
new Sprite
|
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre,
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
|
FillMode = FillMode.Fit,
|
|
|
|
|
Texture = textures.Get(@"Online/supporter-required"),
|
|
|
|
|
},
|
2021-06-24 13:45:38 +08:00
|
|
|
|
supporterRequiredText = new LinkFlowContainer
|
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre,
|
|
|
|
|
AutoSizeAxes = Axes.Both,
|
|
|
|
|
Margin = new MarginPadding { Bottom = 10 },
|
|
|
|
|
},
|
2021-06-19 20:54:24 +08:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-22 13:53:21 +08:00
|
|
|
|
public void UpdateText(List<LocalisableString> filters)
|
2021-06-21 15:31:47 +08:00
|
|
|
|
{
|
2021-06-24 13:45:38 +08:00
|
|
|
|
supporterRequiredText.Clear();
|
2021-06-19 20:54:24 +08:00
|
|
|
|
|
2021-06-24 13:45:38 +08:00
|
|
|
|
supporterRequiredText.AddText(
|
|
|
|
|
BeatmapsStrings.ListingSearchSupporterFilterQuoteDefault(string.Join(" and ", filters), "").ToString(),
|
2021-06-22 13:53:21 +08:00
|
|
|
|
t =>
|
|
|
|
|
{
|
|
|
|
|
t.Font = OsuFont.GetFont(size: 16);
|
|
|
|
|
t.Colour = Colour4.White;
|
|
|
|
|
}
|
2021-06-24 13:45:38 +08:00
|
|
|
|
);
|
2021-06-22 13:53:21 +08:00
|
|
|
|
|
|
|
|
|
supporterRequiredText.AddLink(BeatmapsStrings.ListingSearchSupporterFilterQuoteLinkText.ToString(), @"/store/products/supporter-tag");
|
2021-06-19 20:54:24 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-14 14:35:11 +08:00
|
|
|
|
private const double time_between_fetches = 500;
|
|
|
|
|
|
|
|
|
|
private double lastFetchDisplayedTime;
|
|
|
|
|
|
2020-05-12 02:18:47 +08:00
|
|
|
|
protected override void Update()
|
|
|
|
|
{
|
|
|
|
|
base.Update();
|
|
|
|
|
|
2020-05-14 14:35:11 +08:00
|
|
|
|
const int pagination_scroll_distance = 500;
|
|
|
|
|
|
2021-12-24 17:58:31 +08:00
|
|
|
|
bool shouldShowMore = panelLoadTask?.IsCompleted != false
|
2020-05-14 14:35:11 +08:00
|
|
|
|
&& Time.Current - lastFetchDisplayedTime > time_between_fetches
|
2021-01-18 15:48:12 +08:00
|
|
|
|
&& (ScrollFlow.ScrollableExtent > 0 && ScrollFlow.IsScrolledToEnd(pagination_scroll_distance));
|
2020-05-14 14:35:11 +08:00
|
|
|
|
|
|
|
|
|
if (shouldShowMore)
|
|
|
|
|
filterControl.FetchNextPage();
|
2020-05-12 02:18:47 +08:00
|
|
|
|
}
|
2022-11-05 00:05:03 +08:00
|
|
|
|
|
|
|
|
|
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 22:40:54 +08:00
|
|
|
|
}
|
|
|
|
|
}
|