mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 13:02:06 +00:00

Applications using the Vim emulation engine now support line-wise text selection. We already have support for character-wise text selection, by pressing `v` from normal mode. However now can also trigger line-wise text selection by pressing `shift+v` from normal mode, and then using vertical motion commands (e.g. `j` or `k`) to expand the selection. This is a standard vim feature. In visual line mode the following operations are supported: * `escape`: back to normal mode * `u`: convert to lowercase * `U`: convert to uppercase * `~`: toggle case * `ctrl+d`: move down by 50% of page height * `ctrl+u`: move up by 50% of page height * `d` or `x`: delete selection * `c`: change selection * `y`: copy selection * `page up`: move up by 100% of page height * `page down`: move down by 100% of page height Notably I didn't implement pressing `v` to go to regular (character-wise) visual mode straight from visual line mode. This is tricky to implement in the current code base, and there's an alternative, which is to take a detour via normal mode.
205 lines
6.2 KiB
C++
205 lines
6.2 KiB
C++
/*
|
|
* Copyright (c) 2021, the SerenityOS developers.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/Optional.h>
|
|
#include <LibCore/Object.h>
|
|
#include <LibGUI/EditingEngine.h>
|
|
#include <LibGUI/TextRange.h>
|
|
|
|
namespace GUI {
|
|
|
|
// Wrapper over TextPosition that makes it easier to move it around as a cursor,
|
|
// and to get the current line or character.
|
|
class VimCursor {
|
|
public:
|
|
VimCursor(TextEditor& editor, TextPosition initial_position, bool forwards)
|
|
: m_editor(editor)
|
|
, m_position(initial_position)
|
|
, m_forwards(forwards)
|
|
{
|
|
}
|
|
|
|
void move_forwards();
|
|
void move_backwards();
|
|
|
|
// Move a single character in the current direction.
|
|
void move();
|
|
// Move a single character in reverse.
|
|
void move_reverse();
|
|
// Peek a single character in the current direction.
|
|
u32 peek();
|
|
// Peek a single character in reverse.
|
|
u32 peek_reverse();
|
|
// Get the character the cursor is currently on.
|
|
u32 current_char();
|
|
// Get the line the cursor is currently on.
|
|
TextDocumentLine& current_line();
|
|
// Get the current position.
|
|
TextPosition& current_position() { return m_position; }
|
|
|
|
// Did we hit the edge of the document?
|
|
bool hit_edge() { return m_hit_edge; }
|
|
// Will the next move cross a line boundary?
|
|
bool will_cross_line_boundary();
|
|
// Did we cross a line boundary?
|
|
bool crossed_line_boundary() { return m_crossed_line_boundary; }
|
|
// Are we on an empty line?
|
|
bool on_empty_line();
|
|
// Are we going forwards?
|
|
bool forwards() { return m_forwards; }
|
|
|
|
private:
|
|
TextEditor& m_editor;
|
|
TextPosition m_position;
|
|
bool m_forwards;
|
|
|
|
u32 m_cached_char { 0 };
|
|
|
|
bool m_hit_edge { false };
|
|
bool m_crossed_line_boundary { false };
|
|
};
|
|
|
|
class VimMotion {
|
|
public:
|
|
enum class Unit {
|
|
// The motion isn't complete yet, or was invalid.
|
|
Unknown,
|
|
// Document. Anything non-negative is counted as G while anything else is gg.
|
|
Document,
|
|
// Lines.
|
|
Line,
|
|
// A sequence of letters, digits and underscores, or a sequence of other
|
|
// non-blank characters separated by whitespace.
|
|
Word,
|
|
// A sequence of non-blank characters separated by whitespace.
|
|
// This is how Vim separates w from W.
|
|
WORD,
|
|
// End of a word. This is basically the same as a word but it doesn't
|
|
// trim the spaces at the end.
|
|
EndOfWord,
|
|
// End of a WORD.
|
|
EndOfWORD,
|
|
// Characters (or Unicode code points based on how pedantic you want to
|
|
// get).
|
|
Character,
|
|
// Used for find-mode.
|
|
Find
|
|
};
|
|
enum class FindMode {
|
|
/// Find mode is not enabled.
|
|
None,
|
|
/// Finding until the given character.
|
|
To,
|
|
/// Finding through the given character.
|
|
Find
|
|
};
|
|
|
|
void add_key_code(KeyCode key, bool ctrl, bool shift, bool alt);
|
|
Optional<TextRange> get_range(class VimEditingEngine& engine, bool normalize_for_position = false);
|
|
Optional<TextRange> get_repeat_range(class VimEditingEngine& engine, Unit, bool normalize_for_position = false);
|
|
Optional<TextPosition> get_position(VimEditingEngine& engine, bool in_visual_mode = false);
|
|
void reset();
|
|
|
|
/// Returns whether the motion should consume the next character no matter what.
|
|
/// Used for f and t motions.
|
|
bool should_consume_next_character() { return m_should_consume_next_character; }
|
|
bool is_complete() { return m_is_complete; }
|
|
bool is_cancelled() { return m_is_complete && m_unit == Unit::Unknown; }
|
|
Unit unit() { return m_unit; }
|
|
int amount() { return m_amount; }
|
|
|
|
// FIXME: come up with a better way to signal start/end of line than sentinels?
|
|
static constexpr int START_OF_LINE = NumericLimits<int>::min();
|
|
static constexpr int START_OF_NON_WHITESPACE = NumericLimits<int>::min() + 1;
|
|
static constexpr int END_OF_LINE = NumericLimits<int>::max();
|
|
|
|
private:
|
|
void calculate_document_range(TextEditor&);
|
|
void calculate_line_range(TextEditor&, bool normalize_for_position);
|
|
void calculate_word_range(VimCursor&, int amount, bool normalize_for_position);
|
|
void calculate_WORD_range(VimCursor&, int amount, bool normalize_for_position);
|
|
void calculate_character_range(VimCursor&, int amount, bool normalize_for_position);
|
|
void calculate_find_range(VimCursor&, int amount);
|
|
|
|
Unit m_unit { Unit::Unknown };
|
|
int m_amount { 0 };
|
|
bool m_is_complete { false };
|
|
bool m_guirky_mode { false };
|
|
bool m_should_consume_next_character { false };
|
|
|
|
FindMode m_find_mode { FindMode::None };
|
|
u32 m_next_character { 0 };
|
|
|
|
size_t m_start_line { 0 };
|
|
size_t m_start_column { 0 };
|
|
size_t m_end_line { 0 };
|
|
size_t m_end_column { 0 };
|
|
};
|
|
|
|
class VimEditingEngine final : public EditingEngine {
|
|
|
|
public:
|
|
virtual CursorWidth cursor_width() const override;
|
|
|
|
virtual bool on_key(KeyEvent const& event) override;
|
|
|
|
private:
|
|
enum VimMode {
|
|
Normal,
|
|
Insert,
|
|
Visual,
|
|
VisualLine
|
|
};
|
|
|
|
enum YankType {
|
|
Line,
|
|
Selection
|
|
};
|
|
|
|
enum class Casing {
|
|
Uppercase,
|
|
Lowercase,
|
|
Invertcase
|
|
};
|
|
|
|
VimMode m_vim_mode { VimMode::Normal };
|
|
VimMotion m_motion;
|
|
|
|
YankType m_yank_type {};
|
|
String m_yank_buffer {};
|
|
void yank(YankType);
|
|
void yank(TextRange, YankType yank_type);
|
|
void put_before();
|
|
void put_after();
|
|
|
|
TextPosition m_selection_start_position = {};
|
|
void update_selection_on_cursor_move();
|
|
void clamp_cursor_position();
|
|
void clear_visual_mode_data();
|
|
|
|
KeyCode m_previous_key {};
|
|
void switch_to_normal_mode();
|
|
void switch_to_insert_mode();
|
|
void switch_to_visual_mode();
|
|
void switch_to_visual_line_mode();
|
|
void move_half_page_up();
|
|
void move_half_page_down();
|
|
void move_to_previous_empty_lines_block();
|
|
void move_to_next_empty_lines_block();
|
|
|
|
bool on_key_in_insert_mode(KeyEvent const& event);
|
|
bool on_key_in_normal_mode(KeyEvent const& event);
|
|
bool on_key_in_visual_mode(KeyEvent const& event);
|
|
bool on_key_in_visual_line_mode(KeyEvent const& event);
|
|
|
|
void casefold_selection(Casing);
|
|
|
|
virtual EngineType engine_type() const override { return EngineType::Vim; }
|
|
};
|
|
|
|
}
|