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-05-15 17:07:41 +08:00
|
|
|
|
using System.Threading;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Beatmaps.ControlPoints;
|
2018-11-29 09:56:19 +08:00
|
|
|
|
using osu.Game.Rulesets.Judgements;
|
2020-04-21 15:45:01 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
2023-12-06 14:26:32 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
2019-09-06 14:24:00 +08:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2023-12-06 14:26:32 +08:00
|
|
|
|
using osu.Game.Rulesets.Taiko.Beatmaps;
|
2020-04-21 15:45:01 +08:00
|
|
|
|
using osuTK;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Taiko.Objects
|
|
|
|
|
{
|
2023-08-14 03:47:13 +08:00
|
|
|
|
public class DrumRoll : TaikoStrongableHitObject, IHasPath
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Drum roll distance that results in a duration of 1 speed-adjusted beat length.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private const float base_distance = 100;
|
|
|
|
|
|
2020-02-05 16:12:26 +08:00
|
|
|
|
public double EndTime
|
|
|
|
|
{
|
|
|
|
|
get => StartTime + Duration;
|
|
|
|
|
set => Duration = value - StartTime;
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
public double Duration { get; set; }
|
|
|
|
|
|
2020-04-21 15:45:01 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Velocity of this <see cref="DrumRoll"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Velocity { get; private set; }
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Numer of ticks per beat length.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public int TickRate = 1;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The length (in milliseconds) between ticks of this drumroll.
|
|
|
|
|
/// <para>Half of this value is the hit window of the ticks.</para>
|
|
|
|
|
/// </summary>
|
2022-06-24 19:01:16 +08:00
|
|
|
|
private double tickSpacing = 100;
|
2018-04-13 17:19:50 +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);
|
|
|
|
|
|
|
|
|
|
TimingControlPoint timingPoint = controlPointInfo.TimingPointAt(StartTime);
|
2023-08-14 03:47:13 +08:00
|
|
|
|
EffectControlPoint effectPoint = controlPointInfo.EffectPointAt(StartTime);
|
2020-04-21 15:45:01 +08:00
|
|
|
|
|
2023-12-06 14:26:32 +08:00
|
|
|
|
double scoringDistance = base_distance * (difficulty.SliderMultiplier * TaikoBeatmapConverter.VELOCITY_MULTIPLIER) * effectPoint.ScrollSpeed;
|
2020-04-21 15:45:01 +08:00
|
|
|
|
Velocity = scoringDistance / timingPoint.BeatLength;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2023-06-06 15:54:33 +08:00
|
|
|
|
TickRate = difficulty.SliderTickRate == 3 ? 3 : 4;
|
|
|
|
|
|
2022-06-24 19:01:16 +08:00
|
|
|
|
tickSpacing = timingPoint.BeatLength / TickRate;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-15 17:07:41 +08:00
|
|
|
|
protected override void CreateNestedHitObjects(CancellationToken cancellationToken)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2022-08-05 22:30:07 +08:00
|
|
|
|
createTicks(cancellationToken);
|
2018-08-03 15:11:57 +08:00
|
|
|
|
|
2020-05-15 17:07:41 +08:00
|
|
|
|
base.CreateNestedHitObjects(cancellationToken);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-05 22:30:07 +08:00
|
|
|
|
private void createTicks(CancellationToken cancellationToken)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2022-06-24 19:01:16 +08:00
|
|
|
|
if (tickSpacing == 0)
|
2022-08-05 22:30:07 +08:00
|
|
|
|
return;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
bool first = true;
|
2019-04-01 11:16:05 +08:00
|
|
|
|
|
2022-06-24 19:01:16 +08:00
|
|
|
|
for (double t = StartTime; t < EndTime + tickSpacing / 2; t += tickSpacing)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2020-05-15 18:25:14 +08:00
|
|
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
|
|
2023-06-13 01:33:22 +08:00
|
|
|
|
AddNested(new DrumRollTick(this)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
FirstTick = first,
|
2022-06-24 19:01:16 +08:00
|
|
|
|
TickSpacing = tickSpacing,
|
2018-04-13 17:19:50 +08:00
|
|
|
|
StartTime = t,
|
2023-04-30 05:52:24 +08:00
|
|
|
|
IsStrong = IsStrong,
|
Propagate samples to drum roll/swell ticks for correct playback
In d97daee96be2c10f90709cc30beceb1f369ae225, `DrumSampleTriggerSource`
was changed such that in order to play sounds for the user's inputs, the
bank of the normal sound would always be used.
The problem is that in the case of taiko objects which have nested
objects (swells and drum rolls), the samples were not propagated fully
(drum rolls, where only the finish sample was kept, for the purposes of
determining strongability), or not propagated at all (swells) to ticks.
As ticks of both objects are valid return values of
`GetMostValidHitObject()`, this would lead to the drum making no sounds
if the next object was a drum roll or swell, until that drum roll or
swell was completed. To fix, propagate the full set of samples, so that
`DrumSampleTriggerSource` can retrieve the normal sound to copy the bank
from.
Note that this may not necessarily reproduce prior behaviour. This is
because it is not guaranteed that all realised samples for a given
hitobject have the same bank - some may have been overriden locally on a
given hitobject. Previously, the bank would have been retrieved from the
sample control point, wherein there is only one possible bank to use;
however, when deciding the sound to play on the basis of a constructed
hitobject, it is possible that there are cases wherein the hitnormal
sample was overridden on that given hitobject, and in such cases, this
PR would make samples _play_, but not necessarily the _same_ samples
as prior to #23308.
If that turns out to be the case, this will have to be revisited.
2023-05-21 22:49:17 +08:00
|
|
|
|
Samples = Samples
|
2018-04-13 17:19:50 +08:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
first = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-12-06 16:09:42 +08:00
|
|
|
|
|
2022-08-30 20:44:44 +08:00
|
|
|
|
public override Judgement CreateJudgement() => new IgnoreJudgement();
|
2019-09-02 15:10:30 +08:00
|
|
|
|
|
2019-10-09 18:08:31 +08:00
|
|
|
|
protected override HitWindows CreateHitWindows() => HitWindows.Empty;
|
2020-04-21 15:45:01 +08:00
|
|
|
|
|
2023-05-23 18:07:54 +08:00
|
|
|
|
protected override StrongNestedHitObject CreateStrongNestedHit(double startTime) => new StrongNestedHit(this)
|
2023-05-22 00:28:30 +08:00
|
|
|
|
{
|
|
|
|
|
StartTime = startTime,
|
|
|
|
|
Samples = Samples
|
|
|
|
|
};
|
2020-12-13 19:59:46 +08:00
|
|
|
|
|
|
|
|
|
public class StrongNestedHit : StrongNestedHitObject
|
|
|
|
|
{
|
2022-09-06 16:27:29 +08:00
|
|
|
|
// The strong hit of the drum roll doesn't actually provide any score.
|
|
|
|
|
public override Judgement CreateJudgement() => new IgnoreJudgement();
|
2023-05-09 18:30:54 +08:00
|
|
|
|
|
|
|
|
|
public StrongNestedHit(TaikoHitObject parent)
|
|
|
|
|
: base(parent)
|
|
|
|
|
{
|
|
|
|
|
}
|
2020-12-13 19:59:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-21 15:45:01 +08:00
|
|
|
|
#region LegacyBeatmapEncoder
|
|
|
|
|
|
|
|
|
|
double IHasDistance.Distance => Duration * Velocity;
|
|
|
|
|
|
2020-05-26 16:44:47 +08:00
|
|
|
|
SliderPath IHasPath.Path
|
2023-12-06 14:26:32 +08:00
|
|
|
|
=> new SliderPath(PathType.LINEAR, new[] { Vector2.Zero, new Vector2(1) }, ((IHasDistance)this).Distance / TaikoBeatmapConverter.VELOCITY_MULTIPLIER);
|
2020-04-21 15:45:01 +08:00
|
|
|
|
|
|
|
|
|
#endregion
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
}
|