/**************************************************************************/ /* test_code_edit.h */ /**************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* https://godotengine.org */ /**************************************************************************/ /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /**************************************************************************/ #ifndef TEST_CODE_EDIT_H #define TEST_CODE_EDIT_H #include "scene/gui/code_edit.h" #include "tests/test_macros.h" namespace TestCodeEdit { static inline Array build_array() { return Array(); } template static inline Array build_array(Variant item, Targs... Fargs) { Array a = build_array(Fargs...); a.push_front(item); return a; } TEST_CASE("[SceneTree][CodeEdit] line gutters") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); SUBCASE("[CodeEdit] breakpoints") { SIGNAL_WATCH(code_edit, "breakpoint_toggled"); SUBCASE("[CodeEdit] draw breakpoints gutter") { code_edit->set_draw_breakpoints_gutter(false); CHECK_FALSE(code_edit->is_drawing_breakpoints_gutter()); code_edit->set_draw_breakpoints_gutter(true); CHECK(code_edit->is_drawing_breakpoints_gutter()); } SUBCASE("[CodeEdit] set line as breakpoint") { /* Out of bounds. */ ERR_PRINT_OFF; code_edit->set_line_as_breakpoint(-1, true); CHECK_FALSE(code_edit->is_line_breakpointed(-1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); code_edit->set_line_as_breakpoint(1, true); CHECK_FALSE(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); ERR_PRINT_ON; Array args = build_array(build_array(0)); code_edit->set_line_as_breakpoint(0, true); CHECK(code_edit->is_line_breakpointed(0)); CHECK(code_edit->get_breakpointed_lines()[0] == 0); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->set_line_as_breakpoint(0, false); CHECK_FALSE(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] clear breakpointed lines") { code_edit->clear_breakpointed_lines(); SIGNAL_CHECK_FALSE("breakpoint_toggled"); Array args = build_array(build_array(0)); code_edit->set_line_as_breakpoint(0, true); CHECK(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->clear_breakpointed_lines(); CHECK_FALSE(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and set text") { Array args = build_array(build_array(0)); code_edit->set_text("test\nline"); code_edit->set_line_as_breakpoint(0, true); CHECK(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); /* breakpoint on lines that still exist are kept. */ code_edit->set_text(""); MessageQueue::get_singleton()->flush(); CHECK(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); /* breakpoint on lines that are removed should also be removed. */ code_edit->clear_breakpointed_lines(); SIGNAL_DISCARD("breakpoint_toggled") args = build_array(build_array(1)); code_edit->set_text("test\nline"); code_edit->set_line_as_breakpoint(1, true); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->set_text(""); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_breakpointed(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(1)); ERR_PRINT_ON; SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and clear") { Array args = build_array(build_array(0)); code_edit->set_text("test\nline"); code_edit->set_line_as_breakpoint(0, true); CHECK(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); /* breakpoint on lines that still exist are removed. */ code_edit->clear(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); /* breakpoint on lines that are removed should also be removed. */ code_edit->clear_breakpointed_lines(); SIGNAL_DISCARD("breakpoint_toggled") args = build_array(build_array(1)); code_edit->set_text("test\nline"); code_edit->set_line_as_breakpoint(1, true); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->clear(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_breakpointed(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(1)); ERR_PRINT_ON; SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and new lines no text") { Array args = build_array(build_array(0)); /* No text moves breakpoint. */ code_edit->set_line_as_breakpoint(0, true); CHECK(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); // Normal. args = build_array(build_array(0), build_array(1)); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line_count() == 2); CHECK_FALSE(code_edit->is_line_breakpointed(0)); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); // Non-Breaking. args = build_array(build_array(1), build_array(2)); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line_count() == 3); CHECK_FALSE(code_edit->is_line_breakpointed(1)); CHECK(code_edit->is_line_breakpointed(2)); SIGNAL_CHECK("breakpoint_toggled", args); // Above. args = build_array(build_array(2), build_array(3)); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line_count() == 4); CHECK_FALSE(code_edit->is_line_breakpointed(2)); CHECK(code_edit->is_line_breakpointed(3)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and new lines with text") { Array args = build_array(build_array(0)); /* Having text does not move breakpoint. */ code_edit->insert_text_at_caret("text"); code_edit->set_line_as_breakpoint(0, true); CHECK(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); /* Normal. */ SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line_count() == 2); CHECK(code_edit->is_line_breakpointed(0)); CHECK_FALSE(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); /* Non-Breaking. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line_count() == 3); CHECK(code_edit->is_line_breakpointed(0)); CHECK_FALSE(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); // Above does move. args = build_array(build_array(0), build_array(1)); code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line_count() == 4); CHECK_FALSE(code_edit->is_line_breakpointed(0)); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and backspace") { Array args = build_array(build_array(1)); code_edit->set_text("\n\n"); code_edit->set_line_as_breakpoint(1, true); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->set_caret_line(2); /* backspace onto line does not remove breakpoint */ SEND_GUI_ACTION("ui_text_backspace"); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); /* backspace on breakpointed line removes it */ SEND_GUI_ACTION("ui_text_backspace"); CHECK_FALSE(code_edit->is_line_breakpointed(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(1)); ERR_PRINT_ON; SIGNAL_CHECK("breakpoint_toggled", args); // Backspace above breakpointed line moves it. args = build_array(build_array(2)); code_edit->set_text("\n\n"); code_edit->set_line_as_breakpoint(2, true); CHECK(code_edit->is_line_breakpointed(2)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->set_caret_line(1); args = build_array(build_array(2), build_array(1)); SEND_GUI_ACTION("ui_text_backspace"); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(2)); ERR_PRINT_ON; CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and delete") { Array args = build_array(build_array(1)); code_edit->set_text("\n\n"); code_edit->set_line_as_breakpoint(1, true); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->set_caret_line(1); /* Delete onto breakpointed lines does not remove it. */ SEND_GUI_ACTION("ui_text_delete"); CHECK(code_edit->get_line_count() == 2); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); /* Delete moving breakpointed line up removes it. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_delete"); CHECK(code_edit->get_line_count() == 1); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(1)); ERR_PRINT_ON; SIGNAL_CHECK("breakpoint_toggled", args); // Delete above breakpointed line moves it. args = build_array(build_array(2)); code_edit->set_text("\n\n"); code_edit->set_line_as_breakpoint(2, true); CHECK(code_edit->is_line_breakpointed(2)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->set_caret_line(0); args = build_array(build_array(2), build_array(1)); SEND_GUI_ACTION("ui_text_delete"); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(2)); ERR_PRINT_ON; CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and delete selection") { Array args = build_array(build_array(1)); code_edit->set_text("\n\n"); code_edit->set_line_as_breakpoint(1, true); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->select(0, 0, 2, 0); code_edit->delete_selection(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); // Should handle breakpoint move when deleting selection by adding less text then removed. args = build_array(build_array(9)); code_edit->set_text("\n\n\n\n\n\n\n\n\n"); code_edit->set_line_as_breakpoint(9, true); CHECK(code_edit->is_line_breakpointed(9)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->select(0, 0, 6, 0); args = build_array(build_array(9), build_array(4)); SEND_GUI_ACTION("ui_text_newline"); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_breakpointed(9)); ERR_PRINT_ON; CHECK(code_edit->is_line_breakpointed(4)); SIGNAL_CHECK("breakpoint_toggled", args); // Should handle breakpoint move when deleting selection by adding more text then removed. args = build_array(build_array(9), build_array(14)); code_edit->insert_text_at_caret("\n\n\n\n\n"); MessageQueue::get_singleton()->flush(); SIGNAL_DISCARD("breakpoint_toggled") CHECK(code_edit->is_line_breakpointed(9)); code_edit->select(0, 0, 6, 0); code_edit->insert_text_at_caret("\n\n\n\n\n\n\n\n\n\n\n"); MessageQueue::get_singleton()->flush(); CHECK(code_edit->is_line_breakpointed(14)); SIGNAL_CHECK("breakpoint_toggled", args); } SUBCASE("[CodeEdit] breakpoints and undo") { Array args = build_array(build_array(1)); code_edit->set_text("\n\n"); code_edit->set_line_as_breakpoint(1, true); CHECK(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK("breakpoint_toggled", args); code_edit->select(0, 0, 2, 0); code_edit->delete_selection(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_breakpointed(0)); SIGNAL_CHECK("breakpoint_toggled", args); /* Undo does not restore breakpoint. */ code_edit->undo(); CHECK_FALSE(code_edit->is_line_breakpointed(1)); SIGNAL_CHECK_FALSE("breakpoint_toggled"); } SIGNAL_UNWATCH(code_edit, "breakpoint_toggled"); } SUBCASE("[CodeEdit] bookmarks") { SUBCASE("[CodeEdit] draw bookmarks gutter") { code_edit->set_draw_bookmarks_gutter(false); CHECK_FALSE(code_edit->is_drawing_bookmarks_gutter()); code_edit->set_draw_bookmarks_gutter(true); CHECK(code_edit->is_drawing_bookmarks_gutter()); } SUBCASE("[CodeEdit] set line as bookmarks") { /* Out of bounds. */ ERR_PRINT_OFF; code_edit->set_line_as_bookmarked(-1, true); CHECK_FALSE(code_edit->is_line_bookmarked(-1)); code_edit->set_line_as_bookmarked(1, true); CHECK_FALSE(code_edit->is_line_bookmarked(1)); ERR_PRINT_ON; code_edit->set_line_as_bookmarked(0, true); CHECK(code_edit->get_bookmarked_lines()[0] == 0); CHECK(code_edit->is_line_bookmarked(0)); code_edit->set_line_as_bookmarked(0, false); CHECK_FALSE(code_edit->is_line_bookmarked(0)); } SUBCASE("[CodeEdit] clear bookmarked lines") { code_edit->clear_bookmarked_lines(); code_edit->set_line_as_bookmarked(0, true); CHECK(code_edit->is_line_bookmarked(0)); code_edit->clear_bookmarked_lines(); CHECK_FALSE(code_edit->is_line_bookmarked(0)); } SUBCASE("[CodeEdit] bookmarks and set text") { code_edit->set_text("test\nline"); code_edit->set_line_as_bookmarked(0, true); CHECK(code_edit->is_line_bookmarked(0)); /* bookmarks on lines that still exist are kept. */ code_edit->set_text(""); MessageQueue::get_singleton()->flush(); CHECK(code_edit->is_line_bookmarked(0)); /* bookmarks on lines that are removed should also be removed. */ code_edit->clear_bookmarked_lines(); code_edit->set_text("test\nline"); code_edit->set_line_as_bookmarked(1, true); CHECK(code_edit->is_line_bookmarked(1)); code_edit->set_text(""); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_bookmarked(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_bookmarked(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] bookmarks and clear") { code_edit->set_text("test\nline"); code_edit->set_line_as_bookmarked(0, true); CHECK(code_edit->is_line_bookmarked(0)); /* bookmarks on lines that still exist are removed. */ code_edit->clear(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_bookmarked(0)); /* bookmarks on lines that are removed should also be removed. */ code_edit->clear_bookmarked_lines(); code_edit->set_text("test\nline"); code_edit->set_line_as_bookmarked(1, true); CHECK(code_edit->is_line_bookmarked(1)); code_edit->clear(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_bookmarked(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_bookmarked(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] bookmarks and new lines no text") { /* No text moves bookmarks. */ code_edit->set_line_as_bookmarked(0, true); CHECK(code_edit->is_line_bookmarked(0)); /* Normal. */ SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line_count() == 2); CHECK_FALSE(code_edit->is_line_bookmarked(0)); CHECK(code_edit->is_line_bookmarked(1)); /* Non-Breaking. */ SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line_count() == 3); CHECK_FALSE(code_edit->is_line_bookmarked(1)); CHECK(code_edit->is_line_bookmarked(2)); /* Above. */ SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line_count() == 4); CHECK_FALSE(code_edit->is_line_bookmarked(2)); CHECK(code_edit->is_line_bookmarked(3)); } SUBCASE("[CodeEdit] bookmarks and new lines with text") { /* Having text does not move bookmark. */ code_edit->insert_text_at_caret("text"); code_edit->set_line_as_bookmarked(0, true); CHECK(code_edit->is_line_bookmarked(0)); /* Normal. */ SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line_count() == 2); CHECK(code_edit->is_line_bookmarked(0)); CHECK_FALSE(code_edit->is_line_bookmarked(1)); /* Non-Breaking. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line_count() == 3); CHECK(code_edit->is_line_bookmarked(0)); CHECK_FALSE(code_edit->is_line_bookmarked(1)); /* Above does move. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line_count() == 4); CHECK_FALSE(code_edit->is_line_bookmarked(0)); CHECK(code_edit->is_line_bookmarked(1)); } SUBCASE("[CodeEdit] bookmarks and backspace") { code_edit->set_text("\n\n"); code_edit->set_line_as_bookmarked(1, true); CHECK(code_edit->is_line_bookmarked(1)); code_edit->set_caret_line(2); /* backspace onto line does not remove bookmark */ SEND_GUI_ACTION("ui_text_backspace"); CHECK(code_edit->is_line_bookmarked(1)); /* backspace on bookmarked line removes it */ SEND_GUI_ACTION("ui_text_backspace"); CHECK_FALSE(code_edit->is_line_bookmarked(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_bookmarked(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] bookmarks and delete") { code_edit->set_text("\n\n"); code_edit->set_line_as_bookmarked(1, true); CHECK(code_edit->is_line_bookmarked(1)); code_edit->set_caret_line(1); /* Delete onto bookmarked lines does not remove it. */ SEND_GUI_ACTION("ui_text_delete"); CHECK(code_edit->get_line_count() == 2); CHECK(code_edit->is_line_bookmarked(1)); /* Delete moving bookmarked line up removes it. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_delete"); CHECK(code_edit->get_line_count() == 1); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_bookmarked(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] bookmarks and delete selection") { code_edit->set_text("\n\n"); code_edit->set_line_as_bookmarked(1, true); CHECK(code_edit->is_line_bookmarked(1)); code_edit->select(0, 0, 2, 0); code_edit->delete_selection(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_bookmarked(0)); } SUBCASE("[CodeEdit] bookmarks and undo") { code_edit->set_text("\n\n"); code_edit->set_line_as_bookmarked(1, true); CHECK(code_edit->is_line_bookmarked(1)); code_edit->select(0, 0, 2, 0); code_edit->delete_selection(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_bookmarked(0)); /* Undo does not restore bookmark. */ code_edit->undo(); CHECK_FALSE(code_edit->is_line_bookmarked(1)); } } SUBCASE("[CodeEdit] executing lines") { SUBCASE("[CodeEdit] draw executing lines gutter") { code_edit->set_draw_executing_lines_gutter(false); CHECK_FALSE(code_edit->is_drawing_executing_lines_gutter()); code_edit->set_draw_executing_lines_gutter(true); CHECK(code_edit->is_drawing_executing_lines_gutter()); } SUBCASE("[CodeEdit] set line as executing lines") { /* Out of bounds. */ ERR_PRINT_OFF; code_edit->set_line_as_executing(-1, true); CHECK_FALSE(code_edit->is_line_executing(-1)); code_edit->set_line_as_executing(1, true); CHECK_FALSE(code_edit->is_line_executing(1)); ERR_PRINT_ON; code_edit->set_line_as_executing(0, true); CHECK(code_edit->get_executing_lines()[0] == 0); CHECK(code_edit->is_line_executing(0)); code_edit->set_line_as_executing(0, false); CHECK_FALSE(code_edit->is_line_executing(0)); } SUBCASE("[CodeEdit] clear executing lines lines") { code_edit->clear_executing_lines(); code_edit->set_line_as_executing(0, true); CHECK(code_edit->is_line_executing(0)); code_edit->clear_executing_lines(); CHECK_FALSE(code_edit->is_line_executing(0)); } SUBCASE("[CodeEdit] executing lines and set text") { code_edit->set_text("test\nline"); code_edit->set_line_as_executing(0, true); CHECK(code_edit->is_line_executing(0)); /* executing on lines that still exist are kept. */ code_edit->set_text(""); MessageQueue::get_singleton()->flush(); CHECK(code_edit->is_line_executing(0)); /* executing on lines that are removed should also be removed. */ code_edit->clear_executing_lines(); code_edit->set_text("test\nline"); code_edit->set_line_as_executing(1, true); CHECK(code_edit->is_line_executing(1)); code_edit->set_text(""); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_executing(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_executing(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] executing lines and clear") { code_edit->set_text("test\nline"); code_edit->set_line_as_executing(0, true); CHECK(code_edit->is_line_executing(0)); /* executing on lines that still exist are removed. */ code_edit->clear(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_executing(0)); /* executing on lines that are removed should also be removed. */ code_edit->clear_executing_lines(); code_edit->set_text("test\nline"); code_edit->set_line_as_executing(1, true); CHECK(code_edit->is_line_executing(1)); code_edit->clear(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_executing(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_executing(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] executing lines and new lines no text") { /* No text moves executing lines. */ code_edit->set_line_as_executing(0, true); CHECK(code_edit->is_line_executing(0)); /* Normal. */ SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line_count() == 2); CHECK_FALSE(code_edit->is_line_executing(0)); CHECK(code_edit->is_line_executing(1)); /* Non-Breaking. */ SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line_count() == 3); CHECK_FALSE(code_edit->is_line_executing(1)); CHECK(code_edit->is_line_executing(2)); /* Above. */ SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line_count() == 4); CHECK_FALSE(code_edit->is_line_executing(2)); CHECK(code_edit->is_line_executing(3)); } SUBCASE("[CodeEdit] executing lines and new lines with text") { /* Having text does not move executing lines. */ code_edit->insert_text_at_caret("text"); code_edit->set_line_as_executing(0, true); CHECK(code_edit->is_line_executing(0)); /* Normal. */ SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line_count() == 2); CHECK(code_edit->is_line_executing(0)); CHECK_FALSE(code_edit->is_line_executing(1)); /* Non-Breaking. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line_count() == 3); CHECK(code_edit->is_line_executing(0)); CHECK_FALSE(code_edit->is_line_executing(1)); /* Above does move. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line_count() == 4); CHECK_FALSE(code_edit->is_line_executing(0)); CHECK(code_edit->is_line_executing(1)); } SUBCASE("[CodeEdit] executing lines and backspace") { code_edit->set_text("\n\n"); code_edit->set_line_as_executing(1, true); CHECK(code_edit->is_line_executing(1)); code_edit->set_caret_line(2); /* backspace onto line does not remove executing lines. */ SEND_GUI_ACTION("ui_text_backspace"); CHECK(code_edit->is_line_executing(1)); /* backspace on executing line removes it */ SEND_GUI_ACTION("ui_text_backspace"); CHECK_FALSE(code_edit->is_line_executing(0)); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_executing(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] executing lines and delete") { code_edit->set_text("\n\n"); code_edit->set_line_as_executing(1, true); CHECK(code_edit->is_line_executing(1)); code_edit->set_caret_line(1); /* Delete onto executing lines does not remove it. */ SEND_GUI_ACTION("ui_text_delete"); CHECK(code_edit->get_line_count() == 2); CHECK(code_edit->is_line_executing(1)); /* Delete moving executing line up removes it. */ code_edit->set_caret_line(0); SEND_GUI_ACTION("ui_text_delete"); CHECK(code_edit->get_line_count() == 1); ERR_PRINT_OFF; CHECK_FALSE(code_edit->is_line_executing(1)); ERR_PRINT_ON; } SUBCASE("[CodeEdit] executing lines and delete selection") { code_edit->set_text("\n\n"); code_edit->set_line_as_executing(1, true); CHECK(code_edit->is_line_executing(1)); code_edit->select(0, 0, 2, 0); code_edit->delete_selection(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_executing(0)); } SUBCASE("[CodeEdit] executing lines and undo") { code_edit->set_text("\n\n"); code_edit->set_line_as_executing(1, true); CHECK(code_edit->is_line_executing(1)); code_edit->select(0, 0, 2, 0); code_edit->delete_selection(); MessageQueue::get_singleton()->flush(); CHECK_FALSE(code_edit->is_line_executing(0)); /* Undo does not restore executing lines. */ code_edit->undo(); CHECK_FALSE(code_edit->is_line_executing(1)); } } SUBCASE("[CodeEdit] line numbers") { SUBCASE("[CodeEdit] draw line numbers gutter and padding") { code_edit->set_draw_line_numbers(false); CHECK_FALSE(code_edit->is_draw_line_numbers_enabled()); code_edit->set_draw_line_numbers(true); CHECK(code_edit->is_draw_line_numbers_enabled()); code_edit->set_line_numbers_zero_padded(false); CHECK_FALSE(code_edit->is_line_numbers_zero_padded()); code_edit->set_line_numbers_zero_padded(true); CHECK(code_edit->is_line_numbers_zero_padded()); code_edit->set_line_numbers_zero_padded(false); CHECK_FALSE(code_edit->is_line_numbers_zero_padded()); code_edit->set_draw_line_numbers(false); CHECK_FALSE(code_edit->is_draw_line_numbers_enabled()); code_edit->set_line_numbers_zero_padded(true); CHECK(code_edit->is_line_numbers_zero_padded()); } } SUBCASE("[CodeEdit] line folding") { SUBCASE("[CodeEdit] draw line folding gutter") { code_edit->set_draw_fold_gutter(false); CHECK_FALSE(code_edit->is_drawing_fold_gutter()); code_edit->set_draw_fold_gutter(true); CHECK(code_edit->is_drawing_fold_gutter()); } } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] delimiters") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); const Point2 OUTSIDE_DELIMETER = Point2(-1, -1); code_edit->clear_string_delimiters(); code_edit->clear_comment_delimiters(); SUBCASE("[CodeEdit] add and remove delimiters") { SUBCASE("[CodeEdit] add and remove string delimiters") { /* Add a delimiter.*/ code_edit->add_string_delimiter("\"", "\"", false); CHECK(code_edit->has_string_delimiter("\"")); CHECK(code_edit->get_string_delimiters().size() == 1); ERR_PRINT_OFF; /* Adding a duplicate start key is not allowed. */ code_edit->add_string_delimiter("\"", "\'", false); CHECK(code_edit->get_string_delimiters().size() == 1); /* Adding a duplicate end key is allowed. */ code_edit->add_string_delimiter("'", "\"", false); CHECK(code_edit->has_string_delimiter("'")); CHECK(code_edit->get_string_delimiters().size() == 2); /* Both start and end keys have to be symbols. */ code_edit->add_string_delimiter("f", "\"", false); CHECK_FALSE(code_edit->has_string_delimiter("f")); CHECK(code_edit->get_string_delimiters().size() == 2); code_edit->add_string_delimiter("f", "\"", false); CHECK_FALSE(code_edit->has_string_delimiter("f")); CHECK(code_edit->get_string_delimiters().size() == 2); code_edit->add_string_delimiter("@", "f", false); CHECK_FALSE(code_edit->has_string_delimiter("@")); CHECK(code_edit->get_string_delimiters().size() == 2); code_edit->add_string_delimiter("f", "f", false); CHECK_FALSE(code_edit->has_string_delimiter("f")); CHECK(code_edit->get_string_delimiters().size() == 2); /* Blank start keys are not allowed */ code_edit->add_string_delimiter("", "#", false); CHECK_FALSE(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 2); ERR_PRINT_ON; /* Blank end keys are allowed. */ code_edit->add_string_delimiter("#", "", false); CHECK(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 3); /* Remove a delimiter. */ code_edit->remove_string_delimiter("#"); CHECK_FALSE(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 2); /* Set should override existing, and test multiline */ TypedArray delimiters; delimiters.push_back("^^ ^^"); code_edit->set_string_delimiters(delimiters); CHECK_FALSE(code_edit->has_string_delimiter("\"")); CHECK(code_edit->has_string_delimiter("^^")); CHECK(code_edit->get_string_delimiters().size() == 1); /* clear should remove all. */ code_edit->clear_string_delimiters(); CHECK_FALSE(code_edit->has_string_delimiter("^^")); CHECK(code_edit->get_string_delimiters().size() == 0); } SUBCASE("[CodeEdit] add and remove comment delimiters") { /* Add a delimiter.*/ code_edit->add_comment_delimiter("\"", "\"", false); CHECK(code_edit->has_comment_delimiter("\"")); CHECK(code_edit->get_comment_delimiters().size() == 1); ERR_PRINT_OFF; /* Adding a duplicate start key is not allowed. */ code_edit->add_comment_delimiter("\"", "\'", false); CHECK(code_edit->get_comment_delimiters().size() == 1); /* Adding a duplicate end key is allowed. */ code_edit->add_comment_delimiter("'", "\"", false); CHECK(code_edit->has_comment_delimiter("'")); CHECK(code_edit->get_comment_delimiters().size() == 2); /* Both start and end keys have to be symbols. */ code_edit->add_comment_delimiter("f", "\"", false); CHECK_FALSE(code_edit->has_comment_delimiter("f")); CHECK(code_edit->get_comment_delimiters().size() == 2); code_edit->add_comment_delimiter("f", "\"", false); CHECK_FALSE(code_edit->has_comment_delimiter("f")); CHECK(code_edit->get_comment_delimiters().size() == 2); code_edit->add_comment_delimiter("@", "f", false); CHECK_FALSE(code_edit->has_comment_delimiter("@")); CHECK(code_edit->get_comment_delimiters().size() == 2); code_edit->add_comment_delimiter("f", "f", false); CHECK_FALSE(code_edit->has_comment_delimiter("f")); CHECK(code_edit->get_comment_delimiters().size() == 2); /* Blank start keys are not allowed. */ code_edit->add_comment_delimiter("", "#", false); CHECK_FALSE(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 2); ERR_PRINT_ON; /* Blank end keys are allowed. */ code_edit->add_comment_delimiter("#", "", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 3); /* Remove a delimiter. */ code_edit->remove_comment_delimiter("#"); CHECK_FALSE(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 2); /* Set should override existing, and test multiline. */ TypedArray delimiters; delimiters.push_back("^^ ^^"); code_edit->set_comment_delimiters(delimiters); CHECK_FALSE(code_edit->has_comment_delimiter("\"")); CHECK(code_edit->has_comment_delimiter("^^")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* clear should remove all. */ code_edit->clear_comment_delimiters(); CHECK_FALSE(code_edit->has_comment_delimiter("^^")); CHECK(code_edit->get_comment_delimiters().size() == 0); } SUBCASE("[CodeEdit] add and remove mixed delimiters") { code_edit->add_comment_delimiter("#", "", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); ERR_PRINT_OFF; /* Disallow adding a string with the same start key as comment. */ code_edit->add_string_delimiter("#", "", false); CHECK_FALSE(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 0); code_edit->add_string_delimiter("\"", "\"", false); CHECK(code_edit->has_string_delimiter("\"")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* Disallow adding a comment with the same start key as string. */ code_edit->add_comment_delimiter("\"", "", false); CHECK_FALSE(code_edit->has_comment_delimiter("\"")); CHECK(code_edit->get_comment_delimiters().size() == 1); ERR_PRINT_ON; /* Cannot remove string with remove comment. */ code_edit->remove_comment_delimiter("\""); CHECK(code_edit->has_string_delimiter("\"")); CHECK(code_edit->get_string_delimiters().size() == 1); /* Cannot remove comment with remove string. */ code_edit->remove_string_delimiter("#"); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* Clear comments leave strings. */ code_edit->clear_comment_delimiters(); CHECK(code_edit->has_string_delimiter("\"")); CHECK(code_edit->get_string_delimiters().size() == 1); /* Clear string leave comments. */ code_edit->add_comment_delimiter("#", "", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); code_edit->clear_string_delimiters(); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); } } SUBCASE("[CodeEdit] single line delimiters") { SUBCASE("[CodeEdit] single line string delimiters") { /* Blank end key should set lineonly to true. */ code_edit->add_string_delimiter("#", "", false); CHECK(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 1); /* Insert line above, line with string then line below. */ code_edit->insert_text_at_caret(" \n#\n "); /* Check line above is not in string. */ CHECK(code_edit->is_in_string(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in string. */ CHECK(code_edit->is_in_string(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column after start key is in string and start / end positions are correct. */ CHECK(code_edit->is_in_string(1, 1) != -1); CHECK(code_edit->get_delimiter_start_position(1, 1) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 1) == Point2(2, 1)); /* Check line after is not in string. */ CHECK(code_edit->is_in_string(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Check region metadata. */ int idx = code_edit->is_in_string(1, 1); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == ""); /* Check nested strings are handled correctly. */ code_edit->set_text(" \n# # \n "); /* Check line above is not in string. */ CHECK(code_edit->is_in_string(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before first start key is not in string. */ CHECK(code_edit->is_in_string(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column after the first start key is in string and start / end positions are correct. */ CHECK(code_edit->is_in_string(1, 1) != -1); CHECK(code_edit->get_delimiter_start_position(1, 1) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 1) == Point2(6, 1)); /* Check column after the second start key returns data for the first. */ CHECK(code_edit->is_in_string(1, 5) != -1); CHECK(code_edit->get_delimiter_start_position(1, 5) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 5) == Point2(6, 1)); /* Check line after is not in string. */ CHECK(code_edit->is_in_string(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Check is in string with no column returns true if entire line is comment excluding whitespace. */ code_edit->set_text(" \n # # \n "); CHECK(code_edit->is_in_string(1) != -1); code_edit->set_text(" \n text # # \n "); CHECK(code_edit->is_in_string(1) == -1); /* Removing delimiter should update. */ code_edit->set_text(" \n # # \n "); code_edit->remove_string_delimiter("#"); CHECK_FALSE(code_edit->has_string_delimiter("$")); CHECK(code_edit->get_string_delimiters().size() == 0); CHECK(code_edit->is_in_string(1) == -1); /* Adding and clear should update. */ code_edit->add_string_delimiter("#", "", false); CHECK(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 1); CHECK(code_edit->is_in_string(1) != -1); code_edit->clear_string_delimiters(); CHECK_FALSE(code_edit->has_string_delimiter("$")); CHECK(code_edit->get_string_delimiters().size() == 0); CHECK(code_edit->is_in_string(1) == -1); } SUBCASE("[CodeEdit] single line comment delimiters") { /* Blank end key should set lineonly to true. */ code_edit->add_comment_delimiter("#", "", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* Insert line above, line with comment then line below. */ code_edit->insert_text_at_caret(" \n#\n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column after start key is in comment and start / end positions are correct. */ CHECK(code_edit->is_in_comment(1, 1) != -1); CHECK(code_edit->get_delimiter_start_position(1, 1) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 1) == Point2(2, 1)); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Check region metadata. */ int idx = code_edit->is_in_comment(1, 1); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == ""); /* Check nested comments are handled correctly. */ code_edit->set_text(" \n# # \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before first start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column after the first start key is in comment and start / end positions are correct. */ CHECK(code_edit->is_in_comment(1, 1) != -1); CHECK(code_edit->get_delimiter_start_position(1, 1) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 1) == Point2(6, 1)); /* Check column after the second start key returns data for the first. */ CHECK(code_edit->is_in_comment(1, 5) != -1); CHECK(code_edit->get_delimiter_start_position(1, 5) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 5) == Point2(6, 1)); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Check is in comment with no column returns true if entire line is comment excluding whitespace. */ code_edit->set_text(" \n # # \n "); CHECK(code_edit->is_in_comment(1) != -1); code_edit->set_text(" \n text # # \n "); CHECK(code_edit->is_in_comment(1) == -1); /* Removing delimiter should update. */ code_edit->set_text(" \n # # \n "); code_edit->remove_comment_delimiter("#"); CHECK_FALSE(code_edit->has_comment_delimiter("$")); CHECK(code_edit->get_comment_delimiters().size() == 0); CHECK(code_edit->is_in_comment(1) == -1); /* Adding and clear should update. */ code_edit->add_comment_delimiter("#", "", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); CHECK(code_edit->is_in_comment(1) != -1); code_edit->clear_comment_delimiters(); CHECK_FALSE(code_edit->has_comment_delimiter("$")); CHECK(code_edit->get_comment_delimiters().size() == 0); CHECK(code_edit->is_in_comment(1) == -1); } SUBCASE("[CodeEdit] single line mixed delimiters") { /* Blank end key should set lineonly to true. */ /* Add string delimiter. */ code_edit->add_string_delimiter("&", "", false); CHECK(code_edit->has_string_delimiter("&")); CHECK(code_edit->get_string_delimiters().size() == 1); /* Add comment delimiter. */ code_edit->add_comment_delimiter("#", "", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* Nest a string delimiter inside a comment. */ code_edit->set_text(" \n# & \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before first start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column after the first start key is in comment and start / end positions are correct. */ CHECK(code_edit->is_in_comment(1, 1) != -1); CHECK(code_edit->get_delimiter_start_position(1, 1) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 1) == Point2(6, 1)); /* Check column after the second start key returns data for the first, and does not state string. */ CHECK(code_edit->is_in_comment(1, 5) != -1); CHECK(code_edit->get_delimiter_start_position(1, 5) == Point2(1, 1)); CHECK(code_edit->get_delimiter_end_position(1, 5) == Point2(6, 1)); CHECK(code_edit->is_in_string(1, 5) == -1); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Remove the comment delimiter. */ code_edit->remove_comment_delimiter("#"); CHECK_FALSE(code_edit->has_comment_delimiter("$")); CHECK(code_edit->get_comment_delimiters().size() == 0); /* The "first" comment region is no longer valid. */ CHECK(code_edit->is_in_comment(1, 1) == -1); CHECK(code_edit->get_delimiter_start_position(1, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 1) == OUTSIDE_DELIMETER); /* The "second" region as string is now valid. */ CHECK(code_edit->is_in_string(1, 5) != -1); CHECK(code_edit->get_delimiter_start_position(1, 5) == Point2(4, 1)); CHECK(code_edit->get_delimiter_end_position(1, 5) == Point2(6, 1)); } } SUBCASE("[CodeEdit] multiline delimiters") { SUBCASE("[CodeEdit] multiline string delimiters") { code_edit->clear_string_delimiters(); code_edit->clear_comment_delimiters(); /* Add string delimiter. */ code_edit->add_string_delimiter("#", "#", false); CHECK(code_edit->has_string_delimiter("#")); CHECK(code_edit->get_string_delimiters().size() == 1); /* First test over a single line. */ code_edit->set_text(" \n # # \n "); /* Check line above is not in string. */ CHECK(code_edit->is_in_string(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in string. */ CHECK(code_edit->is_in_string(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column before closing delimiter is in string. */ CHECK(code_edit->is_in_string(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(5, 1)); /* Check column after end key is not in string. */ CHECK(code_edit->is_in_string(1, 6) == -1); CHECK(code_edit->get_delimiter_start_position(1, 6) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 6) == OUTSIDE_DELIMETER); /* Check line after is not in string. */ CHECK(code_edit->is_in_string(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Check the region metadata. */ int idx = code_edit->is_in_string(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Next test over a multiple blank lines. */ code_edit->set_text(" \n # \n\n # \n "); /* Check line above is not in string. */ CHECK(code_edit->is_in_string(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in string. */ CHECK(code_edit->is_in_string(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in string. */ CHECK(code_edit->is_in_string(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(2, 3)); /* Check blank middle line. */ CHECK(code_edit->is_in_string(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(2, 3)); /* Check column just before end key is in string. */ CHECK(code_edit->is_in_string(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(2, 3)); /* Check column after end key is not in string. */ CHECK(code_edit->is_in_string(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in string. */ CHECK(code_edit->is_in_string(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* Next test over a multiple non-blank lines. */ code_edit->set_text(" \n # \n \n # \n "); /* Check line above is not in string. */ CHECK(code_edit->is_in_string(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in string. */ CHECK(code_edit->is_in_string(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in string. */ CHECK(code_edit->is_in_string(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(2, 3)); /* Check middle line. */ CHECK(code_edit->is_in_string(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(2, 3)); /* Check column just before end key is in string. */ CHECK(code_edit->is_in_string(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(2, 3)); /* Check column after end key is not in string. */ CHECK(code_edit->is_in_string(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in string. */ CHECK(code_edit->is_in_string(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* check the region metadata. */ idx = code_edit->is_in_string(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Next test nested strings. */ code_edit->add_string_delimiter("^", "^", false); CHECK(code_edit->has_string_delimiter("^")); CHECK(code_edit->get_string_delimiters().size() == 2); code_edit->set_text(" \n # ^\n \n^ # \n "); /* Check line above is not in string. */ CHECK(code_edit->is_in_string(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in string. */ CHECK(code_edit->is_in_string(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in string. */ CHECK(code_edit->is_in_string(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(3, 3)); /* Check middle line. */ CHECK(code_edit->is_in_string(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(3, 3)); /* Check column just before end key is in string. */ CHECK(code_edit->is_in_string(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(3, 3)); /* Check column after end key is not in string. */ CHECK(code_edit->is_in_string(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in string. */ CHECK(code_edit->is_in_string(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* check the region metadata. */ idx = code_edit->is_in_string(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Next test no end key. */ code_edit->set_text(" \n # \n "); /* check the region metadata. */ idx = code_edit->is_in_string(1, 2); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(-1, -1)); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Check is in string with no column returns true if entire line is string excluding whitespace. */ code_edit->set_text(" \n # \n\n #\n "); CHECK(code_edit->is_in_string(1) != -1); CHECK(code_edit->is_in_string(2) != -1); CHECK(code_edit->is_in_string(3) != -1); code_edit->set_text(" \n test # \n\n # test \n "); CHECK(code_edit->is_in_string(1) == -1); CHECK(code_edit->is_in_string(2) != -1); CHECK(code_edit->is_in_string(3) == -1); /* Next check updating the delimiter cache while typing. */ code_edit->set_text("\n\n"); code_edit->set_caret_line(0); code_edit->set_caret_column(0); CHECK(code_edit->is_in_string(0) == -1); CHECK(code_edit->is_in_string(1) == -1); code_edit->insert_text_at_caret("#"); CHECK(code_edit->is_in_string(0) != -1); CHECK(code_edit->is_in_string(1) != -1); code_edit->insert_text_at_caret("#"); CHECK(code_edit->is_in_string(0) != -1); CHECK(code_edit->is_in_string(1) == -1); } SUBCASE("[CodeEdit] multiline comment delimiters") { /* Add comment delimiter. */ code_edit->add_comment_delimiter("#", "#", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* First test over a single line. */ code_edit->set_text(" \n # # \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column before closing delimiter is in comment. */ CHECK(code_edit->is_in_comment(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(5, 1)); /* Check column after end key is not in comment. */ CHECK(code_edit->is_in_comment(1, 6) == -1); CHECK(code_edit->get_delimiter_start_position(1, 6) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 6) == OUTSIDE_DELIMETER); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(2, 1) == -1); CHECK(code_edit->get_delimiter_start_position(2, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(2, 1) == OUTSIDE_DELIMETER); /* Check the region metadata. */ int idx = code_edit->is_in_comment(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Next test over a multiple blank lines. */ code_edit->set_text(" \n # \n\n # \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in comment. */ CHECK(code_edit->is_in_comment(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(2, 3)); /* Check blank middle line. */ CHECK(code_edit->is_in_comment(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(2, 3)); /* Check column just before end key is in comment. */ CHECK(code_edit->is_in_comment(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(2, 3)); /* Check column after end key is not in comment. */ CHECK(code_edit->is_in_comment(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* Next test over a multiple non-blank lines. */ code_edit->set_text(" \n # \n \n # \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in comment. */ CHECK(code_edit->is_in_comment(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(2, 3)); /* Check middle line. */ CHECK(code_edit->is_in_comment(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(2, 3)); /* Check column just before end key is in comment. */ CHECK(code_edit->is_in_comment(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(2, 3)); /* Check column after end key is not in comment. */ CHECK(code_edit->is_in_comment(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* check the region metadata. */ idx = code_edit->is_in_comment(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Next test nested comments. */ code_edit->add_comment_delimiter("^", "^", false); CHECK(code_edit->has_comment_delimiter("^")); CHECK(code_edit->get_comment_delimiters().size() == 2); code_edit->set_text(" \n # ^\n \n^ # \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in comment. */ CHECK(code_edit->is_in_comment(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(3, 3)); /* Check middle line. */ CHECK(code_edit->is_in_comment(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(3, 3)); /* Check column just before end key is in comment. */ CHECK(code_edit->is_in_comment(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(3, 3)); /* Check column after end key is not in comment. */ CHECK(code_edit->is_in_comment(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* check the region metadata. */ idx = code_edit->is_in_comment(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Next test no end key. */ code_edit->set_text(" \n # \n "); /* check the region metadata. */ idx = code_edit->is_in_comment(1, 2); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(-1, -1)); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Check is in comment with no column returns true if entire line is comment excluding whitespace. */ code_edit->set_text(" \n # \n\n #\n "); CHECK(code_edit->is_in_comment(1) != -1); CHECK(code_edit->is_in_comment(2) != -1); CHECK(code_edit->is_in_comment(3) != -1); code_edit->set_text(" \n test # \n\n # test \n "); CHECK(code_edit->is_in_comment(1) == -1); CHECK(code_edit->is_in_comment(2) != -1); CHECK(code_edit->is_in_comment(3) == -1); /* Next check updating the delimiter cache while typing. */ code_edit->set_text("\n\n"); code_edit->set_caret_line(0); code_edit->set_caret_column(0); CHECK(code_edit->is_in_comment(0) == -1); CHECK(code_edit->is_in_comment(1) == -1); code_edit->insert_text_at_caret("#"); CHECK(code_edit->is_in_comment(0) != -1); CHECK(code_edit->is_in_comment(1) != -1); code_edit->insert_text_at_caret("#"); CHECK(code_edit->is_in_comment(0) != -1); CHECK(code_edit->is_in_comment(1) == -1); } SUBCASE("[CodeEdit] multiline mixed delimiters") { /* Add comment delimiter. */ code_edit->add_comment_delimiter("#", "#", false); CHECK(code_edit->has_comment_delimiter("#")); CHECK(code_edit->get_comment_delimiters().size() == 1); /* Add string delimiter. */ code_edit->add_string_delimiter("^", "^", false); CHECK(code_edit->has_string_delimiter("^")); CHECK(code_edit->get_string_delimiters().size() == 1); /* Nest a string inside a comment. */ code_edit->set_text(" \n # ^\n \n^ # \n "); /* Check line above is not in comment. */ CHECK(code_edit->is_in_comment(0, 1) == -1); CHECK(code_edit->get_delimiter_start_position(0, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(0, 1) == OUTSIDE_DELIMETER); /* Check column before start key is not in comment. */ CHECK(code_edit->is_in_comment(1, 0) == -1); CHECK(code_edit->get_delimiter_start_position(1, 0) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(1, 0) == OUTSIDE_DELIMETER); /* Check column just after start key is in comment. */ CHECK(code_edit->is_in_comment(1, 2) != -1); CHECK(code_edit->get_delimiter_start_position(1, 2) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(1, 2) == Point2(3, 3)); /* Check middle line. */ CHECK(code_edit->is_in_comment(2, 0) != -1); CHECK(code_edit->get_delimiter_start_position(2, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(2, 0) == Point2(3, 3)); /* Check column just before end key is in comment. */ CHECK(code_edit->is_in_comment(3, 0) != -1); CHECK(code_edit->get_delimiter_start_position(3, 0) == Point2(2, 1)); CHECK(code_edit->get_delimiter_end_position(3, 0) == Point2(3, 3)); /* Check column after end key is not in comment. */ CHECK(code_edit->is_in_comment(3, 3) == -1); CHECK(code_edit->get_delimiter_start_position(3, 3) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(3, 3) == OUTSIDE_DELIMETER); /* Check line after is not in comment. */ CHECK(code_edit->is_in_comment(4, 1) == -1); CHECK(code_edit->get_delimiter_start_position(4, 1) == OUTSIDE_DELIMETER); CHECK(code_edit->get_delimiter_end_position(4, 1) == OUTSIDE_DELIMETER); /* check the region metadata. */ int idx = code_edit->is_in_comment(1, 2); CHECK(code_edit->get_delimiter_start_key(idx) == "#"); CHECK(code_edit->get_delimiter_end_key(idx) == "#"); /* Check is in comment with no column returns true as inner delimiter should not be counted. */ CHECK(code_edit->is_in_comment(1) != -1); CHECK(code_edit->is_in_comment(2) != -1); CHECK(code_edit->is_in_comment(3) != -1); } } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] indent") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); SUBCASE("[CodeEdit] indent settings") { code_edit->set_indent_size(10); CHECK(code_edit->get_indent_size() == 10); CHECK(code_edit->get_tab_size() == 10); code_edit->set_auto_indent_enabled(false); CHECK_FALSE(code_edit->is_auto_indent_enabled()); code_edit->set_auto_indent_enabled(true); CHECK(code_edit->is_auto_indent_enabled()); code_edit->set_indent_using_spaces(false); CHECK_FALSE(code_edit->is_indent_using_spaces()); code_edit->set_indent_using_spaces(true); CHECK(code_edit->is_indent_using_spaces()); /* Only the first char is registered. */ TypedArray auto_indent_prefixes; auto_indent_prefixes.push_back("::"); auto_indent_prefixes.push_back("s"); auto_indent_prefixes.push_back("1"); code_edit->set_auto_indent_prefixes(auto_indent_prefixes); auto_indent_prefixes = code_edit->get_auto_indent_prefixes(); CHECK(auto_indent_prefixes.has(":")); CHECK(auto_indent_prefixes.has("s")); CHECK(auto_indent_prefixes.has("1")); } SUBCASE("[CodeEdit] indent tabs") { code_edit->set_indent_size(4); code_edit->set_auto_indent_enabled(true); code_edit->set_indent_using_spaces(false); /* Do nothing if not editable. */ code_edit->set_editable(false); code_edit->do_indent(); CHECK(code_edit->get_line(0).is_empty()); code_edit->indent_lines(); CHECK(code_edit->get_line(0).is_empty()); code_edit->set_editable(true); /* Simple indent. */ code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\t"); /* Check input action. */ SEND_GUI_ACTION("ui_text_indent"); CHECK(code_edit->get_line(0) == "\t\t"); /* Insert in place. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test"); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "test\t"); // Insert in place with multiple carets. code_edit->set_text("test text"); code_edit->set_caret_column(5); code_edit->add_caret(0, 7); code_edit->add_caret(0, 2); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "te\tst \tte\txt"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->get_caret_column(0) == 7); CHECK(code_edit->get_caret_column(1) == 10); CHECK(code_edit->get_caret_column(2) == 3); code_edit->remove_secondary_carets(); // Indent lines does entire line and works without selection. code_edit->set_text(""); code_edit->insert_text_at_caret("test"); code_edit->indent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_caret_column() == 5); /* Selection does entire line. */ code_edit->set_text("test"); code_edit->select_all(); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 5); // Selection does entire line, right to left selection. code_edit->set_text("test"); code_edit->select(0, 4, 0, 0); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 5); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* Handles multiple lines. */ code_edit->set_text("test\ntext"); code_edit->select_all(); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == "\ttext"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 5); /* Do not indent line if last col is zero. */ code_edit->set_text("test\ntext"); code_edit->select(0, 0, 1, 0); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Indent even if last column of first line. */ code_edit->set_text("test\ntext"); code_edit->select(0, 4, 1, 0); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 5); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Indent even if last column of first line, reversed. code_edit->set_text("test\ntext"); code_edit->select(1, 0, 0, 4); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 5); /* Check selection is adjusted. */ code_edit->set_text("test"); code_edit->select(0, 1, 0, 2); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 3); // Indent once with multiple selections. code_edit->set_text("test"); code_edit->select(0, 1, 0, 2); code_edit->add_caret(0, 4); code_edit->select(0, 4, 0, 3, 1); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 0); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 3); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 0); CHECK(code_edit->get_selection_origin_column(1) == 5); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 4); } SUBCASE("[CodeEdit] indent spaces") { code_edit->set_indent_size(4); code_edit->set_auto_indent_enabled(true); code_edit->set_indent_using_spaces(true); /* Do nothing if not editable. */ code_edit->set_editable(false); code_edit->do_indent(); CHECK(code_edit->get_line(0).is_empty()); code_edit->indent_lines(); CHECK(code_edit->get_line(0).is_empty()); code_edit->set_editable(true); /* Simple indent. */ code_edit->do_indent(); CHECK(code_edit->get_line(0) == " "); /* Check input action. */ SEND_GUI_ACTION("ui_text_indent"); CHECK(code_edit->get_line(0) == " "); /* Insert in place. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test"); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "test "); // Insert in place with multiple carets. code_edit->set_text("test text"); code_edit->set_caret_column(5); code_edit->add_caret(0, 7); code_edit->add_caret(0, 2); code_edit->do_indent(); CHECK(code_edit->get_line(0) == "te st te xt"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->get_caret_column(0) == 10); CHECK(code_edit->get_caret_column(1) == 14); CHECK(code_edit->get_caret_column(2) == 4); code_edit->remove_secondary_carets(); // Indent lines does entire line and works without selection. code_edit->set_text(""); code_edit->insert_text_at_caret("test"); code_edit->indent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_caret_column() == 8); /* Selection does entire line. */ code_edit->set_text("test"); code_edit->select_all(); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 8); // Selection does entire line, right to left selection. code_edit->set_text("test"); code_edit->select(0, 4, 0, 0); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* single indent only add required spaces. */ code_edit->set_text(" test"); code_edit->select_all(); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 8); /* Handles multiple lines. */ code_edit->set_text("test\ntext"); code_edit->select_all(); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == " text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 8); /* Do not indent line if last col is zero. */ code_edit->set_text("test\ntext"); code_edit->select(0, 0, 1, 0); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Indent even if last column of first line. */ code_edit->set_text("test\ntext"); code_edit->select(0, 4, 1, 0); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Indent even if last column of first line, right to left selection. code_edit->set_text("test\ntext"); code_edit->select(1, 0, 0, 4); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 8); /* Check selection is adjusted. */ code_edit->set_text("test"); code_edit->select(0, 1, 0, 2); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 5); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 6); // Indent once with multiple selections. code_edit->set_text("test"); code_edit->select(0, 1, 0, 2); code_edit->add_caret(0, 4); code_edit->select(0, 4, 0, 3, 1); code_edit->do_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 0); CHECK(code_edit->get_selection_origin_column(0) == 5); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 6); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 0); CHECK(code_edit->get_selection_origin_column(1) == 8); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 7); } SUBCASE("[CodeEdit] unindent tabs") { code_edit->set_indent_size(4); code_edit->set_auto_indent_enabled(true); code_edit->set_indent_using_spaces(false); /* Do nothing if not editable. */ code_edit->set_text("\t"); code_edit->set_editable(false); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\t"); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\t"); code_edit->set_editable(true); /* Simple unindent. */ code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == ""); /* Backspace does a simple unindent. */ code_edit->set_text(""); code_edit->insert_text_at_caret("\t"); code_edit->backspace(); CHECK(code_edit->get_line(0) == ""); /* Unindent lines does entire line and works without selection. */ code_edit->set_text(""); code_edit->insert_text_at_caret("\ttest"); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_caret_column() == 4); // Unindent lines once with multiple carets. code_edit->set_text("\t\ttest"); code_edit->set_caret_column(1); code_edit->add_caret(0, 3); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_caret_count() == 2); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 2); code_edit->remove_secondary_carets(); /* Caret on col zero unindent line. */ code_edit->set_text("\t\ttest"); code_edit->set_caret_column(0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_caret_column() == 0); /* Check input action. */ code_edit->set_text("\t\ttest"); SEND_GUI_ACTION("ui_text_dedent"); CHECK(code_edit->get_line(0) == "\ttest"); /* Selection does entire line. */ code_edit->set_text("\t\ttest"); code_edit->select_all(); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 5); // Selection does entire line, right to left selection. code_edit->set_text("\t\ttest"); code_edit->select(0, 6, 0, 0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 5); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Handles multiple lines. code_edit->set_text("\t\ttest\n\t\ttext"); code_edit->select_all(); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == "\ttext"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 5); /* Do not unindent line if last col is zero. */ code_edit->set_text("\ttest\n\ttext"); code_edit->select(0, 0, 1, 0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_line(1) == "\ttext"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Do not unindent line if last col is zero, right to left selection. code_edit->set_text("\ttest\n\ttext"); code_edit->select(1, 0, 0, 0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_line(1) == "\ttext"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* Unindent even if last column of first line. */ code_edit->set_text("\ttest\n\ttext"); code_edit->select(0, 5, 1, 1); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 4); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Check selection is adjusted. */ code_edit->set_text("\ttest"); code_edit->select(0, 1, 0, 2); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 1); // Deselect if only the tab was selected. code_edit->set_text("\ttest"); code_edit->select(0, 0, 0, 1); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Unindent once with multiple selections. code_edit->set_text("\t\ttest"); code_edit->select(0, 1, 0, 2); code_edit->add_caret(0, 4); code_edit->select(0, 4, 0, 3, 1); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 0); CHECK(code_edit->get_selection_origin_column(0) == 0); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 1); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 0); CHECK(code_edit->get_selection_origin_column(1) == 3); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 2); } SUBCASE("[CodeEdit] unindent spaces") { code_edit->set_indent_size(4); code_edit->set_auto_indent_enabled(true); code_edit->set_indent_using_spaces(true); /* Do nothing if not editable. */ code_edit->set_text(" "); code_edit->set_editable(false); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " "); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " "); code_edit->set_editable(true); /* Simple unindent. */ code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == ""); /* Backspace does a simple unindent. */ code_edit->set_text(""); code_edit->insert_text_at_caret(" "); code_edit->backspace(); CHECK(code_edit->get_line(0) == ""); /* Backspace with letter. */ code_edit->set_text(""); code_edit->insert_text_at_caret(" a"); code_edit->backspace(); CHECK(code_edit->get_line(0) == " "); /* Unindent lines does entire line and works without selection. */ code_edit->set_text(""); code_edit->insert_text_at_caret(" test"); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_caret_column() == 4); // Unindent lines once with multiple carets. code_edit->set_text(" test"); code_edit->set_caret_column(1); code_edit->add_caret(0, 9); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_caret_count() == 2); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 5); code_edit->remove_secondary_carets(); /* Caret on col zero unindent line. */ code_edit->set_text(" test"); code_edit->set_caret_column(0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_caret_column() == 0); /* Only as far as needed */ code_edit->set_text(" test"); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); /* Check input action. */ code_edit->set_text(" test"); SEND_GUI_ACTION("ui_text_dedent"); CHECK(code_edit->get_line(0) == " test"); /* Selection does entire line. */ code_edit->set_text(" test"); code_edit->select_all(); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 8); // Selection does entire line, right to left selection. code_edit->set_text(" test"); code_edit->select(0, 12, 0, 0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Handles multiple lines. code_edit->set_text(" test\n text"); code_edit->select_all(); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == " text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 8); /* Do not unindent line if last col is zero. */ code_edit->set_text(" test\n text"); code_edit->select(0, 0, 1, 0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_line(1) == " text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Do not unindent line if last col is zero, right to left selection. code_edit->set_text(" test\n text"); code_edit->select(1, 0, 0, 0); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_line(1) == " text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* Unindent even if last column of first line. */ code_edit->set_text(" test\n text"); code_edit->select(0, 5, 1, 1); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->get_line(1) == "text"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 1); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Check selection is adjusted. */ code_edit->set_text(" test"); code_edit->select(0, 4, 0, 5); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 1); // Deselect if only the tab was selected. code_edit->set_text(" test"); code_edit->select(0, 0, 0, 4); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == "test"); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Unindent once with multiple selections. code_edit->set_text(" test"); code_edit->select(0, 1, 0, 2); code_edit->add_caret(0, 4); code_edit->select(0, 12, 0, 10, 1); code_edit->unindent_lines(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_caret_count() == 2); CHECK_FALSE(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 0); CHECK(code_edit->get_selection_origin_column(1) == 8); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 6); } SUBCASE("[CodeEdit] auto indent") { SUBCASE("[CodeEdit] auto indent tabs") { code_edit->set_indent_size(4); code_edit->set_auto_indent_enabled(true); code_edit->set_indent_using_spaces(false); /* Simple indent on new line. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test:"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test:"); CHECK(code_edit->get_line(1) == "\t"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); /* new blank line should still indent. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test:"); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "test:"); CHECK(code_edit->get_line(1) == "\t"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); /* new line above should not indent. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test:"); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == "test:"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* Whitespace between symbol and caret is okay. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test: "); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test: "); CHECK(code_edit->get_line(1) == "\t"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); /* Comment between symbol and caret is okay. */ code_edit->add_comment_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test: # comment"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test: # comment"); CHECK(code_edit->get_line(1) == "\t"); code_edit->remove_comment_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); /* Strings between symbol and caret are not okay. */ code_edit->add_string_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test: # string"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test: # string"); CHECK(code_edit->get_line(1) == ""); code_edit->remove_string_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Non-whitespace prevents auto-indentation. */ code_edit->add_comment_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test := 0 # comment"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test := 0 # comment"); CHECK(code_edit->get_line(1) == ""); code_edit->remove_comment_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Even when there's no comments. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test := 0"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test := 0"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Preserve current indentation. code_edit->set_text("\ttest"); code_edit->set_caret_column(3); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "\tte"); CHECK(code_edit->get_line(1) == "\tst"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Preserve current indentation blank. code_edit->set_text("\ttest"); code_edit->set_caret_column(3); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == "\t"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Preserve current indentation above. code_edit->set_text("\ttest"); code_edit->set_caret_column(3); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == "\t"); CHECK(code_edit->get_line(1) == "\ttest"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 1); // Increase existing indentation. code_edit->set_text("\ttest:"); code_edit->set_caret_column(6); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "\ttest:"); CHECK(code_edit->get_line(1) == "\t\t"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 2); // Increase existing indentation blank. code_edit->set_text("\ttest:"); code_edit->set_caret_column(3); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "\ttest:"); CHECK(code_edit->get_line(1) == "\t\t"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 2); /* If between brace pairs an extra line is added. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test{}"); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test{"); CHECK(code_edit->get_line(1) == "\t"); CHECK(code_edit->get_line(2) == "}"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); /* Except when we are going above. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test{}"); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == "test{}"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* or below. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test{}"); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "test{}"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); } SUBCASE("[CodeEdit] auto indent spaces") { code_edit->set_indent_size(4); code_edit->set_auto_indent_enabled(true); code_edit->set_indent_using_spaces(true); /* Simple indent on new line. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test:"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test:"); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); /* new blank line should still indent. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test:"); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "test:"); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); /* new line above should not indent. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test:"); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == "test:"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* Whitespace between symbol and caret is okay. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test: "); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test: "); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); /* Comment between symbol and caret is okay. */ code_edit->add_comment_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test: # comment"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test: # comment"); CHECK(code_edit->get_line(1) == " "); code_edit->remove_comment_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); /* Strings between symbol and caret are not okay. */ code_edit->add_string_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test: # string"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test: # string"); CHECK(code_edit->get_line(1) == ""); code_edit->remove_string_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Non-whitespace prevents auto-indentation. */ code_edit->add_comment_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test := 0 # comment"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test := 0 # comment"); CHECK(code_edit->get_line(1) == ""); code_edit->remove_comment_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* Even when there's no comments. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test := 0"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test := 0"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Preserve current indentation. code_edit->set_text(" test"); code_edit->set_caret_column(6); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == " te"); CHECK(code_edit->get_line(1) == " st"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); // Preserve current indentation blank. code_edit->set_text(" test"); code_edit->set_caret_column(6); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); // Preserve current indentation above. code_edit->set_text(" test"); code_edit->set_caret_column(6); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == " "); CHECK(code_edit->get_line(1) == " test"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 4); // Increase existing indentation. code_edit->set_text(" test:"); code_edit->set_caret_column(9); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == " test:"); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 8); // Increase existing indentation blank. code_edit->set_text(" test:"); code_edit->set_caret_column(9); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == " test:"); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 8); /* If between brace pairs an extra line is added. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test{}"); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test{"); CHECK(code_edit->get_line(1) == " "); CHECK(code_edit->get_line(2) == "}"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); /* Except when we are going above. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test{}"); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == "test{}"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); /* or below. */ code_edit->set_text(""); code_edit->insert_text_at_caret("test{}"); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "test{}"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); /* If there is something after a colon and there is a colon in the comment it should not indent. */ code_edit->add_comment_delimiter("#", ""); code_edit->set_text(""); code_edit->insert_text_at_caret("test:test#:"); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test:test#:"); CHECK(code_edit->get_line(1) == ""); code_edit->remove_comment_delimiter("#"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); } } SUBCASE("[CodeEdit] convert indent to tabs") { code_edit->set_indent_size(4); code_edit->set_indent_using_spaces(false); // Only line. code_edit->set_text(" test"); code_edit->select(0, 9, 0, 8); code_edit->convert_indent(); CHECK(code_edit->get_line(0) == "\t\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 3); CHECK(code_edit->get_caret_column() == 2); // First line. code_edit->set_text(" test\n"); code_edit->select(0, 8, 0, 9); code_edit->convert_indent(); CHECK(code_edit->get_line(0) == "\t\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_column() == 3); // Middle line. code_edit->set_text("\n test\n"); code_edit->select(1, 8, 1, 9); code_edit->convert_indent(); CHECK(code_edit->get_line(1) == "\t\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_column() == 3); // End line. code_edit->set_text("\n test"); code_edit->select(1, 8, 1, 9); code_edit->convert_indent(); CHECK(code_edit->get_line(1) == "\t\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_column() == 3); // Within provided range. code_edit->set_text(" test\n test\n"); code_edit->select(1, 8, 1, 9); code_edit->convert_indent(1, 1); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->get_line(1) == "\t\ttest"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_column() == 3); } SUBCASE("[CodeEdit] convert indent to spaces") { code_edit->set_indent_size(4); code_edit->set_indent_using_spaces(true); // Only line. code_edit->set_text("\t\ttest"); code_edit->select(0, 3, 0, 2); code_edit->convert_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 9); CHECK(code_edit->get_caret_column() == 8); // First line. code_edit->set_text("\t\ttest\n"); code_edit->select(0, 2, 0, 3); code_edit->convert_indent(); CHECK(code_edit->get_line(0) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_column() == 9); // Middle line. code_edit->set_text("\n\t\ttest\n"); code_edit->select(1, 2, 1, 3); code_edit->convert_indent(); CHECK(code_edit->get_line(1) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_column() == 9); // End line. code_edit->set_text("\n\t\ttest"); code_edit->select(1, 2, 1, 3); code_edit->convert_indent(); CHECK(code_edit->get_line(1) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_column() == 9); // Within provided range. code_edit->set_text("\ttest\n\t\ttest\n"); code_edit->select(1, 2, 1, 3); code_edit->convert_indent(1, 1); CHECK(code_edit->get_line(0) == "\ttest"); CHECK(code_edit->get_line(1) == " test"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_column() == 9); // Outside of range. ERR_PRINT_OFF; code_edit->convert_indent(0, 4); code_edit->convert_indent(4, 5); code_edit->convert_indent(4, 1); ERR_PRINT_ON; } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] folding") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); code_edit->set_line_folding_enabled(true); SUBCASE("[CodeEdit] folding settings") { code_edit->set_line_folding_enabled(true); CHECK(code_edit->is_line_folding_enabled()); code_edit->set_line_folding_enabled(false); CHECK_FALSE(code_edit->is_line_folding_enabled()); } SUBCASE("[CodeEdit] folding") { // No indent. code_edit->set_text("line1\nline2\nline3"); for (int i = 0; i < 2; i++) { CHECK_FALSE(code_edit->can_fold_line(i)); code_edit->fold_line(i); CHECK_FALSE(code_edit->is_line_folded(i)); } CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Indented lines. code_edit->set_text("\tline1\n\tline2\n\tline3"); for (int i = 0; i < 2; i++) { CHECK_FALSE(code_edit->can_fold_line(i)); code_edit->fold_line(i); CHECK_FALSE(code_edit->is_line_folded(i)); } CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Indent. code_edit->set_text("line1\n\tline2\nline3"); CHECK(code_edit->can_fold_line(0)); for (int i = 1; i < 2; i++) { CHECK_FALSE(code_edit->can_fold_line(i)); code_edit->fold_line(i); CHECK_FALSE(code_edit->is_line_folded(i)); } code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 2); // Indent with blank lines. code_edit->set_text("line1\n\tline2\n\n\nline3"); CHECK(code_edit->can_fold_line(0)); for (int i = 1; i < 2; i++) { CHECK_FALSE(code_edit->can_fold_line(i)); code_edit->fold_line(i); CHECK_FALSE(code_edit->is_line_folded(i)); } code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 2); // Nested indents. code_edit->set_text("line1\n\tline2\n\t\tline3\nline4"); CHECK(code_edit->can_fold_line(0)); CHECK(code_edit->can_fold_line(1)); for (int i = 2; i < 3; i++) { CHECK_FALSE(code_edit->can_fold_line(i)); code_edit->fold_line(i); CHECK_FALSE(code_edit->is_line_folded(i)); } code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK(code_edit->is_line_folded(1)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK_FALSE(code_edit->is_line_folded(3)); CHECK(code_edit->get_next_visible_line_offset_from(2, 1) == 2); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK_FALSE(code_edit->is_line_folded(3)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 3); // Check metadata. CHECK(code_edit->get_folded_lines().size() == 1); CHECK((int)code_edit->get_folded_lines()[0] == 0); // Cannot unfold nested. code_edit->unfold_line(1); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // (un)Fold all / toggle. code_edit->unfold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Check metadata. CHECK(code_edit->get_folded_lines().size() == 0); code_edit->fold_all_lines(); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 3); code_edit->unfold_all_lines(); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); code_edit->toggle_foldable_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 3); // Can also unfold from hidden line. code_edit->unfold_line(1); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Blank lines. code_edit->set_text("line1\n\tline2\n\n\n\ttest\n\nline3"); CHECK(code_edit->can_fold_line(0)); for (int i = 1; i < code_edit->get_line_count(); i++) { CHECK_FALSE(code_edit->can_fold_line(i)); code_edit->fold_line(i); CHECK_FALSE(code_edit->is_line_folded(i)); } code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); for (int i = 1; i < code_edit->get_line_count(); i++) { CHECK_FALSE(code_edit->is_line_folded(i)); } CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 5); // End of file. code_edit->set_text("line1\n\tline2"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Comment & string blocks. // Single line block code_edit->add_comment_delimiter("#", "", true); code_edit->set_text("#line1\n#\tline2"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Has to be full line. code_edit->set_text("test #line1\n#\tline2"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); code_edit->set_text("#line1\ntest #\tline2"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // String. code_edit->add_string_delimiter("^", "", true); code_edit->set_text("^line1\n^\tline2"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Has to be full line. code_edit->set_text("test ^line1\n^\tline2"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); code_edit->set_text("^line1\ntest ^\tline2"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Multiline blocks. code_edit->add_comment_delimiter("&", "&", false); code_edit->set_text("&line1\n\tline2&\nline3"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 2); // Multiline comment before last line. code_edit->set_text("&line1\nline2&\ntest"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(2)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 2); // Has to be full line. code_edit->set_text("test &line1\n\tline2&"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); code_edit->set_text("&line1\n\tline2& test"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Strings. code_edit->add_string_delimiter("$", "$", false); code_edit->set_text("$line1\n\tline2$"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Has to be full line. code_edit->set_text("test $line1\n\tline2$"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); code_edit->set_text("$line1\n\tline2$ test"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 1); // Non-indented comments/strings. // Single line code_edit->set_text("test\n\tline1\n#line1\n#line2\n\ttest"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); code_edit->set_text("test\n\tline1\n^line1\n^line2\n\ttest"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); // Indent level 0->1, comment after lines code_edit->set_text("line1\n\tline2\n#test"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 2); // Indent level 0->1, comment between lines code_edit->set_text("line1\n#test\n\tline2\nline3"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(2)); code_edit->fold_line(2); CHECK_FALSE(code_edit->is_line_folded(2)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 3); // Indent level 1->2, comment after lines code_edit->set_text("\tline1\n\t\tline2\n#test"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 2); // Indent level 1->2, comment between lines code_edit->set_text("\tline1\n#test\n\t\tline2\nline3"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(2)); code_edit->fold_line(2); CHECK_FALSE(code_edit->is_line_folded(2)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 3); // Multiline code_edit->set_text("test\n\tline1\n&line1\nline2&\n\ttest"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); code_edit->set_text("test\n\tline1\n$line1\nline2$\n\ttest"); CHECK(code_edit->can_fold_line(0)); CHECK_FALSE(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK_FALSE(code_edit->is_line_folded(1)); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); } SUBCASE("[CodeEdit] folding carets") { // Folding a line moves all carets that would be hidden. code_edit->set_text("test\n\tline1\n\t\tline 2\n"); code_edit->set_caret_line(1); code_edit->set_caret_column(0); code_edit->add_caret(1, 3); code_edit->add_caret(2, 8); code_edit->add_caret(2, 1); code_edit->select(2, 0, 2, 1, 3); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK(code_edit->get_caret_count() == 1); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 4); // Undoing an action that puts the caret on a folded line unfolds it. code_edit->set_text("test\n\tline1"); code_edit->select(1, 1, 1, 2); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test\n\tlline1"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 3); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 2); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 4); code_edit->undo(); CHECK(code_edit->get_text() == "test\n\tline1"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 2); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 1); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); // Redoing doesn't refold. code_edit->redo(); CHECK(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 3); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 2); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); } SUBCASE("[CodeEdit] actions unfold") { // add_selection_for_next_occurrence unfolds. code_edit->set_text("test\n\tline1 test\n\t\tline 2\ntest2"); code_edit->select(0, 0, 0, 4); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); code_edit->add_selection_for_next_occurrence(); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_caret_column() == 4); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_selection_origin_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 11); CHECK(code_edit->get_selection_origin_column(1) == 7); CHECK_FALSE(code_edit->is_line_folded(0)); code_edit->remove_secondary_carets(); // skip_selection_for_next_occurrence unfolds. code_edit->select(0, 0, 0, 4); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); code_edit->skip_selection_for_next_occurrence(); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_caret_column() == 11); CHECK(code_edit->get_selection_origin_column() == 7); CHECK_FALSE(code_edit->is_line_folded(0)); code_edit->remove_secondary_carets(); code_edit->deselect(); } SUBCASE("[CodeEdit] toggle folding carets") { code_edit->set_text("test\n\tline1\ntest2\n\tline2"); // Fold lines with carets on them. code_edit->set_caret_line(0); code_edit->set_caret_column(1); code_edit->toggle_foldable_lines_at_carets(); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(2)); // Toggle fold on lines with carets. code_edit->add_caret(2, 0); code_edit->toggle_foldable_lines_at_carets(); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK(code_edit->is_line_folded(2)); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 1); CHECK(code_edit->get_caret_line(1) == 2); CHECK(code_edit->get_caret_column(1) == 0); // Multiple carets as part of one fold. code_edit->unfold_all_lines(); code_edit->remove_secondary_carets(); code_edit->set_caret_line(0); code_edit->set_caret_column(1); code_edit->add_caret(0, 4); code_edit->add_caret(1, 2); code_edit->toggle_foldable_lines_at_carets(); CHECK(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(2)); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 1); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 4); } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] region folding") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); SUBCASE("[CodeEdit] region tags") { code_edit->set_line_folding_enabled(true); // Region tag detection. code_edit->set_text("#region region_name\nline2\n#endregion"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); CHECK(code_edit->is_line_code_region_start(0)); CHECK_FALSE(code_edit->is_line_code_region_start(1)); CHECK_FALSE(code_edit->is_line_code_region_start(2)); CHECK_FALSE(code_edit->is_line_code_region_end(0)); CHECK_FALSE(code_edit->is_line_code_region_end(1)); CHECK(code_edit->is_line_code_region_end(2)); // Region tag customization. code_edit->set_text("#region region_name\nline2\n#endregion\n#open region_name\nline2\n#close"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); CHECK(code_edit->is_line_code_region_start(0)); CHECK(code_edit->is_line_code_region_end(2)); CHECK_FALSE(code_edit->is_line_code_region_start(3)); CHECK_FALSE(code_edit->is_line_code_region_end(5)); code_edit->set_code_region_tags("open", "close"); CHECK_FALSE(code_edit->is_line_code_region_start(0)); CHECK_FALSE(code_edit->is_line_code_region_end(2)); CHECK(code_edit->is_line_code_region_start(3)); CHECK(code_edit->is_line_code_region_end(5)); code_edit->set_code_region_tags("region", "endregion"); // Setting identical start and end region tags should fail. CHECK(code_edit->get_code_region_start_tag() == "region"); CHECK(code_edit->get_code_region_end_tag() == "endregion"); ERR_PRINT_OFF; code_edit->set_code_region_tags("same_tag", "same_tag"); ERR_PRINT_ON; CHECK(code_edit->get_code_region_start_tag() == "region"); CHECK(code_edit->get_code_region_end_tag() == "endregion"); } SUBCASE("[CodeEdit] create code region") { code_edit->set_line_folding_enabled(true); // Region creation with selection adds start and close region lines. Region name is selected and the region is folded. code_edit->set_text("line1\nline2\nline3"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->select(1, 0, 1, 4); code_edit->create_code_region(); CHECK(code_edit->is_line_code_region_start(1)); CHECK(code_edit->is_line_code_region_end(3)); CHECK(code_edit->get_text() == "line1\n#region New Code Region\nline2\n#endregion\nline3"); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selected_text() == "New Code Region"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 23); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->is_line_folded(1)); // Undo region creation. Line get unfolded. code_edit->undo(); CHECK(code_edit->get_text() == "line1\nline2\nline3"); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 4); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 0); CHECK_FALSE(code_edit->is_line_folded(1)); // Redo region creation. code_edit->redo(); CHECK(code_edit->get_text() == "line1\n#region New Code Region\nline2\n#endregion\nline3"); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selected_text() == "New Code Region"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 23); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 8); CHECK_FALSE(code_edit->is_line_folded(1)); // Region creation without any selection has no effect. code_edit->set_text("line1\nline2\nline3"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->create_code_region(); CHECK(code_edit->get_text() == "line1\nline2\nline3"); // Region creation with multiple selections. Secondary carets are removed and the first region name is selected. code_edit->set_text("line1\nline2\nline3"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->select(0, 0, 0, 4, 0); code_edit->add_caret(2, 5); code_edit->select(2, 0, 2, 5, 1); code_edit->create_code_region(); CHECK(code_edit->get_text() == "#region New Code Region\nline1\n#endregion\nline2\n#region New Code Region\nline3\n#endregion"); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selected_text() == "New Code Region"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 23); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 8); // Region creation with mixed selection and non-selection carets. Regular carets are ignored. code_edit->set_text("line1\nline2\nline3"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->select(0, 0, 0, 4, 0); code_edit->add_caret(2, 5); code_edit->create_code_region(); CHECK(code_edit->get_text() == "#region New Code Region\nline1\n#endregion\nline2\nline3"); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selected_text() == "New Code Region"); // Two selections on the same line create only one region. code_edit->set_text("test line1\ntest line2\ntest line3"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->select(0, 0, 1, 2, 0); code_edit->add_caret(1, 4); code_edit->select(1, 4, 2, 5, 1); code_edit->create_code_region(); CHECK(code_edit->get_text() == "#region New Code Region\ntest line1\ntest line2\ntest line3\n#endregion"); // Region tag with // comment delimiter. code_edit->set_text("//region region_name\nline2\n//endregion"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("//", ""); CHECK(code_edit->is_line_code_region_start(0)); CHECK(code_edit->is_line_code_region_end(2)); // Creating region with no valid one line comment delimiter has no effect. code_edit->set_text("line1\nline2\nline3"); code_edit->clear_comment_delimiters(); code_edit->create_code_region(); CHECK(code_edit->get_text() == "line1\nline2\nline3"); code_edit->add_comment_delimiter("/*", "*/"); code_edit->create_code_region(); CHECK(code_edit->get_text() == "line1\nline2\nline3"); } SUBCASE("[CodeEdit] region comment delimiters") { code_edit->set_line_folding_enabled(true); // Choose one line comment delimiter. code_edit->set_text("//region region_name\nline2\n//endregion"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("/*", "*/"); code_edit->add_comment_delimiter("//", ""); CHECK(code_edit->is_line_code_region_start(0)); CHECK(code_edit->is_line_code_region_end(2)); // Update code region delimiter when removing comment delimiter. code_edit->set_text("#region region_name\nline2\n#endregion\n//region region_name\nline2\n//endregion"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("//", ""); code_edit->add_comment_delimiter("#", ""); // A shorter delimiter has higher priority. CHECK(code_edit->is_line_code_region_start(0)); CHECK(code_edit->is_line_code_region_end(2)); CHECK_FALSE(code_edit->is_line_code_region_start(3)); CHECK_FALSE(code_edit->is_line_code_region_end(5)); code_edit->remove_comment_delimiter("#"); CHECK_FALSE(code_edit->is_line_code_region_start(0)); CHECK_FALSE(code_edit->is_line_code_region_end(2)); CHECK(code_edit->is_line_code_region_start(3)); CHECK(code_edit->is_line_code_region_end(5)); // Update code region delimiter when clearing comment delimiters. code_edit->set_text("//region region_name\nline2\n//endregion"); code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("//", ""); CHECK(code_edit->is_line_code_region_start(0)); CHECK(code_edit->is_line_code_region_end(2)); code_edit->clear_comment_delimiters(); CHECK_FALSE(code_edit->is_line_code_region_start(0)); CHECK_FALSE(code_edit->is_line_code_region_end(2)); } SUBCASE("[CodeEdit] fold region") { code_edit->set_line_folding_enabled(true); // Fold region. code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->set_text("#region region_name\nline2\nline3\n#endregion\nvisible line"); CHECK(code_edit->can_fold_line(0)); for (int i = 1; i < 5; i++) { CHECK_FALSE(code_edit->can_fold_line(i)); } for (int i = 0; i < 5; i++) { CHECK_FALSE(code_edit->is_line_folded(i)); } code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); // Region with no end can't be folded. ERR_PRINT_OFF; code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->set_text("#region region_name\nline2\nline3\n#bad_end_tag\nvisible line"); CHECK_FALSE(code_edit->can_fold_line(0)); ERR_PRINT_ON; // Bad nested region can't be folded. ERR_PRINT_OFF; code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->set_text("#region without end\n#region region2\nline3\n#endregion\n#no_end"); CHECK_FALSE(code_edit->can_fold_line(0)); CHECK(code_edit->can_fold_line(1)); ERR_PRINT_ON; // Nested region folding. ERR_PRINT_OFF; code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->set_text("#region region1\n#region region2\nline3\n#endregion\n#endregion"); CHECK(code_edit->can_fold_line(0)); CHECK(code_edit->can_fold_line(1)); code_edit->fold_line(1); CHECK(code_edit->get_next_visible_line_offset_from(2, 1) == 3); code_edit->fold_line(0); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); ERR_PRINT_ON; // Unfolding a line inside a region unfold whole region. code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->set_text("#region region\ninside\nline3\n#endregion\nvisible"); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 4); code_edit->unfold_line(1); CHECK_FALSE(code_edit->is_line_folded(0)); // Region start and end tags are ignored if in a string and at the start of the line. code_edit->clear_comment_delimiters(); code_edit->add_comment_delimiter("#", ""); code_edit->clear_string_delimiters(); code_edit->add_string_delimiter("\"", "\""); code_edit->set_text("#region region_name1\nline2\n\"\n#region region_name2\n#endregion\n\"\n#endregion\nvisible"); CHECK(code_edit->is_line_code_region_start(0)); CHECK(code_edit->is_line_code_region_end(6)); CHECK(code_edit->can_fold_line(0)); for (int i = 1; i < 7; i++) { if (i == 2) { continue; } CHECK_FALSE(code_edit->can_fold_line(i)); } for (int i = 0; i < 7; i++) { CHECK_FALSE(code_edit->is_line_folded(i)); } code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); CHECK(code_edit->get_next_visible_line_offset_from(1, 1) == 7); } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] completion") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); SUBCASE("[CodeEdit] auto brace completion") { code_edit->set_auto_brace_completion_enabled(true); CHECK(code_edit->is_auto_brace_completion_enabled()); code_edit->set_highlight_matching_braces_enabled(true); CHECK(code_edit->is_highlight_matching_braces_enabled()); /* Try setters, any length. */ Dictionary auto_brace_completion_pairs; auto_brace_completion_pairs["["] = "]"; auto_brace_completion_pairs["'"] = "'"; auto_brace_completion_pairs[";"] = "'"; auto_brace_completion_pairs["'''"] = "'''"; code_edit->set_auto_brace_completion_pairs(auto_brace_completion_pairs); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); CHECK(code_edit->get_auto_brace_completion_pairs()["["] == "]"); CHECK(code_edit->get_auto_brace_completion_pairs()["'"] == "'"); CHECK(code_edit->get_auto_brace_completion_pairs()[";"] == "'"); CHECK(code_edit->get_auto_brace_completion_pairs()["'''"] == "'''"); ERR_PRINT_OFF; /* No duplicate start keys. */ code_edit->add_auto_brace_completion_pair("[", "]"); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); /* No empty keys. */ code_edit->add_auto_brace_completion_pair("[", ""); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); code_edit->add_auto_brace_completion_pair("", "]"); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); code_edit->add_auto_brace_completion_pair("", ""); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); /* Must be a symbol. */ code_edit->add_auto_brace_completion_pair("a", "]"); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); code_edit->add_auto_brace_completion_pair("[", "a"); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); code_edit->add_auto_brace_completion_pair("a", "a"); CHECK(code_edit->get_auto_brace_completion_pairs().size() == 4); ERR_PRINT_ON; /* Check metadata. */ CHECK(code_edit->has_auto_brace_completion_open_key("[")); CHECK(code_edit->has_auto_brace_completion_open_key("'")); CHECK(code_edit->has_auto_brace_completion_open_key(";")); CHECK(code_edit->has_auto_brace_completion_open_key("'''")); CHECK_FALSE(code_edit->has_auto_brace_completion_open_key("(")); CHECK(code_edit->has_auto_brace_completion_close_key("]")); CHECK(code_edit->has_auto_brace_completion_close_key("'")); CHECK(code_edit->has_auto_brace_completion_close_key("'''")); CHECK_FALSE(code_edit->has_auto_brace_completion_close_key(")")); CHECK(code_edit->get_auto_brace_completion_close_key("[") == "]"); CHECK(code_edit->get_auto_brace_completion_close_key("'") == "'"); CHECK(code_edit->get_auto_brace_completion_close_key(";") == "'"); CHECK(code_edit->get_auto_brace_completion_close_key("'''") == "'''"); CHECK(code_edit->get_auto_brace_completion_close_key("(").is_empty()); /* Check typing inserts closing pair. */ code_edit->clear(); SEND_GUI_KEY_EVENT(Key::BRACKETLEFT); CHECK(code_edit->get_line(0) == "[]"); /* Should first match and insert smaller key. */ code_edit->clear(); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); CHECK(code_edit->get_line(0) == "''"); CHECK(code_edit->get_caret_column() == 1); /* Move out from center, Should match and insert larger key. */ SEND_GUI_ACTION("ui_text_caret_right"); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); CHECK(code_edit->get_line(0) == "''''''"); CHECK(code_edit->get_caret_column() == 3); /* Backspace should remove all. */ SEND_GUI_ACTION("ui_text_backspace"); CHECK(code_edit->get_line(0).is_empty()); /* If in between and typing close key should "skip". */ SEND_GUI_KEY_EVENT(Key::BRACKETLEFT); CHECK(code_edit->get_line(0) == "[]"); CHECK(code_edit->get_caret_column() == 1); SEND_GUI_KEY_EVENT(Key::BRACKETRIGHT); CHECK(code_edit->get_line(0) == "[]"); CHECK(code_edit->get_caret_column() == 2); /* If current is char and inserting a string, do not autocomplete. */ code_edit->clear(); SEND_GUI_KEY_EVENT(Key::A); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); CHECK(code_edit->get_line(0) == "A'"); /* If in comment, do not complete. */ code_edit->add_comment_delimiter("#", ""); code_edit->clear(); SEND_GUI_KEY_EVENT(Key::NUMBERSIGN); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); CHECK(code_edit->get_line(0) == "#'"); /* If in string, and inserting string do not complete. */ code_edit->clear(); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); SEND_GUI_KEY_EVENT(Key::QUOTEDBL); CHECK(code_edit->get_line(0) == "'\"'"); /* Wrap single line selection with brackets */ code_edit->clear(); code_edit->insert_text_at_caret("abc"); code_edit->select_all(); SEND_GUI_KEY_EVENT(Key::BRACKETLEFT); CHECK(code_edit->get_line(0) == "[abc]"); /* Caret should be after the last character of the single line selection */ CHECK(code_edit->get_caret_column() == 4); /* Wrap multi line selection with brackets */ code_edit->clear(); code_edit->insert_text_at_caret("abc\nabc"); code_edit->select_all(); SEND_GUI_KEY_EVENT(Key::BRACKETLEFT); CHECK(code_edit->get_text() == "[abc\nabc]"); /* Caret should be after the last character of the multi line selection */ CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 3); /* If inserted character is not a auto brace completion open key, replace selected text with the inserted character */ code_edit->clear(); code_edit->insert_text_at_caret("abc"); code_edit->select_all(); SEND_GUI_KEY_EVENT(Key::KEY_1); CHECK(code_edit->get_text() == "1"); /* If potential multichar and single brace completion is matched, it should wrap the single. */ code_edit->clear(); code_edit->insert_text_at_caret("\'\'abc"); code_edit->select(0, 2, 0, 5); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); CHECK(code_edit->get_text() == "\'\'\'abc\'"); /* If only the potential multichar brace completion is matched, it does not wrap or complete. */ auto_brace_completion_pairs.erase("\'"); code_edit->set_auto_brace_completion_pairs(auto_brace_completion_pairs); CHECK_FALSE(code_edit->has_auto_brace_completion_open_key("\'")); code_edit->clear(); code_edit->insert_text_at_caret("\'\'abc"); code_edit->select(0, 2, 0, 5); SEND_GUI_KEY_EVENT(Key::APOSTROPHE); CHECK(code_edit->get_text() == "\'\'\'"); } SUBCASE("[CodeEdit] autocomplete with brace completion") { code_edit->set_auto_brace_completion_enabled(true); CHECK(code_edit->is_auto_brace_completion_enabled()); code_edit->insert_text_at_caret("(te)"); code_edit->set_caret_column(3); // Full completion. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_FUNCTION, "test()", "test()"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "(test())"); CHECK(code_edit->get_caret_column() == 7); code_edit->undo(); // With "arg". code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_FUNCTION, "test(", "test("); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "(test())"); CHECK(code_edit->get_caret_column() == 6); code_edit->undo(); // brace completion disabled code_edit->set_auto_brace_completion_enabled(false); // Full completion. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_FUNCTION, "test()", "test()"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "(test())"); CHECK(code_edit->get_caret_column() == 7); code_edit->undo(); // With "arg". code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_FUNCTION, "test(", "test("); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "(test()"); CHECK(code_edit->get_caret_column() == 6); // String code_edit->set_auto_brace_completion_enabled(true); code_edit->clear(); code_edit->insert_text_at_caret("\"\""); code_edit->set_caret_column(1); // Full completion. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_NODE_PATH, "\"test\"", "\"test\""); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "\"test\""); CHECK(code_edit->get_caret_column() == 6); code_edit->undo(); // With "arg". code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_NODE_PATH, "\"test", "\"test"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "\"\"test\""); CHECK(code_edit->get_caret_column() == 7); code_edit->undo(); // brace completion disabled code_edit->set_auto_brace_completion_enabled(false); // Full completion. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_NODE_PATH, "\"test\"", "\"test\""); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "\"test\""); CHECK(code_edit->get_caret_column() == 6); code_edit->undo(); // With "arg". code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_NODE_PATH, "\"test", "\"test"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "\"\"test\""); CHECK(code_edit->get_caret_column() == 7); code_edit->undo(); } SUBCASE("[CodeEdit] autocomplete") { code_edit->set_code_completion_enabled(true); CHECK(code_edit->is_code_completion_enabled()); /* Set prefixes, single char only, disallow empty. */ TypedArray completion_prefixes; completion_prefixes.push_back(""); completion_prefixes.push_back("."); completion_prefixes.push_back("."); completion_prefixes.push_back(",,"); ERR_PRINT_OFF; code_edit->set_code_completion_prefixes(completion_prefixes); ERR_PRINT_ON; completion_prefixes = code_edit->get_code_completion_prefixes(); CHECK(completion_prefixes.size() == 2); CHECK(completion_prefixes.has(".")); CHECK(completion_prefixes.has(",")); code_edit->set_text("test\ntest"); CHECK(code_edit->get_text_for_code_completion() == String::chr(0xFFFF) + "test\ntest"); } SUBCASE("[CodeEdit] autocomplete request") { SIGNAL_WATCH(code_edit, "code_completion_requested"); code_edit->set_code_completion_enabled(true); Array signal_args; signal_args.push_back(Array()); /* Force request. */ code_edit->request_code_completion(); SIGNAL_CHECK_FALSE("code_completion_requested"); code_edit->request_code_completion(true); SIGNAL_CHECK("code_completion_requested", signal_args); /* Manual request should force. */ SEND_GUI_ACTION("ui_text_completion_query"); SIGNAL_CHECK("code_completion_requested", signal_args); /* Insert prefix. */ TypedArray completion_prefixes; completion_prefixes.push_back("."); code_edit->set_code_completion_prefixes(completion_prefixes); code_edit->insert_text_at_caret("."); code_edit->request_code_completion(); SIGNAL_CHECK("code_completion_requested", signal_args); /* Should work with space too. */ code_edit->insert_text_at_caret(" "); code_edit->request_code_completion(); SIGNAL_CHECK("code_completion_requested", signal_args); /* Should work when complete ends with prefix. */ code_edit->clear(); code_edit->insert_text_at_caret("t"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "test.", "test."); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "test."); SIGNAL_CHECK("code_completion_requested", signal_args); SIGNAL_UNWATCH(code_edit, "code_completion_requested"); } SUBCASE("[CodeEdit] autocomplete completion") { if (TS->has_feature(TextServer::FEATURE_FONT_DYNAMIC) && TS->has_feature(TextServer::FEATURE_SIMPLE_LAYOUT)) { CHECK(code_edit->get_code_completion_selected_index() == -1); code_edit->set_code_completion_enabled(true); CHECK(code_edit->get_code_completion_selected_index() == -1); code_edit->update_code_completion_options(); code_edit->set_code_completion_selected_index(1); CHECK(code_edit->get_code_completion_selected_index() == -1); CHECK(code_edit->get_code_completion_option(0).size() == 0); CHECK(code_edit->get_code_completion_options().size() == 0); /* Adding does not update the list. */ code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_0.", "item_0"); code_edit->set_code_completion_selected_index(1); CHECK(code_edit->get_code_completion_selected_index() == -1); CHECK(code_edit->get_code_completion_option(0).size() == 0); CHECK(code_edit->get_code_completion_options().size() == 0); /* After update, pending add should not be counted, */ /* also does not work on col 0 */ int before_text_caret_column = code_edit->get_caret_column(); code_edit->insert_text_at_caret("i"); code_edit->update_code_completion_options(); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0", Color(1, 0, 0), Ref(), Color(1, 0, 0)); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_1.", "item_1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_2.", "item_2"); ERR_PRINT_OFF; code_edit->set_code_completion_selected_index(1); ERR_PRINT_ON; CHECK(code_edit->get_code_completion_selected_index() == 0); CHECK(code_edit->get_code_completion_option(0).size() == 7); CHECK(code_edit->get_code_completion_options().size() == 1); /* Check cancel closes completion. */ SEND_GUI_ACTION("ui_cancel"); CHECK(code_edit->get_code_completion_selected_index() == -1); code_edit->update_code_completion_options(); CHECK(code_edit->get_code_completion_selected_index() == 0); code_edit->set_code_completion_selected_index(1); CHECK(code_edit->get_code_completion_selected_index() == 1); CHECK(code_edit->get_code_completion_option(0).size() == 7); CHECK(code_edit->get_code_completion_options().size() == 3); /* Check data. */ Dictionary option = code_edit->get_code_completion_option(0); CHECK((int)option["kind"] == (int)CodeEdit::CodeCompletionKind::KIND_CLASS); CHECK(option["display_text"] == "item_0."); CHECK(option["insert_text"] == "item_0"); CHECK(option["font_color"] == Color(1, 0, 0)); CHECK(option["icon"] == Ref()); CHECK(option["default_value"] == Color(1, 0, 0)); /* Set size for mouse input. */ code_edit->set_size(Size2(100, 100)); /* Test home and end keys close the completion and move the caret */ /* => ui_text_caret_line_start */ code_edit->set_caret_column(before_text_caret_column + 1); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0", Color(1, 0, 0), Ref(), Color(1, 0, 0)); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_1.", "item_1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_2.", "item_2"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_caret_line_start"); code_edit->update_code_completion_options(); CHECK(code_edit->get_code_completion_selected_index() == -1); CHECK(code_edit->get_caret_column() == 0); /* => ui_text_caret_line_end */ code_edit->set_caret_column(before_text_caret_column + 1); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0", Color(1, 0, 0), Ref(), Color(1, 0, 0)); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_1.", "item_1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_2.", "item_2"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_caret_line_end"); code_edit->update_code_completion_options(); CHECK(code_edit->get_code_completion_selected_index() == -1); CHECK(code_edit->get_caret_column() == before_text_caret_column + 1); /* Check input. */ code_edit->set_caret_column(before_text_caret_column + 1); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0", Color(1, 0, 0), Ref(), Color(1, 0, 0)); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_1.", "item_1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "item_2.", "item_2"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_page_down"); CHECK(code_edit->get_code_completion_selected_index() == 2); SEND_GUI_ACTION("ui_page_up"); CHECK(code_edit->get_code_completion_selected_index() == 0); SEND_GUI_ACTION("ui_up"); CHECK(code_edit->get_code_completion_selected_index() == 2); SEND_GUI_ACTION("ui_down"); CHECK(code_edit->get_code_completion_selected_index() == 0); SEND_GUI_KEY_EVENT(Key::T); CHECK(code_edit->get_code_completion_selected_index() == 0); SEND_GUI_ACTION("ui_left"); CHECK(code_edit->get_code_completion_selected_index() == 0); SEND_GUI_ACTION("ui_right"); CHECK(code_edit->get_code_completion_selected_index() == 0); SEND_GUI_ACTION("ui_text_backspace"); CHECK(code_edit->get_code_completion_selected_index() == 0); Point2 caret_pos = code_edit->get_caret_draw_pos(); caret_pos.y += code_edit->get_line_height(); SEND_GUI_MOUSE_BUTTON_EVENT(caret_pos, MouseButton::WHEEL_DOWN, 0, Key::NONE); CHECK(code_edit->get_code_completion_selected_index() == 1); SEND_GUI_MOUSE_BUTTON_EVENT(caret_pos, MouseButton::WHEEL_UP, 0, Key::NONE); CHECK(code_edit->get_code_completion_selected_index() == 0); /* Single click selects. */ caret_pos.y += code_edit->get_line_height() * 2; SEND_GUI_MOUSE_BUTTON_EVENT(caret_pos, MouseButton::LEFT, MouseButtonMask::LEFT, Key::NONE); SEND_GUI_MOUSE_BUTTON_RELEASED_EVENT(caret_pos, MouseButton::LEFT, MouseButtonMask::NONE, Key::NONE); CHECK(code_edit->get_code_completion_selected_index() == 2); /* Double click inserts. */ SEND_GUI_DOUBLE_CLICK(caret_pos, Key::NONE); CHECK(code_edit->get_code_completion_selected_index() == -1); CHECK(code_edit->get_line(0) == "item_2"); code_edit->set_auto_brace_completion_enabled(false); /* Does nothing in readonly. */ code_edit->undo(); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0"); code_edit->update_code_completion_options(); code_edit->set_editable(false); code_edit->confirm_code_completion(); code_edit->set_editable(true); CHECK(code_edit->get_line(0) == "i"); /* Replace */ code_edit->clear(); code_edit->insert_text_at_caret("item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0 test"); /* Replace string. */ code_edit->clear(); code_edit->insert_text_at_caret("\"item_1 test\""); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "\"item_0\""); /* Normal replace if no end is given. */ code_edit->clear(); code_edit->insert_text_at_caret("\"item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "\"item_0\" test"); /* Insert at completion. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_accept"); CHECK(code_edit->get_line(0) == "item_01 test"); /* Insert at completion with string should have same output. */ code_edit->clear(); code_edit->insert_text_at_caret("\"item_1 test\""); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0.", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_accept"); CHECK(code_edit->get_line(0) == "\"item_0\"1 test\""); /* Merge symbol at end on insert text. */ /* End on completion entry. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0(", "item_0("); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0( test"); CHECK(code_edit->get_caret_column() == 7); /* End of text*/ code_edit->clear(); code_edit->insert_text_at_caret("item_1( test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0( test"); CHECK(code_edit->get_caret_column() == 6); /* End of both. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1( test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0(", "item_0("); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0( test"); CHECK(code_edit->get_caret_column() == 7); /* Full set. */ /* End on completion entry. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0()", "item_0()"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 8); /* End of text*/ code_edit->clear(); code_edit->insert_text_at_caret("item_1() test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 6); /* End of both. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1() test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0()", "item_0()"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 8); /* Autobrace completion. */ code_edit->set_auto_brace_completion_enabled(true); /* End on completion entry. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0(", "item_0("); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 7); /* End of text*/ code_edit->clear(); code_edit->insert_text_at_caret("item_1( test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0( test"); CHECK(code_edit->get_caret_column() == 6); /* End of both. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1( test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0(", "item_0("); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0( test"); CHECK(code_edit->get_caret_column() == 7); /* Full set. */ /* End on completion entry. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1 test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0()", "item_0()"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 8); /* End of text*/ code_edit->clear(); code_edit->insert_text_at_caret("item_1() test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0", "item_0"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 6); /* End of both. */ code_edit->clear(); code_edit->insert_text_at_caret("item_1() test"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_CLASS, "item_0()", "item_0()"); code_edit->update_code_completion_options(); SEND_GUI_ACTION("ui_text_completion_replace"); CHECK(code_edit->get_line(0) == "item_0() test"); CHECK(code_edit->get_caret_column() == 8); } } SUBCASE("[CodeEdit] autocomplete suggestion order") { /* Prefer less fragmented suggestion. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "test", "test"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "tset", "tset"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "test"); /* Prefer suggestion starting with the string to complete (matching start). */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "test", "test"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "stest", "stest"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "test"); /* Prefer less fragment over matching start. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "tset", "tset"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "stest", "stest"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "stest"); /* Prefer good capitalization. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "test", "test"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "Test", "Test"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "test"); /* Prefer matching start over good capitalization. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "Test", "Test"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "stest_bis", "test_bis"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "Test"); /* Prefer closer location. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "test", "test"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "test_bis", "test_bis", Color(1, 1, 1), Ref(), Variant::NIL, CodeEdit::LOCATION_LOCAL); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "test_bis"); /* Prefer good capitalization over location. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "test", "test"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "Test", "Test", Color(1, 1, 1), Ref(), Variant::NIL, CodeEdit::LOCATION_LOCAL); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "test"); /* Prefer the start of the string to complete being closest to the start of the suggestion (closest to start). */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "stest", "stest"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "sstest", "sstest"); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "stest"); /* Prefer location over closest to start. */ code_edit->clear(); code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "stest", "stest"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "sstest", "sstest", Color(1, 1, 1), Ref(), Variant::NIL, CodeEdit::LOCATION_LOCAL); code_edit->update_code_completion_options(); code_edit->confirm_code_completion(); CHECK(code_edit->get_line(0) == "sstest"); } SUBCASE("[CodeEdit] autocomplete currently selected option") { code_edit->set_code_completion_enabled(true); REQUIRE(code_edit->is_code_completion_enabled()); // Initially select item 0. code_edit->insert_text_at_caret("te"); code_edit->set_caret_column(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te1", "te1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 0, "Initially selected item should be 0."); // After adding later options shouldn't update selection. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te1", "te1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te4", "te4"); // Added te4. code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 0, "Adding later options shouldn't update selection."); code_edit->set_code_completion_selected_index(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te1", "te1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te4", "te4"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te5", "te5"); // Added te5. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te6", "te6"); // Added te6. code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 2, "Adding later options shouldn't update selection."); // Removing elements after selected element shouldn't update selection. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te1", "te1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te4", "te4"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te5", "te5"); // Removed te6. code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 2, "Removing elements after selected element shouldn't update selection."); // Changing elements after selected element shouldn't update selection. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te1", "te1"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te4", "te4"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te6", "te6"); // Changed te5->te6. code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 2, "Changing elements after selected element shouldn't update selection."); // Changing elements before selected element should reset selection. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te2", "te2"); // Changed te1->te2. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te4", "te4"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te6", "te6"); code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 0, "Changing elements before selected element should reset selection."); // Removing elements before selected element should reset selection. code_edit->set_code_completion_selected_index(2); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te3", "te3"); // Removed te2. code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te4", "te4"); code_edit->add_code_completion_option(CodeEdit::CodeCompletionKind::KIND_VARIABLE, "te6", "te6"); code_edit->update_code_completion_options(); CHECK_MESSAGE(code_edit->get_code_completion_selected_index() == 0, "Removing elements before selected element should reset selection."); } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] symbol lookup") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); code_edit->set_symbol_lookup_on_click_enabled(true); CHECK(code_edit->is_symbol_lookup_on_click_enabled()); if (TS->has_feature(TextServer::FEATURE_FONT_DYNAMIC) && TS->has_feature(TextServer::FEATURE_SIMPLE_LAYOUT)) { /* Set size for mouse input. */ code_edit->set_size(Size2(100, 100)); code_edit->set_text("this is some text"); Point2 caret_pos = code_edit->get_caret_draw_pos(); caret_pos.x += 60; SEND_GUI_MOUSE_BUTTON_EVENT(caret_pos, MouseButton::NONE, 0, Key::NONE); CHECK(code_edit->get_text_for_symbol_lookup() == "this is s" + String::chr(0xFFFF) + "ome text"); SIGNAL_WATCH(code_edit, "symbol_validate"); #ifdef MACOS_ENABLED SEND_GUI_KEY_EVENT(Key::META); #else SEND_GUI_KEY_EVENT(Key::CTRL); #endif Array signal_args = build_array(build_array("some")); SIGNAL_CHECK("symbol_validate", signal_args); SIGNAL_UNWATCH(code_edit, "symbol_validate"); } memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] line length guidelines") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); TypedArray guide_lines; code_edit->set_line_length_guidelines(guide_lines); CHECK(code_edit->get_line_length_guidelines().size() == 0); guide_lines.push_back(80); guide_lines.push_back(120); /* Order should be preserved. */ code_edit->set_line_length_guidelines(guide_lines); CHECK((int)code_edit->get_line_length_guidelines()[0] == 80); CHECK((int)code_edit->get_line_length_guidelines()[1] == 120); memdelete(code_edit); } TEST_CASE("[SceneTree][CodeEdit] text manipulation") { CodeEdit *code_edit = memnew(CodeEdit); SceneTree::get_singleton()->get_root()->add_child(code_edit); code_edit->grab_focus(); SUBCASE("[SceneTree][CodeEdit] backspace") { // Backspace with selection on first line. code_edit->set_text("test backspace"); code_edit->select(0, 0, 0, 5); code_edit->backspace(); CHECK(code_edit->get_line(0) == "backspace"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Backspace with selection on first line and caret at the beginning of file. code_edit->set_text("test backspace"); code_edit->select(0, 5, 0, 0); code_edit->backspace(); CHECK(code_edit->get_line(0) == "backspace"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Move caret up to the previous line on backspace if caret is at the first column. code_edit->set_text("line 1\nline 2"); code_edit->set_caret_line(1); code_edit->set_caret_column(0); code_edit->backspace(); CHECK(code_edit->get_line(0) == "line 1line 2"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 6); // Multiple carets with a caret at the first column. code_edit->set_text("line 1\nline 2"); code_edit->set_caret_line(1); code_edit->set_caret_column(2); code_edit->add_caret(1, 0); code_edit->add_caret(1, 5); code_edit->backspace(); CHECK(code_edit->get_text() == "line 1lne2"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 7); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 6); CHECK(code_edit->get_caret_line(2) == 0); CHECK(code_edit->get_caret_column(2) == 9); code_edit->remove_secondary_carets(); // Multiple carets close together. code_edit->set_text("line 1\nline 2"); code_edit->set_caret_line(1); code_edit->set_caret_column(2); code_edit->add_caret(1, 1); code_edit->backspace(); CHECK(code_edit->get_text() == "line 1\nne 2"); CHECK(code_edit->get_caret_count() == 1); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Backspace delete all text if all text is selected. code_edit->set_text("line 1\nline 2\nline 3"); code_edit->select_all(); code_edit->backspace(); CHECK(code_edit->get_text().is_empty()); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Backspace at the beginning without selection has no effect. code_edit->set_text("line 1\nline 2\nline 3"); code_edit->set_caret_line(0); code_edit->set_caret_column(0); code_edit->backspace(); CHECK(code_edit->get_text() == "line 1\nline 2\nline 3"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Unfold previous folded line on backspace if the caret is at the first column. code_edit->set_line_folding_enabled(true); code_edit->set_text("line 1\n\tline 2\nline 3"); code_edit->set_caret_line(2); code_edit->set_caret_column(0); code_edit->fold_line(0); code_edit->backspace(); CHECK_FALSE(code_edit->is_line_folded(0)); code_edit->set_line_folding_enabled(false); // Do not unfold previous line on backspace if the caret is not at the first column. code_edit->set_line_folding_enabled(true); code_edit->set_text("line 1\n\tline 2\nline 3"); code_edit->set_caret_line(2); code_edit->set_caret_column(4); code_edit->fold_line(0); code_edit->backspace(); CHECK(code_edit->is_line_folded(0)); code_edit->set_line_folding_enabled(false); } SUBCASE("[TextEdit] cut") { DisplayServerMock *DS = (DisplayServerMock *)(DisplayServer::get_singleton()); code_edit->set_line_folding_enabled(true); // Cut without a selection removes the entire line. code_edit->set_text("this is\nsome\n"); code_edit->set_caret_line(0); code_edit->set_caret_column(6); code_edit->cut(); CHECK(DS->clipboard_get() == "this is\n"); CHECK(code_edit->get_text() == "some\n"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 3); // In the default font, this is the same position. // Undo restores the cut text. code_edit->undo(); CHECK(DS->clipboard_get() == "this is\n"); CHECK(code_edit->get_text() == "this is\nsome\n"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 6); // Redo. code_edit->redo(); CHECK(DS->clipboard_get() == "this is\n"); CHECK(code_edit->get_text() == "some\n"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 3); // Cut unfolds the line. code_edit->set_text("this is\n\tsome\n"); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); code_edit->cut(); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK(DS->clipboard_get() == "this is\n"); CHECK(code_edit->get_text() == "\tsome\n"); CHECK(code_edit->get_caret_line() == 0); // Cut with a selection removes just the selection. code_edit->set_text("this is\nsome\n"); code_edit->select(0, 5, 0, 7); SEND_GUI_ACTION("ui_cut"); CHECK(code_edit->get_viewport()->is_input_handled()); CHECK(DS->clipboard_get() == "is"); CHECK(code_edit->get_text() == "this \nsome\n"); CHECK_FALSE(code_edit->get_caret_line()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 5); // Cut does not change the text if not editable. Text is still added to clipboard. code_edit->set_text("this is\nsome\n"); code_edit->set_caret_line(0); code_edit->set_caret_column(5); code_edit->set_editable(false); code_edit->cut(); code_edit->set_editable(true); CHECK(DS->clipboard_get() == "this is\n"); CHECK(code_edit->get_text() == "this is\nsome\n"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 5); // Cut line with multiple carets. code_edit->set_text("this is\nsome\n"); code_edit->set_caret_line(0); code_edit->set_caret_column(3); code_edit->add_caret(0, 2); code_edit->add_caret(0, 4); code_edit->add_caret(2, 0); code_edit->cut(); CHECK(DS->clipboard_get() == "this is\n\n"); CHECK(code_edit->get_text() == "some"); CHECK(code_edit->get_caret_count() == 3); CHECK_FALSE(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 2); // In the default font, this is the same position. // The previous caret at index 1 was merged. CHECK_FALSE(code_edit->has_selection(1)); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 3); // In the default font, this is the same position. CHECK_FALSE(code_edit->has_selection(2)); CHECK(code_edit->get_caret_line(2) == 0); CHECK(code_edit->get_caret_column(2) == 4); code_edit->remove_secondary_carets(); // Cut on the only line removes the contents. code_edit->set_caret_line(0); code_edit->set_caret_column(2); code_edit->cut(); CHECK(DS->clipboard_get() == "some\n"); CHECK(code_edit->get_text() == ""); CHECK(code_edit->get_line_count() == 1); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Cut empty line. code_edit->cut(); CHECK(DS->clipboard_get() == "\n"); CHECK(code_edit->get_text() == ""); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Cut multiple lines, in order. code_edit->set_text("this is\nsome\ntext to\nbe\n\ncut"); code_edit->set_caret_line(2); code_edit->set_caret_column(7); code_edit->add_caret(3, 0); code_edit->add_caret(0, 2); code_edit->cut(); CHECK(DS->clipboard_get() == "this is\ntext to\nbe\n"); CHECK(code_edit->get_text() == "some\n\ncut"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 1); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 2); code_edit->remove_secondary_carets(); // Cut multiple selections, in order. Ignores regular carets. code_edit->set_text("this is\nsome\ntext to\nbe\n\ncut"); code_edit->add_caret(3, 0); code_edit->add_caret(0, 2); code_edit->add_caret(2, 0); code_edit->select(1, 0, 1, 2, 0); code_edit->select(3, 0, 4, 0, 1); code_edit->select(0, 5, 0, 3, 2); code_edit->cut(); CHECK(DS->clipboard_get() == "s \nso\nbe\n"); CHECK(code_edit->get_text() == "thiis\nme\ntext to\n\ncut"); CHECK(code_edit->get_caret_count() == 4); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line(0) == 1); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 3); CHECK(code_edit->get_caret_column(1) == 0); CHECK(code_edit->get_caret_line(2) == 0); CHECK(code_edit->get_caret_column(2) == 3); CHECK(code_edit->get_caret_line(3) == 2); CHECK(code_edit->get_caret_column(3) == 0); } SUBCASE("[SceneTree][CodeEdit] cut when empty selection clipboard disabled") { DisplayServerMock *DS = (DisplayServerMock *)(DisplayServer::get_singleton()); code_edit->set_empty_selection_clipboard_enabled(false); DS->clipboard_set(""); code_edit->set_text("this is\nsome\n"); code_edit->set_caret_line(0); code_edit->set_caret_column(6); MessageQueue::get_singleton()->flush(); SIGNAL_DISCARD("text_set"); SIGNAL_DISCARD("text_changed"); SIGNAL_DISCARD("lines_edited_from"); SIGNAL_DISCARD("caret_changed"); code_edit->cut(); MessageQueue::get_singleton()->flush(); CHECK(DS->clipboard_get() == ""); CHECK(code_edit->get_text() == "this is\nsome\n"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 6); SIGNAL_CHECK_FALSE("caret_changed"); SIGNAL_CHECK_FALSE("text_changed"); SIGNAL_CHECK_FALSE("lines_edited_from"); } SUBCASE("[SceneTree][CodeEdit] new line") { // Add a new line. code_edit->set_text("test new line"); code_edit->set_caret_line(0); code_edit->set_caret_column(13); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test new line"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Split line with new line. code_edit->set_text("test new line"); code_edit->set_caret_line(0); code_edit->set_caret_column(5); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test "); CHECK(code_edit->get_line(1) == "new line"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Delete selection and split with new line. code_edit->set_text("test new line"); code_edit->select(0, 0, 0, 5); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == "new line"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Blank new line below with selection should not split. code_edit->set_text("test new line"); code_edit->select(0, 0, 0, 5); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "test new line"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Blank new line above with selection should not split. code_edit->set_text("test new line"); code_edit->select(0, 0, 0, 5); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == "test new line"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Multiple new lines with multiple carets. code_edit->set_text("test new line"); code_edit->set_caret_line(0); code_edit->set_caret_column(5); code_edit->add_caret(0, 8); SEND_GUI_ACTION("ui_text_newline"); CHECK(code_edit->get_line(0) == "test "); CHECK(code_edit->get_line(1) == "new"); CHECK(code_edit->get_line(2) == " line"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 1); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 2); CHECK(code_edit->get_caret_column(1) == 0); // Multiple blank new lines with multiple carets. code_edit->set_text("test new line"); code_edit->remove_secondary_carets(); code_edit->set_caret_line(0); code_edit->set_caret_column(5); code_edit->add_caret(0, 8); SEND_GUI_ACTION("ui_text_newline_blank"); CHECK(code_edit->get_line(0) == "test new line"); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_line(2) == ""); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 2); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 0); // Multiple new lines above with multiple carets. code_edit->set_text("test new line"); code_edit->remove_secondary_carets(); code_edit->set_caret_line(0); code_edit->set_caret_column(5); code_edit->add_caret(0, 8); SEND_GUI_ACTION("ui_text_newline_above"); CHECK(code_edit->get_line(0) == ""); CHECK(code_edit->get_line(1) == ""); CHECK(code_edit->get_line(2) == "test new line"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 0); // See '[CodeEdit] auto indent' tests for tests about new line with indentation. } SUBCASE("[SceneTree][CodeEdit] move lines up") { code_edit->set_text("test\nlines\nto\n\nmove\naround"); // Move line up with caret on it. code_edit->set_caret_line(2); code_edit->set_caret_column(1); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nto\nlines\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Undo. code_edit->undo(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 1); // Redo. code_edit->redo(); CHECK(code_edit->get_text() == "test\nto\nlines\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Does nothing at the first line. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->set_caret_line(0); code_edit->set_caret_column(1); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 1); // Does nothing at the first line when selection ends at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(0, 0, 1, 0); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nmove\naround"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Works on empty line. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->set_caret_line(3); code_edit->set_caret_column(0); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\n\nto\nmove\naround"); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 0); // Move multiple lines up with selection. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(4, 0, 5, 1); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\nto\nmove\naround\n"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 3); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 4); CHECK(code_edit->get_caret_column() == 1); // Does not affect line with selection end at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(4, 0, 5, 0); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\nto\nmove\n\naround"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 3); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 4); CHECK(code_edit->get_caret_column() == 0); // Move multiple lines up with selection, right to left selection. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(5, 2, 4, 1); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\nto\nmove\naround\n"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 4); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_line() == 3); CHECK(code_edit->get_caret_column() == 1); // Move multiple lines with multiple carets. A line with multiple carets is only moved once. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(5, 2, 5, 4); code_edit->add_caret(4, 0); code_edit->add_caret(4, 4); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "test\nlines\nto\nmove\naround\n"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 4); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 4); CHECK(code_edit->get_caret_column(0) == 4); CHECK_FALSE(code_edit->has_selection(1)); CHECK(code_edit->get_caret_line(1) == 3); CHECK(code_edit->get_caret_column(1) == 0); CHECK_FALSE(code_edit->has_selection(2)); CHECK(code_edit->get_caret_line(2) == 3); CHECK(code_edit->get_caret_column(2) == 4); code_edit->remove_secondary_carets(); // Move multiple separate lines with multiple selections. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(2, 2, 1, 4); code_edit->add_caret(5, 0); code_edit->select(5, 0, 5, 1, 1); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "lines\nto\ntest\n\naround\nmove"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 1); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 4); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 4); CHECK(code_edit->get_selection_origin_column(1) == 0); CHECK(code_edit->get_caret_line(1) == 4); CHECK(code_edit->get_caret_column(1) == 1); code_edit->remove_secondary_carets(); // Move lines with adjacent selections that end at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(1, 2, 2, 0); code_edit->add_caret(2, 2); code_edit->select(2, 2, 3, 0, 1); code_edit->move_lines_up(); CHECK(code_edit->get_text() == "lines\nto\ntest\n\nmove\naround"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 0); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 1); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 1); CHECK(code_edit->get_selection_origin_column(1) == 2); CHECK(code_edit->get_caret_line(1) == 2); CHECK(code_edit->get_caret_column(1) == 0); code_edit->remove_secondary_carets(); code_edit->deselect(); code_edit->set_line_folding_enabled(true); // Move line up into a folded region unfolds it. code_edit->set_text("test\n\tline1 test\n\t\tline 2\ntest2"); code_edit->set_caret_line(3); code_edit->set_caret_column(0); code_edit->fold_line(0); CHECK(code_edit->is_line_folded(0)); code_edit->move_lines_up(); CHECK(code_edit->get_caret_count() == 1); CHECK_FALSE(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 0); CHECK(code_edit->get_text() == "test\n\tline1 test\ntest2\n\t\tline 2"); CHECK_FALSE(code_edit->is_line_folded(0)); } SUBCASE("[SceneTree][CodeEdit] move lines down") { code_edit->set_text("test\nlines\nto\n\nmove\naround"); // Move line down with caret on it. code_edit->set_caret_line(1); code_edit->set_caret_column(1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\nto\nlines\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 1); // Undo. code_edit->undo(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Redo. code_edit->redo(); CHECK(code_edit->get_text() == "test\nto\nlines\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 1); // Does nothing at the last line. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->set_caret_line(5); code_edit->set_caret_column(1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nmove\naround"); CHECK(code_edit->get_caret_line() == 5); CHECK(code_edit->get_caret_column() == 1); // Does nothing at the last line when selection ends at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(4, 0, 5, 0); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nmove\naround"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 4); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 5); CHECK(code_edit->get_caret_column() == 0); // Works on empty line. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->set_caret_line(3); code_edit->set_caret_column(0); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\nlines\nto\nmove\n\naround"); CHECK(code_edit->get_caret_line() == 4); CHECK(code_edit->get_caret_column() == 0); // Move multiple lines down with selection. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(1, 0, 2, 1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\n\nlines\nto\nmove\naround"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 2); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 3); CHECK(code_edit->get_caret_column() == 1); // Does not affect line with selection end at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(1, 0, 2, 0); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\nto\nlines\n\nmove\naround"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 2); CHECK(code_edit->get_selection_origin_column() == 0); CHECK(code_edit->get_caret_line() == 3); CHECK(code_edit->get_caret_column() == 0); // Move multiple lines down with selection, right to left selection. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(2, 2, 1, 1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\n\nlines\nto\nmove\naround"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 3); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 1); // Move multiple lines with multiple carets. A line with multiple carets is only moved once. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(1, 2, 1, 4); code_edit->add_caret(0, 0); code_edit->add_caret(0, 1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "to\ntest\nlines\n\nmove\naround"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 2); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 2); CHECK(code_edit->get_caret_column(0) == 4); CHECK_FALSE(code_edit->has_selection(1)); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 0); CHECK_FALSE(code_edit->has_selection(2)); CHECK(code_edit->get_caret_line(2) == 1); CHECK(code_edit->get_caret_column(2) == 1); // Move multiple separate lines with multiple selections. code_edit->remove_secondary_carets(); code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(0, 2, 1, 4); code_edit->add_caret(4, 0); code_edit->select(4, 0, 4, 2, 1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "to\ntest\nlines\n\naround\nmove"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 1); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 2); CHECK(code_edit->get_caret_column(0) == 4); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 5); CHECK(code_edit->get_selection_origin_column(1) == 0); CHECK(code_edit->get_caret_line(1) == 5); CHECK(code_edit->get_caret_column(1) == 2); // Move lines with adjacent selections that end at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(1, 2, 2, 0); code_edit->add_caret(2, 2); code_edit->select(2, 2, 3, 0, 1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "test\n\nlines\nto\nmove\naround"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 2); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 3); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 3); CHECK(code_edit->get_selection_origin_column(1) == 2); CHECK(code_edit->get_caret_line(1) == 4); CHECK(code_edit->get_caret_column(1) == 0); code_edit->remove_secondary_carets(); // Move lines with disconnected adjacent selections that end at column 0. code_edit->set_text("test\nlines\nto\n\nmove\naround"); code_edit->select(0, 2, 1, 0); code_edit->add_caret(2, 2); code_edit->select(2, 0, 3, 0, 1); code_edit->move_lines_down(); CHECK(code_edit->get_text() == "lines\ntest\n\nto\nmove\naround"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 1); CHECK(code_edit->get_selection_origin_column(0) == 2); CHECK(code_edit->get_caret_line(0) == 2); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 3); CHECK(code_edit->get_selection_origin_column(1) == 0); CHECK(code_edit->get_caret_line(1) == 4); CHECK(code_edit->get_caret_column(1) == 0); code_edit->remove_secondary_carets(); code_edit->deselect(); code_edit->set_line_folding_enabled(true); // Move line down into a folded region unfolds it. code_edit->set_text("test\ntest2\n\tline1 test\n\t\tline 2\ntest2"); code_edit->set_caret_line(0); code_edit->set_caret_column(0); code_edit->fold_line(1); CHECK(code_edit->is_line_folded(1)); code_edit->move_lines_down(); CHECK(code_edit->get_caret_count() == 1); CHECK_FALSE(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); CHECK(code_edit->get_text() == "test2\ntest\n\tline1 test\n\t\tline 2\ntest2"); CHECK_FALSE(code_edit->is_line_folded(0)); CHECK_FALSE(code_edit->is_line_folded(1)); } SUBCASE("[SceneTree][CodeEdit] delete lines") { code_edit->set_text("test\nlines\nto\n\ndelete"); // Delete line with caret on it. code_edit->set_caret_line(1); code_edit->set_caret_column(1); code_edit->delete_lines(); CHECK(code_edit->get_text() == "test\nto\n\ndelete"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Undo. code_edit->undo(); CHECK(code_edit->get_text() == "test\nlines\nto\n\ndelete"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Redo. code_edit->redo(); CHECK(code_edit->get_text() == "test\nto\n\ndelete"); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Delete empty line. code_edit->set_caret_line(2); code_edit->set_caret_column(0); code_edit->delete_lines(); CHECK(code_edit->get_text() == "test\nto\ndelete"); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 0); // Deletes only one line when there are multiple carets on it. Carets move down and the column gets clamped. code_edit->set_caret_line(0); code_edit->set_caret_column(0); code_edit->add_caret(0, 1); code_edit->add_caret(0, 4); code_edit->delete_lines(); CHECK(code_edit->get_text() == "to\ndelete"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 0); CHECK(code_edit->get_caret_column(1) == 1); CHECK(code_edit->get_caret_line(2) == 0); CHECK(code_edit->get_caret_column(2) == 2); // Delete multiple lines with selection. code_edit->remove_secondary_carets(); code_edit->set_text("test\nlines\nto\n\ndelete"); code_edit->select(0, 1, 2, 1); code_edit->delete_lines(); CHECK(code_edit->get_text() == "\ndelete"); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Does not affect line with selection end at column 0. code_edit->set_text("test\nlines\nto\n\ndelete"); code_edit->select(0, 1, 1, 0); code_edit->delete_lines(); CHECK(code_edit->get_text() == "lines\nto\n\ndelete"); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); // Delete multiple lines with multiple carets. code_edit->set_text("test\nlines\nto\n\ndelete"); code_edit->set_caret_line(0); code_edit->set_caret_column(2); code_edit->add_caret(1, 0); code_edit->add_caret(4, 5); code_edit->delete_lines(); CHECK(code_edit->get_text() == "to\n"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 0); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 0); // Delete multiple separate lines with multiple selections. code_edit->remove_secondary_carets(); code_edit->set_text("test\nlines\nto\n\ndelete"); code_edit->add_caret(4, 5); code_edit->select(0, 1, 1, 1); code_edit->select(5, 5, 4, 0, 1); code_edit->delete_lines(); CHECK(code_edit->get_text() == "to\n"); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->get_caret_line(0) == 0); CHECK(code_edit->get_caret_column(0) == 1); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 0); // Deletes contents when there is only one line. code_edit->remove_secondary_carets(); code_edit->set_text("test"); code_edit->set_caret_line(0); code_edit->set_caret_column(4); code_edit->delete_lines(); CHECK(code_edit->get_text() == ""); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 0); CHECK(code_edit->get_caret_column() == 0); } SUBCASE("[SceneTree][CodeEdit] duplicate selection") { code_edit->set_text("test\nlines\nto\n\nduplicate"); // Duplicate selected text. code_edit->select(0, 1, 1, 2); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test\nliest\nlines\nto\n\nduplicate"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 2); // Undo. code_edit->undo(); CHECK(code_edit->get_text() == "test\nlines\nto\n\nduplicate"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 0); CHECK(code_edit->get_selection_origin_column() == 1); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 2); // Redo. code_edit->redo(); CHECK(code_edit->get_text() == "test\nliest\nlines\nto\n\nduplicate"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 1); CHECK(code_edit->get_selection_origin_column() == 2); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 2); // Duplicate selected text, right to left selection. code_edit->set_text("test\nlines\nto\n\nduplicate"); code_edit->select(1, 1, 0, 2); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test\nlst\nlines\nto\n\nduplicate"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 2); CHECK(code_edit->get_selection_origin_column() == 1); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 1); // Duplicate line if there is no selection. code_edit->deselect(); code_edit->set_text("test\nlines\nto\n\nduplicate"); code_edit->set_caret_line(1); code_edit->set_caret_column(2); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test\nlines\nlines\nto\n\nduplicate"); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line() == 2); CHECK(code_edit->get_caret_column() == 2); // Duplicate multiple lines. code_edit->deselect(); code_edit->set_text("test\nlines\nto\n\nduplicate"); code_edit->set_caret_line(1); code_edit->set_caret_column(2); code_edit->add_caret(5, 0); code_edit->add_caret(0, 4); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test\ntest\nlines\nlines\nto\n\nduplicate\nduplicate"); CHECK(code_edit->get_caret_count() == 3); CHECK_FALSE(code_edit->has_selection()); CHECK(code_edit->get_caret_line(0) == 3); CHECK(code_edit->get_caret_column(0) == 2); CHECK(code_edit->get_caret_line(1) == 7); CHECK(code_edit->get_caret_column(1) == 0); CHECK(code_edit->get_caret_line(2) == 1); CHECK(code_edit->get_caret_column(2) == 4); // Duplicate multiple separate selections. code_edit->remove_secondary_carets(); code_edit->set_text("test\nlines\nto\n\nduplicate"); code_edit->add_caret(4, 4); code_edit->add_caret(0, 1); code_edit->add_caret(0, 4); code_edit->select(2, 0, 2, 1, 0); code_edit->select(3, 0, 4, 4, 1); code_edit->select(0, 1, 0, 0, 2); code_edit->select(0, 2, 0, 4, 3); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "ttestst\nlines\ntto\n\ndupl\nduplicate"); CHECK(code_edit->get_caret_count() == 4); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 2); CHECK(code_edit->get_selection_origin_column(0) == 1); CHECK(code_edit->get_caret_line(0) == 2); CHECK(code_edit->get_caret_column(0) == 2); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 4); CHECK(code_edit->get_selection_origin_column(1) == 4); CHECK(code_edit->get_caret_line(1) == 5); CHECK(code_edit->get_caret_column(1) == 4); CHECK(code_edit->has_selection(2)); CHECK(code_edit->get_selection_origin_line(2) == 0); CHECK(code_edit->get_selection_origin_column(2) == 2); CHECK(code_edit->get_caret_line(2) == 0); CHECK(code_edit->get_caret_column(2) == 1); CHECK(code_edit->has_selection(3)); CHECK(code_edit->get_selection_origin_line(3) == 0); CHECK(code_edit->get_selection_origin_column(3) == 5); CHECK(code_edit->get_caret_line(3) == 0); CHECK(code_edit->get_caret_column(3) == 7); // Duplicate adjacent selections. code_edit->remove_secondary_carets(); code_edit->set_text("test\nlines\nto\n\nduplicate"); code_edit->add_caret(1, 2); code_edit->select(1, 0, 1, 1, 0); code_edit->select(1, 1, 1, 4, 1); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test\nllineines\nto\n\nduplicate"); CHECK(code_edit->get_caret_count() == 2); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 1); CHECK(code_edit->get_selection_origin_column(0) == 1); CHECK(code_edit->get_caret_line(0) == 1); CHECK(code_edit->get_caret_column(0) == 2); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 1); CHECK(code_edit->get_selection_origin_column(1) == 5); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 8); // Duplicate lines then duplicate selections when there are both selections and non-selections. code_edit->remove_secondary_carets(); code_edit->set_text("test duplicate"); code_edit->select(0, 14, 0, 13, 0); code_edit->add_caret(0, 8); code_edit->add_caret(0, 4); code_edit->select(0, 2, 0, 4, 2); code_edit->duplicate_selection(); CHECK(code_edit->get_text() == "test duplicate\ntestst duplicatee"); CHECK(code_edit->get_caret_count() == 3); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 1); CHECK(code_edit->get_selection_origin_column(0) == 17); CHECK(code_edit->get_caret_line(0) == 1); CHECK(code_edit->get_caret_column(0) == 16); CHECK_FALSE(code_edit->has_selection(1)); CHECK(code_edit->get_caret_line(1) == 1); CHECK(code_edit->get_caret_column(1) == 10); CHECK(code_edit->has_selection(2)); CHECK(code_edit->get_selection_origin_line(2) == 1); CHECK(code_edit->get_selection_origin_column(2) == 4); CHECK(code_edit->get_caret_line(2) == 1); CHECK(code_edit->get_caret_column(2) == 6); } SUBCASE("[SceneTree][CodeEdit] duplicate lines") { String reset_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) )"; code_edit->set_text(reset_text); // 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) == ""); CHECK(code_edit->get_caret_line() == 1); CHECK(code_edit->get_caret_column() == 0); // Duplicate multiple lines with selection. code_edit->set_text(reset_text); code_edit->select(4, 8, 6, 15); code_edit->duplicate_lines(); CHECK(code_edit->get_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): 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) )"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 7); CHECK(code_edit->get_selection_origin_column() == 8); CHECK(code_edit->get_caret_line() == 9); CHECK(code_edit->get_caret_column() == 15); // Duplicate multiple lines with right to left selection. code_edit->set_text(reset_text); code_edit->select(6, 15, 4, 8); code_edit->duplicate_lines(); CHECK(code_edit->get_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): 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) )"); CHECK(code_edit->has_selection()); CHECK(code_edit->get_selection_origin_line() == 9); CHECK(code_edit->get_selection_origin_column() == 15); CHECK(code_edit->get_caret_line() == 7); CHECK(code_edit->get_caret_column() == 8); // Duplicate single lines with multiple carets. Multiple carets on a single line only duplicate once. code_edit->remove_secondary_carets(); code_edit->deselect(); code_edit->set_text(reset_text); code_edit->set_caret_line(3); code_edit->set_caret_column(1); code_edit->add_caret(5, 1); code_edit->add_caret(5, 5); code_edit->add_caret(4, 2); code_edit->duplicate_lines(); CHECK(code_edit->get_text() == R"(extends Node func _ready(): var a := len(OS.get_cmdline_args()) var a := len(OS.get_cmdline_args()) var b := get_child_count() var b := get_child_count() var c := a + b 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) )"); CHECK(code_edit->get_caret_count() == 4); CHECK_FALSE(code_edit->has_selection(0)); CHECK(code_edit->get_caret_line(0) == 4); CHECK(code_edit->get_caret_column(0) == 1); CHECK_FALSE(code_edit->has_selection(1)); CHECK(code_edit->get_caret_line(1) == 8); CHECK(code_edit->get_caret_column(1) == 1); CHECK_FALSE(code_edit->has_selection(2)); CHECK(code_edit->get_caret_line(2) == 8); CHECK(code_edit->get_caret_column(2) == 5); CHECK_FALSE(code_edit->has_selection(3)); CHECK(code_edit->get_caret_line(3) == 6); CHECK(code_edit->get_caret_column(3) == 2); // Duplicate multiple lines with multiple selections. code_edit->remove_secondary_carets(); code_edit->set_text(reset_text); code_edit->add_caret(4, 2); code_edit->add_caret(6, 0); code_edit->add_caret(7, 8); code_edit->select(0, 0, 2, 5, 0); code_edit->select(3, 0, 4, 2, 1); code_edit->select(7, 1, 6, 0, 2); code_edit->select(7, 3, 7, 8, 3); code_edit->duplicate_lines(); CHECK(code_edit->get_text() == R"(extends Node func _ready(): extends Node func _ready(): var a := len(OS.get_cmdline_args()) var b := get_child_count() 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)) for i in range(c): print("This is the solution: ", sin(i)) var pos = get_index() - 1 print("Make sure this exits: %b" % pos) )"); CHECK(code_edit->get_caret_count() == 4); CHECK(code_edit->has_selection(0)); CHECK(code_edit->get_selection_origin_line(0) == 3); CHECK(code_edit->get_selection_origin_column(0) == 0); CHECK(code_edit->get_caret_line(0) == 5); CHECK(code_edit->get_caret_column(0) == 5); CHECK(code_edit->has_selection(1)); CHECK(code_edit->get_selection_origin_line(1) == 8); CHECK(code_edit->get_selection_origin_column(1) == 0); CHECK(code_edit->get_caret_line(1) == 9); CHECK(code_edit->get_caret_column(1) == 2); CHECK(code_edit->has_selection(2)); CHECK(code_edit->get_selection_origin_line(2) == 14); CHECK(code_edit->get_selection_origin_column(2) == 1); CHECK(code_edit->get_caret_line(2) == 13); CHECK(code_edit->get_caret_column(2) == 0); CHECK(code_edit->has_selection(3)); CHECK(code_edit->get_selection_origin_line(3) == 14); CHECK(code_edit->get_selection_origin_column(3) == 3); CHECK(code_edit->get_caret_line(3) == 14); CHECK(code_edit->get_caret_column(3) == 8); } memdelete(code_edit); } } // namespace TestCodeEdit #endif // TEST_CODE_EDIT_H