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
2022-06-17 07:37:17 +00:00
#nullable disable
2019-10-18 04:18:41 +00:00
using System ;
2017-12-22 12:42:54 +00:00
using System.Collections.Generic ;
2022-03-14 06:45:43 +00:00
using System.Collections.Immutable ;
2020-05-15 09:07:41 +00:00
using System.Threading ;
2019-09-02 08:48:41 +00:00
using JetBrains.Annotations ;
2017-12-07 05:42:36 +00:00
using Newtonsoft.Json ;
2019-10-03 05:23:48 +00:00
using osu.Framework.Bindables ;
2021-09-19 16:48:33 +00:00
using osu.Framework.Extensions.ListExtensions ;
using osu.Framework.Lists ;
2017-04-06 02:41:16 +00:00
using osu.Game.Audio ;
2017-07-26 04:22:46 +00:00
using osu.Game.Beatmaps ;
2017-05-23 04:55:18 +00:00
using osu.Game.Beatmaps.ControlPoints ;
2021-08-30 05:12:30 +00:00
using osu.Game.Beatmaps.Legacy ;
2018-08-01 12:04:03 +00:00
using osu.Game.Rulesets.Judgements ;
2017-04-21 07:18:34 +00:00
using osu.Game.Rulesets.Objects.Types ;
2019-09-06 06:24:00 +00:00
using osu.Game.Rulesets.Scoring ;
2018-04-13 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
namespace osu.Game.Rulesets.Objects
2016-08-31 03:33:01 +00:00
{
/// <summary>
2017-03-13 10:15:25 +00:00
/// A HitObject describes an object in a Beatmap.
/// <para>
/// HitObjects may contain more properties for which you should be checking through the IHas* types.
/// </para>
2016-08-31 03:33:01 +00:00
/// </summary>
2020-02-25 10:07:15 +00:00
public class HitObject
2016-08-31 03:33:01 +00:00
{
2018-12-04 03:01:30 +00:00
/// <summary>
/// A small adjustment to the start time of control points to account for rounding/precision errors.
/// </summary>
private const double control_point_leniency = 1 ;
2019-10-18 04:18:41 +00:00
/// <summary>
/// Invoked after <see cref="ApplyDefaults"/> has completed on this <see cref="HitObject"/>.
/// </summary>
2020-05-08 09:49:19 +00:00
public event Action < HitObject > DefaultsApplied ;
2019-10-18 04:18:41 +00:00
2020-02-01 21:50:29 +00:00
public readonly Bindable < double > StartTimeBindable = new BindableDouble ( ) ;
2019-10-03 05:23:48 +00:00
2017-03-13 10:15:25 +00:00
/// <summary>
/// The time at which the HitObject starts.
/// </summary>
2019-10-03 05:23:48 +00:00
public virtual double StartTime
{
get = > StartTimeBindable . Value ;
set = > StartTimeBindable . Value = value ;
}
2018-04-13 09:19:50 +00:00
2019-11-08 05:04:57 +00:00
public readonly BindableList < HitSampleInfo > SamplesBindable = new BindableList < HitSampleInfo > ( ) ;
2018-04-13 09:19:50 +00:00
2017-03-13 10:15:25 +00:00
/// <summary>
2017-04-06 02:41:16 +00:00
/// The samples to be played when this hit object is hit.
2017-04-21 09:49:49 +00:00
/// <para>
2017-04-21 11:42:13 +00:00
/// In the case of <see cref="IHasRepeats"/> types, this is the sample of the curve body
2017-04-21 09:49:49 +00:00
/// and can be treated as the default samples for the hit object.
/// </para>
2017-03-13 10:15:25 +00:00
/// </summary>
2019-11-08 05:04:57 +00:00
public IList < HitSampleInfo > Samples
2017-12-25 07:41:18 +00:00
{
2019-11-08 05:04:57 +00:00
get = > SamplesBindable ;
set
{
SamplesBindable . Clear ( ) ;
SamplesBindable . AddRange ( value ) ;
}
2017-12-25 07:41:18 +00:00
}
2018-04-13 09:19:50 +00:00
2022-03-14 06:45:43 +00:00
/// <summary>
/// Any samples which may be used by this hit object that are non-standard.
/// This is used only to preload these samples ahead of time.
/// </summary>
public virtual IList < HitSampleInfo > AuxiliarySamples = > ImmutableList < HitSampleInfo > . Empty ;
2021-09-02 10:42:34 +00:00
public SampleControlPoint SampleControlPoint = SampleControlPoint . DEFAULT ;
public DifficultyControlPoint DifficultyControlPoint = DifficultyControlPoint . DEFAULT ;
2021-08-31 07:16:53 +00:00
2017-09-12 01:01:07 +00:00
/// <summary>
/// Whether this <see cref="HitObject"/> is in Kiai time.
/// </summary>
2017-12-07 05:42:36 +00:00
[JsonIgnore]
2017-09-12 01:01:07 +00:00
public bool Kiai { get ; private set ; }
2018-04-13 09:19:50 +00:00
2018-02-02 09:47:10 +00:00
/// <summary>
2018-02-08 08:38:46 +00:00
/// The hit windows for this <see cref="HitObject"/>.
2018-02-02 09:47:10 +00:00
/// </summary>
2020-10-07 04:45:41 +00:00
[JsonIgnore]
2018-05-11 06:30:26 +00:00
public HitWindows HitWindows { get ; set ; }
2018-04-13 09:19:50 +00:00
2018-11-05 03:15:45 +00:00
private readonly List < HitObject > nestedHitObjects = new List < HitObject > ( ) ;
2018-04-13 09:19:50 +00:00
2017-12-22 12:42:54 +00:00
[JsonIgnore]
2021-09-19 16:48:33 +00:00
public SlimReadOnlyListWrapper < HitObject > NestedHitObjects = > nestedHitObjects . AsSlimReadOnly ( ) ;
2018-04-13 09:19:50 +00:00
2017-03-16 07:55:08 +00:00
/// <summary>
/// Applies default values to this HitObject.
/// </summary>
2017-05-23 04:55:18 +00:00
/// <param name="controlPointInfo">The control points.</param>
2017-03-16 08:24:41 +00:00
/// <param name="difficulty">The difficulty settings to use.</param>
2020-05-15 09:07:41 +00:00
/// <param name="cancellationToken">The cancellation token.</param>
2021-10-01 05:56:42 +00:00
public void ApplyDefaults ( ControlPointInfo controlPointInfo , IBeatmapDifficultyInfo difficulty , CancellationToken cancellationToken = default )
2017-12-22 12:42:54 +00:00
{
2021-09-03 07:48:37 +00:00
var legacyInfo = controlPointInfo as LegacyControlPointInfo ;
if ( legacyInfo ! = null )
2021-09-10 05:36:32 +00:00
DifficultyControlPoint = ( DifficultyControlPoint ) legacyInfo . DifficultyPointAt ( StartTime ) . DeepClone ( ) ;
2022-06-20 07:53:03 +00:00
else if ( ReferenceEquals ( DifficultyControlPoint , DifficultyControlPoint . DEFAULT ) )
2021-10-26 06:59:48 +00:00
DifficultyControlPoint = new DifficultyControlPoint ( ) ;
2018-12-03 08:21:27 +00:00
2022-01-11 20:27:22 +00:00
DifficultyControlPoint . Time = StartTime ;
2021-09-01 09:26:00 +00:00
ApplyDefaultsToSelf ( controlPointInfo , difficulty ) ;
2021-09-03 07:48:37 +00:00
// This is done here after ApplyDefaultsToSelf as we may require custom defaults to be applied to have an accurate end time.
if ( legacyInfo ! = null )
2021-09-10 05:36:32 +00:00
SampleControlPoint = ( SampleControlPoint ) legacyInfo . SamplePointAt ( this . GetEndTime ( ) + control_point_leniency ) . DeepClone ( ) ;
2022-06-20 07:53:03 +00:00
else if ( ReferenceEquals ( SampleControlPoint , SampleControlPoint . DEFAULT ) )
2021-10-26 06:59:48 +00:00
SampleControlPoint = new SampleControlPoint ( ) ;
2021-09-03 07:48:37 +00:00
2022-01-11 20:27:22 +00:00
SampleControlPoint . Time = this . GetEndTime ( ) + control_point_leniency ;
2018-10-10 04:03:18 +00:00
nestedHitObjects . Clear ( ) ;
2018-05-17 03:29:33 +00:00
2020-05-15 09:07:41 +00:00
CreateNestedHitObjects ( cancellationToken ) ;
2018-05-17 03:29:33 +00:00
2019-09-26 08:39:19 +00:00
if ( this is IHasComboInformation hasCombo )
{
2021-09-19 16:48:33 +00:00
foreach ( HitObject hitObject in nestedHitObjects )
2019-09-26 08:39:19 +00:00
{
2021-09-19 16:48:33 +00:00
if ( hitObject is IHasComboInformation n )
{
n . ComboIndexBindable . BindTo ( hasCombo . ComboIndexBindable ) ;
n . ComboIndexWithOffsetsBindable . BindTo ( hasCombo . ComboIndexWithOffsetsBindable ) ;
n . IndexInCurrentComboBindable . BindTo ( hasCombo . IndexInCurrentComboBindable ) ;
}
2019-09-26 08:39:19 +00:00
}
}
2018-11-05 03:15:45 +00:00
nestedHitObjects . Sort ( ( h1 , h2 ) = > h1 . StartTime . CompareTo ( h2 . StartTime ) ) ;
2018-10-10 05:58:29 +00:00
foreach ( var h in nestedHitObjects )
2020-05-15 09:07:41 +00:00
h . ApplyDefaults ( controlPointInfo , difficulty , cancellationToken ) ;
2019-10-18 04:18:41 +00:00
2022-01-12 18:24:59 +00:00
// `ApplyDefaults()` may be called multiple times on a single hitobject.
// to prevent subscribing to `StartTimeBindable.ValueChanged` multiple times with the same callback,
// remove the previous subscription (if present) before (re-)registering.
StartTimeBindable . ValueChanged - = onStartTimeChanged ;
// this callback must be (re-)registered after default application
// to ensure that the read of `this.GetEndTime()` within `onStartTimeChanged` doesn't return an invalid value
2022-01-11 20:27:22 +00:00
// if `StartTimeBindable` is changed prior to default application.
2022-01-12 18:24:59 +00:00
StartTimeBindable . ValueChanged + = onStartTimeChanged ;
DefaultsApplied ? . Invoke ( this ) ;
void onStartTimeChanged ( ValueChangedEvent < double > time )
2022-01-11 20:27:22 +00:00
{
double offset = time . NewValue - time . OldValue ;
foreach ( var nested in nestedHitObjects )
nested . StartTime + = offset ;
2022-01-11 21:12:23 +00:00
DifficultyControlPoint . Time = time . NewValue ;
SampleControlPoint . Time = this . GetEndTime ( ) + control_point_leniency ;
2022-01-12 18:24:59 +00:00
}
2017-12-22 12:42:54 +00:00
}
2018-04-13 09:19:50 +00:00
2021-10-01 05:56:42 +00:00
protected virtual void ApplyDefaultsToSelf ( ControlPointInfo controlPointInfo , IBeatmapDifficultyInfo difficulty )
2017-04-05 12:59:07 +00:00
{
2018-12-04 03:01:30 +00:00
Kiai = controlPointInfo . EffectPointAt ( StartTime + control_point_leniency ) . KiaiMode ;
2018-04-13 09:19:50 +00:00
2020-06-03 07:48:44 +00:00
HitWindows ? ? = CreateHitWindows ( ) ;
2018-05-11 06:30:26 +00:00
HitWindows ? . SetDifficulty ( difficulty . OverallDifficulty ) ;
2017-04-05 12:59:07 +00:00
}
2018-04-13 09:19:50 +00:00
2020-05-15 09:07:41 +00:00
protected virtual void CreateNestedHitObjects ( CancellationToken cancellationToken )
2017-12-22 12:42:54 +00:00
{
}
2018-04-13 09:19:50 +00:00
2018-10-10 04:03:18 +00:00
protected void AddNested ( HitObject hitObject ) = > nestedHitObjects . Add ( hitObject ) ;
2018-05-11 06:30:26 +00:00
2018-08-06 01:55:38 +00:00
/// <summary>
/// Creates the <see cref="Judgement"/> that represents the scoring information for this <see cref="HitObject"/>.
/// </summary>
2020-02-23 04:01:30 +00:00
[NotNull]
2020-02-25 10:07:15 +00:00
public virtual Judgement CreateJudgement ( ) = > new Judgement ( ) ;
2018-08-06 01:55:38 +00:00
2018-05-11 06:30:26 +00:00
/// <summary>
/// Creates the <see cref="HitWindows"/> for this <see cref="HitObject"/>.
2019-09-02 08:15:36 +00:00
/// This can be null to indicate that the <see cref="HitObject"/> has no <see cref="HitWindows"/> and timing errors should not be displayed to the user.
2018-05-11 06:52:51 +00:00
/// <para>
2019-04-25 08:36:17 +00:00
/// This will only be invoked if <see cref="HitWindows"/> hasn't been set externally (e.g. from a <see cref="BeatmapConverter{T}"/>.
2018-05-11 06:52:51 +00:00
/// </para>
2018-05-11 06:30:26 +00:00
/// </summary>
2019-10-09 10:08:31 +00:00
[NotNull]
2020-02-25 10:07:15 +00:00
protected virtual HitWindows CreateHitWindows ( ) = > new HitWindows ( ) ;
2016-08-31 03:33:01 +00:00
}
2019-11-25 10:01:24 +00:00
public static class HitObjectExtensions
{
/// <summary>
/// Returns the end time of this object.
/// </summary>
/// <remarks>
2020-05-27 03:38:39 +00:00
/// This returns the <see cref="IHasDuration.EndTime"/> where available, falling back to <see cref="HitObject.StartTime"/> otherwise.
2019-11-25 10:01:24 +00:00
/// </remarks>
/// <param name="hitObject">The object.</param>
/// <returns>The end time of this object.</returns>
2020-05-27 03:38:39 +00:00
public static double GetEndTime ( this HitObject hitObject ) = > ( hitObject as IHasDuration ) ? . EndTime ? ? hitObject . StartTime ;
2019-11-25 10:01:24 +00:00
}
2016-08-31 03:33:01 +00:00
}