123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511 |
- /*
- * Copyright (c) 2020, the SerenityOS developers.
- * 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.
- */
- #include <AK/StringBuilder.h>
- #include <LibLine/Editor.h>
- #include <ctype.h>
- #include <stdio.h>
- namespace {
- constexpr u32 ctrl(char c) { return c & 0x3f; }
- }
- namespace Line {
- Function<bool(Editor&)> Editor::find_internal_function(const StringView& name)
- {
- #define __ENUMERATE(internal_name) \
- if (name == #internal_name) \
- return EDITOR_INTERNAL_FUNCTION(internal_name);
- ENUMERATE_EDITOR_INTERNAL_FUNCTIONS(__ENUMERATE)
- return {};
- }
- void Editor::search_forwards()
- {
- ScopedValueRollback inline_search_cursor_rollback { m_inline_search_cursor };
- StringBuilder builder;
- builder.append(Utf32View { m_buffer.data(), m_inline_search_cursor });
- String search_phrase = builder.to_string();
- if (m_search_offset_state == SearchOffsetState::Backwards)
- --m_search_offset;
- if (m_search_offset > 0) {
- ScopedValueRollback search_offset_rollback { m_search_offset };
- --m_search_offset;
- if (search(search_phrase, true)) {
- m_search_offset_state = SearchOffsetState::Forwards;
- search_offset_rollback.set_override_rollback_value(m_search_offset);
- } else {
- m_search_offset_state = SearchOffsetState::Unbiased;
- }
- } else {
- m_search_offset_state = SearchOffsetState::Unbiased;
- m_chars_touched_in_the_middle = buffer().size();
- m_cursor = 0;
- m_buffer.clear();
- insert(search_phrase);
- m_refresh_needed = true;
- }
- }
- void Editor::search_backwards()
- {
- ScopedValueRollback inline_search_cursor_rollback { m_inline_search_cursor };
- StringBuilder builder;
- builder.append(Utf32View { m_buffer.data(), m_inline_search_cursor });
- String search_phrase = builder.to_string();
- if (m_search_offset_state == SearchOffsetState::Forwards)
- ++m_search_offset;
- if (search(search_phrase, true)) {
- m_search_offset_state = SearchOffsetState::Backwards;
- ++m_search_offset;
- } else {
- m_search_offset_state = SearchOffsetState::Unbiased;
- --m_search_offset;
- }
- }
- void Editor::cursor_left_word()
- {
- if (m_cursor > 0) {
- auto skipped_at_least_one_character = false;
- for (;;) {
- if (m_cursor == 0)
- break;
- if (skipped_at_least_one_character && !isalnum(m_buffer[m_cursor - 1])) // stop *after* a non-alnum, but only if it changes the position
- break;
- skipped_at_least_one_character = true;
- --m_cursor;
- }
- }
- m_inline_search_cursor = m_cursor;
- }
- void Editor::cursor_left_character()
- {
- if (m_cursor > 0)
- --m_cursor;
- m_inline_search_cursor = m_cursor;
- }
- void Editor::cursor_right_word()
- {
- if (m_cursor < m_buffer.size()) {
- // Temporarily put a space at the end of our buffer,
- // doing this greatly simplifies the logic below.
- m_buffer.append(' ');
- for (;;) {
- if (m_cursor >= m_buffer.size())
- break;
- if (!isalnum(m_buffer[++m_cursor]))
- break;
- }
- m_buffer.take_last();
- }
- m_inline_search_cursor = m_cursor;
- m_search_offset = 0;
- }
- void Editor::cursor_right_character()
- {
- if (m_cursor < m_buffer.size()) {
- ++m_cursor;
- }
- m_inline_search_cursor = m_cursor;
- m_search_offset = 0;
- }
- void Editor::erase_character_backwards()
- {
- if (m_is_searching) {
- return;
- }
- if (m_cursor == 0) {
- fputc('\a', stderr);
- fflush(stderr);
- return;
- }
- remove_at_index(m_cursor - 1);
- --m_cursor;
- m_inline_search_cursor = m_cursor;
- // We will have to redraw :(
- m_refresh_needed = true;
- }
- void Editor::erase_character_forwards()
- {
- if (m_cursor == m_buffer.size()) {
- fputc('\a', stderr);
- fflush(stderr);
- return;
- }
- remove_at_index(m_cursor);
- m_refresh_needed = true;
- }
- void Editor::finish_edit()
- {
- fprintf(stderr, "<EOF>\n");
- if (!m_always_refresh) {
- m_input_error = Error::Eof;
- finish();
- really_quit_event_loop();
- }
- }
- void Editor::kill_line()
- {
- for (size_t i = 0; i < m_cursor; ++i)
- remove_at_index(0);
- m_cursor = 0;
- m_refresh_needed = true;
- }
- void Editor::erase_word_backwards()
- {
- // A word here is space-separated. `foo=bar baz` is two words.
- 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;
- }
- erase_character_backwards();
- }
- }
- void Editor::erase_to_end()
- {
- while (m_cursor < m_buffer.size())
- erase_character_forwards();
- }
- void Editor::erase_to_beginning()
- {
- }
- void Editor::transpose_characters()
- {
- if (m_cursor > 0 && m_buffer.size() >= 2) {
- if (m_cursor < m_buffer.size())
- ++m_cursor;
- swap(m_buffer[m_cursor - 1], m_buffer[m_cursor - 2]);
- // FIXME: Update anchored styles too.
- m_refresh_needed = true;
- m_chars_touched_in_the_middle += 2;
- }
- }
- void Editor::enter_search()
- {
- if (m_is_searching) {
- // How did we get here?
- ASSERT_NOT_REACHED();
- } else {
- m_is_searching = true;
- m_search_offset = 0;
- m_pre_search_buffer.clear();
- for (auto code_point : m_buffer)
- m_pre_search_buffer.append(code_point);
- m_pre_search_cursor = m_cursor;
- // Disable our own notifier so as to avoid interfering with the search editor.
- m_notifier->set_enabled(false);
- m_search_editor = Editor::construct(Configuration { Configuration::Eager, Configuration::NoSignalHandlers }); // Has anyone seen 'Inception'?
- m_search_editor->initialize();
- add_child(*m_search_editor);
- m_search_editor->on_display_refresh = [this](Editor& search_editor) {
- StringBuilder builder;
- builder.append(Utf32View { search_editor.buffer().data(), search_editor.buffer().size() });
- if (!search(builder.build(), false, false)) {
- m_chars_touched_in_the_middle = m_buffer.size();
- m_refresh_needed = true;
- m_buffer.clear();
- m_cursor = 0;
- }
- refresh_display();
- };
- // Whenever the search editor gets a ^R, cycle between history entries.
- m_search_editor->register_key_input_callback(ctrl('R'), [this](Editor& search_editor) {
- ++m_search_offset;
- search_editor.m_refresh_needed = true;
- return false; // Do not process this key event
- });
- // ^C should cancel the search.
- m_search_editor->register_key_input_callback(ctrl('C'), [this](Editor& search_editor) {
- search_editor.finish();
- m_reset_buffer_on_search_end = true;
- search_editor.deferred_invoke([&search_editor](auto&) { search_editor.really_quit_event_loop(); });
- return false;
- });
- // Whenever the search editor gets a backspace, cycle back between history entries
- // unless we're at the zeroth entry, in which case, allow the deletion.
- m_search_editor->register_key_input_callback(m_termios.c_cc[VERASE], [this](Editor& search_editor) {
- if (m_search_offset > 0) {
- --m_search_offset;
- search_editor.m_refresh_needed = true;
- return false; // Do not process this key event
- }
- search_editor.erase_character_backwards();
- return false;
- });
- // ^L - This is a source of issues, as the search editor refreshes first,
- // and we end up with the wrong order of prompts, so we will first refresh
- // ourselves, then refresh the search editor, and then tell him not to process
- // this event.
- m_search_editor->register_key_input_callback(ctrl('L'), [this](auto& search_editor) {
- fprintf(stderr, "\033[3J\033[H\033[2J"); // Clear screen.
- // refresh our own prompt
- set_origin(1, 1);
- m_refresh_needed = true;
- refresh_display();
- // move the search prompt below ours
- // and tell it to redraw itself
- auto prompt_end_line = current_prompt_metrics().lines_with_addition(m_cached_buffer_metrics, m_num_columns);
- search_editor.set_origin(prompt_end_line + 1, 1);
- search_editor.m_refresh_needed = true;
- return false;
- });
- // quit without clearing the current buffer
- m_search_editor->register_key_input_callback('\t', [this](Editor& search_editor) {
- search_editor.finish();
- m_reset_buffer_on_search_end = false;
- return false;
- });
- fprintf(stderr, "\n");
- fflush(stderr);
- auto search_prompt = "\x1b[32msearch:\x1b[0m ";
- // While the search editor is active, we do not want editing events.
- m_is_editing = false;
- auto search_string_result = m_search_editor->get_line(search_prompt);
- // Grab where the search origin last was, anything up to this point will be cleared.
- auto search_end_row = m_search_editor->m_origin_row;
- remove_child(*m_search_editor);
- m_search_editor = nullptr;
- m_is_searching = false;
- m_is_editing = true;
- m_search_offset = 0;
- // Re-enable the notifier after discarding the search editor.
- m_notifier->set_enabled(true);
- if (search_string_result.is_error()) {
- // Somethine broke, fail
- m_input_error = search_string_result.error();
- finish();
- return;
- }
- auto& search_string = search_string_result.value();
- // Manually cleanup the search line.
- reposition_cursor();
- auto search_metrics = actual_rendered_string_metrics(search_string);
- auto metrics = actual_rendered_string_metrics(search_prompt);
- VT::clear_lines(0, metrics.lines_with_addition(search_metrics, m_num_columns) + search_end_row - m_origin_row - 1);
- reposition_cursor();
- if (!m_reset_buffer_on_search_end || search_metrics.total_length == 0) {
- // If the entry was empty, or we purposely quit without a newline,
- // do not return anything; instead, just end the search.
- end_search();
- return;
- }
- // Return the string,
- finish();
- }
- }
- void Editor::transpose_words()
- {
- // A word here is contiguous alnums. `foo=bar baz` is three words.
- // 'abcd,.:efg...' should become 'efg...,.:abcd' if caret is after
- // 'efg...'. If it's in 'efg', it should become 'efg,.:abcd...'
- // with the caret after it, which then becomes 'abcd...,.:efg'
- // when alt-t is pressed a second time.
- // Move to end of word under (or after) caret.
- size_t cursor = m_cursor;
- while (cursor < m_buffer.size() && !isalnum(m_buffer[cursor]))
- ++cursor;
- while (cursor < m_buffer.size() && isalnum(m_buffer[cursor]))
- ++cursor;
- // Move left over second word and the space to its right.
- size_t end = cursor;
- size_t start = cursor;
- while (start > 0 && !isalnum(m_buffer[start - 1]))
- --start;
- while (start > 0 && isalnum(m_buffer[start - 1]))
- --start;
- size_t start_second_word = start;
- // Move left over space between the two words.
- while (start > 0 && !isalnum(m_buffer[start - 1]))
- --start;
- size_t start_gap = start;
- // Move left over first word.
- while (start > 0 && isalnum(m_buffer[start - 1]))
- --start;
- if (start != start_gap) {
- // To swap the two words, swap each word (and the gap) individually, and then swap the whole range.
- auto swap_range = [this](auto from, auto to) {
- for (size_t i = 0; i < (to - from) / 2; ++i)
- swap(m_buffer[from + i], m_buffer[to - 1 - i]);
- };
- swap_range(start, start_gap);
- swap_range(start_gap, start_second_word);
- swap_range(start_second_word, end);
- swap_range(start, end);
- m_cursor = cursor;
- // FIXME: Update anchored styles too.
- m_refresh_needed = true;
- m_chars_touched_in_the_middle += end - start;
- }
- }
- void Editor::go_home()
- {
- m_cursor = 0;
- m_inline_search_cursor = m_cursor;
- m_search_offset = 0;
- }
- void Editor::go_end()
- {
- m_cursor = m_buffer.size();
- m_inline_search_cursor = m_cursor;
- m_search_offset = 0;
- }
- void Editor::clear_screen()
- {
- fprintf(stderr, "\033[3J\033[H\033[2J"); // Clear screen.
- VT::move_absolute(1, 1);
- set_origin(1, 1);
- m_refresh_needed = true;
- m_cached_prompt_valid = false;
- }
- void Editor::insert_last_words()
- {
- if (!m_history.is_empty()) {
- // FIXME: This isn't quite right: if the last arg was `"foo bar"` or `foo\ bar` (but not `foo\\ bar`), we should insert that whole arg as last token.
- if (auto last_words = m_history.last().entry.split_view(' '); !last_words.is_empty())
- insert(last_words.last());
- }
- }
- void Editor::erase_alnum_word_backwards()
- {
- // A word here is contiguous alnums. `foo=bar baz` is three words.
- bool has_seen_alnum = false;
- while (m_cursor > 0) {
- if (!isalnum(m_buffer[m_cursor - 1])) {
- if (has_seen_alnum)
- break;
- } else {
- has_seen_alnum = true;
- }
- erase_character_backwards();
- }
- }
- void Editor::erase_alnum_word_forwards()
- {
- // A word here is contiguous alnums. `foo=bar baz` is three words.
- bool has_seen_alnum = false;
- while (m_cursor < m_buffer.size()) {
- if (!isalnum(m_buffer[m_cursor])) {
- if (has_seen_alnum)
- break;
- } else {
- has_seen_alnum = true;
- }
- erase_character_forwards();
- }
- }
- void Editor::case_change_word(Editor::CaseChangeOp change_op)
- {
- // A word here is contiguous alnums. `foo=bar baz` is three words.
- while (m_cursor < m_buffer.size() && !isalnum(m_buffer[m_cursor]))
- ++m_cursor;
- size_t start = m_cursor;
- while (m_cursor < m_buffer.size() && isalnum(m_buffer[m_cursor])) {
- if (change_op == CaseChangeOp::Uppercase || (change_op == CaseChangeOp::Capital && m_cursor == start)) {
- m_buffer[m_cursor] = toupper(m_buffer[m_cursor]);
- } else {
- ASSERT(change_op == CaseChangeOp::Lowercase || (change_op == CaseChangeOp::Capital && m_cursor > start));
- m_buffer[m_cursor] = tolower(m_buffer[m_cursor]);
- }
- ++m_cursor;
- m_refresh_needed = true;
- }
- }
- void Editor::capitalize_word()
- {
- case_change_word(CaseChangeOp::Capital);
- }
- void Editor::lowercase_word()
- {
- case_change_word(CaseChangeOp::Lowercase);
- }
- void Editor::uppercase_word()
- {
- case_change_word(CaseChangeOp::Uppercase);
- }
- }
|