// Copyright (c) 2007-2017 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using System; using System.Collections.Generic; using System.Linq; using OpenTK; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Game.Rulesets.Objects; using osu.Game.Rulesets.Objects.Drawables; namespace osu.Game.Rulesets.Mania.Timing.Drawables { /// /// Represents a container in which contains hit objects and moves relative to the current time. /// public abstract class DrawableTimingChange : Container { public readonly TimingChange TimingChange; protected override Container Content => content; private readonly Container content; private readonly Axes scrollingAxes; /// /// Creates a new drawable timing change which contains hit objects and scrolls relative to the current time. /// /// The encapsulated timing change that provides the speed changes. /// The axes through which this timing change scrolls. protected DrawableTimingChange(TimingChange timingChange, Axes scrollingAxes) { this.scrollingAxes = scrollingAxes; TimingChange = timingChange; // We have to proxy the hit objects to an internal container since we're // going to be modifying our height to apply speed changes AddInternal(content = new RelativeCoordinateAutoSizingContainer(scrollingAxes) { RelativeSizeAxes = Axes.Both, RelativePositionAxes = Axes.Both }); } public override Axes RelativeSizeAxes { get { return Axes.Both; } set { throw new InvalidOperationException($"{nameof(DrawableTimingChange)} must always be relatively-sized."); } } protected override void Update() { var parent = Parent as IHasTimeSpan; if (parent == null) return; // Adjust our size to account for the speed changes float speedAdjustedSize = (float)(1000 / TimingChange.BeatLength / TimingChange.SpeedMultiplier); Size = new Vector2((scrollingAxes & Axes.X) > 0 ? speedAdjustedSize : 1, (scrollingAxes & Axes.Y) > 0 ? speedAdjustedSize : 1); RelativeCoordinateSpace = new Vector2((scrollingAxes & Axes.X) > 0 ? (float)parent.TimeSpan.X : 1, (scrollingAxes & Axes.Y) > 0 ? (float)parent.TimeSpan.Y : 1); } public override void Add(DrawableHitObject hitObject) { // The previously relatively-positioned hit object will now become relative to content, but since the hit object has no knowledge of content, // we need to offset it back by content's position (timing change time) so that it becomes correctly relatively-positioned to content // This can be removed if hit objects were stored such that either their StartTime or their "beat offset" was relative to the timing change // they belonged to, but this requires a radical change to the beatmap format which we're not ready to do just yet hitObject.Position = new Vector2((scrollingAxes & Axes.X) > 0 ? hitObject.X - (float)TimingChange.Time : hitObject.X, (scrollingAxes & Axes.Y) > 0 ? hitObject.Y - (float)TimingChange.Time : hitObject.Y); base.Add(hitObject); } /// /// Whether this timing change can contain a hit object. This is true if the hit object occurs "after" after this timing change. /// public bool CanContain(DrawableHitObject hitObject) => TimingChange.Time <= hitObject.HitObject.StartTime; /// /// A container which cann be relatively-sized while auto-sizing to its children on desired axes. The relative coordinate space of /// this container follows its auto-sized height. /// private class RelativeCoordinateAutoSizingContainer : Container { protected override IComparer DepthComparer => new HitObjectReverseStartTimeComparer(); private readonly Axes autoSizingAxes; /// /// The axes which this container should calculate its size from its children on. /// Note that this is not the same as , because that would not allow this container /// to be relatively sized - desired in the case where the playfield re-defines . /// /// public RelativeCoordinateAutoSizingContainer(Axes autoSizingAxes) { this.autoSizingAxes = autoSizingAxes; } public override void InvalidateFromChild(Invalidation invalidation) { // We only want to re-compute our size when a child's size or position has changed if ((invalidation & Invalidation.Geometry) == 0) { base.InvalidateFromChild(invalidation); return; } if (!Children.Any()) return; float height = Children.Select(child => child.Y + child.Height).Max(); float width = Children.Select(child => child.X + child.Width).Max(); Size = new Vector2((autoSizingAxes & Axes.X) > 0 ? width : base.Size.X, (autoSizingAxes & Axes.Y) > 0 ? height : base.Size.Y); RelativeCoordinateSpace = new Vector2((autoSizingAxes & Axes.X) > 0 ? width : 1, (autoSizingAxes & Axes.Y) > 0 ? height : 1); base.InvalidateFromChild(invalidation); } } } }