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

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

178 lines
6.9 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.
2022-06-17 07:37:17 +00:00
#nullable disable
2020-04-09 11:48:59 +00:00
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
2020-04-09 11:48:59 +00:00
using DiffPlex;
2022-06-13 06:40:11 +00:00
using DiffPlex.Model;
2020-04-09 11:48:59 +00:00
using osu.Framework.Audio.Track;
using osu.Framework.Graphics.Textures;
using osu.Game.Beatmaps;
using osu.Game.Beatmaps.ControlPoints;
2022-06-13 06:40:11 +00:00
using osu.Game.Beatmaps.Formats;
2020-04-09 11:48:59 +00:00
using osu.Game.IO;
using osu.Game.Skinning;
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);
2022-06-13 06:40:11 +00:00
IBeatmap newBeatmap = null;
editorBeatmap.BeginChange();
processHitObjects(result, () => newBeatmap ??= readBeatmap(newState));
2022-06-21 03:11:44 +00:00
processTimingPoints(() => newBeatmap ??= readBeatmap(newState));
2022-06-13 06:40:11 +00:00
editorBeatmap.EndChange();
}
2022-06-21 03:11:44 +00:00
private void processTimingPoints(Func<IBeatmap> getNewBeatmap)
2022-06-13 06:40:11 +00:00
{
ControlPointInfo newControlPoints = EditorBeatmap.ConvertControlPoints(getNewBeatmap().ControlPointInfo);
2022-06-13 06:40:11 +00:00
// Remove all groups from the current beatmap which don't have a corresponding equal group in the new beatmap.
foreach (var oldGroup in editorBeatmap.ControlPointInfo.Groups.ToArray())
{
var newGroup = newControlPoints.GroupAt(oldGroup.Time);
2022-06-13 06:40:11 +00:00
if (!oldGroup.Equals(newGroup))
editorBeatmap.ControlPointInfo.RemoveGroup(oldGroup);
}
// Add all groups from the new beatmap which don't have a corresponding equal group in the old beatmap.
foreach (var newGroup in newControlPoints.Groups)
{
var oldGroup = editorBeatmap.ControlPointInfo.GroupAt(newGroup.Time);
if (!newGroup.Equals(oldGroup))
{
foreach (var point in newGroup.ControlPoints)
editorBeatmap.ControlPointInfo.Add(newGroup.Time, point);
}
}
2022-06-13 06:40:11 +00:00
}
private void processHitObjects(DiffResult result, Func<IBeatmap> getNewBeatmap)
{
findChangedIndices(result, LegacyDecoder<Beatmap>.Section.HitObjects, out var removedIndices, out var addedIndices);
2022-06-13 08:36:32 +00:00
for (int i = removedIndices.Count - 1; i >= 0; i--)
editorBeatmap.RemoveAt(removedIndices[i]);
2022-06-13 06:40:11 +00:00
if (addedIndices.Count > 0)
{
var newBeatmap = getNewBeatmap();
foreach (int i in addedIndices)
editorBeatmap.Insert(i, newBeatmap.HitObjects[i]);
}
}
private void findChangedIndices(DiffResult result, LegacyDecoder<Beatmap>.Section section, out List<int> removedIndices, out List<int> addedIndices)
{
removedIndices = new List<int>();
addedIndices = new List<int>();
2020-04-09 11:48:59 +00:00
2022-06-13 08:36:32 +00:00
// Find the start and end indices of the relevant section headers in both the old and the new beatmap file. Lines changed outside of the modified ranges are ignored.
2022-06-13 06:40:11 +00:00
int oldSectionStartIndex = Array.IndexOf(result.PiecesOld, $"[{section}]");
2022-06-13 07:56:08 +00:00
if (oldSectionStartIndex == -1)
return;
2022-06-13 06:40:11 +00:00
2022-06-13 07:56:08 +00:00
int oldSectionEndIndex = Array.FindIndex(result.PiecesOld, oldSectionStartIndex + 1, s => s.StartsWith('['));
2022-06-13 06:40:11 +00:00
if (oldSectionEndIndex == -1)
oldSectionEndIndex = result.PiecesOld.Length;
int newSectionStartIndex = Array.IndexOf(result.PiecesNew, $"[{section}]");
2022-06-13 07:56:08 +00:00
if (newSectionStartIndex == -1)
return;
2020-04-09 11:48:59 +00:00
2022-06-13 07:56:08 +00:00
int newSectionEndIndex = Array.FindIndex(result.PiecesNew, newSectionStartIndex + 1, s => s.StartsWith('['));
2022-06-13 06:40:11 +00:00
if (newSectionEndIndex == -1)
2022-06-13 07:55:46 +00:00
newSectionEndIndex = result.PiecesNew.Length;
2020-11-07 15:18:25 +00:00
2020-04-09 11:48:59 +00:00
foreach (var block in result.DiffBlocks)
{
2022-06-13 06:40:11 +00:00
// Removed indices
2020-04-09 11:48:59 +00:00
for (int i = 0; i < block.DeleteCountA; i++)
{
2022-06-13 06:40:11 +00:00
int objectIndex = block.DeleteStartA + i;
2020-04-09 11:48:59 +00:00
2022-06-13 06:40:11 +00:00
if (objectIndex <= oldSectionStartIndex || objectIndex >= oldSectionEndIndex)
2020-04-09 11:48:59 +00:00
continue;
2022-06-13 06:40:11 +00:00
removedIndices.Add(objectIndex - oldSectionStartIndex - 1);
2020-04-09 11:48:59 +00:00
}
2022-06-13 06:40:11 +00:00
// Added indices
2020-04-09 11:48:59 +00:00
for (int i = 0; i < block.InsertCountB; i++)
{
2022-06-13 06:40:11 +00:00
int objectIndex = block.InsertStartB + i;
2020-04-09 11:48:59 +00:00
2022-06-13 06:40:11 +00:00
if (objectIndex <= newSectionStartIndex || objectIndex >= newSectionEndIndex)
2020-04-09 11:48:59 +00:00
continue;
2022-06-13 06:40:11 +00:00
addedIndices.Add(objectIndex - newSectionStartIndex - 1);
2020-04-09 11:48:59 +00:00
}
}
// 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.
2022-06-13 06:40:11 +00:00
removedIndices.Sort();
addedIndices.Sort();
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();
protected internal override ISkin GetSkin() => throw new NotImplementedException();
public override Stream GetStream(string storagePath) => throw new NotImplementedException();
2020-04-09 11:48:59 +00:00
}
}
}