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
2022-06-17 15:38:35 +08:00
#nullable disable
2018-11-20 15:51:59 +08:00
using osuTK ;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Objects.Types ;
using System ;
using System.Collections.Generic ;
2018-07-02 13:20:35 +08:00
using System.IO ;
2018-04-13 17:19:50 +08:00
using osu.Game.Beatmaps.Formats ;
using osu.Game.Audio ;
using System.Linq ;
2018-08-22 14:35:29 +08:00
using JetBrains.Annotations ;
2021-02-25 14:38:56 +08:00
using osu.Framework.Extensions.EnumExtensions ;
2020-01-09 12:43:44 +08:00
using osu.Framework.Utils ;
2019-12-10 19:19:16 +08:00
using osu.Game.Beatmaps.Legacy ;
2020-06-21 22:43:21 +08:00
using osu.Game.Skinning ;
2020-12-01 14:37:51 +08:00
using osu.Game.Utils ;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Objects.Legacy
{
/// <summary>
/// A HitObjectParser to parse legacy Beatmaps.
/// </summary>
public abstract class ConvertHitObjectParser : HitObjectParser
{
2018-08-15 09:24:56 +08:00
/// <summary>
/// The offset to apply to all time values.
/// </summary>
2018-08-15 10:47:31 +08:00
protected readonly double Offset ;
2018-08-15 09:24:56 +08:00
/// <summary>
/// The beatmap version.
/// </summary>
2018-08-15 10:47:31 +08:00
protected readonly int FormatVersion ;
2018-08-15 09:24:56 +08:00
2018-08-15 10:47:54 +08:00
protected bool FirstObject { get ; private set ; } = true ;
2018-08-15 09:24:56 +08:00
protected ConvertHitObjectParser ( double offset , int formatVersion )
2018-04-30 15:43:32 +08:00
{
2018-08-15 09:24:56 +08:00
Offset = offset ;
2018-08-15 09:53:25 +08:00
FormatVersion = formatVersion ;
2018-04-30 15:43:32 +08:00
}
2018-08-22 14:35:29 +08:00
[CanBeNull]
2018-08-15 09:24:56 +08:00
public override HitObject Parse ( string text )
2018-04-13 17:19:50 +08:00
{
2019-08-08 13:44:04 +08:00
string [ ] split = text . Split ( ',' ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
Vector2 pos = new Vector2 ( ( int ) Parsing . ParseFloat ( split [ 0 ] , Parsing . MAX_COORDINATE_VALUE ) , ( int ) Parsing . ParseFloat ( split [ 1 ] , Parsing . MAX_COORDINATE_VALUE ) ) ;
2018-07-19 23:47:55 +08:00
2019-08-08 13:44:04 +08:00
double startTime = Parsing . ParseDouble ( split [ 2 ] ) + Offset ;
2019-03-12 19:31:15 +08:00
2019-12-10 19:19:16 +08:00
LegacyHitObjectType type = ( LegacyHitObjectType ) Parsing . ParseInt ( split [ 3 ] ) ;
2018-08-15 09:48:42 +08:00
2019-12-10 19:19:16 +08:00
int comboOffset = ( int ) ( type & LegacyHitObjectType . ComboOffset ) > > 4 ;
type & = ~ LegacyHitObjectType . ComboOffset ;
2018-08-15 09:48:42 +08:00
2021-02-25 14:38:56 +08:00
bool combo = type . HasFlagFast ( LegacyHitObjectType . NewCombo ) ;
2019-12-10 19:19:16 +08:00
type & = ~ LegacyHitObjectType . NewCombo ;
2018-04-13 17:19:50 +08:00
2019-12-10 19:04:37 +08:00
var soundType = ( LegacyHitSoundType ) Parsing . ParseInt ( split [ 4 ] ) ;
2019-08-08 13:44:04 +08:00
var bankInfo = new SampleBankInfo ( ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
HitObject result = null ;
2018-04-13 17:19:50 +08:00
2021-02-25 14:38:56 +08:00
if ( type . HasFlagFast ( LegacyHitObjectType . Circle ) )
2019-08-08 13:44:04 +08:00
{
result = CreateHit ( pos , combo , comboOffset ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
if ( split . Length > 5 )
readCustomSampleBanks ( split [ 5 ] , bankInfo ) ;
}
2021-02-25 14:38:56 +08:00
else if ( type . HasFlagFast ( LegacyHitObjectType . Slider ) )
2019-08-08 13:44:04 +08:00
{
double? length = null ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
int repeatCount = Parsing . ParseInt ( split [ 6 ] ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
if ( repeatCount > 9000 )
2019-11-28 22:21:21 +08:00
throw new FormatException ( @"Repeat count is way too high" ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
// osu-stable treated the first span of the slider as a repeat, but no repeats are happening
repeatCount = Math . Max ( 0 , repeatCount - 1 ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
if ( split . Length > 7 )
{
2020-03-11 17:07:11 +08:00
length = Math . Max ( 0 , Parsing . ParseDouble ( split [ 7 ] , Parsing . MAX_COORDINATE_VALUE ) ) ;
2019-08-08 13:44:04 +08:00
if ( length = = 0 )
length = null ;
}
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
if ( split . Length > 10 )
readCustomSampleBanks ( split [ 10 ] , bankInfo ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
// One node for each repeat + the start and end nodes
int nodes = repeatCount + 2 ;
2019-04-01 11:16:05 +08:00
2019-08-08 13:44:04 +08:00
// Populate node sample bank infos with the default hit object sample bank
var nodeBankInfos = new List < SampleBankInfo > ( ) ;
for ( int i = 0 ; i < nodes ; i + + )
nodeBankInfos . Add ( bankInfo . Clone ( ) ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
// Read any per-node sample banks
if ( split . Length > 9 & & split [ 9 ] . Length > 0 )
{
string [ ] sets = split [ 9 ] . Split ( '|' ) ;
2018-04-13 17:19:50 +08:00
for ( int i = 0 ; i < nodes ; i + + )
{
2019-08-08 13:44:04 +08:00
if ( i > = sets . Length )
break ;
2019-04-01 11:16:05 +08:00
2019-08-08 13:44:04 +08:00
SampleBankInfo info = nodeBankInfos [ i ] ;
readCustomSampleBanks ( sets [ i ] , info ) ;
2018-04-13 17:19:50 +08:00
}
}
2019-03-12 19:31:15 +08:00
2019-08-08 13:44:04 +08:00
// Populate node sound types with the default hit object sound type
2019-12-10 19:04:37 +08:00
var nodeSoundTypes = new List < LegacyHitSoundType > ( ) ;
2019-08-08 13:44:04 +08:00
for ( int i = 0 ; i < nodes ; i + + )
nodeSoundTypes . Add ( soundType ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
// Read any per-node sound types
if ( split . Length > 8 & & split [ 8 ] . Length > 0 )
2018-04-13 17:19:50 +08:00
{
2019-08-08 13:44:04 +08:00
string [ ] adds = split [ 8 ] . Split ( '|' ) ;
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
for ( int i = 0 ; i < nodes ; i + + )
2018-04-13 17:19:50 +08:00
{
2019-08-08 13:44:04 +08:00
if ( i > = adds . Length )
break ;
2018-04-13 17:19:50 +08:00
2021-10-27 12:04:41 +08:00
int . TryParse ( adds [ i ] , out int sound ) ;
2019-12-10 19:04:37 +08:00
nodeSoundTypes [ i ] = ( LegacyHitSoundType ) sound ;
2019-08-08 13:44:04 +08:00
}
2018-08-22 14:35:29 +08:00
}
2018-04-13 17:19:50 +08:00
2019-08-08 13:44:04 +08:00
// Generate the final per-node samples
2019-11-08 13:04:57 +08:00
var nodeSamples = new List < IList < HitSampleInfo > > ( nodes ) ;
2019-08-08 13:44:04 +08:00
for ( int i = 0 ; i < nodes ; i + + )
nodeSamples . Add ( convertSoundType ( nodeSoundTypes [ i ] , nodeBankInfos [ i ] ) ) ;
2018-10-16 16:10:24 +08:00
2020-10-12 18:16:37 +08:00
result = CreateSlider ( pos , combo , comboOffset , convertPathString ( split [ 5 ] , pos ) , length , repeatCount , nodeSamples ) ;
2018-04-13 17:19:50 +08:00
}
2021-02-25 14:38:56 +08:00
else if ( type . HasFlagFast ( LegacyHitObjectType . Spinner ) )
2018-04-13 17:19:50 +08:00
{
2020-05-27 11:37:44 +08:00
double duration = Math . Max ( 0 , Parsing . ParseDouble ( split [ 5 ] ) + Offset - startTime ) ;
2019-08-08 13:44:04 +08:00
2020-05-27 11:37:44 +08:00
result = CreateSpinner ( new Vector2 ( 512 , 384 ) / 2 , combo , comboOffset , duration ) ;
2019-08-08 13:44:04 +08:00
if ( split . Length > 6 )
readCustomSampleBanks ( split [ 6 ] , bankInfo ) ;
2018-04-13 17:19:50 +08:00
}
2021-02-25 14:38:56 +08:00
else if ( type . HasFlagFast ( LegacyHitObjectType . Hold ) )
2019-03-13 10:30:33 +08:00
{
2019-08-08 13:44:04 +08:00
// Note: Hold is generated by BMS converts
double endTime = Math . Max ( startTime , Parsing . ParseDouble ( split [ 2 ] ) ) ;
if ( split . Length > 5 & & ! string . IsNullOrEmpty ( split [ 5 ] ) )
{
string [ ] ss = split [ 5 ] . Split ( ':' ) ;
endTime = Math . Max ( startTime , Parsing . ParseDouble ( ss [ 0 ] ) ) ;
2020-10-16 17:52:29 +08:00
readCustomSampleBanks ( string . Join ( ':' , ss . Skip ( 1 ) ) , bankInfo ) ;
2019-08-08 13:44:04 +08:00
}
2020-05-27 11:37:44 +08:00
result = CreateHold ( pos , combo , comboOffset , endTime + Offset - startTime ) ;
2019-03-13 10:30:33 +08:00
}
2019-08-08 13:44:04 +08:00
if ( result = = null )
2019-08-08 13:44:49 +08:00
throw new InvalidDataException ( $"Unknown hit object type: {split[3]}" ) ;
2019-08-08 13:44:04 +08:00
result . StartTime = startTime ;
if ( result . Samples . Count = = 0 )
result . Samples = convertSoundType ( soundType , bankInfo ) ;
FirstObject = false ;
return result ;
2018-04-13 17:19:50 +08:00
}
private void readCustomSampleBanks ( string str , SampleBankInfo bankInfo )
{
if ( string . IsNullOrEmpty ( str ) )
return ;
string [ ] split = str . Split ( ':' ) ;
2019-12-10 19:23:15 +08:00
var bank = ( LegacySampleBank ) Parsing . ParseInt ( split [ 0 ] ) ;
2021-12-10 13:15:00 +08:00
var addBank = ( LegacySampleBank ) Parsing . ParseInt ( split [ 1 ] ) ;
2018-04-13 17:19:50 +08:00
2018-07-25 13:37:05 +08:00
string stringBank = bank . ToString ( ) . ToLowerInvariant ( ) ;
2018-04-13 17:19:50 +08:00
if ( stringBank = = @"none" )
stringBank = null ;
2021-12-10 13:15:00 +08:00
string stringAddBank = addBank . ToString ( ) . ToLowerInvariant ( ) ;
2018-04-13 17:19:50 +08:00
if ( stringAddBank = = @"none" )
stringAddBank = null ;
2022-10-25 13:55:33 +08:00
bankInfo . BankForNormal = stringBank ;
bankInfo . BankForAdditions = string . IsNullOrEmpty ( stringAddBank ) ? stringBank : stringAddBank ;
2018-04-13 17:19:50 +08:00
2018-07-20 14:12:44 +08:00
if ( split . Length > 2 )
2019-03-12 19:31:15 +08:00
bankInfo . CustomSampleBank = Parsing . ParseInt ( split [ 2 ] ) ;
2018-07-20 14:12:44 +08:00
2018-04-13 17:19:50 +08:00
if ( split . Length > 3 )
2019-03-13 11:35:05 +08:00
bankInfo . Volume = Math . Max ( 0 , Parsing . ParseInt ( split [ 3 ] ) ) ;
2018-07-02 13:29:18 +08:00
bankInfo . Filename = split . Length > 4 ? split [ 4 ] : null ;
2018-04-13 17:19:50 +08:00
}
2020-10-12 17:04:28 +08:00
private PathType convertPathType ( string input )
{
switch ( input [ 0 ] )
{
default :
case 'C' :
return PathType . Catmull ;
case 'B' :
return PathType . Bezier ;
case 'L' :
return PathType . Linear ;
case 'P' :
return PathType . PerfectCurve ;
}
}
/// <summary>
2020-10-12 18:16:37 +08:00
/// Converts a given point string into a set of path control points.
2020-10-12 17:04:28 +08:00
/// </summary>
2020-10-12 18:16:37 +08:00
/// <remarks>
/// A point string takes the form: X|1:1|2:2|2:2|3:3|Y|1:1|2:2.
/// This has three segments:
/// <list type="number">
/// <item>
/// <description>X: { (1,1), (2,2) } (implicit segment)</description>
/// </item>
/// <item>
/// <description>X: { (2,2), (3,3) } (implicit segment)</description>
/// </item>
/// <item>
/// <description>Y: { (3,3), (1,1), (2, 2) } (explicit segment)</description>
/// </item>
/// </list>
/// </remarks>
/// <param name="pointString">The point string.</param>
/// <param name="offset">The positional offset to apply to the control points.</param>
/// <returns>All control points in the resultant path.</returns>
private PathControlPoint [ ] convertPathString ( string pointString , Vector2 offset )
{
// This code takes on the responsibility of handling explicit segments of the path ("X" & "Y" from above). Implicit segments are handled by calls to convertPoints().
string [ ] pointSplit = pointString . Split ( '|' ) ;
var controlPoints = new List < Memory < PathControlPoint > > ( ) ;
int startIndex = 0 ;
int endIndex = 0 ;
bool first = true ;
while ( + + endIndex < pointSplit . Length )
{
// Keep incrementing endIndex while it's not the start of a new segment (indicated by having a type descriptor of length 1).
if ( pointSplit [ endIndex ] . Length > 1 )
continue ;
// Multi-segmented sliders DON'T contain the end point as part of the current segment as it's assumed to be the start of the next segment.
// The start of the next segment is the index after the type descriptor.
string endPoint = endIndex < pointSplit . Length - 1 ? pointSplit [ endIndex + 1 ] : null ;
controlPoints . AddRange ( convertPoints ( pointSplit . AsMemory ( ) . Slice ( startIndex , endIndex - startIndex ) , endPoint , first , offset ) ) ;
startIndex = endIndex ;
first = false ;
}
if ( endIndex > startIndex )
controlPoints . AddRange ( convertPoints ( pointSplit . AsMemory ( ) . Slice ( startIndex , endIndex - startIndex ) , null , first , offset ) ) ;
return mergePointsLists ( controlPoints ) ;
}
/// <summary>
/// Converts a given point list into a set of path segments.
/// </summary>
/// <param name="points">The point list.</param>
2020-10-12 17:04:28 +08:00
/// <param name="endPoint">Any extra endpoint to consider as part of the points. This will NOT be returned.</param>
2020-10-12 18:16:37 +08:00
/// <param name="first">Whether this is the first segment in the set. If <c>true</c> the first of the returned segments will contain a zero point.</param>
2020-10-12 17:04:28 +08:00
/// <param name="offset">The positional offset to apply to the control points.</param>
2020-10-12 18:16:37 +08:00
/// <returns>The set of points contained by <paramref name="points"/> as one or more segments of the path, prepended by an extra zero point if <paramref name="first"/> is <c>true</c>.</returns>
private IEnumerable < Memory < PathControlPoint > > convertPoints ( ReadOnlyMemory < string > points , string endPoint , bool first , Vector2 offset )
2019-12-05 18:53:31 +08:00
{
2020-10-12 18:16:37 +08:00
PathType type = convertPathType ( points . Span [ 0 ] ) ;
2020-10-12 17:04:28 +08:00
int readOffset = first ? 1 : 0 ; // First control point is zero for the first segment.
2020-10-12 18:16:37 +08:00
int readablePoints = points . Length - 1 ; // Total points readable from the base point span.
2020-10-12 17:04:28 +08:00
int endPointLength = endPoint ! = null ? 1 : 0 ; // Extra length if an endpoint is given that lies outside the base point span.
var vertices = new PathControlPoint [ readOffset + readablePoints + endPointLength ] ;
// Fill any non-read points.
for ( int i = 0 ; i < readOffset ; i + + )
vertices [ i ] = new PathControlPoint ( ) ;
// Parse into control points.
2020-10-12 18:16:37 +08:00
for ( int i = 1 ; i < points . Length ; i + + )
readPoint ( points . Span [ i ] , offset , out vertices [ readOffset + i - 1 ] ) ;
2020-10-12 17:04:28 +08:00
2020-10-12 18:22:34 +08:00
// If an endpoint is given, add it to the end.
2020-10-12 17:04:28 +08:00
if ( endPoint ! = null )
readPoint ( endPoint , offset , out vertices [ ^ 1 ] ) ;
2020-10-12 18:22:34 +08:00
// Edge-case rules (to match stable).
2019-12-05 18:53:31 +08:00
if ( type = = PathType . PerfectCurve )
{
2019-12-10 12:12:54 +08:00
if ( vertices . Length ! = 3 )
type = PathType . Bezier ;
else if ( isLinear ( vertices ) )
2019-12-05 18:53:31 +08:00
{
// osu-stable special-cased colinear perfect curves to a linear path
2019-12-10 12:12:54 +08:00
type = PathType . Linear ;
2019-12-05 18:53:31 +08:00
}
}
2020-10-12 18:22:34 +08:00
// The first control point must have a definite type.
2021-08-26 00:42:57 +08:00
vertices [ 0 ] . Type = type ;
2019-12-05 18:53:31 +08:00
2020-10-12 18:22:34 +08:00
// A path can have multiple implicit segments of the same type if there are two sequential control points with the same position.
2020-10-12 18:16:37 +08:00
// To handle such cases, this code may return multiple path segments with the final control point in each segment having a non-null type.
2020-10-12 18:22:34 +08:00
// For the point string X|1:1|2:2|2:2|3:3, this code returns the segments:
// X: { (1,1), (2, 2) }
// X: { (3, 3) }
// Note: (2, 2) is not returned in the second segments, as it is implicit in the path.
2020-10-12 18:16:37 +08:00
int startIndex = 0 ;
int endIndex = 0 ;
while ( + + endIndex < vertices . Length - endPointLength )
2019-12-05 18:53:31 +08:00
{
2022-05-18 16:11:08 +08:00
// Keep incrementing while an implicit segment doesn't need to be started.
2021-08-26 00:42:57 +08:00
if ( vertices [ endIndex ] . Position ! = vertices [ endIndex - 1 ] . Position )
2020-10-12 18:16:37 +08:00
continue ;
2022-05-24 10:47:42 +08:00
// Legacy Catmull sliders don't support multiple segments, so adjacent Catmull segments should be treated as a single one.
// Importantly, this is not applied to the first control point, which may duplicate the slider path's position
// resulting in a duplicate (0,0) control point in the resultant list.
if ( type = = PathType . Catmull & & endIndex > 1 & & FormatVersion < LegacyBeatmapEncoder . FIRST_LAZER_VERSION )
2022-05-18 16:11:08 +08:00
continue ;
2021-04-05 16:49:36 +08:00
// The last control point of each segment is not allowed to start a new implicit segment.
if ( endIndex = = vertices . Length - endPointLength - 1 )
continue ;
2020-10-12 18:16:37 +08:00
// Force a type on the last point, and return the current control point set as a segment.
2021-08-26 00:42:57 +08:00
vertices [ endIndex - 1 ] . Type = type ;
2020-10-12 18:16:37 +08:00
yield return vertices . AsMemory ( ) . Slice ( startIndex , endIndex - startIndex ) ;
// Skip the current control point - as it's the same as the one that's just been returned.
startIndex = endIndex + 1 ;
2020-10-12 17:04:28 +08:00
}
2019-12-05 18:53:31 +08:00
2020-10-12 18:16:37 +08:00
if ( endIndex > startIndex )
yield return vertices . AsMemory ( ) . Slice ( startIndex , endIndex - startIndex ) ;
2020-10-12 17:04:28 +08:00
static void readPoint ( string value , Vector2 startPos , out PathControlPoint point )
{
string [ ] vertexSplit = value . Split ( ':' ) ;
Vector2 pos = new Vector2 ( ( int ) Parsing . ParseDouble ( vertexSplit [ 0 ] , Parsing . MAX_COORDINATE_VALUE ) , ( int ) Parsing . ParseDouble ( vertexSplit [ 1 ] , Parsing . MAX_COORDINATE_VALUE ) ) - startPos ;
2021-08-26 00:42:57 +08:00
point = new PathControlPoint { Position = pos } ;
2019-12-05 18:53:31 +08:00
}
2021-08-26 00:42:57 +08:00
static bool isLinear ( PathControlPoint [ ] p ) = > Precision . AlmostEquals ( 0 , ( p [ 1 ] . Position . Y - p [ 0 ] . Position . Y ) * ( p [ 2 ] . Position . X - p [ 0 ] . Position . X )
- ( p [ 1 ] . Position . X - p [ 0 ] . Position . X ) * ( p [ 2 ] . Position . Y - p [ 0 ] . Position . Y ) ) ;
2020-10-12 17:04:28 +08:00
}
2020-10-12 18:16:37 +08:00
private PathControlPoint [ ] mergePointsLists ( List < Memory < PathControlPoint > > controlPointList )
2020-10-12 17:04:28 +08:00
{
int totalCount = 0 ;
foreach ( var arr in controlPointList )
totalCount + = arr . Length ;
var mergedArray = new PathControlPoint [ totalCount ] ;
var mergedArrayMemory = mergedArray . AsMemory ( ) ;
int copyIndex = 0 ;
foreach ( var arr in controlPointList )
{
arr . CopyTo ( mergedArrayMemory . Slice ( copyIndex ) ) ;
copyIndex + = arr . Length ;
}
2019-12-05 18:53:31 +08:00
2020-10-12 17:04:28 +08:00
return mergedArray ;
2019-12-05 18:53:31 +08:00
}
2018-04-13 17:19:50 +08:00
/// <summary>
/// Creates a legacy Hit-type hit object.
/// </summary>
/// <param name="position">The position of the hit object.</param>
/// <param name="newCombo">Whether the hit object creates a new combo.</param>
2018-08-15 09:48:42 +08:00
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2018-04-13 17:19:50 +08:00
/// <returns>The hit object.</returns>
2018-08-15 09:48:42 +08:00
protected abstract HitObject CreateHit ( Vector2 position , bool newCombo , int comboOffset ) ;
2018-04-13 17:19:50 +08:00
/// <summary>
/// Creats a legacy Slider-type hit object.
/// </summary>
/// <param name="position">The position of the hit object.</param>
/// <param name="newCombo">Whether the hit object creates a new combo.</param>
2018-08-15 09:48:42 +08:00
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2018-04-13 17:19:50 +08:00
/// <param name="controlPoints">The slider control points.</param>
/// <param name="length">The slider length.</param>
/// <param name="repeatCount">The slider repeat count.</param>
2018-10-16 16:10:24 +08:00
/// <param name="nodeSamples">The samples to be played when the slider nodes are hit. This includes the head and tail of the slider.</param>
2018-04-13 17:19:50 +08:00
/// <returns>The hit object.</returns>
2019-12-05 18:53:31 +08:00
protected abstract HitObject CreateSlider ( Vector2 position , bool newCombo , int comboOffset , PathControlPoint [ ] controlPoints , double? length , int repeatCount ,
2021-10-23 16:59:07 +08:00
IList < IList < HitSampleInfo > > nodeSamples ) ;
2018-04-13 17:19:50 +08:00
/// <summary>
/// Creates a legacy Spinner-type hit object.
/// </summary>
/// <param name="position">The position of the hit object.</param>
2018-08-15 09:48:42 +08:00
/// <param name="newCombo">Whether the hit object creates a new combo.</param>
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2020-05-27 11:37:44 +08:00
/// <param name="duration">The spinner duration.</param>
2018-04-13 17:19:50 +08:00
/// <returns>The hit object.</returns>
2020-05-27 11:37:44 +08:00
protected abstract HitObject CreateSpinner ( Vector2 position , bool newCombo , int comboOffset , double duration ) ;
2018-04-13 17:19:50 +08:00
/// <summary>
/// Creates a legacy Hold-type hit object.
/// </summary>
/// <param name="position">The position of the hit object.</param>
/// <param name="newCombo">Whether the hit object creates a new combo.</param>
2018-08-15 09:48:42 +08:00
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2020-05-31 21:39:03 +08:00
/// <param name="duration">The hold duration.</param>
2020-05-27 11:37:44 +08:00
protected abstract HitObject CreateHold ( Vector2 position , bool newCombo , int comboOffset , double duration ) ;
2018-04-13 17:19:50 +08:00
2019-12-10 19:04:37 +08:00
private List < HitSampleInfo > convertSoundType ( LegacyHitSoundType type , SampleBankInfo bankInfo )
2018-04-13 17:19:50 +08:00
{
2023-02-10 05:18:12 +08:00
var soundTypes = new List < HitSampleInfo > ( ) ;
if ( string . IsNullOrEmpty ( bankInfo . Filename ) )
2019-03-05 17:21:29 +08:00
{
2023-02-10 05:18:12 +08:00
soundTypes . Add ( new LegacyHitSampleInfo ( HitSampleInfo . HIT_NORMAL , bankInfo . BankForNormal , bankInfo . Volume , bankInfo . CustomSampleBank ,
// if the sound type doesn't have the Normal flag set, attach it anyway as a layered sample.
// None also counts as a normal non-layered sample: https://osu.ppy.sh/help/wiki/osu!_File_Formats/Osu_(file_format)#hitsounds
type ! = LegacyHitSoundType . None & & ! type . HasFlagFast ( LegacyHitSoundType . Normal ) ) ) ;
2019-03-05 17:21:29 +08:00
}
2023-02-10 05:18:12 +08:00
else
2018-04-13 17:19:50 +08:00
{
2023-02-10 05:18:12 +08:00
// Todo: This should set the normal SampleInfo if the specified sample file isn't found, but that's a pretty edge-case scenario
soundTypes . Add ( new FileHitSampleInfo ( bankInfo . Filename , bankInfo . Volume ) ) ;
}
2018-04-13 17:19:50 +08:00
2021-02-25 14:38:56 +08:00
if ( type . HasFlagFast ( LegacyHitSoundType . Finish ) )
2022-10-25 13:55:33 +08:00
soundTypes . Add ( new LegacyHitSampleInfo ( HitSampleInfo . HIT_FINISH , bankInfo . BankForAdditions , bankInfo . Volume , bankInfo . CustomSampleBank ) ) ;
2018-04-13 17:19:50 +08:00
2021-02-25 14:38:56 +08:00
if ( type . HasFlagFast ( LegacyHitSoundType . Whistle ) )
2022-10-25 13:55:33 +08:00
soundTypes . Add ( new LegacyHitSampleInfo ( HitSampleInfo . HIT_WHISTLE , bankInfo . BankForAdditions , bankInfo . Volume , bankInfo . CustomSampleBank ) ) ;
2018-04-13 17:19:50 +08:00
2021-02-25 14:38:56 +08:00
if ( type . HasFlagFast ( LegacyHitSoundType . Clap ) )
2022-10-25 13:55:33 +08:00
soundTypes . Add ( new LegacyHitSampleInfo ( HitSampleInfo . HIT_CLAP , bankInfo . BankForAdditions , bankInfo . Volume , bankInfo . CustomSampleBank ) ) ;
2018-04-13 17:19:50 +08:00
return soundTypes ;
}
private class SampleBankInfo
{
2022-10-25 13:55:33 +08:00
/// <summary>
/// An optional overriding filename which causes all bank/sample specifications to be ignored.
/// </summary>
2018-07-02 13:20:35 +08:00
public string Filename ;
2022-10-25 13:55:33 +08:00
/// <summary>
/// The bank identifier to use for the base ("hitnormal") sample.
/// Transferred to <see cref="HitSampleInfo.Bank"/> when appropriate.
/// </summary>
public string BankForNormal ;
/// <summary>
/// The bank identifier to use for additions ("hitwhistle", "hitfinish", "hitclap").
/// Transferred to <see cref="HitSampleInfo.Bank"/> when appropriate.
/// </summary>
public string BankForAdditions ;
/// <summary>
/// Hit sample volume (0-100).
/// See <see cref="HitSampleInfo.Volume"/>.
/// </summary>
2018-04-13 17:19:50 +08:00
public int Volume ;
2022-10-25 13:55:33 +08:00
/// <summary>
/// The index of the custom sample bank. Is only used if 2 or above for "reasons".
/// This will add a suffix to lookups, allowing extended bank lookups (ie. "normal-hitnormal-2").
/// See <see cref="HitSampleInfo.Suffix"/>.
/// </summary>
2018-07-20 14:12:44 +08:00
public int CustomSampleBank ;
2018-07-02 13:20:35 +08:00
public SampleBankInfo Clone ( ) = > ( SampleBankInfo ) MemberwiseClone ( ) ;
}
2020-12-01 14:37:51 +08:00
#nullable enable
2020-12-01 14:44:16 +08:00
public class LegacyHitSampleInfo : HitSampleInfo , IEquatable < LegacyHitSampleInfo >
2018-07-20 14:12:44 +08:00
{
2020-12-01 14:37:51 +08:00
public readonly int CustomSampleBank ;
2020-06-21 22:43:21 +08:00
/// <summary>
/// Whether this hit sample is layered.
/// </summary>
/// <remarks>
/// Layered hit samples are automatically added in all modes (except osu!mania), but can be disabled
2021-10-22 13:41:59 +08:00
/// using the <see cref="SkinConfiguration.LegacySetting.LayeredHitSounds"/> skin config option.
2020-06-21 22:43:21 +08:00
/// </remarks>
2020-12-01 14:37:51 +08:00
public readonly bool IsLayered ;
2020-12-01 17:09:28 +08:00
public LegacyHitSampleInfo ( string name , string? bank = null , int volume = 0 , int customSampleBank = 0 , bool isLayered = false )
2020-12-01 14:37:51 +08:00
: base ( name , bank , customSampleBank > = 2 ? customSampleBank . ToString ( ) : null , volume )
{
CustomSampleBank = customSampleBank ;
IsLayered = isLayered ;
}
2020-12-02 09:55:48 +08:00
public sealed override HitSampleInfo With ( Optional < string > newName = default , Optional < string? > newBank = default , Optional < string? > newSuffix = default , Optional < int > newVolume = default )
= > With ( newName , newBank , newVolume ) ;
2020-12-01 14:37:51 +08:00
2022-10-25 13:55:33 +08:00
public virtual LegacyHitSampleInfo With ( Optional < string > newName = default , Optional < string? > newBank = default , Optional < int > newVolume = default ,
Optional < int > newCustomSampleBank = default ,
2020-12-02 09:55:48 +08:00
Optional < bool > newIsLayered = default )
= > new LegacyHitSampleInfo ( newName . GetOr ( Name ) , newBank . GetOr ( Bank ) , newVolume . GetOr ( Volume ) , newCustomSampleBank . GetOr ( CustomSampleBank ) , newIsLayered . GetOr ( IsLayered ) ) ;
2020-12-01 14:44:16 +08:00
public bool Equals ( LegacyHitSampleInfo ? other )
2022-03-23 19:18:44 +08:00
// The additions to equality checks here are *required* to ensure that pooling works correctly.
// Of note, `IsLayered` may cause the usage of `SampleVirtual` instead of an actual sample (in cases playback is not required).
// Removing it would cause samples which may actually require playback to potentially source for a `SampleVirtual` sample pool.
2022-03-23 19:15:17 +08:00
= > base . Equals ( other ) & & CustomSampleBank = = other . CustomSampleBank & & IsLayered = = other . IsLayered ;
2020-12-01 14:44:16 +08:00
2020-12-01 17:09:21 +08:00
public override bool Equals ( object? obj )
= > obj is LegacyHitSampleInfo other & & Equals ( other ) ;
2020-12-01 14:44:16 +08:00
2022-03-23 19:15:13 +08:00
public override int GetHashCode ( ) = > HashCode . Combine ( base . GetHashCode ( ) , CustomSampleBank , IsLayered ) ;
2018-07-20 14:12:44 +08:00
}
2020-12-01 14:44:16 +08:00
private class FileHitSampleInfo : LegacyHitSampleInfo , IEquatable < FileHitSampleInfo >
2018-07-02 13:20:35 +08:00
{
2020-12-01 14:37:51 +08:00
public readonly string Filename ;
2018-07-02 13:20:35 +08:00
2020-12-01 14:37:51 +08:00
public FileHitSampleInfo ( string filename , int volume )
// Force CSS=1 to make sure that the LegacyBeatmapSkin does not fall back to the user skin.
2020-04-14 20:05:07 +08:00
// Note that this does not change the lookup names, as they are overridden locally.
2020-12-01 14:37:51 +08:00
: base ( string . Empty , customSampleBank : 1 , volume : volume )
{
Filename = filename ;
2020-04-13 19:09:17 +08:00
}
2018-07-02 13:20:35 +08:00
public override IEnumerable < string > LookupNames = > new [ ]
2018-04-13 17:19:50 +08:00
{
2018-07-02 13:20:35 +08:00
Filename ,
Path . ChangeExtension ( Filename , null )
} ;
2020-12-01 14:37:51 +08:00
2022-10-25 13:55:33 +08:00
public sealed override LegacyHitSampleInfo With ( Optional < string > newName = default , Optional < string? > newBank = default , Optional < int > newVolume = default ,
Optional < int > newCustomSampleBank = default ,
2020-12-02 09:55:48 +08:00
Optional < bool > newIsLayered = default )
= > new FileHitSampleInfo ( Filename , newVolume . GetOr ( Volume ) ) ;
2020-12-01 14:44:16 +08:00
public bool Equals ( FileHitSampleInfo ? other )
= > base . Equals ( other ) & & Filename = = other . Filename ;
public override bool Equals ( object? obj )
2020-12-01 17:09:21 +08:00
= > obj is FileHitSampleInfo other & & Equals ( other ) ;
2020-12-01 14:44:16 +08:00
public override int GetHashCode ( ) = > HashCode . Combine ( base . GetHashCode ( ) , Filename ) ;
2018-04-13 17:19:50 +08:00
}
2020-12-01 14:37:51 +08:00
#nullable disable
2018-04-13 17:19:50 +08:00
}
}