2018-04-13 09:19:50 +00:00
|
|
|
|
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
|
|
|
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
2018-05-14 08:41:35 +00:00
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Configuration;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Input;
|
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Graphics.Containers;
|
|
|
|
|
using OpenTK;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Overlays.Music
|
|
|
|
|
{
|
|
|
|
|
public class PlaylistList : CompositeDrawable
|
|
|
|
|
{
|
2018-05-14 08:41:35 +00:00
|
|
|
|
public Action<BeatmapSetInfo> Selected;
|
|
|
|
|
public Action<BeatmapSetInfo, int> OrderChanged;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
private readonly ItemsScrollContainer items;
|
|
|
|
|
|
|
|
|
|
public PlaylistList()
|
|
|
|
|
{
|
|
|
|
|
InternalChild = items = new ItemsScrollContainer
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
2018-05-14 08:41:35 +00:00
|
|
|
|
Selected = set => Selected?.Invoke(set),
|
|
|
|
|
OrderChanged = (s, i) => OrderChanged?.Invoke(s, i)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public new MarginPadding Padding
|
|
|
|
|
{
|
|
|
|
|
get { return base.Padding; }
|
|
|
|
|
set { base.Padding = value; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public BeatmapSetInfo FirstVisibleSet => items.FirstVisibleSet;
|
|
|
|
|
|
|
|
|
|
public void Filter(string searchTerm) => items.SearchTerm = searchTerm;
|
|
|
|
|
|
|
|
|
|
private class ItemsScrollContainer : OsuScrollContainer
|
|
|
|
|
{
|
2018-05-14 08:41:35 +00:00
|
|
|
|
public Action<BeatmapSetInfo> Selected;
|
|
|
|
|
public Action<BeatmapSetInfo, int> OrderChanged;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
private readonly SearchContainer search;
|
|
|
|
|
private readonly FillFlowContainer<PlaylistItem> items;
|
|
|
|
|
|
2018-05-14 08:41:35 +00:00
|
|
|
|
private readonly IBindable<WorkingBeatmap> beatmapBacking = new Bindable<WorkingBeatmap>();
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
public ItemsScrollContainer()
|
|
|
|
|
{
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
search = new SearchContainer
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
items = new ItemSearchContainer
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-14 08:41:35 +00:00
|
|
|
|
[BackgroundDependencyLoader]
|
2018-05-23 08:37:39 +00:00
|
|
|
|
private void load(BeatmapManager beatmaps, IGameBeatmap beatmap)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-05-14 08:41:35 +00:00
|
|
|
|
beatmaps.GetAllUsableBeatmapSets().ForEach(addBeatmapSet);
|
|
|
|
|
beatmaps.ItemAdded += addBeatmapSet;
|
|
|
|
|
beatmaps.ItemRemoved += removeBeatmapSet;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-05-23 08:37:39 +00:00
|
|
|
|
beatmapBacking.BindTo(beatmap);
|
2018-05-14 08:41:35 +00:00
|
|
|
|
beatmapBacking.ValueChanged += _ => updateSelectedSet();
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-05-14 08:41:35 +00:00
|
|
|
|
private void addBeatmapSet(BeatmapSetInfo obj)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-05-14 08:41:35 +00:00
|
|
|
|
var newItem = new PlaylistItem(obj) { OnSelect = set => Selected?.Invoke(set) };
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
items.Add(newItem);
|
2018-05-14 08:41:35 +00:00
|
|
|
|
items.SetLayoutPosition(newItem, items.Count - 1);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-05-14 08:41:35 +00:00
|
|
|
|
private void removeBeatmapSet(BeatmapSetInfo obj)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-05-14 08:41:35 +00:00
|
|
|
|
var itemToRemove = items.FirstOrDefault(i => i.BeatmapSetInfo.ID == obj.ID);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
if (itemToRemove != null)
|
|
|
|
|
items.Remove(itemToRemove);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-14 08:41:35 +00:00
|
|
|
|
private void updateSelectedSet()
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-05-14 08:41:35 +00:00
|
|
|
|
foreach (PlaylistItem s in items.Children)
|
|
|
|
|
s.Selected = s.BeatmapSetInfo.ID == beatmapBacking.Value.BeatmapSetInfo.ID;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string SearchTerm
|
|
|
|
|
{
|
|
|
|
|
get { return search.SearchTerm; }
|
|
|
|
|
set { search.SearchTerm = value; }
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public BeatmapSetInfo FirstVisibleSet => items.FirstOrDefault(i => i.MatchingFilter)?.BeatmapSetInfo;
|
|
|
|
|
|
|
|
|
|
private Vector2 nativeDragPosition;
|
|
|
|
|
private PlaylistItem draggedItem;
|
|
|
|
|
|
|
|
|
|
protected override bool OnDragStart(InputState state)
|
|
|
|
|
{
|
|
|
|
|
nativeDragPosition = state.Mouse.NativeState.Position;
|
|
|
|
|
draggedItem = items.FirstOrDefault(d => d.IsDraggable);
|
|
|
|
|
return draggedItem != null || base.OnDragStart(state);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override bool OnDrag(InputState state)
|
|
|
|
|
{
|
|
|
|
|
nativeDragPosition = state.Mouse.NativeState.Position;
|
|
|
|
|
if (draggedItem == null)
|
|
|
|
|
return base.OnDrag(state);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override bool OnDragEnd(InputState state)
|
|
|
|
|
{
|
|
|
|
|
nativeDragPosition = state.Mouse.NativeState.Position;
|
|
|
|
|
var handled = draggedItem != null || base.OnDragEnd(state);
|
|
|
|
|
draggedItem = null;
|
|
|
|
|
|
|
|
|
|
return handled;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void Update()
|
|
|
|
|
{
|
|
|
|
|
base.Update();
|
|
|
|
|
|
|
|
|
|
if (draggedItem == null)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
updateScrollPosition();
|
|
|
|
|
updateDragPosition();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void updateScrollPosition()
|
|
|
|
|
{
|
|
|
|
|
const float start_offset = 10;
|
|
|
|
|
const double max_power = 50;
|
|
|
|
|
const double exp_base = 1.05;
|
|
|
|
|
|
|
|
|
|
var localPos = ToLocalSpace(nativeDragPosition);
|
|
|
|
|
|
|
|
|
|
if (localPos.Y < start_offset)
|
|
|
|
|
{
|
|
|
|
|
if (Current <= 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
var power = Math.Min(max_power, Math.Abs(start_offset - localPos.Y));
|
|
|
|
|
ScrollBy(-(float)Math.Pow(exp_base, power));
|
|
|
|
|
}
|
|
|
|
|
else if (localPos.Y > DrawHeight - start_offset)
|
|
|
|
|
{
|
|
|
|
|
if (IsScrolledToEnd())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
var power = Math.Min(max_power, Math.Abs(DrawHeight - start_offset - localPos.Y));
|
|
|
|
|
ScrollBy((float)Math.Pow(exp_base, power));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void updateDragPosition()
|
|
|
|
|
{
|
|
|
|
|
var itemsPos = items.ToLocalSpace(nativeDragPosition);
|
|
|
|
|
|
|
|
|
|
int srcIndex = (int)items.GetLayoutPosition(draggedItem);
|
|
|
|
|
|
|
|
|
|
// Find the last item with position < mouse position. Note we can't directly use
|
|
|
|
|
// the item positions as they are being transformed
|
|
|
|
|
float heightAccumulator = 0;
|
|
|
|
|
int dstIndex = 0;
|
|
|
|
|
for (; dstIndex < items.Count; dstIndex++)
|
|
|
|
|
{
|
|
|
|
|
// Using BoundingBox here takes care of scale, paddings, etc...
|
|
|
|
|
heightAccumulator += items[dstIndex].BoundingBox.Height;
|
|
|
|
|
if (heightAccumulator > itemsPos.Y)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dstIndex = MathHelper.Clamp(dstIndex, 0, items.Count - 1);
|
|
|
|
|
|
|
|
|
|
if (srcIndex == dstIndex)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (srcIndex < dstIndex)
|
|
|
|
|
{
|
|
|
|
|
for (int i = srcIndex + 1; i <= dstIndex; i++)
|
|
|
|
|
items.SetLayoutPosition(items[i], i - 1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (int i = dstIndex; i < srcIndex; i++)
|
|
|
|
|
items.SetLayoutPosition(items[i], i + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
items.SetLayoutPosition(draggedItem, dstIndex);
|
2018-05-14 08:41:35 +00:00
|
|
|
|
OrderChanged?.Invoke(draggedItem.BeatmapSetInfo, dstIndex);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private class ItemSearchContainer : FillFlowContainer<PlaylistItem>, IHasFilterableChildren
|
|
|
|
|
{
|
|
|
|
|
public IEnumerable<string> FilterTerms => new string[] { };
|
|
|
|
|
|
|
|
|
|
public bool MatchingFilter
|
|
|
|
|
{
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (value)
|
|
|
|
|
InvalidateLayout();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IEnumerable<IFilterable> FilterableChildren => Children;
|
|
|
|
|
|
|
|
|
|
public ItemSearchContainer()
|
|
|
|
|
{
|
|
|
|
|
LayoutDuration = 200;
|
|
|
|
|
LayoutEasing = Easing.OutQuint;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|