mirror of
https://github.com/ppy/osu.git
synced 2025-01-07 23:23:12 +08:00
Preserve storyboard events when saving a beatmap in the editor
Until we have full encoding support for storyboards, this stop-gap measure ensures that storyboards don't just disappear from existence.
This commit is contained in:
parent
0c45a482eb
commit
fd3f4a9e7b
@ -37,6 +37,21 @@ namespace osu.Game.Tests.Beatmaps.Formats
|
|||||||
|
|
||||||
private static IEnumerable<string> allBeatmaps = beatmaps_resource_store.GetAvailableResources().Where(res => res.EndsWith(".osu", StringComparison.Ordinal));
|
private static IEnumerable<string> allBeatmaps = beatmaps_resource_store.GetAvailableResources().Where(res => res.EndsWith(".osu", StringComparison.Ordinal));
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestUnsupportedStoryboardEvents()
|
||||||
|
{
|
||||||
|
const string name = "Resources/storyboard_only_video.osu";
|
||||||
|
|
||||||
|
var decoded = decodeFromLegacy(beatmaps_resource_store.GetStream(name), name);
|
||||||
|
var decodedAfterEncode = decodeFromLegacy(encodeToLegacy(decoded), name);
|
||||||
|
|
||||||
|
Assert.That(decoded.beatmap.UnhandledEventLines.Count, Is.EqualTo(1));
|
||||||
|
Assert.That(decoded.beatmap.UnhandledEventLines.Single(), Is.EqualTo("Video,0,\"video.avi\""));
|
||||||
|
|
||||||
|
Assert.That(decodedAfterEncode.beatmap.UnhandledEventLines.Count, Is.EqualTo(1));
|
||||||
|
Assert.That(decodedAfterEncode.beatmap.UnhandledEventLines.Single(), Is.EqualTo("Video,0,\"video.avi\""));
|
||||||
|
}
|
||||||
|
|
||||||
[TestCaseSource(nameof(allBeatmaps))]
|
[TestCaseSource(nameof(allBeatmaps))]
|
||||||
public void TestEncodeDecodeStability(string name)
|
public void TestEncodeDecodeStability(string name)
|
||||||
{
|
{
|
||||||
|
@ -63,6 +63,8 @@ namespace osu.Game.Beatmaps
|
|||||||
|
|
||||||
public List<BreakPeriod> Breaks { get; set; } = new List<BreakPeriod>();
|
public List<BreakPeriod> Breaks { get; set; } = new List<BreakPeriod>();
|
||||||
|
|
||||||
|
public List<string> UnhandledEventLines { get; set; } = new List<string>();
|
||||||
|
|
||||||
[JsonIgnore]
|
[JsonIgnore]
|
||||||
public double TotalBreakTime => Breaks.Sum(b => b.Duration);
|
public double TotalBreakTime => Breaks.Sum(b => b.Duration);
|
||||||
|
|
||||||
|
@ -66,6 +66,7 @@ namespace osu.Game.Beatmaps
|
|||||||
beatmap.ControlPointInfo = original.ControlPointInfo;
|
beatmap.ControlPointInfo = original.ControlPointInfo;
|
||||||
beatmap.HitObjects = convertHitObjects(original.HitObjects, original, cancellationToken).OrderBy(s => s.StartTime).ToList();
|
beatmap.HitObjects = convertHitObjects(original.HitObjects, original, cancellationToken).OrderBy(s => s.StartTime).ToList();
|
||||||
beatmap.Breaks = original.Breaks;
|
beatmap.Breaks = original.Breaks;
|
||||||
|
beatmap.UnhandledEventLines = original.UnhandledEventLines;
|
||||||
|
|
||||||
return beatmap;
|
return beatmap;
|
||||||
}
|
}
|
||||||
|
@ -420,6 +420,10 @@ namespace osu.Game.Beatmaps.Formats
|
|||||||
if (!Enum.TryParse(split[0], out LegacyEventType type))
|
if (!Enum.TryParse(split[0], out LegacyEventType type))
|
||||||
throw new InvalidDataException($@"Unknown event type: {split[0]}");
|
throw new InvalidDataException($@"Unknown event type: {split[0]}");
|
||||||
|
|
||||||
|
// Until we have full storyboard encoder coverage, let's track any lines which aren't handled
|
||||||
|
// and store them to a temporary location such that they aren't lost on editor save / export.
|
||||||
|
bool lineSupportedByEncoder = false;
|
||||||
|
|
||||||
switch (type)
|
switch (type)
|
||||||
{
|
{
|
||||||
case LegacyEventType.Sprite:
|
case LegacyEventType.Sprite:
|
||||||
@ -427,7 +431,11 @@ namespace osu.Game.Beatmaps.Formats
|
|||||||
// In some older beatmaps, it is not present and replaced by a storyboard-level background instead.
|
// 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.
|
// Allow the first sprite (by file order) to act as the background in such cases.
|
||||||
if (string.IsNullOrEmpty(beatmap.BeatmapInfo.Metadata.BackgroundFile))
|
if (string.IsNullOrEmpty(beatmap.BeatmapInfo.Metadata.BackgroundFile))
|
||||||
|
{
|
||||||
beatmap.BeatmapInfo.Metadata.BackgroundFile = CleanFilename(split[3]);
|
beatmap.BeatmapInfo.Metadata.BackgroundFile = CleanFilename(split[3]);
|
||||||
|
lineSupportedByEncoder = true;
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LegacyEventType.Video:
|
case LegacyEventType.Video:
|
||||||
@ -439,12 +447,14 @@ namespace osu.Game.Beatmaps.Formats
|
|||||||
if (!OsuGameBase.VIDEO_EXTENSIONS.Contains(Path.GetExtension(filename).ToLowerInvariant()))
|
if (!OsuGameBase.VIDEO_EXTENSIONS.Contains(Path.GetExtension(filename).ToLowerInvariant()))
|
||||||
{
|
{
|
||||||
beatmap.BeatmapInfo.Metadata.BackgroundFile = filename;
|
beatmap.BeatmapInfo.Metadata.BackgroundFile = filename;
|
||||||
|
lineSupportedByEncoder = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LegacyEventType.Background:
|
case LegacyEventType.Background:
|
||||||
beatmap.BeatmapInfo.Metadata.BackgroundFile = CleanFilename(split[2]);
|
beatmap.BeatmapInfo.Metadata.BackgroundFile = CleanFilename(split[2]);
|
||||||
|
lineSupportedByEncoder = true;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LegacyEventType.Break:
|
case LegacyEventType.Break:
|
||||||
@ -452,8 +462,12 @@ namespace osu.Game.Beatmaps.Formats
|
|||||||
double end = Math.Max(start, getOffsetTime(Parsing.ParseDouble(split[2])));
|
double end = Math.Max(start, getOffsetTime(Parsing.ParseDouble(split[2])));
|
||||||
|
|
||||||
beatmap.Breaks.Add(new BreakPeriod(start, end));
|
beatmap.Breaks.Add(new BreakPeriod(start, end));
|
||||||
|
lineSupportedByEncoder = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!lineSupportedByEncoder)
|
||||||
|
beatmap.UnhandledEventLines.Add(line);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleTimingPoint(string line)
|
private void handleTimingPoint(string line)
|
||||||
|
@ -156,6 +156,9 @@ namespace osu.Game.Beatmaps.Formats
|
|||||||
|
|
||||||
foreach (var b in beatmap.Breaks)
|
foreach (var b in beatmap.Breaks)
|
||||||
writer.WriteLine(FormattableString.Invariant($"{(int)LegacyEventType.Break},{b.StartTime},{b.EndTime}"));
|
writer.WriteLine(FormattableString.Invariant($"{(int)LegacyEventType.Break},{b.StartTime},{b.EndTime}"));
|
||||||
|
|
||||||
|
foreach (string l in beatmap.UnhandledEventLines)
|
||||||
|
writer.WriteLine(l);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleControlPoints(TextWriter writer)
|
private void handleControlPoints(TextWriter writer)
|
||||||
|
@ -42,6 +42,12 @@ namespace osu.Game.Beatmaps
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
List<BreakPeriod> Breaks { get; }
|
List<BreakPeriod> Breaks { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// All lines from the [Events] section which aren't handled in the encoding process yet.
|
||||||
|
/// These lines shoule be written out to the beatmap file on save or export.
|
||||||
|
/// </summary>
|
||||||
|
List<string> UnhandledEventLines { get; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Total amount of break time in the beatmap.
|
/// Total amount of break time in the beatmap.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -330,6 +330,8 @@ namespace osu.Game.Rulesets.Difficulty
|
|||||||
}
|
}
|
||||||
|
|
||||||
public List<BreakPeriod> Breaks => baseBeatmap.Breaks;
|
public List<BreakPeriod> Breaks => baseBeatmap.Breaks;
|
||||||
|
public List<string> UnhandledEventLines => baseBeatmap.UnhandledEventLines;
|
||||||
|
|
||||||
public double TotalBreakTime => baseBeatmap.TotalBreakTime;
|
public double TotalBreakTime => baseBeatmap.TotalBreakTime;
|
||||||
public IEnumerable<BeatmapStatistic> GetStatistics() => baseBeatmap.GetStatistics();
|
public IEnumerable<BeatmapStatistic> GetStatistics() => baseBeatmap.GetStatistics();
|
||||||
public double GetMostCommonBeatLength() => baseBeatmap.GetMostCommonBeatLength();
|
public double GetMostCommonBeatLength() => baseBeatmap.GetMostCommonBeatLength();
|
||||||
|
@ -174,6 +174,8 @@ namespace osu.Game.Screens.Edit
|
|||||||
|
|
||||||
public List<BreakPeriod> Breaks => PlayableBeatmap.Breaks;
|
public List<BreakPeriod> Breaks => PlayableBeatmap.Breaks;
|
||||||
|
|
||||||
|
public List<string> UnhandledEventLines => PlayableBeatmap.UnhandledEventLines;
|
||||||
|
|
||||||
public double TotalBreakTime => PlayableBeatmap.TotalBreakTime;
|
public double TotalBreakTime => PlayableBeatmap.TotalBreakTime;
|
||||||
|
|
||||||
public IReadOnlyList<HitObject> HitObjects => PlayableBeatmap.HitObjects;
|
public IReadOnlyList<HitObject> HitObjects => PlayableBeatmap.HitObjects;
|
||||||
|
@ -27,6 +27,7 @@ namespace osu.Game.Tests.Beatmaps
|
|||||||
BeatmapInfo = baseBeatmap.BeatmapInfo;
|
BeatmapInfo = baseBeatmap.BeatmapInfo;
|
||||||
ControlPointInfo = baseBeatmap.ControlPointInfo;
|
ControlPointInfo = baseBeatmap.ControlPointInfo;
|
||||||
Breaks = baseBeatmap.Breaks;
|
Breaks = baseBeatmap.Breaks;
|
||||||
|
UnhandledEventLines = baseBeatmap.UnhandledEventLines;
|
||||||
|
|
||||||
if (withHitObjects)
|
if (withHitObjects)
|
||||||
HitObjects = baseBeatmap.HitObjects;
|
HitObjects = baseBeatmap.HitObjects;
|
||||||
|
Loading…
Reference in New Issue
Block a user