// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Collections.Generic; using System.Linq; using Humanizer; using osu.Framework.Allocation; using osu.Framework.Bindables; using osu.Framework.Extensions; using osu.Framework.Extensions.IEnumerableExtensions; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Cursor; using osu.Framework.Graphics.UserInterface; using osu.Framework.Input.Events; using osu.Game.Audio; using osu.Game.Graphics; using osu.Game.Graphics.UserInterface; using osu.Game.Graphics.UserInterfaceV2; using osu.Game.Rulesets.Objects; using osu.Game.Screens.Edit.Components.TernaryButtons; using osu.Game.Screens.Edit.Timing; using osuTK; using osuTK.Graphics; using osuTK.Input; namespace osu.Game.Screens.Edit.Compose.Components.Timeline { public partial class SamplePointPiece : HitObjectPointPiece, IHasPopover { public readonly HitObject HitObject; public SamplePointPiece(HitObject hitObject) { HitObject = hitObject; } public bool AlternativeColor { get; init; } protected override Color4 GetRepresentingColour(OsuColour colours) => AlternativeColor ? colours.Purple : colours.Pink; [BackgroundDependencyLoader] private void load() { HitObject.DefaultsApplied += _ => updateText(); updateText(); } protected override bool OnClick(ClickEvent e) { this.ShowPopover(); return true; } private void updateText() { Label.Text = $"{abbreviateBank(GetBankValue(GetSamples()))} {GetVolumeValue(GetSamples())}"; } private static string? abbreviateBank(string? bank) { return bank switch { "normal" => "N", "soft" => "S", "drum" => "D", _ => bank }; } public static string? GetBankValue(IEnumerable samples) { return samples.FirstOrDefault(o => o.Name == HitSampleInfo.HIT_NORMAL)?.Bank; } public static string? GetAdditionBankValue(IEnumerable samples) { return samples.FirstOrDefault(o => o.Name != HitSampleInfo.HIT_NORMAL)?.Bank ?? GetBankValue(samples); } public static int GetVolumeValue(ICollection samples) { return samples.Count == 0 ? 0 : samples.Max(o => o.Volume); } protected virtual IList GetSamples() => HitObject.Samples; public virtual Popover GetPopover() => new SampleEditPopover(HitObject); public partial class SampleEditPopover : OsuPopover { private readonly HitObject hitObject; private LabelledTextBox bank = null!; private LabelledTextBox additionBank = null!; private IndeterminateSliderWithTextBoxInput volume = null!; private FillFlowContainer togglesCollection = null!; protected virtual IList GetSamples(HitObject ho) => ho.Samples; [Resolved(canBeNull: true)] private EditorBeatmap beatmap { get; set; } = null!; public SampleEditPopover(HitObject hitObject) { this.hitObject = hitObject; } [BackgroundDependencyLoader] private void load() { FillFlowContainer flow; Children = new Drawable[] { flow = new FillFlowContainer { Width = 200, Direction = FillDirection.Vertical, AutoSizeAxes = Axes.Y, Spacing = new Vector2(0, 10), Children = new Drawable[] { togglesCollection = new FillFlowContainer { RelativeSizeAxes = Axes.X, AutoSizeAxes = Axes.Y, Direction = FillDirection.Horizontal, Spacing = new Vector2(5, 5), }, bank = new LabelledTextBox { Label = "Bank Name", }, additionBank = new LabelledTextBox { Label = "Addition Bank", }, volume = new IndeterminateSliderWithTextBoxInput("Volume", new BindableInt(100) { MinValue = 0, MaxValue = 100, }) } } }; bank.TabbableContentContainer = flow; additionBank.TabbableContentContainer = flow; volume.TabbableContentContainer = flow; // if the piece belongs to a currently selected object, assume that the user wants to change all selected objects. // if the piece belongs to an unselected object, operate on that object alone, independently of the selection. var relevantObjects = (beatmap.SelectedHitObjects.Contains(hitObject) ? beatmap.SelectedHitObjects : hitObject.Yield()).ToArray(); var relevantSamples = relevantObjects.Select(GetSamples).ToArray(); // even if there are multiple objects selected, we can still display sample volume or bank if they all have the same value. string? commonBank = getCommonBank(relevantSamples); if (!string.IsNullOrEmpty(commonBank)) bank.Current.Value = commonBank; string? commonAdditionBank = getCommonAdditionBank(relevantSamples); if (!string.IsNullOrEmpty(commonAdditionBank)) additionBank.Current.Value = commonAdditionBank; int? commonVolume = getCommonVolume(relevantSamples); if (commonVolume != null) volume.Current.Value = commonVolume.Value; updateBankPlaceholderText(relevantObjects); bank.Current.BindValueChanged(val => { updateBankFor(relevantObjects, val.NewValue); updateBankPlaceholderText(relevantObjects); }); // on commit, ensure that the value is correct by sourcing it from the objects' samples again. // this ensures that committing empty text causes a revert to the previous value. bank.OnCommit += (_, _) => bank.Current.Value = getCommonBank(relevantSamples); updateAdditionBankPlaceholderText(relevantObjects); additionBank.Current.BindValueChanged(val => { updateAdditionBankFor(relevantObjects, val.NewValue); updateAdditionBankPlaceholderText(relevantObjects); }); additionBank.OnCommit += (_, _) => additionBank.Current.Value = getCommonAdditionBank(relevantSamples); volume.Current.BindValueChanged(val => updateVolumeFor(relevantObjects, val.NewValue)); createStateBindables(relevantObjects); updateTernaryStates(relevantObjects); togglesCollection.AddRange(createTernaryButtons().Select(b => new DrawableTernaryButton(b) { RelativeSizeAxes = Axes.None, Size = new Vector2(40, 40) })); } protected override void LoadComplete() { base.LoadComplete(); ScheduleAfterChildren(() => GetContainingInputManager().ChangeFocus(volume)); } private static string? getCommonBank(IList[] relevantSamples) => relevantSamples.Select(GetBankValue).Distinct().Count() == 1 ? GetBankValue(relevantSamples.First()) : null; private static string? getCommonAdditionBank(IList[] relevantSamples) => relevantSamples.Select(GetAdditionBankValue).Distinct().Count() == 1 ? GetAdditionBankValue(relevantSamples.First()) : null; private static int? getCommonVolume(IList[] relevantSamples) => relevantSamples.Select(GetVolumeValue).Distinct().Count() == 1 ? GetVolumeValue(relevantSamples.First()) : null; private void updateFor(IEnumerable objects, Action> updateAction) { beatmap.BeginChange(); foreach (var h in objects) { var samples = GetSamples(h); updateAction(h, samples); beatmap.Update(h); } beatmap.EndChange(); } private void updateBankFor(IEnumerable objects, string? newBank) { if (string.IsNullOrEmpty(newBank)) return; updateFor(objects, (_, samples) => { for (int i = 0; i < samples.Count; i++) { if (samples[i].Name != HitSampleInfo.HIT_NORMAL) continue; samples[i] = samples[i].With(newBank: newBank); } }); } private void updateAdditionBankFor(IEnumerable objects, string? newBank) { if (string.IsNullOrEmpty(newBank)) return; updateFor(objects, (_, samples) => { for (int i = 0; i < samples.Count; i++) { if (samples[i].Name == HitSampleInfo.HIT_NORMAL) continue; samples[i] = samples[i].With(newBank: newBank); } }); } private void updateBankPlaceholderText(IEnumerable objects) { string? commonBank = getCommonBank(objects.Select(GetSamples).ToArray()); bank.PlaceholderText = string.IsNullOrEmpty(commonBank) ? "(multiple)" : string.Empty; } private void updateAdditionBankPlaceholderText(IEnumerable objects) { string? commonAdditionBank = getCommonAdditionBank(objects.Select(GetSamples).ToArray()); additionBank.PlaceholderText = string.IsNullOrEmpty(commonAdditionBank) ? "(multiple)" : string.Empty; } private void updateVolumeFor(IEnumerable objects, int? newVolume) { if (newVolume == null) return; updateFor(objects, (_, samples) => { for (int i = 0; i < samples.Count; i++) { samples[i] = samples[i].With(newVolume: newVolume.Value); } }); } #region hitsound toggles private readonly Dictionary> selectionSampleStates = new Dictionary>(); private void createStateBindables(IEnumerable objects) { foreach (string sampleName in HitSampleInfo.AllAdditions) { var bindable = new Bindable { Description = sampleName.Replace("hit", string.Empty).Titleize() }; bindable.ValueChanged += state => { switch (state.NewValue) { case TernaryState.False: removeHitSampleFor(objects, sampleName); break; case TernaryState.True: addHitSampleFor(objects, sampleName); break; } }; selectionSampleStates[sampleName] = bindable; } } private void updateTernaryStates(IEnumerable objects) { foreach ((string sampleName, var bindable) in selectionSampleStates) { bindable.Value = SelectionHandler.GetStateFromSelection(objects, h => GetSamples(h).Any(s => s.Name == sampleName)); } } private IEnumerable createTernaryButtons() { foreach ((string sampleName, var bindable) in selectionSampleStates) yield return new TernaryButton(bindable, string.Empty, () => ComposeBlueprintContainer.GetIconForSample(sampleName)); } private void addHitSampleFor(IEnumerable objects, string sampleName) { if (string.IsNullOrEmpty(sampleName)) return; updateFor(objects, (h, samples) => { // Make sure there isn't already an existing sample if (samples.Any(s => s.Name == sampleName)) return; samples.Add(h.CreateHitSampleInfo(sampleName)); }); } private void removeHitSampleFor(IEnumerable objects, string sampleName) { if (string.IsNullOrEmpty(sampleName)) return; updateFor(objects, (_, samples) => { for (int i = 0; i < samples.Count; i++) { if (samples[i].Name == sampleName) samples.RemoveAt(i--); } }); } protected override bool OnKeyDown(KeyDownEvent e) { if (e.ControlPressed || e.AltPressed || e.SuperPressed || e.ShiftPressed || !checkRightToggleFromKey(e.Key, out int rightIndex)) return base.OnKeyDown(e); var item = togglesCollection.ElementAtOrDefault(rightIndex); if (item is not DrawableTernaryButton button) return base.OnKeyDown(e); button.Button.Toggle(); return true; } private bool checkRightToggleFromKey(Key key, out int index) { switch (key) { case Key.W: index = 0; break; case Key.E: index = 1; break; case Key.R: index = 2; break; default: index = -1; break; } return index >= 0; } #endregion } } }