Add swap journaling and group undo/redo with extensive tests.
- Introduced SwapManager for sidecar journaling of buffer mutations, with a safe recovery mechanism. - Added group undo/redo functionality, allowing atomic grouping of related edits. - Implemented `SwapRecorder` and integrated it as a callback interface for mutations. - Added unit tests for swap journaling (save/load/replay) and undo grouping. - Refactored undo to support group tracking and ID management. - Updated CMake to include the new tests and swap journaling logic.
This commit is contained in:
91
tests/test_command_semantics.cc
Normal file
91
tests/test_command_semantics.cc
Normal file
@@ -0,0 +1,91 @@
|
||||
#include "Test.h"
|
||||
|
||||
#include "TestHarness.h"
|
||||
|
||||
using ktet::TestHarness;
|
||||
|
||||
|
||||
TEST (CommandSemantics_KillToEOL_KillChain_And_Yank)
|
||||
{
|
||||
TestHarness h;
|
||||
Editor &ed = h.EditorRef();
|
||||
Buffer &b = h.Buf();
|
||||
|
||||
b.insert_text(0, 0, std::string("abc\ndef"));
|
||||
b.SetCursor(1, 0); // a|bc
|
||||
|
||||
ed.KillRingClear();
|
||||
ed.SetKillChain(false);
|
||||
|
||||
ASSERT_TRUE(h.Exec(CommandId::KillToEOL));
|
||||
ASSERT_EQ(h.Text(), std::string("a\ndef"));
|
||||
ASSERT_EQ(ed.KillRingHead(), std::string("bc"));
|
||||
|
||||
// At EOL, KillToEOL kills the newline (join).
|
||||
ASSERT_TRUE(h.Exec(CommandId::KillToEOL));
|
||||
ASSERT_EQ(h.Text(), std::string("adef"));
|
||||
ASSERT_EQ(ed.KillRingHead(), std::string("bc\n"));
|
||||
|
||||
// Yank pastes the kill ring head and breaks the kill chain.
|
||||
ASSERT_TRUE(h.Exec(CommandId::Yank));
|
||||
ASSERT_EQ(h.Text(), std::string("abc\ndef"));
|
||||
ASSERT_EQ(ed.KillRingHead(), std::string("bc\n"));
|
||||
ASSERT_EQ(ed.KillChain(), false);
|
||||
}
|
||||
|
||||
|
||||
TEST (CommandSemantics_ToggleMark_JumpToMark)
|
||||
{
|
||||
TestHarness h;
|
||||
Buffer &b = h.Buf();
|
||||
|
||||
b.insert_text(0, 0, std::string("hello"));
|
||||
b.SetCursor(2, 0);
|
||||
ASSERT_EQ(b.MarkSet(), false);
|
||||
|
||||
ASSERT_TRUE(h.Exec(CommandId::ToggleMark));
|
||||
ASSERT_EQ(b.MarkSet(), true);
|
||||
ASSERT_EQ(b.MarkCurx(), (std::size_t) 2);
|
||||
ASSERT_EQ(b.MarkCury(), (std::size_t) 0);
|
||||
|
||||
b.SetCursor(4, 0);
|
||||
ASSERT_TRUE(h.Exec(CommandId::JumpToMark));
|
||||
ASSERT_EQ(b.Curx(), (std::size_t) 2);
|
||||
ASSERT_EQ(b.Cury(), (std::size_t) 0);
|
||||
// Jump-to-mark swaps: mark becomes previous cursor.
|
||||
ASSERT_EQ(b.MarkSet(), true);
|
||||
ASSERT_EQ(b.MarkCurx(), (std::size_t) 4);
|
||||
ASSERT_EQ(b.MarkCury(), (std::size_t) 0);
|
||||
}
|
||||
|
||||
|
||||
TEST (CommandSemantics_CopyRegion_And_KillRegion)
|
||||
{
|
||||
TestHarness h;
|
||||
Editor &ed = h.EditorRef();
|
||||
Buffer &b = h.Buf();
|
||||
|
||||
b.insert_text(0, 0, std::string("hello world"));
|
||||
b.SetCursor(0, 0);
|
||||
|
||||
ed.KillRingClear();
|
||||
ed.SetKillChain(false);
|
||||
|
||||
// Copy "hello" (region [0,5)).
|
||||
ASSERT_TRUE(h.Exec(CommandId::ToggleMark));
|
||||
b.SetCursor(5, 0);
|
||||
ASSERT_TRUE(h.Exec(CommandId::CopyRegion));
|
||||
ASSERT_EQ(ed.KillRingHead(), std::string("hello"));
|
||||
ASSERT_EQ(b.MarkSet(), false);
|
||||
ASSERT_EQ(h.Text(), std::string("hello world"));
|
||||
|
||||
// Kill "world" (region [6,11)).
|
||||
ed.SetKillChain(false);
|
||||
b.SetCursor(6, 0);
|
||||
ASSERT_TRUE(h.Exec(CommandId::ToggleMark));
|
||||
b.SetCursor(11, 0);
|
||||
ASSERT_TRUE(h.Exec(CommandId::KillRegion));
|
||||
ASSERT_EQ(ed.KillRingHead(), std::string("world"));
|
||||
ASSERT_EQ(b.MarkSet(), false);
|
||||
ASSERT_EQ(h.Text(), std::string("hello "));
|
||||
}
|
||||
Reference in New Issue
Block a user