1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-28 04:02:57 +08:00

Fix beatmap updates causing one extra carousel selection

This commit is contained in:
Dean Herbert 2023-12-20 19:09:07 +09:00
parent 64f62e7d90
commit e003462f7d
No known key found for this signature in database
2 changed files with 54 additions and 39 deletions

View File

@ -455,30 +455,13 @@ namespace osu.Game.Screens.Select
private void updateBeatmapSet(BeatmapSetInfo beatmapSet) private void updateBeatmapSet(BeatmapSetInfo beatmapSet)
{ {
Guid? previouslySelectedID = null;
originalBeatmapSetsDetached.RemoveAll(set => set.ID == beatmapSet.ID); originalBeatmapSetsDetached.RemoveAll(set => set.ID == beatmapSet.ID);
originalBeatmapSetsDetached.Add(beatmapSet.Detach()); originalBeatmapSetsDetached.Add(beatmapSet.Detach());
// If the selected beatmap is about to be removed, store its ID so it can be re-selected if required var newSets = new List<CarouselBeatmapSet>();
if (selectedBeatmapSet?.BeatmapSet.ID == beatmapSet.ID)
previouslySelectedID = selectedBeatmap?.BeatmapInfo.ID;
var removedSets = root.RemoveItemsByID(beatmapSet.ID);
foreach (var removedSet in removedSets)
{
// If we don't remove this here, it may remain in a hidden state until scrolled off screen.
// Doesn't really affect anything during actual user interaction, but makes testing annoying.
var removedDrawable = Scroll.FirstOrDefault(c => c.Item == removedSet);
if (removedDrawable != null)
expirePanelImmediately(removedDrawable);
}
if (beatmapsSplitOut) if (beatmapsSplitOut)
{ {
var newSets = new List<CarouselBeatmapSet>();
foreach (var beatmap in beatmapSet.Beatmaps) foreach (var beatmap in beatmapSet.Beatmaps)
{ {
var newSet = createCarouselSet(new BeatmapSetInfo(new[] { beatmap }) var newSet = createCarouselSet(new BeatmapSetInfo(new[] { beatmap })
@ -489,18 +472,7 @@ namespace osu.Game.Screens.Select
}); });
if (newSet != null) if (newSet != null)
{
newSets.Add(newSet); newSets.Add(newSet);
root.AddItem(newSet);
}
}
// check if we can/need to maintain our current selection.
if (previouslySelectedID != null)
{
var toSelect = newSets.FirstOrDefault(s => s.Beatmaps.Any(b => b.BeatmapInfo.ID == previouslySelectedID))
?? newSets.FirstOrDefault();
select(toSelect);
} }
} }
else else
@ -508,13 +480,18 @@ namespace osu.Game.Screens.Select
var newSet = createCarouselSet(beatmapSet); var newSet = createCarouselSet(beatmapSet);
if (newSet != null) if (newSet != null)
{ newSets.Add(newSet);
root.AddItem(newSet); }
// check if we can/need to maintain our current selection. var removedSets = root.ReplaceItem(beatmapSet, newSets);
if (previouslySelectedID != null)
select((CarouselItem?)newSet.Beatmaps.FirstOrDefault(b => b.BeatmapInfo.ID == previouslySelectedID) ?? newSet); // If we don't remove these here, it may remain in a hidden state until scrolled off screen.
} // Doesn't really affect anything during actual user interaction, but makes testing annoying.
foreach (var removedSet in removedSets)
{
var removedDrawable = Scroll.FirstOrDefault(c => c.Item == removedSet);
if (removedDrawable != null)
expirePanelImmediately(removedDrawable);
} }
} }
@ -1207,6 +1184,44 @@ namespace osu.Game.Screens.Select
base.AddItem(i); base.AddItem(i);
} }
/// <summary>
/// A special method to handle replace operations (general for updating a beatmap).
/// Avoids event-driven selection flip-flopping during the remove/add process.
/// </summary>
/// <param name="oldItem">The beatmap set to be replaced.</param>
/// <param name="newItems">All new items to replace the removed beatmap set.</param>
/// <returns>All removed items, for any further processing.</returns>
public IEnumerable<CarouselBeatmapSet> ReplaceItem(BeatmapSetInfo oldItem, List<CarouselBeatmapSet> newItems)
{
// Without doing this, the removal of the old beatmap will cause carousel's eager selection
// logic to invoke, causing one unnecessary selection.
DisableSelection = true;
var removedSets = RemoveItemsByID(oldItem.ID);
DisableSelection = false;
foreach (var set in newItems)
AddItem(set);
Guid? previouslySelectedID = null;
var selectedBeatmap = (LastSelected as CarouselBeatmap)?.BeatmapInfo;
// If the selected beatmap is about to be removed, store its ID so it can be re-selected if required
if (selectedBeatmap?.BeatmapSet?.ID == oldItem.ID)
previouslySelectedID = selectedBeatmap.ID;
// check if we can/need to maintain our current selection.
if (previouslySelectedID != null)
{
var toSelect = newItems.FirstOrDefault(s => s.Beatmaps.Any(b => b.BeatmapInfo.ID == previouslySelectedID))
?? newItems.First();
toSelect.State.Value = CarouselItemState.Selected;
}
return removedSets;
}
public IEnumerable<CarouselBeatmapSet> RemoveItemsByID(Guid beatmapSetID) public IEnumerable<CarouselBeatmapSet> RemoveItemsByID(Guid beatmapSetID)
{ {
if (BeatmapSetsByID.TryGetValue(beatmapSetID, out var carouselBeatmapSets)) if (BeatmapSetsByID.TryGetValue(beatmapSetID, out var carouselBeatmapSets))

View File

@ -36,13 +36,13 @@ namespace osu.Game.Screens.Select.Carousel
/// items have been filtered. This bool will be true during the base <see cref="Filter(FilterCriteria)"/> /// items have been filtered. This bool will be true during the base <see cref="Filter(FilterCriteria)"/>
/// operation. /// operation.
/// </summary> /// </summary>
private bool filteringItems; protected bool DisableSelection;
public override void Filter(FilterCriteria criteria) public override void Filter(FilterCriteria criteria)
{ {
filteringItems = true; DisableSelection = true;
base.Filter(criteria); base.Filter(criteria);
filteringItems = false; DisableSelection = false;
attemptSelection(); attemptSelection();
} }
@ -95,7 +95,7 @@ namespace osu.Game.Screens.Select.Carousel
private void attemptSelection() private void attemptSelection()
{ {
if (filteringItems) return; if (DisableSelection) return;
// we only perform eager selection if we are a currently selected group. // we only perform eager selection if we are a currently selected group.
if (State.Value != CarouselItemState.Selected) return; if (State.Value != CarouselItemState.Selected) return;