Merge pull request #66553 from PucklaJ/duplicate_lines
Add Duplicate Lines shortcut to CodeTextEditor
This commit is contained in:
commit
1aa2d8ba19
@ -152,6 +152,12 @@
|
||||
Perform an indent as if the user activated the "ui_text_indent" action.
|
||||
</description>
|
||||
</method>
|
||||
<method name="duplicate_lines">
|
||||
<return type="void" />
|
||||
<description>
|
||||
Duplicates all lines currently selected with any caret. Duplicates the entire line beneath the current one no matter where the caret is within the line.
|
||||
</description>
|
||||
</method>
|
||||
<method name="fold_all_lines">
|
||||
<return type="void" />
|
||||
<description>
|
||||
|
@ -805,6 +805,11 @@ void CodeTextEditor::input(const Ref<InputEvent> &event) {
|
||||
accept_event();
|
||||
return;
|
||||
}
|
||||
if (ED_IS_SHORTCUT("script_text_editor/duplicate_lines", key_event)) {
|
||||
text_editor->duplicate_lines();
|
||||
accept_event();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
void CodeTextEditor::_text_editor_gui_input(const Ref<InputEvent> &p_event) {
|
||||
|
@ -1306,6 +1306,9 @@ void ScriptTextEditor::_edit_option(int p_op) {
|
||||
case EDIT_DUPLICATE_SELECTION: {
|
||||
code_editor->duplicate_selection();
|
||||
} break;
|
||||
case EDIT_DUPLICATE_LINES: {
|
||||
code_editor->get_text_editor()->duplicate_lines();
|
||||
} break;
|
||||
case EDIT_TOGGLE_FOLD_LINE: {
|
||||
int previous_line = -1;
|
||||
for (int caret_idx : tx->get_caret_index_edit_order()) {
|
||||
@ -2173,6 +2176,7 @@ void ScriptTextEditor::_enable_code_editor() {
|
||||
edit_menu->get_popup()->add_separator();
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("ui_text_select_all"), EDIT_SELECT_ALL);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/duplicate_selection"), EDIT_DUPLICATE_SELECTION);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/duplicate_lines"), EDIT_DUPLICATE_LINES);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/evaluate_selection"), EDIT_EVALUATE);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/toggle_word_wrap"), EDIT_TOGGLE_WORD_WRAP);
|
||||
edit_menu->get_popup()->add_separator();
|
||||
@ -2395,6 +2399,8 @@ void ScriptTextEditor::register_editor() {
|
||||
ED_SHORTCUT("script_text_editor/unfold_all_lines", TTR("Unfold All Lines"), Key::NONE);
|
||||
ED_SHORTCUT("script_text_editor/duplicate_selection", TTR("Duplicate Selection"), KeyModifierMask::SHIFT | KeyModifierMask::CTRL | Key::D);
|
||||
ED_SHORTCUT_OVERRIDE("script_text_editor/duplicate_selection", "macos", KeyModifierMask::SHIFT | KeyModifierMask::META | Key::C);
|
||||
ED_SHORTCUT("script_text_editor/duplicate_lines", TTR("Duplicate Lines"), KeyModifierMask::CMD_OR_CTRL | KeyModifierMask::ALT | Key::DOWN);
|
||||
ED_SHORTCUT_OVERRIDE("script_text_editor/duplicate_lines", "macos", KeyModifierMask::CMD_OR_CTRL | KeyModifierMask::SHIFT | Key::DOWN);
|
||||
ED_SHORTCUT("script_text_editor/evaluate_selection", TTR("Evaluate Selection"), KeyModifierMask::CMD_OR_CTRL | KeyModifierMask::SHIFT | Key::E);
|
||||
ED_SHORTCUT("script_text_editor/toggle_word_wrap", TTR("Toggle Word Wrap"), KeyModifierMask::ALT | Key::Z);
|
||||
ED_SHORTCUT("script_text_editor/trim_trailing_whitespace", TTR("Trim Trailing Whitespace"), KeyModifierMask::CMD_OR_CTRL | KeyModifierMask::ALT | Key::T);
|
||||
|
@ -126,6 +126,7 @@ class ScriptTextEditor : public ScriptEditorBase {
|
||||
EDIT_UNINDENT,
|
||||
EDIT_DELETE_LINE,
|
||||
EDIT_DUPLICATE_SELECTION,
|
||||
EDIT_DUPLICATE_LINES,
|
||||
EDIT_PICK_COLOR,
|
||||
EDIT_TO_UPPERCASE,
|
||||
EDIT_TO_LOWERCASE,
|
||||
|
@ -392,6 +392,9 @@ void TextEditor::_edit_option(int p_op) {
|
||||
case EDIT_DUPLICATE_SELECTION: {
|
||||
code_editor->duplicate_selection();
|
||||
} break;
|
||||
case EDIT_DUPLICATE_LINES: {
|
||||
code_editor->get_text_editor()->duplicate_lines();
|
||||
} break;
|
||||
case EDIT_TOGGLE_FOLD_LINE: {
|
||||
int previous_line = -1;
|
||||
for (int caret_idx : tx->get_caret_index_edit_order()) {
|
||||
@ -651,6 +654,7 @@ TextEditor::TextEditor() {
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/unfold_all_lines"), EDIT_UNFOLD_ALL_LINES);
|
||||
edit_menu->get_popup()->add_separator();
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/duplicate_selection"), EDIT_DUPLICATE_SELECTION);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/duplicate_lines"), EDIT_DUPLICATE_LINES);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/toggle_word_wrap"), EDIT_TOGGLE_WORD_WRAP);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/trim_trailing_whitespace"), EDIT_TRIM_TRAILING_WHITESAPCE);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/convert_indent_to_spaces"), EDIT_CONVERT_INDENT_TO_SPACES);
|
||||
|
@ -71,6 +71,7 @@ private:
|
||||
EDIT_UNINDENT,
|
||||
EDIT_DELETE_LINE,
|
||||
EDIT_DUPLICATE_SELECTION,
|
||||
EDIT_DUPLICATE_LINES,
|
||||
EDIT_TO_UPPERCASE,
|
||||
EDIT_TO_LOWERCASE,
|
||||
EDIT_CAPITALIZE,
|
||||
|
@ -671,6 +671,9 @@ void TextShaderEditor::_menu_option(int p_option) {
|
||||
case EDIT_DUPLICATE_SELECTION: {
|
||||
shader_editor->duplicate_selection();
|
||||
} break;
|
||||
case EDIT_DUPLICATE_LINES: {
|
||||
shader_editor->get_text_editor()->duplicate_lines();
|
||||
} break;
|
||||
case EDIT_TOGGLE_WORD_WRAP: {
|
||||
TextEdit::LineWrappingMode wrap = shader_editor->get_text_editor()->get_line_wrapping_mode();
|
||||
shader_editor->get_text_editor()->set_line_wrapping_mode(wrap == TextEdit::LINE_WRAPPING_BOUNDARY ? TextEdit::LINE_WRAPPING_NONE : TextEdit::LINE_WRAPPING_BOUNDARY);
|
||||
@ -1122,6 +1125,7 @@ TextShaderEditor::TextShaderEditor() {
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/delete_line"), EDIT_DELETE_LINE);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/toggle_comment"), EDIT_TOGGLE_COMMENT);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/duplicate_selection"), EDIT_DUPLICATE_SELECTION);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/duplicate_lines"), EDIT_DUPLICATE_LINES);
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("script_text_editor/toggle_word_wrap"), EDIT_TOGGLE_WORD_WRAP);
|
||||
edit_menu->get_popup()->add_separator();
|
||||
edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("ui_text_completion_query"), EDIT_COMPLETE);
|
||||
|
@ -120,6 +120,7 @@ class TextShaderEditor : public MarginContainer {
|
||||
EDIT_UNINDENT,
|
||||
EDIT_DELETE_LINE,
|
||||
EDIT_DUPLICATE_SELECTION,
|
||||
EDIT_DUPLICATE_LINES,
|
||||
EDIT_TOGGLE_WORD_WRAP,
|
||||
EDIT_TOGGLE_COMMENT,
|
||||
EDIT_COMPLETE,
|
||||
|
@ -2399,6 +2399,68 @@ void CodeEdit::set_symbol_lookup_word_as_valid(bool p_valid) {
|
||||
}
|
||||
}
|
||||
|
||||
/* Text manipulation */
|
||||
void CodeEdit::duplicate_lines() {
|
||||
begin_complex_operation();
|
||||
|
||||
Vector<int> caret_edit_order = get_caret_index_edit_order();
|
||||
for (const int &caret_index : caret_edit_order) {
|
||||
// The text that will be inserted. All lines in one string.
|
||||
String insert_text;
|
||||
|
||||
// The new line position of the caret after the operation.
|
||||
int new_caret_line = get_caret_line(caret_index);
|
||||
// The new column position of the caret after the operation.
|
||||
int new_caret_column = get_caret_column(caret_index);
|
||||
// The caret positions of the selection. Stays -1 if there is no selection.
|
||||
int select_from_line = -1;
|
||||
int select_to_line = -1;
|
||||
int select_from_column = -1;
|
||||
int select_to_column = -1;
|
||||
// Number of lines of the selection.
|
||||
int select_num_lines = -1;
|
||||
|
||||
if (has_selection(caret_index)) {
|
||||
select_from_line = get_selection_from_line(caret_index);
|
||||
select_to_line = get_selection_to_line(caret_index);
|
||||
select_from_column = get_selection_from_column(caret_index);
|
||||
select_to_column = get_selection_to_column(caret_index);
|
||||
select_num_lines = select_to_line - select_from_line + 1;
|
||||
|
||||
for (int i = select_from_line; i <= select_to_line; i++) {
|
||||
insert_text += "\n" + get_line(i);
|
||||
unfold_line(i);
|
||||
}
|
||||
new_caret_line = select_to_line + select_num_lines;
|
||||
} else {
|
||||
insert_text = "\n" + get_line(new_caret_line);
|
||||
new_caret_line++;
|
||||
|
||||
unfold_line(get_caret_line(caret_index));
|
||||
}
|
||||
|
||||
// The text will be inserted at the end of the current line.
|
||||
set_caret_column(get_line(get_caret_line(caret_index)).length(), false, caret_index);
|
||||
|
||||
deselect(caret_index);
|
||||
|
||||
insert_text_at_caret(insert_text, caret_index);
|
||||
set_caret_line(new_caret_line, false, true, 0, caret_index);
|
||||
set_caret_column(new_caret_column, true, caret_index);
|
||||
|
||||
if (select_from_line != -1) {
|
||||
// Advance the selection by the number of duplicated lines.
|
||||
select_from_line += select_num_lines;
|
||||
select_to_line += select_num_lines;
|
||||
|
||||
select(select_from_line, select_from_column, select_to_line, select_to_column, caret_index);
|
||||
}
|
||||
}
|
||||
|
||||
end_complex_operation();
|
||||
queue_redraw();
|
||||
}
|
||||
|
||||
/* Visual */
|
||||
Color CodeEdit::_get_brace_mismatch_color() const {
|
||||
return theme_cache.brace_mismatch_color;
|
||||
@ -2598,6 +2660,9 @@ void CodeEdit::_bind_methods() {
|
||||
|
||||
ClassDB::bind_method(D_METHOD("set_symbol_lookup_word_as_valid", "valid"), &CodeEdit::set_symbol_lookup_word_as_valid);
|
||||
|
||||
/* Text manipulation */
|
||||
ClassDB::bind_method(D_METHOD("duplicate_lines"), &CodeEdit::duplicate_lines);
|
||||
|
||||
/* Inspector */
|
||||
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "symbol_lookup_on_click"), "set_symbol_lookup_on_click_enabled", "is_symbol_lookup_on_click_enabled");
|
||||
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "line_folding"), "set_line_folding_enabled", "is_line_folding_enabled");
|
||||
|
@ -486,6 +486,9 @@ public:
|
||||
|
||||
void set_symbol_lookup_word_as_valid(bool p_valid);
|
||||
|
||||
/* Text manipulation */
|
||||
void duplicate_lines();
|
||||
|
||||
CodeEdit();
|
||||
~CodeEdit();
|
||||
};
|
||||
|
@ -3886,6 +3886,84 @@ TEST_CASE("[SceneTree][CodeEdit] New Line") {
|
||||
memdelete(code_edit);
|
||||
}
|
||||
|
||||
TEST_CASE("[SceneTree][CodeEdit] Duplicate Lines") {
|
||||
CodeEdit *code_edit = memnew(CodeEdit);
|
||||
SceneTree::get_singleton()->get_root()->add_child(code_edit);
|
||||
code_edit->grab_focus();
|
||||
|
||||
code_edit->set_text(R"(extends Node
|
||||
|
||||
func _ready():
|
||||
var a := len(OS.get_cmdline_args())
|
||||
var b := get_child_count()
|
||||
var c := a + b
|
||||
for i in range(c):
|
||||
print("This is the solution: ", sin(i))
|
||||
var pos = get_index() - 1
|
||||
print("Make sure this exits: %b" % pos)
|
||||
)");
|
||||
|
||||
/* Duplicate a single line without selection. */
|
||||
code_edit->set_caret_line(0);
|
||||
code_edit->duplicate_lines();
|
||||
CHECK(code_edit->get_line(0) == "extends Node");
|
||||
CHECK(code_edit->get_line(1) == "extends Node");
|
||||
CHECK(code_edit->get_line(2) == "");
|
||||
|
||||
/* Duplicate multiple lines with selection. */
|
||||
code_edit->set_caret_line(6);
|
||||
code_edit->set_caret_column(15);
|
||||
code_edit->select(4, 8, 6, 15);
|
||||
code_edit->duplicate_lines();
|
||||
CHECK(code_edit->get_line(6) == "\tvar c := a + b");
|
||||
CHECK(code_edit->get_line(7) == "\tvar a := len(OS.get_cmdline_args())");
|
||||
CHECK(code_edit->get_line(8) == "\tvar b := get_child_count()");
|
||||
CHECK(code_edit->get_line(9) == "\tvar c := a + b");
|
||||
CHECK(code_edit->get_line(10) == "\tfor i in range(c):");
|
||||
|
||||
/* Duplicate single lines with multiple carets. */
|
||||
code_edit->deselect();
|
||||
code_edit->set_caret_line(10);
|
||||
code_edit->set_caret_column(1);
|
||||
code_edit->add_caret(11, 2);
|
||||
code_edit->add_caret(12, 1);
|
||||
code_edit->duplicate_lines();
|
||||
CHECK(code_edit->get_line(9) == "\tvar c := a + b");
|
||||
CHECK(code_edit->get_line(10) == "\tfor i in range(c):");
|
||||
CHECK(code_edit->get_line(11) == "\tfor i in range(c):");
|
||||
CHECK(code_edit->get_line(12) == "\t\tprint(\"This is the solution: \", sin(i))");
|
||||
CHECK(code_edit->get_line(13) == "\t\tprint(\"This is the solution: \", sin(i))");
|
||||
CHECK(code_edit->get_line(14) == "\tvar pos = get_index() - 1");
|
||||
CHECK(code_edit->get_line(15) == "\tvar pos = get_index() - 1");
|
||||
CHECK(code_edit->get_line(16) == "\tprint(\"Make sure this exits: %b\" % pos)");
|
||||
|
||||
/* Duplicate multiple lines with multiple carets. */
|
||||
code_edit->select(0, 0, 1, 2, 0);
|
||||
code_edit->select(3, 0, 4, 2, 1);
|
||||
code_edit->select(16, 0, 17, 0, 2);
|
||||
code_edit->set_caret_line(1, false, true, 0, 0);
|
||||
code_edit->set_caret_column(2, false, 0);
|
||||
code_edit->set_caret_line(4, false, true, 0, 1);
|
||||
code_edit->set_caret_column(2, false, 1);
|
||||
code_edit->set_caret_line(17, false, true, 0, 2);
|
||||
code_edit->set_caret_column(0, false, 2);
|
||||
code_edit->duplicate_lines();
|
||||
CHECK(code_edit->get_line(1) == "extends Node");
|
||||
CHECK(code_edit->get_line(2) == "extends Node");
|
||||
CHECK(code_edit->get_line(3) == "extends Node");
|
||||
CHECK(code_edit->get_line(4) == "");
|
||||
CHECK(code_edit->get_line(6) == "\tvar a := len(OS.get_cmdline_args())");
|
||||
CHECK(code_edit->get_line(7) == "func _ready():");
|
||||
CHECK(code_edit->get_line(8) == "\tvar a := len(OS.get_cmdline_args())");
|
||||
CHECK(code_edit->get_line(9) == "\tvar b := get_child_count()");
|
||||
CHECK(code_edit->get_line(20) == "\tprint(\"Make sure this exits: %b\" % pos)");
|
||||
CHECK(code_edit->get_line(21) == "");
|
||||
CHECK(code_edit->get_line(22) == "\tprint(\"Make sure this exits: %b\" % pos)");
|
||||
CHECK(code_edit->get_line(23) == "");
|
||||
|
||||
memdelete(code_edit);
|
||||
}
|
||||
|
||||
} // namespace TestCodeEdit
|
||||
|
||||
#endif // TEST_CODE_EDIT_H
|
||||
|
Loading…
Reference in New Issue
Block a user