osu/osu.Game/Rulesets/Objects/Legacy/ConvertHitObjectParser.cs

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

579 lines
26 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 09:19:50 +00:00
2022-06-17 07:38:35 +00:00
#nullable disable
2018-11-20 07:51:59 +00:00
using osuTK;
2017-04-18 07:05:58 +00:00
using osu.Game.Rulesets.Objects.Types;
2016-12-06 09:56:20 +00:00
using System;
using System.Collections.Generic;
2018-07-02 05:20:35 +00:00
using System.IO;
using osu.Game.Beatmaps.Formats;
using osu.Game.Audio;
2017-05-12 07:35:57 +00:00
using System.Linq;
using JetBrains.Annotations;
2021-02-25 06:38:56 +00:00
using osu.Framework.Extensions.EnumExtensions;
2020-01-09 04:43:44 +00:00
using osu.Framework.Utils;
2023-05-16 07:29:24 +00:00
using osu.Game.Beatmaps.ControlPoints;
2019-12-10 11:19:16 +00:00
using osu.Game.Beatmaps.Legacy;
using osu.Game.Skinning;
2020-12-01 06:37:51 +00:00
using osu.Game.Utils;
2018-04-13 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
namespace osu.Game.Rulesets.Objects.Legacy
{
2017-04-18 00:13:36 +00:00
/// <summary>
/// A HitObjectParser to parse legacy Beatmaps.
/// </summary>
2017-11-21 03:11:29 +00:00
public abstract class ConvertHitObjectParser : HitObjectParser
{
/// <summary>
/// The offset to apply to all time values.
/// </summary>
2018-08-15 02:47:31 +00:00
protected readonly double Offset;
/// <summary>
/// The beatmap version.
/// </summary>
2018-08-15 02:47:31 +00:00
protected readonly int FormatVersion;
protected bool FirstObject { get; private set; } = true;
protected ConvertHitObjectParser(double offset, int formatVersion)
{
Offset = offset;
2018-08-15 01:53:25 +00:00
FormatVersion = formatVersion;
}
public override HitObject Parse(string text)
{
string[] split = text.Split(',');
2018-04-13 09:19:50 +00:00
Vector2 pos = new Vector2((int)Parsing.ParseFloat(split[0], Parsing.MAX_COORDINATE_VALUE), (int)Parsing.ParseFloat(split[1], Parsing.MAX_COORDINATE_VALUE));
double startTime = Parsing.ParseDouble(split[2]) + Offset;
2019-12-10 11:19:16 +00:00
LegacyHitObjectType type = (LegacyHitObjectType)Parsing.ParseInt(split[3]);
2019-12-10 11:19:16 +00:00
int comboOffset = (int)(type & LegacyHitObjectType.ComboOffset) >> 4;
type &= ~LegacyHitObjectType.ComboOffset;
2021-02-25 06:38:56 +00:00
bool combo = type.HasFlagFast(LegacyHitObjectType.NewCombo);
2019-12-10 11:19:16 +00:00
type &= ~LegacyHitObjectType.NewCombo;
2018-04-13 09:19:50 +00:00
2019-12-10 11:04:37 +00:00
var soundType = (LegacyHitSoundType)Parsing.ParseInt(split[4]);
var bankInfo = new SampleBankInfo();
2018-04-13 09:19:50 +00:00
HitObject result = null;
2018-04-13 09:19:50 +00:00
2021-02-25 06:38:56 +00:00
if (type.HasFlagFast(LegacyHitObjectType.Circle))
{
result = CreateHit(pos, combo, comboOffset);
2018-04-13 09:19:50 +00:00
if (split.Length > 5)
readCustomSampleBanks(split[5], bankInfo);
}
2021-02-25 06:38:56 +00:00
else if (type.HasFlagFast(LegacyHitObjectType.Slider))
{
double? length = null;
2018-04-13 09:19:50 +00:00
int repeatCount = Parsing.ParseInt(split[6]);
2018-04-13 09:19:50 +00:00
if (repeatCount > 9000)
2019-11-28 14:21:21 +00:00
throw new FormatException(@"Repeat count is way too high");
2018-04-13 09:19:50 +00: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 09:19:50 +00:00
if (split.Length > 7)
{
length = Math.Max(0, Parsing.ParseDouble(split[7], Parsing.MAX_COORDINATE_VALUE));
if (length == 0)
length = null;
}
2018-04-13 09:19:50 +00:00
if (split.Length > 10)
readCustomSampleBanks(split[10], bankInfo, true);
2018-04-13 09:19:50 +00:00
// One node for each repeat + the start and end nodes
int nodes = repeatCount + 2;
2019-04-01 03:16:05 +00: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 09:19:50 +00:00
// Read any per-node sample banks
if (split.Length > 9 && split[9].Length > 0)
{
string[] sets = split[9].Split('|');
2018-04-13 09:19:50 +00:00
for (int i = 0; i < nodes; i++)
{
if (i >= sets.Length)
break;
2019-04-01 03:16:05 +00:00
SampleBankInfo info = nodeBankInfos[i];
readCustomSampleBanks(sets[i], info);
}
}
// Populate node sound types with the default hit object sound type
2019-12-10 11:04:37 +00:00
var nodeSoundTypes = new List<LegacyHitSoundType>();
for (int i = 0; i < nodes; i++)
nodeSoundTypes.Add(soundType);
2018-04-13 09:19:50 +00:00
// Read any per-node sound types
if (split.Length > 8 && split[8].Length > 0)
{
string[] adds = split[8].Split('|');
2018-04-13 09:19:50 +00:00
for (int i = 0; i < nodes; i++)
{
if (i >= adds.Length)
break;
2018-04-13 09:19:50 +00:00
int.TryParse(adds[i], out int sound);
2019-12-10 11:04:37 +00:00
nodeSoundTypes[i] = (LegacyHitSoundType)sound;
}
}
2018-04-13 09:19:50 +00:00
// Generate the final per-node samples
2019-11-08 05:04:57 +00:00
var nodeSamples = new List<IList<HitSampleInfo>>(nodes);
for (int i = 0; i < nodes; i++)
nodeSamples.Add(convertSoundType(nodeSoundTypes[i], nodeBankInfos[i]));
result = CreateSlider(pos, combo, comboOffset, convertPathString(split[5], pos), length, repeatCount, nodeSamples);
}
2021-02-25 06:38:56 +00:00
else if (type.HasFlagFast(LegacyHitObjectType.Spinner))
{
2020-05-27 03:37:44 +00:00
double duration = Math.Max(0, Parsing.ParseDouble(split[5]) + Offset - startTime);
2020-05-27 03:37:44 +00:00
result = CreateSpinner(new Vector2(512, 384) / 2, combo, comboOffset, duration);
if (split.Length > 6)
readCustomSampleBanks(split[6], bankInfo);
}
2021-02-25 06:38:56 +00:00
else if (type.HasFlagFast(LegacyHitObjectType.Hold))
2019-03-13 02:30:33 +00: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 09:52:29 +00:00
readCustomSampleBanks(string.Join(':', ss.Skip(1)), bankInfo);
}
2020-05-27 03:37:44 +00:00
result = CreateHold(pos, combo, comboOffset, endTime + Offset - startTime);
2019-03-13 02:30:33 +00:00
}
if (result == null)
2019-08-08 05:44:49 +00:00
throw new InvalidDataException($"Unknown hit object type: {split[3]}");
result.StartTime = startTime;
if (result.Samples.Count == 0)
result.Samples = convertSoundType(soundType, bankInfo);
FirstObject = false;
return result;
}
2018-04-13 09:19:50 +00:00
private void readCustomSampleBanks(string str, SampleBankInfo bankInfo, bool banksOnly = false)
{
if (string.IsNullOrEmpty(str))
return;
2018-04-13 09:19:50 +00:00
string[] split = str.Split(':');
2018-04-13 09:19:50 +00:00
2019-12-10 11:23:15 +00:00
var bank = (LegacySampleBank)Parsing.ParseInt(split[0]);
if (!Enum.IsDefined(bank))
bank = LegacySampleBank.Normal;
var addBank = (LegacySampleBank)Parsing.ParseInt(split[1]);
if (!Enum.IsDefined(addBank))
addBank = LegacySampleBank.Normal;
2018-04-13 09:19:50 +00:00
2018-07-25 05:37:05 +00:00
string stringBank = bank.ToString().ToLowerInvariant();
if (stringBank == @"none")
2017-04-06 03:27:35 +00:00
stringBank = null;
string stringAddBank = addBank.ToString().ToLowerInvariant();
if (stringAddBank == @"none")
2017-04-06 03:27:35 +00:00
stringAddBank = null;
2018-04-13 09:19:50 +00:00
bankInfo.BankForNormal = stringBank;
bankInfo.BankForAdditions = string.IsNullOrEmpty(stringAddBank) ? stringBank : stringAddBank;
2018-04-13 09:19:50 +00:00
if (banksOnly) return;
if (split.Length > 2)
bankInfo.CustomSampleBank = Parsing.ParseInt(split[2]);
if (split.Length > 3)
bankInfo.Volume = Math.Max(0, Parsing.ParseInt(split[3]));
2018-07-02 05:29:18 +00:00
bankInfo.Filename = split.Length > 4 ? split[4] : null;
}
2018-04-13 09:19:50 +00:00
private PathType convertPathType(string input)
{
switch (input[0])
{
default:
case 'C':
2023-11-13 07:24:09 +00:00
return PathType.CATMULL;
case 'B':
if (input.Length > 1 && int.TryParse(input.Substring(1), out int degree) && degree > 0)
2023-11-13 07:24:09 +00:00
return PathType.BSpline(degree);
2023-11-11 14:02:06 +00:00
return PathType.BEZIER;
case 'L':
2023-11-13 07:24:09 +00:00
return PathType.LINEAR;
case 'P':
2023-11-13 07:24:09 +00:00
return PathType.PERFECT_CURVE;
}
}
/// <summary>
/// Converts a given point string into a set of path control points.
/// </summary>
/// <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('|');
Span<Vector2> points = stackalloc Vector2[pointSplit.Length];
Span<(PathType Type, int StartIndex)> segments = stackalloc (PathType Type, int StartIndex)[pointSplit.Length];
int pointsCount = 0;
int segmentsCount = 0;
foreach (string s in pointSplit)
{
if (char.IsLetter(s[0]))
{
// The start of a new segment(indicated by having an alpha character at position 0).
var pathType = convertPathType(s);
segments[segmentsCount++] = (pathType, pointsCount);
2024-02-28 14:42:08 +00:00
// First segment is prepended by an extra zero point
if (pointsCount == 0)
points[pointsCount++] = Vector2.Zero;
}
else
{
points[pointsCount++] = readPoint(s, offset);
}
}
var controlPoints = new List<PathControlPoint>(pointsCount);
for (int i = 0; i < segmentsCount; i++)
{
int startIndex = segments[i].StartIndex;
int endIndex = i < segmentsCount - 1 ? segments[i + 1].StartIndex : pointsCount;
Vector2? endPoint = i < segmentsCount - 1 ? points[endIndex] : null;
controlPoints.AddRange(convertPoints(segments[i].Type, points[startIndex..endIndex], endPoint));
}
return controlPoints.ToArray();
static Vector2 readPoint(string value, Vector2 startPos)
{
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;
return pos;
}
}
/// <summary>
/// Converts a given point list into a set of path segments.
/// </summary>
2024-02-28 14:42:08 +00:00
/// <param name="type">The path type of the point list.</param>
/// <param name="points">The point list.</param>
/// <param name="endPoint">Any extra endpoint to consider as part of the points. This will NOT be returned.</param>
2024-02-28 14:42:08 +00:00
/// <returns>The set of points contained by <paramref name="points"/> as one or more segments of the path.</returns>
private IEnumerable<PathControlPoint> convertPoints(PathType type, ReadOnlySpan<Vector2> points, Vector2? endPoint)
{
2024-02-28 14:42:08 +00:00
var vertices = new PathControlPoint[points.Length];
var result = new List<PathControlPoint>();
// Parse into control points.
2024-02-28 14:42:08 +00:00
for (int i = 0; i < points.Length; i++)
vertices[i] = new PathControlPoint { Position = points[i] };
2020-10-12 10:22:34 +00:00
// Edge-case rules (to match stable).
2023-11-13 07:24:09 +00:00
if (type == PathType.PERFECT_CURVE)
{
2024-02-28 14:42:08 +00:00
int endPointLength = endPoint is null ? 0 : 1;
if (vertices.Length + endPointLength != 3)
type = PathType.BEZIER;
2024-02-28 14:42:08 +00:00
else if (isLinear(stackalloc[] { points[0], points[1], endPoint ?? points[2] }))
{
// osu-stable special-cased colinear perfect curves to a linear path
type = PathType.LINEAR;
}
}
2020-10-12 10:22:34 +00:00
// The first control point must have a definite type.
vertices[0].Type = type;
2020-10-12 10:22:34 +00:00
// A path can have multiple implicit segments of the same type if there are two sequential control points with the same position.
// 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 10:22:34 +00: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.
int startIndex = 0;
int endIndex = 0;
2024-02-28 14:42:08 +00:00
while (++endIndex < vertices.Length)
{
// Keep incrementing while an implicit segment doesn't need to be started.
if (vertices[endIndex].Position != vertices[endIndex - 1].Position)
continue;
// 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)
continue;
// The last control point of each segment is not allowed to start a new implicit segment.
2024-02-28 14:42:08 +00:00
if (endIndex == vertices.Length - 1)
continue;
// Force a type on the last point, and return the current control point set as a segment.
vertices[endIndex - 1].Type = type;
2024-02-28 14:42:08 +00:00
for (int i = startIndex; i < endIndex; i++)
result.Add(vertices[i]);
// Skip the current control point - as it's the same as the one that's just been returned.
startIndex = endIndex + 1;
}
2024-02-28 14:42:08 +00:00
for (int i = startIndex; i < endIndex; i++)
result.Add(vertices[i]);
2024-02-28 14:42:08 +00:00
return result;
2024-02-28 14:42:08 +00:00
static bool isLinear(ReadOnlySpan<Vector2> p) => Precision.AlmostEquals(0, (p[1].Y - p[0].Y) * (p[2].X - p[0].X)
- (p[1].X - p[0].X) * (p[2].Y - p[0].Y));
}
2017-04-18 00:13:36 +00: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>
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2017-04-18 00:13:36 +00:00
/// <returns>The hit object.</returns>
protected abstract HitObject CreateHit(Vector2 position, bool newCombo, int comboOffset);
2018-04-13 09:19:50 +00:00
2017-04-18 00:13:36 +00: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>
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2017-04-18 00:13:36 +00:00
/// <param name="controlPoints">The slider control points.</param>
/// <param name="length">The slider length.</param>
/// <param name="repeatCount">The slider repeat count.</param>
/// <param name="nodeSamples">The samples to be played when the slider nodes are hit. This includes the head and tail of the slider.</param>
2017-04-18 00:13:36 +00:00
/// <returns>The hit object.</returns>
protected abstract HitObject CreateSlider(Vector2 position, bool newCombo, int comboOffset, PathControlPoint[] controlPoints, double? length, int repeatCount,
IList<IList<HitSampleInfo>> nodeSamples);
2018-04-13 09:19:50 +00:00
2017-04-18 00:13:36 +00:00
/// <summary>
/// Creates a legacy Spinner-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>
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2020-05-27 03:37:44 +00:00
/// <param name="duration">The spinner duration.</param>
2017-04-18 00:13:36 +00:00
/// <returns>The hit object.</returns>
2020-05-27 03:37:44 +00:00
protected abstract HitObject CreateSpinner(Vector2 position, bool newCombo, int comboOffset, double duration);
2018-04-13 09:19:50 +00:00
2017-05-12 07:35:57 +00: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>
/// <param name="comboOffset">When starting a new combo, the offset of the new combo relative to the current one.</param>
2020-05-31 13:39:03 +00:00
/// <param name="duration">The hold duration.</param>
2020-05-27 03:37:44 +00:00
protected abstract HitObject CreateHold(Vector2 position, bool newCombo, int comboOffset, double duration);
2018-04-13 09:19:50 +00:00
2019-12-10 11:04:37 +00:00
private List<HitSampleInfo> convertSoundType(LegacyHitSoundType type, SampleBankInfo bankInfo)
{
var soundTypes = new List<HitSampleInfo>();
if (string.IsNullOrEmpty(bankInfo.Filename))
{
soundTypes.Add(new LegacyHitSampleInfo(HitSampleInfo.HIT_NORMAL, bankInfo.BankForNormal, bankInfo.Volume, bankInfo.CustomSampleBank,
2023-05-16 07:29:24 +00:00
// 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)));
}
else
{
// 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 09:19:50 +00:00
2021-02-25 06:38:56 +00:00
if (type.HasFlagFast(LegacyHitSoundType.Finish))
soundTypes.Add(new LegacyHitSampleInfo(HitSampleInfo.HIT_FINISH, bankInfo.BankForAdditions, bankInfo.Volume, bankInfo.CustomSampleBank));
2018-04-13 09:19:50 +00:00
2021-02-25 06:38:56 +00:00
if (type.HasFlagFast(LegacyHitSoundType.Whistle))
soundTypes.Add(new LegacyHitSampleInfo(HitSampleInfo.HIT_WHISTLE, bankInfo.BankForAdditions, bankInfo.Volume, bankInfo.CustomSampleBank));
2018-04-13 09:19:50 +00:00
2021-02-25 06:38:56 +00:00
if (type.HasFlagFast(LegacyHitSoundType.Clap))
soundTypes.Add(new LegacyHitSampleInfo(HitSampleInfo.HIT_CLAP, bankInfo.BankForAdditions, bankInfo.Volume, bankInfo.CustomSampleBank));
2018-04-13 09:19:50 +00:00
return soundTypes;
}
2018-04-13 09:19:50 +00:00
private class SampleBankInfo
{
/// <summary>
/// An optional overriding filename which causes all bank/sample specifications to be ignored.
/// </summary>
2018-07-02 05:20:35 +00:00
public string Filename;
/// <summary>
/// The bank identifier to use for the base ("hitnormal") sample.
/// Transferred to <see cref="HitSampleInfo.Bank"/> when appropriate.
/// </summary>
2023-05-16 07:29:24 +00:00
[CanBeNull]
public string BankForNormal;
/// <summary>
/// The bank identifier to use for additions ("hitwhistle", "hitfinish", "hitclap").
/// Transferred to <see cref="HitSampleInfo.Bank"/> when appropriate.
/// </summary>
2023-05-16 07:29:24 +00:00
[CanBeNull]
public string BankForAdditions;
/// <summary>
/// Hit sample volume (0-100).
/// See <see cref="HitSampleInfo.Volume"/>.
/// </summary>
2017-04-21 10:51:23 +00:00
public int Volume;
2018-04-13 09:19:50 +00: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>
public int CustomSampleBank;
2018-07-02 05:20:35 +00:00
public SampleBankInfo Clone() => (SampleBankInfo)MemberwiseClone();
}
2020-12-01 06:37:51 +00:00
#nullable enable
public class LegacyHitSampleInfo : HitSampleInfo, IEquatable<LegacyHitSampleInfo>
{
2020-12-01 06:37:51 +00:00
public readonly int CustomSampleBank;
/// <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
/// using the <see cref="SkinConfiguration.LegacySetting.LayeredHitSounds"/> skin config option.
/// </remarks>
2020-12-01 06:37:51 +00:00
public readonly bool IsLayered;
2023-05-16 07:29:24 +00:00
/// <summary>
/// Whether a bank was specified locally to the relevant hitobject.
/// If <c>false</c>, a bank will be retrieved from the closest control point.
/// </summary>
public bool BankSpecified;
2020-12-01 09:09:28 +00:00
public LegacyHitSampleInfo(string name, string? bank = null, int volume = 0, int customSampleBank = 0, bool isLayered = false)
2023-05-16 07:29:24 +00:00
: base(name, bank ?? SampleControlPoint.DEFAULT_BANK, customSampleBank >= 2 ? customSampleBank.ToString() : null, volume)
2020-12-01 06:37:51 +00:00
{
CustomSampleBank = customSampleBank;
2023-05-16 07:29:24 +00:00
BankSpecified = !string.IsNullOrEmpty(bank);
2020-12-01 06:37:51 +00:00
IsLayered = isLayered;
}
public sealed override HitSampleInfo With(Optional<string> newName = default, Optional<string> newBank = default, Optional<string?> newSuffix = default, Optional<int> newVolume = default)
2020-12-02 01:55:48 +00:00
=> With(newName, newBank, newVolume);
2020-12-01 06:37:51 +00:00
public virtual LegacyHitSampleInfo With(Optional<string> newName = default, Optional<string> newBank = default, Optional<int> newVolume = default,
Optional<int> newCustomSampleBank = default,
2020-12-02 01:55:48 +00:00
Optional<bool> newIsLayered = default)
=> new LegacyHitSampleInfo(newName.GetOr(Name), newBank.GetOr(Bank), newVolume.GetOr(Volume), newCustomSampleBank.GetOr(CustomSampleBank), newIsLayered.GetOr(IsLayered));
public bool Equals(LegacyHitSampleInfo? other)
// 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.
=> base.Equals(other) && CustomSampleBank == other.CustomSampleBank && IsLayered == other.IsLayered;
public override bool Equals(object? obj)
=> obj is LegacyHitSampleInfo other && Equals(other);
public override int GetHashCode() => HashCode.Combine(base.GetHashCode(), CustomSampleBank, IsLayered);
}
private class FileHitSampleInfo : LegacyHitSampleInfo, IEquatable<FileHitSampleInfo>
2018-07-02 05:20:35 +00:00
{
2020-12-01 06:37:51 +00:00
public readonly string Filename;
2018-07-02 05:20:35 +00:00
2020-12-01 06:37:51 +00: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 12:05:07 +00:00
// Note that this does not change the lookup names, as they are overridden locally.
2020-12-01 06:37:51 +00:00
: base(string.Empty, customSampleBank: 1, volume: volume)
{
Filename = filename;
2020-04-13 11:09:17 +00:00
}
2018-07-02 05:20:35 +00:00
public override IEnumerable<string> LookupNames => new[]
{
2018-07-02 05:20:35 +00:00
Filename,
Path.ChangeExtension(Filename, null)
};
2020-12-01 06:37:51 +00: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 01:55:48 +00:00
Optional<bool> newIsLayered = default)
=> new FileHitSampleInfo(Filename, newVolume.GetOr(Volume));
public bool Equals(FileHitSampleInfo? other)
=> base.Equals(other) && Filename == other.Filename;
public override bool Equals(object? obj)
=> obj is FileHitSampleInfo other && Equals(other);
public override int GetHashCode() => HashCode.Combine(base.GetHashCode(), Filename);
}
}
}