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

200 lines
8.7 KiB
C#
Raw Normal View History

2018-01-05 19:21:19 +08:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
2017-03-11 23:34:21 +08:00
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
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.Game.IO.Serialization;
using osu.Game.Audio;
using osu.Game.Beatmaps.ControlPoints;
2017-03-11 23:34:21 +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
{
2017-04-03 19:32:03 +08:00
/// <summary>
/// osu! is generally slower than taiko, so a factor is added to increase
/// speed. This must be used everywhere slider length or beat length is used.
/// </summary>
private const float legacy_velocity_multiplier = 1.4f;
/// <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;
/// <summary>
/// Base osu! slider scoring distance.
/// </summary>
private const float osu_base_scoring_distance = 100;
2017-03-17 13:44:48 +08:00
2017-04-03 14:24:30 +08:00
/// <summary>
/// Drum roll distance that results in a duration of 1 speed-adjusted beat length.
/// </summary>
2017-04-03 19:32:03 +08:00
private const float taiko_base_distance = 100;
2017-04-03 14:24:30 +08:00
2017-08-22 13:21:28 +08:00
private readonly bool isForCurrentRuleset;
2017-04-18 13:24:16 +08:00
protected override IEnumerable<Type> ValidConversionTypes { get; } = new[] { typeof(HitObject) };
2017-04-17 14:44:46 +08:00
2017-08-22 13:18:17 +08:00
public TaikoBeatmapConverter(bool isForCurrentRuleset)
2017-03-11 23:34:21 +08:00
{
this.isForCurrentRuleset = isForCurrentRuleset;
2017-08-22 13:18:17 +08:00
}
2017-08-22 13:18:17 +08:00
protected override Beatmap<TaikoHitObject> ConvertBeatmap(Beatmap original)
{
2017-04-18 13:24:16 +08:00
// Rewrite the beatmap info to add the slider velocity multiplier
BeatmapInfo info = original.BeatmapInfo.DeepClone();
2017-10-19 13:05:11 +08:00
info.BaseDifficulty.SliderMultiplier *= legacy_velocity_multiplier;
2017-08-22 13:18:17 +08:00
Beatmap<TaikoHitObject> converted = base.ConvertBeatmap(original);
2017-04-18 13:24:16 +08:00
2017-10-14 13:28:25 +08:00
if (original.BeatmapInfo.RulesetID == 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();
if (x.Skip(1).Any())
first.IsStrong = true;
return first;
}).ToList();
}
2017-04-18 13:24:16 +08:00
return converted;
}
2017-04-18 13:24:16 +08:00
protected override IEnumerable<TaikoHitObject> ConvertHitObject(HitObject obj, Beatmap beatmap)
{
var distanceData = obj as IHasDistance;
var repeatsData = obj as IHasRepeats;
var endTimeData = obj as IHasEndTime;
var curveData = obj as IHasCurve;
2017-03-29 09:59:35 +08:00
// Old osu! used hit sounding to determine various hit type information
2017-12-25 14:35:28 +08:00
List<SampleInfo> samples = obj.Samples;
2017-03-29 09:59:35 +08:00
2017-04-06 11:14:06 +08:00
bool strong = samples.Any(s => s.Name == SampleInfo.HIT_FINISH);
if (distanceData != null)
{
int repeats = repeatsData?.RepeatCount ?? 1;
TimingControlPoint timingPoint = beatmap.ControlPointInfo.TimingPointAt(obj.StartTime);
DifficultyControlPoint difficultyPoint = beatmap.ControlPointInfo.DifficultyPointAt(obj.StartTime);
double speedAdjustment = difficultyPoint.SpeedMultiplier;
double speedAdjustedBeatLength = timingPoint.BeatLength / speedAdjustment;
// The true distance, accounting for any repeats. This ends up being the drum roll distance later
2017-04-03 19:32:03 +08:00
double distance = distanceData.Distance * repeats * legacy_velocity_multiplier;
// The velocity of the taiko hit object - calculated as the velocity of a drum roll
2017-10-19 13:05:11 +08:00
double taikoVelocity = taiko_base_distance * beatmap.BeatmapInfo.BaseDifficulty.SliderMultiplier * legacy_velocity_multiplier / speedAdjustedBeatLength;
// The duration of the taiko hit object
double taikoDuration = distance / taikoVelocity;
// For some reason, old osu! always uses speedAdjustment to determine the taiko velocity, but
// only uses it to determine osu! velocity if beatmap version < 8. Let's account for that here.
if (beatmap.BeatmapInfo.BeatmapVersion >= 8)
speedAdjustedBeatLength *= speedAdjustment;
// The velocity of the osu! hit object - calculated as the velocity of a slider
2017-10-19 13:05:11 +08:00
double osuVelocity = osu_base_scoring_distance * beatmap.BeatmapInfo.BaseDifficulty.SliderMultiplier * legacy_velocity_multiplier / speedAdjustedBeatLength;
// The duration of the osu! hit object
double osuDuration = distance / osuVelocity;
// If the drum roll is to be split into hit circles, assume the ticks are 1/8 spaced within the duration of one beat
2017-10-19 13:05:11 +08:00
double tickSpacing = Math.Min(speedAdjustedBeatLength / beatmap.BeatmapInfo.BaseDifficulty.SliderTickRate, taikoDuration / repeats);
if (!isForCurrentRuleset && tickSpacing > 0 && osuDuration < 2 * speedAdjustedBeatLength)
{
2017-12-25 14:35:28 +08:00
List<List<SampleInfo>> allSamples = curveData != null ? curveData.RepeatSamples : new List<List<SampleInfo>>(new[] { samples });
int i = 0;
for (double j = obj.StartTime; j <= obj.StartTime + taikoDuration + tickSpacing / 8; j += tickSpacing)
{
2017-12-25 14:35:28 +08:00
List<SampleInfo> currentSamples = allSamples[i];
bool isRim = currentSamples.Any(s => s.Name == SampleInfo.HIT_CLAP || s.Name == SampleInfo.HIT_WHISTLE);
strong = currentSamples.Any(s => s.Name == SampleInfo.HIT_FINISH);
if (isRim)
{
yield return new RimHit
{
StartTime = j,
Samples = currentSamples,
IsStrong = strong
};
}
else
{
yield return new CentreHit
{
StartTime = j,
Samples = currentSamples,
IsStrong = strong,
};
}
i = (i + 1) % allSamples.Count;
}
}
else
{
yield return new DrumRoll
{
StartTime = obj.StartTime,
Samples = obj.Samples,
IsStrong = strong,
2017-04-05 12:52:53 +08:00
Duration = taikoDuration,
2017-10-19 13:05:11 +08:00
TickRate = beatmap.BeatmapInfo.BaseDifficulty.SliderTickRate == 3 ? 3 : 4,
};
}
}
else if (endTimeData != null)
{
2017-10-19 13:05:11 +08:00
double hitMultiplier = BeatmapDifficulty.DifficultyRange(beatmap.BeatmapInfo.BaseDifficulty.OverallDifficulty, 3, 5, 7.5) * swell_hit_multiplier;
yield return new Swell
{
StartTime = obj.StartTime,
Samples = obj.Samples,
2017-03-28 09:02:41 +08:00
IsStrong = strong,
2017-04-05 12:52:53 +08:00
Duration = endTimeData.Duration,
RequiredHits = (int)Math.Max(1, endTimeData.Duration / 1000 * hitMultiplier),
};
}
else
2017-03-30 14:51:16 +08:00
{
2017-04-06 11:14:06 +08:00
bool isRim = samples.Any(s => s.Name == SampleInfo.HIT_CLAP || s.Name == SampleInfo.HIT_WHISTLE);
if (isRim)
2017-03-30 14:51:16 +08:00
{
yield return new RimHit
{
StartTime = obj.StartTime,
Samples = obj.Samples,
IsStrong = strong,
};
}
else
{
yield return new CentreHit
{
StartTime = obj.StartTime,
Samples = obj.Samples,
IsStrong = strong,
};
}
2017-03-30 14:51:16 +08:00
}
2017-03-11 23:34:21 +08:00
}
}
}