2017-03-14 12:02:42 +08:00
|
|
|
|
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
|
|
|
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
|
|
|
|
|
|
|
|
|
using OpenTK;
|
2017-04-18 15:05:58 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
2016-12-06 17:56:20 +08:00
|
|
|
|
using System;
|
2016-11-14 21:03:39 +08:00
|
|
|
|
using System.Collections.Generic;
|
2016-11-17 20:29:35 +08:00
|
|
|
|
using System.Globalization;
|
2017-04-06 08:43:47 +08:00
|
|
|
|
using osu.Game.Beatmaps.Formats;
|
2017-04-06 10:41:16 +08:00
|
|
|
|
using osu.Game.Audio;
|
2016-11-14 21:03:39 +08:00
|
|
|
|
|
2017-04-18 15:05:58 +08:00
|
|
|
|
namespace osu.Game.Rulesets.Objects.Legacy
|
2016-11-14 21:03:39 +08:00
|
|
|
|
{
|
2017-04-18 08:13:36 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// A HitObjectParser to parse legacy Beatmaps.
|
|
|
|
|
/// </summary>
|
2017-04-17 16:23:11 +08:00
|
|
|
|
internal abstract class HitObjectParser : Objects.HitObjectParser
|
2016-11-14 21:03:39 +08:00
|
|
|
|
{
|
|
|
|
|
public override HitObject Parse(string text)
|
|
|
|
|
{
|
|
|
|
|
string[] split = text.Split(',');
|
2017-04-17 16:23:11 +08:00
|
|
|
|
var type = (HitObjectType)int.Parse(split[3]) & ~HitObjectType.ColourHax;
|
|
|
|
|
bool combo = type.HasFlag(HitObjectType.NewCombo);
|
|
|
|
|
type &= ~HitObjectType.NewCombo;
|
2017-03-13 18:15:25 +08:00
|
|
|
|
|
2017-04-21 13:36:28 +08:00
|
|
|
|
var soundType = (LegacySoundType)int.Parse(split[4]);
|
|
|
|
|
|
|
|
|
|
SampleBankInfo bankInfo = new SampleBankInfo();
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
2017-03-13 18:15:25 +08:00
|
|
|
|
HitObject result;
|
2017-03-14 18:25:04 +08:00
|
|
|
|
|
2017-04-17 16:23:11 +08:00
|
|
|
|
if ((type & HitObjectType.Circle) > 0)
|
2016-11-14 21:03:39 +08:00
|
|
|
|
{
|
2017-04-17 16:23:11 +08:00
|
|
|
|
result = CreateHit(new Vector2(int.Parse(split[0]), int.Parse(split[1])), combo);
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
|
|
|
|
if (split.Length > 5)
|
2017-04-21 13:36:28 +08:00
|
|
|
|
readCustomSampleBanks(split[5], bankInfo);
|
2017-03-14 18:25:04 +08:00
|
|
|
|
}
|
2017-04-17 16:23:11 +08:00
|
|
|
|
else if ((type & HitObjectType.Slider) > 0)
|
2017-03-14 18:25:04 +08:00
|
|
|
|
{
|
|
|
|
|
CurveType curveType = CurveType.Catmull;
|
|
|
|
|
double length = 0;
|
2017-04-17 16:23:11 +08:00
|
|
|
|
var points = new List<Vector2> { new Vector2(int.Parse(split[0]), int.Parse(split[1])) };
|
2016-11-17 20:29:35 +08:00
|
|
|
|
|
2017-03-14 18:25:04 +08:00
|
|
|
|
string[] pointsplit = split[5].Split('|');
|
|
|
|
|
foreach (string t in pointsplit)
|
|
|
|
|
{
|
|
|
|
|
if (t.Length == 1)
|
2016-11-17 20:29:35 +08:00
|
|
|
|
{
|
2017-03-14 18:25:04 +08:00
|
|
|
|
switch (t)
|
2016-11-17 20:29:35 +08:00
|
|
|
|
{
|
2017-03-14 18:25:04 +08:00
|
|
|
|
case @"C":
|
|
|
|
|
curveType = CurveType.Catmull;
|
|
|
|
|
break;
|
|
|
|
|
case @"B":
|
|
|
|
|
curveType = CurveType.Bezier;
|
|
|
|
|
break;
|
|
|
|
|
case @"L":
|
|
|
|
|
curveType = CurveType.Linear;
|
|
|
|
|
break;
|
|
|
|
|
case @"P":
|
|
|
|
|
curveType = CurveType.PerfectCurve;
|
|
|
|
|
break;
|
2016-11-17 20:29:35 +08:00
|
|
|
|
}
|
2017-03-14 18:25:04 +08:00
|
|
|
|
continue;
|
2016-11-17 20:29:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-14 18:25:04 +08:00
|
|
|
|
string[] temp = t.Split(':');
|
2017-04-17 16:23:11 +08:00
|
|
|
|
points.Add(new Vector2((int)Convert.ToDouble(temp[0], CultureInfo.InvariantCulture), (int)Convert.ToDouble(temp[1], CultureInfo.InvariantCulture)));
|
2017-03-14 18:25:04 +08:00
|
|
|
|
}
|
2016-11-17 20:29:35 +08:00
|
|
|
|
|
2017-03-14 18:25:04 +08:00
|
|
|
|
int repeatCount = Convert.ToInt32(split[6], CultureInfo.InvariantCulture);
|
2016-11-17 20:29:35 +08:00
|
|
|
|
|
2017-03-14 18:25:04 +08:00
|
|
|
|
if (repeatCount > 9000)
|
|
|
|
|
throw new ArgumentOutOfRangeException(nameof(repeatCount), @"Repeat count is way too high");
|
2016-11-17 20:29:35 +08:00
|
|
|
|
|
2017-03-14 18:25:04 +08:00
|
|
|
|
if (split.Length > 7)
|
|
|
|
|
length = Convert.ToDouble(split[7], CultureInfo.InvariantCulture);
|
2017-04-21 13:36:28 +08:00
|
|
|
|
|
|
|
|
|
if (split.Length > 10)
|
|
|
|
|
readCustomSampleBanks(split[10], bankInfo);
|
2017-03-14 18:25:04 +08:00
|
|
|
|
|
2017-04-17 16:23:11 +08:00
|
|
|
|
result = CreateSlider(new Vector2(int.Parse(split[0]), int.Parse(split[1])), combo, points, length, curveType, repeatCount);
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
2017-03-14 18:25:04 +08:00
|
|
|
|
}
|
2017-04-17 16:23:11 +08:00
|
|
|
|
else if ((type & HitObjectType.Spinner) > 0)
|
2017-03-14 18:25:04 +08:00
|
|
|
|
{
|
2017-04-17 16:23:11 +08:00
|
|
|
|
result = CreateSpinner(new Vector2(512, 384) / 2, Convert.ToDouble(split[5], CultureInfo.InvariantCulture));
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
|
|
|
|
if (split.Length > 6)
|
2017-04-21 13:36:28 +08:00
|
|
|
|
readCustomSampleBanks(split[6], bankInfo);
|
2016-11-14 21:03:39 +08:00
|
|
|
|
}
|
2017-04-17 16:23:11 +08:00
|
|
|
|
else if ((type & HitObjectType.Hold) > 0)
|
2017-03-14 18:25:04 +08:00
|
|
|
|
{
|
|
|
|
|
// Note: Hold is generated by BMS converts
|
2017-04-06 10:41:16 +08:00
|
|
|
|
|
|
|
|
|
// Todo: Apparently end time is determined by samples??
|
2017-04-06 10:47:54 +08:00
|
|
|
|
// Shouldn't need implementation until mania
|
2017-04-06 10:41:16 +08:00
|
|
|
|
|
2017-04-17 16:23:11 +08:00
|
|
|
|
result = new Hold
|
2017-03-14 18:25:04 +08:00
|
|
|
|
{
|
|
|
|
|
Position = new Vector2(int.Parse(split[0]), int.Parse(split[1])),
|
|
|
|
|
NewCombo = combo
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
throw new InvalidOperationException($@"Unknown hit object type {type}");
|
|
|
|
|
|
2017-02-14 17:55:54 +08:00
|
|
|
|
result.StartTime = Convert.ToDouble(split[2], CultureInfo.InvariantCulture);
|
2017-04-21 13:36:28 +08:00
|
|
|
|
result.Samples = convertSoundType(soundType, bankInfo);
|
2017-03-13 18:15:25 +08:00
|
|
|
|
|
2016-11-14 21:03:39 +08:00
|
|
|
|
return result;
|
|
|
|
|
}
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
2017-04-21 13:36:28 +08:00
|
|
|
|
private void readCustomSampleBanks(string str, SampleBankInfo bankInfo)
|
2017-04-06 08:43:47 +08:00
|
|
|
|
{
|
|
|
|
|
if (string.IsNullOrEmpty(str))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
string[] split = str.Split(':');
|
|
|
|
|
|
2017-04-06 10:41:16 +08:00
|
|
|
|
var bank = (OsuLegacyDecoder.LegacySampleBank)Convert.ToInt32(split[0]);
|
|
|
|
|
var addbank = (OsuLegacyDecoder.LegacySampleBank)Convert.ToInt32(split[1]);
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
2017-04-06 10:47:54 +08:00
|
|
|
|
// Let's not implement this for now, because this doesn't fit nicely into the bank structure
|
2017-04-06 08:43:47 +08:00
|
|
|
|
//string sampleFile = split2.Length > 4 ? split2[4] : string.Empty;
|
|
|
|
|
|
2017-04-06 10:41:16 +08:00
|
|
|
|
string stringBank = bank.ToString().ToLower();
|
|
|
|
|
if (stringBank == @"none")
|
2017-04-06 11:27:35 +08:00
|
|
|
|
stringBank = null;
|
2017-04-06 10:41:16 +08:00
|
|
|
|
string stringAddBank = addbank.ToString().ToLower();
|
|
|
|
|
if (stringAddBank == @"none")
|
2017-04-06 11:27:35 +08:00
|
|
|
|
stringAddBank = null;
|
2017-04-06 08:43:47 +08:00
|
|
|
|
|
2017-04-21 13:36:28 +08:00
|
|
|
|
bankInfo.Normal = stringBank;
|
|
|
|
|
bankInfo.Add = stringAddBank;
|
|
|
|
|
|
|
|
|
|
if (split.Length > 3)
|
|
|
|
|
bankInfo.Volume = int.Parse(split[3]);
|
2017-04-06 08:43:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-18 08:13:36 +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>
|
|
|
|
|
/// <returns>The hit object.</returns>
|
|
|
|
|
protected abstract HitObject CreateHit(Vector2 position, bool newCombo);
|
|
|
|
|
|
|
|
|
|
/// <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="controlPoints">The slider control points.</param>
|
|
|
|
|
/// <param name="length">The slider length.</param>
|
|
|
|
|
/// <param name="curveType">The slider curve type.</param>
|
|
|
|
|
/// <param name="repeatCount">The slider repeat count.</param>
|
|
|
|
|
/// <returns>The hit object.</returns>
|
|
|
|
|
protected abstract HitObject CreateSlider(Vector2 position, bool newCombo, List<Vector2> controlPoints, double length, CurveType curveType, int repeatCount);
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a legacy Spinner-type hit object.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="position">The position of the hit object.</param>
|
|
|
|
|
/// <param name="endTime">The spinner end time.</param>
|
|
|
|
|
/// <returns>The hit object.</returns>
|
|
|
|
|
protected abstract HitObject CreateSpinner(Vector2 position, double endTime);
|
|
|
|
|
|
2017-04-21 13:36:28 +08:00
|
|
|
|
private List<SampleInfo> convertSoundType(LegacySoundType type, SampleBankInfo bankInfo)
|
2017-04-21 12:51:40 +08:00
|
|
|
|
{
|
|
|
|
|
List<SampleInfo> soundTypes = new List<SampleInfo>();
|
|
|
|
|
|
|
|
|
|
if ((type & LegacySoundType.Normal) > 0)
|
|
|
|
|
{
|
|
|
|
|
soundTypes.Add(new SampleInfo
|
|
|
|
|
{
|
2017-04-21 13:36:28 +08:00
|
|
|
|
Bank = bankInfo.Normal,
|
|
|
|
|
Name = SampleInfo.HIT_NORMAL,
|
|
|
|
|
Volume = bankInfo.Volume
|
2017-04-21 12:51:40 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((type & LegacySoundType.Finish) > 0)
|
|
|
|
|
{
|
|
|
|
|
soundTypes.Add(new SampleInfo
|
|
|
|
|
{
|
2017-04-21 13:36:28 +08:00
|
|
|
|
Bank = bankInfo.Add,
|
|
|
|
|
Name = SampleInfo.HIT_FINISH,
|
|
|
|
|
Volume = bankInfo.Volume
|
2017-04-21 12:51:40 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((type & LegacySoundType.Whistle) > 0)
|
|
|
|
|
{
|
|
|
|
|
soundTypes.Add(new SampleInfo
|
|
|
|
|
{
|
2017-04-21 13:36:28 +08:00
|
|
|
|
Bank = bankInfo.Add,
|
|
|
|
|
Name = SampleInfo.HIT_WHISTLE,
|
|
|
|
|
Volume = bankInfo.Volume
|
2017-04-21 12:51:40 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((type & LegacySoundType.Clap) > 0)
|
|
|
|
|
{
|
|
|
|
|
soundTypes.Add(new SampleInfo
|
|
|
|
|
{
|
2017-04-21 13:36:28 +08:00
|
|
|
|
Bank = bankInfo.Add,
|
|
|
|
|
Name = SampleInfo.HIT_CLAP,
|
|
|
|
|
Volume = bankInfo.Volume
|
2017-04-21 12:51:40 +08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return soundTypes;
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-21 13:36:28 +08:00
|
|
|
|
private class SampleBankInfo
|
|
|
|
|
{
|
|
|
|
|
public string Normal = null;
|
|
|
|
|
public string Add = null;
|
|
|
|
|
public int Volume = 0;
|
|
|
|
|
|
|
|
|
|
public SampleBankInfo Clone()
|
|
|
|
|
{
|
|
|
|
|
return (SampleBankInfo)MemberwiseClone();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-06 08:43:47 +08:00
|
|
|
|
[Flags]
|
|
|
|
|
private enum LegacySoundType
|
|
|
|
|
{
|
|
|
|
|
None = 0,
|
|
|
|
|
Normal = 1,
|
|
|
|
|
Whistle = 2,
|
|
|
|
|
Finish = 4,
|
|
|
|
|
Clap = 8
|
|
|
|
|
}
|
2016-11-14 21:03:39 +08:00
|
|
|
|
}
|
|
|
|
|
}
|