2017-06-09 15:11:31 +08:00
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
2017-06-02 17:20:14 +08:00
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2017-06-09 15:11:31 +08:00
using System.Collections.Generic ;
using System.Linq ;
using osu.Framework.Caching ;
2017-06-09 18:57:03 +08:00
using osu.Framework.Configuration ;
2017-06-01 13:26:21 +08:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2017-06-09 15:11:31 +08:00
using osu.Game.Rulesets.Objects ;
2017-06-01 13:26:21 +08:00
using osu.Game.Rulesets.Objects.Drawables ;
2017-06-02 18:53:30 +08:00
using OpenTK ;
2017-06-01 13:26:21 +08:00
2017-06-09 18:57:03 +08:00
namespace osu.Game.Rulesets.Timing
2017-06-01 13:26:21 +08:00
{
2017-06-02 18:27:00 +08:00
/// <summary>
2017-06-09 15:11:31 +08:00
/// A collection of hit objects which scrolls within a <see cref="SpeedAdjustmentContainer"/>.
///
/// <para>
/// This container handles the conversion between time and position through <see cref="Container{T}.RelativeChildSize"/> and
/// <see cref="Container{T}.RelativeChildOffset"/> such that hit objects added to this container should have time values set as their
/// positions/sizes to make proper use of this container.
/// </para>
///
/// <para>
2017-06-12 14:20:34 +08:00
/// This container will auto-size to the total duration of the contained hit objects along the desired auto-sizing axes such that the resulting size
/// of this container will be a value representing the total duration of all contained hit objects.
2017-06-09 15:11:31 +08:00
/// </para>
///
/// <para>
2017-06-12 14:20:34 +08:00
/// This container is and must always be relatively-sized and positioned to its such that the parent can utilise <see cref="Container{T}.RelativeChildSize"/>
/// and <see cref="Container{T}.RelativeChildOffset"/> to apply further time offsets to this collection of hit objects.
2017-06-09 15:11:31 +08:00
/// </para>
2017-06-02 18:27:00 +08:00
/// </summary>
2017-06-08 22:40:24 +08:00
public abstract class DrawableTimingSection : Container < DrawableHitObject >
2017-06-01 13:26:21 +08:00
{
2017-06-09 15:11:31 +08:00
private readonly BindableDouble visibleTimeRange = new BindableDouble ( ) ;
2017-06-12 14:20:34 +08:00
/// <summary>
/// Gets or sets the range of time that is visible by the length of this container.
/// </summary>
2017-06-09 15:11:31 +08:00
public BindableDouble VisibleTimeRange
{
get { return visibleTimeRange ; }
set { visibleTimeRange . BindTo ( value ) ; }
}
protected override IComparer < Drawable > DepthComparer = > new HitObjectReverseStartTimeComparer ( ) ;
2017-06-01 13:26:21 +08:00
2017-06-09 15:11:31 +08:00
private readonly Axes autoSizingAxes ;
2017-06-01 13:26:21 +08:00
2017-06-09 15:11:31 +08:00
private Cached layout = new Cached ( ) ;
2017-06-02 18:27:00 +08:00
/// <summary>
2017-06-09 00:14:14 +08:00
/// Creates a new <see cref="DrawableTimingSection"/>.
2017-06-02 18:27:00 +08:00
/// </summary>
2017-06-09 15:11:31 +08:00
/// <param name="autoSizingAxes">The axes on which to auto-size to the total size of items in the container.</param>
protected DrawableTimingSection ( Axes autoSizingAxes )
2017-06-01 13:26:21 +08:00
{
2017-06-09 15:11:31 +08:00
this . autoSizingAxes = autoSizingAxes ;
2017-06-01 13:26:21 +08:00
2017-06-12 14:20:34 +08:00
RelativeSizeAxes = Axes . Both ;
2017-06-12 12:09:02 +08:00
RelativePositionAxes = Axes . Both ;
2017-06-02 17:11:36 +08:00
}
2017-06-09 15:11:31 +08:00
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 . RequiredParentSizeToFit ) = = 0 )
{
base . InvalidateFromChild ( invalidation ) ;
2017-06-02 18:27:00 +08:00
return ;
2017-06-09 15:11:31 +08:00
}
2017-06-02 17:11:36 +08:00
2017-06-09 15:11:31 +08:00
layout . Invalidate ( ) ;
2017-06-02 15:39:31 +08:00
2017-06-09 15:11:31 +08:00
base . InvalidateFromChild ( invalidation ) ;
2017-06-01 13:26:21 +08:00
}
2017-06-09 15:11:31 +08:00
protected override void UpdateAfterChildren ( )
{
base . UpdateAfterChildren ( ) ;
2017-06-01 13:26:21 +08:00
2017-06-09 15:11:31 +08:00
if ( ! layout . EnsureValid ( ) )
{
layout . Refresh ( ( ) = >
{
if ( ! Children . Any ( ) )
return ;
//double maxDuration = Children.Select(c => (c.HitObject as IHasEndTime)?.EndTime ?? c.HitObject.StartTime).Max();
//float width = (float)maxDuration - RelativeChildOffset.X;
//float height = (float)maxDuration - RelativeChildOffset.Y;
// Auto-size to the total size of our children
// This ends up being the total duration of our children, however for now this is a more sure-fire way to calculate this
// than the above due to some undesired masking optimisations causing some hit objects to be culled...
// Todo: When this is investigated more we should use the above method as it is a little more exact
2017-06-15 15:15:41 +08:00
// Todo: This is not working correctly in the case that hit objects are absolutely-sized - needs a proper looking into in osu!framework
2017-06-09 15:11:31 +08:00
float width = Children . Select ( child = > child . X + child . Width ) . Max ( ) - RelativeChildOffset . X ;
float height = Children . Select ( child = > child . Y + child . Height ) . Max ( ) - RelativeChildOffset . Y ;
// Consider that width/height are time values. To have ourselves span these time values 1:1, we first need to set our size
Size = new Vector2 ( ( autoSizingAxes & Axes . X ) > 0 ? width : Size . X , ( autoSizingAxes & Axes . Y ) > 0 ? height : Size . Y ) ;
// Then to make our position-space be time values again, we need our relative child size to follow our size
RelativeChildSize = Size ;
} ) ;
}
}
2017-06-01 13:26:21 +08:00
}
2017-06-09 15:11:31 +08:00
}