15#include <initializer_list>
44#define DEFINE_CONSOLEV2_PROPERTIES
45#define WIN32_LEAN_AND_MEAN
60#if defined(__EMSCRIPTEN__)
61#include <emscripten.h>
75 auto* screen = App::Active();
77 screen->RequestAnimationFrame();
82#if defined(__EMSCRIPTEN__)
85void ftxui_on_resize(
int columns,
int rows) {
98 App* suspended_screen_ =
nullptr;
100 const bool use_alternative_screen_;
102 bool track_mouse_ =
true;
104 std::string set_cursor_position_;
105 std::string reset_cursor_position_;
107 std::atomic<bool> quit_{
false};
108 bool installed_ =
false;
109 bool animation_requested_ =
false;
115 std::uint64_t frame_count_ = 0;
116 bool mouse_captured =
false;
117 bool previous_frame_resized_ =
false;
119 bool frame_valid_ =
false;
121 bool force_handle_ctrl_c_ =
true;
122 bool force_handle_ctrl_z_ =
true;
124 int cursor_reset_shape_ = 1;
127 bool handle_piped_input_ =
true;
128 bool is_stdin_a_tty_ =
false;
129 bool is_stdout_a_tty_ =
false;
133 std::string terminal_name_ =
"unknown";
134 int terminal_version_ = 0;
136 std::string terminal_emulator_name_ =
"unknown";
137 std::string terminal_emulator_version_ =
"unknown";
139 std::vector<int> terminal_capabilities_;
143 struct SelectionData {
149 bool operator==(
const SelectionData& other)
const {
150 if (empty && other.empty) {
153 if (empty || other.empty) {
156 return start_x == other.start_x && start_y == other.start_y &&
157 end_x == other.end_x && end_y == other.end_y;
159 bool operator!=(
const SelectionData& other)
const {
160 return !(*
this == other);
163 SelectionData selection_data_;
164 SelectionData selection_data_previous_;
165 std::unique_ptr<Selection> selection_;
166 std::function<void()> selection_on_change_;
171 TerminalInputParser terminal_input_parser;
172 task::TaskRunner task_runner;
173 std::chrono::time_point<std::chrono::steady_clock> last_char_time =
174 std::chrono::steady_clock::now();
175 std::string output_buffer;
177 class ThrottledRequest {
179 ThrottledRequest(App::Internal* internal, std::function<
void()> send)
180 : internal_(internal), send_(std::move(send)) {}
182 void Request(
bool force =
false) {
183 if (!internal_->is_stdin_a_tty_) {
198 const auto now = std::chrono::steady_clock::now();
199 const auto delta = now - last_request_time_;
200 const auto delay = std::chrono::milliseconds(500) - delta;
202 if (delay <= std::chrono::milliseconds(0)) {
207 request_queued_ =
true;
210 request_queued_ =
false;
216 void OnReply() { pending_request_ =
false; }
218 bool HasPending()
const {
219 if (pending_request_) {
220 const auto now = std::chrono::steady_clock::now();
221 if (now - last_sent_time_ < std::chrono::seconds(5)) {
225 return request_queued_;
230 last_sent_time_ = std::chrono::steady_clock::now();
231 pending_request_ =
true;
235 App::Internal* internal_;
236 std::function<void()> send_;
237 bool pending_request_ =
false;
238 std::chrono::steady_clock::time_point last_request_time_ =
239 std::chrono::steady_clock::now() - std::chrono::hours(1);
240 std::chrono::steady_clock::time_point last_sent_time_ =
241 std::chrono::steady_clock::now() - std::chrono::hours(1);
242 bool request_queued_ =
false;
245 ThrottledRequest cursor_position_request;
247 MultiReceiverBuffer<Event> event_buffer;
248 std::unique_ptr<MultiReceiverBuffer<Event>::Receiver> setup_receiver;
249 std::unique_ptr<MultiReceiverBuffer<Event>::Receiver> main_loop_receiver;
251 Internal(App* app,
AppDimension dimension,
bool use_alternative_screen);
259 void RunOnce(
const Component& component);
260 void RunOnceBlocking(
Component component);
262 bool HandleSelection(
bool handled, Event event);
264 std::string ResetCursorPosition();
265 void RequestCursorPosition(
bool force =
false);
266 void TerminalSend(std::string_view);
267 void TerminalFlush();
268 void InstallPipedInputHandling();
269 void InstallTerminalInfo();
270 void Signal(
int signal);
271 size_t FetchTerminalEvents();
272 void PostAnimationTask();
277App* g_active_screen =
nullptr;
279std::stack<Closure> on_exit_functions;
282 while (!on_exit_functions.empty()) {
283 on_exit_functions.top()();
284 on_exit_functions.pop();
289const std::string CSI =
"\x1b[";
292const std::string DCS =
"\x1bP";
295const std::string ST =
"\x1b\\";
299const std::string DECRQSS_DECSCUSR = DCS +
"$q q" + ST;
302enum class DECMode : std::uint16_t {
308 kMouseVt200Highlight = 1001,
310 kMouseBtnEventMouse = 1002,
311 kMouseAnyEvent = 1003,
314 kMouseSgrExtMode = 1006,
315 kMouseUrxvtMode = 1015,
316 kMouseSgrPixelsMode = 1016,
317 kAlternateScreen = 1049,
321enum class DSRMode : std::uint8_t {
325std::string Serialize(
const std::vector<DECMode>& parameters) {
328 for (
const DECMode parameter : parameters) {
332 out += std::to_string(
int(parameter));
339std::string Set(
const std::vector<DECMode>& parameters) {
340 return CSI +
"?" + Serialize(parameters) +
"h";
344std::string Reset(
const std::vector<DECMode>& parameters) {
345 return CSI +
"?" + Serialize(parameters) +
"l";
349std::string DeviceStatusReport(DSRMode ps) {
350 return CSI + std::to_string(
int(ps)) +
"n";
353class CapturedMouseImpl :
public CapturedMouseInterface {
355 explicit CapturedMouseImpl(std::function<
void(
void)> callback)
356 : callback_(std::move(callback)) {}
357 ~CapturedMouseImpl()
override { callback_(); }
358 CapturedMouseImpl(
const CapturedMouseImpl&) =
delete;
359 CapturedMouseImpl(CapturedMouseImpl&&) =
delete;
360 CapturedMouseImpl& operator=(
const CapturedMouseImpl&) =
delete;
361 CapturedMouseImpl& operator=(CapturedMouseImpl&&) =
delete;
364 std::function<void(
void)> callback_;
368std::atomic<int> g_signal_exit_count = 0;
369std::atomic<int> g_signal_stop_count = 0;
370std::atomic<int> g_signal_resize_count = 0;
372std::atomic<int> g_signal_exit_count = 0;
377std::atomic<bool> g_terminal_is_raw{
false};
381std::atomic<int> g_last_signal{0};
384using SignalHandler = void (*)(int);
386std::map<int, SignalHandler> g_old_signal_handlers;
389DWORD g_original_stdout_mode = 0;
390DWORD g_original_stdin_mode = 0;
391bool g_has_original_console_mode =
false;
394std::map<int, struct sigaction> g_old_sigactions;
397struct termios g_original_termios;
398bool g_has_original_termios =
false;
404void RestoreSignalHandlerAndRaise(
int signal) {
406 auto it = g_old_signal_handlers.find(signal);
407 auto old_handler = (it != g_old_signal_handlers.end()) ? it->second : SIG_DFL;
408 std::signal(signal, old_handler);
410 auto it = g_old_sigactions.find(signal);
411 if (it != g_old_sigactions.end()) {
412 sigaction(signal, &it->second,
nullptr);
415 sa.sa_handler = SIG_DFL;
416 sigemptyset(&sa.sa_mask);
418 sigaction(signal, &sa,
nullptr);
426void RestoreTerminalEmergency() {
427 if (!g_terminal_is_raw.exchange(
false)) {
431 if (g_has_original_console_mode) {
432 auto stdout_handle = GetStdHandle(STD_OUTPUT_HANDLE);
433 auto stdin_handle = GetStdHandle(STD_INPUT_HANDLE);
434 SetConsoleMode(stdout_handle, g_original_stdout_mode);
435 SetConsoleMode(stdin_handle, g_original_stdin_mode);
438 if (g_has_original_termios && g_tty_fd >= 0) {
439 const char restore_seq[] =
448 std::ignore = write(STDOUT_FILENO, restore_seq,
sizeof(restore_seq) - 1);
449 tcsetattr(g_tty_fd, TCSANOW, &g_original_termios);
455void RecordSignal(
int signal) {
472 RestoreTerminalEmergency();
473 RestoreSignalHandlerAndRaise(signal);
487 g_last_signal.store(signal);
488 g_signal_exit_count++;
494 g_signal_stop_count++;
499 g_signal_resize_count++;
508void ExecuteSignalHandlers() {
509 if (g_last_signal.load() != 0) {
510 App::Private::Signal(*g_active_screen, SIGABRT);
513 int signal_exit_count = g_signal_exit_count.exchange(0);
514 while (signal_exit_count--) {
515 App::Private::Signal(*g_active_screen, SIGABRT);
519 int signal_stop_count = g_signal_stop_count.exchange(0);
520 while (signal_stop_count--) {
521 App::Private::Signal(*g_active_screen, SIGTSTP);
524 int signal_resize_count = g_signal_resize_count.exchange(0);
525 while (signal_resize_count--) {
526 App::Private::Signal(*g_active_screen, SIGWINCH);
531void InstallSignalHandler(
int sig) {
533 auto old_signal_handler = std::signal(sig, RecordSignal);
534 g_old_signal_handlers[sig] = old_signal_handler;
535 on_exit_functions.emplace(
536 [=] { std::ignore = std::signal(sig, old_signal_handler); });
539 sa.sa_handler = RecordSignal;
540 sigemptyset(&sa.sa_mask);
541 sa.sa_flags = SA_RESTART;
542 struct sigaction old_sa;
543 sigaction(sig, &sa, &old_sa);
544 g_old_sigactions[sig] = old_sa;
545 on_exit_functions.emplace([=] { sigaction(sig, &old_sa,
nullptr); });
551App::Internal::Internal(App* app,
553 bool use_alternative_screen)
555 dimension_(dimension),
556 use_alternative_screen_(use_alternative_screen),
557 terminal_input_parser([&](Event event) {
558 event_buffer.Push(std::move(event));
559 public_->RequestAnimationFrame();
561 cursor_position_request(
this, [
this] {
562 TerminalSend(DeviceStatusReport(DSRMode::kCursor));
564 setup_receiver = event_buffer.CreateReceiver();
565 main_loop_receiver = event_buffer.CreateReceiver();
568void App::Internal::ExitNow() {
572void App::Internal::Install() {
573 frame_valid_ =
false;
582 InstallPipedInputHandling();
586 on_exit_functions.emplace([
this] { TerminalFlush(); });
590 for (
const int signal : {SIGTERM, SIGSEGV, SIGINT, SIGILL, SIGABRT, SIGFPE}) {
591 InstallSignalHandler(signal);
597 auto stdout_handle = GetStdHandle(STD_OUTPUT_HANDLE);
598 auto stdin_handle = GetStdHandle(STD_INPUT_HANDLE);
602 GetConsoleMode(stdout_handle, &out_mode);
603 GetConsoleMode(stdin_handle, &in_mode);
604 g_original_stdout_mode = out_mode;
605 g_original_stdin_mode = in_mode;
606 g_has_original_console_mode =
true;
607 on_exit_functions.push([=] { SetConsoleMode(stdout_handle, out_mode); });
608 on_exit_functions.push([=] { SetConsoleMode(stdin_handle, in_mode); });
611 const int enable_virtual_terminal_processing = 0x0004;
612 const int disable_newline_auto_return = 0x0008;
613 out_mode |= enable_virtual_terminal_processing;
614 out_mode |= disable_newline_auto_return;
617 const int enable_line_input = 0x0002;
618 const int enable_echo_input = 0x0004;
619 const int enable_virtual_terminal_input = 0x0200;
620 const int enable_window_input = 0x0008;
621 in_mode &= ~enable_echo_input;
622 in_mode &= ~enable_line_input;
623 in_mode |= enable_virtual_terminal_input;
624 in_mode |= enable_window_input;
626 SetConsoleMode(stdin_handle, in_mode);
627 SetConsoleMode(stdout_handle, out_mode);
629 for (
const int signal :
630 {SIGWINCH, SIGTSTP, SIGBUS, SIGSYS, SIGQUIT, SIGHUP}) {
631 InstallSignalHandler(signal);
634 struct termios terminal;
635 tcgetattr(tty_fd_, &terminal);
636 g_original_termios = terminal;
638 g_has_original_termios =
true;
639 on_exit_functions.emplace([terminal = terminal, tty_fd_ = tty_fd_] {
640 tcsetattr(tty_fd_, TCSANOW, &terminal);
644 terminal.c_iflag &= ~IGNBRK;
645 terminal.c_iflag &= ~BRKINT;
647 terminal.c_iflag &= ~PARMRK;
648 terminal.c_iflag &= ~ISTRIP;
649 terminal.c_iflag &= ~INLCR;
650 terminal.c_iflag &= ~IGNCR;
651 terminal.c_iflag &= ~ICRNL;
652 terminal.c_iflag &= ~IXON;
654 terminal.c_lflag &= ~ECHO;
655 terminal.c_lflag &= ~ECHONL;
656 terminal.c_lflag &= ~ICANON;
657 terminal.c_lflag &= ~ISIG;
662 terminal.c_lflag &= ~IEXTEN;
663 terminal.c_cflag |= CS8;
665 terminal.c_cc[VMIN] = 0;
667 terminal.c_cc[VTIME] = 0;
669 tcsetattr(tty_fd_, TCSANOW, &terminal);
673 auto enable = [&](
const std::vector<DECMode>& parameters) {
674 TerminalSend(Set(parameters));
675 on_exit_functions.emplace(
676 [
this, parameters] { TerminalSend(Reset(parameters)); });
679 auto disable = [&](
const std::vector<DECMode>& parameters) {
680 TerminalSend(Reset(parameters));
681 on_exit_functions.emplace(
682 [
this, parameters] { TerminalSend(Set(parameters)); });
685 if (use_alternative_screen_) {
687 DECMode::kAlternateScreen,
696 enable({DECMode::kMouseVt200});
697 enable({DECMode::kMouseAnyEvent});
698 enable({DECMode::kMouseUrxvtMode});
699 enable({DECMode::kMouseSgrExtMode});
706 InstallTerminalInfo();
713 g_terminal_is_raw =
true;
716void App::Internal::Uninstall() {
717 g_terminal_is_raw =
false;
721 if (is_stdin_a_tty_ && is_stdout_a_tty_) {
722 auto closing_receiver =
723 event_buffer.CreateReceiverAt(main_loop_receiver->index());
724 auto start = std::chrono::steady_clock::now();
725 while (cursor_position_request.HasPending()) {
726 FetchTerminalEvents();
728 while (closing_receiver->Has()) {
729 const auto event = closing_receiver->Pop();
730 if (event.is_cursor_position()) {
731 cursor_x_ =
event.cursor_x();
732 cursor_y_ =
event.cursor_y();
733 cursor_position_request.OnReply();
737 task_runner.RunUntilIdle();
739 if (std::chrono::steady_clock::now() - start >
740 std::chrono::milliseconds(400)) {
743 std::this_thread::sleep_for(std::chrono::milliseconds(10));
750void App::Internal::PreMain() {
752 if (g_active_screen) {
753 std::swap(suspended_screen_, g_active_screen);
755 suspended_screen_->internal_->TerminalSend(
756 suspended_screen_->internal_->ResetCursorPosition());
757 suspended_screen_->ResetPosition(
758 suspended_screen_->internal_->output_buffer,
760 suspended_screen_->dimx_ = 0;
761 suspended_screen_->dimy_ = 0;
764 suspended_screen_->internal_->Uninstall();
768 g_active_screen = public_;
769 g_active_screen->internal_->Install();
771 previous_animation_time_ = animation::Clock::now();
774void App::Internal::PostMain() {
776 TerminalSend(ResetCursorPosition());
778 g_active_screen =
nullptr;
781 if (suspended_screen_) {
783 public_->ResetPosition(output_buffer,
true);
787 std::swap(g_active_screen, suspended_screen_);
788 g_active_screen->internal_->Install();
795 if (!use_alternative_screen_) {
798 std::cout << std::flush;
801 int sig = g_last_signal.exchange(0);
803 RestoreSignalHandlerAndRaise(sig);
807bool App::Internal::HasQuitted() {
811void App::Internal::RunOnce(
const Component& component) {
812 const AutoReset set_component(&component_, component);
813 ExecuteSignalHandlers();
814 FetchTerminalEvents();
816 while (!quit_ && main_loop_receiver->Has()) {
817 public_->Post(main_loop_receiver->Pop());
821 const size_t executed_task = task_runner.ExecutedTasks();
822 task_runner.RunUntilIdle();
824 if (executed_task == task_runner.ExecutedTasks()) {
828 ExecuteSignalHandlers();
831 if (selection_data_previous_ != selection_data_) {
832 selection_data_previous_ = selection_data_;
833 if (selection_on_change_) {
834 selection_on_change_();
835 public_->Post(Event::Custom);
840void App::Internal::RunOnceBlocking(
Component component) {
842 const auto time_per_frame = std::chrono::microseconds(16666);
844 auto time = std::chrono::steady_clock::now();
845 const size_t executed_task = task_runner.ExecutedTasks();
848 while (executed_task == task_runner.ExecutedTasks() && !HasQuitted()) {
851 const auto now = std::chrono::steady_clock::now();
852 const auto delta = now - time;
855 if (delta < time_per_frame) {
856 const auto sleep_duration = time_per_frame - delta;
857 std::this_thread::sleep_for(sleep_duration);
862void App::Internal::HandleTask(
Component component,
Task& task) {
865 using T = std::decay_t<
decltype(arg)>;
869 if constexpr (std::is_same_v<T, Event>) {
871 if (arg.is_cursor_position()) {
872 cursor_x_ = arg.cursor_x();
873 cursor_y_ = arg.cursor_y();
874 cursor_position_request.OnReply();
878 if (arg.is_cursor_shape()) {
879 cursor_reset_shape_ = arg.cursor_shape();
883 if (arg.IsTerminalCapabilities()) {
884 terminal_capabilities_ = arg.TerminalCapabilities();
888 if (arg.IsTerminalNameVersion()) {
889 terminal_name_ = arg.TerminalName();
890 terminal_version_ = arg.TerminalVersion();
894 if (arg.IsTerminalEmulator()) {
895 terminal_emulator_name_ = arg.TerminalEmulatorName();
896 terminal_emulator_version_ = arg.TerminalEmulatorVersion();
900 if (arg.is_mouse()) {
901 arg.mouse().x -= cursor_x_;
902 arg.mouse().y -= cursor_y_;
905 arg.screen_ = public_;
907 bool handled = component->OnEvent(arg);
908 handled = HandleSelection(handled, arg);
910 if (arg == Event::CtrlC && (!handled || force_handle_ctrl_c_)) {
911 RecordSignal(SIGINT);
915 if (arg == Event::CtrlZ && (!handled || force_handle_ctrl_z_)) {
916 RecordSignal(SIGTSTP);
920 frame_valid_ =
false;
925 if constexpr (std::is_same_v<T, Closure>) {
931 if constexpr (std::is_same_v<T, AnimationTask>) {
932 if (!animation_requested_) {
936 animation_requested_ =
false;
939 previous_animation_time_ = now;
941 animation::Params params(delta);
942 component->OnAnimation(params);
943 frame_valid_ =
false;
951bool App::Internal::HandleSelection(
bool handled, Event event) {
953 selection_pending_ =
nullptr;
954 selection_data_.empty =
true;
955 selection_ =
nullptr;
959 if (!event.is_mouse()) {
963 auto& mouse =
event.mouse();
964 if (mouse.button != Mouse::Left) {
968 if (mouse.motion == Mouse::Pressed) {
969 selection_pending_ = public_->CaptureMouse();
970 selection_data_.start_x = mouse.x;
971 selection_data_.start_y = mouse.y;
972 selection_data_.end_x = mouse.x;
973 selection_data_.end_y = mouse.y;
977 if (!selection_pending_) {
981 if (mouse.motion == Mouse::Moved) {
982 if ((mouse.x != selection_data_.end_x) ||
983 (mouse.y != selection_data_.end_y)) {
984 selection_data_.end_x = mouse.x;
985 selection_data_.end_y = mouse.y;
986 selection_data_.empty =
false;
992 if (mouse.motion == Mouse::Released) {
993 selection_pending_ =
nullptr;
994 selection_data_.end_x = mouse.x;
995 selection_data_.end_y = mouse.y;
996 selection_data_.empty =
false;
1003void App::Internal::Draw(
Component component) {
1007 auto document = component->Render();
1011 document->ComputeRequirement();
1012 switch (dimension_) {
1014 dimx = public_->dimx_;
1015 dimy = public_->dimy_;
1018 dimx = terminal.dimx;
1019 dimy =
util::clamp(document->requirement().min_y, 0, terminal.dimy);
1022 dimx = terminal.dimx;
1023 dimy = terminal.dimy;
1026 dimx =
util::clamp(document->requirement().min_x, 0, terminal.dimx);
1027 dimy =
util::clamp(document->requirement().min_y, 0, terminal.dimy);
1032 TerminalSend(
"\033[?25l");
1034 const bool resized =
1035 frame_count_ == 0 || (dimx != public_->dimx_) || (dimy != public_->dimy_);
1036 TerminalSend(ResetCursorPosition());
1038 if (frame_count_ != 0) {
1041 public_->ResetPosition(output_buffer, resized);
1045 if ((dimx < public_->dimx_) && !use_alternative_screen_) {
1046 TerminalSend(
"\033[J");
1047 TerminalSend(
"\033[H");
1053 public_->dimx_ = dimx;
1054 public_->dimy_ = dimy;
1055 public_->cells_ = std::vector<Cell>(
static_cast<size_t>(dimx) *
1056 static_cast<size_t>(dimy));
1057 Cursor cursor = public_->cursor_;
1058 cursor.x = dimx - 1;
1059 cursor.y = dimy - 1;
1060 public_->SetCursor(cursor);
1066 if (!use_alternative_screen_ && is_stdout_a_tty_) {
1067 RequestCursorPosition(previous_frame_resized_);
1069 previous_frame_resized_ = resized;
1071 selection_ = selection_data_.empty
1072 ? std::make_unique<Selection>()
1073 : std::make_unique<Selection>(
1074 selection_data_.start_x, selection_data_.start_y,
1075 selection_data_.end_x, selection_data_.end_y);
1076 Render(*public_, document.get(), *selection_);
1080 const int dx = public_->dimx_ - 1 - public_->cursor_.x +
1081 int(public_->dimx_ != terminal.dimx);
1082 const int dy = public_->dimy_ - 1 - public_->cursor_.y;
1084 set_cursor_position_.clear();
1085 reset_cursor_position_.clear();
1088 set_cursor_position_ +=
"\x1B[" + std::to_string(dy) +
"A";
1089 reset_cursor_position_ +=
"\x1B[" + std::to_string(dy) +
"B";
1093 set_cursor_position_ +=
"\x1B[" + std::to_string(dx) +
"D";
1094 reset_cursor_position_ +=
"\x1B[" + std::to_string(dx) +
"C";
1097 if (public_->cursor_.shape != Screen::Cursor::Hidden) {
1098 set_cursor_position_ +=
"\033[?25h";
1099 set_cursor_position_ +=
1100 "\033[" + std::to_string(
int(public_->cursor_.shape)) +
" q";
1104 public_->ToString(output_buffer);
1105 TerminalSend(set_cursor_position_);
1109 frame_valid_ =
true;
1113std::string App::Internal::ResetCursorPosition() {
1114 std::string result = std::move(reset_cursor_position_);
1115 reset_cursor_position_ =
"";
1119void App::Internal::RequestCursorPosition(
bool force) {
1120 cursor_position_request.Request(force);
1123void App::Internal::TerminalSend(std::string_view s) {
1127void App::Internal::TerminalFlush() {
1129 output_buffer +=
'\0';
1130 std::cout << output_buffer << std::flush;
1131 output_buffer.clear();
1134void App::Internal::InstallPipedInputHandling() {
1135 is_stdin_a_tty_ =
false;
1136 is_stdout_a_tty_ =
false;
1137#if defined(__EMSCRIPTEN__)
1138 is_stdin_a_tty_ =
true;
1139 is_stdout_a_tty_ =
true;
1140#elif defined(_WIN32)
1141 is_stdin_a_tty_ = _isatty(_fileno(stdin));
1142 is_stdout_a_tty_ = _isatty(_fileno(stdout));
1144 tty_fd_ = STDIN_FILENO;
1145 is_stdout_a_tty_ = isatty(STDOUT_FILENO);
1149 if (!handle_piped_input_) {
1150 is_stdin_a_tty_ = isatty(STDIN_FILENO);
1151 }
else if (isatty(STDIN_FILENO)) {
1152 is_stdin_a_tty_ =
true;
1155 tty_fd_ = open(
"/dev/tty", O_RDONLY);
1158 tty_fd_ = STDIN_FILENO;
1159 is_stdin_a_tty_ = isatty(STDIN_FILENO);
1161 is_stdin_a_tty_ =
true;
1163 on_exit_functions.emplace([
this] {
1172void App::Internal::InstallTerminalInfo() {
1175 if (is_stdout_a_tty_) {
1176 TerminalSend(DECRQSS_DECSCUSR);
1177 TerminalSend(
"\033[>q");
1178 TerminalSend(
"\033[>c");
1179 TerminalSend(
"\033[c");
1184 if (is_stdin_a_tty_ && is_stdout_a_tty_) {
1185 auto start = std::chrono::steady_clock::now();
1186 bool terminal_capabilities_received =
false;
1189 FetchTerminalEvents();
1190 while (setup_receiver->Has()) {
1191 const auto event = setup_receiver->Pop();
1192 if (event.is_cursor_shape()) {
1193 cursor_reset_shape_ =
event.cursor_shape();
1196 if (event.IsTerminalCapabilities()) {
1197 terminal_capabilities_ =
event.TerminalCapabilities();
1198 terminal_capabilities_received =
true;
1201 if (event.IsTerminalNameVersion()) {
1202 terminal_name_ =
event.TerminalName();
1203 terminal_version_ =
event.TerminalVersion();
1206 if (event.IsTerminalEmulator()) {
1207 terminal_emulator_name_ =
event.TerminalEmulatorName();
1208 terminal_emulator_version_ =
event.TerminalEmulatorVersion();
1215 if (terminal_capabilities_received) {
1219 if (std::chrono::steady_clock::now() - start >
1220 std::chrono::milliseconds(500)) {
1223 std::this_thread::sleep_for(std::chrono::milliseconds(10));
1230 auto safe_getenv = [](
const char* name) -> std::string_view {
1231 const char*
value = std::getenv(name);
1236 safe_getenv(
"TERM"), safe_getenv(
"COLORTERM"),
1237 safe_getenv(
"TERM_PROGRAM"), terminal_name_, terminal_emulator_name_,
1238 terminal_capabilities_);
1240 quirks.SetColorSupport(color_support);
1242 const bool is_modern_emulator = (terminal_emulator_name_ !=
"unknown");
1243 const bool is_vt220_plus =
1244 (terminal_name_ !=
"vt100" && terminal_name_ !=
"unknown");
1245 bool reports_utf8 =
false;
1246 for (
const int x : terminal_capabilities_) {
1248 reports_utf8 =
true;
1259 bool modern = is_modern_emulator || is_vt220_plus || reports_utf8;
1261 quirks.SetBlockCharacters(
true);
1262 quirks.SetCursorHiding(
true);
1263 quirks.SetComponentAscii(
false);
1268 on_exit_functions.emplace([
this] {
1269 TerminalSend(
"\033[?25h");
1270 if (is_stdout_a_tty_) {
1271 TerminalSend(
"\033[" + std::to_string(cursor_reset_shape_) +
" q");
1276void App::Internal::Signal(
int signal) {
1277 if (signal == SIGABRT) {
1284 if (signal == SIGTSTP) {
1286 TerminalSend(ResetCursorPosition());
1287 public_->ResetPosition(output_buffer,
true);
1291 (void)std::raise(SIGTSTP);
1297 if (signal == SIGWINCH) {
1298 public_->Post(Event::Special({0}));
1304size_t App::Internal::FetchTerminalEvents() {
1306 auto get_input_records = [&]() -> std::vector<INPUT_RECORD> {
1308 auto console = GetStdHandle(STD_INPUT_HANDLE);
1309 DWORD number_of_events = 0;
1310 if (!GetNumberOfConsoleInputEvents(console, &number_of_events)) {
1311 return std::vector<INPUT_RECORD>();
1313 if (number_of_events <= 0) {
1315 return std::vector<INPUT_RECORD>();
1318 std::vector<INPUT_RECORD> records(number_of_events);
1319 DWORD number_of_events_read = 0;
1320 if (!ReadConsoleInput(console, records.data(), (DWORD)records.size(),
1321 &number_of_events_read)) {
1322 return std::vector<INPUT_RECORD>();
1324 records.resize(number_of_events_read);
1328 auto records = get_input_records();
1329 if (records.size() == 0) {
1330 const auto timeout = std::chrono::steady_clock::now() - last_char_time;
1331 const size_t timeout_microseconds =
1332 std::chrono::duration_cast<std::chrono::microseconds>(timeout).count();
1333 terminal_input_parser.Timeout(timeout_microseconds);
1336 last_char_time = std::chrono::steady_clock::now();
1341 std::wstring wstring;
1342 for (
const auto& r : records) {
1343 switch (r.EventType) {
1345 auto key_event = r.Event.KeyEvent;
1347 if (key_event.bKeyDown == FALSE) {
1350 const wchar_t wc = key_event.uChar.UnicodeChar;
1352 if (wc >= 0xd800 && wc <= 0xdbff) {
1357 terminal_input_parser.Add(it);
1361 case WINDOW_BUFFER_SIZE_EVENT:
1362 public_->Post(Event::Special({0}));
1371 return records.size();
1372#elif defined(__EMSCRIPTEN__)
1375 std::array<char, 128> out{};
1376 const ssize_t l = read(STDIN_FILENO, out.data(), out.size());
1378 const auto timeout = std::chrono::steady_clock::now() - last_char_time;
1379 const size_t timeout_microseconds =
1380 std::chrono::duration_cast<std::chrono::microseconds>(timeout).count();
1381 terminal_input_parser.Timeout(timeout_microseconds);
1384 last_char_time = std::chrono::steady_clock::now();
1387 for (ssize_t i = 0; i < l; ++i) {
1388 terminal_input_parser.Add(out.at(
static_cast<size_t>(i)));
1392 struct pollfd pfd = {tty_fd_, POLLIN, 0};
1393 const int poll_result = poll(&pfd, 1, 0);
1394 if (poll_result <= 0) {
1395 const auto timeout = std::chrono::steady_clock::now() - last_char_time;
1396 const size_t timeout_ms =
1397 std::chrono::duration_cast<std::chrono::milliseconds>(timeout).count();
1398 terminal_input_parser.Timeout(
static_cast<int>(timeout_ms));
1401 last_char_time = std::chrono::steady_clock::now();
1404 std::array<char, 128> out{};
1405 const ssize_t l = read(tty_fd_, out.data(), out.size());
1411 for (ssize_t i = 0; i < l; ++i) {
1412 terminal_input_parser.Add(out.at(
static_cast<size_t>(i)));
1418void App::Internal::PostAnimationTask() {
1419 public_->Post(AnimationTask());
1423 task_runner.PostDelayedTask([
this] { PostAnimationTask(); },
1424 std::chrono::milliseconds(15));
1427App::App(std::unique_ptr<Internal> internal,
int dimx,
int dimy)
1428 : Screen(dimx, dimy), internal_(std::move(internal)) {
1429 internal_->public_ =
this;
1432App::App(App&& other) noexcept : Screen(std::move(other)) {
1433 internal_ = std::move(other.internal_);
1435 internal_->public_ =
this;
1439App& App::operator=(App&& other)
noexcept {
1440 Screen::operator=(std::move(other));
1441 internal_ = std::move(other.internal_);
1443 internal_->public_ =
this;
1448App::~App() =
default;
1451App App::FixedSize(
int dimx,
int dimy) {
1454 return App(std::move(internal), dimx, dimy);
1458App App::Fullscreen() {
1459 return FullscreenAlternateScreen();
1463App App::FullscreenPrimaryScreen() {
1467 return App(std::move(internal), terminal.dimx, terminal.dimy);
1471App App::FullscreenAlternateScreen() {
1475 return App(std::move(internal), terminal.dimx, terminal.dimy);
1479App App::FitComponent() {
1483 return App(std::move(internal), terminal.dimx, terminal.dimy);
1487App App::TerminalOutput() {
1491 return App(std::move(internal), terminal.dimx, terminal.dimy);
1494void App::TrackMouse(
bool enable) {
1495 internal_->track_mouse_ = enable;
1498void App::HandlePipedInput(
bool enable) {
1499 internal_->handle_piped_input_ = enable;
1504 return g_active_screen;
1508 class Loop loop(this, std::move(component));
1513 Post([
this] { internal_->ExitNow(); });
1516Closure App::ExitLoopClosure() {
1517 return [
this] { Exit(); };
1520void App::Post(
Task task) {
1521 internal_->task_runner.PostTask([
this, task = std::move(task)]()
mutable {
1522 if (internal_->component_) {
1523 internal_->HandleTask(internal_->component_, task);
1528 if (std::holds_alternative<Closure>(task)) {
1529 std::get<Closure>(task)();
1534void App::PostEvent(Event event) {
1535 internal_->event_buffer.Push(std::move(event));
1536 RequestAnimationFrame();
1540void App::PostEventOrExecute(
Closure closure) {
1544 if (
auto* app = App::Active()) {
1545 app->Post(std::move(closure));
1551void App::RequestAnimationFrame() {
1552 if (internal_->animation_requested_) {
1555 internal_->animation_requested_ =
true;
1556 auto now = animation::Clock::now();
1557 const auto time_histeresis = std::chrono::milliseconds(33);
1558 if (now - internal_->previous_animation_time_ >= time_histeresis) {
1559 internal_->previous_animation_time_ = now;
1564 if (internal_->mouse_captured) {
1567 internal_->mouse_captured =
true;
1568 return std::make_unique<CapturedMouseImpl>(
1569 [
this] { internal_->mouse_captured =
false; });
1574 internal_->Uninstall();
1576 internal_->Install();
1580void App::ForceHandleCtrlC(
bool force) {
1581 internal_->force_handle_ctrl_c_ = force;
1584void App::ForceHandleCtrlZ(
bool force) {
1585 internal_->force_handle_ctrl_z_ = force;
1588std::string App::GetSelection() {
1589 if (!internal_->selection_) {
1592 return internal_->selection_->GetParts();
1595void App::SelectionChange(std::function<
void()> callback) {
1596 internal_->selection_on_change_ = std::move(callback);
1599const std::string& App::TerminalName()
const {
1600 return internal_->terminal_name_;
1603int App::TerminalVersion()
const {
1604 return internal_->terminal_version_;
1607const std::string& App::TerminalEmulatorName()
const {
1608 return internal_->terminal_emulator_name_;
1611const std::string& App::TerminalEmulatorVersion()
const {
1612 return internal_->terminal_emulator_version_;
1615const std::vector<int>& App::TerminalCapabilities()
const {
1616 return internal_->terminal_capabilities_;
1619std::vector<std::string> App::TerminalCapabilityNames()
const {
1620 return Event::TerminalCapabilities(
"", internal_->terminal_capabilities_)
1621 .TerminalCapabilityNames();
1626void App::ExitNow() {
1627 internal_->ExitNow();
1629void App::Install() {
1630 internal_->Install();
1632void App::Uninstall() {
1633 internal_->Uninstall();
1635void App::PreMain() {
1636 internal_->PreMain();
1638void App::PostMain() {
1639 internal_->PostMain();
1641bool App::HasQuitted() {
1642 return internal_->HasQuitted();
1644void App::RunOnce(
const Component& component) {
1645 internal_->RunOnce(component);
1647void App::RunOnceBlocking(
Component component) {
1648 internal_->RunOnceBlocking(component);
1651 internal_->HandleTask(component, task);
1653bool App::HandleSelection(
bool handled, Event event) {
1654 return internal_->HandleSelection(handled, event);
1657 internal_->Draw(component);
1659std::string App::ResetCursorPosition() {
1660 return internal_->ResetCursorPosition();
1662void App::RequestCursorPosition(
bool force) {
1663 internal_->RequestCursorPosition(force);
1665void App::TerminalSend(std::string_view s) {
1666 internal_->TerminalSend(s);
1668void App::TerminalFlush() {
1669 internal_->TerminalFlush();
1671void App::InstallPipedInputHandling() {
1672 internal_->InstallPipedInputHandling();
1674void App::InstallTerminalInfo() {
1675 internal_->InstallTerminalInfo();
1677void App::Signal(
int signal) {
1678 internal_->Signal(signal);
1680size_t App::FetchTerminalEvents() {
1681 return internal_->FetchTerminalEvents();
1683void App::PostAnimationTask() {
1684 internal_->PostAnimationTask();
1687Loop::Loop(App* screen,
Component component)
1688 : screen_(screen), component_(std::move(component)) {
1693 screen_->PostMain();
1696bool Loop::HasQuitted() {
1697 return screen_->HasQuitted();
1700void Loop::RunOnce() {
1701 screen_->RunOnce(component_);
1704void Loop::RunOnceBlocking() {
1705 screen_->RunOnceBlocking(component_);
1709 while (!HasQuitted()) {
auto PostDelayedTask(Task task, std::chrono::steady_clock::duration duration) -> void
Schedules a task to be executed after a certain duration.
Quirks GetQuirks()
Get the terminal quirks.
Dimensions Size()
Get the terminal size.
void SetQuirks(const Quirks &quirks)
Override terminal quirks.
The FTXUI ftxui::animation:: namespace.
void SetFallbackSize(const Dimensions &fallbackSize)
Override terminal size in case auto-detection fails.
Color ComputeColorSupport(std::string_view term, std::string_view colorterm, std::string_view term_program, std::string_view terminal_name, std::string_view terminal_emulator_name, const std::vector< int > &capabilities)
Compute the color support based on environment variables and terminal identification.
std::chrono::duration< float > Duration
std::chrono::time_point< Clock > TimePoint
void RequestAnimationFrame()
constexpr const T & clamp(const T &v, const T &lo, const T &hi)
The FTXUI ftxui:: namespace.
std::unique_ptr< CapturedMouseInterface > CapturedMouse
std::string to_string(std::wstring_view s)
Convert a std::wstring into a UTF8 std::string.
std::variant< Event, Closure, AnimationTask > Task
void Render(Screen &screen, Node *node, Selection &selection)
std::function< void()> Closure
std::shared_ptr< ComponentBase > Component