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
using System.Collections.Generic ;
2019-09-02 08:48:41 +00:00
using JetBrains.Annotations ;
2018-04-13 09:19:50 +00:00
using Newtonsoft.Json ;
using osu.Game.Audio ;
using osu.Game.Beatmaps ;
using osu.Game.Beatmaps.ControlPoints ;
2018-08-01 12:04:03 +00:00
using osu.Game.Rulesets.Judgements ;
2018-04-13 09:19:50 +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
namespace osu.Game.Rulesets.Objects
{
/// <summary>
/// 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>
/// </summary>
public class HitObject
{
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 ;
2018-04-13 09:19:50 +00:00
/// <summary>
/// The time at which the HitObject starts.
/// </summary>
public virtual double StartTime { get ; set ; }
2019-06-30 12:58:30 +00:00
private List < HitSampleInfo > samples ;
2018-04-13 09:19:50 +00:00
/// <summary>
/// The samples to be played when this hit object is hit.
/// <para>
/// In the case of <see cref="IHasRepeats"/> types, this is the sample of the curve body
/// and can be treated as the default samples for the hit object.
/// </para>
/// </summary>
2019-06-30 12:58:30 +00:00
public List < HitSampleInfo > Samples
2018-04-13 09:19:50 +00:00
{
2019-06-30 12:58:30 +00:00
get = > samples ? ? ( samples = new List < HitSampleInfo > ( ) ) ;
2018-04-13 09:19:50 +00:00
set = > samples = value ;
}
[JsonIgnore]
public SampleControlPoint SampleControlPoint ;
/// <summary>
/// Whether this <see cref="HitObject"/> is in Kiai time.
/// </summary>
[JsonIgnore]
public bool Kiai { get ; private set ; }
/// <summary>
/// The hit windows for this <see cref="HitObject"/>.
/// </summary>
2019-09-02 08:48:41 +00:00
[CanBeNull]
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
[JsonIgnore]
2018-10-10 04:03:18 +00:00
public IReadOnlyList < HitObject > NestedHitObjects = > nestedHitObjects ;
2018-04-13 09:19:50 +00:00
/// <summary>
/// Applies default values to this HitObject.
/// </summary>
/// <param name="controlPointInfo">The control points.</param>
/// <param name="difficulty">The difficulty settings to use.</param>
public void ApplyDefaults ( ControlPointInfo controlPointInfo , BeatmapDifficulty difficulty )
{
ApplyDefaultsToSelf ( controlPointInfo , difficulty ) ;
2018-12-03 08:21:27 +00:00
// This is done here since ApplyDefaultsToSelf may be used to determine the end time
2018-12-04 03:01:30 +00:00
SampleControlPoint = controlPointInfo . SamplePointAt ( ( ( this as IHasEndTime ) ? . EndTime ? ? StartTime ) + control_point_leniency ) ;
2018-12-03 08:21:27 +00:00
2018-10-10 04:03:18 +00:00
nestedHitObjects . Clear ( ) ;
2018-05-17 03:29:33 +00:00
2018-04-13 09:19:50 +00:00
CreateNestedHitObjects ( ) ;
2018-05-17 03:29:33 +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 )
2018-05-11 06:30:26 +00:00
{
2018-10-10 04:03:18 +00:00
h . HitWindows = HitWindows ;
h . ApplyDefaults ( controlPointInfo , difficulty ) ;
2018-10-10 05:58:29 +00:00
}
2018-04-13 09:19:50 +00:00
}
protected virtual void ApplyDefaultsToSelf ( ControlPointInfo controlPointInfo , BeatmapDifficulty difficulty )
{
2018-12-04 03:01:30 +00:00
Kiai = controlPointInfo . EffectPointAt ( StartTime + control_point_leniency ) . KiaiMode ;
2018-04-13 09:19:50 +00:00
2018-05-11 06:30:26 +00:00
if ( HitWindows = = null )
HitWindows = CreateHitWindows ( ) ;
HitWindows ? . SetDifficulty ( difficulty . OverallDifficulty ) ;
2018-04-13 09:19:50 +00:00
}
protected virtual void CreateNestedHitObjects ( )
{
}
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"/>.
/// May be null.
/// </summary>
2018-08-06 02:50:18 +00:00
public virtual Judgement CreateJudgement ( ) = > null ;
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-09-02 08:48:41 +00:00
[CanBeNull]
2018-05-11 06:52:51 +00:00
protected virtual HitWindows CreateHitWindows ( ) = > new HitWindows ( ) ;
2018-04-13 09:19:50 +00:00
}
}