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
|
|
|
|
|
2020-12-09 23:26:35 +08:00
|
|
|
|
using System;
|
2019-02-21 18:04:31 +08:00
|
|
|
|
using osu.Framework.Bindables;
|
2017-07-26 12:22:46 +08:00
|
|
|
|
using osu.Game.Beatmaps;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
using osu.Game.Beatmaps.ControlPoints;
|
2017-04-18 15:05:58 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
2023-10-20 17:57:14 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Legacy;
|
2017-04-18 15:05:58 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
2019-09-06 14:24:00 +08:00
|
|
|
|
using osu.Game.Rulesets.Osu.Scoring;
|
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
using osuTK;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-04-18 15:05:58 +08:00
|
|
|
|
namespace osu.Game.Rulesets.Osu.Objects
|
2016-09-02 17:27:38 +08:00
|
|
|
|
{
|
2024-10-09 02:29:06 +08:00
|
|
|
|
public abstract class OsuHitObject : HitObject, IHasComboInformation, IHasMutablePosition, IHasTimePreempt
|
2016-09-02 17:27:38 +08:00
|
|
|
|
{
|
2019-10-17 15:36:47 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The radius of hit objects (ie. the radius of a <see cref="HitCircle"/>).
|
|
|
|
|
/// </summary>
|
2019-07-18 14:59:22 +08:00
|
|
|
|
public const float OBJECT_RADIUS = 64;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2023-09-20 11:48:15 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The width and height any element participating in display of a hitcircle (or similarly sized object) should be.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public static readonly Vector2 OBJECT_DIMENSIONS = new Vector2(OBJECT_RADIUS * 2);
|
|
|
|
|
|
2019-10-17 15:36:47 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Scoring distance with a speed-adjusted beat length of 1 second (ie. the speed slider balls move through their track).
|
|
|
|
|
/// </summary>
|
|
|
|
|
internal const float BASE_SCORING_DISTANCE = 100;
|
|
|
|
|
|
2021-02-10 21:06:19 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Minimum preempt time at AR=10.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public const double PREEMPT_MIN = 450;
|
|
|
|
|
|
2023-12-15 02:54:23 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Median preempt time at AR=5.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public const double PREEMPT_MID = 1200;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Maximum preempt time at AR=0.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public const double PREEMPT_MAX = 1800;
|
|
|
|
|
|
2024-07-23 19:30:10 +08:00
|
|
|
|
public double TimePreempt { get; set; } = 600;
|
2018-07-05 10:32:09 +08:00
|
|
|
|
public double TimeFadeIn = 400;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<Vector2> position;
|
|
|
|
|
|
|
|
|
|
public Bindable<Vector2> PositionBindable => position.Bindable;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-10-25 17:16:25 +08:00
|
|
|
|
public virtual Vector2 Position
|
2018-03-09 22:12:34 +08:00
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => position.Value;
|
|
|
|
|
set => position.Value = value;
|
2018-03-09 22:12:34 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2024-10-09 03:25:03 +08:00
|
|
|
|
public float X
|
|
|
|
|
{
|
|
|
|
|
get => Position.X;
|
|
|
|
|
set => Position = Position with { X = value };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public float Y
|
|
|
|
|
{
|
|
|
|
|
get => Position.Y;
|
|
|
|
|
set => Position = Position with { Y = value };
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-02-09 15:29:21 +08:00
|
|
|
|
public Vector2 StackedPosition => Position + StackOffset;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2016-12-06 20:14:38 +08:00
|
|
|
|
public virtual Vector2 EndPosition => Position;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 08:15:43 +08:00
|
|
|
|
public Vector2 StackedEndPosition => EndPosition + StackOffset;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<int> stackHeight;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<int> StackHeightBindable => stackHeight.Bindable;
|
2018-10-03 16:06:18 +08:00
|
|
|
|
|
|
|
|
|
public int StackHeight
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => stackHeight.Value;
|
|
|
|
|
set => stackHeight.Value = value;
|
2018-10-03 16:06:18 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-26 08:15:43 +08:00
|
|
|
|
public virtual Vector2 StackOffset => new Vector2(StackHeight * Scale * -6.4f);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-03-06 22:26:57 +08:00
|
|
|
|
public double Radius => OBJECT_RADIUS * Scale;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<float> scale = new HitObjectProperty<float>(1);
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<float> ScaleBindable => scale.Bindable;
|
2018-10-26 14:26:19 +08:00
|
|
|
|
|
|
|
|
|
public float Scale
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => scale.Value;
|
|
|
|
|
set => scale.Value = value;
|
2018-10-26 14:26:19 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-03-14 16:01:46 +08:00
|
|
|
|
public virtual bool NewCombo { get; set; }
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<int> comboOffset;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<int> ComboOffsetBindable => comboOffset.Bindable;
|
2021-07-20 19:15:43 +08:00
|
|
|
|
|
|
|
|
|
public int ComboOffset
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => comboOffset.Value;
|
|
|
|
|
set => comboOffset.Value = value;
|
2021-07-20 19:15:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<int> indexInCurrentCombo;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<int> IndexInCurrentComboBindable => indexInCurrentCombo.Bindable;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
|
|
|
|
|
public virtual int IndexInCurrentCombo
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => indexInCurrentCombo.Value;
|
|
|
|
|
set => indexInCurrentCombo.Value = value;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<int> comboIndex;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<int> ComboIndexBindable => comboIndex.Bindable;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2019-09-26 15:55:08 +08:00
|
|
|
|
public virtual int ComboIndex
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => comboIndex.Value;
|
|
|
|
|
set => comboIndex.Value = value;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<int> comboIndexWithOffsets;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<int> ComboIndexWithOffsetsBindable => comboIndexWithOffsets.Bindable;
|
2021-07-22 21:22:42 +08:00
|
|
|
|
|
|
|
|
|
public int ComboIndexWithOffsets
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => comboIndexWithOffsets.Value;
|
|
|
|
|
set => comboIndexWithOffsets.Value = value;
|
2021-07-22 21:22:42 +08:00
|
|
|
|
}
|
2021-07-20 19:15:43 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
private HitObjectProperty<bool> lastInCombo;
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
2022-07-19 12:52:12 +08:00
|
|
|
|
public Bindable<bool> LastInComboBindable => lastInCombo.Bindable;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
|
|
|
|
|
public bool LastInCombo
|
|
|
|
|
{
|
2022-07-19 12:52:12 +08:00
|
|
|
|
get => lastInCombo.Value;
|
|
|
|
|
set => lastInCombo.Value = value;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2019-10-21 15:15:41 +08:00
|
|
|
|
protected OsuHitObject()
|
|
|
|
|
{
|
|
|
|
|
StackHeightBindable.BindValueChanged(height =>
|
|
|
|
|
{
|
2024-02-17 20:46:38 +08:00
|
|
|
|
foreach (var nested in NestedHitObjects)
|
|
|
|
|
{
|
|
|
|
|
if (nested is OsuHitObject osuHitObject)
|
|
|
|
|
osuHitObject.StackHeight = height.NewValue;
|
|
|
|
|
}
|
2019-10-21 15:15:41 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-01 13:56:42 +08:00
|
|
|
|
protected override void ApplyDefaultsToSelf(ControlPointInfo controlPointInfo, IBeatmapDifficultyInfo difficulty)
|
2016-12-11 17:11:22 +08:00
|
|
|
|
{
|
2017-12-22 20:42:54 +08:00
|
|
|
|
base.ApplyDefaultsToSelf(controlPointInfo, difficulty);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2023-12-15 02:54:23 +08:00
|
|
|
|
TimePreempt = (float)IBeatmapDifficultyInfo.DifficultyRange(difficulty.ApproachRate, PREEMPT_MAX, PREEMPT_MID, PREEMPT_MIN);
|
2021-02-10 21:06:19 +08:00
|
|
|
|
|
|
|
|
|
// Preempt time can go below 450ms. Normally, this is achieved via the DT mod which uniformly speeds up all animations game wide regardless of AR.
|
|
|
|
|
// This uniform speedup is hard to match 1:1, however we can at least make AR>10 (via mods) feel good by extending the upper linear function above.
|
|
|
|
|
// Note that this doesn't exactly match the AR>10 visuals as they're classically known, but it feels good.
|
|
|
|
|
// This adjustment is necessary for AR>10, otherwise TimePreempt can become smaller leading to hitcircles not fully fading in.
|
|
|
|
|
TimeFadeIn = 400 * Math.Min(1, TimePreempt / PREEMPT_MIN);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2023-10-20 17:57:14 +08:00
|
|
|
|
Scale = LegacyRulesetExtensions.CalculateScaleFromCircleSize(difficulty.CircleSize, true);
|
2016-12-11 17:11:22 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2023-11-23 12:41:01 +08:00
|
|
|
|
public void UpdateComboInformation(IHasComboInformation? lastObj)
|
|
|
|
|
{
|
2023-11-24 09:25:23 +08:00
|
|
|
|
// Note that this implementation is shared with the osu!catch ruleset's implementation.
|
|
|
|
|
// If a change is made here, CatchHitObject.cs should also be updated.
|
2023-11-23 12:41:01 +08:00
|
|
|
|
ComboIndex = lastObj?.ComboIndex ?? 0;
|
|
|
|
|
ComboIndexWithOffsets = lastObj?.ComboIndexWithOffsets ?? 0;
|
|
|
|
|
IndexInCurrentCombo = (lastObj?.IndexInCurrentCombo + 1) ?? 0;
|
|
|
|
|
|
|
|
|
|
if (this is Spinner)
|
|
|
|
|
{
|
|
|
|
|
// For the purpose of combo colours, spinners never start a new combo even if they are flagged as doing so.
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// At decode time, the first hitobject in the beatmap and the first hitobject after a spinner are both enforced to be a new combo,
|
|
|
|
|
// but this isn't directly enforced by the editor so the extra checks against the last hitobject are duplicated here.
|
|
|
|
|
if (NewCombo || lastObj == null || lastObj is Spinner)
|
|
|
|
|
{
|
|
|
|
|
IndexInCurrentCombo = 0;
|
|
|
|
|
ComboIndex++;
|
|
|
|
|
ComboIndexWithOffsets += ComboOffset + 1;
|
|
|
|
|
|
|
|
|
|
if (lastObj != null)
|
|
|
|
|
lastObj.LastInCombo = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-17 12:35:06 +08:00
|
|
|
|
protected override HitWindows CreateHitWindows() => new OsuHitWindows();
|
2017-02-14 08:40:26 +08:00
|
|
|
|
}
|
2016-09-02 17:27:38 +08:00
|
|
|
|
}
|