2019-01-24 16:43:03 +08: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 17:19:50 +08:00
2022-06-17 15:37:17 +08:00
#nullable disable
2019-10-18 12:18:41 +08:00
using System ;
2017-12-22 20:42:54 +08:00
using System.Collections.Generic ;
2022-03-14 14:45:43 +08:00
using System.Collections.Immutable ;
2022-10-11 15:31:37 +08:00
using System.Linq ;
2020-05-15 17:07:41 +08:00
using System.Threading ;
2019-09-02 16:48:41 +08:00
using JetBrains.Annotations ;
2017-12-07 13:42:36 +08:00
using Newtonsoft.Json ;
2019-10-03 13:23:48 +08:00
using osu.Framework.Bindables ;
2021-09-20 00:48:33 +08:00
using osu.Framework.Extensions.ListExtensions ;
2024-06-13 14:18:39 +08:00
using osu.Framework.Extensions.TypeExtensions ;
2021-09-20 00:48:33 +08:00
using osu.Framework.Lists ;
2017-04-06 10:41:16 +08:00
using osu.Game.Audio ;
2017-07-26 12:22:46 +08:00
using osu.Game.Beatmaps ;
2017-05-23 12:55:18 +08:00
using osu.Game.Beatmaps.ControlPoints ;
2018-08-01 20:04:03 +08:00
using osu.Game.Rulesets.Judgements ;
2017-04-21 15:18:34 +08:00
using osu.Game.Rulesets.Objects.Types ;
2019-09-06 14:24:00 +08:00
using osu.Game.Rulesets.Scoring ;
2018-04-13 17:19:50 +08:00
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Objects
2016-08-31 11:33:01 +08:00
{
/// <summary>
2017-03-13 18:15:25 +08: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 11:33:01 +08:00
/// </summary>
2023-04-26 19:10:57 +08:00
public class HitObject
2016-08-31 11:33:01 +08:00
{
2018-12-04 11:01:30 +08: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 12:18:41 +08:00
/// <summary>
/// Invoked after <see cref="ApplyDefaults"/> has completed on this <see cref="HitObject"/>.
/// </summary>
2024-01-29 13:57:19 +08:00
// TODO: This has no implicit unbind flow. Currently, if a Playfield manages HitObjects it will leave a bound event on this and cause the
// playfield to remain in memory.
2020-05-08 17:49:19 +08:00
public event Action < HitObject > DefaultsApplied ;
2019-10-18 12:18:41 +08:00
2020-02-02 05:50:29 +08:00
public readonly Bindable < double > StartTimeBindable = new BindableDouble ( ) ;
2019-10-03 13:23:48 +08:00
2017-03-13 18:15:25 +08:00
/// <summary>
/// The time at which the HitObject starts.
/// </summary>
2019-10-03 13:23:48 +08:00
public virtual double StartTime
{
get = > StartTimeBindable . Value ;
set = > StartTimeBindable . Value = value ;
}
2018-04-13 17:19:50 +08:00
2019-11-08 13:04:57 +08:00
public readonly BindableList < HitSampleInfo > SamplesBindable = new BindableList < HitSampleInfo > ( ) ;
2018-04-13 17:19:50 +08:00
2017-03-13 18:15:25 +08:00
/// <summary>
2017-04-06 10:41:16 +08:00
/// The samples to be played when this hit object is hit.
2017-04-21 17:49:49 +08:00
/// <para>
2017-04-21 19:42:13 +08:00
/// In the case of <see cref="IHasRepeats"/> types, this is the sample of the curve body
2017-04-21 17:49:49 +08:00
/// and can be treated as the default samples for the hit object.
/// </para>
2017-03-13 18:15:25 +08:00
/// </summary>
2019-11-08 13:04:57 +08:00
public IList < HitSampleInfo > Samples
2017-12-25 15:41:18 +08:00
{
2019-11-08 13:04:57 +08:00
get = > SamplesBindable ;
set
{
SamplesBindable . Clear ( ) ;
SamplesBindable . AddRange ( value ) ;
}
2017-12-25 15:41:18 +08:00
}
2018-04-13 17:19:50 +08:00
2022-03-14 14:45:43 +08: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 ;
2017-09-12 09:01:07 +08:00
/// <summary>
/// Whether this <see cref="HitObject"/> is in Kiai time.
/// </summary>
2017-12-07 13:42:36 +08:00
[JsonIgnore]
2017-09-12 09:01:07 +08:00
public bool Kiai { get ; private set ; }
2018-04-13 17:19:50 +08:00
2018-02-02 17:47:10 +08:00
/// <summary>
2018-02-08 16:38:46 +08:00
/// The hit windows for this <see cref="HitObject"/>.
2018-02-02 17:47:10 +08:00
/// </summary>
2020-10-07 12:45:41 +08:00
[JsonIgnore]
2018-05-11 14:30:26 +08:00
public HitWindows HitWindows { get ; set ; }
2018-04-13 17:19:50 +08:00
2018-11-05 11:15:45 +08:00
private readonly List < HitObject > nestedHitObjects = new List < HitObject > ( ) ;
2018-04-13 17:19:50 +08:00
2017-12-22 20:42:54 +08:00
[JsonIgnore]
2021-09-20 00:48:33 +08:00
public SlimReadOnlyListWrapper < HitObject > NestedHitObjects = > nestedHitObjects . AsSlimReadOnly ( ) ;
2018-04-13 17:19:50 +08:00
2017-03-16 15:55:08 +08:00
/// <summary>
/// Applies default values to this HitObject.
/// </summary>
2017-05-23 12:55:18 +08:00
/// <param name="controlPointInfo">The control points.</param>
2017-03-16 16:24:41 +08:00
/// <param name="difficulty">The difficulty settings to use.</param>
2020-05-15 17:07:41 +08:00
/// <param name="cancellationToken">The cancellation token.</param>
2021-10-01 13:56:42 +08:00
public void ApplyDefaults ( ControlPointInfo controlPointInfo , IBeatmapDifficultyInfo difficulty , CancellationToken cancellationToken = default )
2017-12-22 20:42:54 +08:00
{
2023-04-26 17:46:05 +08:00
ApplyDefaultsToSelf ( controlPointInfo , difficulty ) ;
2018-10-10 12:03:18 +08:00
nestedHitObjects . Clear ( ) ;
2018-05-17 11:29:33 +08:00
2020-05-15 17:07:41 +08:00
CreateNestedHitObjects ( cancellationToken ) ;
2018-05-17 11:29:33 +08:00
2019-09-26 16:39:19 +08:00
if ( this is IHasComboInformation hasCombo )
{
2021-09-20 00:48:33 +08:00
foreach ( HitObject hitObject in nestedHitObjects )
2019-09-26 16:39:19 +08:00
{
2021-09-20 00:48:33 +08:00
if ( hitObject is IHasComboInformation n )
{
n . ComboIndexBindable . BindTo ( hasCombo . ComboIndexBindable ) ;
n . ComboIndexWithOffsetsBindable . BindTo ( hasCombo . ComboIndexWithOffsetsBindable ) ;
n . IndexInCurrentComboBindable . BindTo ( hasCombo . IndexInCurrentComboBindable ) ;
}
2019-09-26 16:39:19 +08:00
}
}
2018-11-05 11:15:45 +08:00
nestedHitObjects . Sort ( ( h1 , h2 ) = > h1 . StartTime . CompareTo ( h2 . StartTime ) ) ;
2018-10-10 13:58:29 +08:00
foreach ( var h in nestedHitObjects )
2020-05-15 17:07:41 +08:00
h . ApplyDefaults ( controlPointInfo , difficulty , cancellationToken ) ;
2019-10-18 12:18:41 +08:00
2022-01-13 02:24:59 +08: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-12 04:27:22 +08:00
// if `StartTimeBindable` is changed prior to default application.
2022-01-13 02:24:59 +08:00
StartTimeBindable . ValueChanged + = onStartTimeChanged ;
DefaultsApplied ? . Invoke ( this ) ;
void onStartTimeChanged ( ValueChangedEvent < double > time )
2022-01-12 04:27:22 +08:00
{
double offset = time . NewValue - time . OldValue ;
foreach ( var nested in nestedHitObjects )
nested . StartTime + = offset ;
2022-01-13 02:24:59 +08:00
}
2017-12-22 20:42:54 +08:00
}
2018-04-13 17:19:50 +08:00
2021-10-01 13:56:42 +08:00
protected virtual void ApplyDefaultsToSelf ( ControlPointInfo controlPointInfo , IBeatmapDifficultyInfo difficulty )
2017-04-05 20:59:07 +08:00
{
2018-12-04 11:01:30 +08:00
Kiai = controlPointInfo . EffectPointAt ( StartTime + control_point_leniency ) . KiaiMode ;
2018-04-13 17:19:50 +08:00
2020-06-03 15:48:44 +08:00
HitWindows ? ? = CreateHitWindows ( ) ;
2018-05-11 14:30:26 +08:00
HitWindows ? . SetDifficulty ( difficulty . OverallDifficulty ) ;
2017-04-05 20:59:07 +08:00
}
2018-04-13 17:19:50 +08:00
2020-05-15 17:07:41 +08:00
protected virtual void CreateNestedHitObjects ( CancellationToken cancellationToken )
2017-12-22 20:42:54 +08:00
{
}
2018-04-13 17:19:50 +08:00
2018-10-10 12:03:18 +08:00
protected void AddNested ( HitObject hitObject ) = > nestedHitObjects . Add ( hitObject ) ;
2018-05-11 14:30:26 +08:00
2024-02-10 04:20:31 +08:00
/// <summary>
/// The <see cref="Judgement"/> that represents the scoring information for this <see cref="HitObject"/>.
/// </summary>
[JsonIgnore]
public Judgement Judgement = > judgement ? ? = CreateJudgement ( ) ;
private Judgement judgement ;
2018-08-06 09:55:38 +08:00
/// <summary>
2024-02-18 17:54:29 +08:00
/// Should be overridden to create a <see cref="Judgement"/> that represents the scoring information for this <see cref="HitObject"/>.
2018-08-06 09:55:38 +08:00
/// </summary>
2024-02-17 01:24:02 +08:00
/// <remarks>
2024-02-18 17:54:29 +08:00
/// For read access, use <see cref="Judgement"/> to avoid unnecessary allocations.
2024-02-17 01:24:02 +08:00
/// </remarks>
2020-02-23 12:01:30 +08:00
[NotNull]
2024-02-17 01:24:02 +08:00
public virtual Judgement CreateJudgement ( ) = > new Judgement ( ) ;
2018-08-06 09:55:38 +08:00
2018-05-11 14:30:26 +08:00
/// <summary>
/// Creates the <see cref="HitWindows"/> for this <see cref="HitObject"/>.
2019-09-02 16:15:36 +08: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 14:52:51 +08:00
/// <para>
2019-04-25 16:36:17 +08: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 14:52:51 +08:00
/// </para>
2018-05-11 14:30:26 +08:00
/// </summary>
2019-10-09 18:08:31 +08:00
[NotNull]
2020-02-25 18:07:15 +08:00
protected virtual HitWindows CreateHitWindows ( ) = > new HitWindows ( ) ;
2022-10-11 15:31:37 +08:00
2023-01-19 19:53:35 +08:00
/// <summary>
/// The maximum offset from the end time of <see cref="HitObject"/> at which this <see cref="HitObject"/> can be judged.
/// <para>
/// Defaults to the miss window.
/// </para>
/// </summary>
public virtual double MaximumJudgementOffset = > HitWindows ? . WindowFor ( HitResult . Miss ) ? ? 0 ;
2022-10-11 15:31:37 +08:00
public IList < HitSampleInfo > CreateSlidingSamples ( )
{
var slidingSamples = new List < HitSampleInfo > ( ) ;
var normalSample = Samples . FirstOrDefault ( s = > s . Name = = HitSampleInfo . HIT_NORMAL ) ;
if ( normalSample ! = null )
slidingSamples . Add ( normalSample . With ( "sliderslide" ) ) ;
var whistleSample = Samples . FirstOrDefault ( s = > s . Name = = HitSampleInfo . HIT_WHISTLE ) ;
if ( whistleSample ! = null )
slidingSamples . Add ( whistleSample . With ( "sliderwhistle" ) ) ;
return slidingSamples ;
}
2023-04-30 05:52:24 +08:00
/// <summary>
2023-05-17 13:07:48 +08:00
/// Create a <see cref="HitSampleInfo"/> based on the sample settings of the first <see cref="HitSampleInfo.HIT_NORMAL"/> sample in <see cref="Samples"/>.
/// If no sample is available, sane default settings will be used instead.
2023-04-30 05:52:24 +08:00
/// </summary>
2023-05-17 13:07:48 +08:00
/// <remarks>
/// In the case an existing sample exists, all settings apart from the sample name will be inherited. This includes volume, bank and suffix.
/// </remarks>
2023-04-30 05:52:24 +08:00
/// <param name="sampleName">The name of the sample.</param>
/// <returns>A populated <see cref="HitSampleInfo"/>.</returns>
2023-05-17 13:07:48 +08:00
public HitSampleInfo CreateHitSampleInfo ( string sampleName = HitSampleInfo . HIT_NORMAL )
2023-04-30 05:52:24 +08:00
{
2023-05-30 13:04:02 +08:00
// As per stable, all non-normal "addition" samples should use the same bank.
if ( sampleName ! = HitSampleInfo . HIT_NORMAL )
{
if ( Samples . FirstOrDefault ( s = > s . Name ! = HitSampleInfo . HIT_NORMAL ) is HitSampleInfo existingAddition )
return existingAddition . With ( newName : sampleName ) ;
}
// Fall back to using the normal sample bank otherwise.
if ( Samples . FirstOrDefault ( s = > s . Name = = HitSampleInfo . HIT_NORMAL ) is HitSampleInfo existingNormal )
2024-08-30 03:54:47 +08:00
return existingNormal . With ( newName : sampleName , newEditorAutoBank : true ) ;
2023-05-17 13:07:48 +08:00
return new HitSampleInfo ( sampleName ) ;
2023-04-30 05:52:24 +08:00
}
2024-06-13 14:18:39 +08:00
public override string ToString ( ) = > $"{GetType().ReadableName()} @ {StartTime}" ;
2016-08-31 11:33:01 +08:00
}
2019-11-25 18:01:24 +08:00
public static class HitObjectExtensions
{
/// <summary>
/// Returns the end time of this object.
/// </summary>
/// <remarks>
2020-05-27 11:38:39 +08:00
/// This returns the <see cref="IHasDuration.EndTime"/> where available, falling back to <see cref="HitObject.StartTime"/> otherwise.
2019-11-25 18:01:24 +08:00
/// </remarks>
/// <param name="hitObject">The object.</param>
/// <returns>The end time of this object.</returns>
2020-05-27 11:38:39 +08:00
public static double GetEndTime ( this HitObject hitObject ) = > ( hitObject as IHasDuration ) ? . EndTime ? ? hitObject . StartTime ;
2019-11-25 18:01:24 +08:00
}
2016-08-31 11:33:01 +08:00
}