* Removed using namespace directive from header files and put these in cpp files where they are used * Fixed tabbing issues by replacing them with spaces. Also regrouped the using directives. * Update src/host/exemain.cpp Co-Authored-By: Mike Griese <migrie@microsoft.com> * Update src/interactivity/win32/find.cpp Co-Authored-By: Mike Griese <migrie@microsoft.com>
1969 lines
72 KiB
C++
1969 lines
72 KiB
C++
// Copyright (c) Microsoft Corporation.
|
|
// Licensed under the MIT license.
|
|
|
|
#include "precomp.h"
|
|
#include "WexTestClass.h"
|
|
#include "../inc/consoletaeftemplates.hpp"
|
|
|
|
#include "CommonState.hpp"
|
|
|
|
#include "globals.h"
|
|
#include "../buffer/out/textBuffer.hpp"
|
|
#include "../buffer/out/CharRow.hpp"
|
|
|
|
#include "input.h"
|
|
#include "_stream.h"
|
|
|
|
#include "../interactivity/inc/ServiceLocator.hpp"
|
|
#include "../renderer/inc/DummyRenderTarget.hpp"
|
|
|
|
using namespace Microsoft::Console::Types;
|
|
using namespace Microsoft::Console::Interactivity;
|
|
using namespace Microsoft::Console::VirtualTerminal;
|
|
using namespace WEX::Common;
|
|
using namespace WEX::Logging;
|
|
using namespace WEX::TestExecution;
|
|
|
|
class TextBufferTests
|
|
{
|
|
DummyRenderTarget _renderTarget;
|
|
CommonState* m_state;
|
|
|
|
TEST_CLASS(TextBufferTests);
|
|
|
|
TEST_CLASS_SETUP(ClassSetup)
|
|
{
|
|
m_state = new CommonState();
|
|
|
|
m_state->PrepareGlobalFont();
|
|
m_state->PrepareGlobalScreenBuffer();
|
|
|
|
return true;
|
|
}
|
|
|
|
TEST_CLASS_CLEANUP(ClassCleanup)
|
|
{
|
|
m_state->CleanupGlobalScreenBuffer();
|
|
m_state->CleanupGlobalFont();
|
|
|
|
delete m_state;
|
|
|
|
return true;
|
|
}
|
|
|
|
TEST_METHOD_SETUP(MethodSetup)
|
|
{
|
|
m_state->PrepareNewTextBufferInfo();
|
|
|
|
return true;
|
|
}
|
|
|
|
TEST_METHOD_CLEANUP(MethodCleanup)
|
|
{
|
|
m_state->CleanupNewTextBufferInfo();
|
|
|
|
return true;
|
|
}
|
|
|
|
TEST_METHOD(TestBufferCreate);
|
|
|
|
TextBuffer& GetTbi();
|
|
|
|
SHORT GetBufferWidth();
|
|
|
|
SHORT GetBufferHeight();
|
|
|
|
TEST_METHOD(TestBufferRowByOffset);
|
|
|
|
TEST_METHOD(TestWrapFlag);
|
|
|
|
TEST_METHOD(TestDoubleBytePadFlag);
|
|
|
|
void DoBoundaryTest(PWCHAR const pwszInputString,
|
|
short const cLength,
|
|
short const cMax,
|
|
short const cLeft,
|
|
short const cRight);
|
|
|
|
TEST_METHOD(TestBoundaryMeasuresRegularString);
|
|
|
|
TEST_METHOD(TestBoundaryMeasuresFloatingString);
|
|
|
|
TEST_METHOD(TestCopyProperties);
|
|
|
|
TEST_METHOD(TestInsertCharacter);
|
|
|
|
TEST_METHOD(TestIncrementCursor);
|
|
|
|
TEST_METHOD(TestNewlineCursor);
|
|
|
|
void TestLastNonSpace(short const cursorPosY);
|
|
|
|
TEST_METHOD(TestGetLastNonSpaceCharacter);
|
|
|
|
TEST_METHOD(TestSetWrapOnCurrentRow);
|
|
|
|
TEST_METHOD(TestIncrementCircularBuffer);
|
|
|
|
TEST_METHOD(TestMixedRgbAndLegacyForeground);
|
|
TEST_METHOD(TestMixedRgbAndLegacyBackground);
|
|
TEST_METHOD(TestMixedRgbAndLegacyUnderline);
|
|
TEST_METHOD(TestMixedRgbAndLegacyBrightness);
|
|
|
|
TEST_METHOD(TestRgbEraseLine);
|
|
|
|
TEST_METHOD(TestUnBold);
|
|
TEST_METHOD(TestUnBoldRgb);
|
|
TEST_METHOD(TestComplexUnBold);
|
|
|
|
TEST_METHOD(CopyAttrs);
|
|
|
|
TEST_METHOD(EmptySgrTest);
|
|
|
|
TEST_METHOD(TestReverseReset);
|
|
|
|
TEST_METHOD(CopyLastAttr);
|
|
|
|
TEST_METHOD(TestRgbThenBold);
|
|
TEST_METHOD(TestResetClearsBoldness);
|
|
|
|
TEST_METHOD(TestBackspaceRightSideVt);
|
|
|
|
TEST_METHOD(TestBackspaceStrings);
|
|
TEST_METHOD(TestBackspaceStringsAPI);
|
|
|
|
TEST_METHOD(TestRepeatCharacter);
|
|
|
|
TEST_METHOD(ResizeTraditional);
|
|
|
|
TEST_METHOD(ResizeTraditionalRotationPreservesHighUnicode);
|
|
TEST_METHOD(ScrollBufferRotationPreservesHighUnicode);
|
|
|
|
TEST_METHOD(ResizeTraditionalHighUnicodeRowRemoval);
|
|
TEST_METHOD(ResizeTraditionalHighUnicodeColumnRemoval);
|
|
|
|
TEST_METHOD(TestBurrito);
|
|
|
|
};
|
|
|
|
void TextBufferTests::TestBufferCreate()
|
|
{
|
|
VERIFY_SUCCESS_NTSTATUS(m_state->GetTextBufferInfoInitResult());
|
|
}
|
|
|
|
TextBuffer& TextBufferTests::GetTbi()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
return gci.GetActiveOutputBuffer().GetTextBuffer();
|
|
}
|
|
|
|
SHORT TextBufferTests::GetBufferWidth()
|
|
{
|
|
return GetTbi().GetSize().Width();
|
|
}
|
|
|
|
SHORT TextBufferTests::GetBufferHeight()
|
|
{
|
|
return GetTbi().GetSize().Height();
|
|
}
|
|
|
|
void TextBufferTests::TestBufferRowByOffset()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
SHORT csBufferHeight = GetBufferHeight();
|
|
|
|
VERIFY_IS_TRUE(csBufferHeight > 20);
|
|
|
|
short sId = csBufferHeight / 2 - 5;
|
|
|
|
const ROW& row = textBuffer.GetRowByOffset(sId);
|
|
VERIFY_ARE_EQUAL(row.GetId(), sId);
|
|
}
|
|
|
|
void TextBufferTests::TestWrapFlag()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
ROW& Row = textBuffer._GetFirstRow();
|
|
|
|
// no wrap by default
|
|
VERIFY_IS_FALSE(Row.GetCharRow().WasWrapForced());
|
|
|
|
// try set wrap and check
|
|
Row.GetCharRow().SetWrapForced(true);
|
|
VERIFY_IS_TRUE(Row.GetCharRow().WasWrapForced());
|
|
|
|
// try unset wrap and check
|
|
Row.GetCharRow().SetWrapForced(false);
|
|
VERIFY_IS_FALSE(Row.GetCharRow().WasWrapForced());
|
|
}
|
|
|
|
void TextBufferTests::TestDoubleBytePadFlag()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
ROW& Row = textBuffer._GetFirstRow();
|
|
|
|
// no padding by default
|
|
VERIFY_IS_FALSE(Row.GetCharRow().WasDoubleBytePadded());
|
|
|
|
// try set and check
|
|
Row.GetCharRow().SetDoubleBytePadded(true);
|
|
VERIFY_IS_TRUE(Row.GetCharRow().WasDoubleBytePadded());
|
|
|
|
// try unset and check
|
|
Row.GetCharRow().SetDoubleBytePadded(false);
|
|
VERIFY_IS_FALSE(Row.GetCharRow().WasDoubleBytePadded());
|
|
}
|
|
|
|
|
|
void TextBufferTests::DoBoundaryTest(PWCHAR const pwszInputString,
|
|
short const cLength,
|
|
short const cMax,
|
|
short const cLeft,
|
|
short const cRight)
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
CharRow& charRow = textBuffer._GetFirstRow().GetCharRow();
|
|
|
|
// copy string into buffer
|
|
for (size_t i = 0; i < static_cast<size_t>(cLength); ++i)
|
|
{
|
|
charRow.GlyphAt(i) = { &pwszInputString[i], 1 };
|
|
}
|
|
|
|
// space pad the rest of the string
|
|
if (cLength < cMax)
|
|
{
|
|
for (short cStart = cLength; cStart < cMax; cStart++)
|
|
{
|
|
charRow.ClearGlyph(cStart);
|
|
}
|
|
}
|
|
|
|
// left edge should be 0 since there are no leading spaces
|
|
VERIFY_ARE_EQUAL(charRow.MeasureLeft(), static_cast<size_t>(cLeft));
|
|
// right edge should be one past the index of the last character or the string length
|
|
VERIFY_ARE_EQUAL(charRow.MeasureRight(), static_cast<size_t>(cRight));
|
|
}
|
|
|
|
void TextBufferTests::TestBoundaryMeasuresRegularString()
|
|
{
|
|
SHORT csBufferWidth = GetBufferWidth();
|
|
|
|
// length 44, left 0, right 44
|
|
const PWCHAR pwszLazyDog = L"The quick brown fox jumps over the lazy dog.";
|
|
DoBoundaryTest(pwszLazyDog, 44, csBufferWidth, 0, 44);
|
|
}
|
|
|
|
void TextBufferTests::TestBoundaryMeasuresFloatingString()
|
|
{
|
|
SHORT csBufferWidth = GetBufferWidth();
|
|
|
|
// length 5 spaces + 4 chars + 5 spaces = 14, left 5, right 9
|
|
const PWCHAR pwszOffsets = L" C:\\> ";
|
|
DoBoundaryTest(pwszOffsets, 14, csBufferWidth, 5, 9);
|
|
}
|
|
|
|
void TextBufferTests::TestCopyProperties()
|
|
{
|
|
TextBuffer& otherTbi = GetTbi();
|
|
|
|
std::unique_ptr<TextBuffer> testTextBuffer = std::make_unique<TextBuffer>(otherTbi.GetSize().Dimensions(),
|
|
otherTbi._currentAttributes,
|
|
12,
|
|
otherTbi._renderTarget);
|
|
VERIFY_IS_NOT_NULL(testTextBuffer.get());
|
|
|
|
// set initial mapping values
|
|
testTextBuffer->GetCursor().SetHasMoved(false);
|
|
otherTbi.GetCursor().SetHasMoved(true);
|
|
|
|
testTextBuffer->GetCursor().SetIsVisible(false);
|
|
otherTbi.GetCursor().SetIsVisible(true);
|
|
|
|
testTextBuffer->GetCursor().SetIsOn(false);
|
|
otherTbi.GetCursor().SetIsOn(true);
|
|
|
|
testTextBuffer->GetCursor().SetIsDouble(false);
|
|
otherTbi.GetCursor().SetIsDouble(true);
|
|
|
|
testTextBuffer->GetCursor().SetDelay(false);
|
|
otherTbi.GetCursor().SetDelay(true);
|
|
|
|
// run copy
|
|
testTextBuffer->CopyProperties(otherTbi);
|
|
|
|
// test that new now contains values from other
|
|
VERIFY_IS_TRUE(testTextBuffer->GetCursor().HasMoved());
|
|
VERIFY_IS_TRUE(testTextBuffer->GetCursor().IsVisible());
|
|
VERIFY_IS_TRUE(testTextBuffer->GetCursor().IsOn());
|
|
VERIFY_IS_TRUE(testTextBuffer->GetCursor().IsDouble());
|
|
VERIFY_IS_TRUE(testTextBuffer->GetCursor().GetDelay());
|
|
}
|
|
|
|
void TextBufferTests::TestInsertCharacter()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
// get starting cursor position
|
|
COORD const coordCursorBefore = textBuffer.GetCursor().GetPosition();
|
|
|
|
// Get current row from the buffer
|
|
ROW& Row = textBuffer.GetRowByOffset(coordCursorBefore.Y);
|
|
|
|
// create some sample test data
|
|
const auto wch = L'Z';
|
|
const std::wstring_view wchTest(&wch, 1);
|
|
DbcsAttribute dbcsAttribute;
|
|
dbcsAttribute.SetTrailing();
|
|
WORD const wAttrTest = BACKGROUND_INTENSITY | FOREGROUND_INTENSITY | FOREGROUND_RED | FOREGROUND_BLUE;
|
|
TextAttribute TestAttributes = TextAttribute(wAttrTest);
|
|
|
|
CharRow& charRow = Row.GetCharRow();
|
|
charRow.DbcsAttrAt(coordCursorBefore.X).SetLeading();
|
|
// ensure that the buffer didn't start with these fields
|
|
VERIFY_ARE_NOT_EQUAL(charRow.GlyphAt(coordCursorBefore.X), wchTest);
|
|
VERIFY_ARE_NOT_EQUAL(charRow.DbcsAttrAt(coordCursorBefore.X), dbcsAttribute);
|
|
|
|
auto attr = Row.GetAttrRow().GetAttrByColumn(coordCursorBefore.X);
|
|
|
|
VERIFY_ARE_NOT_EQUAL(attr, TestAttributes);
|
|
|
|
// now apply the new data to the buffer
|
|
textBuffer.InsertCharacter(wchTest, dbcsAttribute, TestAttributes);
|
|
|
|
// ensure that the buffer position where the cursor WAS contains the test items
|
|
VERIFY_ARE_EQUAL(charRow.GlyphAt(coordCursorBefore.X), wchTest);
|
|
VERIFY_ARE_EQUAL(charRow.DbcsAttrAt(coordCursorBefore.X), dbcsAttribute);
|
|
|
|
attr = Row.GetAttrRow().GetAttrByColumn(coordCursorBefore.X);
|
|
VERIFY_ARE_EQUAL(attr, TestAttributes);
|
|
|
|
// ensure that the cursor moved to a new position (X or Y or both have changed)
|
|
VERIFY_IS_TRUE((coordCursorBefore.X != textBuffer.GetCursor().GetPosition().X) ||
|
|
(coordCursorBefore.Y != textBuffer.GetCursor().GetPosition().Y));
|
|
// the proper advancement of the cursor (e.g. which position it goes to) is validated in other tests
|
|
}
|
|
|
|
void TextBufferTests::TestIncrementCursor()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
// only checking X increments here
|
|
// Y increments are covered in the NewlineCursor test
|
|
|
|
short const sBufferWidth = textBuffer.GetSize().Width();
|
|
|
|
short const sBufferHeight = textBuffer.GetSize().Height();
|
|
VERIFY_IS_TRUE(sBufferWidth > 1 && sBufferHeight > 1);
|
|
|
|
Log::Comment(L"Test normal case of moving once to the right within a single line");
|
|
textBuffer.GetCursor().SetXPosition(0);
|
|
textBuffer.GetCursor().SetYPosition(0);
|
|
|
|
COORD coordCursorBefore = textBuffer.GetCursor().GetPosition();
|
|
|
|
textBuffer.IncrementCursor();
|
|
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().X, 1); // X should advance by 1
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().Y, coordCursorBefore.Y); // Y shouldn't have moved
|
|
|
|
Log::Comment(L"Test line wrap case where cursor is on the right edge of the line");
|
|
textBuffer.GetCursor().SetXPosition(sBufferWidth - 1);
|
|
textBuffer.GetCursor().SetYPosition(0);
|
|
|
|
coordCursorBefore = textBuffer.GetCursor().GetPosition();
|
|
|
|
textBuffer.IncrementCursor();
|
|
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().X, 0); // position should be reset to the left edge when passing right edge
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().Y - 1, coordCursorBefore.Y); // the cursor should be moved one row down from where it used to be
|
|
}
|
|
|
|
void TextBufferTests::TestNewlineCursor()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
|
|
const short sBufferHeight = textBuffer.GetSize().Height();
|
|
|
|
const short sBufferWidth = textBuffer.GetSize().Width();
|
|
// width and height are sufficiently large for upcoming math
|
|
VERIFY_IS_TRUE(sBufferWidth > 4 && sBufferHeight > 4);
|
|
|
|
Log::Comment(L"Verify standard row increment from somewhere in the buffer");
|
|
|
|
// set cursor X position to non zero, any position in buffer
|
|
textBuffer.GetCursor().SetXPosition(3);
|
|
|
|
// set cursor Y position to not-the-final row in the buffer
|
|
textBuffer.GetCursor().SetYPosition(3);
|
|
|
|
COORD coordCursorBefore = textBuffer.GetCursor().GetPosition();
|
|
|
|
// perform operation
|
|
textBuffer.NewlineCursor();
|
|
|
|
// verify
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().X, 0); // move to left edge of buffer
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().Y, coordCursorBefore.Y + 1); // move down one row
|
|
|
|
Log::Comment(L"Verify increment when already on last row of buffer");
|
|
|
|
// X position still doesn't matter
|
|
textBuffer.GetCursor().SetXPosition(3);
|
|
|
|
// Y position needs to be on the last row of the buffer
|
|
textBuffer.GetCursor().SetYPosition(sBufferHeight - 1);
|
|
|
|
coordCursorBefore = textBuffer.GetCursor().GetPosition();
|
|
|
|
// perform operation
|
|
textBuffer.NewlineCursor();
|
|
|
|
// verify
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().X, 0); // move to left edge
|
|
VERIFY_ARE_EQUAL(textBuffer.GetCursor().GetPosition().Y, coordCursorBefore.Y); // cursor Y position should not have moved. stays on same logical final line of buffer
|
|
|
|
// This is okay because the backing circular buffer changes, not the logical screen position (final visible line of the buffer)
|
|
}
|
|
|
|
void TextBufferTests::TestLastNonSpace(short const cursorPosY)
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
textBuffer.GetCursor().SetYPosition(cursorPosY);
|
|
|
|
COORD coordLastNonSpace = textBuffer.GetLastNonSpaceCharacter();
|
|
|
|
// We expect the last non space character to be the last printable character in the row.
|
|
// The .Right property on a row is 1 past the last printable character in the row.
|
|
// If there is one character in the row, the last character would be 0.
|
|
// If there are no characters in the row, the last character would be -1 and we need to seek backwards to find the previous row with a character.
|
|
|
|
// start expected position from cursor
|
|
COORD coordExpected = textBuffer.GetCursor().GetPosition();
|
|
|
|
// Try to get the X position from the current cursor position.
|
|
coordExpected.X = static_cast<short>(textBuffer.GetRowByOffset(coordExpected.Y).GetCharRow().MeasureRight()) - 1;
|
|
|
|
// If we went negative, this row was empty and we need to continue seeking upward...
|
|
// - As long as X is negative (empty rows)
|
|
// - As long as we have space before the top of the buffer (Y isn't the 0th/top row).
|
|
while (coordExpected.X < 0 && coordExpected.Y > 0)
|
|
{
|
|
coordExpected.Y--;
|
|
coordExpected.X = static_cast<short>(textBuffer.GetRowByOffset(coordExpected.Y).GetCharRow().MeasureRight()) - 1;
|
|
}
|
|
|
|
VERIFY_ARE_EQUAL(coordLastNonSpace.X, coordExpected.X);
|
|
VERIFY_ARE_EQUAL(coordLastNonSpace.Y, coordExpected.Y);
|
|
}
|
|
|
|
void TextBufferTests::TestGetLastNonSpaceCharacter()
|
|
{
|
|
m_state->FillTextBuffer(); // fill buffer with some text, it should be 4 rows. See CommonState for details
|
|
|
|
Log::Comment(L"Test with cursor inside last row of text");
|
|
TestLastNonSpace(3);
|
|
|
|
Log::Comment(L"Test with cursor one beyond last row of text");
|
|
TestLastNonSpace(4);
|
|
|
|
Log::Comment(L"Test with cursor way beyond last row of text");
|
|
TestLastNonSpace(14);
|
|
}
|
|
|
|
void TextBufferTests::TestSetWrapOnCurrentRow()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
short sCurrentRow = textBuffer.GetCursor().GetPosition().Y;
|
|
|
|
ROW& Row = textBuffer.GetRowByOffset(sCurrentRow);
|
|
|
|
Log::Comment(L"Testing off to on");
|
|
|
|
// turn wrap status off first
|
|
Row.GetCharRow().SetWrapForced(false);
|
|
|
|
// trigger wrap
|
|
textBuffer._SetWrapOnCurrentRow();
|
|
|
|
// ensure this row was flipped
|
|
VERIFY_IS_TRUE(Row.GetCharRow().WasWrapForced());
|
|
|
|
Log::Comment(L"Testing on stays on");
|
|
|
|
// make sure wrap status is on
|
|
Row.GetCharRow().SetWrapForced(true);
|
|
|
|
// trigger wrap
|
|
textBuffer._SetWrapOnCurrentRow();
|
|
|
|
// ensure row is still on
|
|
VERIFY_IS_TRUE(Row.GetCharRow().WasWrapForced());
|
|
}
|
|
|
|
void TextBufferTests::TestIncrementCircularBuffer()
|
|
{
|
|
TextBuffer& textBuffer = GetTbi();
|
|
|
|
short const sBufferHeight = textBuffer.GetSize().Height();
|
|
|
|
VERIFY_IS_TRUE(sBufferHeight > 4); // buffer should be sufficiently large
|
|
|
|
Log::Comment(L"Test 1 = FirstRow of circular buffer is not the final row of the buffer");
|
|
Log::Comment(L"Test 2 = FirstRow of circular buffer IS THE FINAL ROW of the buffer (and therefore circles)");
|
|
short rgRowsToTest[] = { 2, sBufferHeight - 1 };
|
|
|
|
for (UINT iTestIndex = 0; iTestIndex < ARRAYSIZE(rgRowsToTest); iTestIndex++)
|
|
{
|
|
const short iRowToTestIndex = rgRowsToTest[iTestIndex];
|
|
|
|
short iNextRowIndex = iRowToTestIndex + 1;
|
|
// if we're at or crossing the height, loop back to 0 (circular buffer)
|
|
if (iNextRowIndex >= sBufferHeight)
|
|
{
|
|
iNextRowIndex = 0;
|
|
}
|
|
|
|
textBuffer._firstRow = iRowToTestIndex;
|
|
|
|
// fill first row with some stuff
|
|
ROW& FirstRow = textBuffer._GetFirstRow();
|
|
CharRow& charRow = FirstRow.GetCharRow();
|
|
const auto stuff = L'A';
|
|
charRow.GlyphAt(0) = { &stuff, 1 };
|
|
|
|
// ensure it does say that it contains text
|
|
VERIFY_IS_TRUE(FirstRow.GetCharRow().ContainsText());
|
|
|
|
// try increment
|
|
textBuffer.IncrementCircularBuffer();
|
|
|
|
// validate that first row has moved
|
|
VERIFY_ARE_EQUAL(textBuffer._firstRow, iNextRowIndex); // first row has incremented
|
|
VERIFY_ARE_NOT_EQUAL(textBuffer._GetFirstRow(), FirstRow); // the old first row is no longer the first
|
|
|
|
// ensure old first row has been emptied
|
|
VERIFY_IS_FALSE(FirstRow.GetCharRow().ContainsText());
|
|
}
|
|
}
|
|
|
|
void TextBufferTests::TestMixedRgbAndLegacyForeground()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
|
|
// Case 1 -
|
|
// Write '\E[m\E[38;2;64;128;255mX\E[49mX\E[m'
|
|
// Make sure that the second X has RGB attributes (FG and BG)
|
|
// FG = rgb(64;128;255), BG = rgb(default)
|
|
Log::Comment(L"Case 1 \"\\E[m\\E[38;2;64;128;255mX\\E[49mX\\E[m\"");
|
|
|
|
wchar_t* sequence = L"\x1b[m\x1b[38;2;64;128;255mX\x1b[49mX\x1b[m";
|
|
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
const short x = cursor.GetPosition().X;
|
|
const short y = cursor.GetPosition().Y;
|
|
const ROW& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), RGB(64, 128, 255));
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), gci.LookupBackgroundColor(si.GetAttributes()));
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), RGB(64, 128, 255));
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), gci.LookupBackgroundColor(si.GetAttributes()));
|
|
|
|
wchar_t* reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(reset, std::wcslen(reset));
|
|
|
|
}
|
|
|
|
void TextBufferTests::TestMixedRgbAndLegacyBackground()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
|
|
// Case 2 -
|
|
// \E[m\E[48;2;64;128;255mX\E[39mX\E[m
|
|
// Make sure that the second X has RGB attributes (FG and BG)
|
|
// FG = rgb(default), BG = rgb(64;128;255)
|
|
Log::Comment(L"Case 2 \"\\E[m\\E[48;2;64;128;255mX\\E[39mX\\E[m\"");
|
|
|
|
wchar_t* sequence = L"\x1b[m\x1b[48;2;64;128;255mX\x1b[39mX\x1b[m";
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), RGB(64, 128, 255));
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), gci.LookupForegroundColor(si.GetAttributes()));
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), RGB(64, 128, 255));
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), gci.LookupForegroundColor(si.GetAttributes()));
|
|
|
|
wchar_t* reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(reset, std::wcslen(reset));
|
|
}
|
|
|
|
void TextBufferTests::TestMixedRgbAndLegacyUnderline()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
|
|
// Case 3 -
|
|
// '\E[m\E[48;2;64;128;255mX\E[4mX\E[m'
|
|
// Make sure that the second X has RGB attributes AND underline
|
|
Log::Comment(L"Case 3 \"\\E[m\\E[48;2;64;128;255mX\\E[4mX\\E[m\"");
|
|
wchar_t* sequence = L"\x1b[m\x1b[48;2;64;128;255mX\x1b[4mX\x1b[m";
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), RGB(64, 128, 255));
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), gci.LookupForegroundColor(si.GetAttributes()));
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), RGB(64, 128, 255));
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), gci.LookupForegroundColor(si.GetAttributes()));
|
|
|
|
VERIFY_ARE_EQUAL(attrA.GetLegacyAttributes()&COMMON_LVB_UNDERSCORE, 0);
|
|
VERIFY_ARE_EQUAL(attrB.GetLegacyAttributes()&COMMON_LVB_UNDERSCORE, COMMON_LVB_UNDERSCORE);
|
|
|
|
wchar_t* reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(reset, std::wcslen(reset));
|
|
|
|
}
|
|
|
|
void TextBufferTests::TestMixedRgbAndLegacyBrightness()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
// Case 4 -
|
|
// '\E[m\E[32mX\E[1mX'
|
|
// Make sure that the second X is a BRIGHT green, not white.
|
|
Log::Comment(L"Case 4 ;\"\\E[m\\E[32mX\\E[1mX\"");
|
|
const auto dark_green = gci.GetColorTableEntry(2);
|
|
const auto bright_green = gci.GetColorTableEntry(10);
|
|
VERIFY_ARE_NOT_EQUAL(dark_green, bright_green);
|
|
|
|
wchar_t* sequence = L"\x1b[m\x1b[32mX\x1b[1mX";
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), dark_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), bright_green);
|
|
|
|
wchar_t* reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(reset, std::wcslen(reset));
|
|
}
|
|
|
|
void TextBufferTests::TestRgbEraseLine()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
// Case 1 -
|
|
// Write '\E[m\E[48;2;64;128;255X\E[48;2;128;128;255\E[KX'
|
|
// Make sure that all the characters after the first have the rgb attrs
|
|
// BG = rgb(128;128;255)
|
|
{
|
|
std::wstring sequence = L"\x1b[m\x1b[48;2;64;128;255m";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
sequence = L"X";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
sequence = L"\x1b[48;2;128;128;255m";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
sequence = L"\x1b[K";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
sequence = L"X";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_ARE_EQUAL(x, 2);
|
|
VERIFY_ARE_EQUAL(y, 0);
|
|
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
|
|
const auto attr0 = attrs[0];
|
|
|
|
VERIFY_ARE_EQUAL(attr0.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr0), RGB(64, 128, 255));
|
|
|
|
for (auto i = 1; i < len; i++)
|
|
{
|
|
const auto attr = attrs[i];
|
|
LOG_ATTR(attr);
|
|
VERIFY_ARE_EQUAL(attr.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr), RGB(128, 128, 255));
|
|
}
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
}
|
|
|
|
void TextBufferTests::TestUnBold()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
// Case 1 -
|
|
// Write '\E[1;32mX\E[22mX'
|
|
// The first X should be bright green.
|
|
// The second x should be dark green.
|
|
std::wstring sequence = L"\x1b[1;32mX\x1b[22mX";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto dark_green = gci.GetColorTableEntry(2);
|
|
const auto bright_green = gci.GetColorTableEntry(10);
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_ARE_EQUAL(x, 2);
|
|
VERIFY_ARE_EQUAL(y, 0);
|
|
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), bright_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), dark_green);
|
|
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
|
|
void TextBufferTests::TestUnBoldRgb()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
// Case 2 -
|
|
// Write '\E[1;32m\E[48;2;1;2;3mX\E[22mX'
|
|
// The first X should be bright green, and not legacy.
|
|
// The second X should be dark green, and not legacy.
|
|
// BG = rgb(1;2;3)
|
|
std::wstring sequence = L"\x1b[1;32m\x1b[48;2;1;2;3mX\x1b[22mX";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto dark_green = gci.GetColorTableEntry(2);
|
|
const auto bright_green = gci.GetColorTableEntry(10);
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_ARE_EQUAL(x, 2);
|
|
VERIFY_ARE_EQUAL(y, 0);
|
|
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), bright_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), dark_green);
|
|
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
|
|
void TextBufferTests::TestComplexUnBold()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
// Case 3 -
|
|
// Write '\E[1;32m\E[48;2;1;2;3mA\E[22mB\E[38;2;32;32;32mC\E[1mD\E[38;2;64;64;64mE\E[22mF'
|
|
// The A should be bright green, and not legacy.
|
|
// The B should be dark green, and not legacy.
|
|
// The C should be rgb(32, 32, 32), and not legacy.
|
|
// The D should be unchanged from the third.
|
|
// The E should be rgb(64, 64, 64), and not legacy.
|
|
// The F should be rgb(64, 64, 64), and not legacy.
|
|
// BG = rgb(1;2;3)
|
|
std::wstring sequence = L"\x1b[1;32m\x1b[48;2;1;2;3mA\x1b[22mB\x1b[38;2;32;32;32mC\x1b[1mD\x1b[38;2;64;64;64mE\x1b[22mF";
|
|
Log::Comment(NoThrowString().Format(sequence.c_str()));
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto dark_green = gci.GetColorTableEntry(2);
|
|
const auto bright_green = gci.GetColorTableEntry(10);
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_ARE_EQUAL(x, 6);
|
|
VERIFY_ARE_EQUAL(y, 0);
|
|
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 6];
|
|
const auto attrB = attrs[x - 5];
|
|
const auto attrC = attrs[x - 4];
|
|
const auto attrD = attrs[x - 3];
|
|
const auto attrE = attrs[x - 2];
|
|
const auto attrF = attrs[x - 1];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
Log::Comment(NoThrowString().Format(
|
|
L"attrA=%s", VerifyOutputTraits<TextAttribute>::ToString(attrA).GetBuffer()
|
|
));
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
LOG_ATTR(attrC);
|
|
LOG_ATTR(attrD);
|
|
LOG_ATTR(attrE);
|
|
LOG_ATTR(attrF);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrC.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrD.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrE.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrF.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), bright_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), RGB(1, 2, 3));
|
|
VERIFY_IS_TRUE(attrA.IsBold());
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), dark_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), RGB(1, 2, 3));
|
|
VERIFY_IS_FALSE(attrB.IsBold());
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrC), RGB(32, 32, 32));
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrC), RGB(1, 2, 3));
|
|
VERIFY_IS_FALSE(attrC.IsBold());
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrD), gci.LookupForegroundColor(attrC));
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrD), gci.LookupBackgroundColor(attrC));
|
|
VERIFY_IS_TRUE(attrD.IsBold());
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrE), RGB(64, 64, 64));
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrE), RGB(1, 2, 3));
|
|
VERIFY_IS_TRUE(attrE.IsBold());
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrF), RGB(64, 64, 64));
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrF), RGB(1, 2, 3));
|
|
VERIFY_IS_FALSE(attrF.IsBold());
|
|
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
|
|
|
|
void TextBufferTests::CopyAttrs()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
cursor.SetYPosition(0);
|
|
// Write '\E[32mX\E[33mX\n\E[34mX\E[35mX\E[H\E[M'
|
|
// The first two X's should get deleted.
|
|
// The third X should be blue
|
|
// The fourth X should be magenta
|
|
std::wstring sequence = L"\x1b[32mX\x1b[33mX\n\x1b[34mX\x1b[35mX\x1b[H\x1b[M";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto dark_blue = gci.GetColorTableEntry(1);
|
|
const auto dark_magenta = gci.GetColorTableEntry(5);
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_ARE_EQUAL(x, 0);
|
|
VERIFY_ARE_EQUAL(y, 0);
|
|
|
|
const auto& row = tbi.GetRowByOffset(0);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[0];
|
|
const auto attrB = attrs[1];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), dark_blue);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), dark_magenta);
|
|
|
|
}
|
|
|
|
void TextBufferTests::EmptySgrTest()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
cursor.SetXPosition(0);
|
|
cursor.SetYPosition(0);
|
|
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
const COLORREF defaultFg = gci.LookupForegroundColor(si.GetAttributes());
|
|
const COLORREF defaultBg = gci.LookupBackgroundColor(si.GetAttributes());
|
|
|
|
// Case 1 -
|
|
// Write '\x1b[0mX\x1b[31mX\x1b[31;m'
|
|
// The first X should be default colors.
|
|
// The second X should be (darkRed,default).
|
|
// The third X should be default colors.
|
|
std::wstring sequence = L"\x1b[0mX\x1b[31mX\x1b[31;mX";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const COLORREF darkRed = gci.GetColorTableEntry(4);
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_IS_TRUE(x >= 3);
|
|
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 3];
|
|
const auto attrB = attrs[x - 2];
|
|
const auto attrC = attrs[x - 1];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
LOG_ATTR(attrC);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), defaultFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), defaultBg);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), darkRed);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), defaultBg);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrC), defaultFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrC), defaultBg);
|
|
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
|
|
void TextBufferTests::TestReverseReset()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
cursor.SetYPosition(0);
|
|
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
const COLORREF defaultFg = gci.LookupForegroundColor(si.GetAttributes());
|
|
const COLORREF defaultBg = gci.LookupBackgroundColor(si.GetAttributes());
|
|
|
|
// Case 1 -
|
|
// Write '\E[42m\E[38;2;128;5;255mX\E[7mX\E[27mX'
|
|
// The first X should be (fg,bg) = (rgb(128;5;255), dark_green)
|
|
// The second X should be (fg,bg) = (dark_green, rgb(128;5;255))
|
|
// The third X should be (fg,bg) = (rgb(128;5;255), dark_green)
|
|
std::wstring sequence = L"\x1b[42m\x1b[38;2;128;5;255mX\x1b[7mX\x1b[27mX";
|
|
stateMachine.ProcessString(&sequence[0], sequence.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto dark_green = gci.GetColorTableEntry(2);
|
|
const COLORREF rgbColor = RGB(128, 5, 255);
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
VERIFY_IS_TRUE(x >= 3);
|
|
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const auto len = tbi.GetSize().Width();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 3];
|
|
const auto attrB = attrs[x - 2];
|
|
const auto attrC = attrs[x - 1];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
LOG_ATTR(attrC);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), rgbColor);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), dark_green);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), dark_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), rgbColor);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrC), rgbColor);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrC), dark_green);
|
|
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
|
|
void TextBufferTests::CopyLastAttr()
|
|
{
|
|
DisableVerifyExceptions disable;
|
|
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
cursor.SetYPosition(0);
|
|
|
|
std::wstring reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
const COLORREF defaultFg = gci.LookupForegroundColor(si.GetAttributes());
|
|
const COLORREF defaultBg = gci.LookupBackgroundColor(si.GetAttributes());
|
|
|
|
const COLORREF solFg = RGB(101, 123, 131);
|
|
const COLORREF solBg = RGB(0, 43, 54);
|
|
const COLORREF solCyan = RGB(42, 161, 152);
|
|
|
|
std::wstring solFgSeq = L"\x1b[38;2;101;123;131m";
|
|
std::wstring solBgSeq = L"\x1b[48;2;0;43;54m";
|
|
std::wstring solCyanSeq = L"\x1b[38;2;42;161;152m";
|
|
|
|
// Make sure that the color table has certain values we expect
|
|
const COLORREF defaultBrightBlack = RGB(118, 118, 118);
|
|
const COLORREF defaultBrightYellow = RGB(249, 241, 165);
|
|
const COLORREF defaultBrightCyan = RGB(97, 214, 214);
|
|
|
|
gci.SetColorTableEntry(8, defaultBrightBlack);
|
|
gci.SetColorTableEntry(14, defaultBrightYellow);
|
|
gci.SetColorTableEntry(11, defaultBrightCyan);
|
|
|
|
// Write (solFg, solBG) X \n
|
|
// (solFg, solBG) X (solCyan, solBG) X \n
|
|
// (solFg, solBG) X (solCyan, solBG) X (solFg, solBG) X
|
|
// then go home, and insert a line.
|
|
|
|
// Row 1
|
|
stateMachine.ProcessString(&solFgSeq[0], solFgSeq.length());
|
|
stateMachine.ProcessString(&solBgSeq[0], solBgSeq.length());
|
|
stateMachine.ProcessString(L"X", 1);
|
|
stateMachine.ProcessString(L"\n", 1);
|
|
|
|
// Row 2
|
|
// Remember that the colors from before persist here too, so we don't need
|
|
// to emit both the FG and BG if they haven't changed.
|
|
stateMachine.ProcessString(L"X", 1);
|
|
stateMachine.ProcessString(&solCyanSeq[0], solCyanSeq.length());
|
|
stateMachine.ProcessString(L"X", 1);
|
|
stateMachine.ProcessString(L"\n", 1);
|
|
|
|
// Row 3
|
|
stateMachine.ProcessString(&solFgSeq[0], solFgSeq.length());
|
|
stateMachine.ProcessString(&solBgSeq[0], solBgSeq.length());
|
|
stateMachine.ProcessString(L"X", 1);
|
|
stateMachine.ProcessString(&solCyanSeq[0], solCyanSeq.length());
|
|
stateMachine.ProcessString(L"X", 1);
|
|
stateMachine.ProcessString(&solFgSeq[0], solFgSeq.length());
|
|
stateMachine.ProcessString(L"X", 1);
|
|
|
|
std::wstring insertLineAtHome = L"\x1b[H\x1b[L";
|
|
stateMachine.ProcessString(&insertLineAtHome[0], insertLineAtHome.length());
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
const ROW& row1 = tbi.GetRowByOffset(y + 1);
|
|
const ROW& row2 = tbi.GetRowByOffset(y + 2);
|
|
const ROW& row3 = tbi.GetRowByOffset(y + 3);
|
|
const auto len = tbi.GetSize().Width();
|
|
|
|
const std::vector<TextAttribute> attrs1{ row1.GetAttrRow().begin(), row1.GetAttrRow().end() };
|
|
const std::vector<TextAttribute> attrs2{ row2.GetAttrRow().begin(), row2.GetAttrRow().end() };
|
|
const std::vector<TextAttribute> attrs3{ row3.GetAttrRow().begin(), row3.GetAttrRow().end() };
|
|
|
|
const auto attr1A = attrs1[0];
|
|
|
|
const auto attr2A = attrs2[0];
|
|
const auto attr2B = attrs2[1];
|
|
|
|
const auto attr3A = attrs3[0];
|
|
const auto attr3B = attrs3[1];
|
|
const auto attr3C = attrs3[2];
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
|
|
LOG_ATTR(attr1A);
|
|
LOG_ATTR(attr2A);
|
|
LOG_ATTR(attr2A);
|
|
LOG_ATTR(attr3A);
|
|
LOG_ATTR(attr3B);
|
|
LOG_ATTR(attr3C);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attr1A), solFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr1A), solBg);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attr2A), solFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr2A), solBg);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attr2B), solCyan);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr2B), solBg);
|
|
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attr3A), solFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr3A), solBg);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attr3B), solCyan);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr3B), solBg);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attr3C), solFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attr3C), solBg);
|
|
|
|
stateMachine.ProcessString(&reset[0], reset.length());
|
|
}
|
|
|
|
void TextBufferTests::TestRgbThenBold()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
// See MSFT:16398982
|
|
Log::Comment(NoThrowString().Format(
|
|
L"Test that a bold following a RGB color doesn't remove the RGB color"
|
|
));
|
|
Log::Comment(L"\"\\x1b[38;2;40;40;40m\\x1b[48;2;168;153;132mX\\x1b[1mX\\x1b[m\"");
|
|
const auto foreground = RGB(40, 40, 40);
|
|
const auto background = RGB(168, 153, 132);
|
|
|
|
const wchar_t* const sequence = L"\x1b[38;2;40;40;40m\x1b[48;2;168;153;132mX\x1b[1mX\x1b[m";
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x - 2];
|
|
const auto attrB = attrs[x - 1];
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
Log::Comment(NoThrowString().Format(
|
|
L"attrA should be RGB, and attrB should be the same as attrA, NOT bolded"
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
|
|
VERIFY_ARE_EQUAL(attrA.IsLegacy(), false);
|
|
VERIFY_ARE_EQUAL(attrB.IsLegacy(), false);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), foreground);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrA), background);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), foreground);
|
|
VERIFY_ARE_EQUAL(gci.LookupBackgroundColor(attrB), background);
|
|
|
|
wchar_t* reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(reset, std::wcslen(reset));
|
|
}
|
|
|
|
void TextBufferTests::TestResetClearsBoldness()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
Log::Comment(NoThrowString().Format(
|
|
L"Test that resetting bold attributes clears the boldness."
|
|
));
|
|
const auto x0 = cursor.GetPosition().X;
|
|
|
|
// Test assumes that the background/foreground were default attribute when it starts up,
|
|
// so set that here.
|
|
TextAttribute defaultAttribute;
|
|
si.SetAttributes(defaultAttribute);
|
|
|
|
const COLORREF defaultFg = gci.LookupForegroundColor(si.GetAttributes());
|
|
const COLORREF defaultBg = gci.LookupBackgroundColor(si.GetAttributes());
|
|
const auto dark_green = gci.GetColorTableEntry(2);
|
|
const auto bright_green = gci.GetColorTableEntry(10);
|
|
|
|
wchar_t* sequence = L"\x1b[32mA\x1b[1mB\x1b[0mC\x1b[32mD";
|
|
Log::Comment(NoThrowString().Format(sequence));
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
|
|
const auto x = cursor.GetPosition().X;
|
|
const auto y = cursor.GetPosition().Y;
|
|
const auto& row = tbi.GetRowByOffset(y);
|
|
const auto attrRow = &row.GetAttrRow();
|
|
const std::vector<TextAttribute> attrs{ attrRow->begin(), attrRow->end() };
|
|
const auto attrA = attrs[x0];
|
|
const auto attrB = attrs[x0 + 1];
|
|
const auto attrC = attrs[x0 + 2];
|
|
const auto attrD = attrs[x0 + 3];
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x, y
|
|
));
|
|
Log::Comment(NoThrowString().Format(
|
|
L"attrA should be RGB, and attrB should be the same as attrA, NOT bolded"
|
|
));
|
|
|
|
LOG_ATTR(attrA);
|
|
LOG_ATTR(attrB);
|
|
LOG_ATTR(attrC);
|
|
LOG_ATTR(attrD);
|
|
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrA), dark_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrB), bright_green);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrC), defaultFg);
|
|
VERIFY_ARE_EQUAL(gci.LookupForegroundColor(attrD), dark_green);
|
|
|
|
VERIFY_IS_FALSE(attrA.IsBold());
|
|
VERIFY_IS_TRUE(attrB.IsBold());
|
|
VERIFY_IS_FALSE(attrC.IsBold());
|
|
VERIFY_IS_FALSE(attrD.IsBold());
|
|
|
|
wchar_t* reset = L"\x1b[0m";
|
|
stateMachine.ProcessString(reset, std::wcslen(reset));
|
|
}
|
|
|
|
void TextBufferTests::TestBackspaceRightSideVt()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
|
|
Log::Comment(L"verify that backspace has the same behavior as a vt CUB sequence once "
|
|
L"we've traversed to the right side of the current row");
|
|
|
|
const wchar_t* const sequence = L"\033[1000Cx\by\n";
|
|
Log::Comment(NoThrowString().Format(sequence));
|
|
|
|
const auto preCursorPosition = cursor.GetPosition();
|
|
stateMachine.ProcessString(sequence, std::wcslen(sequence));
|
|
const auto postCursorPosition = cursor.GetPosition();
|
|
|
|
// make sure newline was handled correctly
|
|
VERIFY_ARE_EQUAL(0, postCursorPosition.X);
|
|
VERIFY_ARE_EQUAL(preCursorPosition.Y, postCursorPosition.Y - 1);
|
|
|
|
// make sure "yx" was written to the end of the line the cursor started on
|
|
const auto& row = tbi.GetRowByOffset(preCursorPosition.Y);
|
|
const auto rowText = row.GetText();
|
|
auto it = rowText.crbegin();
|
|
VERIFY_ARE_EQUAL(*it, L'x');
|
|
++it;
|
|
VERIFY_ARE_EQUAL(*it, L'y');
|
|
}
|
|
|
|
void TextBufferTests::TestBackspaceStrings()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
|
|
const auto x0 = cursor.GetPosition().X;
|
|
const auto y0 = cursor.GetPosition().Y;
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x0, y0
|
|
));
|
|
std::wstring seq = L"a\b \b";
|
|
stateMachine.ProcessString(seq.c_str(), seq.length());
|
|
|
|
const auto x1 = cursor.GetPosition().X;
|
|
const auto y1 = cursor.GetPosition().Y;
|
|
|
|
VERIFY_ARE_EQUAL(x1, x0);
|
|
VERIFY_ARE_EQUAL(y1, y0);
|
|
|
|
seq = L"a";
|
|
stateMachine.ProcessString(seq.c_str(), seq.length());
|
|
seq = L"\b";
|
|
stateMachine.ProcessString(seq.c_str(), seq.length());
|
|
seq = L" ";
|
|
stateMachine.ProcessString(seq.c_str(), seq.length());
|
|
seq = L"\b";
|
|
stateMachine.ProcessString(seq.c_str(), seq.length());
|
|
|
|
const auto x2 = cursor.GetPosition().X;
|
|
const auto y2 = cursor.GetPosition().Y;
|
|
|
|
VERIFY_ARE_EQUAL(x2, x0);
|
|
VERIFY_ARE_EQUAL(y2, y0);
|
|
}
|
|
|
|
void TextBufferTests::TestBackspaceStringsAPI()
|
|
{
|
|
// Pretty much the same as the above test, but explicitly DOESNT use the
|
|
// state machine.
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
const TextBuffer& tbi = si.GetTextBuffer();
|
|
const Cursor& cursor = tbi.GetCursor();
|
|
|
|
gci.SetVirtTermLevel(0);
|
|
WI_ClearFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
const auto x0 = cursor.GetPosition().X;
|
|
const auto y0 = cursor.GetPosition().Y;
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"cursor={X:%d,Y:%d}",
|
|
x0, y0
|
|
));
|
|
|
|
// We're going to write an "a" to the buffer in various ways, then try
|
|
// backspacing it with "\b \b".
|
|
// Regardless of how we write those sequences of characters, the end result
|
|
// should be the same.
|
|
std::unique_ptr<WriteData> waiter;
|
|
|
|
size_t aCb = 2;
|
|
VERIFY_SUCCEEDED(DoWriteConsole(L"a", &aCb, si, waiter));
|
|
|
|
size_t seqCb = 6;
|
|
Log::Comment(NoThrowString().Format(
|
|
L"Using WriteCharsLegacy, write \\b \\b as a single string."
|
|
));
|
|
{
|
|
wchar_t* str = L"\b \b";
|
|
VERIFY_SUCCESS_NTSTATUS(WriteCharsLegacy(si, str, str, str, &seqCb, nullptr, cursor.GetPosition().X, 0, nullptr));
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, x0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, y0);
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"Using DoWriteConsole, write \\b \\b as a single string."
|
|
));
|
|
VERIFY_SUCCEEDED(DoWriteConsole(L"a", &aCb, si, waiter));
|
|
|
|
VERIFY_SUCCEEDED(DoWriteConsole(str, &seqCb, si, waiter));
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, x0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, y0);
|
|
}
|
|
|
|
seqCb = 2;
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"Using DoWriteConsole, write \\b \\b as seperate strings."
|
|
));
|
|
|
|
VERIFY_SUCCEEDED(DoWriteConsole(L"a", &seqCb, si, waiter));
|
|
VERIFY_SUCCEEDED(DoWriteConsole(L"\b", &seqCb, si, waiter));
|
|
VERIFY_SUCCEEDED(DoWriteConsole(L" ", &seqCb, si, waiter));
|
|
VERIFY_SUCCEEDED(DoWriteConsole(L"\b", &seqCb, si, waiter));
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, x0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, y0);
|
|
|
|
|
|
Log::Comment(NoThrowString().Format(
|
|
L"Using WriteCharsLegacy, write \\b \\b as seperate strings."
|
|
));
|
|
{
|
|
wchar_t* str = L"a";
|
|
VERIFY_SUCCESS_NTSTATUS(WriteCharsLegacy(si, str, str, str, &seqCb, nullptr, cursor.GetPosition().X, 0, nullptr));
|
|
}
|
|
{
|
|
wchar_t* str = L"\b";
|
|
VERIFY_SUCCESS_NTSTATUS(WriteCharsLegacy(si, str, str, str, &seqCb, nullptr, cursor.GetPosition().X, 0, nullptr));
|
|
}
|
|
{
|
|
wchar_t* str = L" ";
|
|
VERIFY_SUCCESS_NTSTATUS(WriteCharsLegacy(si, str, str, str, &seqCb, nullptr, cursor.GetPosition().X, 0, nullptr));
|
|
}
|
|
{
|
|
wchar_t* str = L"\b";
|
|
VERIFY_SUCCESS_NTSTATUS(WriteCharsLegacy(si, str, str, str, &seqCb, nullptr, cursor.GetPosition().X, 0, nullptr));
|
|
}
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, x0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, y0);
|
|
|
|
}
|
|
|
|
void TextBufferTests::TestRepeatCharacter()
|
|
{
|
|
CONSOLE_INFORMATION& gci = ServiceLocator::LocateGlobals().getConsoleInformation();
|
|
SCREEN_INFORMATION& si = gci.GetActiveOutputBuffer().GetActiveBuffer();
|
|
TextBuffer& tbi = si.GetTextBuffer();
|
|
StateMachine& stateMachine = si.GetStateMachine();
|
|
Cursor& cursor = tbi.GetCursor();
|
|
|
|
WI_SetFlag(si.OutputMode, ENABLE_VIRTUAL_TERMINAL_PROCESSING);
|
|
|
|
cursor.SetXPosition(0);
|
|
cursor.SetYPosition(0);
|
|
|
|
Log::Comment(
|
|
L"Test 0: Simply repeat a single character."
|
|
);
|
|
|
|
std::wstring sequence = L"X";
|
|
stateMachine.ProcessString(sequence);
|
|
|
|
sequence = L"\x1b[b";
|
|
stateMachine.ProcessString(sequence);
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 2);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, 0);
|
|
|
|
{
|
|
const auto& row0 = tbi.GetRowByOffset(0);
|
|
const auto row0Text = row0.GetText();
|
|
VERIFY_ARE_EQUAL(L'X', row0Text[0]);
|
|
VERIFY_ARE_EQUAL(L'X', row0Text[1]);
|
|
VERIFY_ARE_EQUAL(L' ', row0Text[2]);
|
|
}
|
|
|
|
Log::Comment(
|
|
L"Test 1: Try repeating characters after another VT action. It should do nothing."
|
|
);
|
|
|
|
stateMachine.ProcessString(L"\n");
|
|
stateMachine.ProcessString(L"A");
|
|
stateMachine.ProcessString(L"B");
|
|
stateMachine.ProcessString(L"\x1b[A");
|
|
stateMachine.ProcessString(L"\x1b[b");
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 2);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, 0);
|
|
|
|
{
|
|
const auto& row0 = tbi.GetRowByOffset(0);
|
|
const auto& row1 = tbi.GetRowByOffset(1);
|
|
const auto row0Text = row0.GetText();
|
|
const auto row1Text = row1.GetText();
|
|
VERIFY_ARE_EQUAL(L'X', row0Text[0]);
|
|
VERIFY_ARE_EQUAL(L'X', row0Text[1]);
|
|
VERIFY_ARE_EQUAL(L' ', row0Text[2]);
|
|
VERIFY_ARE_EQUAL(L'A', row1Text[0]);
|
|
VERIFY_ARE_EQUAL(L'B', row1Text[1]);
|
|
VERIFY_ARE_EQUAL(L' ', row1Text[2]);
|
|
}
|
|
|
|
Log::Comment(
|
|
L"Test 2: Repeat a character lots of times"
|
|
);
|
|
|
|
stateMachine.ProcessString(L"\x1b[3;H");
|
|
stateMachine.ProcessString(L"C");
|
|
stateMachine.ProcessString(L"\x1b[5b");
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 6);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, 2);
|
|
|
|
{
|
|
const auto& row2 = tbi.GetRowByOffset(2);
|
|
const auto row2Text = row2.GetText();
|
|
VERIFY_ARE_EQUAL(L'C', row2Text[0]);
|
|
VERIFY_ARE_EQUAL(L'C', row2Text[1]);
|
|
VERIFY_ARE_EQUAL(L'C', row2Text[2]);
|
|
VERIFY_ARE_EQUAL(L'C', row2Text[3]);
|
|
VERIFY_ARE_EQUAL(L'C', row2Text[4]);
|
|
VERIFY_ARE_EQUAL(L'C', row2Text[5]);
|
|
VERIFY_ARE_EQUAL(L' ', row2Text[6]);
|
|
}
|
|
|
|
Log::Comment(
|
|
L"Test 3: try repeating a non-graphical character. It should do nothing."
|
|
);
|
|
|
|
stateMachine.ProcessString(L"\r\n");
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, 3);
|
|
stateMachine.ProcessString(L"D\n");
|
|
stateMachine.ProcessString(L"\x1b[b");
|
|
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, 4);
|
|
|
|
|
|
Log::Comment(
|
|
L"Test 4: try repeating multiple times. It should do nothing."
|
|
);
|
|
|
|
stateMachine.ProcessString(L"\r\n");
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 0);
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().Y, 5);
|
|
stateMachine.ProcessString(L"E");
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 1);
|
|
stateMachine.ProcessString(L"\x1b[b");
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 2);
|
|
stateMachine.ProcessString(L"\x1b[b");
|
|
VERIFY_ARE_EQUAL(cursor.GetPosition().X, 2);
|
|
|
|
{
|
|
const auto& row5 = tbi.GetRowByOffset(5);
|
|
const auto row5Text = row5.GetText();
|
|
VERIFY_ARE_EQUAL(L'E', row5Text[0]);
|
|
VERIFY_ARE_EQUAL(L'E', row5Text[1]);
|
|
VERIFY_ARE_EQUAL(L' ', row5Text[2]);
|
|
}
|
|
}
|
|
|
|
void TextBufferTests::ResizeTraditional()
|
|
{
|
|
BEGIN_TEST_METHOD_PROPERTIES()
|
|
TEST_METHOD_PROPERTY(L"Data:shrinkX", L"{false, true}")
|
|
TEST_METHOD_PROPERTY(L"Data:shrinkY", L"{false, true}")
|
|
END_TEST_METHOD_PROPERTIES();
|
|
|
|
bool shrinkX;
|
|
VERIFY_SUCCEEDED(TestData::TryGetValue(L"shrinkX", shrinkX), L"Shrink X = true, Grow X = false");
|
|
|
|
bool shrinkY;
|
|
VERIFY_SUCCEEDED(TestData::TryGetValue(L"shrinkY", shrinkY), L"Shrink Y = true, Grow Y = false");
|
|
|
|
const COORD smallSize = { 5, 5 };
|
|
TextAttribute defaultAttr;
|
|
defaultAttr.SetFromLegacy(0);
|
|
|
|
TextBuffer buffer(smallSize, defaultAttr, 12, _renderTarget);
|
|
|
|
Log::Comment(L"Fill buffer with some data and do assorted resize operations.");
|
|
|
|
wchar_t expectedChar = L'A';
|
|
const std::wstring_view expectedView(&expectedChar, 1);
|
|
TextAttribute expectedAttr(FOREGROUND_RED);
|
|
OutputCellIterator it(expectedChar, expectedAttr);
|
|
const auto finalIt = buffer.Write(it);
|
|
VERIFY_ARE_EQUAL(smallSize.X * smallSize.Y, finalIt.GetCellDistance(it), L"Verify we said we filled every cell.");
|
|
|
|
const Viewport writtenView = Viewport::FromDimensions({ 0, 0 }, smallSize);
|
|
|
|
Log::Comment(L"Ensure every cell has our test pattern value.");
|
|
{
|
|
TextBufferCellIterator viewIt(buffer, { 0, 0 });
|
|
while (viewIt)
|
|
{
|
|
VERIFY_ARE_EQUAL(expectedView, viewIt->Chars());
|
|
VERIFY_ARE_EQUAL(expectedAttr, viewIt->TextAttr());
|
|
viewIt++;
|
|
}
|
|
}
|
|
|
|
Log::Comment(L"Resize to X and Y.");
|
|
COORD newSize = smallSize;
|
|
|
|
if (shrinkX)
|
|
{
|
|
newSize.X -= 2;
|
|
}
|
|
else
|
|
{
|
|
newSize.X += 2;
|
|
}
|
|
|
|
if (shrinkY)
|
|
{
|
|
newSize.Y -= 2;
|
|
}
|
|
else
|
|
{
|
|
newSize.Y += 2;
|
|
}
|
|
|
|
// When we grow, we extend the last color. Therefore, this region covers the area colored the same as the letters but filled with a blank.
|
|
const auto widthAdjustedView = Viewport::FromDimensions(writtenView.Origin(), { newSize.X, smallSize.Y });
|
|
|
|
// When we resize, we expect the attributes to be unchanged, but the new cells
|
|
// to be filled with spaces
|
|
wchar_t expectedSpace = UNICODE_SPACE;
|
|
std::wstring_view expectedSpaceView(&expectedSpace, 1);
|
|
|
|
VERIFY_SUCCEEDED(buffer.ResizeTraditional(newSize));
|
|
|
|
Log::Comment(L"Verify every cell in the X dimension is still the same as when filled and the new Y row is just empty default cells.");
|
|
{
|
|
TextBufferCellIterator viewIt(buffer, { 0, 0 });
|
|
while (viewIt)
|
|
{
|
|
Log::Comment(NoThrowString().Format(L"Checking cell (Y=%d, X=%d)", viewIt._pos.Y, viewIt._pos.X));
|
|
if (writtenView.IsInBounds(viewIt._pos))
|
|
{
|
|
Log::Comment(L"This position is inside our original write area. It should have the original character and color.");
|
|
// If the position is in bounds with what we originally wrote, it should have that character and color.
|
|
VERIFY_ARE_EQUAL(expectedView, viewIt->Chars());
|
|
VERIFY_ARE_EQUAL(expectedAttr, viewIt->TextAttr());
|
|
}
|
|
else if (widthAdjustedView.IsInBounds(viewIt._pos))
|
|
{
|
|
Log::Comment(L"This position is right of our original write area. It should have extended the color rightward and filled with a space.");
|
|
// If we missed the original fill, but we're still in region defined by the adjusted width, then
|
|
// the color was extended outward but without the character value.
|
|
VERIFY_ARE_EQUAL(expectedSpaceView, viewIt->Chars());
|
|
VERIFY_ARE_EQUAL(expectedAttr, viewIt->TextAttr());
|
|
}
|
|
else
|
|
{
|
|
Log::Comment(L"This position is below our ouriginal write area. It should have filled blank lines (space lines) with the default fill color.");
|
|
// Otherwise, we use the default.
|
|
VERIFY_ARE_EQUAL(expectedSpaceView, viewIt->Chars());
|
|
VERIFY_ARE_EQUAL(defaultAttr, viewIt->TextAttr());
|
|
}
|
|
viewIt++;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// This tests that when buffer storage rows are rotated around during a resize traditional operation,
|
|
// that the Unicode Storage-held high unicode items like emoji rotate properly with it.
|
|
void TextBufferTests::ResizeTraditionalRotationPreservesHighUnicode()
|
|
{
|
|
// Set up a text buffer for us
|
|
const COORD bufferSize{ 80, 10 };
|
|
const UINT cursorSize = 12;
|
|
const TextAttribute attr{ 0x7f };
|
|
auto _buffer = std::make_unique<TextBuffer>(bufferSize, attr, cursorSize, _renderTarget);
|
|
|
|
// Get a position inside the buffer
|
|
const COORD pos{ 2, 1 };
|
|
auto position = _buffer->_storage[pos.Y].GetCharRow().GlyphAt(pos.X);
|
|
|
|
// Fill it up with a sequence that will have to hit the high unicode storage.
|
|
// This is the negative squared latin capital letter B emoji: 🅱
|
|
// It's encoded in UTF-16, as needed by the buffer.
|
|
const auto bbutton = L"\xD83C\xDD71";
|
|
position = bbutton;
|
|
|
|
// Read back the text at that position and ensure that it matches what we wrote.
|
|
const auto readBack = _buffer->GetTextDataAt(pos);
|
|
const auto readBackText = *readBack;
|
|
VERIFY_ARE_EQUAL(String(bbutton), String(readBackText.data(), gsl::narrow<int>(readBackText.size())));
|
|
|
|
// Make it the first row in the buffer so it will rotate around when we resize and cause renumbering
|
|
const SHORT delta = _buffer->GetFirstRowIndex() - pos.Y;
|
|
const COORD newPos{ pos.X, pos.Y + delta };
|
|
|
|
_buffer->_SetFirstRowIndex(pos.Y);
|
|
|
|
// Perform resize to rotate the rows around
|
|
VERIFY_NT_SUCCESS(_buffer->ResizeTraditional(bufferSize));
|
|
|
|
// Retrieve the text at the old and new positions.
|
|
const auto shouldBeEmptyText = *_buffer->GetTextDataAt(pos);
|
|
const auto shouldBeEmojiText = *_buffer->GetTextDataAt(newPos);
|
|
|
|
VERIFY_ARE_EQUAL(String(L" "), String(shouldBeEmptyText.data(), gsl::narrow<int>(shouldBeEmptyText.size())));
|
|
VERIFY_ARE_EQUAL(String(bbutton), String(shouldBeEmojiText.data(), gsl::narrow<int>(shouldBeEmojiText.size())));
|
|
}
|
|
|
|
// This tests that when buffer storage rows are rotated around during a scroll buffer operation,
|
|
// that the Unicode Storage-held high unicode items like emoji rotate properly with it.
|
|
void TextBufferTests::ScrollBufferRotationPreservesHighUnicode()
|
|
{
|
|
// Set up a text buffer for us
|
|
const COORD bufferSize{ 80, 10 };
|
|
const UINT cursorSize = 12;
|
|
const TextAttribute attr{ 0x7f };
|
|
auto _buffer = std::make_unique<TextBuffer>(bufferSize, attr, cursorSize, _renderTarget);
|
|
|
|
// Get a position inside the buffer
|
|
const COORD pos{ 2, 1 };
|
|
auto position = _buffer->_storage[pos.Y].GetCharRow().GlyphAt(pos.X);
|
|
|
|
// Fill it up with a sequence that will have to hit the high unicode storage.
|
|
// This is the fire emoji: 🔥
|
|
// It's encoded in UTF-16, as needed by the buffer.
|
|
const auto fire = L"\xD83D\xDD25";
|
|
position = fire;
|
|
|
|
// Read back the text at that position and ensure that it matches what we wrote.
|
|
const auto readBack = _buffer->GetTextDataAt(pos);
|
|
const auto readBackText = *readBack;
|
|
VERIFY_ARE_EQUAL(String(fire), String(readBackText.data(), gsl::narrow<int>(readBackText.size())));
|
|
|
|
// Prepare a delta and the new position we expect the symbol to be moved into.
|
|
const SHORT delta = 5;
|
|
const COORD newPos{ pos.X, pos.Y + delta };
|
|
|
|
// Scroll the row with our data by delta.
|
|
_buffer->ScrollRows(pos.Y, 1, delta);
|
|
|
|
// Retrieve the text at the old and new positions.
|
|
const auto shouldBeEmptyText = *_buffer->GetTextDataAt(pos);
|
|
const auto shouldBeFireText = *_buffer->GetTextDataAt(newPos);
|
|
|
|
VERIFY_ARE_EQUAL(String(L" "), String(shouldBeEmptyText.data(), gsl::narrow<int>(shouldBeEmptyText.size())));
|
|
VERIFY_ARE_EQUAL(String(fire), String(shouldBeFireText.data(), gsl::narrow<int>(shouldBeFireText.size())));
|
|
}
|
|
|
|
// This tests that rows removed from the buffer while resizing traditionally will also drop the high unicode
|
|
// characters from the Unicode Storage buffer
|
|
void TextBufferTests::ResizeTraditionalHighUnicodeRowRemoval()
|
|
{
|
|
// Set up a text buffer for us
|
|
const COORD bufferSize{ 80, 10 };
|
|
const UINT cursorSize = 12;
|
|
const TextAttribute attr{ 0x7f };
|
|
auto _buffer = std::make_unique<TextBuffer>(bufferSize, attr, cursorSize, _renderTarget);
|
|
|
|
// Get a position inside the buffer in the bottom row
|
|
const COORD pos{ 0, bufferSize.Y - 1 };
|
|
auto position = _buffer->_storage[pos.Y].GetCharRow().GlyphAt(pos.X);
|
|
|
|
// Fill it up with a sequence that will have to hit the high unicode storage.
|
|
// This is the eggplant emoji: 🍆
|
|
// It's encoded in UTF-16, as needed by the buffer.
|
|
const auto emoji = L"\xD83C\xDF46";
|
|
position = emoji;
|
|
|
|
// Read back the text at that position and ensure that it matches what we wrote.
|
|
const auto readBack = _buffer->GetTextDataAt(pos);
|
|
const auto readBackText = *readBack;
|
|
VERIFY_ARE_EQUAL(String(emoji), String(readBackText.data(), gsl::narrow<int>(readBackText.size())));
|
|
|
|
VERIFY_ARE_EQUAL(1u, _buffer->GetUnicodeStorage()._map.size(), L"There should be one item in the map.");
|
|
|
|
// Perform resize to trim off the row of the buffer that included the emoji
|
|
COORD trimmedBufferSize{ bufferSize.X, bufferSize.Y - 1 };
|
|
|
|
VERIFY_NT_SUCCESS(_buffer->ResizeTraditional(trimmedBufferSize));
|
|
|
|
VERIFY_IS_TRUE(_buffer->GetUnicodeStorage()._map.empty(), L"The map should now be empty.");
|
|
}
|
|
|
|
// This tests that columns removed from the buffer while resizing traditionally will also drop the high unicode
|
|
// characters from the Unicode Storage buffer
|
|
void TextBufferTests::ResizeTraditionalHighUnicodeColumnRemoval()
|
|
{
|
|
// Set up a text buffer for us
|
|
const COORD bufferSize{ 80, 10 };
|
|
const UINT cursorSize = 12;
|
|
const TextAttribute attr{ 0x7f };
|
|
auto _buffer = std::make_unique<TextBuffer>(bufferSize, attr, cursorSize, _renderTarget);
|
|
|
|
// Get a position inside the buffer in the last column
|
|
const COORD pos{ bufferSize.X - 1, 0 };
|
|
auto position = _buffer->_storage[pos.Y].GetCharRow().GlyphAt(pos.X);
|
|
|
|
// Fill it up with a sequence that will have to hit the high unicode storage.
|
|
// This is the peach emoji: 🍑
|
|
// It's encoded in UTF-16, as needed by the buffer.
|
|
const auto emoji = L"\xD83C\xDF51";
|
|
position = emoji;
|
|
|
|
// Read back the text at that position and ensure that it matches what we wrote.
|
|
const auto readBack = _buffer->GetTextDataAt(pos);
|
|
const auto readBackText = *readBack;
|
|
VERIFY_ARE_EQUAL(String(emoji), String(readBackText.data(), gsl::narrow<int>(readBackText.size())));
|
|
|
|
VERIFY_ARE_EQUAL(1u, _buffer->GetUnicodeStorage()._map.size(), L"There should be one item in the map.");
|
|
|
|
// Perform resize to trim off the column of the buffer that included the emoji
|
|
COORD trimmedBufferSize{ bufferSize.X - 1, bufferSize.Y};
|
|
|
|
VERIFY_NT_SUCCESS(_buffer->ResizeTraditional(trimmedBufferSize));
|
|
|
|
VERIFY_IS_TRUE(_buffer->GetUnicodeStorage()._map.empty(), L"The map should now be empty.");
|
|
}
|
|
|
|
void TextBufferTests::TestBurrito()
|
|
{
|
|
COORD bufferSize{ 80, 9001 };
|
|
UINT cursorSize = 12;
|
|
TextAttribute attr{ 0x7f };
|
|
auto _buffer = std::make_unique<TextBuffer>(bufferSize, attr, cursorSize, _renderTarget);
|
|
|
|
// This is the burrito emoji: 🌯
|
|
// It's encoded in UTF-16, as needed by the buffer.
|
|
const auto burrito = L"\xD83C\xDF2F";
|
|
OutputCellIterator burriter{ burrito };
|
|
|
|
auto afterFIter = _buffer->Write({ L"F" });
|
|
_buffer->IncrementCursor();
|
|
|
|
auto afterBurritoIter = _buffer->Write(burriter);
|
|
_buffer->IncrementCursor();
|
|
_buffer->IncrementCursor();
|
|
VERIFY_IS_FALSE(afterBurritoIter);
|
|
}
|