// 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; using System.Collections.Generic; using osu.Framework.Bindables; using osu.Game.Beatmaps; using osu.Game.Beatmaps.ControlPoints; using osu.Game.Beatmaps.Timing; using osu.Game.Rulesets.Objects; namespace osu.Game.Screens.Edit { public class EditorBeatmap : IEditorBeatmap { /// /// Invoked when a is added to this . /// public event Action HitObjectAdded; /// /// Invoked when a is removed from this . /// public event Action HitObjectRemoved; /// /// Invoked when the start time of a in this was changed. /// public event Action StartTimeChanged; private readonly Dictionary> startTimeBindables = new Dictionary>(); private readonly IBeatmap beatmap; public EditorBeatmap(IBeatmap beatmap) { this.beatmap = beatmap; foreach (var obj in HitObjects) trackStartTime(obj); } public BeatmapInfo BeatmapInfo { get => beatmap.BeatmapInfo; set => beatmap.BeatmapInfo = value; } public BeatmapMetadata Metadata => beatmap.Metadata; public ControlPointInfo ControlPointInfo => beatmap.ControlPointInfo; public List Breaks => beatmap.Breaks; public double TotalBreakTime => beatmap.TotalBreakTime; public IReadOnlyList HitObjects => beatmap.HitObjects; IReadOnlyList IBeatmap.HitObjects => beatmap.HitObjects; public IEnumerable GetStatistics() => beatmap.GetStatistics(); public IBeatmap Clone() => (EditorBeatmap)MemberwiseClone(); private IList mutableHitObjects => (IList)beatmap.HitObjects; /// /// Adds a to this . /// /// The to add. public void Add(HitObject hitObject) { trackStartTime(hitObject); // Preserve existing sorting order in the beatmap var insertionIndex = findInsertionIndex(beatmap.HitObjects, hitObject.StartTime); mutableHitObjects.Insert(insertionIndex + 1, hitObject); HitObjectAdded?.Invoke(hitObject); } /// /// Removes a from this . /// /// The to add. public void Remove(HitObject hitObject) { if (!mutableHitObjects.Contains(hitObject)) return; mutableHitObjects.Remove(hitObject); var bindable = startTimeBindables[hitObject]; bindable.UnbindAll(); startTimeBindables.Remove(hitObject); HitObjectRemoved?.Invoke(hitObject); } private void trackStartTime(HitObject hitObject) { startTimeBindables[hitObject] = hitObject.StartTimeBindable.GetBoundCopy(); startTimeBindables[hitObject].ValueChanged += _ => { // For now we'll remove and re-add the hitobject. This is not optimal and can be improved if required. mutableHitObjects.Remove(hitObject); var insertionIndex = findInsertionIndex(beatmap.HitObjects, hitObject.StartTime); mutableHitObjects.Insert(insertionIndex + 1, hitObject); StartTimeChanged?.Invoke(hitObject); }; } private int findInsertionIndex(IReadOnlyList list, double startTime) { for (int i = 0; i < list.Count; i++) { if (list[i].StartTime > startTime) return i - 1; } return list.Count - 1; } } }