2021-04-27 14:40:35 +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.
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2024-10-01 19:22:20 +08:00
|
|
|
|
using System.Collections.Specialized;
|
2021-04-27 14:40:35 +08:00
|
|
|
|
using System.Linq;
|
|
|
|
|
using Humanizer;
|
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
|
using osu.Framework.Graphics.UserInterface;
|
2024-07-18 17:20:31 +08:00
|
|
|
|
using osu.Framework.Input.Bindings;
|
2021-04-27 14:40:35 +08:00
|
|
|
|
using osu.Game.Audio;
|
|
|
|
|
using osu.Game.Graphics.UserInterface;
|
|
|
|
|
using osu.Game.Rulesets.Edit;
|
|
|
|
|
using osu.Game.Rulesets.Objects;
|
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Screens.Edit.Compose.Components
|
|
|
|
|
{
|
2021-04-28 12:43:16 +08:00
|
|
|
|
public partial class EditorSelectionHandler : SelectionHandler<HitObject>
|
2021-04-27 14:40:35 +08:00
|
|
|
|
{
|
2023-05-24 16:11:12 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// A special bank name that is only used in the editor UI.
|
|
|
|
|
/// When selected and in placement mode, the bank of the last hit object will always be used.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public const string HIT_BANK_AUTO = "auto";
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
[Resolved]
|
|
|
|
|
protected EditorBeatmap EditorBeatmap { get; private set; } = null!;
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load()
|
|
|
|
|
{
|
|
|
|
|
createStateBindables();
|
|
|
|
|
|
|
|
|
|
// bring in updates from selection changes
|
|
|
|
|
EditorBeatmap.HitObjectUpdated += _ => Scheduler.AddOnce(UpdateTernaryStates);
|
|
|
|
|
|
2024-10-01 19:22:20 +08:00
|
|
|
|
SelectedItems.CollectionChanged += onSelectedItemsChanged;
|
2021-04-27 14:40:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void DeleteItems(IEnumerable<HitObject> items) => EditorBeatmap.RemoveRange(items);
|
|
|
|
|
|
|
|
|
|
#region Selection State
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The state of "new combo" for all selected hitobjects.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly Bindable<TernaryState> SelectionNewComboState = new Bindable<TernaryState>();
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The state of each sample type for all selected hitobjects. Keys match with <see cref="HitSampleInfo"/> constant specifications.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly Dictionary<string, Bindable<TernaryState>> SelectionSampleStates = new Dictionary<string, Bindable<TernaryState>>();
|
|
|
|
|
|
2022-10-19 19:53:18 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The state of each sample bank type for all selected hitobjects.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly Dictionary<string, Bindable<TernaryState>> SelectionBankStates = new Dictionary<string, Bindable<TernaryState>>();
|
|
|
|
|
|
2024-07-15 02:43:16 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The state of each sample addition bank type for all selected hitobjects.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly Dictionary<string, Bindable<TernaryState>> SelectionAdditionBankStates = new Dictionary<string, Bindable<TernaryState>>();
|
|
|
|
|
|
2024-10-24 03:25:37 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether there is no selection and the auto <see cref="SelectionBankStates"/> can be used.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly Bindable<bool> AutoSelectionBankEnabled = new Bindable<bool>();
|
|
|
|
|
|
2024-10-01 19:07:25 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the selection contains any addition samples and the <see cref="SelectionAdditionBankStates"/> can be used.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly Bindable<bool> SelectionAdditionBanksEnabled = new Bindable<bool>();
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Set up ternary state bindables and bind them to selection/hitobject changes (in both directions)
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void createStateBindables()
|
|
|
|
|
{
|
2023-05-24 16:11:12 +08:00
|
|
|
|
foreach (string bankName in HitSampleInfo.AllBanks.Prepend(HIT_BANK_AUTO))
|
2022-10-19 19:53:18 +08:00
|
|
|
|
{
|
|
|
|
|
var bindable = new Bindable<TernaryState>
|
|
|
|
|
{
|
|
|
|
|
Description = bankName.Titleize()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
bindable.ValueChanged += state =>
|
|
|
|
|
{
|
|
|
|
|
switch (state.NewValue)
|
|
|
|
|
{
|
|
|
|
|
case TernaryState.False:
|
2022-10-19 20:35:08 +08:00
|
|
|
|
if (SelectedItems.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
// Ensure that if this is the last selected bank, it should remain selected.
|
|
|
|
|
if (SelectionBankStates.Values.All(b => b.Value == TernaryState.False))
|
|
|
|
|
bindable.Value = TernaryState.True;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2023-05-25 16:57:37 +08:00
|
|
|
|
// Auto should never apply when there is a selection made.
|
|
|
|
|
// This is also required to stop a bindable feedback loop when a HitObject has zero samples (and LINQ `All` below becomes true).
|
|
|
|
|
if (bankName == HIT_BANK_AUTO)
|
|
|
|
|
break;
|
|
|
|
|
|
2022-10-19 20:35:08 +08:00
|
|
|
|
// Never remove a sample bank.
|
|
|
|
|
// These are basically radio buttons, not toggles.
|
2024-07-15 02:43:16 +08:00
|
|
|
|
if (SelectedItems.All(h => h.Samples.Where(o => o.Name == HitSampleInfo.HIT_NORMAL).All(s => s.Bank == bankName)))
|
2022-10-19 20:35:08 +08:00
|
|
|
|
bindable.Value = TernaryState.True;
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-19 19:53:18 +08:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TernaryState.True:
|
2022-10-19 20:35:08 +08:00
|
|
|
|
if (SelectedItems.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
// Ensure the user can't stack multiple bank selections when there's no hitobject selection.
|
|
|
|
|
// Note that in normal scenarios this is sorted out by the feedback from applying the bank to the selected objects.
|
|
|
|
|
foreach (var other in SelectionBankStates.Values)
|
|
|
|
|
{
|
|
|
|
|
if (other != bindable)
|
|
|
|
|
other.Value = TernaryState.False;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2023-05-24 16:11:12 +08:00
|
|
|
|
// Auto should just not apply if there's a selection already made.
|
|
|
|
|
// Maybe we could make it a disabled button in the future, but right now the editor buttons don't support disabled state.
|
|
|
|
|
if (bankName == HIT_BANK_AUTO)
|
|
|
|
|
{
|
|
|
|
|
bindable.Value = TernaryState.False;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-02 22:22:15 +08:00
|
|
|
|
SetSampleBank(bankName);
|
2022-10-19 20:35:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-10-19 19:53:18 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
SelectionBankStates[bankName] = bindable;
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-15 02:43:16 +08:00
|
|
|
|
foreach (string bankName in HitSampleInfo.AllBanks.Prepend(HIT_BANK_AUTO))
|
|
|
|
|
{
|
|
|
|
|
var bindable = new Bindable<TernaryState>
|
|
|
|
|
{
|
|
|
|
|
Description = bankName.Titleize()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
bindable.ValueChanged += state =>
|
|
|
|
|
{
|
|
|
|
|
switch (state.NewValue)
|
|
|
|
|
{
|
|
|
|
|
case TernaryState.False:
|
|
|
|
|
if (SelectedItems.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
// Ensure that if this is the last selected bank, it should remain selected.
|
|
|
|
|
if (SelectionAdditionBankStates.Values.All(b => b.Value == TernaryState.False))
|
|
|
|
|
bindable.Value = TernaryState.True;
|
|
|
|
|
}
|
2024-07-16 17:30:52 +08:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Completely empty selections should be allowed in the case that none of the selected objects have any addition samples.
|
|
|
|
|
// This is also required to stop a bindable feedback loop when a HitObject has zero addition samples (and LINQ `All` below becomes true).
|
|
|
|
|
if (SelectedItems.SelectMany(enumerateAllSamples).All(h => h.All(o => o.Name == HitSampleInfo.HIT_NORMAL)))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
// Never remove a sample bank.
|
|
|
|
|
// These are basically radio buttons, not toggles.
|
2024-08-30 03:54:47 +08:00
|
|
|
|
if (bankName == HIT_BANK_AUTO)
|
|
|
|
|
{
|
|
|
|
|
if (SelectedItems.SelectMany(enumerateAllSamples).All(h => h.Where(o => o.Name != HitSampleInfo.HIT_NORMAL).All(s => s.EditorAutoBank)))
|
|
|
|
|
bindable.Value = TernaryState.True;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (SelectedItems.SelectMany(enumerateAllSamples).All(h => h.Where(o => o.Name != HitSampleInfo.HIT_NORMAL).All(s => s.Bank == bankName && !s.EditorAutoBank)))
|
|
|
|
|
bindable.Value = TernaryState.True;
|
|
|
|
|
}
|
2024-07-16 17:30:52 +08:00
|
|
|
|
}
|
2024-07-15 02:43:16 +08:00
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TernaryState.True:
|
|
|
|
|
if (SelectedItems.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
// Ensure the user can't stack multiple bank selections when there's no hitobject selection.
|
|
|
|
|
// Note that in normal scenarios this is sorted out by the feedback from applying the bank to the selected objects.
|
|
|
|
|
foreach (var other in SelectionAdditionBankStates.Values)
|
|
|
|
|
{
|
|
|
|
|
if (other != bindable)
|
|
|
|
|
other.Value = TernaryState.False;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2024-07-15 04:50:15 +08:00
|
|
|
|
// If none of the selected objects have any addition samples, we should not apply the addition bank.
|
|
|
|
|
if (SelectedItems.SelectMany(enumerateAllSamples).All(h => h.All(o => o.Name == HitSampleInfo.HIT_NORMAL)))
|
2024-07-15 02:43:16 +08:00
|
|
|
|
{
|
|
|
|
|
bindable.Value = TernaryState.False;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SetSampleAdditionBank(bankName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
SelectionAdditionBankStates[bankName] = bindable;
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-01 19:22:20 +08:00
|
|
|
|
resetTernaryStates();
|
2022-10-19 20:35:08 +08:00
|
|
|
|
|
2021-10-27 12:04:41 +08:00
|
|
|
|
foreach (string sampleName in HitSampleInfo.AllAdditions)
|
2021-04-27 14:40:35 +08:00
|
|
|
|
{
|
|
|
|
|
var bindable = new Bindable<TernaryState>
|
|
|
|
|
{
|
|
|
|
|
Description = sampleName.Replace("hit", string.Empty).Titleize()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
bindable.ValueChanged += state =>
|
|
|
|
|
{
|
|
|
|
|
switch (state.NewValue)
|
|
|
|
|
{
|
|
|
|
|
case TernaryState.False:
|
|
|
|
|
RemoveHitSample(sampleName);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TernaryState.True:
|
|
|
|
|
AddHitSample(sampleName);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
SelectionSampleStates[sampleName] = bindable;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// new combo
|
|
|
|
|
SelectionNewComboState.ValueChanged += state =>
|
|
|
|
|
{
|
|
|
|
|
switch (state.NewValue)
|
|
|
|
|
{
|
|
|
|
|
case TernaryState.False:
|
|
|
|
|
SetNewCombo(false);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TernaryState.True:
|
|
|
|
|
SetNewCombo(true);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-01 19:22:20 +08:00
|
|
|
|
private void resetTernaryStates()
|
|
|
|
|
{
|
2024-11-26 15:04:37 +08:00
|
|
|
|
if (SelectedItems.Count > 0)
|
|
|
|
|
return;
|
|
|
|
|
|
2024-11-18 22:06:13 +08:00
|
|
|
|
SelectionNewComboState.Value = TernaryState.False;
|
2024-10-24 03:25:37 +08:00
|
|
|
|
AutoSelectionBankEnabled.Value = true;
|
2024-10-24 19:17:49 +08:00
|
|
|
|
SelectionAdditionBanksEnabled.Value = true;
|
2024-10-01 19:22:20 +08:00
|
|
|
|
SelectionBankStates[HIT_BANK_AUTO].Value = TernaryState.True;
|
|
|
|
|
SelectionAdditionBankStates[HIT_BANK_AUTO].Value = TernaryState.True;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Called when context menu ternary states may need to be recalculated (selection changed or hitobject updated).
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected virtual void UpdateTernaryStates()
|
|
|
|
|
{
|
2024-10-12 05:22:15 +08:00
|
|
|
|
if (SelectedItems.Any())
|
|
|
|
|
SelectionNewComboState.Value = GetStateFromSelection(SelectedItems.OfType<IHasComboInformation>(), h => h.NewCombo);
|
2024-10-24 03:25:37 +08:00
|
|
|
|
AutoSelectionBankEnabled.Value = SelectedItems.Count == 0;
|
2021-04-27 14:40:35 +08:00
|
|
|
|
|
2024-07-02 22:22:15 +08:00
|
|
|
|
var samplesInSelection = SelectedItems.SelectMany(enumerateAllSamples).ToArray();
|
|
|
|
|
|
2021-10-27 12:04:41 +08:00
|
|
|
|
foreach ((string sampleName, var bindable) in SelectionSampleStates)
|
2021-04-27 14:40:35 +08:00
|
|
|
|
{
|
2024-07-02 22:22:15 +08:00
|
|
|
|
bindable.Value = GetStateFromSelection(samplesInSelection, h => h.Any(s => s.Name == sampleName));
|
2021-04-27 14:40:35 +08:00
|
|
|
|
}
|
2022-10-19 19:53:18 +08:00
|
|
|
|
|
|
|
|
|
foreach ((string bankName, var bindable) in SelectionBankStates)
|
|
|
|
|
{
|
2024-07-15 02:43:16 +08:00
|
|
|
|
bindable.Value = GetStateFromSelection(samplesInSelection.SelectMany(s => s).Where(o => o.Name == HitSampleInfo.HIT_NORMAL), h => h.Bank == bankName);
|
2024-07-02 22:22:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-10-01 19:07:25 +08:00
|
|
|
|
SelectionAdditionBanksEnabled.Value = samplesInSelection.SelectMany(s => s).Any(o => o.Name != HitSampleInfo.HIT_NORMAL);
|
|
|
|
|
|
2024-07-15 02:43:16 +08:00
|
|
|
|
foreach ((string bankName, var bindable) in SelectionAdditionBankStates)
|
2024-07-02 22:22:15 +08:00
|
|
|
|
{
|
2024-08-30 03:54:47 +08:00
|
|
|
|
bindable.Value = GetStateFromSelection(samplesInSelection.SelectMany(s => s).Where(o => o.Name != HitSampleInfo.HIT_NORMAL), h => (bankName != HIT_BANK_AUTO && h.Bank == bankName && !h.EditorAutoBank) || (bankName == HIT_BANK_AUTO && h.EditorAutoBank));
|
2024-07-02 22:22:15 +08:00
|
|
|
|
}
|
2024-07-15 04:50:15 +08:00
|
|
|
|
}
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
2024-10-01 19:22:20 +08:00
|
|
|
|
private void onSelectedItemsChanged(object? sender, NotifyCollectionChangedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
// Reset the ternary states when the selection is cleared.
|
|
|
|
|
if (e.OldStartingIndex >= 0 && e.NewStartingIndex < 0)
|
|
|
|
|
Scheduler.AddOnce(resetTernaryStates);
|
|
|
|
|
else
|
|
|
|
|
Scheduler.AddOnce(UpdateTernaryStates);
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-15 04:50:15 +08:00
|
|
|
|
private IEnumerable<IList<HitSampleInfo>> enumerateAllSamples(HitObject hitObject)
|
|
|
|
|
{
|
|
|
|
|
yield return hitObject.Samples;
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
2024-07-15 04:50:15 +08:00
|
|
|
|
if (hitObject is IHasRepeats withRepeats)
|
|
|
|
|
{
|
|
|
|
|
foreach (var node in withRepeats.NodeSamples)
|
|
|
|
|
yield return node;
|
2022-10-19 19:53:18 +08:00
|
|
|
|
}
|
2021-04-27 14:40:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2021-04-28 10:42:10 +08:00
|
|
|
|
#region Ternary state changes
|
2021-04-27 14:40:35 +08:00
|
|
|
|
|
2022-10-19 19:53:18 +08:00
|
|
|
|
/// <summary>
|
2024-07-02 22:22:15 +08:00
|
|
|
|
/// Sets the sample bank for all selected <see cref="HitObject"/>s.
|
2022-10-19 19:53:18 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="bankName">The name of the sample bank.</param>
|
2024-07-02 22:22:15 +08:00
|
|
|
|
public void SetSampleBank(string bankName)
|
2022-10-19 19:53:18 +08:00
|
|
|
|
{
|
2024-07-02 22:22:15 +08:00
|
|
|
|
bool hasRelevantBank(HitObject hitObject)
|
|
|
|
|
{
|
2024-07-15 02:43:16 +08:00
|
|
|
|
bool result = hitObject.Samples.Where(o => o.Name == HitSampleInfo.HIT_NORMAL).All(s => s.Bank == bankName);
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
|
|
|
|
if (hitObject is IHasRepeats hasRepeats)
|
|
|
|
|
{
|
|
|
|
|
foreach (var node in hasRepeats.NodeSamples)
|
2024-07-15 02:43:16 +08:00
|
|
|
|
result &= node.Where(o => o.Name == HitSampleInfo.HIT_NORMAL).All(s => s.Bank == bankName);
|
2024-07-02 22:22:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (SelectedItems.All(hasRelevantBank))
|
Fix performance overhead from ternary state bindable callbacks when selection is changing
Closes https://github.com/ppy/osu/issues/28369.
The reporter of the issue was incorrect; it's not the beat snap grid
that is causing the problem, it's something far stupider than that.
When the current selection changes,
`EditorSelectionHandler.UpdateTernaryStates()` is supposed to update the
state of ternary bindables to reflect the reality of the current
selection. This in turn will fire bindable change callbacks for said
ternary toggles, which heavily use `EditorBeatmap.PerformOnSelection()`.
The thing about that method is that it will attempt to check whether any
changes were actually made to avoid producing empty undo states, *but*
to do this, it must *serialise out the entire beatmap to a stream* and
then *binary equality check that* to determine whether any changes were
actually made:
https://github.com/ppy/osu/blob/7b14c77e43e4ee96775a9fcb6843324170fa70bb/osu.Game/Screens/Edit/EditorChangeHandler.cs#L65-L69
As goes without saying, this is very expensive and unnecessary, which
leads to stuff like keeping a selection box active while a taiko beatmap
is playing under it dog slow. So to attempt to mitigate that, add
precondition checks to every single ternary callback of this sort to
avoid this serialisation overhead.
And yes, those precondition checks use linq, and that is *still* faster
than not having them.
2024-06-04 16:25:08 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2022-10-19 19:53:18 +08:00
|
|
|
|
EditorBeatmap.PerformOnSelection(h =>
|
|
|
|
|
{
|
2024-08-20 18:36:13 +08:00
|
|
|
|
if (hasRelevantBank(h))
|
2022-10-19 19:53:18 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2024-07-15 02:43:16 +08:00
|
|
|
|
h.Samples = h.Samples.Select(s => s.Name == HitSampleInfo.HIT_NORMAL ? s.With(newBank: bankName) : s).ToList();
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
|
|
|
|
if (h is IHasRepeats hasRepeats)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < hasRepeats.NodeSamples.Count; ++i)
|
2024-07-15 02:43:16 +08:00
|
|
|
|
hasRepeats.NodeSamples[i] = hasRepeats.NodeSamples[i].Select(s => s.Name == HitSampleInfo.HIT_NORMAL ? s.With(newBank: bankName) : s).ToList();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EditorBeatmap.Update(h);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Sets the sample addition bank for all selected <see cref="HitObject"/>s.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="bankName">The name of the sample bank.</param>
|
|
|
|
|
public void SetSampleAdditionBank(string bankName)
|
|
|
|
|
{
|
2024-08-30 03:54:47 +08:00
|
|
|
|
bool hasRelevantBank(HitObject hitObject) =>
|
|
|
|
|
bankName == HIT_BANK_AUTO
|
|
|
|
|
? enumerateAllSamples(hitObject).SelectMany(o => o).Where(o => o.Name != HitSampleInfo.HIT_NORMAL).All(s => s.EditorAutoBank)
|
|
|
|
|
: enumerateAllSamples(hitObject).SelectMany(o => o).Where(o => o.Name != HitSampleInfo.HIT_NORMAL).All(s => s.Bank == bankName && !s.EditorAutoBank);
|
2024-07-15 02:43:16 +08:00
|
|
|
|
|
|
|
|
|
if (SelectedItems.All(hasRelevantBank))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
EditorBeatmap.PerformOnSelection(h =>
|
|
|
|
|
{
|
2024-08-30 03:54:47 +08:00
|
|
|
|
if (hasRelevantBank(h))
|
2024-07-15 02:43:16 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2024-08-30 03:54:47 +08:00
|
|
|
|
string normalBank = h.Samples.FirstOrDefault(s => s.Name == HitSampleInfo.HIT_NORMAL)?.Bank ?? HitSampleInfo.BANK_SOFT;
|
|
|
|
|
h.Samples = h.Samples.Select(s => s.Name != HitSampleInfo.HIT_NORMAL ? bankName == HIT_BANK_AUTO ? s.With(newBank: normalBank, newEditorAutoBank: true) : s.With(newBank: bankName, newEditorAutoBank: false) : s).ToList();
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
|
|
|
|
if (h is IHasRepeats hasRepeats)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < hasRepeats.NodeSamples.Count; ++i)
|
2024-08-30 03:54:47 +08:00
|
|
|
|
{
|
|
|
|
|
normalBank = hasRepeats.NodeSamples[i].FirstOrDefault(s => s.Name == HitSampleInfo.HIT_NORMAL)?.Bank ?? HitSampleInfo.BANK_SOFT;
|
|
|
|
|
hasRepeats.NodeSamples[i] = hasRepeats.NodeSamples[i].Select(s => s.Name != HitSampleInfo.HIT_NORMAL ? bankName == HIT_BANK_AUTO ? s.With(newBank: normalBank, newEditorAutoBank: true) : s.With(newBank: bankName, newEditorAutoBank: false) : s).ToList();
|
|
|
|
|
}
|
2024-07-02 22:22:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-10-19 19:53:18 +08:00
|
|
|
|
EditorBeatmap.Update(h);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-02 22:22:15 +08:00
|
|
|
|
private bool hasRelevantSample(HitObject hitObject, string sampleName)
|
|
|
|
|
{
|
|
|
|
|
bool result = hitObject.Samples.Any(s => s.Name == sampleName);
|
|
|
|
|
|
|
|
|
|
if (hitObject is IHasRepeats hasRepeats)
|
|
|
|
|
{
|
|
|
|
|
foreach (var node in hasRepeats.NodeSamples)
|
|
|
|
|
result &= node.Any(s => s.Name == sampleName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Adds a hit sample to all selected <see cref="HitObject"/>s.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="sampleName">The name of the hit sample.</param>
|
|
|
|
|
public void AddHitSample(string sampleName)
|
|
|
|
|
{
|
2024-07-02 22:22:15 +08:00
|
|
|
|
if (SelectedItems.All(h => hasRelevantSample(h, sampleName)))
|
Fix performance overhead from ternary state bindable callbacks when selection is changing
Closes https://github.com/ppy/osu/issues/28369.
The reporter of the issue was incorrect; it's not the beat snap grid
that is causing the problem, it's something far stupider than that.
When the current selection changes,
`EditorSelectionHandler.UpdateTernaryStates()` is supposed to update the
state of ternary bindables to reflect the reality of the current
selection. This in turn will fire bindable change callbacks for said
ternary toggles, which heavily use `EditorBeatmap.PerformOnSelection()`.
The thing about that method is that it will attempt to check whether any
changes were actually made to avoid producing empty undo states, *but*
to do this, it must *serialise out the entire beatmap to a stream* and
then *binary equality check that* to determine whether any changes were
actually made:
https://github.com/ppy/osu/blob/7b14c77e43e4ee96775a9fcb6843324170fa70bb/osu.Game/Screens/Edit/EditorChangeHandler.cs#L65-L69
As goes without saying, this is very expensive and unnecessary, which
leads to stuff like keeping a selection box active while a taiko beatmap
is playing under it dog slow. So to attempt to mitigate that, add
precondition checks to every single ternary callback of this sort to
avoid this serialisation overhead.
And yes, those precondition checks use linq, and that is *still* faster
than not having them.
2024-06-04 16:25:08 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
EditorBeatmap.PerformOnSelection(h =>
|
|
|
|
|
{
|
|
|
|
|
// Make sure there isn't already an existing sample
|
2024-08-20 18:36:13 +08:00
|
|
|
|
if (h.Samples.All(s => s.Name != sampleName))
|
|
|
|
|
h.Samples.Add(h.CreateHitSampleInfo(sampleName));
|
2023-05-30 13:04:02 +08:00
|
|
|
|
|
2024-07-02 22:22:15 +08:00
|
|
|
|
if (h is IHasRepeats hasRepeats)
|
|
|
|
|
{
|
|
|
|
|
foreach (var node in hasRepeats.NodeSamples)
|
|
|
|
|
{
|
|
|
|
|
if (node.Any(s => s.Name == sampleName))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
var hitSample = h.CreateHitSampleInfo(sampleName);
|
|
|
|
|
|
2024-08-30 03:54:47 +08:00
|
|
|
|
HitSampleInfo? existingAddition = node.FirstOrDefault(s => s.Name != HitSampleInfo.HIT_NORMAL);
|
|
|
|
|
if (existingAddition != null)
|
|
|
|
|
hitSample = hitSample.With(newBank: existingAddition.Bank, newEditorAutoBank: existingAddition.EditorAutoBank);
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
|
|
|
|
node.Add(hitSample);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-23 20:19:38 +08:00
|
|
|
|
EditorBeatmap.Update(h);
|
2021-04-27 14:40:35 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 10:42:10 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Removes a hit sample from all selected <see cref="HitObject"/>s.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="sampleName">The name of the hit sample.</param>
|
|
|
|
|
public void RemoveHitSample(string sampleName)
|
|
|
|
|
{
|
2024-07-02 22:22:15 +08:00
|
|
|
|
if (SelectedItems.All(h => !hasRelevantSample(h, sampleName)))
|
Fix performance overhead from ternary state bindable callbacks when selection is changing
Closes https://github.com/ppy/osu/issues/28369.
The reporter of the issue was incorrect; it's not the beat snap grid
that is causing the problem, it's something far stupider than that.
When the current selection changes,
`EditorSelectionHandler.UpdateTernaryStates()` is supposed to update the
state of ternary bindables to reflect the reality of the current
selection. This in turn will fire bindable change callbacks for said
ternary toggles, which heavily use `EditorBeatmap.PerformOnSelection()`.
The thing about that method is that it will attempt to check whether any
changes were actually made to avoid producing empty undo states, *but*
to do this, it must *serialise out the entire beatmap to a stream* and
then *binary equality check that* to determine whether any changes were
actually made:
https://github.com/ppy/osu/blob/7b14c77e43e4ee96775a9fcb6843324170fa70bb/osu.Game/Screens/Edit/EditorChangeHandler.cs#L65-L69
As goes without saying, this is very expensive and unnecessary, which
leads to stuff like keeping a selection box active while a taiko beatmap
is playing under it dog slow. So to attempt to mitigate that, add
precondition checks to every single ternary callback of this sort to
avoid this serialisation overhead.
And yes, those precondition checks use linq, and that is *still* faster
than not having them.
2024-06-04 16:25:08 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2021-05-23 20:19:38 +08:00
|
|
|
|
EditorBeatmap.PerformOnSelection(h =>
|
|
|
|
|
{
|
|
|
|
|
h.SamplesBindable.RemoveAll(s => s.Name == sampleName);
|
2024-07-02 22:22:15 +08:00
|
|
|
|
|
|
|
|
|
if (h is IHasRepeats hasRepeats)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < hasRepeats.NodeSamples.Count; ++i)
|
|
|
|
|
hasRepeats.NodeSamples[i] = hasRepeats.NodeSamples[i].Where(s => s.Name != sampleName).ToList();
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-23 20:19:38 +08:00
|
|
|
|
EditorBeatmap.Update(h);
|
|
|
|
|
});
|
2021-04-28 10:42:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Set the new combo state of all selected <see cref="HitObject"/>s.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="state">Whether to set or unset.</param>
|
|
|
|
|
/// <exception cref="InvalidOperationException">Throws if any selected object doesn't implement <see cref="IHasComboInformation"/></exception>
|
|
|
|
|
public void SetNewCombo(bool state)
|
|
|
|
|
{
|
Fix performance overhead from ternary state bindable callbacks when selection is changing
Closes https://github.com/ppy/osu/issues/28369.
The reporter of the issue was incorrect; it's not the beat snap grid
that is causing the problem, it's something far stupider than that.
When the current selection changes,
`EditorSelectionHandler.UpdateTernaryStates()` is supposed to update the
state of ternary bindables to reflect the reality of the current
selection. This in turn will fire bindable change callbacks for said
ternary toggles, which heavily use `EditorBeatmap.PerformOnSelection()`.
The thing about that method is that it will attempt to check whether any
changes were actually made to avoid producing empty undo states, *but*
to do this, it must *serialise out the entire beatmap to a stream* and
then *binary equality check that* to determine whether any changes were
actually made:
https://github.com/ppy/osu/blob/7b14c77e43e4ee96775a9fcb6843324170fa70bb/osu.Game/Screens/Edit/EditorChangeHandler.cs#L65-L69
As goes without saying, this is very expensive and unnecessary, which
leads to stuff like keeping a selection box active while a taiko beatmap
is playing under it dog slow. So to attempt to mitigate that, add
precondition checks to every single ternary callback of this sort to
avoid this serialisation overhead.
And yes, those precondition checks use linq, and that is *still* faster
than not having them.
2024-06-04 16:25:08 +08:00
|
|
|
|
if (SelectedItems.OfType<IHasComboInformation>().All(h => h.NewCombo == state))
|
|
|
|
|
return;
|
|
|
|
|
|
2021-04-27 14:40:35 +08:00
|
|
|
|
EditorBeatmap.PerformOnSelection(h =>
|
|
|
|
|
{
|
|
|
|
|
var comboInfo = h as IHasComboInformation;
|
|
|
|
|
|
|
|
|
|
if (comboInfo == null || comboInfo.NewCombo == state) return;
|
|
|
|
|
|
|
|
|
|
comboInfo.NewCombo = state;
|
|
|
|
|
EditorBeatmap.Update(h);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region Context Menu
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Provide context menu items relevant to current selection. Calling base is not required.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="selection">The current selection.</param>
|
|
|
|
|
/// <returns>The relevant menu items.</returns>
|
2021-04-28 12:43:16 +08:00
|
|
|
|
protected override IEnumerable<MenuItem> GetContextMenuItemsForSelection(IEnumerable<SelectionBlueprint<HitObject>> selection)
|
|
|
|
|
{
|
|
|
|
|
if (SelectedBlueprints.All(b => b.Item is IHasComboInformation))
|
|
|
|
|
{
|
2024-07-18 17:20:31 +08:00
|
|
|
|
yield return new TernaryStateToggleMenuItem("New combo")
|
|
|
|
|
{
|
|
|
|
|
State = { BindTarget = SelectionNewComboState },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.Q))
|
|
|
|
|
};
|
2021-04-28 12:43:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-07-18 17:20:31 +08:00
|
|
|
|
yield return new OsuMenuItem("Sample") { Items = getSampleSubmenuItems().ToArray(), };
|
|
|
|
|
yield return new OsuMenuItem("Bank") { Items = getBankSubmenuItems().ToArray(), };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private IEnumerable<MenuItem> getSampleSubmenuItems()
|
|
|
|
|
{
|
|
|
|
|
var whistle = SelectionSampleStates[HitSampleInfo.HIT_WHISTLE];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(whistle.Description)
|
|
|
|
|
{
|
|
|
|
|
State = { BindTarget = whistle },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.W))
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var finish = SelectionSampleStates[HitSampleInfo.HIT_FINISH];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(finish.Description)
|
|
|
|
|
{
|
|
|
|
|
State = { BindTarget = finish },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.E))
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var clap = SelectionSampleStates[HitSampleInfo.HIT_CLAP];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(clap.Description)
|
|
|
|
|
{
|
|
|
|
|
State = { BindTarget = clap },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.R))
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private IEnumerable<MenuItem> getBankSubmenuItems()
|
|
|
|
|
{
|
|
|
|
|
var auto = SelectionBankStates[HIT_BANK_AUTO];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(auto.Description)
|
|
|
|
|
{
|
|
|
|
|
State = { BindTarget = auto },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.Shift, InputKey.Q))
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var normal = SelectionBankStates[HitSampleInfo.BANK_NORMAL];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(normal.Description)
|
|
|
|
|
{
|
|
|
|
|
State = { BindTarget = normal },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.Shift, InputKey.W))
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var soft = SelectionBankStates[HitSampleInfo.BANK_SOFT];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(soft.Description)
|
2021-04-28 12:43:16 +08:00
|
|
|
|
{
|
2024-07-18 17:20:31 +08:00
|
|
|
|
State = { BindTarget = soft },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.Shift, InputKey.E))
|
2021-04-28 12:43:16 +08:00
|
|
|
|
};
|
2022-10-19 19:53:18 +08:00
|
|
|
|
|
2024-07-18 17:20:31 +08:00
|
|
|
|
var drum = SelectionBankStates[HitSampleInfo.BANK_DRUM];
|
|
|
|
|
yield return new TernaryStateToggleMenuItem(drum.Description)
|
2022-10-19 19:53:18 +08:00
|
|
|
|
{
|
2024-07-18 17:20:31 +08:00
|
|
|
|
State = { BindTarget = drum },
|
|
|
|
|
Hotkey = new Hotkey(new KeyCombination(InputKey.Shift, InputKey.R))
|
2022-10-19 19:53:18 +08:00
|
|
|
|
};
|
2024-07-15 02:43:16 +08:00
|
|
|
|
|
|
|
|
|
yield return new OsuMenuItem("Addition bank")
|
|
|
|
|
{
|
|
|
|
|
Items = SelectionAdditionBankStates.Select(kvp =>
|
|
|
|
|
new TernaryStateToggleMenuItem(kvp.Value.Description) { State = { BindTarget = kvp.Value } }).ToArray()
|
|
|
|
|
};
|
2021-04-28 12:43:16 +08:00
|
|
|
|
}
|
2021-04-27 14:40:35 +08:00
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
}
|
|
|
|
|
}
|