mirror of
https://github.com/ppy/osu.git
synced 2025-01-16 13:32:54 +08:00
4f6263ef86
This is important when using dynamic compiling to rapidly iterate. Until we actually split projects out into pieces (like the abstract ruleset project we have talked about) there is no advantage to using internal in the osu! game code.
629 lines
22 KiB
C#
629 lines
22 KiB
C#
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
|
|
|
using OpenTK;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Containers;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using osu.Game.Beatmaps.Drawables;
|
|
using osu.Game.Configuration;
|
|
using osu.Framework.Input;
|
|
using OpenTK.Input;
|
|
using osu.Framework.MathUtils;
|
|
using System.Diagnostics;
|
|
using System.Threading.Tasks;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Threading;
|
|
using osu.Framework.Configuration;
|
|
using osu.Game.Beatmaps;
|
|
using osu.Game.Graphics.Containers;
|
|
using osu.Game.Graphics.Cursor;
|
|
|
|
namespace osu.Game.Screens.Select
|
|
{
|
|
public class BeatmapCarousel : OsuScrollContainer
|
|
{
|
|
public BeatmapInfo SelectedBeatmap => selectedPanel?.Beatmap;
|
|
|
|
public override bool HandleInput => AllowSelection;
|
|
|
|
public Action BeatmapsChanged;
|
|
|
|
public IEnumerable<BeatmapSetInfo> Beatmaps
|
|
{
|
|
get { return groups.Select(g => g.BeatmapSet); }
|
|
|
|
set
|
|
{
|
|
scrollableContent.Clear(false);
|
|
panels.Clear();
|
|
groups.Clear();
|
|
|
|
List<BeatmapGroup> newGroups = null;
|
|
|
|
Task.Run(() =>
|
|
{
|
|
newGroups = value.Select(createGroup).ToList();
|
|
criteria.Filter(newGroups);
|
|
}).ContinueWith(t =>
|
|
{
|
|
Schedule(() =>
|
|
{
|
|
foreach (var g in newGroups)
|
|
if (g != null) addGroup(g);
|
|
|
|
computeYPositions();
|
|
BeatmapsChanged?.Invoke();
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
private readonly List<float> yPositions = new List<float>();
|
|
|
|
/// <summary>
|
|
/// Required for now unfortunately.
|
|
/// </summary>
|
|
private BeatmapManager manager;
|
|
|
|
private readonly Container<Panel> scrollableContent;
|
|
|
|
private readonly List<BeatmapGroup> groups = new List<BeatmapGroup>();
|
|
|
|
private Bindable<SelectionRandomType> randomType;
|
|
private readonly List<BeatmapGroup> seenGroups = new List<BeatmapGroup>();
|
|
|
|
private readonly List<Panel> panels = new List<Panel>();
|
|
|
|
private readonly Stack<KeyValuePair<BeatmapGroup, BeatmapPanel>> randomSelectedBeatmaps = new Stack<KeyValuePair<BeatmapGroup, BeatmapPanel>>();
|
|
|
|
private BeatmapGroup selectedGroup;
|
|
private BeatmapPanel selectedPanel;
|
|
|
|
public BeatmapCarousel()
|
|
{
|
|
Add(new OsuContextMenuContainer
|
|
{
|
|
RelativeSizeAxes = Axes.X,
|
|
AutoSizeAxes = Axes.Y,
|
|
Child = scrollableContent = new Container<Panel>
|
|
{
|
|
RelativeSizeAxes = Axes.X,
|
|
}
|
|
});
|
|
}
|
|
|
|
public void AddBeatmap(BeatmapSetInfo beatmapSet)
|
|
{
|
|
Schedule(() =>
|
|
{
|
|
var group = createGroup(beatmapSet);
|
|
|
|
if (group == null)
|
|
return;
|
|
|
|
addGroup(group);
|
|
computeYPositions();
|
|
if (selectedGroup == null)
|
|
selectGroup(group);
|
|
});
|
|
}
|
|
|
|
public void RemoveBeatmap(BeatmapSetInfo beatmapSet)
|
|
{
|
|
Schedule(() => removeGroup(groups.Find(b => b.BeatmapSet.ID == beatmapSet.ID)));
|
|
}
|
|
|
|
public void UpdateBeatmap(BeatmapInfo beatmap)
|
|
{
|
|
// todo: this method should not run more than once for the same BeatmapSetInfo.
|
|
var set = manager.QueryBeatmapSet(s => s.ID == beatmap.BeatmapSetInfoID);
|
|
|
|
// todo: this method should be smarter as to not recreate panels that haven't changed, etc.
|
|
var group = groups.Find(b => b.BeatmapSet.ID == set.ID);
|
|
|
|
if (group == null)
|
|
return;
|
|
|
|
int i = groups.IndexOf(group);
|
|
groups.RemoveAt(i);
|
|
|
|
var newGroup = createGroup(set);
|
|
|
|
if (newGroup != null)
|
|
groups.Insert(i, newGroup);
|
|
|
|
bool hadSelection = selectedGroup == group;
|
|
|
|
if (hadSelection && newGroup == null)
|
|
selectedGroup = null;
|
|
|
|
Filter(null, false);
|
|
|
|
//check if we can/need to maintain our current selection.
|
|
if (hadSelection && newGroup != null)
|
|
{
|
|
var newSelection =
|
|
newGroup.BeatmapPanels.Find(p => p.Beatmap.ID == selectedPanel?.Beatmap.ID) ??
|
|
newGroup.BeatmapPanels[Math.Min(newGroup.BeatmapPanels.Count - 1, group.BeatmapPanels.IndexOf(selectedPanel))];
|
|
|
|
selectGroup(newGroup, newSelection);
|
|
}
|
|
}
|
|
|
|
public void SelectBeatmap(BeatmapInfo beatmap, bool animated = true)
|
|
{
|
|
if (beatmap == null || beatmap.Hidden)
|
|
{
|
|
SelectNext();
|
|
return;
|
|
}
|
|
|
|
if (beatmap == SelectedBeatmap) return;
|
|
|
|
foreach (BeatmapGroup group in groups)
|
|
{
|
|
var panel = group.BeatmapPanels.FirstOrDefault(p => p.Beatmap.Equals(beatmap));
|
|
if (panel != null)
|
|
{
|
|
selectGroup(group, panel, animated);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
public Action<BeatmapInfo> SelectionChanged;
|
|
|
|
public Action StartRequested;
|
|
|
|
public Action<BeatmapSetInfo> DeleteRequested;
|
|
|
|
public Action<BeatmapSetInfo> RestoreRequested;
|
|
|
|
public Action<BeatmapInfo> EditRequested;
|
|
|
|
public Action<BeatmapInfo> HideDifficultyRequested;
|
|
|
|
private void selectNullBeatmap()
|
|
{
|
|
selectedGroup = null;
|
|
selectedPanel = null;
|
|
SelectionChanged?.Invoke(null);
|
|
}
|
|
|
|
public void SelectNext(int direction = 1, bool skipDifficulties = true)
|
|
{
|
|
if (groups.All(g => g.State == BeatmapGroupState.Hidden))
|
|
{
|
|
selectNullBeatmap();
|
|
return;
|
|
}
|
|
|
|
if (!skipDifficulties && selectedGroup != null)
|
|
{
|
|
int i = selectedGroup.BeatmapPanels.IndexOf(selectedPanel) + direction;
|
|
|
|
if (i >= 0 && i < selectedGroup.BeatmapPanels.Count)
|
|
{
|
|
//changing difficulty panel, not set.
|
|
selectGroup(selectedGroup, selectedGroup.BeatmapPanels[i]);
|
|
return;
|
|
}
|
|
}
|
|
|
|
int startIndex = Math.Max(0, groups.IndexOf(selectedGroup));
|
|
int index = startIndex;
|
|
|
|
do
|
|
{
|
|
index = (index + direction + groups.Count) % groups.Count;
|
|
if (groups[index].State != BeatmapGroupState.Hidden)
|
|
{
|
|
if (skipDifficulties)
|
|
SelectBeatmap(groups[index].SelectedPanel != null ? groups[index].SelectedPanel.Beatmap : groups[index].BeatmapPanels.First().Beatmap);
|
|
else
|
|
SelectBeatmap(direction == 1 ? groups[index].BeatmapPanels.First().Beatmap : groups[index].BeatmapPanels.Last().Beatmap);
|
|
|
|
return;
|
|
}
|
|
} while (index != startIndex);
|
|
}
|
|
|
|
private IEnumerable<BeatmapGroup> getVisibleGroups() => groups.Where(selectGroup => selectGroup.State != BeatmapGroupState.Hidden);
|
|
|
|
public void SelectNextRandom()
|
|
{
|
|
if (groups.Count == 0)
|
|
return;
|
|
|
|
var visibleGroups = getVisibleGroups();
|
|
if (!visibleGroups.Any())
|
|
return;
|
|
|
|
if (selectedGroup != null)
|
|
randomSelectedBeatmaps.Push(new KeyValuePair<BeatmapGroup, BeatmapPanel>(selectedGroup, selectedGroup.SelectedPanel));
|
|
|
|
BeatmapGroup group;
|
|
|
|
if (randomType == SelectionRandomType.RandomPermutation)
|
|
{
|
|
var notSeenGroups = visibleGroups.Except(seenGroups);
|
|
if (!notSeenGroups.Any())
|
|
{
|
|
seenGroups.Clear();
|
|
notSeenGroups = visibleGroups;
|
|
}
|
|
|
|
group = notSeenGroups.ElementAt(RNG.Next(notSeenGroups.Count()));
|
|
seenGroups.Add(group);
|
|
}
|
|
else
|
|
group = visibleGroups.ElementAt(RNG.Next(visibleGroups.Count()));
|
|
|
|
BeatmapPanel panel = group.BeatmapPanels[RNG.Next(group.BeatmapPanels.Count)];
|
|
|
|
selectGroup(group, panel);
|
|
}
|
|
|
|
public void SelectPreviousRandom()
|
|
{
|
|
if (!randomSelectedBeatmaps.Any())
|
|
return;
|
|
|
|
var visibleGroups = getVisibleGroups();
|
|
if (!visibleGroups.Any())
|
|
return;
|
|
|
|
while (randomSelectedBeatmaps.Any())
|
|
{
|
|
var beatmapCoordinates = randomSelectedBeatmaps.Pop();
|
|
var group = beatmapCoordinates.Key;
|
|
if (visibleGroups.Contains(group))
|
|
{
|
|
selectGroup(group, beatmapCoordinates.Value);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
private FilterCriteria criteria = new FilterCriteria();
|
|
|
|
private ScheduledDelegate filterTask;
|
|
|
|
public bool AllowSelection = true;
|
|
|
|
public void FlushPendingFilters()
|
|
{
|
|
if (filterTask?.Completed == false)
|
|
Filter(null, false);
|
|
}
|
|
|
|
public void Filter(FilterCriteria newCriteria = null, bool debounce = true)
|
|
{
|
|
if (newCriteria != null)
|
|
criteria = newCriteria;
|
|
|
|
if (!IsLoaded) return;
|
|
|
|
Action perform = delegate
|
|
{
|
|
filterTask = null;
|
|
|
|
criteria.Filter(groups);
|
|
|
|
var filtered = new List<BeatmapGroup>(groups);
|
|
|
|
scrollableContent.Clear(false);
|
|
panels.Clear();
|
|
groups.Clear();
|
|
|
|
foreach (var g in filtered)
|
|
addGroup(g);
|
|
|
|
computeYPositions();
|
|
|
|
if (selectedGroup == null || selectedGroup.State == BeatmapGroupState.Hidden)
|
|
SelectNext();
|
|
else
|
|
selectGroup(selectedGroup, selectedPanel);
|
|
};
|
|
|
|
filterTask?.Cancel();
|
|
filterTask = null;
|
|
|
|
if (debounce)
|
|
filterTask = Scheduler.AddDelayed(perform, 250);
|
|
else
|
|
perform();
|
|
}
|
|
|
|
public void ScrollToSelected(bool animated = true)
|
|
{
|
|
float selectedY = computeYPositions(animated);
|
|
ScrollTo(selectedY, animated);
|
|
}
|
|
|
|
private BeatmapGroup createGroup(BeatmapSetInfo beatmapSet)
|
|
{
|
|
if (beatmapSet.Beatmaps.All(b => b.Hidden))
|
|
return null;
|
|
|
|
foreach (var b in beatmapSet.Beatmaps)
|
|
{
|
|
if (b.Metadata == null)
|
|
b.Metadata = beatmapSet.Metadata;
|
|
}
|
|
|
|
return new BeatmapGroup(beatmapSet, manager)
|
|
{
|
|
SelectionChanged = (g, p) => selectGroup(g, p),
|
|
StartRequested = b => StartRequested?.Invoke(),
|
|
DeleteRequested = b => DeleteRequested?.Invoke(b),
|
|
RestoreHiddenRequested = s => RestoreRequested?.Invoke(s),
|
|
EditRequested = b => EditRequested?.Invoke(b),
|
|
HideDifficultyRequested = b => HideDifficultyRequested?.Invoke(b),
|
|
State = BeatmapGroupState.Collapsed
|
|
};
|
|
}
|
|
|
|
[BackgroundDependencyLoader(permitNulls: true)]
|
|
private void load(BeatmapManager manager, OsuConfigManager config)
|
|
{
|
|
this.manager = manager;
|
|
|
|
randomType = config.GetBindable<SelectionRandomType>(OsuSetting.SelectionRandomType);
|
|
}
|
|
|
|
private void addGroup(BeatmapGroup group)
|
|
{
|
|
groups.Add(group);
|
|
panels.Add(group.Header);
|
|
panels.AddRange(group.BeatmapPanels);
|
|
}
|
|
|
|
private void removeGroup(BeatmapGroup group)
|
|
{
|
|
if (group == null)
|
|
return;
|
|
|
|
if (selectedGroup == group)
|
|
{
|
|
if (getVisibleGroups().Count() == 1)
|
|
selectNullBeatmap();
|
|
else
|
|
SelectNext();
|
|
}
|
|
|
|
groups.Remove(group);
|
|
panels.Remove(group.Header);
|
|
foreach (var p in group.BeatmapPanels)
|
|
panels.Remove(p);
|
|
|
|
scrollableContent.Remove(group.Header);
|
|
scrollableContent.RemoveRange(group.BeatmapPanels);
|
|
|
|
computeYPositions();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Computes the target Y positions for every panel in the carousel.
|
|
/// </summary>
|
|
/// <returns>The Y position of the currently selected panel.</returns>
|
|
private float computeYPositions(bool animated = true)
|
|
{
|
|
yPositions.Clear();
|
|
|
|
float currentY = DrawHeight / 2;
|
|
float selectedY = currentY;
|
|
|
|
foreach (BeatmapGroup group in groups)
|
|
{
|
|
movePanel(group.Header, group.State != BeatmapGroupState.Hidden, animated, ref currentY);
|
|
|
|
if (group.State == BeatmapGroupState.Expanded)
|
|
{
|
|
group.Header.MoveToX(-100, 500, Easing.OutExpo);
|
|
var headerY = group.Header.Position.Y;
|
|
|
|
foreach (BeatmapPanel panel in group.BeatmapPanels)
|
|
{
|
|
if (panel == selectedPanel)
|
|
selectedY = currentY + panel.DrawHeight / 2 - DrawHeight / 2;
|
|
|
|
panel.MoveToX(-50, 500, Easing.OutExpo);
|
|
|
|
//on first display we want to begin hidden under our group's header.
|
|
if (panel.Alpha == 0)
|
|
panel.MoveToY(headerY);
|
|
|
|
movePanel(panel, true, animated, ref currentY);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
group.Header.MoveToX(0, 500, Easing.OutExpo);
|
|
|
|
foreach (BeatmapPanel panel in group.BeatmapPanels)
|
|
{
|
|
panel.MoveToX(0, 500, Easing.OutExpo);
|
|
movePanel(panel, false, animated, ref currentY);
|
|
}
|
|
}
|
|
}
|
|
|
|
currentY += DrawHeight / 2;
|
|
scrollableContent.Height = currentY;
|
|
|
|
return selectedY;
|
|
}
|
|
|
|
private void movePanel(Panel panel, bool advance, bool animated, ref float currentY)
|
|
{
|
|
yPositions.Add(currentY);
|
|
panel.MoveToY(currentY, animated ? 750 : 0, Easing.OutExpo);
|
|
|
|
if (advance)
|
|
currentY += panel.DrawHeight + 5;
|
|
}
|
|
|
|
private void selectGroup(BeatmapGroup group, BeatmapPanel panel = null, bool animated = true)
|
|
{
|
|
try
|
|
{
|
|
if (panel == null)
|
|
panel = group.BeatmapPanels.First();
|
|
|
|
if (selectedPanel == panel) return;
|
|
|
|
Trace.Assert(group.BeatmapPanels.Contains(panel), @"Selected panel must be in provided group");
|
|
|
|
if (selectedGroup != null && selectedGroup != group && selectedGroup.State != BeatmapGroupState.Hidden)
|
|
selectedGroup.State = BeatmapGroupState.Collapsed;
|
|
|
|
group.State = BeatmapGroupState.Expanded;
|
|
group.SelectedPanel = panel;
|
|
|
|
panel.State = PanelSelectedState.Selected;
|
|
|
|
if (selectedPanel == panel) return;
|
|
|
|
selectedPanel = panel;
|
|
selectedGroup = group;
|
|
|
|
SelectionChanged?.Invoke(panel.Beatmap);
|
|
}
|
|
finally
|
|
{
|
|
ScrollToSelected(animated);
|
|
}
|
|
}
|
|
|
|
protected override bool OnKeyDown(InputState state, KeyDownEventArgs args)
|
|
{
|
|
int direction = 0;
|
|
bool skipDifficulties = false;
|
|
|
|
switch (args.Key)
|
|
{
|
|
case Key.Up:
|
|
direction = -1;
|
|
break;
|
|
case Key.Down:
|
|
direction = 1;
|
|
break;
|
|
case Key.Left:
|
|
direction = -1;
|
|
skipDifficulties = true;
|
|
break;
|
|
case Key.Right:
|
|
direction = 1;
|
|
skipDifficulties = true;
|
|
break;
|
|
}
|
|
|
|
if (direction == 0)
|
|
return base.OnKeyDown(state, args);
|
|
|
|
SelectNext(direction, skipDifficulties);
|
|
return true;
|
|
}
|
|
|
|
protected override void Update()
|
|
{
|
|
base.Update();
|
|
|
|
float drawHeight = DrawHeight;
|
|
|
|
// Remove all panels that should no longer be on-screen
|
|
scrollableContent.RemoveAll(delegate(Panel p)
|
|
{
|
|
float panelPosY = p.Position.Y;
|
|
bool remove = panelPosY < Current - p.DrawHeight || panelPosY > Current + drawHeight || !p.IsPresent;
|
|
return remove;
|
|
});
|
|
|
|
// Find index range of all panels that should be on-screen
|
|
Trace.Assert(panels.Count == yPositions.Count);
|
|
|
|
int firstIndex = yPositions.BinarySearch(Current - Panel.MAX_HEIGHT);
|
|
if (firstIndex < 0) firstIndex = ~firstIndex;
|
|
int lastIndex = yPositions.BinarySearch(Current + drawHeight);
|
|
if (lastIndex < 0)
|
|
{
|
|
lastIndex = ~lastIndex;
|
|
|
|
// Add the first panel of the last visible beatmap group to preload its data.
|
|
if (lastIndex != 0 && panels[lastIndex - 1] is BeatmapSetHeader)
|
|
lastIndex++;
|
|
}
|
|
|
|
// Add those panels within the previously found index range that should be displayed.
|
|
for (int i = firstIndex; i < lastIndex; ++i)
|
|
{
|
|
Panel panel = panels[i];
|
|
if (panel.State == PanelSelectedState.Hidden)
|
|
continue;
|
|
|
|
// Only add if we're not already part of the content.
|
|
if (!scrollableContent.Contains(panel))
|
|
{
|
|
// Makes sure headers are always _below_ panels,
|
|
// and depth flows downward.
|
|
panel.Depth = i + (panel is BeatmapSetHeader ? panels.Count : 0);
|
|
scrollableContent.Add(panel);
|
|
}
|
|
}
|
|
|
|
// Update externally controlled state of currently visible panels
|
|
// (e.g. x-offset and opacity).
|
|
float halfHeight = drawHeight / 2;
|
|
foreach (Panel p in scrollableContent.Children)
|
|
updatePanel(p, halfHeight);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Computes the x-offset of currently visible panels. Makes the carousel appear round.
|
|
/// </summary>
|
|
/// <param name="dist">
|
|
/// Vertical distance from the center of the carousel container
|
|
/// ranging from -1 to 1.
|
|
/// </param>
|
|
/// <param name="halfHeight">Half the height of the carousel container.</param>
|
|
private static float offsetX(float dist, float halfHeight)
|
|
{
|
|
// The radius of the circle the carousel moves on.
|
|
const float circle_radius = 3;
|
|
double discriminant = Math.Max(0, circle_radius * circle_radius - dist * dist);
|
|
float x = (circle_radius - (float)Math.Sqrt(discriminant)) * halfHeight;
|
|
|
|
return 125 + x;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update a panel's x position and multiplicative alpha based on its y position and
|
|
/// the current scroll position.
|
|
/// </summary>
|
|
/// <param name="p">The panel to be updated.</param>
|
|
/// <param name="halfHeight">Half the draw height of the carousel container.</param>
|
|
private void updatePanel(Panel p, float halfHeight)
|
|
{
|
|
var height = p.IsPresent ? p.DrawHeight : 0;
|
|
|
|
float panelDrawY = p.Position.Y - Current + height / 2;
|
|
float dist = Math.Abs(1f - panelDrawY / halfHeight);
|
|
|
|
// Setting the origin position serves as an additive position on top of potential
|
|
// local transformation we may want to apply (e.g. when a panel gets selected, we
|
|
// may want to smoothly transform it leftwards.)
|
|
p.OriginPosition = new Vector2(-offsetX(dist, halfHeight), 0);
|
|
|
|
// We are applying a multiplicative alpha (which is internally done by nesting an
|
|
// additional container and setting that container's alpha) such that we can
|
|
// layer transformations on top, with a similar reasoning to the previous comment.
|
|
p.SetMultiplicativeAlpha(MathHelper.Clamp(1.75f - 1.5f * dist, 0, 1));
|
|
}
|
|
}
|
|
}
|