| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org> | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 1. Redistributions of source code must retain the above copyright notice, this | 
					
						
							|  |  |  |  *    list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | 
					
						
							|  |  |  |  *    this list of conditions and the following disclaimer in the documentation | 
					
						
							|  |  |  |  *    and/or other materials provided with the distribution. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
					
						
							|  |  |  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
					
						
							|  |  |  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
					
						
							|  |  |  |  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
					
						
							|  |  |  |  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
					
						
							|  |  |  |  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
					
						
							|  |  |  |  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
					
						
							|  |  |  |  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
					
						
							|  |  |  |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:57 +02:00
										 |  |  | #include "Editor.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:24:10 +02:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | #include <ctype.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <sys/ioctl.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | namespace Line { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Editor::Editor() | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     struct winsize ws; | 
					
						
							|  |  |  |     if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) < 0) | 
					
						
							|  |  |  |         m_num_columns = 80; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         m_num_columns = ws.ws_col; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | Editor::~Editor() | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-30 22:42:50 +04:30
										 |  |  |     tcsetattr(0, TCSANOW, &m_default_termios); | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::add_to_history(const String& line) | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     if ((m_history.size() + 1) > m_history_capacity) | 
					
						
							|  |  |  |         m_history.take_first(); | 
					
						
							|  |  |  |     m_history.append(line); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::clear_line() | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     for (size_t i = 0; i < m_cursor; ++i) | 
					
						
							|  |  |  |         fputc(0x8, stdout); | 
					
						
							|  |  |  |     fputs("\033[K", stdout); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  |     m_buffer.clear(); | 
					
						
							|  |  |  |     m_cursor = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::insert(const String& string) | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     fputs(string.characters(), stdout); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_cursor == m_buffer.size()) { | 
					
						
							|  |  |  |         m_buffer.append(string.characters(), string.length()); | 
					
						
							|  |  |  |         m_cursor = m_buffer.size(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vt_save_cursor(); | 
					
						
							|  |  |  |     vt_clear_to_end_of_line(); | 
					
						
							|  |  |  |     for (size_t i = m_cursor; i < m_buffer.size(); ++i) | 
					
						
							|  |  |  |         fputc(m_buffer[i], stdout); | 
					
						
							|  |  |  |     vt_restore_cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_buffer.ensure_capacity(m_buffer.size() + string.length()); | 
					
						
							|  |  |  |     for (size_t i = 0; i < string.length(); ++i) | 
					
						
							|  |  |  |         m_buffer.insert(m_cursor + i, string[i]); | 
					
						
							|  |  |  |     m_cursor += string.length(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::insert(const char ch) | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     putchar(ch); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_cursor == m_buffer.size()) { | 
					
						
							|  |  |  |         m_buffer.append(ch); | 
					
						
							|  |  |  |         m_cursor = m_buffer.size(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vt_save_cursor(); | 
					
						
							|  |  |  |     vt_clear_to_end_of_line(); | 
					
						
							|  |  |  |     for (size_t i = m_cursor; i < m_buffer.size(); ++i) | 
					
						
							|  |  |  |         fputc(m_buffer[i], stdout); | 
					
						
							|  |  |  |     vt_restore_cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_buffer.insert(m_cursor, ch); | 
					
						
							|  |  |  |     ++m_cursor; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:37:01 +02:00
										 |  |  | void Editor::register_character_input_callback(char ch, Function<bool(Editor&)> callback) | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     if (m_key_callbacks.contains(ch)) { | 
					
						
							|  |  |  |         dbg() << "Key callback registered twice for " << ch; | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     m_key_callbacks.set(ch, make<KeyCallback>(move(callback))); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::cut_mismatching_chars(String& completion, const String& other, size_t start_compare) | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     size_t i = start_compare; | 
					
						
							|  |  |  |     while (i < completion.length() && i < other.length() && completion[i] == other[i]) | 
					
						
							|  |  |  |         ++i; | 
					
						
							|  |  |  |     completion = completion.substring(0, i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | String Editor::get_line(const String& prompt) | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     fputs(prompt.characters(), stdout); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_history_cursor = m_history.size(); | 
					
						
							|  |  |  |     m_cursor = 0; | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							|  |  |  |         char keybuf[16]; | 
					
						
							|  |  |  |         ssize_t nread = read(0, keybuf, sizeof(keybuf)); | 
					
						
							|  |  |  |         // FIXME: exit()ing here is a bit off. Should communicate failure to caller somehow instead.
 | 
					
						
							|  |  |  |         if (nread == 0) | 
					
						
							|  |  |  |             exit(0); | 
					
						
							|  |  |  |         if (nread < 0) { | 
					
						
							|  |  |  |             if (errno == EINTR) { | 
					
						
							|  |  |  |                 if (m_was_interrupted) { | 
					
						
							|  |  |  |                     m_was_interrupted = false; | 
					
						
							|  |  |  |                     if (!m_buffer.is_empty()) | 
					
						
							|  |  |  |                         printf("^C"); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (m_was_resized) { | 
					
						
							|  |  |  |                     m_was_resized = false; | 
					
						
							|  |  |  |                     printf("\033[2K\r"); | 
					
						
							|  |  |  |                     m_buffer.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     struct winsize ws; | 
					
						
							|  |  |  |                     int rc = ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws); | 
					
						
							|  |  |  |                     ASSERT(rc == 0); | 
					
						
							|  |  |  |                     m_num_columns = ws.ws_col; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     return String::empty(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 m_buffer.clear(); | 
					
						
							|  |  |  |                 putchar('\n'); | 
					
						
							|  |  |  |                 return String::empty(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             perror("read failed"); | 
					
						
							|  |  |  |             // FIXME: exit()ing here is a bit off. Should communicate failure to caller somehow instead.
 | 
					
						
							|  |  |  |             exit(2); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto do_delete = [&] { | 
					
						
							|  |  |  |             if (m_cursor == m_buffer.size()) { | 
					
						
							|  |  |  |                 fputc('\a', stdout); | 
					
						
							|  |  |  |                 fflush(stdout); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             m_buffer.remove(m_cursor); | 
					
						
							|  |  |  |             fputs("\033[3~", stdout); | 
					
						
							|  |  |  |             fflush(stdout); | 
					
						
							|  |  |  |             vt_save_cursor(); | 
					
						
							|  |  |  |             vt_clear_to_end_of_line(); | 
					
						
							|  |  |  |             for (size_t i = m_cursor; i < m_buffer.size(); ++i) | 
					
						
							|  |  |  |                 fputc(m_buffer[i], stdout); | 
					
						
							|  |  |  |             vt_restore_cursor(); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (ssize_t i = 0; i < nread; ++i) { | 
					
						
							|  |  |  |             char ch = keybuf[i]; | 
					
						
							|  |  |  |             if (ch == 0) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (m_state) { | 
					
						
							|  |  |  |             case InputState::ExpectBracket: | 
					
						
							|  |  |  |                 if (ch == '[') { | 
					
						
							|  |  |  |                     m_state = InputState::ExpectFinal; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             case InputState::ExpectFinal: | 
					
						
							|  |  |  |                 switch (ch) { | 
					
						
							|  |  |  |                 case 'A': // up
 | 
					
						
							|  |  |  |                     if (m_history_cursor > 0) | 
					
						
							|  |  |  |                         --m_history_cursor; | 
					
						
							|  |  |  |                     clear_line(); | 
					
						
							|  |  |  |                     if (m_history_cursor < m_history.size()) | 
					
						
							|  |  |  |                         insert(m_history[m_history_cursor]); | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 case 'B': // down
 | 
					
						
							|  |  |  |                     if (m_history_cursor < m_history.size()) | 
					
						
							|  |  |  |                         ++m_history_cursor; | 
					
						
							|  |  |  |                     clear_line(); | 
					
						
							|  |  |  |                     if (m_history_cursor < m_history.size()) | 
					
						
							|  |  |  |                         insert(m_history[m_history_cursor]); | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 case 'D': // left
 | 
					
						
							|  |  |  |                     if (m_cursor > 0) { | 
					
						
							|  |  |  |                         --m_cursor; | 
					
						
							|  |  |  |                         fputs("\033[D", stdout); | 
					
						
							|  |  |  |                         fflush(stdout); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 case 'C': // right
 | 
					
						
							|  |  |  |                     if (m_cursor < m_buffer.size()) { | 
					
						
							|  |  |  |                         ++m_cursor; | 
					
						
							|  |  |  |                         fputs("\033[C", stdout); | 
					
						
							|  |  |  |                         fflush(stdout); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 case 'H': | 
					
						
							|  |  |  |                     if (m_cursor > 0) { | 
					
						
							|  |  |  |                         fprintf(stdout, "\033[%zuD", m_cursor); | 
					
						
							|  |  |  |                         fflush(stdout); | 
					
						
							|  |  |  |                         m_cursor = 0; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 case 'F': | 
					
						
							|  |  |  |                     if (m_cursor < m_buffer.size()) { | 
					
						
							|  |  |  |                         fprintf(stdout, "\033[%zuC", m_buffer.size() - m_cursor); | 
					
						
							|  |  |  |                         fflush(stdout); | 
					
						
							|  |  |  |                         m_cursor = m_buffer.size(); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 case '3': | 
					
						
							|  |  |  |                     do_delete(); | 
					
						
							|  |  |  |                     m_state = InputState::ExpectTerminator; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 default: | 
					
						
							| 
									
										
										
										
											2020-03-31 18:59:15 +02:00
										 |  |  |                     dbgprintf("Shell: Unhandled final: %02x (%c)\n", ch, ch); | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  |                     m_state = InputState::Free; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case InputState::ExpectTerminator: | 
					
						
							|  |  |  |                 m_state = InputState::Free; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             case InputState::Free: | 
					
						
							|  |  |  |                 if (ch == 27) { | 
					
						
							|  |  |  |                     m_state = InputState::ExpectBracket; | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto cb = m_key_callbacks.get(ch); | 
					
						
							|  |  |  |             if (cb.has_value()) { | 
					
						
							|  |  |  |                 if (!cb.value()->callback(*this)) { | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (ch == '\t') { | 
					
						
							|  |  |  |                 if (!on_tab_complete_first_token || !on_tab_complete_other_token) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 bool is_empty_token = m_cursor == 0 || m_buffer[m_cursor - 1] == ' '; | 
					
						
							|  |  |  |                 m_times_tab_pressed++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 int token_start = m_cursor - 1; | 
					
						
							|  |  |  |                 if (!is_empty_token) { | 
					
						
							|  |  |  |                     while (token_start >= 0 && m_buffer[token_start] != ' ') | 
					
						
							|  |  |  |                         --token_start; | 
					
						
							|  |  |  |                     ++token_start; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 bool is_first_token = true; | 
					
						
							|  |  |  |                 for (int i = token_start - 1; i >= 0; --i) { | 
					
						
							|  |  |  |                     if (m_buffer[i] != ' ') { | 
					
						
							|  |  |  |                         is_first_token = false; | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 String token = is_empty_token ? String() : String(&m_buffer[token_start], m_cursor - token_start); | 
					
						
							|  |  |  |                 Vector<String> suggestions; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (is_first_token) | 
					
						
							|  |  |  |                     suggestions = on_tab_complete_first_token(token); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     suggestions = on_tab_complete_other_token(token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (m_times_tab_pressed > 1 && !suggestions.is_empty()) { | 
					
						
							|  |  |  |                     size_t longest_suggestion_length = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     for (auto& suggestion : suggestions) | 
					
						
							|  |  |  |                         longest_suggestion_length = max(longest_suggestion_length, suggestion.length()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     size_t num_printed = 0; | 
					
						
							|  |  |  |                     putchar('\n'); | 
					
						
							|  |  |  |                     for (auto& suggestion : suggestions) { | 
					
						
							|  |  |  |                         size_t next_column = num_printed + suggestion.length() + longest_suggestion_length + 2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (next_column > m_num_columns) { | 
					
						
							|  |  |  |                             putchar('\n'); | 
					
						
							|  |  |  |                             num_printed = 0; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         num_printed += fprintf(stderr, "%-*s", static_cast<int>(longest_suggestion_length) + 2, suggestion.characters()); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 11:24:10 +02:00
										 |  |  |                     StringBuilder builder; | 
					
						
							|  |  |  |                     builder.append('\n'); | 
					
						
							|  |  |  |                     builder.append(prompt); | 
					
						
							|  |  |  |                     builder.append(m_buffer.data(), m_cursor); | 
					
						
							|  |  |  |                     builder.append(m_buffer.data() + m_cursor, m_buffer.size() - m_cursor); | 
					
						
							|  |  |  |                     fputs(builder.to_string().characters(), stdout); | 
					
						
							|  |  |  |                     fflush(stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 18:59:15 +02:00
										 |  |  |                     m_cursor = m_buffer.size(); | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 suggestions.clear_with_capacity(); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             m_times_tab_pressed = 0; // Safe to say if we get here, the user didn't press TAB
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto do_backspace = [&] { | 
					
						
							|  |  |  |                 if (m_cursor == 0) { | 
					
						
							|  |  |  |                     fputc('\a', stdout); | 
					
						
							|  |  |  |                     fflush(stdout); | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 m_buffer.remove(m_cursor - 1); | 
					
						
							|  |  |  |                 --m_cursor; | 
					
						
							|  |  |  |                 putchar(8); | 
					
						
							|  |  |  |                 vt_save_cursor(); | 
					
						
							|  |  |  |                 vt_clear_to_end_of_line(); | 
					
						
							|  |  |  |                 for (size_t i = m_cursor; i < m_buffer.size(); ++i) | 
					
						
							|  |  |  |                     fputc(m_buffer[i], stdout); | 
					
						
							|  |  |  |                 vt_restore_cursor(); | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (ch == 8 || ch == m_termios.c_cc[VERASE]) { | 
					
						
							|  |  |  |                 do_backspace(); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (ch == m_termios.c_cc[VWERASE]) { | 
					
						
							|  |  |  |                 bool has_seen_nonspace = false; | 
					
						
							|  |  |  |                 while (m_cursor > 0) { | 
					
						
							|  |  |  |                     if (isspace(m_buffer[m_cursor - 1])) { | 
					
						
							|  |  |  |                         if (has_seen_nonspace) | 
					
						
							|  |  |  |                             break; | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         has_seen_nonspace = true; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     do_backspace(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (ch == m_termios.c_cc[VKILL]) { | 
					
						
							|  |  |  |                 while (m_cursor > 0) | 
					
						
							|  |  |  |                     do_backspace(); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  |             if (ch == 0xc) {                    // ^L
 | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  |                 printf("\033[3J\033[H\033[2J"); // Clear screen.
 | 
					
						
							|  |  |  |                 fputs(prompt.characters(), stdout); | 
					
						
							|  |  |  |                 for (size_t i = 0; i < m_buffer.size(); ++i) | 
					
						
							|  |  |  |                     fputc(m_buffer[i], stdout); | 
					
						
							|  |  |  |                 if (m_cursor < m_buffer.size()) | 
					
						
							|  |  |  |                     printf("\033[%zuD", m_buffer.size() - m_cursor); // Move cursor N steps left.
 | 
					
						
							|  |  |  |                 fflush(stdout); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (ch == 0x01) { // ^A
 | 
					
						
							|  |  |  |                 if (m_cursor > 0) { | 
					
						
							|  |  |  |                     printf("\033[%zuD", m_cursor); | 
					
						
							|  |  |  |                     fflush(stdout); | 
					
						
							|  |  |  |                     m_cursor = 0; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (ch == m_termios.c_cc[VEOF]) { // Normally ^D
 | 
					
						
							|  |  |  |                 if (m_buffer.is_empty()) { | 
					
						
							|  |  |  |                     printf("<EOF>\n"); | 
					
						
							|  |  |  |                     exit(0); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (ch == 0x05) { // ^E
 | 
					
						
							|  |  |  |                 if (m_cursor < m_buffer.size()) { | 
					
						
							|  |  |  |                     printf("\033[%zuC", m_buffer.size() - m_cursor); | 
					
						
							|  |  |  |                     fflush(stdout); | 
					
						
							|  |  |  |                     m_cursor = m_buffer.size(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (ch == '\n') { | 
					
						
							|  |  |  |                 putchar('\n'); | 
					
						
							|  |  |  |                 fflush(stdout); | 
					
						
							|  |  |  |                 auto string = String::copy(m_buffer); | 
					
						
							|  |  |  |                 m_buffer.clear(); | 
					
						
							|  |  |  |                 return string; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             insert(ch); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::vt_save_cursor() | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     fputs("\033[s", stdout); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::vt_restore_cursor() | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     fputs("\033[u", stdout); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | void Editor::vt_clear_to_end_of_line() | 
					
						
							| 
									
										
										
										
											2020-03-30 21:29:04 +04:30
										 |  |  | { | 
					
						
							|  |  |  |     fputs("\033[K", stdout); | 
					
						
							|  |  |  |     fflush(stdout); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-31 13:34:06 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | } |