From 530dc2a76e5a2c2c6ce05ec95ca37dbe53fa8fcb Mon Sep 17 00:00:00 2001 From: yyc12345 Date: Mon, 30 Mar 2026 10:40:50 +0800 Subject: [PATCH] feat: add before and after support for tas operation --- .../BallanceTasEditor/App.xaml.cs | 4 - .../BallanceTasEditor/Backend/TasOperation.cs | 39 ++++++++- .../Frontend/Views/NewFileDialog.xaml | 2 +- .../Backend/TasOperationTests.cs | 86 +++++++++++++++++++ .../Backend/TasSequenceTests.cs | 56 ++++++------ 5 files changed, 150 insertions(+), 37 deletions(-) create mode 100644 BallanceTasEditor/BallanceTasEditorTests/Backend/TasOperationTests.cs diff --git a/BallanceTasEditor/BallanceTasEditor/App.xaml.cs b/BallanceTasEditor/BallanceTasEditor/App.xaml.cs index 6beb8d9..9ee5511 100644 --- a/BallanceTasEditor/BallanceTasEditor/App.xaml.cs +++ b/BallanceTasEditor/BallanceTasEditor/App.xaml.cs @@ -1,7 +1,3 @@ -// Import LanguageExt globally -global using LanguageExt; -global using static LanguageExt.Prelude; - using System.Configuration; using System.Data; using System.Windows; diff --git a/BallanceTasEditor/BallanceTasEditor/Backend/TasOperation.cs b/BallanceTasEditor/BallanceTasEditor/Backend/TasOperation.cs index fd42dd1..dc046d7 100644 --- a/BallanceTasEditor/BallanceTasEditor/Backend/TasOperation.cs +++ b/BallanceTasEditor/BallanceTasEditor/Backend/TasOperation.cs @@ -1,5 +1,6 @@ using System; using System.Collections.Generic; +using System.Diagnostics; using System.Diagnostics.CodeAnalysis; using System.Linq; using System.Text; @@ -354,13 +355,19 @@ namespace BallanceTasEditor.Backend { } } + public enum InsertFrameOperationKind { + Before, After + } + public class InsertFrameOperation : ITasRevocableOperation { - public InsertFrameOperation(int index, IExactSizeEnumerable frames) { + public InsertFrameOperation(InsertFrameOperationKind kind, int index, IExactSizeEnumerable frames) { + m_Kind = kind; m_Index = index; m_InsertedFrames = frames.Select((frame) => frame.ToRaw()).ToArray(); m_IsExecuted = false; } + private InsertFrameOperationKind m_Kind; private int m_Index; private RawTasFrame[] m_InsertedFrames; private bool m_IsExecuted; @@ -375,7 +382,18 @@ namespace BallanceTasEditor.Backend { } // Check arguments - ArgumentOutOfRangeException.ThrowIfGreaterThan(m_Index, seq.GetCount()); + // If we insert before some frame, the valid index can be [0, count], + // however, if we insert after some frame, the valid index is [0, count), + switch (m_Kind) { + case InsertFrameOperationKind.Before: + ArgumentOutOfRangeException.ThrowIfGreaterThan(m_Index, seq.GetCount()); + break; + case InsertFrameOperationKind.After: + ArgumentOutOfRangeException.ThrowIfGreaterThanOrEqual(m_Index, seq.GetCount()); + break; + default: + throw new UnreachableException("Unknown InsertFrameOperationKind"); + } // Skip if count is zero var count = m_InsertedFrames.Length; @@ -383,8 +401,14 @@ namespace BallanceTasEditor.Backend { // Prepare iterator var iter = m_InsertedFrames.Select((frame) => TasFrame.FromRaw(frame)); var exactSizedIter = new ExactSizeEnumerableAdapter(iter, count); + // Compute the insert index + var index = m_Kind switch { + InsertFrameOperationKind.Before => m_Index, + InsertFrameOperationKind.After => m_Index + 1, + _ => throw new UnreachableException("Unknown InsertFrameOperationKind"), + }; // Execute inserting. - seq.Insert(m_Index, exactSizedIter); + seq.Insert(index, exactSizedIter); } // Set execution status @@ -399,7 +423,14 @@ namespace BallanceTasEditor.Backend { // Arguments were checked so we directly restore them. var count = m_InsertedFrames.Length; if (count != 0) { - seq.Remove(m_Index, m_Index + count - 1); + // Compute the index for removing + var index = m_Kind switch { + InsertFrameOperationKind.Before => m_Index, + InsertFrameOperationKind.After => m_Index + 1, + _ => throw new UnreachableException("Unknown InsertFrameOperationKind"), + }; + // Execute removing. + seq.Remove(index, index + count - 1); } // Modify execution status diff --git a/BallanceTasEditor/BallanceTasEditor/Frontend/Views/NewFileDialog.xaml b/BallanceTasEditor/BallanceTasEditor/Frontend/Views/NewFileDialog.xaml index fdcfda6..760b55e 100644 --- a/BallanceTasEditor/BallanceTasEditor/Frontend/Views/NewFileDialog.xaml +++ b/BallanceTasEditor/BallanceTasEditor/Frontend/Views/NewFileDialog.xaml @@ -3,7 +3,7 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" - xmlns:local="clr-namespace:BallanceTasEditor.Views" + xmlns:local="clr-namespace:BallanceTasEditor.Frontend.Views" xmlns:vm="clr-namespace:BallanceTasEditor.Frontend.ViewModels" xmlns:conveter="clr-namespace:BallanceTasEditor.Frontend.Converters" d:DataContext="{d:DesignInstance vm:NewFileDialog}" diff --git a/BallanceTasEditor/BallanceTasEditorTests/Backend/TasOperationTests.cs b/BallanceTasEditor/BallanceTasEditorTests/Backend/TasOperationTests.cs new file mode 100644 index 0000000..89db9ee --- /dev/null +++ b/BallanceTasEditor/BallanceTasEditorTests/Backend/TasOperationTests.cs @@ -0,0 +1,86 @@ +using BallanceTasEditor.Backend; +using Microsoft.VisualStudio.TestTools.UnitTesting; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace BallanceTasEditorTests.Backend { + [TestClass] + public class TasOperationTests { + + private static IEnumerable TasSequenceInstanceProvider { + get { + yield return new object[] { new ListTasSequence() }; + yield return new object[] { new LegacyTasSequence() }; + // TODO: Add GapBufferTasSequence once we finish it. + //yield return new object[] { new GapBufferTasSequence() }; + } + } + + /// + /// CellKeysOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void CellKeysOperationTest(ITasSequence sequence, CellKeysOperationKind kind, int startIndex, int endIndex, TasKey startKey, TasKey endKey) { + throw new NotImplementedException(); + } + + /// + /// FrameFpsOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void FrameFpsOperationTest(ITasSequence sequence) { + throw new NotImplementedException(); + } + + /// + /// RemoveFrameOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void RemoveFrameOperationTest(ITasSequence sequence) { + throw new NotImplementedException(); + } + + /// + /// AddFrameOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void AddFrameOperationTest(ITasSequence sequence) { + throw new NotImplementedException(); + } + + /// + /// InsertFrameOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void InsertFrameOperationTest(ITasSequence sequence) { + throw new NotImplementedException(); + } + + /// + /// ClearKeysOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void ClearKeysOperationTest(ITasSequence sequence) { + throw new NotImplementedException(); + } + + /// + /// UniformFpsOperation测试。 + /// + [DataTestMethod] + [DynamicData(nameof(TasSequenceInstanceProvider))] + public void UniformFpsOperationTest(ITasSequence sequence) { + throw new NotImplementedException(); + } + + } +} diff --git a/BallanceTasEditor/BallanceTasEditorTests/Backend/TasSequenceTests.cs b/BallanceTasEditor/BallanceTasEditorTests/Backend/TasSequenceTests.cs index 16eef19..f22d162 100644 --- a/BallanceTasEditor/BallanceTasEditorTests/Backend/TasSequenceTests.cs +++ b/BallanceTasEditor/BallanceTasEditorTests/Backend/TasSequenceTests.cs @@ -13,19 +13,19 @@ namespace BallanceTasEditorTests.Backend { private static readonly TasFrame[] BLANK = { }; private static readonly TasFrame[] PROBE = { - new TasFrame(10), - new TasFrame(20), - new TasFrame(30), - new TasFrame(40), - new TasFrame(50), + TasFrame.FromFps(10), + TasFrame.FromFps(20), + TasFrame.FromFps(30), + TasFrame.FromFps(40), + TasFrame.FromFps(50), }; - private static CountableEnumerable GetCountableProbe() { - return new CountableEnumerable(PROBE); + private static IExactSizeEnumerable GetExactSizeProbe() { + return new ExactSizeEnumerableAdapter(PROBE, PROBE.Length); } - private static CountableEnumerable GetCountableBlank() { - return new CountableEnumerable(BLANK); + private static IExactSizeEnumerable GetExactSizeBlank() { + return new ExactSizeEnumerableAdapter(BLANK, BLANK.Length); } private static IEnumerable TasSequenceInstanceProvider { @@ -40,7 +40,7 @@ namespace BallanceTasEditorTests.Backend { /// /// Visit函数独立测试。 /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void VisitTest(ITasSequence sequence) { // 空时访问 @@ -49,7 +49,7 @@ namespace BallanceTasEditorTests.Backend { AssertExtension.ThrowsDerivedException(() => sequence.Visit(1)); // 设置数据 - sequence.Insert(0, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); // 访问数据 AssertExtension.ThrowsDerivedException(() => sequence.Visit(-1)); for (int i = 0; i < PROBE.Length; i++) { @@ -61,16 +61,16 @@ namespace BallanceTasEditorTests.Backend { /// /// Insert函数独立测试。 /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void InsertTest(ITasSequence sequence) { // 需要在不同的存储器上,分别检测在空的时候插入, // 和在非空时的头,中,尾分别插入的结果。 // 先检测空插入 - AssertExtension.ThrowsDerivedException(() => sequence.Insert(-1, GetCountableProbe())); - AssertExtension.ThrowsDerivedException(() => sequence.Insert(1, GetCountableProbe())); - sequence.Insert(0, GetCountableProbe()); + AssertExtension.ThrowsDerivedException(() => sequence.Insert(-1, GetExactSizeProbe())); + AssertExtension.ThrowsDerivedException(() => sequence.Insert(1, GetExactSizeProbe())); + sequence.Insert(0, GetExactSizeProbe()); for (int i = 0; i < PROBE.Length; i++) { Assert.AreEqual(sequence.Visit(i), PROBE[i]); } @@ -80,8 +80,8 @@ namespace BallanceTasEditorTests.Backend { foreach (var index in indices) { // 清空,一次插入,然后二次插入 sequence.Clear(); - sequence.Insert(0, GetCountableProbe()); - sequence.Insert(index, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); + sequence.Insert(index, GetExactSizeProbe()); // 用List做正确模拟 var expected = new List(); @@ -100,7 +100,7 @@ namespace BallanceTasEditorTests.Backend { /// /// Remove函数独立测试。 /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void RemoveTest(ITasSequence sequence) { // 在空的时候删除0项 @@ -111,7 +111,7 @@ namespace BallanceTasEditorTests.Backend { foreach (var index in indices) { // 清空,插入,删除 sequence.Clear(); - sequence.Insert(0, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); sequence.Remove(index, 1); // 用List做正确模拟 @@ -130,11 +130,11 @@ namespace BallanceTasEditorTests.Backend { /// /// Clear函数独立测试。 /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void ClearTest(ITasSequence sequence) { // 设置数据后清空 - sequence.Insert(0, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); sequence.Clear(); // 检查是否为空 @@ -144,28 +144,28 @@ namespace BallanceTasEditorTests.Backend { /// /// IsEmpty函数独立测试。 /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void IsEmptyTest(ITasSequence sequence) { // 检查是否为空 Assert.IsTrue(sequence.IsEmpty()); // 插入数据后再检查 - sequence.Insert(0, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); Assert.IsFalse(sequence.IsEmpty()); } /// /// GetCount函数独立测试。 /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void GetCountTest(ITasSequence sequence) { // 检查长度为0 Assert.AreEqual(sequence.GetCount(), 0); // 插入数据后再检查 - sequence.Insert(0, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); Assert.AreEqual(sequence.GetCount(), PROBE.Length); } @@ -173,7 +173,7 @@ namespace BallanceTasEditorTests.Backend { /// 混合检查Visit,Clear,GetCount,IsEmpty。 /// /// - [TestMethod] + [DataTestMethod] [DynamicData(nameof(TasSequenceInstanceProvider))] public void HybridTest(ITasSequence sequence) { // 检查空和大小 @@ -181,7 +181,7 @@ namespace BallanceTasEditorTests.Backend { Assert.AreEqual(sequence.GetCount(), 0); // 设置内容 - sequence.Insert(0, GetCountableProbe()); + sequence.Insert(0, GetExactSizeProbe()); // 并再次检查大小 Assert.IsFalse(sequence.IsEmpty()); Assert.AreEqual(sequence.GetCount(), PROBE.Length); @@ -200,7 +200,7 @@ namespace BallanceTasEditorTests.Backend { // 清空后插入0项,然后确认 sequence.Clear(); - sequence.Insert(0, GetCountableBlank()); + sequence.Insert(0, GetExactSizeBlank()); AssertExtension.ThrowsDerivedException(() => sequence.Visit(-1)); AssertExtension.ThrowsDerivedException(() => sequence.Visit(0)); AssertExtension.ThrowsDerivedException(() => sequence.Visit(1));