1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-06 02:37:25 +08:00
osu-lazer/osu.Game/Beatmaps/Formats/LegacyBeatmapDecoder.cs

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

526 lines
19 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
2022-06-17 15:37:17 +08:00
#nullable disable
using System;
2019-10-25 18:58:42 +08:00
using System.Collections.Generic;
using System.IO;
2018-03-09 20:23:03 +08:00
using System.Linq;
2019-11-19 20:34:35 +08:00
using osu.Framework.Extensions;
2021-02-25 14:38:56 +08:00
using osu.Framework.Extensions.EnumExtensions;
using osu.Framework.Logging;
using osu.Game.Beatmaps.ControlPoints;
2019-12-10 19:19:31 +08:00
using osu.Game.Beatmaps.Legacy;
using osu.Game.Beatmaps.Timing;
using osu.Game.IO;
using osu.Game.Rulesets;
using osu.Game.Rulesets.Objects.Legacy;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Beatmaps.Formats
{
2018-03-09 20:23:03 +08:00
public class LegacyBeatmapDecoder : LegacyDecoder<Beatmap>
{
/// <summary>
/// An offset which needs to be applied to old beatmaps (v4 and lower) to correct timing changes that were applied at a game client level.
/// </summary>
public const int EARLY_VERSION_TIMING_OFFSET = 24;
internal static RulesetStore RulesetStore;
private Beatmap beatmap;
2018-04-13 17:19:50 +08:00
private ConvertHitObjectParser parser;
2018-04-13 17:19:50 +08:00
private LegacySampleBank defaultSampleBank;
private int defaultSampleVolume = 100;
2018-04-13 17:19:50 +08:00
2018-03-09 20:23:03 +08:00
public static void Register()
{
AddDecoder<Beatmap>(@"osu file format v", m => new LegacyBeatmapDecoder(Parsing.ParseInt(m.Split('v').Last())));
SetFallbackDecoder<Beatmap>(() => new LegacyBeatmapDecoder());
2018-03-09 20:23:03 +08:00
}
2018-04-13 17:19:50 +08:00
2018-03-04 21:13:43 +08:00
/// <summary>
/// Whether or not beatmap or runtime offsets should be applied. Defaults on; only disable for testing purposes.
/// </summary>
public bool ApplyOffsets = true;
2018-04-13 17:19:50 +08:00
private readonly int offset;
2018-04-13 17:19:50 +08:00
public LegacyBeatmapDecoder(int version = LATEST_VERSION)
: base(version)
{
if (RulesetStore == null)
{
Logger.Log($"A {nameof(RulesetStore)} was not provided via {nameof(Decoder)}.{nameof(RegisterDependencies)}; falling back to default {nameof(AssemblyRulesetStore)}.");
RulesetStore = new AssemblyRulesetStore();
}
offset = FormatVersion < 5 ? EARLY_VERSION_TIMING_OFFSET : 0;
}
2018-04-13 17:19:50 +08:00
protected override Beatmap CreateTemplateObject()
{
var templateBeatmap = base.CreateTemplateObject();
templateBeatmap.ControlPointInfo = new LegacyControlPointInfo();
return templateBeatmap;
}
protected override void ParseStreamInto(LineBufferedReader stream, Beatmap beatmap)
{
this.beatmap = beatmap;
2018-03-12 10:33:12 +08:00
this.beatmap.BeatmapInfo.BeatmapVersion = FormatVersion;
2018-04-13 17:19:50 +08:00
applyLegacyDefaults(this.beatmap.BeatmapInfo);
2018-03-09 20:23:03 +08:00
base.ParseStreamInto(stream, beatmap);
2018-04-13 17:19:50 +08:00
2019-10-25 18:58:42 +08:00
flushPendingPoints();
2018-05-16 12:59:51 +08:00
// Objects may be out of order *only* if a user has manually edited an .osu file.
// Unfortunately there are ranked maps in this state (example: https://osu.ppy.sh/s/594828).
// OrderBy is used to guarantee that the parsing order of hitobjects with equal start times is maintained (stably-sorted)
// The parsing order of hitobjects matters in mania difficulty calculation
2018-05-16 12:30:48 +08:00
this.beatmap.HitObjects = this.beatmap.HitObjects.OrderBy(h => h.StartTime).ToList();
2018-04-13 17:19:50 +08:00
foreach (var hitObject in this.beatmap.HitObjects)
hitObject.ApplyDefaults(this.beatmap.ControlPointInfo, this.beatmap.Difficulty);
}
2018-04-13 17:19:50 +08:00
/// <summary>
/// Some `BeatmapInfo` members have default values that differ from the default values used by stable.
/// In addition, legacy beatmaps will sometimes not contain some configuration keys, in which case
/// the legacy default values should be used.
/// This method's intention is to restore those legacy defaults.
/// See also: https://osu.ppy.sh/wiki/en/Client/File_formats/Osu_%28file_format%29
/// </summary>
private void applyLegacyDefaults(BeatmapInfo beatmapInfo)
{
beatmapInfo.WidescreenStoryboard = false;
beatmapInfo.SamplesMatchPlaybackRate = false;
}
protected override bool ShouldSkipLine(string line) => base.ShouldSkipLine(line) || line.StartsWith(' ') || line.StartsWith('_');
2018-04-13 17:19:50 +08:00
2018-03-09 20:23:03 +08:00
protected override void ParseLine(Beatmap beatmap, Section section, string line)
{
switch (section)
{
case Section.General:
handleGeneral(line);
2018-03-13 18:13:50 +08:00
return;
2019-04-01 11:16:05 +08:00
case Section.Editor:
handleEditor(line);
2018-03-13 18:13:50 +08:00
return;
2019-04-01 11:16:05 +08:00
case Section.Metadata:
handleMetadata(line);
2018-03-13 18:13:50 +08:00
return;
2019-04-01 11:16:05 +08:00
case Section.Difficulty:
handleDifficulty(line);
2018-03-13 18:13:50 +08:00
return;
2019-04-01 11:16:05 +08:00
case Section.Events:
handleEvent(line);
2018-03-13 18:13:50 +08:00
return;
2019-04-01 11:16:05 +08:00
case Section.TimingPoints:
handleTimingPoint(line);
2018-03-13 18:13:50 +08:00
return;
2019-04-01 11:16:05 +08:00
case Section.HitObjects:
handleHitObject(line);
2018-03-13 18:13:50 +08:00
return;
}
2018-04-13 17:19:50 +08:00
2018-03-13 18:13:50 +08:00
base.ParseLine(beatmap, section, line);
}
2018-04-13 17:19:50 +08:00
private void handleGeneral(string line)
{
2018-03-14 17:41:48 +08:00
var pair = SplitKeyVal(line);
2018-04-13 17:19:50 +08:00
var metadata = beatmap.BeatmapInfo.Metadata;
2019-04-01 11:16:05 +08:00
switch (pair.Key)
{
case @"AudioFilename":
2019-12-11 16:06:56 +08:00
metadata.AudioFile = pair.Value.ToStandardisedPath();
break;
2019-04-01 11:16:05 +08:00
case @"AudioLeadIn":
beatmap.BeatmapInfo.AudioLeadIn = Parsing.ParseInt(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"PreviewTime":
metadata.PreviewTime = getOffsetTime(Parsing.ParseInt(pair.Value));
break;
2019-04-01 11:16:05 +08:00
case @"SampleSet":
defaultSampleBank = (LegacySampleBank)Enum.Parse(typeof(LegacySampleBank), pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"SampleVolume":
defaultSampleVolume = Parsing.ParseInt(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"StackLeniency":
beatmap.BeatmapInfo.StackLeniency = Parsing.ParseFloat(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"Mode":
int rulesetID = Parsing.ParseInt(pair.Value);
2018-04-13 17:19:50 +08:00
beatmap.BeatmapInfo.Ruleset = RulesetStore.GetRuleset(rulesetID) ?? throw new ArgumentException("Ruleset is not available locally.");
switch (rulesetID)
{
case 0:
parser = new Rulesets.Objects.Legacy.Osu.ConvertHitObjectParser(getOffsetTime(), FormatVersion);
break;
2019-04-01 11:16:05 +08:00
case 1:
parser = new Rulesets.Objects.Legacy.Taiko.ConvertHitObjectParser(getOffsetTime(), FormatVersion);
break;
2019-04-01 11:16:05 +08:00
case 2:
parser = new Rulesets.Objects.Legacy.Catch.ConvertHitObjectParser(getOffsetTime(), FormatVersion);
break;
2019-04-01 11:16:05 +08:00
case 3:
parser = new Rulesets.Objects.Legacy.Mania.ConvertHitObjectParser(getOffsetTime(), FormatVersion);
break;
}
break;
2019-04-01 11:16:05 +08:00
case @"LetterboxInBreaks":
beatmap.BeatmapInfo.LetterboxInBreaks = Parsing.ParseInt(pair.Value) == 1;
break;
2019-04-01 11:16:05 +08:00
case @"SpecialStyle":
beatmap.BeatmapInfo.SpecialStyle = Parsing.ParseInt(pair.Value) == 1;
break;
2019-04-01 11:16:05 +08:00
case @"WidescreenStoryboard":
beatmap.BeatmapInfo.WidescreenStoryboard = Parsing.ParseInt(pair.Value) == 1;
break;
2020-10-20 05:53:41 +08:00
2020-07-20 18:36:42 +08:00
case @"EpilepsyWarning":
beatmap.BeatmapInfo.EpilepsyWarning = Parsing.ParseInt(pair.Value) == 1;
break;
case @"SamplesMatchPlaybackRate":
beatmap.BeatmapInfo.SamplesMatchPlaybackRate = Parsing.ParseInt(pair.Value) == 1;
break;
case @"Countdown":
beatmap.BeatmapInfo.Countdown = (CountdownType)Enum.Parse(typeof(CountdownType), pair.Value);
break;
case @"CountdownOffset":
beatmap.BeatmapInfo.CountdownOffset = Parsing.ParseInt(pair.Value);
break;
}
}
2018-04-13 17:19:50 +08:00
private void handleEditor(string line)
{
2018-03-14 17:41:48 +08:00
var pair = SplitKeyVal(line);
2018-04-13 17:19:50 +08:00
switch (pair.Key)
{
case @"Bookmarks":
beatmap.BeatmapInfo.Bookmarks = pair.Value.Split(',').Select(v =>
{
bool result = int.TryParse(v, out int val);
return new { result, val };
}).Where(p => p.result).Select(p => p.val).ToArray();
break;
2019-04-01 11:16:05 +08:00
case @"DistanceSpacing":
beatmap.BeatmapInfo.DistanceSpacing = Math.Max(0, Parsing.ParseDouble(pair.Value));
break;
2019-04-01 11:16:05 +08:00
case @"BeatDivisor":
beatmap.BeatmapInfo.BeatDivisor = Parsing.ParseInt(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"GridSize":
beatmap.BeatmapInfo.GridSize = Parsing.ParseInt(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"TimelineZoom":
beatmap.BeatmapInfo.TimelineZoom = Math.Max(0, Parsing.ParseDouble(pair.Value));
break;
}
}
2018-04-13 17:19:50 +08:00
private void handleMetadata(string line)
{
2018-03-14 17:41:48 +08:00
var pair = SplitKeyVal(line);
2018-04-13 17:19:50 +08:00
var metadata = beatmap.BeatmapInfo.Metadata;
2019-04-01 11:16:05 +08:00
switch (pair.Key)
{
case @"Title":
metadata.Title = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"TitleUnicode":
metadata.TitleUnicode = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"Artist":
metadata.Artist = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"ArtistUnicode":
metadata.ArtistUnicode = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"Creator":
2022-01-18 22:30:40 +08:00
metadata.Author.Username = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"Version":
beatmap.BeatmapInfo.DifficultyName = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"Source":
metadata.Source = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"Tags":
metadata.Tags = pair.Value;
break;
2019-04-01 11:16:05 +08:00
case @"BeatmapID":
beatmap.BeatmapInfo.OnlineID = Parsing.ParseInt(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"BeatmapSetID":
beatmap.BeatmapInfo.BeatmapSet = new BeatmapSetInfo { OnlineID = Parsing.ParseInt(pair.Value) };
break;
}
}
2018-04-13 17:19:50 +08:00
private void handleDifficulty(string line)
{
2018-03-14 17:41:48 +08:00
var pair = SplitKeyVal(line);
2018-04-13 17:19:50 +08:00
var difficulty = beatmap.Difficulty;
2019-04-01 11:16:05 +08:00
switch (pair.Key)
{
case @"HPDrainRate":
difficulty.DrainRate = Parsing.ParseFloat(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"CircleSize":
difficulty.CircleSize = Parsing.ParseFloat(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"OverallDifficulty":
difficulty.OverallDifficulty = Parsing.ParseFloat(pair.Value);
if (!hasApproachRate)
difficulty.ApproachRate = difficulty.OverallDifficulty;
break;
2019-04-01 11:16:05 +08:00
case @"ApproachRate":
difficulty.ApproachRate = Parsing.ParseFloat(pair.Value);
hasApproachRate = true;
break;
2019-04-01 11:16:05 +08:00
case @"SliderMultiplier":
difficulty.SliderMultiplier = Parsing.ParseDouble(pair.Value);
break;
2019-04-01 11:16:05 +08:00
case @"SliderTickRate":
difficulty.SliderTickRate = Parsing.ParseDouble(pair.Value);
break;
}
}
2018-04-13 17:19:50 +08:00
2018-04-02 19:07:18 +08:00
private void handleEvent(string line)
{
string[] split = line.Split(',');
2018-04-13 17:19:50 +08:00
2019-12-10 19:23:15 +08:00
if (!Enum.TryParse(split[0], out LegacyEventType type))
throw new InvalidDataException($@"Unknown event type: {split[0]}");
2018-04-13 17:19:50 +08:00
switch (type)
{
case LegacyEventType.Sprite:
// Generally, the background is the first thing defined in a beatmap file.
// In some older beatmaps, it is not present and replaced by a storyboard-level background instead.
// Allow the first sprite (by file order) to act as the background in such cases.
if (string.IsNullOrEmpty(beatmap.BeatmapInfo.Metadata.BackgroundFile))
beatmap.BeatmapInfo.Metadata.BackgroundFile = CleanFilename(split[3]);
break;
2019-12-10 19:23:15 +08:00
case LegacyEventType.Background:
2020-01-25 00:05:27 +08:00
beatmap.BeatmapInfo.Metadata.BackgroundFile = CleanFilename(split[2]);
2019-08-31 04:19:34 +08:00
break;
2019-12-10 19:23:15 +08:00
case LegacyEventType.Break:
double start = getOffsetTime(Parsing.ParseDouble(split[1]));
double end = Math.Max(start, getOffsetTime(Parsing.ParseDouble(split[2])));
2020-10-09 20:04:56 +08:00
beatmap.Breaks.Add(new BreakPeriod(start, end));
break;
}
}
2018-04-13 17:19:50 +08:00
2018-04-02 19:07:18 +08:00
private void handleTimingPoint(string line)
{
string[] split = line.Split(',');
double time = getOffsetTime(Parsing.ParseDouble(split[0].Trim()));
2022-08-24 14:10:19 +08:00
// beatLength is allowed to be NaN to handle an edge case in which some beatmaps use NaN slider velocity to disable slider tick generation (see LegacyDifficultyControlPoint).
double beatLength = Parsing.ParseDouble(split[1].Trim(), allowNaN: true);
// If beatLength is NaN, speedMultiplier should still be 1 because all comparisons against NaN are false.
double speedMultiplier = beatLength < 0 ? 100.0 / -beatLength : 1;
TimeSignature timeSignature = TimeSignature.SimpleQuadruple;
if (split.Length >= 3)
timeSignature = split[2][0] == '0' ? TimeSignature.SimpleQuadruple : new TimeSignature(Parsing.ParseInt(split[2]));
LegacySampleBank sampleSet = defaultSampleBank;
if (split.Length >= 4)
sampleSet = (LegacySampleBank)Parsing.ParseInt(split[3]);
int customSampleBank = 0;
if (split.Length >= 5)
customSampleBank = Parsing.ParseInt(split[4]);
int sampleVolume = defaultSampleVolume;
if (split.Length >= 6)
sampleVolume = Parsing.ParseInt(split[5]);
bool timingChange = true;
if (split.Length >= 7)
timingChange = split[6][0] == '1';
bool kiaiMode = false;
bool omitFirstBarSignature = false;
if (split.Length >= 8)
2018-04-02 19:08:40 +08:00
{
2019-12-10 19:19:31 +08:00
LegacyEffectFlags effectFlags = (LegacyEffectFlags)Parsing.ParseInt(split[7]);
2021-02-25 14:38:56 +08:00
kiaiMode = effectFlags.HasFlagFast(LegacyEffectFlags.Kiai);
omitFirstBarSignature = effectFlags.HasFlagFast(LegacyEffectFlags.OmitFirstBarLine);
2018-04-02 19:08:40 +08:00
}
string stringSampleSet = sampleSet.ToString().ToLowerInvariant();
if (stringSampleSet == @"none")
stringSampleSet = @"normal";
if (timingChange)
2019-03-13 10:30:33 +08:00
{
if (double.IsNaN(beatLength))
throw new InvalidDataException("Beat length cannot be NaN in a timing control point");
var controlPoint = CreateTimingControlPoint();
2019-10-25 18:58:42 +08:00
controlPoint.BeatLength = beatLength;
controlPoint.TimeSignature = timeSignature;
2019-10-25 18:58:42 +08:00
addControlPoint(time, controlPoint, true);
2019-03-13 10:30:33 +08:00
}
2019-10-25 18:58:42 +08:00
int onlineRulesetID = beatmap.BeatmapInfo.Ruleset.OnlineID;
#pragma warning disable 618
addControlPoint(time, new LegacyDifficultyControlPoint(onlineRulesetID, beatLength)
#pragma warning restore 618
{
SliderVelocity = speedMultiplier,
}, timingChange);
var effectPoint = new EffectControlPoint
{
KiaiMode = kiaiMode,
OmitFirstBarLine = omitFirstBarSignature,
};
// osu!taiko and osu!mania use effect points rather than difficulty points for scroll speed adjustments.
if (onlineRulesetID == 1 || onlineRulesetID == 3)
effectPoint.ScrollSpeed = speedMultiplier;
addControlPoint(time, effectPoint, timingChange);
2019-10-25 18:58:42 +08:00
addControlPoint(time, new LegacySampleControlPoint
{
SampleBank = stringSampleSet,
SampleVolume = sampleVolume,
CustomSampleBank = customSampleBank,
2019-10-25 18:58:42 +08:00
}, timingChange);
}
private readonly List<ControlPoint> pendingControlPoints = new List<ControlPoint>();
private readonly HashSet<Type> pendingControlPointTypes = new HashSet<Type>();
2019-10-25 18:58:42 +08:00
private double pendingControlPointsTime;
private bool hasApproachRate;
2019-10-25 18:58:42 +08:00
private void addControlPoint(double time, ControlPoint point, bool timingChange)
{
if (time != pendingControlPointsTime)
flushPendingPoints();
2019-10-25 18:58:42 +08:00
if (timingChange)
pendingControlPoints.Insert(0, point);
else
pendingControlPoints.Add(point);
2019-10-25 18:58:42 +08:00
pendingControlPointsTime = time;
}
private void flushPendingPoints()
{
// Changes from non-timing-points are added to the end of the list (see addControlPoint()) and should override any changes from timing-points (added to the start of the list).
for (int i = pendingControlPoints.Count - 1; i >= 0; i--)
{
var type = pendingControlPoints[i].GetType();
if (pendingControlPointTypes.Contains(type))
continue;
pendingControlPointTypes.Add(type);
beatmap.ControlPointInfo.Add(pendingControlPointsTime, pendingControlPoints[i]);
}
2019-10-25 18:58:42 +08:00
pendingControlPoints.Clear();
pendingControlPointTypes.Clear();
}
2018-04-13 17:19:50 +08:00
2018-04-02 19:07:18 +08:00
private void handleHitObject(string line)
{
// If the ruleset wasn't specified, assume the osu!standard ruleset.
parser ??= new Rulesets.Objects.Legacy.Osu.ConvertHitObjectParser(getOffsetTime(), FormatVersion);
2018-04-13 17:19:50 +08:00
var obj = parser.Parse(line);
if (obj != null)
{
obj.ApplyDefaults(beatmap.ControlPointInfo, beatmap.Difficulty);
beatmap.HitObjects.Add(obj);
}
}
2018-04-13 17:19:50 +08:00
2018-03-04 21:13:43 +08:00
private int getOffsetTime(int time) => time + (ApplyOffsets ? offset : 0);
2018-04-13 17:19:50 +08:00
private double getOffsetTime() => ApplyOffsets ? offset : 0;
2018-03-04 21:13:43 +08:00
private double getOffsetTime(double time) => time + (ApplyOffsets ? offset : 0);
2018-07-16 15:26:37 +08:00
protected virtual TimingControlPoint CreateTimingControlPoint() => new TimingControlPoint();
}
}