mirror of
https://github.com/ArthurSonzogni/FTXUI.git
synced 2025-12-16 01:48:56 +08:00
Warn users they have defined the min/max macros which is not compatible with other code from the standard library or FTXUI. Co-authored-by: Sylko Olzscher <sylko.olzscher@solostec.ch> Co-authored-by: ArthurSonzogni <sonzogniarthur@gmail.com>
482 lines
14 KiB
C++
482 lines
14 KiB
C++
// Copyright 2020 Arthur Sonzogni. All rights reserved.
|
||
// Use of this source code is governed by the MIT license that can be found in
|
||
// the LICENSE file.
|
||
#include <ftxui/component/mouse.hpp> // for Mouse, Mouse::Left, Mouse::Middle, Mouse::Pressed, Mouse::Released, Mouse::Right
|
||
#include <functional> // for function
|
||
#include <initializer_list> // for initializer_list
|
||
#include <memory> // for allocator, unique_ptr
|
||
#include <vector> // for vector
|
||
|
||
#include "ftxui/component/event.hpp" // for Event, Event::Return, Event::ArrowDown, Event::ArrowLeft, Event::ArrowRight, Event::ArrowUp, Event::Backspace, Event::End, Event::Home, Event::Custom, Event::Delete, Event::F1, Event::F10, Event::F11, Event::F12, Event::F2, Event::F3, Event::F4, Event::F5, Event::F6, Event::F7, Event::F8, Event::F9, Event::PageDown, Event::PageUp, Event::Tab, Event::TabReverse, Event::Escape
|
||
#include "ftxui/component/terminal_input_parser.hpp"
|
||
#include "gtest/gtest.h" // for AssertionResult, Test, Message, TestPartResult, EXPECT_EQ, EXPECT_TRUE, TEST, EXPECT_FALSE
|
||
|
||
// NOLINTBEGIN
|
||
namespace ftxui {
|
||
|
||
// Test char |c| to are trivially converted into |Event::Character(c)|.
|
||
TEST(Event, Character) {
|
||
std::vector<char> basic_char;
|
||
for (char c = 'a'; c <= 'z'; ++c) {
|
||
basic_char.push_back(c);
|
||
}
|
||
for (char c = 'A'; c <= 'Z'; ++c) {
|
||
basic_char.push_back(c);
|
||
}
|
||
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
for (char c : basic_char) {
|
||
parser.Add(c);
|
||
}
|
||
|
||
for (size_t i = 0; i < basic_char.size(); ++i) {
|
||
EXPECT_TRUE(received_events[i].is_character());
|
||
EXPECT_EQ(basic_char[i], received_events[i].character()[0]);
|
||
}
|
||
EXPECT_EQ(received_events.size(), basic_char.size());
|
||
}
|
||
|
||
TEST(Event, EscapeKeyWithoutWaiting) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
|
||
EXPECT_TRUE(received_events.empty());
|
||
}
|
||
|
||
TEST(Event, EscapeKeyNotEnoughWait) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Timeout(49);
|
||
|
||
EXPECT_TRUE(received_events.empty());
|
||
}
|
||
|
||
TEST(Event, EscapeKeyEnoughWait) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Timeout(50);
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_EQ(received_events[0], Event::Escape);
|
||
}
|
||
|
||
TEST(Event, EscapeFast) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('a');
|
||
parser.Add('');
|
||
parser.Add('b');
|
||
parser.Timeout(49);
|
||
|
||
EXPECT_EQ(2, received_events.size());
|
||
EXPECT_EQ(received_events[0], Event::AltA);
|
||
EXPECT_EQ(received_events[1], Event::AltB);
|
||
}
|
||
|
||
TEST(Event, MouseLeftClickPressed) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('[');
|
||
parser.Add('0');
|
||
parser.Add(';');
|
||
parser.Add('1');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('4');
|
||
parser.Add('2');
|
||
parser.Add('M');
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_mouse());
|
||
EXPECT_EQ(Mouse::Left, received_events[0].mouse().button);
|
||
EXPECT_EQ(12, received_events[0].mouse().x);
|
||
EXPECT_EQ(42, received_events[0].mouse().y);
|
||
EXPECT_EQ(received_events[0].mouse().motion, Mouse::Pressed);
|
||
}
|
||
|
||
TEST(Event, MouseLeftMoved) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('[');
|
||
parser.Add('3');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('1');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('4');
|
||
parser.Add('2');
|
||
parser.Add('M');
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_mouse());
|
||
EXPECT_EQ(Mouse::Left, received_events[0].mouse().button);
|
||
EXPECT_EQ(12, received_events[0].mouse().x);
|
||
EXPECT_EQ(42, received_events[0].mouse().y);
|
||
EXPECT_EQ(received_events[0].mouse().motion, Mouse::Moved);
|
||
}
|
||
|
||
TEST(Event, MouseLeftClickReleased) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('[');
|
||
parser.Add('0');
|
||
parser.Add(';');
|
||
parser.Add('1');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('4');
|
||
parser.Add('2');
|
||
parser.Add('m');
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_mouse());
|
||
EXPECT_EQ(Mouse::Left, received_events[0].mouse().button);
|
||
EXPECT_EQ(12, received_events[0].mouse().x);
|
||
EXPECT_EQ(42, received_events[0].mouse().y);
|
||
EXPECT_EQ(received_events[0].mouse().motion, Mouse::Released);
|
||
}
|
||
|
||
TEST(Event, MouseReporting) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('[');
|
||
parser.Add('1');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('4');
|
||
parser.Add('2');
|
||
parser.Add('R');
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_cursor_position());
|
||
EXPECT_EQ(42, received_events[0].cursor_x());
|
||
EXPECT_EQ(12, received_events[0].cursor_y());
|
||
}
|
||
|
||
TEST(Event, MouseMiddleClick) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('[');
|
||
parser.Add('3');
|
||
parser.Add('3');
|
||
parser.Add(';');
|
||
parser.Add('1');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('4');
|
||
parser.Add('2');
|
||
parser.Add('M');
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_mouse());
|
||
EXPECT_EQ(Mouse::Middle, received_events[0].mouse().button);
|
||
EXPECT_EQ(12, received_events[0].mouse().x);
|
||
EXPECT_EQ(42, received_events[0].mouse().y);
|
||
}
|
||
|
||
TEST(Event, MouseRightClick) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add('');
|
||
parser.Add('[');
|
||
parser.Add('3');
|
||
parser.Add('4');
|
||
parser.Add(';');
|
||
parser.Add('1');
|
||
parser.Add('2');
|
||
parser.Add(';');
|
||
parser.Add('4');
|
||
parser.Add('2');
|
||
parser.Add('M');
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_mouse());
|
||
EXPECT_EQ(Mouse::Right, received_events[0].mouse().button);
|
||
EXPECT_EQ(12, received_events[0].mouse().x);
|
||
EXPECT_EQ(42, received_events[0].mouse().y);
|
||
}
|
||
|
||
TEST(Event, UTF8) {
|
||
struct {
|
||
std::vector<unsigned char> input;
|
||
bool valid;
|
||
} kTestCase[] = {
|
||
// Basic characters.
|
||
{{'a'}, true},
|
||
{{'z'}, true},
|
||
{{'A'}, true},
|
||
{{'Z'}, true},
|
||
{{'0'}, true},
|
||
{{'9'}, true},
|
||
|
||
// UTF-8 of various size:
|
||
{{0b0100'0001}, true},
|
||
{{0b1100'0010, 0b1000'0000}, true},
|
||
{{0b1110'0010, 0b1000'0000, 0b1000'0000}, true},
|
||
{{0b1111'0010, 0b1000'0000, 0b1000'0000, 0b1000'0000}, true},
|
||
|
||
// Overlong UTF-8 encoding:
|
||
{{0b1100'0000, 0b1000'0000}, false},
|
||
{{0b1110'0000, 0b1000'0000, 0b1000'0000}, false},
|
||
{{0b1111'0000, 0b1000'0000, 0b1000'0000, 0b1000'0000}, false},
|
||
|
||
// Test limits in between the various legal regions
|
||
// https://unicode.org/versions/corrigendum1.html
|
||
// Limit in between the valid and ina
|
||
// {{0x7F}, true}, => Special sequence.
|
||
{{0x80}, false},
|
||
// ---
|
||
{{0xC1, 0x80}, false},
|
||
{{0xC2, 0x7F}, false},
|
||
{{0xC2, 0x80}, true},
|
||
// ---
|
||
{{0xDF, 0xBF}, true},
|
||
{{0xDF, 0xC0}, false},
|
||
// ---
|
||
{{0xE0, 0x9F, 0x80}, false},
|
||
{{0xE0, 0xA0, 0x7F}, false},
|
||
{{0xE0, 0xA0, 0x80}, true},
|
||
// ---
|
||
{{0xE0, 0xBF, 0xBF}, true},
|
||
// ---
|
||
{{0xE1, 0x7F, 0x80}, false},
|
||
{{0xE1, 0x80, 0x7f}, false},
|
||
{{0xE1, 0x80, 0x80}, true},
|
||
// --
|
||
{{0xEF, 0xBF, 0xBF}, true},
|
||
{{0xEF, 0xC0, 0xBF}, false},
|
||
{{0xEF, 0xBF, 0xC0}, false},
|
||
// --
|
||
{{0xF0, 0x90, 0x80}, false},
|
||
{{0xF0, 0x8F, 0x80, 0x80}, false},
|
||
{{0xF0, 0x90, 0x80, 0x7F}, false},
|
||
{{0xF0, 0x90, 0x80, 0x80}, true},
|
||
// --
|
||
{{0xF1, 0x80, 0x80, 0x80}, true},
|
||
// --
|
||
{{0xF1, 0xBF, 0xBF, 0xBF}, true},
|
||
// --
|
||
{{0xF2, 0x80, 0x80, 0x80}, true},
|
||
// --
|
||
{{0xF4, 0x8F, 0xBF, 0xBF}, true},
|
||
{{0xF4, 0x90, 0xBF, 0xBF}, false},
|
||
|
||
};
|
||
for (auto test : kTestCase) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
for (auto input : test.input) {
|
||
parser.Add(input);
|
||
}
|
||
|
||
if (test.valid) {
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_character());
|
||
} else {
|
||
EXPECT_TRUE(received_events.empty());
|
||
}
|
||
}
|
||
}
|
||
|
||
TEST(Event, NewLine) {
|
||
for (char newline : {'\r', '\n'}) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add(newline);
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0] == Event::Return);
|
||
}
|
||
}
|
||
|
||
TEST(Event, Control) {
|
||
struct TestCase {
|
||
char input;
|
||
bool cancel;
|
||
};
|
||
std::vector<TestCase> cases;
|
||
for (int i = 0; i < 32; ++i) {
|
||
if (i == 8 || i == 13 || i == 24 || i == 26 || i == 27) {
|
||
continue;
|
||
}
|
||
cases.push_back({char(i), false});
|
||
}
|
||
cases.push_back({char(24), false});
|
||
cases.push_back({char(26), false});
|
||
cases.push_back({char(127), false});
|
||
|
||
for (auto test : cases) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add(test.input);
|
||
|
||
if (test.cancel) {
|
||
EXPECT_TRUE(received_events.empty());
|
||
} else {
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_EQ(received_events[0], Event::Special({test.input}));
|
||
}
|
||
}
|
||
}
|
||
|
||
TEST(Event, Special) {
|
||
auto str = [](std::string input) {
|
||
std::vector<unsigned char> output;
|
||
for (auto it : input) {
|
||
output.push_back(it);
|
||
}
|
||
return output;
|
||
};
|
||
|
||
struct {
|
||
std::vector<unsigned char> input;
|
||
Event expected;
|
||
} kTestCase[] = {
|
||
// Arrow (default cursor mode)
|
||
{str("[A"), Event::ArrowUp},
|
||
{str("[B"), Event::ArrowDown},
|
||
{str("[C"), Event::ArrowRight},
|
||
{str("[D"), Event::ArrowLeft},
|
||
{str("[H"), Event::Home},
|
||
{str("[F"), Event::End},
|
||
|
||
// Arrow (application cursor mode)
|
||
{str("\x1BOA"), Event::ArrowUp},
|
||
{str("\x1BOB"), Event::ArrowDown},
|
||
{str("\x1BOC"), Event::ArrowRight},
|
||
{str("\x1BOD"), Event::ArrowLeft},
|
||
{str("\x1BOH"), Event::Home},
|
||
{str("\x1BOF"), Event::End},
|
||
|
||
// Backspace & Quirk for:
|
||
// https://github.com/ArthurSonzogni/FTXUI/issues/508
|
||
{{127}, Event::Backspace},
|
||
{{8}, Event::Backspace},
|
||
|
||
// Delete
|
||
{str("\x1B[3~"), Event::Delete},
|
||
|
||
// Return
|
||
{{13}, Event::Return},
|
||
{{10}, Event::Return},
|
||
|
||
// Tabs:
|
||
{{9}, Event::Tab},
|
||
{{27, 91, 90}, Event::TabReverse},
|
||
|
||
// Function keys
|
||
{str("\x1BOP"), Event::F1},
|
||
{str("\x1BOQ"), Event::F2},
|
||
{str("\x1BOR"), Event::F3},
|
||
{str("\x1BOS"), Event::F4},
|
||
{str("\x1B[15~"), Event::F5},
|
||
{str("\x1B[17~"), Event::F6},
|
||
{str("\x1B[18~"), Event::F7},
|
||
{str("\x1B[19~"), Event::F8},
|
||
{str("\x1B[20~"), Event::F9},
|
||
{str("\x1B[21~"), Event::F10},
|
||
{str("\x1B[23~"), Event::F11},
|
||
{str("\x1B[24~"), Event::F12},
|
||
|
||
// Function keys for virtual terminal:
|
||
{str("\x1B[[A"), Event::F1},
|
||
{str("\x1B[[B"), Event::F2},
|
||
{str("\x1B[[C"), Event::F3},
|
||
{str("\x1B[[D"), Event::F4},
|
||
{str("\x1B[[E"), Event::F5},
|
||
|
||
// Function keys for xterm-r5, xterm-r6, rxvt
|
||
{str("\x1B[11~"), Event::F1},
|
||
{str("\x1B[12~"), Event::F2},
|
||
{str("\x1B[13~"), Event::F3},
|
||
{str("\x1B[14~"), Event::F4},
|
||
|
||
// Function keys for vt100
|
||
{str("\x1BOt"), Event::F5},
|
||
{str("\x1BOu"), Event::F6},
|
||
{str("\x1BOv"), Event::F7},
|
||
{str("\x1BOl"), Event::F8},
|
||
{str("\x1BOw"), Event::F9},
|
||
{str("\x1BOx"), Event::F10},
|
||
|
||
// Function keys for scoansi
|
||
{str("\x1B[M"), Event::F1},
|
||
{str("\x1B[N"), Event::F2},
|
||
{str("\x1B[O"), Event::F3},
|
||
{str("\x1B[P"), Event::F4},
|
||
{str("\x1B[Q"), Event::F5},
|
||
{str("\x1B[R"), Event::F6},
|
||
{str("\x1B[S"), Event::F7},
|
||
{str("\x1B[T"), Event::F8},
|
||
{str("\x1B[U"), Event::F9},
|
||
{str("\x1B[V"), Event::F10},
|
||
{str("\x1B[W"), Event::F11},
|
||
{str("\x1B[X"), Event::F12},
|
||
|
||
// Page up and down:
|
||
{str("\x1B[5~"), Event::PageUp},
|
||
{str("\x1B[6~"), Event::PageDown},
|
||
|
||
// Custom:
|
||
{{0}, Event::Custom},
|
||
};
|
||
|
||
for (auto test : kTestCase) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
for (auto input : test.input) {
|
||
parser.Add(input);
|
||
}
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_EQ(received_events[0], test.expected);
|
||
}
|
||
}
|
||
|
||
TEST(Event, DeviceControlString) {
|
||
std::vector<Event> received_events;
|
||
auto parser = TerminalInputParser(
|
||
[&](Event event) { received_events.push_back(std::move(event)); });
|
||
parser.Add(27); // ESC
|
||
parser.Add(80); // P
|
||
parser.Add(49); // 1
|
||
parser.Add(36); // $
|
||
parser.Add(114); // r
|
||
parser.Add(49); // 1
|
||
parser.Add(32); // SP
|
||
parser.Add(113); // q
|
||
parser.Add(27); // ESC
|
||
parser.Add(92); // (backslash)
|
||
|
||
EXPECT_EQ(1, received_events.size());
|
||
EXPECT_TRUE(received_events[0].is_cursor_shape());
|
||
EXPECT_EQ(1, received_events[0].cursor_shape());
|
||
}
|
||
|
||
} // namespace ftxui
|
||
// NOLINTEND
|