osu/osu.Game/Screens/Select/Carousel/DrawableCarouselBeatmapSet.cs

235 lines
8.3 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 09:19:50 +00:00
using System;
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Cursor;
using osu.Framework.Graphics.UserInterface;
using osu.Game.Beatmaps;
2020-09-02 12:08:31 +00:00
using osu.Game.Collections;
2018-04-13 09:19:50 +00:00
using osu.Game.Graphics.UserInterface;
using osu.Game.Overlays;
namespace osu.Game.Screens.Select.Carousel
{
public class DrawableCarouselBeatmapSet : DrawableCarouselItem, IHasContextMenu
{
public const float HEIGHT = MAX_HEIGHT;
2018-04-13 09:19:50 +00:00
private Action<BeatmapSetInfo> restoreHiddenRequested;
private Action<int> viewDetails;
2020-02-14 13:59:51 +00:00
[Resolved(CanBeNull = true)]
2020-02-14 13:14:00 +00:00
private DialogOverlay dialogOverlay { get; set; }
2020-02-14 13:30:27 +00:00
2020-09-09 06:39:15 +00:00
[Resolved(CanBeNull = true)]
private CollectionManager collectionManager { get; set; }
2020-09-02 12:08:31 +00:00
2020-09-04 18:52:07 +00:00
[Resolved(CanBeNull = true)]
2020-09-07 12:08:48 +00:00
private ManageCollectionsDialog manageCollectionsDialog { get; set; }
2020-09-04 18:52:07 +00:00
public override IEnumerable<DrawableCarouselItem> ChildItems => beatmapContainer?.Children ?? base.ChildItems;
2020-10-12 06:36:03 +00:00
private BeatmapSetInfo beatmapSet => (Item as CarouselBeatmapSet)?.BeatmapSet;
2018-04-13 09:19:50 +00:00
private Container<DrawableCarouselItem> beatmapContainer;
2020-10-12 06:36:03 +00:00
[Resolved]
private BeatmapManager manager { get; set; }
2018-04-13 09:19:50 +00:00
protected override void FreeAfterUse()
{
base.FreeAfterUse();
Item = null;
ClearTransforms();
}
2018-04-13 09:19:50 +00:00
[BackgroundDependencyLoader(true)]
2020-10-12 06:36:03 +00:00
private void load(BeatmapSetOverlay beatmapOverlay)
2018-04-13 09:19:50 +00:00
{
restoreHiddenRequested = s => s.Beatmaps.ForEach(manager.Restore);
2018-04-13 09:19:50 +00:00
if (beatmapOverlay != null)
viewDetails = beatmapOverlay.FetchAndShowBeatmapSet;
2020-10-12 06:36:03 +00:00
}
protected override void UpdateItem()
{
base.UpdateItem();
Content.Clear();
beatmapContainer = null;
if (Item == null)
return;
DelayedLoadWrapper background;
DelayedLoadWrapper mainFlow;
Header.Children = new Drawable[]
2018-04-13 09:19:50 +00:00
{
background = new DelayedLoadWrapper(new SetPanelBackground(manager.GetWorkingBeatmap(beatmapSet.Beatmaps.FirstOrDefault()))
2018-04-13 09:19:50 +00:00
{
RelativeSizeAxes = Axes.Both,
}, 300),
mainFlow = new DelayedLoadWrapper(new SetPanelContent((CarouselBeatmapSet)Item), 100),
};
background.DelayedLoadComplete += fadeContentIn;
mainFlow.DelayedLoadComplete += fadeContentIn;
}
private void fadeContentIn(Drawable d)
{
d.FadeInFromZero(1000, Easing.OutQuint);
}
protected override void Deselected()
{
base.Deselected();
2020-10-12 09:19:10 +00:00
MovementContainer.MoveToX(0, 500, Easing.OutExpo);
if (beatmapContainer != null)
{
foreach (var beatmap in beatmapContainer)
beatmap.MoveToY(0, 800, Easing.OutQuint);
}
}
protected override void Selected()
{
base.Selected();
2020-10-12 06:36:03 +00:00
MovementContainer.MoveToX(-100, 500, Easing.OutExpo);
2020-10-12 06:36:03 +00:00
updateBeatmapDifficulties();
}
private void updateBeatmapDifficulties()
{
var carouselBeatmapSet = (CarouselBeatmapSet)Item;
var visibleBeatmaps = carouselBeatmapSet.Children
.Where(c => c.Visible)
.ToArray();
// if we are already displaying all the correct beatmaps, only run animation updates.
// note that the displayed beatmaps may change due to the applied filter.
// a future optimisation could add/remove only changed difficulties rather than reinitialise.
if (beatmapContainer != null && visibleBeatmaps.Length == beatmapContainer.Count && visibleBeatmaps.All(b => beatmapContainer.Any(c => c.Item == b)))
{
updateBeatmapYPositions();
}
else
{
// on selection we show our child beatmaps.
// for now this is a simple drawable construction each selection.
// can be improved in the future.
beatmapContainer = new Container<DrawableCarouselItem>
{
X = 100,
RelativeSizeAxes = Axes.Both,
// ToArray() in this line is required due to framework oversight: https://github.com/ppy/osu-framework/pull/3929
ChildrenEnumerable = visibleBeatmaps.Select(c => c.CreateDrawableRepresentation()).ToArray()
};
2020-10-12 06:36:03 +00:00
LoadComponentAsync(beatmapContainer, loaded =>
{
// make sure the pooled target hasn't changed.
if (carouselBeatmapSet != Item)
return;
Content.Child = loaded;
updateBeatmapYPositions();
});
}
void updateBeatmapYPositions()
{
2020-10-12 11:26:20 +00:00
float yPos = DrawableCarouselBeatmap.CAROUSEL_BEATMAP_SPACING;
foreach (var panel in beatmapContainer.Children)
{
panel.MoveToY(yPos, 800, Easing.OutQuint);
yPos += panel.Item.TotalHeight;
}
}
2018-04-13 09:19:50 +00:00
}
public MenuItem[] ContextMenuItems
{
get
{
List<MenuItem> items = new List<MenuItem>();
2019-02-21 09:56:34 +00:00
if (Item.State.Value == CarouselItemState.NotSelected)
2018-04-13 09:19:50 +00:00
items.Add(new OsuMenuItem("Expand", MenuItemType.Highlighted, () => Item.State.Value = CarouselItemState.Selected));
if (beatmapSet.OnlineBeatmapSetID != null && viewDetails != null)
items.Add(new OsuMenuItem("Details...", MenuItemType.Standard, () => viewDetails(beatmapSet.OnlineBeatmapSetID.Value)));
2018-04-13 09:19:50 +00:00
2020-09-09 06:39:15 +00:00
if (collectionManager != null)
{
var collectionItems = collectionManager.Collections.Select(createCollectionMenuItem).ToList();
if (manageCollectionsDialog != null)
collectionItems.Add(new OsuMenuItem("Manage...", MenuItemType.Standard, manageCollectionsDialog.Show));
2020-09-04 18:52:07 +00:00
2020-09-09 06:39:15 +00:00
items.Add(new OsuMenuItem("Collections") { Items = collectionItems });
}
2020-09-02 12:08:31 +00:00
2020-09-08 03:04:35 +00:00
if (beatmapSet.Beatmaps.Any(b => b.Hidden))
items.Add(new OsuMenuItem("Restore all hidden", MenuItemType.Standard, () => restoreHiddenRequested(beatmapSet)));
if (dialogOverlay != null)
items.Add(new OsuMenuItem("Delete...", MenuItemType.Destructive, () => dialogOverlay.Push(new BeatmapDeleteDialog(beatmapSet))));
2018-04-13 09:19:50 +00:00
return items.ToArray();
}
}
2020-09-02 12:08:31 +00:00
private MenuItem createCollectionMenuItem(BeatmapCollection collection)
{
TernaryState state;
var countExisting = beatmapSet.Beatmaps.Count(b => collection.Beatmaps.Contains(b));
if (countExisting == beatmapSet.Beatmaps.Count)
state = TernaryState.True;
else if (countExisting > 0)
state = TernaryState.Indeterminate;
else
state = TernaryState.False;
2020-09-04 19:43:51 +00:00
return new TernaryStateMenuItem(collection.Name.Value, MenuItemType.Standard, s =>
2020-09-02 12:08:31 +00:00
{
foreach (var b in beatmapSet.Beatmaps)
{
switch (s)
{
case TernaryState.True:
if (collection.Beatmaps.Contains(b))
continue;
collection.Beatmaps.Add(b);
break;
case TernaryState.False:
collection.Beatmaps.Remove(b);
break;
}
}
})
{
State = { Value = state }
};
}
2018-04-13 09:19:50 +00:00
}
}