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-09-18 23:29:08 +08:00
|
|
|
|
using System;
|
2021-06-22 16:33:32 +08:00
|
|
|
|
using Newtonsoft.Json;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
using osu.Framework.Bindables;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Beatmaps.ControlPoints;
|
2020-07-01 23:21:45 +08:00
|
|
|
|
using osu.Game.Rulesets.Catch.UI;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
2023-10-20 17:57:14 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Legacy;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
2019-09-06 14:24:00 +08:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2021-07-14 13:38:38 +08:00
|
|
|
|
using osuTK;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Catch.Objects
|
|
|
|
|
{
|
2024-07-23 19:30:10 +08:00
|
|
|
|
public abstract class CatchHitObject : HitObject, IHasPosition, IHasComboInformation, IHasTimePreempt
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2020-02-17 18:16:40 +08:00
|
|
|
|
public const float OBJECT_RADIUS = 64;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<float> originalX;
|
|
|
|
|
|
|
|
|
|
public Bindable<float> OriginalXBindable => originalX.Bindable;
|
2020-11-27 09:31:18 +08:00
|
|
|
|
|
2020-12-14 10:15:49 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The horizontal position of the hit object between 0 and <see cref="CatchPlayfield.WIDTH"/>.
|
|
|
|
|
/// </summary>
|
2021-06-22 16:33:32 +08:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Only setter is exposed.
|
|
|
|
|
/// Use <see cref="OriginalX"/> or <see cref="EffectiveX"/> to get the horizontal position.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
[JsonIgnore]
|
2020-12-14 12:39:07 +08:00
|
|
|
|
public float X
|
2020-12-09 16:46:36 +08:00
|
|
|
|
{
|
2022-09-19 23:26:04 +08:00
|
|
|
|
set => originalX.Value = value;
|
2020-12-09 16:46:36 +08:00
|
|
|
|
}
|
2019-08-01 12:33:00 +08:00
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<float> xOffset;
|
|
|
|
|
|
|
|
|
|
public Bindable<float> XOffsetBindable => xOffset.Bindable;
|
2020-12-14 10:25:09 +08:00
|
|
|
|
|
2020-12-14 12:18:14 +08:00
|
|
|
|
/// <summary>
|
2020-12-14 12:39:07 +08:00
|
|
|
|
/// A random offset applied to the horizontal position, set by the beatmap processing.
|
2020-12-14 12:18:14 +08:00
|
|
|
|
/// </summary>
|
2020-12-14 12:39:07 +08:00
|
|
|
|
public float XOffset
|
2020-12-14 12:18:14 +08:00
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => xOffset.Value;
|
|
|
|
|
set => xOffset.Value = value;
|
2020-12-14 12:18:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-14 10:15:49 +08:00
|
|
|
|
/// <summary>
|
2020-12-14 12:39:07 +08:00
|
|
|
|
/// The horizontal position of the hit object between 0 and <see cref="CatchPlayfield.WIDTH"/>.
|
2020-12-14 10:15:49 +08:00
|
|
|
|
/// </summary>
|
2020-12-14 12:39:07 +08:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// This value is the original <see cref="X"/> value specified in the beatmap, not affected by the beatmap processing.
|
|
|
|
|
/// Use <see cref="EffectiveX"/> for a gameplay.
|
|
|
|
|
/// </remarks>
|
2021-06-22 16:33:32 +08:00
|
|
|
|
public float OriginalX
|
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => originalX.Value;
|
|
|
|
|
set => originalX.Value = value;
|
2021-06-22 16:33:32 +08:00
|
|
|
|
}
|
2020-11-27 09:31:18 +08:00
|
|
|
|
|
2019-08-01 12:33:00 +08:00
|
|
|
|
/// <summary>
|
2020-12-14 12:39:07 +08:00
|
|
|
|
/// The effective horizontal position of the hit object between 0 and <see cref="CatchPlayfield.WIDTH"/>.
|
2019-08-01 12:33:00 +08:00
|
|
|
|
/// </summary>
|
2020-12-14 12:39:07 +08:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// This value is the original <see cref="X"/> value plus the offset applied by the beatmap processing.
|
|
|
|
|
/// Use <see cref="OriginalX"/> if a value not affected by the offset is desired.
|
|
|
|
|
/// </remarks>
|
2022-09-19 23:26:04 +08:00
|
|
|
|
public float EffectiveX => Math.Clamp(OriginalX + XOffset, 0, CatchPlayfield.WIDTH);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-06-22 16:33:32 +08:00
|
|
|
|
public double TimePreempt { get; set; } = 1000;
|
2019-06-28 16:34:04 +08:00
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<int> indexInBeatmap;
|
|
|
|
|
|
|
|
|
|
public Bindable<int> IndexInBeatmapBindable => indexInBeatmap.Bindable;
|
2020-11-27 09:31:18 +08:00
|
|
|
|
|
|
|
|
|
public int IndexInBeatmap
|
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => indexInBeatmap.Value;
|
|
|
|
|
set => indexInBeatmap.Value = value;
|
2020-11-27 09:31:18 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
public virtual bool NewCombo { get; set; }
|
|
|
|
|
|
2021-07-20 19:15:43 +08:00
|
|
|
|
public int ComboOffset { get; set; }
|
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<int> indexInCurrentCombo;
|
|
|
|
|
|
|
|
|
|
public Bindable<int> IndexInCurrentComboBindable => indexInCurrentCombo.Bindable;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2019-09-26 15:55:08 +08:00
|
|
|
|
public int IndexInCurrentCombo
|
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => indexInCurrentCombo.Value;
|
|
|
|
|
set => indexInCurrentCombo.Value = value;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<int> comboIndex;
|
|
|
|
|
|
|
|
|
|
public Bindable<int> ComboIndexBindable => comboIndex.Bindable;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
|
|
|
|
|
public int ComboIndex
|
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => comboIndex.Value;
|
|
|
|
|
set => comboIndex.Value = value;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-07-19 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<int> comboIndexWithOffsets;
|
|
|
|
|
|
|
|
|
|
public Bindable<int> ComboIndexWithOffsetsBindable => comboIndexWithOffsets.Bindable;
|
2021-07-22 21:22:42 +08:00
|
|
|
|
|
|
|
|
|
public int ComboIndexWithOffsets
|
|
|
|
|
{
|
2022-07-19 09:35:07 +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 09:35:07 +08:00
|
|
|
|
private HitObjectProperty<bool> lastInCombo;
|
|
|
|
|
|
|
|
|
|
public Bindable<bool> LastInComboBindable => lastInCombo.Bindable;
|
2019-09-26 15:55:08 +08:00
|
|
|
|
|
2018-05-21 09:58:46 +08:00
|
|
|
|
/// <summary>
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// The next fruit starts a new combo. Used for explodey.
|
|
|
|
|
/// </summary>
|
2019-09-26 15:55:08 +08:00
|
|
|
|
public virtual bool LastInCombo
|
|
|
|
|
{
|
2022-07-19 09:35:07 +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
|
|
|
|
|
2022-07-19 11:57:31 +08:00
|
|
|
|
private HitObjectProperty<float> scale = new HitObjectProperty<float>(1);
|
2022-07-19 09:35:07 +08:00
|
|
|
|
|
|
|
|
|
public Bindable<float> ScaleBindable => scale.Bindable;
|
2020-11-27 09:31:18 +08:00
|
|
|
|
|
|
|
|
|
public float Scale
|
|
|
|
|
{
|
2022-07-19 09:35:07 +08:00
|
|
|
|
get => scale.Value;
|
|
|
|
|
set => scale.Value = value;
|
2020-11-27 09:31:18 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-12-02 16:12:30 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The seed value used for visual randomness such as fruit rotation.
|
2020-12-02 19:53:47 +08:00
|
|
|
|
/// The value is <see cref="HitObject.StartTime"/> truncated to an integer.
|
2020-12-02 16:12:30 +08:00
|
|
|
|
/// </summary>
|
2020-12-02 19:53:47 +08:00
|
|
|
|
public int RandomSeed => (int)StartTime;
|
2020-12-02 16:12:30 +08:00
|
|
|
|
|
2021-10-01 13:56:42 +08:00
|
|
|
|
protected override void ApplyDefaultsToSelf(ControlPointInfo controlPointInfo, IBeatmapDifficultyInfo difficulty)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
base.ApplyDefaultsToSelf(controlPointInfo, difficulty);
|
|
|
|
|
|
2023-12-15 02:54:23 +08:00
|
|
|
|
TimePreempt = (float)IBeatmapDifficultyInfo.DifficultyRange(difficulty.ApproachRate, PREEMPT_MAX, PREEMPT_MID, PREEMPT_MIN);
|
2019-06-28 16:34:04 +08:00
|
|
|
|
|
2023-10-20 17:57:14 +08:00
|
|
|
|
Scale = LegacyRulesetExtensions.CalculateScaleFromCircleSize(difficulty.CircleSize);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
2018-05-11 14:52:51 +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! ruleset's implementation.
|
|
|
|
|
// If a change is made here, OsuHitObject.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 BananaShower)
|
|
|
|
|
{
|
|
|
|
|
// 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 banana shower 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 BananaShower)
|
|
|
|
|
{
|
|
|
|
|
IndexInCurrentCombo = 0;
|
|
|
|
|
ComboIndex++;
|
|
|
|
|
ComboIndexWithOffsets += ComboOffset + 1;
|
|
|
|
|
|
|
|
|
|
if (lastObj != null)
|
|
|
|
|
lastObj.LastInCombo = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-09 18:08:31 +08:00
|
|
|
|
protected override HitWindows CreateHitWindows() => HitWindows.Empty;
|
2021-07-14 13:38:38 +08:00
|
|
|
|
|
|
|
|
|
#region Hit object conversion
|
|
|
|
|
|
|
|
|
|
// The half of the height of the osu! playfield.
|
|
|
|
|
public const float DEFAULT_LEGACY_CONVERT_Y = 192;
|
|
|
|
|
|
2023-12-15 02:54:23 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Minimum preempt time at AR=10.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public const double PREEMPT_MIN = 450;
|
|
|
|
|
|
|
|
|
|
/// <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;
|
|
|
|
|
|
2021-07-14 13:38:38 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The Y position of the hit object is not used in the normal osu!catch gameplay.
|
|
|
|
|
/// It is preserved to maximize the backward compatibility with the legacy editor, in which the mappers use the Y position to organize the patterns.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public float LegacyConvertedY { get; set; } = DEFAULT_LEGACY_CONVERT_Y;
|
|
|
|
|
|
2024-12-27 17:56:52 +08:00
|
|
|
|
float IHasXPosition.X
|
|
|
|
|
{
|
|
|
|
|
get => OriginalX;
|
|
|
|
|
set => OriginalX = value;
|
|
|
|
|
}
|
2021-07-14 13:38:38 +08:00
|
|
|
|
|
2024-12-27 17:56:52 +08:00
|
|
|
|
float IHasYPosition.Y
|
|
|
|
|
{
|
|
|
|
|
get => LegacyConvertedY;
|
|
|
|
|
set => LegacyConvertedY = value;
|
|
|
|
|
}
|
2021-07-14 13:38:38 +08:00
|
|
|
|
|
2024-12-27 17:56:52 +08:00
|
|
|
|
Vector2 IHasPosition.Position
|
|
|
|
|
{
|
|
|
|
|
get => new Vector2(OriginalX, LegacyConvertedY);
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
((IHasXPosition)this).X = value.X;
|
|
|
|
|
((IHasYPosition)this).Y = value.Y;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-07-14 13:38:38 +08:00
|
|
|
|
|
|
|
|
|
#endregion
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
}
|