2019-01-24 08:43:03 +00: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.
2018-04-13 09:19:50 +00:00
2020-11-10 13:49:02 +00:00
using System ;
2018-04-13 09:19:50 +00:00
using System.Collections.Generic ;
2020-11-10 13:49:02 +00:00
using System.Diagnostics ;
2018-04-13 09:19:50 +00:00
using System.Linq ;
2020-11-10 13:49:02 +00:00
using osu.Framework.Allocation ;
2019-12-17 17:56:29 +00:00
using osu.Framework.Bindables ;
2020-11-12 05:54:33 +00:00
using osu.Framework.Extensions.TypeExtensions ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2020-10-29 06:20:10 +00:00
using osu.Framework.Graphics.Performance ;
2020-11-10 13:49:02 +00:00
using osu.Game.Rulesets.Judgements ;
using osu.Game.Rulesets.Objects ;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets.Objects.Drawables ;
namespace osu.Game.Rulesets.UI
{
2018-12-13 05:55:28 +00:00
public class HitObjectContainer : LifetimeManagementContainer
2018-04-13 09:19:50 +00:00
{
2020-11-10 13:49:02 +00:00
/// <summary>
/// All currently in-use <see cref="DrawableHitObject"/>s.
/// </summary>
2020-11-11 10:27:07 +00:00
public IEnumerable < DrawableHitObject > Objects = > InternalChildren . Cast < DrawableHitObject > ( ) . OrderBy ( h = > h . HitObject . StartTime ) ;
2020-11-10 13:49:02 +00:00
/// <summary>
/// All currently in-use <see cref="DrawableHitObject"/>s that are alive.
/// </summary>
/// <remarks>
/// If this <see cref="HitObjectContainer"/> uses pooled objects, this is equivalent to <see cref="Objects"/>.
/// </remarks>
2020-11-11 10:27:07 +00:00
public IEnumerable < DrawableHitObject > AliveObjects = > AliveInternalChildren . Cast < DrawableHitObject > ( ) . OrderBy ( h = > h . HitObject . StartTime ) ;
2020-11-10 13:49:02 +00:00
2020-11-10 14:32:30 +00:00
/// <summary>
/// Invoked when a <see cref="DrawableHitObject"/> is judged.
/// </summary>
2020-11-10 13:49:02 +00:00
public event Action < DrawableHitObject , JudgementResult > NewResult ;
2020-11-10 14:32:30 +00:00
/// <summary>
/// Invoked when a <see cref="DrawableHitObject"/> judgement is reverted.
/// </summary>
2020-11-10 13:49:02 +00:00
public event Action < DrawableHitObject , JudgementResult > RevertResult ;
2020-11-12 09:32:20 +00:00
/// <summary>
/// Invoked when a <see cref="HitObject"/> becomes used by a <see cref="DrawableHitObject"/>.
/// </summary>
/// <remarks>
/// If this <see cref="HitObjectContainer"/> uses pooled objects, this represents the time when the <see cref="HitObject"/>s become alive.
/// </remarks>
2020-11-26 05:01:46 +00:00
internal event Action < HitObject > HitObjectUsageBegan ;
2020-11-12 09:32:20 +00:00
/// <summary>
/// Invoked when a <see cref="HitObject"/> becomes unused by a <see cref="DrawableHitObject"/>.
/// </summary>
/// <remarks>
/// If this <see cref="HitObjectContainer"/> uses pooled objects, this represents the time when the <see cref="HitObject"/>s become dead.
/// </remarks>
2020-11-26 05:01:46 +00:00
internal event Action < HitObject > HitObjectUsageFinished ;
2020-11-12 09:32:20 +00:00
2020-11-12 09:34:50 +00:00
/// <summary>
/// The amount of time prior to the current time within which <see cref="HitObject"/>s should be considered alive.
/// </summary>
2020-11-13 09:54:49 +00:00
internal double PastLifetimeExtension { get ; set ; }
2020-11-12 09:34:50 +00:00
/// <summary>
/// The amount of time after the current time within which <see cref="HitObject"/>s should be considered alive.
/// </summary>
2020-11-13 09:54:49 +00:00
internal double FutureLifetimeExtension { get ; set ; }
2020-11-12 09:34:50 +00:00
2020-11-10 13:49:02 +00:00
private readonly Dictionary < DrawableHitObject , IBindable > startTimeMap = new Dictionary < DrawableHitObject , IBindable > ( ) ;
private readonly Dictionary < HitObjectLifetimeEntry , DrawableHitObject > drawableMap = new Dictionary < HitObjectLifetimeEntry , DrawableHitObject > ( ) ;
private readonly LifetimeEntryManager lifetimeManager = new LifetimeEntryManager ( ) ;
[Resolved(CanBeNull = true)]
2020-11-13 15:54:57 +00:00
private IPooledHitObjectProvider pooledObjectProvider { get ; set ; }
2019-12-17 17:56:29 +00:00
2018-09-21 05:35:50 +00:00
public HitObjectContainer ( )
{
RelativeSizeAxes = Axes . Both ;
2020-11-10 13:49:02 +00:00
lifetimeManager . EntryBecameAlive + = entryBecameAlive ;
lifetimeManager . EntryBecameDead + = entryBecameDead ;
2018-09-21 05:35:50 +00:00
}
2020-11-12 08:07:20 +00:00
protected override void LoadAsyncComplete ( )
2020-11-12 07:58:40 +00:00
{
2020-11-12 08:07:20 +00:00
base . LoadAsyncComplete ( ) ;
2020-11-12 07:58:40 +00:00
2020-11-12 08:07:20 +00:00
// Application of hitobjects during load() may have changed their start times, so ensure the correct sorting order.
2020-11-12 07:58:40 +00:00
SortInternal ( ) ;
}
2020-11-10 13:49:02 +00:00
#region Pooling support
public void Add ( HitObjectLifetimeEntry entry ) = > lifetimeManager . AddEntry ( entry ) ;
2020-11-10 14:32:30 +00:00
public bool Remove ( HitObjectLifetimeEntry entry ) = > lifetimeManager . RemoveEntry ( entry ) ;
2020-11-10 13:49:02 +00:00
private void entryBecameAlive ( LifetimeEntry entry ) = > addDrawable ( ( HitObjectLifetimeEntry ) entry ) ;
private void entryBecameDead ( LifetimeEntry entry ) = > removeDrawable ( ( HitObjectLifetimeEntry ) entry ) ;
private void addDrawable ( HitObjectLifetimeEntry entry )
{
Debug . Assert ( ! drawableMap . ContainsKey ( entry ) ) ;
2020-12-03 10:46:42 +00:00
var drawable = pooledObjectProvider ? . GetPooledDrawableRepresentation ( entry . HitObject , null ) ;
2020-11-12 05:54:33 +00:00
if ( drawable = = null )
throw new InvalidOperationException ( $"A drawable representation could not be retrieved for hitobject type: {entry.HitObject.GetType().ReadableName()}." ) ;
2020-11-10 13:49:02 +00:00
drawable . OnNewResult + = onNewResult ;
drawable . OnRevertResult + = onRevertResult ;
bindStartTime ( drawable ) ;
AddInternal ( drawableMap [ entry ] = drawable , false ) ;
2020-11-27 04:36:40 +00:00
OnAdd ( drawable ) ;
2020-11-12 09:32:20 +00:00
2020-11-26 05:01:46 +00:00
HitObjectUsageBegan ? . Invoke ( entry . HitObject ) ;
2020-11-10 13:49:02 +00:00
}
private void removeDrawable ( HitObjectLifetimeEntry entry )
2019-12-17 17:56:29 +00:00
{
2020-11-10 13:49:02 +00:00
Debug . Assert ( drawableMap . ContainsKey ( entry ) ) ;
2020-11-06 15:57:33 +00:00
2020-11-10 13:49:02 +00:00
var drawable = drawableMap [ entry ] ;
2021-01-20 22:55:54 +00:00
// OnKilled can potentially change the hitobject's result, so it needs to run first before unbinding.
drawable . OnKilled ( ) ;
2020-11-10 13:49:02 +00:00
drawable . OnNewResult - = onNewResult ;
drawable . OnRevertResult - = onRevertResult ;
drawableMap . Remove ( entry ) ;
2020-11-27 04:36:40 +00:00
OnRemove ( drawable ) ;
2020-11-10 13:49:02 +00:00
unbindStartTime ( drawable ) ;
RemoveInternal ( drawable ) ;
2020-11-26 05:01:46 +00:00
HitObjectUsageFinished ? . Invoke ( entry . HitObject ) ;
2020-11-10 13:49:02 +00:00
}
#endregion
#region Non - pooling support
public virtual void Add ( DrawableHitObject hitObject )
{
bindStartTime ( hitObject ) ;
hitObject . OnNewResult + = onNewResult ;
hitObject . OnRevertResult + = onRevertResult ;
2020-11-12 03:55:42 +00:00
AddInternal ( hitObject ) ;
2020-11-27 04:36:40 +00:00
OnAdd ( hitObject ) ;
2019-12-17 17:56:29 +00:00
}
public virtual bool Remove ( DrawableHitObject hitObject )
{
2020-11-27 04:36:40 +00:00
OnRemove ( hitObject ) ;
2019-12-18 03:03:15 +00:00
if ( ! RemoveInternal ( hitObject ) )
return false ;
2019-12-17 17:56:29 +00:00
2020-11-10 13:49:02 +00:00
hitObject . OnNewResult - = onNewResult ;
hitObject . OnRevertResult - = onRevertResult ;
unbindStartTime ( hitObject ) ;
2019-12-17 17:56:29 +00:00
2019-12-18 03:03:15 +00:00
return true ;
2019-12-17 17:56:29 +00:00
}
2020-11-10 13:49:02 +00:00
public int IndexOf ( DrawableHitObject hitObject ) = > IndexOfInternal ( hitObject ) ;
protected override void OnChildLifetimeBoundaryCrossed ( LifetimeBoundaryCrossedEvent e )
2020-04-23 02:16:59 +00:00
{
2020-11-10 13:49:02 +00:00
if ( ! ( e . Child is DrawableHitObject hitObject ) )
return ;
if ( ( e . Kind = = LifetimeBoundaryKind . End & & e . Direction = = LifetimeBoundaryCrossingDirection . Forward )
| | ( e . Kind = = LifetimeBoundaryKind . Start & & e . Direction = = LifetimeBoundaryCrossingDirection . Backward ) )
{
hitObject . OnKilled ( ) ;
}
2020-09-21 09:27:15 +00:00
}
2020-04-23 02:16:59 +00:00
2020-11-10 13:49:02 +00:00
#endregion
2020-11-27 04:36:40 +00:00
/// <summary>
/// Invoked when a <see cref="DrawableHitObject"/> is added to this container.
/// </summary>
/// <remarks>
/// This method is not invoked for nested <see cref="DrawableHitObject"/>s.
/// </remarks>
protected virtual void OnAdd ( DrawableHitObject drawableHitObject )
{
}
/// <summary>
/// Invoked when a <see cref="DrawableHitObject"/> is removed from this container.
/// </summary>
/// <remarks>
/// This method is not invoked for nested <see cref="DrawableHitObject"/>s.
/// </remarks>
protected virtual void OnRemove ( DrawableHitObject drawableHitObject )
{
}
2020-11-06 15:57:33 +00:00
public virtual void Clear ( bool disposeChildren = true )
2020-09-21 09:27:15 +00:00
{
2020-11-10 13:49:02 +00:00
lifetimeManager . ClearEntries ( ) ;
2020-11-06 15:57:33 +00:00
ClearInternal ( disposeChildren ) ;
2020-11-10 13:49:02 +00:00
unbindAllStartTimes ( ) ;
2020-04-23 02:16:59 +00:00
}
2020-11-12 03:55:42 +00:00
protected override bool CheckChildrenLife ( )
{
bool aliveChanged = base . CheckChildrenLife ( ) ;
2020-11-12 09:34:50 +00:00
aliveChanged | = lifetimeManager . Update ( Time . Current - PastLifetimeExtension , Time . Current + FutureLifetimeExtension ) ;
2020-11-12 03:55:42 +00:00
return aliveChanged ;
}
2020-11-10 13:49:02 +00:00
private void onNewResult ( DrawableHitObject d , JudgementResult r ) = > NewResult ? . Invoke ( d , r ) ;
private void onRevertResult ( DrawableHitObject d , JudgementResult r ) = > RevertResult ? . Invoke ( d , r ) ;
#region Comparator + StartTime tracking
private void bindStartTime ( DrawableHitObject hitObject )
2019-12-17 17:56:29 +00:00
{
2020-11-10 13:49:02 +00:00
var bindable = hitObject . StartTimeBindable . GetBoundCopy ( ) ;
2020-11-12 07:58:40 +00:00
bindable . BindValueChanged ( _ = >
{
2020-11-12 08:07:20 +00:00
if ( LoadState > = LoadState . Ready )
2020-11-12 07:58:40 +00:00
SortInternal ( ) ;
} ) ;
2020-11-06 13:15:00 +00:00
2020-11-10 13:49:02 +00:00
startTimeMap [ hitObject ] = bindable ;
}
2020-11-06 13:15:00 +00:00
2020-11-10 13:49:02 +00:00
private void unbindStartTime ( DrawableHitObject hitObject )
2020-11-06 13:15:00 +00:00
{
2020-11-10 13:49:02 +00:00
startTimeMap [ hitObject ] . UnbindAll ( ) ;
startTimeMap . Remove ( hitObject ) ;
}
2018-04-13 09:19:50 +00:00
2020-11-10 13:49:02 +00:00
private void unbindAllStartTimes ( )
{
foreach ( var kvp in startTimeMap )
kvp . Value . UnbindAll ( ) ;
startTimeMap . Clear ( ) ;
2020-11-06 15:57:33 +00:00
}
2020-11-06 13:15:00 +00:00
2018-04-13 09:19:50 +00:00
protected override int Compare ( Drawable x , Drawable y )
{
if ( ! ( x is DrawableHitObject xObj ) | | ! ( y is DrawableHitObject yObj ) )
return base . Compare ( x , y ) ;
// Put earlier hitobjects towards the end of the list, so they handle input first
2020-11-10 13:49:02 +00:00
int i = yObj . HitObject . StartTime . CompareTo ( xObj . HitObject . StartTime ) ;
2018-04-13 09:19:50 +00:00
return i = = 0 ? CompareReverseChildID ( x , y ) : i ;
}
2019-01-29 06:25:27 +00:00
2020-11-10 13:49:02 +00:00
#endregion
2020-11-06 15:57:33 +00:00
2020-11-10 13:49:02 +00:00
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
unbindAllStartTimes ( ) ;
2019-01-29 06:25:27 +00:00
}
2018-04-13 09:19:50 +00:00
}
}