osu/osu.Game/Screens/Edit/LegacyEditorBeatmapPatcher.cs

124 lines
4.2 KiB
C#
Raw Normal View History

2020-04-09 11:48:59 +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.
using System;
using System.Collections.Generic;
2020-11-07 15:18:25 +00:00
using System.Diagnostics;
2020-04-09 11:48:59 +00:00
using System.IO;
using System.Text;
2020-04-09 11:48:59 +00:00
using DiffPlex;
using osu.Framework.Audio.Track;
using osu.Framework.Graphics.Textures;
using osu.Game.Beatmaps;
using osu.Game.IO;
using Decoder = osu.Game.Beatmaps.Formats.Decoder;
2020-04-09 11:48:59 +00:00
namespace osu.Game.Screens.Edit
{
2020-04-13 08:20:01 +00:00
/// <summary>
/// Patches an <see cref="EditorBeatmap"/> based on the difference between two legacy (.osu) states.
/// </summary>
public class LegacyEditorBeatmapPatcher
2020-04-09 11:48:59 +00:00
{
private readonly EditorBeatmap editorBeatmap;
2020-04-13 08:20:01 +00:00
public LegacyEditorBeatmapPatcher(EditorBeatmap editorBeatmap)
2020-04-09 11:48:59 +00:00
{
this.editorBeatmap = editorBeatmap;
}
public void Patch(byte[] currentState, byte[] newState)
2020-04-09 11:48:59 +00:00
{
// Diff the beatmaps
var result = new Differ().CreateLineDiffs(readString(currentState), readString(newState), true, false);
// Find the index of [HitObject] sections. Lines changed prior to this index are ignored.
int oldHitObjectsIndex = Array.IndexOf(result.PiecesOld, "[HitObjects]");
int newHitObjectsIndex = Array.IndexOf(result.PiecesNew, "[HitObjects]");
2020-11-07 15:18:25 +00:00
Debug.Assert(oldHitObjectsIndex >= 0);
Debug.Assert(newHitObjectsIndex >= 0);
2020-04-09 11:48:59 +00:00
var toRemove = new List<int>();
var toAdd = new List<int>();
foreach (var block in result.DiffBlocks)
{
// Removed hitobjects
2020-04-09 11:48:59 +00:00
for (int i = 0; i < block.DeleteCountA; i++)
{
int hoIndex = block.DeleteStartA + i - oldHitObjectsIndex - 1;
if (hoIndex < 0)
continue;
toRemove.Add(hoIndex);
}
// Added hitobjects
2020-04-09 11:48:59 +00:00
for (int i = 0; i < block.InsertCountB; i++)
{
int hoIndex = block.InsertStartB + i - newHitObjectsIndex - 1;
if (hoIndex < 0)
continue;
toAdd.Add(hoIndex);
}
}
// Sort the indices to ensure that removal + insertion indices don't get jumbled up post-removal or post-insertion.
// This isn't strictly required, but the differ makes no guarantees about order.
2020-04-09 11:48:59 +00:00
toRemove.Sort();
toAdd.Sort();
2020-04-09 11:48:59 +00:00
editorBeatmap.BeginChange();
// Apply the changes.
for (int i = toRemove.Count - 1; i >= 0; i--)
editorBeatmap.RemoveAt(toRemove[i]);
if (toAdd.Count > 0)
2020-04-09 11:48:59 +00:00
{
IBeatmap newBeatmap = readBeatmap(newState);
foreach (var i in toAdd)
editorBeatmap.Insert(i, newBeatmap.HitObjects[i]);
}
editorBeatmap.EndChange();
2020-04-09 11:48:59 +00:00
}
private string readString(byte[] state) => Encoding.UTF8.GetString(state);
2020-04-09 11:48:59 +00:00
private IBeatmap readBeatmap(byte[] state)
2020-04-09 11:48:59 +00:00
{
using (var stream = new MemoryStream(state))
2020-04-09 11:48:59 +00:00
using (var reader = new LineBufferedReader(stream, true))
2020-04-30 11:03:46 +00:00
{
var decoded = Decoder.GetDecoder<Beatmap>(reader).Decode(reader);
decoded.BeatmapInfo.Ruleset = editorBeatmap.BeatmapInfo.Ruleset;
return new PassThroughWorkingBeatmap(decoded).GetPlayableBeatmap(editorBeatmap.BeatmapInfo.Ruleset);
}
2020-04-09 11:48:59 +00:00
}
private class PassThroughWorkingBeatmap : WorkingBeatmap
{
private readonly IBeatmap beatmap;
public PassThroughWorkingBeatmap(IBeatmap beatmap)
: base(beatmap.BeatmapInfo, null)
{
this.beatmap = beatmap;
}
protected override IBeatmap GetBeatmap() => beatmap;
protected override Texture GetBackground() => throw new NotImplementedException();
2020-08-07 13:31:41 +00:00
protected override Track GetBeatmapTrack() => throw new NotImplementedException();
public override Stream GetStream(string storagePath) => throw new NotImplementedException();
2020-04-09 11:48:59 +00:00
}
}
}