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
|
|
|
|
|
2021-06-22 08:33:32 +00:00
|
|
|
|
using Newtonsoft.Json;
|
2019-09-26 07:55:08 +00:00
|
|
|
|
using osu.Framework.Bindables;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Beatmaps.ControlPoints;
|
2020-07-01 15:21:45 +00:00
|
|
|
|
using osu.Game.Rulesets.Catch.UI;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
2019-09-06 06:24:00 +00:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2021-07-14 05:38:38 +00:00
|
|
|
|
using osuTK;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Catch.Objects
|
|
|
|
|
{
|
2021-07-14 05:38:38 +00:00
|
|
|
|
public abstract class CatchHitObject : HitObject, IHasPosition, IHasComboInformation
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2020-02-17 10:16:40 +00:00
|
|
|
|
public const float OBJECT_RADIUS = 64;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-12-09 08:46:36 +00:00
|
|
|
|
public readonly Bindable<float> OriginalXBindable = new Bindable<float>();
|
2020-11-27 01:31:18 +00:00
|
|
|
|
|
2020-12-14 02:15:49 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The horizontal position of the hit object between 0 and <see cref="CatchPlayfield.WIDTH"/>.
|
|
|
|
|
/// </summary>
|
2021-06-22 08:33:32 +00:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Only setter is exposed.
|
|
|
|
|
/// Use <see cref="OriginalX"/> or <see cref="EffectiveX"/> to get the horizontal position.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
[JsonIgnore]
|
2020-12-14 04:39:07 +00:00
|
|
|
|
public float X
|
2020-12-09 08:46:36 +00:00
|
|
|
|
{
|
|
|
|
|
set => OriginalXBindable.Value = value;
|
|
|
|
|
}
|
2019-08-01 04:33:00 +00:00
|
|
|
|
|
2020-12-14 04:39:07 +00:00
|
|
|
|
public readonly Bindable<float> XOffsetBindable = new Bindable<float>();
|
2020-12-14 02:25:09 +00:00
|
|
|
|
|
2020-12-14 04:18:14 +00:00
|
|
|
|
/// <summary>
|
2020-12-14 04:39:07 +00:00
|
|
|
|
/// A random offset applied to the horizontal position, set by the beatmap processing.
|
2020-12-14 04:18:14 +00:00
|
|
|
|
/// </summary>
|
2020-12-14 04:39:07 +00:00
|
|
|
|
public float XOffset
|
2020-12-14 04:18:14 +00:00
|
|
|
|
{
|
2021-06-22 08:33:32 +00:00
|
|
|
|
get => XOffsetBindable.Value;
|
2020-12-14 04:39:07 +00:00
|
|
|
|
set => XOffsetBindable.Value = value;
|
2020-12-14 04:18:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-14 02:15:49 +00:00
|
|
|
|
/// <summary>
|
2020-12-14 04:39:07 +00:00
|
|
|
|
/// The horizontal position of the hit object between 0 and <see cref="CatchPlayfield.WIDTH"/>.
|
2020-12-14 02:15:49 +00:00
|
|
|
|
/// </summary>
|
2020-12-14 04:39:07 +00: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 08:33:32 +00:00
|
|
|
|
public float OriginalX
|
|
|
|
|
{
|
|
|
|
|
get => OriginalXBindable.Value;
|
|
|
|
|
set => OriginalXBindable.Value = value;
|
|
|
|
|
}
|
2020-11-27 01:31:18 +00:00
|
|
|
|
|
2019-08-01 04:33:00 +00:00
|
|
|
|
/// <summary>
|
2020-12-14 04:39:07 +00:00
|
|
|
|
/// The effective horizontal position of the hit object between 0 and <see cref="CatchPlayfield.WIDTH"/>.
|
2019-08-01 04:33:00 +00:00
|
|
|
|
/// </summary>
|
2020-12-14 04:39:07 +00: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>
|
2021-06-22 08:35:30 +00:00
|
|
|
|
public float EffectiveX => OriginalX + XOffset;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2021-06-22 08:33:32 +00:00
|
|
|
|
public double TimePreempt { get; set; } = 1000;
|
2019-06-28 08:34:04 +00:00
|
|
|
|
|
2020-11-27 01:31:18 +00:00
|
|
|
|
public readonly Bindable<int> IndexInBeatmapBindable = new Bindable<int>();
|
|
|
|
|
|
|
|
|
|
public int IndexInBeatmap
|
|
|
|
|
{
|
|
|
|
|
get => IndexInBeatmapBindable.Value;
|
|
|
|
|
set => IndexInBeatmapBindable.Value = value;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
public virtual bool NewCombo { get; set; }
|
|
|
|
|
|
2019-09-26 07:55:08 +00:00
|
|
|
|
public Bindable<int> IndexInCurrentComboBindable { get; } = new Bindable<int>();
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-09-26 07:55:08 +00:00
|
|
|
|
public int IndexInCurrentCombo
|
|
|
|
|
{
|
|
|
|
|
get => IndexInCurrentComboBindable.Value;
|
|
|
|
|
set => IndexInCurrentComboBindable.Value = value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Bindable<int> ComboIndexBindable { get; } = new Bindable<int>();
|
|
|
|
|
|
|
|
|
|
public int ComboIndex
|
|
|
|
|
{
|
|
|
|
|
get => ComboIndexBindable.Value;
|
|
|
|
|
set => ComboIndexBindable.Value = value;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-09-26 07:55:08 +00:00
|
|
|
|
public Bindable<bool> LastInComboBindable { get; } = new Bindable<bool>();
|
|
|
|
|
|
2018-05-21 01:58:46 +00:00
|
|
|
|
/// <summary>
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// The next fruit starts a new combo. Used for explodey.
|
|
|
|
|
/// </summary>
|
2019-09-26 07:55:08 +00:00
|
|
|
|
public virtual bool LastInCombo
|
|
|
|
|
{
|
|
|
|
|
get => LastInComboBindable.Value;
|
|
|
|
|
set => LastInComboBindable.Value = value;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-11-27 01:31:18 +00:00
|
|
|
|
public readonly Bindable<float> ScaleBindable = new Bindable<float>(1);
|
|
|
|
|
|
|
|
|
|
public float Scale
|
|
|
|
|
{
|
|
|
|
|
get => ScaleBindable.Value;
|
|
|
|
|
set => ScaleBindable.Value = value;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-12-02 08:12:30 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The seed value used for visual randomness such as fruit rotation.
|
2020-12-02 11:53:47 +00:00
|
|
|
|
/// The value is <see cref="HitObject.StartTime"/> truncated to an integer.
|
2020-12-02 08:12:30 +00:00
|
|
|
|
/// </summary>
|
2020-12-02 11:53:47 +00:00
|
|
|
|
public int RandomSeed => (int)StartTime;
|
2020-12-02 08:12:30 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
protected override void ApplyDefaultsToSelf(ControlPointInfo controlPointInfo, BeatmapDifficulty difficulty)
|
|
|
|
|
{
|
|
|
|
|
base.ApplyDefaultsToSelf(controlPointInfo, difficulty);
|
|
|
|
|
|
2019-06-28 08:34:04 +00:00
|
|
|
|
TimePreempt = (float)BeatmapDifficulty.DifficultyRange(difficulty.ApproachRate, 1800, 1200, 450);
|
|
|
|
|
|
2020-02-19 00:52:29 +00:00
|
|
|
|
Scale = (1.0f - 0.7f * (difficulty.CircleSize - 5) / 5) / 2;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
2018-05-11 06:52:51 +00:00
|
|
|
|
|
2019-10-09 10:08:31 +00:00
|
|
|
|
protected override HitWindows CreateHitWindows() => HitWindows.Empty;
|
2021-07-14 05:38:38 +00:00
|
|
|
|
|
|
|
|
|
#region Hit object conversion
|
|
|
|
|
|
|
|
|
|
// The half of the height of the osu! playfield.
|
|
|
|
|
public const float DEFAULT_LEGACY_CONVERT_Y = 192;
|
|
|
|
|
|
|
|
|
|
/// <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;
|
|
|
|
|
|
|
|
|
|
float IHasXPosition.X => OriginalX;
|
|
|
|
|
|
|
|
|
|
float IHasYPosition.Y => LegacyConvertedY;
|
|
|
|
|
|
|
|
|
|
Vector2 IHasPosition.Position => new Vector2(OriginalX, LegacyConvertedY);
|
|
|
|
|
|
|
|
|
|
#endregion
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|