1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-10 19:43:10 +08:00
osu-lazer/osu.Game.Rulesets.Taiko/Beatmaps/TaikoBeatmapConverter.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

226 lines
9.6 KiB
C#
Raw Normal View History

// 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
2017-03-11 23:34:21 +08:00
using osu.Game.Beatmaps;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.Objects.Types;
using osu.Game.Rulesets.Taiko.Objects;
using System;
2017-03-11 23:34:21 +08:00
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Utils;
using System.Threading;
using osu.Game.Audio;
using osu.Game.Beatmaps.ControlPoints;
using osu.Game.Rulesets.Objects.Legacy;
2018-04-13 17:19:50 +08:00
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Taiko.Beatmaps
2017-03-11 23:34:21 +08:00
{
internal class TaikoBeatmapConverter : BeatmapConverter<TaikoHitObject>
2017-03-11 23:34:21 +08:00
{
2023-12-06 14:26:32 +08:00
/// <summary>
/// A speed multiplier applied globally to osu!taiko.
/// </summary>
/// <remarks>
/// osu! is generally slower than taiko, so a factor was historically added to increase speed for converts.
/// This must be used everywhere slider length or beat length is used in taiko.
///
/// Of note, this has never been exposed to the end user, and is considered a hidden internal multiplier.
/// </remarks>
public const float VELOCITY_MULTIPLIER = 1.4f;
2017-04-03 19:32:03 +08:00
/// <summary>
/// Because swells are easier in taiko than spinners are in osu!,
/// legacy taiko multiplies a factor when converting the number of required hits.
/// </summary>
private const float swell_hit_multiplier = 1.65f;
2018-04-13 17:19:50 +08:00
2017-04-03 19:32:03 +08:00
/// <summary>
/// Base osu! slider scoring distance.
/// </summary>
private const float osu_base_scoring_distance = 100;
2018-04-13 17:19:50 +08:00
2017-08-22 13:21:28 +08:00
private readonly bool isForCurrentRuleset;
2018-04-13 17:19:50 +08:00
public TaikoBeatmapConverter(IBeatmap beatmap, Ruleset ruleset)
: base(beatmap, ruleset)
2017-03-11 23:34:21 +08:00
{
isForCurrentRuleset = beatmap.BeatmapInfo.Ruleset.Equals(ruleset.RulesetInfo);
2017-08-22 13:18:17 +08:00
}
2018-04-13 17:19:50 +08:00
public override bool CanConvert() => true;
protected override Beatmap<TaikoHitObject> ConvertBeatmap(IBeatmap original, CancellationToken cancellationToken)
2017-08-22 13:18:17 +08:00
{
Beatmap<TaikoHitObject> converted = base.ConvertBeatmap(original, cancellationToken);
2018-04-13 17:19:50 +08:00
2022-10-27 13:25:50 +08:00
if (original.BeatmapInfo.Ruleset.OnlineID == 0)
{
// Post processing step to transform standard slider velocity changes into scroll speed changes
double lastScrollSpeed = 1;
foreach (HitObject hitObject in original.HitObjects)
{
if (hitObject is not IHasSliderVelocity hasSliderVelocity) continue;
double nextScrollSpeed = hasSliderVelocity.SliderVelocityMultiplier;
EffectControlPoint currentEffectPoint = converted.ControlPointInfo.EffectPointAt(hitObject.StartTime);
2022-10-27 13:25:50 +08:00
if (!Precision.AlmostEquals(lastScrollSpeed, nextScrollSpeed, acceptableDifference: currentEffectPoint.ScrollSpeedBindable.Precision))
2022-10-27 13:25:50 +08:00
{
converted.ControlPointInfo.Add(hitObject.StartTime, new EffectControlPoint
{
KiaiMode = currentEffectPoint.KiaiMode,
ScrollSpeed = lastScrollSpeed = nextScrollSpeed,
});
2022-10-27 13:25:50 +08:00
}
}
}
if (original.BeatmapInfo.Ruleset.OnlineID == 3)
2017-03-11 23:34:21 +08:00
{
// Post processing step to transform mania hit objects with the same start time into strong hits
converted.HitObjects = converted.HitObjects.GroupBy(t => t.StartTime).Select(x =>
{
TaikoHitObject first = x.First();
2020-12-15 04:46:02 +08:00
if (x.Skip(1).Any() && first is TaikoStrongableHitObject strong)
strong.IsStrong = true;
return first;
}).ToList();
}
2018-04-13 17:19:50 +08:00
// TODO: stable makes the last tick of a drumroll non-required when the next object is too close.
// This probably needs to be reimplemented:
//
// List<HitObject> hitobjects = hitObjectManager.hitObjects;
// int ind = hitobjects.IndexOf(this);
// if (i < hitobjects.Count - 1 && hitobjects[i + 1].HittableStartTime - (EndTime + (int)TickSpacing) <= (int)TickSpacing)
// lastTickHittable = false;
2017-04-18 13:24:16 +08:00
return converted;
}
2018-04-13 17:19:50 +08:00
protected override IEnumerable<TaikoHitObject> ConvertHitObject(HitObject obj, IBeatmap beatmap, CancellationToken cancellationToken)
{
2017-03-29 09:59:35 +08:00
// Old osu! used hit sounding to determine various hit type information
2019-11-08 13:04:57 +08:00
IList<HitSampleInfo> samples = obj.Samples;
2018-04-13 17:19:50 +08:00
2019-11-12 18:16:51 +08:00
switch (obj)
{
case IHasPath pathData:
2019-11-12 18:16:51 +08:00
{
if (shouldConvertSliderToHits(obj, beatmap, pathData, out int taikoDuration, out double tickSpacing))
{
IList<IList<HitSampleInfo>> allSamples = obj is IHasPathWithRepeats curveData ? curveData.NodeSamples : new List<IList<HitSampleInfo>>(new[] { samples });
2018-04-13 17:19:50 +08:00
2019-11-12 18:16:51 +08:00
int i = 0;
for (double j = obj.StartTime; j <= obj.StartTime + taikoDuration + tickSpacing / 8; j += tickSpacing)
{
2019-11-12 18:16:51 +08:00
IList<HitSampleInfo> currentSamples = allSamples[i];
yield return new Hit
{
StartTime = j,
Samples = currentSamples,
};
2019-11-12 18:16:51 +08:00
i = (i + 1) % allSamples.Count;
if (Precision.AlmostEquals(0, tickSpacing))
break;
}
}
2019-11-12 18:16:51 +08:00
else
{
2019-11-12 18:16:51 +08:00
yield return new DrumRoll
{
StartTime = obj.StartTime,
Samples = obj.Samples,
Duration = taikoDuration,
};
}
break;
}
2018-04-13 17:19:50 +08:00
2020-05-27 11:38:39 +08:00
case IHasDuration endTimeData:
{
double hitMultiplier = IBeatmapDifficultyInfo.DifficultyRange(beatmap.Difficulty.OverallDifficulty, 3, 5, 7.5) * swell_hit_multiplier;
2018-04-13 17:19:50 +08:00
2019-11-12 18:16:51 +08:00
yield return new Swell
{
StartTime = obj.StartTime,
Samples = obj.Samples,
2019-11-12 18:16:51 +08:00
Duration = endTimeData.Duration,
RequiredHits = (int)Math.Max(1, endTimeData.Duration / 1000 * hitMultiplier)
};
2019-11-12 18:16:51 +08:00
break;
}
2019-11-12 18:16:51 +08:00
default:
{
yield return new Hit
2019-11-12 18:16:51 +08:00
{
StartTime = obj.StartTime,
Samples = samples,
};
2019-11-12 18:16:51 +08:00
break;
}
2017-03-30 14:51:16 +08:00
}
2017-03-11 23:34:21 +08:00
}
2018-05-07 09:51:30 +08:00
private bool shouldConvertSliderToHits(HitObject obj, IBeatmap beatmap, IHasPath pathData, out int taikoDuration, out double tickSpacing)
{
// DO NOT CHANGE OR REFACTOR ANYTHING IN HERE WITHOUT TESTING AGAINST _ALL_ BEATMAPS.
// Some of these calculations look redundant, but they are not - extremely small floating point errors are introduced to maintain 1:1 compatibility with stable.
// Rounding cannot be used as an alternative since the error deltas have been observed to be between 1e-2 and 1e-6.
// The true distance, accounting for any repeats. This ends up being the drum roll distance later
int spans = (obj as IHasRepeats)?.SpanCount() ?? 1;
double distance = pathData.Path.ExpectedDistance.Value ?? 0;
// Do not combine the following two lines!
2023-12-06 14:26:32 +08:00
distance *= VELOCITY_MULTIPLIER;
distance *= spans;
TimingControlPoint timingPoint = beatmap.ControlPointInfo.TimingPointAt(obj.StartTime);
double beatLength;
if (obj is IHasSliderVelocity hasSliderVelocity)
2023-09-15 17:13:04 +08:00
beatLength = LegacyRulesetExtensions.GetPrecisionAdjustedBeatLength(hasSliderVelocity, timingPoint, TaikoRuleset.SHORT_NAME);
else
beatLength = timingPoint.BeatLength;
double sliderScoringPointDistance = osu_base_scoring_distance * (beatmap.Difficulty.SliderMultiplier * VELOCITY_MULTIPLIER) / beatmap.Difficulty.SliderTickRate;
// The velocity and duration of the taiko hit object - calculated as the velocity of a drum roll.
double taikoVelocity = sliderScoringPointDistance * beatmap.Difficulty.SliderTickRate;
taikoDuration = (int)(distance / taikoVelocity * beatLength);
if (isForCurrentRuleset)
{
tickSpacing = 0;
return false;
}
double osuVelocity = taikoVelocity * (1000f / beatLength);
// osu-stable always uses the speed-adjusted beatlength to determine the osu! velocity, but only uses it for conversion if beatmap version < 8
if (beatmap.BeatmapInfo.BeatmapVersion >= 8)
beatLength = timingPoint.BeatLength;
// If the drum roll is to be split into hit circles, assume the ticks are 1/8 spaced within the duration of one beat
tickSpacing = Math.Min(beatLength / beatmap.Difficulty.SliderTickRate, (double)taikoDuration / spans);
return tickSpacing > 0
&& distance / osuVelocity * 1000 < 2 * beatLength;
}
2018-05-07 09:51:30 +08:00
protected override Beatmap<TaikoHitObject> CreateBeatmap() => new TaikoBeatmap();
2017-03-11 23:34:21 +08:00
}
}