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
2023-05-03 04:19:13 +00:00
#pragma warning disable 618
2017-12-01 16:43:33 +00:00
using System ;
2019-10-25 10:58:42 +00:00
using System.Collections.Generic ;
2017-12-01 16:43:33 +00:00
using System.IO ;
2018-03-09 12:23:03 +00:00
using System.Linq ;
2019-11-19 12:34:35 +00:00
using osu.Framework.Extensions ;
2021-02-25 06:38:56 +00:00
using osu.Framework.Extensions.EnumExtensions ;
2022-02-18 07:48:30 +00:00
using osu.Framework.Logging ;
2022-10-19 11:34:41 +00:00
using osu.Game.Audio ;
2017-12-01 16:43:33 +00:00
using osu.Game.Beatmaps.ControlPoints ;
2019-12-10 11:19:31 +00:00
using osu.Game.Beatmaps.Legacy ;
2020-03-07 22:08:13 +00:00
using osu.Game.Beatmaps.Timing ;
using osu.Game.IO ;
2022-02-16 08:12:57 +00:00
using osu.Game.Rulesets ;
2023-04-25 10:12:46 +00:00
using osu.Game.Rulesets.Objects ;
2020-03-07 22:08:13 +00:00
using osu.Game.Rulesets.Objects.Legacy ;
2023-04-25 10:52:21 +00:00
using osu.Game.Rulesets.Objects.Types ;
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
namespace osu.Game.Beatmaps.Formats
{
2018-03-09 12:23:03 +00:00
public class LegacyBeatmapDecoder : LegacyDecoder < Beatmap >
2017-12-01 16:43:33 +00:00
{
2022-03-24 07:43:41 +00:00
/// <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 ;
2023-04-25 10:52:21 +00:00
/// <summary>
/// A small adjustment to the start time of control points to account for rounding/precision errors.
/// </summary>
private const double control_point_leniency = 1 ;
2023-08-16 10:34:37 +00:00
internal static RulesetStore ? RulesetStore ;
2022-02-16 08:12:57 +00:00
2023-08-16 10:34:37 +00:00
private Beatmap beatmap = null ! ;
2018-04-13 09:19:50 +00:00
2023-08-16 10:34:37 +00:00
private ConvertHitObjectParser ? parser ;
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
private LegacySampleBank defaultSampleBank ;
private int defaultSampleVolume = 100 ;
2018-04-13 09:19:50 +00:00
2018-03-09 12:23:03 +00:00
public static void Register ( )
{
2019-03-13 04:56:31 +00:00
AddDecoder < Beatmap > ( @"osu file format v" , m = > new LegacyBeatmapDecoder ( Parsing . ParseInt ( m . Split ( 'v' ) . Last ( ) ) ) ) ;
2019-09-10 20:06:10 +00:00
SetFallbackDecoder < Beatmap > ( ( ) = > new LegacyBeatmapDecoder ( ) ) ;
2018-03-09 12:23:03 +00:00
}
2018-04-13 09:19:50 +00:00
2018-03-04 13:13:43 +00: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 09:19:50 +00:00
2018-05-27 18:00:21 +00:00
private readonly int offset ;
2018-04-13 09:19:50 +00:00
2018-06-08 06:26:27 +00:00
public LegacyBeatmapDecoder ( int version = LATEST_VERSION )
: base ( version )
2017-12-01 18:11:52 +00:00
{
2022-02-16 08:12:57 +00:00
if ( RulesetStore = = null )
2022-02-18 07:48:30 +00:00
{
Logger . Log ( $"A {nameof(RulesetStore)} was not provided via {nameof(Decoder)}.{nameof(RegisterDependencies)}; falling back to default {nameof(AssemblyRulesetStore)}." ) ;
RulesetStore = new AssemblyRulesetStore ( ) ;
}
2022-02-16 08:12:57 +00:00
2022-03-24 07:43:41 +00:00
offset = FormatVersion < 5 ? EARLY_VERSION_TIMING_OFFSET : 0 ;
2017-12-01 18:11:52 +00:00
}
2018-04-13 09:19:50 +00:00
2021-08-25 09:00:57 +00:00
protected override Beatmap CreateTemplateObject ( )
{
var templateBeatmap = base . CreateTemplateObject ( ) ;
templateBeatmap . ControlPointInfo = new LegacyControlPointInfo ( ) ;
return templateBeatmap ;
}
2019-09-09 22:43:30 +00:00
protected override void ParseStreamInto ( LineBufferedReader stream , Beatmap beatmap )
2017-12-01 21:05:01 +00:00
{
this . beatmap = beatmap ;
2018-03-12 02:33:12 +00:00
this . beatmap . BeatmapInfo . BeatmapVersion = FormatVersion ;
2018-04-13 09:19:50 +00:00
2022-01-27 20:41:30 +00:00
applyLegacyDefaults ( this . beatmap . BeatmapInfo ) ;
2018-03-09 12:23:03 +00:00
base . ParseStreamInto ( stream , beatmap ) ;
2018-04-13 09:19:50 +00:00
2019-10-25 10:58:42 +00:00
flushPendingPoints ( ) ;
2018-05-16 04:59:51 +00: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 04:30:48 +00:00
this . beatmap . HitObjects = this . beatmap . HitObjects . OrderBy ( h = > h . StartTime ) . ToList ( ) ;
2018-04-13 09:19:50 +00:00
2017-12-01 21:05:01 +00:00
foreach ( var hitObject in this . beatmap . HitObjects )
2023-04-25 09:34:09 +00:00
{
2023-05-03 04:30:45 +00:00
applyDefaults ( hitObject ) ;
applySamples ( hitObject ) ;
2023-04-25 09:34:09 +00:00
}
2017-12-01 21:05:01 +00:00
}
2018-04-13 09:19:50 +00:00
2023-05-03 04:30:45 +00:00
private void applyDefaults ( HitObject hitObject )
2023-04-25 10:12:46 +00:00
{
2023-05-03 04:30:45 +00:00
DifficultyControlPoint difficultyControlPoint = ( beatmap . ControlPointInfo as LegacyControlPointInfo ) ? . DifficultyPointAt ( hitObject . StartTime ) ? ? DifficultyControlPoint . DEFAULT ;
2023-05-03 04:19:13 +00:00
2023-04-25 10:12:46 +00:00
if ( difficultyControlPoint is LegacyDifficultyControlPoint legacyDifficultyControlPoint )
2023-04-26 11:10:57 +00:00
{
hitObject . LegacyBpmMultiplier = legacyDifficultyControlPoint . BpmMultiplier ;
if ( hitObject is IHasGenerateTicks hasGenerateTicks )
hasGenerateTicks . GenerateTicks = legacyDifficultyControlPoint . GenerateTicks ;
}
2023-04-25 10:12:46 +00:00
2023-04-25 10:52:21 +00:00
if ( hitObject is IHasSliderVelocity hasSliderVelocity )
2023-09-06 09:59:15 +00:00
hasSliderVelocity . SliderVelocityMultiplier = difficultyControlPoint . SliderVelocity ;
2023-04-25 10:52:21 +00:00
hitObject . ApplyDefaults ( beatmap . ControlPointInfo , beatmap . Difficulty ) ;
2023-05-03 04:30:45 +00:00
}
private void applySamples ( HitObject hitObject )
{
SampleControlPoint sampleControlPoint = ( beatmap . ControlPointInfo as LegacyControlPointInfo ) ? . SamplePointAt ( hitObject . GetEndTime ( ) + control_point_leniency ) ? ? SampleControlPoint . DEFAULT ;
2023-04-25 10:52:21 +00:00
2023-04-26 12:32:12 +00:00
hitObject . Samples = hitObject . Samples . Select ( o = > sampleControlPoint . ApplyTo ( o ) ) . ToList ( ) ;
2023-04-25 10:52:21 +00:00
2023-05-03 04:26:50 +00:00
if ( hitObject is IHasRepeats hasRepeats )
2023-04-25 10:52:21 +00:00
{
2023-05-03 04:26:50 +00:00
for ( int i = 0 ; i < hasRepeats . NodeSamples . Count ; i + + )
{
double time = hitObject . StartTime + i * hasRepeats . Duration / hasRepeats . SpanCount ( ) + control_point_leniency ;
2023-05-03 04:30:45 +00:00
var nodeSamplePoint = ( beatmap . ControlPointInfo as LegacyControlPointInfo ) ? . SamplePointAt ( time ) ? ? SampleControlPoint . DEFAULT ;
2023-04-25 10:52:21 +00:00
2023-05-03 04:30:45 +00:00
hasRepeats . NodeSamples [ i ] = hasRepeats . NodeSamples [ i ] . Select ( o = > nodeSamplePoint . ApplyTo ( o ) ) . ToList ( ) ;
2023-05-03 04:26:50 +00:00
}
2023-04-25 10:52:21 +00:00
}
2017-12-01 21:05:01 +00:00
}
2018-04-13 09:19:50 +00:00
2022-01-27 20:41:30 +00: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 ;
}
2020-02-08 17:05:27 +00:00
protected override bool ShouldSkipLine ( string line ) = > base . ShouldSkipLine ( line ) | | line . StartsWith ( ' ' ) | | line . StartsWith ( '_' ) ;
2018-04-13 09:19:50 +00:00
2018-03-09 12:23:03 +00:00
protected override void ParseLine ( Beatmap beatmap , Section section , string line )
2017-12-01 16:43:33 +00:00
{
switch ( section )
{
case Section . General :
2021-03-18 07:30:30 +00:00
handleGeneral ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case Section . Editor :
2021-03-18 07:30:30 +00:00
handleEditor ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case Section . Metadata :
handleMetadata ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case Section . Difficulty :
2021-03-18 07:30:30 +00:00
handleDifficulty ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case Section . Events :
2021-03-18 07:30:30 +00:00
handleEvent ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case Section . TimingPoints :
2021-03-18 07:30:30 +00:00
handleTimingPoint ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case Section . HitObjects :
2021-03-18 07:30:30 +00:00
handleHitObject ( line ) ;
2018-03-13 10:13:50 +00:00
return ;
2017-12-01 16:43:33 +00:00
}
2018-04-13 09:19:50 +00:00
2018-03-13 10:13:50 +00:00
base . ParseLine ( beatmap , section , line ) ;
2017-12-01 16:43:33 +00:00
}
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
private void handleGeneral ( string line )
{
2018-03-14 09:41:48 +00:00
var pair = SplitKeyVal ( line ) ;
2018-04-13 09:19:50 +00:00
2017-12-01 21:05:01 +00:00
var metadata = beatmap . BeatmapInfo . Metadata ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
switch ( pair . Key )
{
case @"AudioFilename" :
2019-12-11 08:06:56 +00:00
metadata . AudioFile = pair . Value . ToStandardisedPath ( ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"AudioLeadIn" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . AudioLeadIn = Parsing . ParseInt ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"PreviewTime" :
2019-03-13 04:56:31 +00:00
metadata . PreviewTime = getOffsetTime ( Parsing . ParseInt ( pair . Value ) ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"SampleSet" :
2022-12-26 19:36:39 +00:00
defaultSampleBank = Enum . Parse < LegacySampleBank > ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"SampleVolume" :
2019-03-13 04:56:31 +00:00
defaultSampleVolume = Parsing . ParseInt ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"StackLeniency" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . StackLeniency = Parsing . ParseFloat ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"Mode" :
2022-01-27 06:19:48 +00:00
int rulesetID = Parsing . ParseInt ( pair . Value ) ;
2018-04-13 09:19:50 +00:00
2023-08-16 10:34:37 +00:00
beatmap . BeatmapInfo . Ruleset = RulesetStore ? . GetRuleset ( rulesetID ) ? ? throw new ArgumentException ( "Ruleset is not available locally." ) ;
2022-01-27 06:19:48 +00:00
switch ( rulesetID )
2017-12-01 16:43:33 +00:00
{
case 0 :
2018-08-15 01:24:56 +00:00
parser = new Rulesets . Objects . Legacy . Osu . ConvertHitObjectParser ( getOffsetTime ( ) , FormatVersion ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case 1 :
2018-08-15 01:24:56 +00:00
parser = new Rulesets . Objects . Legacy . Taiko . ConvertHitObjectParser ( getOffsetTime ( ) , FormatVersion ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case 2 :
2018-08-15 01:24:56 +00:00
parser = new Rulesets . Objects . Legacy . Catch . ConvertHitObjectParser ( getOffsetTime ( ) , FormatVersion ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case 3 :
2018-08-15 01:24:56 +00:00
parser = new Rulesets . Objects . Legacy . Mania . ConvertHitObjectParser ( getOffsetTime ( ) , FormatVersion ) ;
2017-12-01 16:43:33 +00:00
break ;
}
2018-06-08 06:26:27 +00:00
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"LetterboxInBreaks" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . LetterboxInBreaks = Parsing . ParseInt ( pair . Value ) = = 1 ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"SpecialStyle" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . SpecialStyle = Parsing . ParseInt ( pair . Value ) = = 1 ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"WidescreenStoryboard" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . WidescreenStoryboard = Parsing . ParseInt ( pair . Value ) = = 1 ;
2017-12-01 16:43:33 +00:00
break ;
2020-10-19 21:53:41 +00:00
2020-07-20 10:36:42 +00:00
case @"EpilepsyWarning" :
beatmap . BeatmapInfo . EpilepsyWarning = Parsing . ParseInt ( pair . Value ) = = 1 ;
break ;
2021-08-24 18:53:27 +00:00
2021-09-12 14:45:27 +00:00
case @"SamplesMatchPlaybackRate" :
beatmap . BeatmapInfo . SamplesMatchPlaybackRate = Parsing . ParseInt ( pair . Value ) = = 1 ;
break ;
2021-08-24 18:53:27 +00:00
case @"Countdown" :
2022-12-26 19:38:35 +00:00
beatmap . BeatmapInfo . Countdown = Enum . Parse < CountdownType > ( pair . Value ) ;
2021-08-24 18:53:27 +00:00
break ;
case @"CountdownOffset" :
beatmap . BeatmapInfo . CountdownOffset = Parsing . ParseInt ( pair . Value ) ;
break ;
2017-12-01 16:43:33 +00:00
}
}
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
private void handleEditor ( string line )
{
2018-03-14 09:41:48 +00:00
var pair = SplitKeyVal ( line ) ;
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
switch ( pair . Key )
{
case @"Bookmarks" :
2021-11-22 16:02:22 +00:00
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 ( ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"DistanceSpacing" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . DistanceSpacing = Math . Max ( 0 , Parsing . ParseDouble ( pair . Value ) ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"BeatDivisor" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . BeatDivisor = Parsing . ParseInt ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"GridSize" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . GridSize = Parsing . ParseInt ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"TimelineZoom" :
2019-03-13 04:56:31 +00:00
beatmap . BeatmapInfo . TimelineZoom = Math . Max ( 0 , Parsing . ParseDouble ( pair . Value ) ) ;
2017-12-01 16:43:33 +00:00
break ;
}
}
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
private void handleMetadata ( string line )
{
2018-03-14 09:41:48 +00:00
var pair = SplitKeyVal ( line ) ;
2018-04-13 09:19:50 +00:00
2017-12-01 21:05:01 +00:00
var metadata = beatmap . BeatmapInfo . Metadata ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
switch ( pair . Key )
{
case @"Title" :
metadata . Title = pair . Value ;
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"TitleUnicode" :
metadata . TitleUnicode = pair . Value ;
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"Artist" :
metadata . Artist = pair . Value ;
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"ArtistUnicode" :
metadata . ArtistUnicode = pair . Value ;
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"Creator" :
2022-01-18 14:30:40 +00:00
metadata . Author . Username = pair . Value ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"Version" :
2021-11-11 08:19:53 +00:00
beatmap . BeatmapInfo . DifficultyName = pair . Value ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"Source" :
2020-01-23 15:23:53 +00:00
metadata . Source = pair . Value ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"Tags" :
2020-01-23 15:23:53 +00:00
metadata . Tags = pair . Value ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"BeatmapID" :
2021-11-12 08:45:05 +00:00
beatmap . BeatmapInfo . OnlineID = Parsing . ParseInt ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"BeatmapSetID" :
2021-11-12 08:50:31 +00:00
beatmap . BeatmapInfo . BeatmapSet = new BeatmapSetInfo { OnlineID = Parsing . ParseInt ( pair . Value ) } ;
2017-12-01 16:43:33 +00:00
break ;
}
}
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
private void handleDifficulty ( string line )
{
2018-03-14 09:41:48 +00:00
var pair = SplitKeyVal ( line ) ;
2018-04-13 09:19:50 +00:00
2021-10-02 03:34:29 +00:00
var difficulty = beatmap . Difficulty ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
switch ( pair . Key )
{
case @"HPDrainRate" :
2019-03-13 04:56:31 +00:00
difficulty . DrainRate = Parsing . ParseFloat ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"CircleSize" :
2019-03-13 04:56:31 +00:00
difficulty . CircleSize = Parsing . ParseFloat ( pair . Value ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"OverallDifficulty" :
2019-03-13 04:56:31 +00:00
difficulty . OverallDifficulty = Parsing . ParseFloat ( pair . Value ) ;
2022-01-28 09:53:28 +00:00
if ( ! hasApproachRate )
difficulty . ApproachRate = difficulty . OverallDifficulty ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"ApproachRate" :
2019-03-13 04:56:31 +00:00
difficulty . ApproachRate = Parsing . ParseFloat ( pair . Value ) ;
2022-01-28 09:53:28 +00:00
hasApproachRate = true ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"SliderMultiplier" :
2023-05-08 05:05:59 +00:00
difficulty . SliderMultiplier = Math . Clamp ( Parsing . ParseDouble ( pair . Value ) , 0.4 , 3.6 ) ;
2017-12-01 16:43:33 +00:00
break ;
2019-04-01 03:16:05 +00:00
2017-12-01 16:43:33 +00:00
case @"SliderTickRate" :
2023-05-08 05:05:59 +00:00
difficulty . SliderTickRate = Math . Clamp ( Parsing . ParseDouble ( pair . Value ) , 0.5 , 8 ) ;
2017-12-01 16:43:33 +00:00
break ;
}
}
2018-04-13 09:19:50 +00:00
2018-04-02 11:07:18 +00:00
private void handleEvent ( string line )
2017-12-01 16:43:33 +00:00
{
string [ ] split = line . Split ( ',' ) ;
2018-04-13 09:19:50 +00:00
2019-12-10 11:23:15 +00:00
if ( ! Enum . TryParse ( split [ 0 ] , out LegacyEventType type ) )
2019-08-08 05:44:04 +00:00
throw new InvalidDataException ( $@"Unknown event type: {split[0]}" ) ;
2018-04-13 09:19:50 +00:00
2017-12-01 16:43:33 +00:00
switch ( type )
{
2022-10-19 07:01:07 +00:00
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 ;
2023-03-13 09:10:16 +00:00
case LegacyEventType . Video :
string filename = CleanFilename ( split [ 2 ] ) ;
// Some very old beatmaps had incorrect type specifications for their backgrounds (ie. using 1 for VIDEO
// instead of 0 for BACKGROUND). To handle this gracefully, check the file extension against known supported
// video extensions and handle similar to a background if it doesn't match.
2023-04-21 00:35:28 +00:00
if ( ! OsuGameBase . VIDEO_EXTENSIONS . Contains ( Path . GetExtension ( filename ) . ToLowerInvariant ( ) ) )
2023-03-13 09:10:16 +00:00
{
beatmap . BeatmapInfo . Metadata . BackgroundFile = filename ;
}
break ;
2019-12-10 11:23:15 +00:00
case LegacyEventType . Background :
2020-01-24 16:05:27 +00:00
beatmap . BeatmapInfo . Metadata . BackgroundFile = CleanFilename ( split [ 2 ] ) ;
2019-08-30 20:19:34 +00:00
break ;
2019-12-10 11:23:15 +00:00
case LegacyEventType . Break :
2019-03-13 04:56:31 +00:00
double start = getOffsetTime ( Parsing . ParseDouble ( split [ 1 ] ) ) ;
2020-04-05 18:29:03 +00:00
double end = Math . Max ( start , getOffsetTime ( Parsing . ParseDouble ( split [ 2 ] ) ) ) ;
2019-03-13 04:56:31 +00:00
2020-10-09 12:04:56 +00:00
beatmap . Breaks . Add ( new BreakPeriod ( start , end ) ) ;
2017-12-01 16:43:33 +00:00
break ;
}
}
2018-04-13 09:19:50 +00:00
2018-04-02 11:07:18 +00:00
private void handleTimingPoint ( string line )
2017-12-01 16:43:33 +00:00
{
2019-08-08 05:44:04 +00:00
string [ ] split = line . Split ( ',' ) ;
double time = getOffsetTime ( Parsing . ParseDouble ( split [ 0 ] . Trim ( ) ) ) ;
2022-08-24 06:10:19 +00: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).
2022-08-23 01:44:25 +00:00
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.
2019-08-08 05:44:04 +00:00
double speedMultiplier = beatLength < 0 ? 100.0 / - beatLength : 1 ;
2022-01-22 16:27:27 +00:00
TimeSignature timeSignature = TimeSignature . SimpleQuadruple ;
2019-08-08 05:44:04 +00:00
if ( split . Length > = 3 )
2022-01-22 16:27:27 +00:00
timeSignature = split [ 2 ] [ 0 ] = = '0' ? TimeSignature . SimpleQuadruple : new TimeSignature ( Parsing . ParseInt ( split [ 2 ] ) ) ;
2019-08-08 05:44:04 +00:00
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 11:08:40 +00:00
{
2019-12-10 11:19:31 +00:00
LegacyEffectFlags effectFlags = ( LegacyEffectFlags ) Parsing . ParseInt ( split [ 7 ] ) ;
2021-02-25 06:38:56 +00:00
kiaiMode = effectFlags . HasFlagFast ( LegacyEffectFlags . Kiai ) ;
omitFirstBarSignature = effectFlags . HasFlagFast ( LegacyEffectFlags . OmitFirstBarLine ) ;
2018-04-02 11:08:40 +00:00
}
2019-08-08 05:44:04 +00:00
string stringSampleSet = sampleSet . ToString ( ) . ToLowerInvariant ( ) ;
if ( stringSampleSet = = @"none" )
2022-10-19 14:54:12 +00:00
stringSampleSet = HitSampleInfo . BANK_NORMAL ;
2019-08-08 05:44:04 +00:00
if ( timingChange )
2019-03-13 02:30:33 +00:00
{
2022-08-23 01:44:25 +00:00
if ( double . IsNaN ( beatLength ) )
throw new InvalidDataException ( "Beat length cannot be NaN in a timing control point" ) ;
2019-08-08 05:44:04 +00:00
var controlPoint = CreateTimingControlPoint ( ) ;
2019-10-25 10:58:42 +00:00
2019-08-08 05:44:04 +00:00
controlPoint . BeatLength = beatLength ;
controlPoint . TimeSignature = timeSignature ;
2023-02-28 10:29:31 +00:00
controlPoint . OmitFirstBarLine = omitFirstBarSignature ;
2019-08-08 05:44:04 +00:00
2019-10-25 10:58:42 +00:00
addControlPoint ( time , controlPoint , true ) ;
2019-03-13 02:30:33 +00:00
}
2019-10-25 10:58:42 +00:00
2022-10-13 06:05:15 +00:00
int onlineRulesetID = beatmap . BeatmapInfo . Ruleset . OnlineID ;
addControlPoint ( time , new LegacyDifficultyControlPoint ( onlineRulesetID , beatLength )
2021-09-18 13:32:08 +00:00
{
SliderVelocity = speedMultiplier ,
} , timingChange ) ;
2019-08-08 05:44:04 +00:00
2021-09-18 12:24:50 +00:00
var effectPoint = new EffectControlPoint
2019-08-08 05:44:04 +00:00
{
KiaiMode = kiaiMode ,
2021-09-18 12:24:50 +00:00
} ;
2022-05-08 12:49:42 +00:00
// osu!taiko and osu!mania use effect points rather than difficulty points for scroll speed adjustments.
if ( onlineRulesetID = = 1 | | onlineRulesetID = = 3 )
2021-09-18 12:24:50 +00:00
effectPoint . ScrollSpeed = speedMultiplier ;
addControlPoint ( time , effectPoint , timingChange ) ;
2019-08-08 05:44:04 +00:00
2019-10-25 10:58:42 +00:00
addControlPoint ( time , new LegacySampleControlPoint
2019-08-08 05:44:04 +00:00
{
SampleBank = stringSampleSet ,
SampleVolume = sampleVolume ,
CustomSampleBank = customSampleBank ,
2019-10-25 10:58:42 +00:00
} , timingChange ) ;
}
private readonly List < ControlPoint > pendingControlPoints = new List < ControlPoint > ( ) ;
2020-04-21 05:19:05 +00:00
private readonly HashSet < Type > pendingControlPointTypes = new HashSet < Type > ( ) ;
2019-10-25 10:58:42 +00:00
private double pendingControlPointsTime ;
2022-01-28 09:53:28 +00:00
private bool hasApproachRate ;
2019-10-25 10:58:42 +00:00
private void addControlPoint ( double time , ControlPoint point , bool timingChange )
{
2019-10-30 09:02:18 +00:00
if ( time ! = pendingControlPointsTime )
flushPendingPoints ( ) ;
2019-10-25 10:58:42 +00:00
if ( timingChange )
2020-04-21 05:19:05 +00:00
pendingControlPoints . Insert ( 0 , point ) ;
else
pendingControlPoints . Add ( point ) ;
2019-10-25 10:58:42 +00:00
pendingControlPointsTime = time ;
}
private void flushPendingPoints ( )
{
2020-04-21 05:19:05 +00:00
// 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 10:58:42 +00:00
pendingControlPoints . Clear ( ) ;
2020-04-21 05:19:05 +00:00
pendingControlPointTypes . Clear ( ) ;
2017-12-01 16:43:33 +00:00
}
2018-04-13 09:19:50 +00:00
2018-04-02 11:07:18 +00:00
private void handleHitObject ( string line )
2017-12-01 16:43:33 +00:00
{
// If the ruleset wasn't specified, assume the osu!standard ruleset.
2020-06-03 07:48:44 +00:00
parser ? ? = new Rulesets . Objects . Legacy . Osu . ConvertHitObjectParser ( getOffsetTime ( ) , FormatVersion ) ;
2018-04-13 09:19:50 +00:00
2018-08-15 01:24:56 +00:00
var obj = parser . Parse ( line ) ;
2021-08-25 09:00:57 +00:00
2017-12-01 16:43:33 +00:00
if ( obj ! = null )
2021-08-25 09:00:57 +00:00
{
2021-10-02 03:34:29 +00:00
obj . ApplyDefaults ( beatmap . ControlPointInfo , beatmap . Difficulty ) ;
2021-08-25 09:00:57 +00:00
2017-12-01 21:05:01 +00:00
beatmap . HitObjects . Add ( obj ) ;
2021-08-25 09:00:57 +00:00
}
2017-12-01 16:43:33 +00:00
}
2018-04-13 09:19:50 +00:00
2018-03-04 13:13:43 +00:00
private int getOffsetTime ( int time ) = > time + ( ApplyOffsets ? offset : 0 ) ;
2018-04-13 09:19:50 +00:00
2018-04-30 07:43:32 +00:00
private double getOffsetTime ( ) = > ApplyOffsets ? offset : 0 ;
2018-03-04 13:13:43 +00:00
private double getOffsetTime ( double time ) = > time + ( ApplyOffsets ? offset : 0 ) ;
2018-07-16 07:26:37 +00:00
2018-10-09 02:34:38 +00:00
protected virtual TimingControlPoint CreateTimingControlPoint ( ) = > new TimingControlPoint ( ) ;
2017-12-01 16:43:33 +00:00
}
}