summaryrefslogtreecommitdiff
path: root/tests
diff options
context:
space:
mode:
Diffstat (limited to 'tests')
-rw-r--r--tests/SCsub5
-rw-r--r--tests/core/io/test_config_file.h2
-rw-r--r--tests/core/io/test_file_access.h4
-rw-r--r--tests/core/io/test_image.h10
-rw-r--r--tests/core/io/test_pck_packer.h4
-rw-r--r--tests/core/io/test_resource.h6
-rw-r--r--tests/core/math/test_astar.h14
-rw-r--r--tests/core/math/test_color.h24
-rw-r--r--tests/core/math/test_geometry_3d.h4
-rw-r--r--tests/core/math/test_math.cpp690
-rw-r--r--tests/core/math/test_math.h41
-rw-r--r--tests/core/math/test_vector2.h3
-rw-r--r--tests/core/math/test_vector3.h3
-rw-r--r--tests/core/object/test_class_db.h4
-rw-r--r--tests/core/object/test_object.h2
-rw-r--r--tests/core/templates/test_oa_hash_map.cpp301
-rw-r--r--tests/core/templates/test_oa_hash_map.h41
-rw-r--r--tests/scene/test_code_edit.h575
-rw-r--r--tests/scene/test_gui.cpp259
-rw-r--r--tests/scene/test_gui.h41
-rw-r--r--tests/scene/test_text_edit.h3507
-rw-r--r--tests/servers/test_physics_2d.cpp374
-rw-r--r--tests/servers/test_physics_2d.h41
-rw-r--r--tests/servers/test_physics_3d.cpp410
-rw-r--r--tests/servers/test_physics_3d.h41
-rw-r--r--tests/servers/test_render.cpp232
-rw-r--r--tests/servers/test_render.h41
-rw-r--r--tests/servers/test_shader_lang.cpp364
-rw-r--r--tests/servers/test_shader_lang.h41
-rw-r--r--tests/servers/test_text_server.h66
-rw-r--r--tests/test_macros.h77
-rw-r--r--tests/test_main.cpp16
32 files changed, 3957 insertions, 3286 deletions
diff --git a/tests/SCsub b/tests/SCsub
index 25b06f2312..c59ce69b92 100644
--- a/tests/SCsub
+++ b/tests/SCsub
@@ -18,11 +18,6 @@ if env_tests["platform"] == "windows":
if env_tests.msvc:
env_tests.Append(CCFLAGS=["/bigobj"])
-env_tests.add_source_files(env.tests_sources, "core/*.cpp")
-env_tests.add_source_files(env.tests_sources, "core/math/*.cpp")
-env_tests.add_source_files(env.tests_sources, "core/templates/*.cpp")
-env_tests.add_source_files(env.tests_sources, "scene/*.cpp")
-env_tests.add_source_files(env.tests_sources, "servers/*.cpp")
env_tests.add_source_files(env.tests_sources, "*.cpp")
lib = env_tests.add_library("tests", env.tests_sources)
diff --git a/tests/core/io/test_config_file.h b/tests/core/io/test_config_file.h
index 6e393c7a2d..355aca479e 100644
--- a/tests/core/io/test_config_file.h
+++ b/tests/core/io/test_config_file.h
@@ -155,7 +155,7 @@ antiAliasing=false
"a=b"=7
)");
- FileAccessRef file = FileAccess::open(config_path, FileAccess::READ);
+ Ref<FileAccess> file = FileAccess::open(config_path, FileAccess::READ);
CHECK_MESSAGE(file->get_as_utf8_string() == contents,
"The saved configuration file should match the expected format.");
}
diff --git a/tests/core/io/test_file_access.h b/tests/core/io/test_file_access.h
index eee57048cf..f0e1cceacf 100644
--- a/tests/core/io/test_file_access.h
+++ b/tests/core/io/test_file_access.h
@@ -38,7 +38,7 @@
namespace TestFileAccess {
TEST_CASE("[FileAccess] CSV read") {
- FileAccessRef f = FileAccess::open(TestUtils::get_data_path("translations.csv"), FileAccess::READ);
+ Ref<FileAccess> f = FileAccess::open(TestUtils::get_data_path("translations.csv"), FileAccess::READ);
Vector<String> header = f->get_csv_line(); // Default delimiter: ",".
REQUIRE(header.size() == 3);
@@ -77,8 +77,6 @@ TEST_CASE("[FileAccess] CSV read") {
CHECK(row5[0] == "What about");
CHECK(row5[1] == "tab separated");
CHECK(row5[2] == "lines, good?");
-
- f->close();
}
} // namespace TestFileAccess
diff --git a/tests/core/io/test_image.h b/tests/core/io/test_image.h
index dcf21dd7b0..1c778c3228 100644
--- a/tests/core/io/test_image.h
+++ b/tests/core/io/test_image.h
@@ -106,7 +106,7 @@ TEST_CASE("[Image] Saving and loading") {
// Load BMP
Ref<Image> image_bmp = memnew(Image());
- FileAccessRef f_bmp = FileAccess::open(TestUtils::get_data_path("images/icon.bmp"), FileAccess::READ, &err);
+ Ref<FileAccess> f_bmp = FileAccess::open(TestUtils::get_data_path("images/icon.bmp"), FileAccess::READ, &err);
PackedByteArray data_bmp;
data_bmp.resize(f_bmp->get_length() + 1);
f_bmp->get_buffer(data_bmp.ptrw(), f_bmp->get_length());
@@ -116,7 +116,7 @@ TEST_CASE("[Image] Saving and loading") {
// Load JPG
Ref<Image> image_jpg = memnew(Image());
- FileAccessRef f_jpg = FileAccess::open(TestUtils::get_data_path("images/icon.jpg"), FileAccess::READ, &err);
+ Ref<FileAccess> f_jpg = FileAccess::open(TestUtils::get_data_path("images/icon.jpg"), FileAccess::READ, &err);
PackedByteArray data_jpg;
data_jpg.resize(f_jpg->get_length() + 1);
f_jpg->get_buffer(data_jpg.ptrw(), f_jpg->get_length());
@@ -126,7 +126,7 @@ TEST_CASE("[Image] Saving and loading") {
// Load WEBP
Ref<Image> image_webp = memnew(Image());
- FileAccessRef f_webp = FileAccess::open(TestUtils::get_data_path("images/icon.webp"), FileAccess::READ, &err);
+ Ref<FileAccess> f_webp = FileAccess::open(TestUtils::get_data_path("images/icon.webp"), FileAccess::READ, &err);
PackedByteArray data_webp;
data_webp.resize(f_webp->get_length() + 1);
f_webp->get_buffer(data_webp.ptrw(), f_webp->get_length());
@@ -136,7 +136,7 @@ TEST_CASE("[Image] Saving and loading") {
// Load PNG
Ref<Image> image_png = memnew(Image());
- FileAccessRef f_png = FileAccess::open(TestUtils::get_data_path("images/icon.png"), FileAccess::READ, &err);
+ Ref<FileAccess> f_png = FileAccess::open(TestUtils::get_data_path("images/icon.png"), FileAccess::READ, &err);
PackedByteArray data_png;
data_png.resize(f_png->get_length() + 1);
f_png->get_buffer(data_png.ptrw(), f_png->get_length());
@@ -146,7 +146,7 @@ TEST_CASE("[Image] Saving and loading") {
// Load TGA
Ref<Image> image_tga = memnew(Image());
- FileAccessRef f_tga = FileAccess::open(TestUtils::get_data_path("images/icon.tga"), FileAccess::READ, &err);
+ Ref<FileAccess> f_tga = FileAccess::open(TestUtils::get_data_path("images/icon.tga"), FileAccess::READ, &err);
PackedByteArray data_tga;
data_tga.resize(f_tga->get_length() + 1);
f_tga->get_buffer(data_tga.ptrw(), f_tga->get_length());
diff --git a/tests/core/io/test_pck_packer.h b/tests/core/io/test_pck_packer.h
index 95adca6d68..d21fbdaf50 100644
--- a/tests/core/io/test_pck_packer.h
+++ b/tests/core/io/test_pck_packer.h
@@ -52,7 +52,7 @@ TEST_CASE("[PCKPacker] Pack an empty PCK file") {
"Flushing the PCK should return an OK error code.");
Error err;
- FileAccessRef f = FileAccess::open(output_pck_path, FileAccess::READ, &err);
+ Ref<FileAccess> f = FileAccess::open(output_pck_path, FileAccess::READ, &err);
CHECK_MESSAGE(
err == OK,
"The generated empty PCK file should be opened successfully.");
@@ -106,7 +106,7 @@ TEST_CASE("[PCKPacker] Pack a PCK file with some files and directories") {
"Flushing the PCK should return an OK error code.");
Error err;
- FileAccessRef f = FileAccess::open(output_pck_path, FileAccess::READ, &err);
+ Ref<FileAccess> f = FileAccess::open(output_pck_path, FileAccess::READ, &err);
CHECK_MESSAGE(
err == OK,
"The generated non-empty PCK file should be opened successfully.");
diff --git a/tests/core/io/test_resource.h b/tests/core/io/test_resource.h
index b3983bb06d..84d651b63f 100644
--- a/tests/core/io/test_resource.h
+++ b/tests/core/io/test_resource.h
@@ -69,7 +69,7 @@ TEST_CASE("[Resource] Duplication") {
TEST_CASE("[Resource] Saving and loading") {
Ref<Resource> resource = memnew(Resource);
resource->set_name("Hello world");
- resource->set_meta(" ExampleMetadata ", Vector2i(40, 80));
+ resource->set_meta("ExampleMetadata", Vector2i(40, 80));
resource->set_meta("string", "The\nstring\nwith\nunnecessary\nline\n\t\\\nbreaks");
Ref<Resource> child_resource = memnew(Resource);
child_resource->set_name("I'm a child resource");
@@ -84,7 +84,7 @@ TEST_CASE("[Resource] Saving and loading") {
loaded_resource_binary->get_name() == "Hello world",
"The loaded resource name should be equal to the expected value.");
CHECK_MESSAGE(
- loaded_resource_binary->get_meta(" ExampleMetadata ") == Vector2i(40, 80),
+ loaded_resource_binary->get_meta("ExampleMetadata") == Vector2i(40, 80),
"The loaded resource metadata should be equal to the expected value.");
CHECK_MESSAGE(
loaded_resource_binary->get_meta("string") == "The\nstring\nwith\nunnecessary\nline\n\t\\\nbreaks",
@@ -99,7 +99,7 @@ TEST_CASE("[Resource] Saving and loading") {
loaded_resource_text->get_name() == "Hello world",
"The loaded resource name should be equal to the expected value.");
CHECK_MESSAGE(
- loaded_resource_text->get_meta(" ExampleMetadata ") == Vector2i(40, 80),
+ loaded_resource_text->get_meta("ExampleMetadata") == Vector2i(40, 80),
"The loaded resource metadata should be equal to the expected value.");
CHECK_MESSAGE(
loaded_resource_text->get_meta("string") == "The\nstring\nwith\nunnecessary\nline\n\t\\\nbreaks",
diff --git a/tests/core/math/test_astar.h b/tests/core/math/test_astar.h
index 859172dca3..1306d3c20e 100644
--- a/tests/core/math/test_astar.h
+++ b/tests/core/math/test_astar.h
@@ -37,7 +37,7 @@
namespace TestAStar {
-class ABCX : public AStar {
+class ABCX : public AStar3D {
public:
enum {
A,
@@ -66,7 +66,7 @@ public:
}
};
-TEST_CASE("[AStar] ABC path") {
+TEST_CASE("[AStar3D] ABC path") {
ABCX abcx;
Vector<int> path = abcx.get_id_path(ABCX::A, ABCX::C);
REQUIRE(path.size() == 3);
@@ -75,7 +75,7 @@ TEST_CASE("[AStar] ABC path") {
CHECK(path[2] == ABCX::C);
}
-TEST_CASE("[AStar] ABCX path") {
+TEST_CASE("[AStar3D] ABCX path") {
ABCX abcx;
Vector<int> path = abcx.get_id_path(ABCX::X, ABCX::C);
REQUIRE(path.size() == 4);
@@ -85,8 +85,8 @@ TEST_CASE("[AStar] ABCX path") {
CHECK(path[3] == ABCX::C);
}
-TEST_CASE("[AStar] Add/Remove") {
- AStar a;
+TEST_CASE("[AStar3D] Add/Remove") {
+ AStar3D a;
// Manual tests.
a.add_point(1, Vector3(0, 0, 0));
@@ -213,13 +213,13 @@ TEST_CASE("[AStar] Add/Remove") {
// It's been great work, cheers. \(^ ^)/
}
-TEST_CASE("[Stress][AStar] Find paths") {
+TEST_CASE("[Stress][AStar3D] Find paths") {
// Random stress tests with Floyd-Warshall.
const int N = 30;
Math::seed(0);
for (int test = 0; test < 1000; test++) {
- AStar a;
+ AStar3D a;
Vector3 p[N];
bool adj[N][N] = { { false } };
diff --git a/tests/core/math/test_color.h b/tests/core/math/test_color.h
index e62ce6ec60..51c3bc8bdc 100644
--- a/tests/core/math/test_color.h
+++ b/tests/core/math/test_color.h
@@ -144,17 +144,35 @@ TEST_CASE("[Color] Conversion methods") {
"The string representation should match the expected value.");
}
+TEST_CASE("[Color] Linear <-> sRGB conversion") {
+ const Color color = Color(0.35, 0.5, 0.6, 0.7);
+ const Color color_linear = color.srgb_to_linear();
+ const Color color_srgb = color.linear_to_srgb();
+ CHECK_MESSAGE(
+ color_linear.is_equal_approx(Color(0.100481, 0.214041, 0.318547, 0.7)),
+ "The color converted to linear color space should match the expected value.");
+ CHECK_MESSAGE(
+ color_srgb.is_equal_approx(Color(0.62621, 0.735357, 0.797738, 0.7)),
+ "The color converted to sRGB color space should match the expected value.");
+ CHECK_MESSAGE(
+ color_linear.linear_to_srgb().is_equal_approx(Color(0.35, 0.5, 0.6, 0.7)),
+ "The linear color converted back to sRGB color space should match the expected value.");
+ CHECK_MESSAGE(
+ color_srgb.srgb_to_linear().is_equal_approx(Color(0.35, 0.5, 0.6, 0.7)),
+ "The sRGB color converted back to linear color space should match the expected value.");
+}
+
TEST_CASE("[Color] Named colors") {
CHECK_MESSAGE(
- Color::named("red").is_equal_approx(Color(1, 0, 0)),
+ Color::named("red").is_equal_approx(Color::hex(0xFF0000FF)),
"The named color \"red\" should match the expected value.");
// Named colors have their names automatically normalized.
CHECK_MESSAGE(
- Color::named("white_smoke").is_equal_approx(Color(0.96, 0.96, 0.96)),
+ Color::named("white_smoke").is_equal_approx(Color::hex(0xF5F5F5FF)),
"The named color \"white_smoke\" should match the expected value.");
CHECK_MESSAGE(
- Color::named("Slate Blue").is_equal_approx(Color(0.42, 0.35, 0.80)),
+ Color::named("Slate Blue").is_equal_approx(Color::hex(0x6A5ACDFF)),
"The named color \"Slate Blue\" should match the expected value.");
ERR_PRINT_OFF;
diff --git a/tests/core/math/test_geometry_3d.h b/tests/core/math/test_geometry_3d.h
index 1b8d2eee34..99a4ef2d46 100644
--- a/tests/core/math/test_geometry_3d.h
+++ b/tests/core/math/test_geometry_3d.h
@@ -288,7 +288,7 @@ TEST_CASE("[Geometry3D] Is Point in Projected Triangle") {
TEST_CASE("[Geometry3D] Does Ray Intersect Triangle") {
struct Case {
Vector3 from, direction, v_1, v_2, v_3;
- Vector3 *result;
+ Vector3 *result = nullptr;
bool want;
Case(){};
Case(Vector3 p_from, Vector3 p_direction, Vector3 p_v_1, Vector3 p_v_2, Vector3 p_v_3, bool p_want) :
@@ -390,7 +390,7 @@ TEST_CASE("[Geometry3D] Triangle and Box Overlap") {
struct Case {
Vector3 box_centre;
Vector3 box_half_size;
- Vector3 *tri_verts;
+ Vector3 *tri_verts = nullptr;
bool want;
Case(){};
Case(Vector3 p_centre, Vector3 p_half_size, Vector3 *p_verts, bool p_want) :
diff --git a/tests/core/math/test_math.cpp b/tests/core/math/test_math.cpp
deleted file mode 100644
index 4182455b7a..0000000000
--- a/tests/core/math/test_math.cpp
+++ /dev/null
@@ -1,690 +0,0 @@
-/*************************************************************************/
-/* test_math.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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. */
-/*************************************************************************/
-
-#include "test_math.h"
-
-#include "core/math/camera_matrix.h"
-#include "core/math/delaunay_3d.h"
-#include "core/math/geometry_2d.h"
-#include "core/os/main_loop.h"
-#include "core/os/os.h"
-
-namespace TestMath {
-
-class GetClassAndNamespace {
- String code;
- int idx;
- int line;
- String error_str;
- bool error;
- Variant value;
-
- String class_name;
-
- enum Token {
- TK_BRACKET_OPEN,
- TK_BRACKET_CLOSE,
- TK_CURLY_BRACKET_OPEN,
- TK_CURLY_BRACKET_CLOSE,
- TK_PERIOD,
- TK_COLON,
- TK_COMMA,
- TK_SYMBOL,
- TK_IDENTIFIER,
- TK_STRING,
- TK_NUMBER,
- TK_EOF,
- TK_ERROR
- };
-
- Token get_token() {
- while (true) {
- switch (code[idx]) {
- case '\n': {
- line++;
- idx++;
- break;
- };
- case 0: {
- return TK_EOF;
-
- } break;
- case '{': {
- idx++;
- return TK_CURLY_BRACKET_OPEN;
- };
- case '}': {
- idx++;
- return TK_CURLY_BRACKET_CLOSE;
- };
- case '[': {
- idx++;
- return TK_BRACKET_OPEN;
- };
- case ']': {
- idx++;
- return TK_BRACKET_CLOSE;
- };
- case ':': {
- idx++;
- return TK_COLON;
- };
- case ',': {
- idx++;
- return TK_COMMA;
- };
- case '.': {
- idx++;
- return TK_PERIOD;
- };
- case '#': {
- //compiler directive
- while (code[idx] != '\n' && code[idx] != 0) {
- idx++;
- }
- continue;
- } break;
- case '/': {
- switch (code[idx + 1]) {
- case '*': { // block comment
-
- idx += 2;
- while (true) {
- if (code[idx] == 0) {
- error_str = "Unterminated comment";
- error = true;
- return TK_ERROR;
- } else if (code[idx] == '*' && code[idx + 1] == '/') {
- idx += 2;
- break;
- } else if (code[idx] == '\n') {
- line++;
- }
-
- idx++;
- }
-
- } break;
- case '/': { // line comment skip
-
- while (code[idx] != '\n' && code[idx] != 0) {
- idx++;
- }
-
- } break;
- default: {
- value = "/";
- idx++;
- return TK_SYMBOL;
- }
- }
-
- continue; // a comment
- } break;
- case '\'':
- case '"': {
- char32_t begin_str = code[idx];
- idx++;
- String tk_string = String();
- while (true) {
- if (code[idx] == 0) {
- error_str = "Unterminated String";
- error = true;
- return TK_ERROR;
- } else if (code[idx] == begin_str) {
- idx++;
- break;
- } else if (code[idx] == '\\') {
- //escaped characters...
- idx++;
- char32_t next = code[idx];
- if (next == 0) {
- error_str = "Unterminated String";
- error = true;
- return TK_ERROR;
- }
- char32_t res = 0;
-
- switch (next) {
- case 'b':
- res = 8;
- break;
- case 't':
- res = 9;
- break;
- case 'n':
- res = 10;
- break;
- case 'f':
- res = 12;
- break;
- case 'r':
- res = 13;
- break;
- case '\"':
- res = '\"';
- break;
- case '\\':
- res = '\\';
- break;
- default: {
- res = next;
- } break;
- }
-
- tk_string += res;
-
- } else {
- if (code[idx] == '\n') {
- line++;
- }
- tk_string += code[idx];
- }
- idx++;
- }
-
- value = tk_string;
-
- return TK_STRING;
-
- } break;
- default: {
- if (code[idx] <= 32) {
- idx++;
- break;
- }
-
- if ((code[idx] >= 33 && code[idx] <= 47) || (code[idx] >= 58 && code[idx] <= 64) || (code[idx] >= 91 && code[idx] <= 96) || (code[idx] >= 123 && code[idx] <= 127)) {
- value = String::chr(code[idx]);
- idx++;
- return TK_SYMBOL;
- }
-
- if (code[idx] == '-' || is_digit(code[idx])) {
- //a number
- const char32_t *rptr;
- double number = String::to_float(&code[idx], &rptr);
- idx += (rptr - &code[idx]);
- value = number;
- return TK_NUMBER;
-
- } else if (is_ascii_char(code[idx]) || code[idx] > 127) {
- String id;
-
- while (is_ascii_char(code[idx]) || code[idx] > 127) {
- id += code[idx];
- idx++;
- }
-
- value = id;
- return TK_IDENTIFIER;
- } else {
- error_str = "Unexpected character.";
- error = true;
- return TK_ERROR;
- }
- }
- }
- }
- }
-
-public:
- Error parse(const String &p_code, const String &p_known_class_name = String()) {
- code = p_code;
- idx = 0;
- line = 0;
- error_str = String();
- error = false;
- value = Variant();
- class_name = String();
-
- bool use_next_class = false;
- Token tk = get_token();
-
- Map<int, String> namespace_stack;
- int curly_stack = 0;
-
- while (!error || tk != TK_EOF) {
- if (tk == TK_BRACKET_OPEN) {
- tk = get_token();
- if (tk == TK_IDENTIFIER && String(value) == "ScriptClass") {
- if (get_token() == TK_BRACKET_CLOSE) {
- use_next_class = true;
- }
- }
- } else if (tk == TK_IDENTIFIER && String(value) == "class") {
- tk = get_token();
- if (tk == TK_IDENTIFIER) {
- String name = value;
- if (use_next_class || p_known_class_name == name) {
- for (const KeyValue<int, String> &E : namespace_stack) {
- class_name += E.value + ".";
- }
- class_name += String(value);
- break;
- }
- }
-
- } else if (tk == TK_IDENTIFIER && String(value) == "namespace") {
- String name;
- int at_level = curly_stack;
- while (true) {
- tk = get_token();
- if (tk == TK_IDENTIFIER) {
- name += String(value);
- }
-
- tk = get_token();
- if (tk == TK_PERIOD) {
- name += ".";
- } else if (tk == TK_CURLY_BRACKET_OPEN) {
- curly_stack++;
- break;
- } else {
- break; //whatever else
- }
- }
-
- if (!name.is_empty()) {
- namespace_stack[at_level] = name;
- }
-
- } else if (tk == TK_CURLY_BRACKET_OPEN) {
- curly_stack++;
- } else if (tk == TK_CURLY_BRACKET_CLOSE) {
- curly_stack--;
- if (namespace_stack.has(curly_stack)) {
- namespace_stack.erase(curly_stack);
- }
- }
-
- tk = get_token();
- }
-
- if (error) {
- return ERR_PARSE_ERROR;
- }
-
- return OK;
- }
-
- String get_error() {
- return error_str;
- }
-
- String get_class() {
- return class_name;
- }
-};
-
-void test_vec(Plane p_vec) {
- CameraMatrix cm;
- cm.set_perspective(45, 1, 0, 100);
- Plane v0 = cm.xform4(p_vec);
-
- print_line("out: " + v0);
- v0.normal.z = (v0.d / 100.0 * 2.0 - 1.0) * v0.d;
- print_line("out_F: " + v0);
-}
-
-uint32_t ihash(uint32_t a) {
- a = (a + 0x7ed55d16) + (a << 12);
- a = (a ^ 0xc761c23c) ^ (a >> 19);
- a = (a + 0x165667b1) + (a << 5);
- a = (a + 0xd3a2646c) ^ (a << 9);
- a = (a + 0xfd7046c5) + (a << 3);
- a = (a ^ 0xb55a4f09) ^ (a >> 16);
- return a;
-}
-
-uint32_t ihash2(uint32_t a) {
- a = (a ^ 61) ^ (a >> 16);
- a = a + (a << 3);
- a = a ^ (a >> 4);
- a = a * 0x27d4eb2d;
- a = a ^ (a >> 15);
- return a;
-}
-
-uint32_t ihash3(uint32_t a) {
- a = (a + 0x479ab41d) + (a << 8);
- a = (a ^ 0xe4aa10ce) ^ (a >> 5);
- a = (a + 0x9942f0a6) - (a << 14);
- a = (a ^ 0x5aedd67d) ^ (a >> 3);
- a = (a + 0x17bea992) + (a << 7);
- return a;
-}
-
-MainLoop *test() {
- {
- Vector<Vector3> points;
- points.push_back(Vector3(0, 0, 0));
- points.push_back(Vector3(0, 0, 1));
- points.push_back(Vector3(0, 1, 0));
- points.push_back(Vector3(0, 1, 1));
- points.push_back(Vector3(1, 1, 0));
- points.push_back(Vector3(1, 0, 0));
- points.push_back(Vector3(1, 0, 1));
- points.push_back(Vector3(1, 1, 1));
-
- for (int i = 0; i < 800; i++) {
- points.push_back(Vector3(Math::randf() * 2.0 - 1.0, Math::randf() * 2.0 - 1.0, Math::randf() * 2.0 - 1.0) * Vector3(25, 30, 33));
- }
-
- Vector<Delaunay3D::OutputSimplex> os = Delaunay3D::tetrahedralize(points);
- print_line("simplices in the end: " + itos(os.size()));
- for (int i = 0; i < os.size(); i++) {
- print_line("Simplex " + itos(i) + ": ");
- print_line(points[os[i].points[0]]);
- print_line(points[os[i].points[1]]);
- print_line(points[os[i].points[2]]);
- print_line(points[os[i].points[3]]);
- }
-
- {
- FileAccessRef f = FileAccess::open("res://bsp.obj", FileAccess::WRITE);
- for (int i = 0; i < os.size(); i++) {
- f->store_line("o Simplex" + itos(i));
- for (int j = 0; j < 4; j++) {
- f->store_line(vformat("v %f %f %f", points[os[i].points[j]].x, points[os[i].points[j]].y, points[os[i].points[j]].z));
- }
- static const int face_order[4][3] = {
- { 1, 2, 3 },
- { 1, 3, 4 },
- { 1, 2, 4 },
- { 2, 3, 4 }
- };
-
- for (int j = 0; j < 4; j++) {
- f->store_line(vformat("f %d %d %d", 4 * i + face_order[j][0], 4 * i + face_order[j][1], 4 * i + face_order[j][2]));
- }
- }
- f->close();
- }
-
- return nullptr;
- }
-
- {
- float r = 1;
- float g = 0.5;
- float b = 0.1;
-
- const float pow2to9 = 512.0f;
- const float B = 15.0f;
- const float N = 9.0f;
-
- float sharedexp = 65408.000f;
-
- float cRed = MAX(0.0f, MIN(sharedexp, r));
- float cGreen = MAX(0.0f, MIN(sharedexp, g));
- float cBlue = MAX(0.0f, MIN(sharedexp, b));
-
- float cMax = MAX(cRed, MAX(cGreen, cBlue));
-
- float expp = MAX(-B - 1.0f, floor(Math::log(cMax) / Math_LN2)) + 1.0f + B;
-
- float sMax = (float)floor((cMax / Math::pow(2.0f, expp - B - N)) + 0.5f);
-
- float exps = expp + 1.0f;
-
- if (0.0 <= sMax && sMax < pow2to9) {
- exps = expp;
- }
-
- float sRed = Math::floor((cRed / pow(2.0f, exps - B - N)) + 0.5f);
- float sGreen = Math::floor((cGreen / pow(2.0f, exps - B - N)) + 0.5f);
- float sBlue = Math::floor((cBlue / pow(2.0f, exps - B - N)) + 0.5f);
-
- print_line("R: " + rtos(sRed) + " G: " + rtos(sGreen) + " B: " + rtos(sBlue) + " EXP: " + rtos(exps));
-
- uint32_t rgbe = (Math::fast_ftoi(sRed) & 0x1FF) | ((Math::fast_ftoi(sGreen) & 0x1FF) << 9) | ((Math::fast_ftoi(sBlue) & 0x1FF) << 18) | ((Math::fast_ftoi(exps) & 0x1F) << 27);
-
- float rb = rgbe & 0x1ff;
- float gb = (rgbe >> 9) & 0x1ff;
- float bb = (rgbe >> 18) & 0x1ff;
- float eb = (rgbe >> 27);
- float mb = Math::pow(2.0, eb - 15.0 - 9.0);
- float rd = rb * mb;
- float gd = gb * mb;
- float bd = bb * mb;
-
- print_line("RGBE: " + Color(rd, gd, bd));
- }
-
- Vector<int> ints;
- ints.resize(20);
-
- {
- int *w;
- w = ints.ptrw();
- for (int i = 0; i < ints.size(); i++) {
- w[i] = i;
- }
- }
-
- Vector<int> posho = ints;
-
- {
- const int *r = posho.ptr();
- for (int i = 0; i < posho.size(); i++) {
- print_line(itos(i) + " : " + itos(r[i]));
- }
- }
-
- List<String> cmdlargs = OS::get_singleton()->get_cmdline_args();
-
- if (cmdlargs.is_empty()) {
- //try editor!
- return nullptr;
- }
-
- String test = cmdlargs.back()->get();
- if (test == "math") {
- // Not a file name but the test name, abort.
- // FIXME: This test is ugly as heck, needs fixing :)
- return nullptr;
- }
-
- FileAccess *fa = FileAccess::open(test, FileAccess::READ);
- ERR_FAIL_COND_V_MSG(!fa, nullptr, "Could not open file: " + test);
-
- Vector<uint8_t> buf;
- uint64_t flen = fa->get_length();
- buf.resize(fa->get_length() + 1);
- fa->get_buffer(buf.ptrw(), flen);
- buf.write[flen] = 0;
-
- String code;
- code.parse_utf8((const char *)&buf[0]);
-
- GetClassAndNamespace getclass;
- if (getclass.parse(code)) {
- print_line("Parse error: " + getclass.get_error());
- } else {
- print_line("Found class: " + getclass.get_class());
- }
-
- {
- Vector<int> hashes;
- List<StringName> tl;
- ClassDB::get_class_list(&tl);
-
- for (const StringName &E : tl) {
- Vector<uint8_t> m5b = E.operator String().md5_buffer();
- hashes.push_back(hashes.size());
- }
-
- for (int i = nearest_shift(hashes.size()); i < 20; i++) {
- bool success = true;
- for (int s = 0; s < 10000; s++) {
- Set<uint32_t> existing;
- success = true;
-
- for (int j = 0; j < hashes.size(); j++) {
- uint32_t eh = ihash2(ihash3(hashes[j] + ihash(s) + s)) & ((1 << i) - 1);
- if (existing.has(eh)) {
- success = false;
- break;
- }
- existing.insert(eh);
- }
-
- if (success) {
- print_line("success at " + itos(i) + "/" + itos(nearest_shift(hashes.size())) + " shift " + itos(s));
- break;
- }
- }
- if (success) {
- break;
- }
- }
-
- print_line("DONE");
- }
-
- {
- print_line("NUM: " + itos(-128));
- }
-
- {
- Vector3 v(1, 2, 3);
- v.normalize();
- real_t a = 0.3;
-
- Basis m(v, a);
-
- Vector3 v2(7, 3, 1);
- v2.normalize();
- real_t a2 = 0.8;
-
- Basis m2(v2, a2);
-
- Quaternion q = m;
- Quaternion q2 = m2;
-
- Basis m3 = m.inverse() * m2;
- Quaternion q3 = (q.inverse() * q2); //.normalized();
-
- print_line(Quaternion(m3));
- print_line(q3);
-
- print_line("before v: " + v + " a: " + rtos(a));
- q.get_axis_angle(v, a);
- print_line("after v: " + v + " a: " + rtos(a));
- }
-
- String ret;
-
- List<String> args;
- args.push_back("-l");
- Error err = OS::get_singleton()->execute("/bin/ls", args, &ret);
- print_line("error: " + itos(err));
- print_line(ret);
-
- Basis m3;
- m3.rotate(Vector3(1, 0, 0), 0.2);
- m3.rotate(Vector3(0, 1, 0), 1.77);
- m3.rotate(Vector3(0, 0, 1), 212);
- Basis m32;
- m32.set_euler(m3.get_euler());
- print_line("ELEULEEEEEEEEEEEEEEEEEER: " + m3.get_euler() + " vs " + m32.get_euler());
-
- {
- Dictionary d;
- d["momo"] = 1;
- Dictionary b = d;
- b["44"] = 4;
- }
-
- print_line("inters: " + rtos(Geometry2D::segment_intersects_circle(Vector2(-5, 0), Vector2(-2, 0), Vector2(), 1.0)));
-
- print_line("cross: " + Vector3(1, 2, 3).cross(Vector3(4, 5, 7)));
- print_line("dot: " + rtos(Vector3(1, 2, 3).dot(Vector3(4, 5, 7))));
- print_line("abs: " + Vector3(-1, 2, -3).abs());
- print_line("distance_to: " + rtos(Vector3(1, 2, 3).distance_to(Vector3(4, 5, 7))));
- print_line("distance_squared_to: " + rtos(Vector3(1, 2, 3).distance_squared_to(Vector3(4, 5, 7))));
- print_line("plus: " + (Vector3(1, 2, 3) + Vector3(Vector3(4, 5, 7))));
- print_line("minus: " + (Vector3(1, 2, 3) - Vector3(Vector3(4, 5, 7))));
- print_line("mul: " + (Vector3(1, 2, 3) * Vector3(Vector3(4, 5, 7))));
- print_line("div: " + (Vector3(1, 2, 3) / Vector3(Vector3(4, 5, 7))));
- print_line("mul scalar: " + (Vector3(1, 2, 3) * 2.0));
- print_line("premul scalar: " + (2.0 * Vector3(1, 2, 3)));
- print_line("div scalar: " + (Vector3(1, 2, 3) / 3.0));
- print_line("length: " + rtos(Vector3(1, 2, 3).length()));
- print_line("length squared: " + rtos(Vector3(1, 2, 3).length_squared()));
- print_line("normalized: " + Vector3(1, 2, 3).normalized());
- print_line("inverse: " + Vector3(1, 2, 3).inverse());
-
- {
- Vector3 v(4, 5, 7);
- v.normalize();
- print_line("normalize: " + v);
- }
-
- {
- Vector3 v(4, 5, 7);
- v += Vector3(1, 2, 3);
- print_line("+=: " + v);
- }
-
- {
- Vector3 v(4, 5, 7);
- v -= Vector3(1, 2, 3);
- print_line("-=: " + v);
- }
-
- {
- Vector3 v(4, 5, 7);
- v *= Vector3(1, 2, 3);
- print_line("*=: " + v);
- }
-
- {
- Vector3 v(4, 5, 7);
- v /= Vector3(1, 2, 3);
- print_line("/=: " + v);
- }
-
- {
- Vector3 v(4, 5, 7);
- v *= 2.0;
- print_line("scalar *=: " + v);
- }
-
- {
- Vector3 v(4, 5, 7);
- v /= 2.0;
- print_line("scalar /=: " + v);
- }
-
- return nullptr;
-}
-} // namespace TestMath
diff --git a/tests/core/math/test_math.h b/tests/core/math/test_math.h
deleted file mode 100644
index a8aa8f6847..0000000000
--- a/tests/core/math/test_math.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_math.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_MATH_H
-#define TEST_MATH_H
-
-class MainLoop;
-
-namespace TestMath {
-
-MainLoop *test();
-}
-
-#endif
diff --git a/tests/core/math/test_vector2.h b/tests/core/math/test_vector2.h
index ff60467bf4..9b7800164a 100644
--- a/tests/core/math/test_vector2.h
+++ b/tests/core/math/test_vector2.h
@@ -90,6 +90,9 @@ TEST_CASE("[Vector2] Interpolation methods") {
Vector2(5, 0).slerp(Vector2(0, 5), 0.5).is_equal_approx(Vector2(5, 5) * Math_SQRT12),
"Vector2 slerp with non-normalized values should work as expected.");
CHECK_MESSAGE(
+ Vector2(1, 1).slerp(Vector2(2, 2), 0.5).is_equal_approx(Vector2(1.5, 1.5)),
+ "Vector2 slerp with colinear inputs should behave as expected.");
+ CHECK_MESSAGE(
Vector2().slerp(Vector2(), 0.5) == Vector2(),
"Vector2 slerp with both inputs as zero vectors should return a zero vector.");
CHECK_MESSAGE(
diff --git a/tests/core/math/test_vector3.h b/tests/core/math/test_vector3.h
index 847a7c0b3f..6f99fada2b 100644
--- a/tests/core/math/test_vector3.h
+++ b/tests/core/math/test_vector3.h
@@ -111,6 +111,9 @@ TEST_CASE("[Vector3] Interpolation methods") {
Vector3(5, 0, 0).slerp(Vector3(0, 3, 4), 0.5).is_equal_approx(Vector3(3.535533905029296875, 2.121320486068725586, 2.828427314758300781)),
"Vector3 slerp with non-normalized values should work as expected.");
CHECK_MESSAGE(
+ Vector3(1, 1, 1).slerp(Vector3(2, 2, 2), 0.5).is_equal_approx(Vector3(1.5, 1.5, 1.5)),
+ "Vector3 slerp with colinear inputs should behave as expected.");
+ CHECK_MESSAGE(
Vector3().slerp(Vector3(), 0.5) == Vector3(),
"Vector3 slerp with both inputs as zero vectors should return a zero vector.");
CHECK_MESSAGE(
diff --git a/tests/core/object/test_class_db.h b/tests/core/object/test_class_db.h
index e4145c8408..5cf5403a50 100644
--- a/tests/core/object/test_class_db.h
+++ b/tests/core/object/test_class_db.h
@@ -671,10 +671,6 @@ void add_exposed_classes(Context &r_context) {
} else {
exposed_class.methods.push_back(method);
}
-
- if (method.is_virtual) {
- TEST_COND(String(method.name)[0] != '_', "Virtual method ", String(method.name), " does not start with underscore.");
- }
}
// Add signals
diff --git a/tests/core/object/test_object.h b/tests/core/object/test_object.h
index e44b93bb66..5b9d9cab53 100644
--- a/tests/core/object/test_object.h
+++ b/tests/core/object/test_object.h
@@ -133,7 +133,7 @@ TEST_CASE("[Object] Core getters") {
}
TEST_CASE("[Object] Metadata") {
- const String meta_path = "hello/world complex métadata\n\n\t\tpath";
+ const String meta_path = "complex_metadata_path";
Object object;
object.set_meta(meta_path, Color(0, 1, 0));
diff --git a/tests/core/templates/test_oa_hash_map.cpp b/tests/core/templates/test_oa_hash_map.cpp
deleted file mode 100644
index 87bf9feb83..0000000000
--- a/tests/core/templates/test_oa_hash_map.cpp
+++ /dev/null
@@ -1,301 +0,0 @@
-/*************************************************************************/
-/* test_oa_hash_map.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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. */
-/*************************************************************************/
-
-#include "test_oa_hash_map.h"
-
-#include "core/os/os.h"
-#include "core/templates/oa_hash_map.h"
-
-namespace TestOAHashMap {
-
-struct CountedItem {
- static int count;
-
- int id = -1;
- bool destroyed = false;
-
- CountedItem() {
- count++;
- }
-
- CountedItem(int p_id) :
- id(p_id) {
- count++;
- }
-
- CountedItem(const CountedItem &p_other) :
- id(p_other.id) {
- count++;
- }
-
- void operator=(const CountedItem &p_other) {
- id = p_other.id;
- count++;
- }
-
- ~CountedItem() {
- CRASH_COND(destroyed);
- count--;
- destroyed = true;
- }
-};
-
-int CountedItem::count;
-
-MainLoop *test() {
- OS::get_singleton()->print("\n\n\nHello from test\n");
-
- // test element tracking.
- {
- OAHashMap<int, int> map;
-
- map.set(42, 1337);
- map.set(1337, 21);
- map.set(42, 11880);
-
- int value = 0;
- map.lookup(42, value);
-
- OS::get_singleton()->print("capacity %d\n", map.get_capacity());
- OS::get_singleton()->print("elements %d\n", map.get_num_elements());
-
- OS::get_singleton()->print("map[42] = %d\n", value);
- }
-
- // rehashing and deletion
- {
- OAHashMap<int, int> map;
-
- for (int i = 0; i < 500; i++) {
- map.set(i, i * 2);
- }
-
- for (int i = 0; i < 500; i += 2) {
- map.remove(i);
- }
-
- uint32_t num_elems = 0;
- for (int i = 0; i < 500; i++) {
- int tmp;
- if (map.lookup(i, tmp) && tmp == i * 2) {
- num_elems++;
- }
- }
-
- OS::get_singleton()->print("elements %d == %d.\n", map.get_num_elements(), num_elems);
- }
-
- // iteration
- {
- OAHashMap<String, int> map;
-
- map.set("Hello", 1);
- map.set("World", 2);
- map.set("Godot rocks", 42);
-
- for (OAHashMap<String, int>::Iterator it = map.iter(); it.valid; it = map.next_iter(it)) {
- OS::get_singleton()->print("map[\"%s\"] = %d\n", it.key->utf8().get_data(), *it.value);
- }
- }
-
- // stress test / test for issue #22928
- {
- OAHashMap<int, int> map;
- int dummy = 0;
- const int N = 1000;
- uint32_t *keys = new uint32_t[N];
-
- Math::seed(0);
-
- // insert a couple of random keys (with a dummy value, which is ignored)
- for (int i = 0; i < N; i++) {
- keys[i] = Math::rand();
- map.set(keys[i], dummy);
-
- if (!map.lookup(keys[i], dummy)) {
- OS::get_singleton()->print("could not find 0x%X despite it was just inserted!\n", unsigned(keys[i]));
- }
- }
-
- // check whether the keys are still present
- for (int i = 0; i < N; i++) {
- if (!map.lookup(keys[i], dummy)) {
- OS::get_singleton()->print("could not find 0x%X despite it has been inserted previously! (not checking the other keys, breaking...)\n", unsigned(keys[i]));
- break;
- }
- }
-
- delete[] keys;
- }
-
- // regression test / test for issue related to #31402
- {
- OS::get_singleton()->print("test for issue #31402 started...\n");
-
- const int num_test_values = 12;
- int test_values[num_test_values] = { 0, 24, 48, 72, 96, 120, 144, 168, 192, 216, 240, 264 };
-
- int dummy = 0;
- OAHashMap<int, int> map;
- map.clear();
-
- for (int i = 0; i < num_test_values; ++i) {
- map.set(test_values[i], dummy);
- }
-
- OS::get_singleton()->print("test for issue #31402 passed.\n");
- }
-
- // test collision resolution, should not crash or run indefinitely
- {
- OAHashMap<int, int> map(4);
- map.set(1, 1);
- map.set(5, 1);
- map.set(9, 1);
- map.set(13, 1);
- map.remove(5);
- map.remove(9);
- map.remove(13);
- map.set(5, 1);
- }
-
- // test memory management of items, should not crash or leak items
- {
- // Exercise different patterns of removal
- for (int i = 0; i < 4; ++i) {
- {
- OAHashMap<String, CountedItem> map;
- int id = 0;
- for (int j = 0; j < 100; ++j) {
- map.insert(itos(j), CountedItem(id));
- }
- if (i <= 1) {
- for (int j = 0; j < 100; ++j) {
- map.remove(itos(j));
- }
- }
- if (i % 2 == 0) {
- map.clear();
- }
- }
-
- if (CountedItem::count != 0) {
- OS::get_singleton()->print("%d != 0 (not performing the other test sub-cases, breaking...)\n", CountedItem::count);
- break;
- }
- }
- }
-
- // Test map with 0 capacity.
- {
- OAHashMap<int, String> original_map(0);
- original_map.set(1, "1");
- OS::get_singleton()->print("OAHashMap 0 capacity initialization passed.\n");
- }
-
- // Test copy constructor.
- {
- OAHashMap<int, String> original_map;
- original_map.set(1, "1");
- original_map.set(2, "2");
- original_map.set(3, "3");
- original_map.set(4, "4");
- original_map.set(5, "5");
-
- OAHashMap<int, String> map_copy(original_map);
-
- bool pass = true;
- for (
- OAHashMap<int, String>::Iterator it = original_map.iter();
- it.valid;
- it = original_map.next_iter(it)) {
- if (map_copy.lookup_ptr(*it.key) == nullptr) {
- pass = false;
- }
- if (*it.value != *map_copy.lookup_ptr(*it.key)) {
- pass = false;
- }
- }
- if (pass) {
- OS::get_singleton()->print("OAHashMap copy constructor test passed.\n");
- } else {
- OS::get_singleton()->print("OAHashMap copy constructor test FAILED.\n");
- }
-
- map_copy.set(1, "Random String");
- if (*map_copy.lookup_ptr(1) == *original_map.lookup_ptr(1)) {
- OS::get_singleton()->print("OAHashMap copy constructor, atomic copy test FAILED.\n");
- } else {
- OS::get_singleton()->print("OAHashMap copy constructor, atomic copy test passed.\n");
- }
- }
-
- // Test assign operator.
- {
- OAHashMap<int, String> original_map;
- original_map.set(1, "1");
- original_map.set(2, "2");
- original_map.set(3, "3");
- original_map.set(4, "4");
- original_map.set(5, "5");
-
- OAHashMap<int, String> map_copy(100000);
- map_copy.set(1, "Just a string.");
- map_copy = original_map;
-
- bool pass = true;
- for (
- OAHashMap<int, String>::Iterator it = map_copy.iter();
- it.valid;
- it = map_copy.next_iter(it)) {
- if (original_map.lookup_ptr(*it.key) == nullptr) {
- pass = false;
- }
- if (*it.value != *original_map.lookup_ptr(*it.key)) {
- pass = false;
- }
- }
- if (pass) {
- OS::get_singleton()->print("OAHashMap assign operation test passed.\n");
- } else {
- OS::get_singleton()->print("OAHashMap assign operation test FAILED.\n");
- }
-
- map_copy.set(1, "Random String");
- if (*map_copy.lookup_ptr(1) == *original_map.lookup_ptr(1)) {
- OS::get_singleton()->print("OAHashMap assign operation atomic copy test FAILED.\n");
- } else {
- OS::get_singleton()->print("OAHashMap assign operation atomic copy test passed.\n");
- }
- }
-
- return nullptr;
-}
-} // namespace TestOAHashMap
diff --git a/tests/core/templates/test_oa_hash_map.h b/tests/core/templates/test_oa_hash_map.h
deleted file mode 100644
index d4b72af2ac..0000000000
--- a/tests/core/templates/test_oa_hash_map.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_oa_hash_map.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_OA_HASH_MAP_H
-#define TEST_OA_HASH_MAP_H
-
-class MainLoop;
-
-namespace TestOAHashMap {
-
-MainLoop *test();
-}
-
-#endif // TEST_OA_HASH_MAP_H
diff --git a/tests/scene/test_code_edit.h b/tests/scene/test_code_edit.h
index 8bd35df107..574cacda95 100644
--- a/tests/scene/test_code_edit.h
+++ b/tests/scene/test_code_edit.h
@@ -40,6 +40,7 @@ namespace TestCodeEdit {
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");
@@ -881,6 +882,7 @@ TEST_CASE("[SceneTree][CodeEdit] line gutters") {
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);
@@ -1759,6 +1761,7 @@ TEST_CASE("[SceneTree][CodeEdit] delimiters") {
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);
@@ -2288,6 +2291,7 @@ TEST_CASE("[SceneTree][CodeEdit] indent") {
TEST_CASE("[SceneTree][CodeEdit] folding") {
CodeEdit *code_edit = memnew(CodeEdit);
SceneTree::get_singleton()->get_root()->add_child(code_edit);
+ code_edit->grab_focus();
SUBCASE("[CodeEdit] folding settings") {
code_edit->set_line_folding_enabled(true);
@@ -2672,6 +2676,7 @@ TEST_CASE("[SceneTree][CodeEdit] folding") {
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);
@@ -2901,291 +2906,293 @@ TEST_CASE("[SceneTree][CodeEdit] completion") {
}
SUBCASE("[CodeEdit] autocomplete completion") {
- 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 */
- 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), RES(), 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() == 6);
- CHECK(code_edit->get_code_completion_options().size() == 1);
-
- /* Check cancel closes completion. */
- SEND_GUI_ACTION(code_edit, "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() == 6);
- 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"] == RES());
- CHECK(option["default_value"] == Color(1, 0, 0));
-
- /* Set size for mouse input. */
- code_edit->set_size(Size2(100, 100));
-
- /* Check input. */
- SEND_GUI_ACTION(code_edit, "ui_end");
- CHECK(code_edit->get_code_completion_selected_index() == 2);
-
- SEND_GUI_ACTION(code_edit, "ui_home");
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- SEND_GUI_ACTION(code_edit, "ui_page_down");
- CHECK(code_edit->get_code_completion_selected_index() == 2);
-
- SEND_GUI_ACTION(code_edit, "ui_page_up");
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- SEND_GUI_ACTION(code_edit, "ui_up");
- CHECK(code_edit->get_code_completion_selected_index() == 2);
-
- SEND_GUI_ACTION(code_edit, "ui_down");
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- SEND_GUI_KEY_EVENT(code_edit, Key::T);
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- SEND_GUI_ACTION(code_edit, "ui_left");
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- SEND_GUI_ACTION(code_edit, "ui_right");
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- SEND_GUI_ACTION(code_edit, "ui_text_backspace");
- CHECK(code_edit->get_code_completion_selected_index() == 0);
+ 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 */
+ 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), RES(), 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");
- Point2 caret_pos = code_edit->get_caret_draw_pos();
- caret_pos.y -= code_edit->get_line_height();
- SEND_GUI_MOUSE_EVENT(code_edit, caret_pos, MouseButton::WHEEL_DOWN, MouseButton::NONE);
- CHECK(code_edit->get_code_completion_selected_index() == 1);
-
- SEND_GUI_MOUSE_EVENT(code_edit, caret_pos, MouseButton::WHEEL_UP, MouseButton::NONE);
- CHECK(code_edit->get_code_completion_selected_index() == 0);
-
- /* Single click selects. */
- SEND_GUI_MOUSE_EVENT(code_edit, caret_pos, MouseButton::LEFT, MouseButton::MASK_LEFT);
- CHECK(code_edit->get_code_completion_selected_index() == 2);
-
- /* Double click inserts. */
- SEND_GUI_DOUBLE_CLICK(code_edit, caret_pos);
- CHECK(code_edit->get_code_completion_selected_index() == -1);
- CHECK(code_edit->get_line(0) == "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() == 6);
+ CHECK(code_edit->get_code_completion_options().size() == 1);
- code_edit->set_auto_brace_completion_enabled(false);
+ /* Check cancel closes completion. */
+ SEND_GUI_ACTION(code_edit, "ui_cancel");
+ CHECK(code_edit->get_code_completion_selected_index() == -1);
- /* 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");
+ 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() == 6);
+ CHECK(code_edit->get_code_completion_options().size() == 3);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0 test");
+ /* 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"] == RES());
+ CHECK(option["default_value"] == Color(1, 0, 0));
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "\"item_0\"");
+ /* Set size for mouse input. */
+ code_edit->set_size(Size2(100, 100));
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "\"item_0\" test");
+ /* Check input. */
+ SEND_GUI_ACTION(code_edit, "ui_end");
+ CHECK(code_edit->get_code_completion_selected_index() == 2);
- /* 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(code_edit, "ui_text_completion_accept");
- CHECK(code_edit->get_line(0) == "item_01 test");
+ SEND_GUI_ACTION(code_edit, "ui_home");
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_accept");
- CHECK(code_edit->get_line(0) == "\"item_0\"1 test\"");
+ SEND_GUI_ACTION(code_edit, "ui_page_down");
+ CHECK(code_edit->get_code_completion_selected_index() == 2);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0( test");
- CHECK(code_edit->get_caret_column() == 7);
+ SEND_GUI_ACTION(code_edit, "ui_page_up");
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0( test");
- CHECK(code_edit->get_caret_column() == 6);
+ SEND_GUI_ACTION(code_edit, "ui_up");
+ CHECK(code_edit->get_code_completion_selected_index() == 2);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0( test");
- CHECK(code_edit->get_caret_column() == 7);
+ SEND_GUI_ACTION(code_edit, "ui_down");
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 8);
+ SEND_GUI_KEY_EVENT(code_edit, Key::T);
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 6);
+ SEND_GUI_ACTION(code_edit, "ui_left");
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 8);
+ SEND_GUI_ACTION(code_edit, "ui_right");
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* Autobrace completion. */
- code_edit->set_auto_brace_completion_enabled(true);
+ SEND_GUI_ACTION(code_edit, "ui_text_backspace");
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 7);
+ Point2 caret_pos = code_edit->get_caret_draw_pos();
+ caret_pos.y -= code_edit->get_line_height();
+ SEND_GUI_MOUSE_BUTTON_EVENT(code_edit, caret_pos, MouseButton::WHEEL_DOWN, MouseButton::NONE, Key::NONE);
+ CHECK(code_edit->get_code_completion_selected_index() == 1);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0( test");
- CHECK(code_edit->get_caret_column() == 6);
+ SEND_GUI_MOUSE_BUTTON_EVENT(code_edit, caret_pos, MouseButton::WHEEL_UP, MouseButton::NONE, Key::NONE);
+ CHECK(code_edit->get_code_completion_selected_index() == 0);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0( test");
- CHECK(code_edit->get_caret_column() == 7);
+ /* Single click selects. */
+ SEND_GUI_MOUSE_BUTTON_EVENT(code_edit, caret_pos, MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(code_edit->get_code_completion_selected_index() == 2);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 8);
+ /* Double click inserts. */
+ SEND_GUI_DOUBLE_CLICK(code_edit, caret_pos, Key::NONE);
+ CHECK(code_edit->get_code_completion_selected_index() == -1);
+ CHECK(code_edit->get_line(0) == "item_2");
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 6);
+ code_edit->set_auto_brace_completion_enabled(false);
- /* 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(code_edit, "ui_text_completion_replace");
- CHECK(code_edit->get_line(0) == "item_0() test");
- CHECK(code_edit->get_caret_column() == 8);
+ /* 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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "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(code_edit, "ui_text_completion_replace");
+ CHECK(code_edit->get_line(0) == "item_0() test");
+ CHECK(code_edit->get_caret_column() == 8);
+ }
}
memdelete(code_edit);
@@ -3194,42 +3201,46 @@ TEST_CASE("[SceneTree][CodeEdit] completion") {
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());
- /* Set size for mouse input. */
- code_edit->set_size(Size2(100, 100));
+ 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");
+ code_edit->set_text("this is some text");
- Point2 caret_pos = code_edit->get_caret_draw_pos();
- caret_pos.x += 58;
- SEND_GUI_MOUSE_EVENT(code_edit, caret_pos, MouseButton::NONE, MouseButton::NONE);
- CHECK(code_edit->get_text_for_symbol_lookup() == "this is s" + String::chr(0xFFFF) + "ome text");
+ Point2 caret_pos = code_edit->get_caret_draw_pos();
+ caret_pos.x += 58;
+ SEND_GUI_MOUSE_BUTTON_EVENT(code_edit, caret_pos, MouseButton::NONE, MouseButton::NONE, Key::NONE);
+ CHECK(code_edit->get_text_for_symbol_lookup() == "this is s" + String::chr(0xFFFF) + "ome text");
- SIGNAL_WATCH(code_edit, "symbol_validate");
+ SIGNAL_WATCH(code_edit, "symbol_validate");
#ifdef OSX_ENABLED
- SEND_GUI_KEY_EVENT(code_edit, Key::META);
+ SEND_GUI_KEY_EVENT(code_edit, Key::META);
#else
- SEND_GUI_KEY_EVENT(code_edit, Key::CTRL);
+ SEND_GUI_KEY_EVENT(code_edit, Key::CTRL);
#endif
- Array signal_args;
- Array arg;
- arg.push_back("some");
- signal_args.push_back(arg);
- SIGNAL_CHECK("symbol_validate", signal_args);
+ Array signal_args;
+ Array arg;
+ arg.push_back("some");
+ signal_args.push_back(arg);
+ SIGNAL_CHECK("symbol_validate", signal_args);
- SIGNAL_UNWATCH(code_edit, "symbol_validate");
+ SIGNAL_UNWATCH(code_edit, "symbol_validate");
- memdelete(code_edit);
+ 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<int> guide_lines;
@@ -3250,6 +3261,7 @@ TEST_CASE("[SceneTree][CodeEdit] line length guidelines") {
TEST_CASE("[SceneTree][CodeEdit] Backspace delete") {
CodeEdit *code_edit = memnew(CodeEdit);
SceneTree::get_singleton()->get_root()->add_child(code_edit);
+ code_edit->grab_focus();
/* Backspace with selection on first line. */
code_edit->set_text("");
@@ -3297,6 +3309,7 @@ TEST_CASE("[SceneTree][CodeEdit] Backspace delete") {
TEST_CASE("[SceneTree][CodeEdit] New Line") {
CodeEdit *code_edit = memnew(CodeEdit);
SceneTree::get_singleton()->get_root()->add_child(code_edit);
+ code_edit->grab_focus();
/* Add a new line. */
code_edit->set_text("");
diff --git a/tests/scene/test_gui.cpp b/tests/scene/test_gui.cpp
deleted file mode 100644
index cd5624b70c..0000000000
--- a/tests/scene/test_gui.cpp
+++ /dev/null
@@ -1,259 +0,0 @@
-/*************************************************************************/
-/* test_gui.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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 _3D_DISABLED
-
-#include "test_gui.h"
-
-#include "scene/gui/button.h"
-#include "scene/gui/label.h"
-#include "scene/gui/line_edit.h"
-#include "scene/gui/menu_button.h"
-#include "scene/gui/option_button.h"
-#include "scene/gui/panel.h"
-#include "scene/gui/progress_bar.h"
-#include "scene/gui/rich_text_label.h"
-#include "scene/gui/scroll_bar.h"
-#include "scene/gui/spin_box.h"
-#include "scene/gui/tab_container.h"
-#include "scene/gui/tree.h"
-
-namespace TestGUI {
-
-class TestMainLoop : public SceneTree {
-public:
- virtual void request_quit() {
- quit();
- }
- virtual void initialize() {
- SceneTree::initialize();
-
- Panel *frame = memnew(Panel);
- frame->set_anchor(SIDE_RIGHT, Control::ANCHOR_END);
- frame->set_anchor(SIDE_BOTTOM, Control::ANCHOR_END);
- frame->set_end(Point2(0, 0));
-
- Ref<Theme> t = memnew(Theme);
- frame->set_theme(t);
-
- get_root()->add_child(frame);
-
- Label *label = memnew(Label);
-
- label->set_position(Point2(80, 90));
- label->set_size(Point2(170, 80));
- label->set_horizontal_alignment(HORIZONTAL_ALIGNMENT_FILL);
- label->set_text("There was once upon a time a beautiful unicorn that loved to play with little girls...");
-
- frame->add_child(label);
-
- Button *button = memnew(Button);
-
- button->set_position(Point2(20, 20));
- button->set_size(Point2(1, 1));
- button->set_text("This is a biggie button");
-
- frame->add_child(button);
-
- Tree *tree = memnew(Tree);
- tree->set_columns(2);
-
- tree->set_position(Point2(230, 210));
- tree->set_size(Point2(150, 250));
-
- TreeItem *item = tree->create_item();
- item->set_editable(0, true);
- item->set_text(0, "root");
- item = tree->create_item(tree->get_root());
- item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK);
- item->set_editable(0, true);
- item->set_text(0, "check");
- item->set_cell_mode(1, TreeItem::CELL_MODE_CHECK);
- item->set_editable(1, true);
- item->set_text(1, "check2");
- item = tree->create_item(tree->get_root());
- item->set_cell_mode(0, TreeItem::CELL_MODE_RANGE);
- item->set_editable(0, true);
- item->set_range_config(0, 0, 20, 0.1);
- item->set_range(0, 2);
- item->add_button(0, Theme::get_default()->get_icon("folder", "FileDialog"));
- item->set_cell_mode(1, TreeItem::CELL_MODE_RANGE);
- item->set_editable(1, true);
- item->set_range_config(1, 0, 20, 0.1);
- item->set_range(1, 3);
-
- item = tree->create_item(tree->get_root());
- item->set_cell_mode(0, TreeItem::CELL_MODE_RANGE);
- item->set_editable(0, true);
- item->set_text(0, "Have,Many,Several,Options!");
- item->set_range(0, 2);
-
- item = tree->create_item(item);
- item->set_editable(0, true);
- item->set_text(0, "Gershwin!");
-
- frame->add_child(tree);
-
- LineEdit *line_edit = memnew(LineEdit);
-
- line_edit->set_position(Point2(30, 190));
- line_edit->set_size(Point2(180, 1));
-
- frame->add_child(line_edit);
-
- HScrollBar *hscroll = memnew(HScrollBar);
-
- hscroll->set_position(Point2(30, 290));
- hscroll->set_size(Point2(180, 1));
- hscroll->set_max(10);
- hscroll->set_page(4);
-
- frame->add_child(hscroll);
-
- SpinBox *spin = memnew(SpinBox);
-
- spin->set_position(Point2(30, 260));
- spin->set_size(Point2(120, 1));
-
- frame->add_child(spin);
- hscroll->share(spin);
-
- ProgressBar *progress = memnew(ProgressBar);
-
- progress->set_position(Point2(30, 330));
- progress->set_size(Point2(120, 1));
-
- frame->add_child(progress);
- hscroll->share(progress);
-
- MenuButton *menu_button = memnew(MenuButton);
-
- menu_button->set_text("I'm a menu!");
- menu_button->set_position(Point2(30, 380));
- menu_button->set_size(Point2(1, 1));
-
- frame->add_child(menu_button);
-
- PopupMenu *popup = menu_button->get_popup();
-
- popup->add_item("Hello, testing");
- popup->add_item("My Dearest");
- popup->add_separator();
- popup->add_item("Popup");
- popup->add_check_item("Check Popup");
- popup->set_item_checked(4, true);
- popup->add_separator();
- popup->add_radio_check_item("Option A");
- popup->set_item_checked(6, true);
- popup->add_radio_check_item("Option B");
-
- OptionButton *options = memnew(OptionButton);
-
- options->add_item("Hello, testing");
- options->add_item("My Dearest");
-
- options->set_position(Point2(230, 180));
- options->set_size(Point2(1, 1));
-
- frame->add_child(options);
-
- RichTextLabel *richtext = memnew(RichTextLabel);
-
- richtext->set_position(Point2(600, 210));
- richtext->set_size(Point2(180, 250));
- richtext->set_anchor_and_offset(SIDE_RIGHT, Control::ANCHOR_END, -20);
-
- frame->add_child(richtext);
-
- richtext->add_text("Hello, My Friends!\n\nWelcome to the amazing world of ");
-
- richtext->add_newline();
- richtext->add_newline();
-
- richtext->push_color(Color(1, 0.5, 0.5));
- richtext->add_text("leprechauns");
- richtext->pop();
-
- richtext->add_text(" and ");
- richtext->push_color(Color(0, 1.0, 0.5));
- richtext->add_text("faeries.\n");
- richtext->pop();
- richtext->add_text("In this new episode, we will attempt to ");
- richtext->push_font(richtext->get_theme_font(SNAME("mono_font"), SNAME("Fonts")));
- richtext->push_color(Color(0.7, 0.5, 1.0));
- richtext->add_text("deliver something nice");
- richtext->pop();
- richtext->pop();
- richtext->add_text(" to all the viewers! Unfortunately, I need to ");
- richtext->push_underline();
- richtext->add_text("keep writing a lot of text");
- richtext->pop();
- richtext->add_text(" so the label control overflows and the scrollbar appears.\n");
- richtext->push_meta("http://www.scrollingcapabilities.xz");
- richtext->add_text("This allows to test for the scrolling capabilities ");
- richtext->pop();
- richtext->add_text("of the rich text label for huge text (not like this text will really be huge but, you know).\nAs long as it is so long that it will work nicely for a test/demo, then it's welcomed in my book...\nChanging subject, the day is cloudy today and I'm wondering if I'll get che chance to travel somewhere nice. Sometimes, watching the clouds from satellite images may give a nice insight about how pressure zones in our planet work, although it also makes it pretty obvious to see why most weather forecasts get it wrong so often.\nClouds are so difficult to predict!\nBut it's pretty cool how our civilization has adapted to having water falling from the sky each time it rains...");
-
- TabContainer *tabc = memnew(TabContainer);
-
- Control *ctl = memnew(Control);
- ctl->set_name("tab 1");
- tabc->add_child(ctl);
-
- ctl = memnew(Control);
- ctl->set_name("tab 2");
- tabc->add_child(ctl);
- label = memnew(Label);
- label->set_text("Some Label");
- label->set_position(Point2(20, 20));
- ctl->add_child(label);
-
- ctl = memnew(Control);
- ctl->set_name("tab 3");
- button = memnew(Button);
- button->set_text("Some Button");
- button->set_position(Point2(30, 50));
- ctl->add_child(button);
-
- tabc->add_child(ctl);
-
- frame->add_child(tabc);
-
- tabc->set_position(Point2(400, 210));
- tabc->set_size(Point2(180, 250));
- }
-};
-
-MainLoop *test() {
- return memnew(TestMainLoop);
-}
-} // namespace TestGUI
-
-#endif // _3D_DISABLED
diff --git a/tests/scene/test_gui.h b/tests/scene/test_gui.h
deleted file mode 100644
index a1807ed15c..0000000000
--- a/tests/scene/test_gui.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_gui.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_GUI_H
-#define TEST_GUI_H
-
-class MainLoop;
-
-namespace TestGUI {
-
-MainLoop *test();
-}
-
-#endif
diff --git a/tests/scene/test_text_edit.h b/tests/scene/test_text_edit.h
new file mode 100644
index 0000000000..249b645fae
--- /dev/null
+++ b/tests/scene/test_text_edit.h
@@ -0,0 +1,3507 @@
+/*************************************************************************/
+/* test_text_edit.h */
+/*************************************************************************/
+/* This file is part of: */
+/* GODOT ENGINE */
+/* https://godotengine.org */
+/*************************************************************************/
+/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
+/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
+/* */
+/* 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_TEXT_EDIT_H
+#define TEST_TEXT_EDIT_H
+
+#include "scene/gui/text_edit.h"
+
+#include "tests/test_macros.h"
+
+namespace TestTextEdit {
+
+TEST_CASE("[SceneTree][TextEdit] text entry") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+ text_edit->grab_focus();
+
+ Array empty_singal_args;
+ empty_singal_args.push_back(Array());
+
+ SUBCASE("[TextEdit] text entry") {
+ SIGNAL_WATCH(text_edit, "text_set");
+ SIGNAL_WATCH(text_edit, "text_changed");
+ SIGNAL_WATCH(text_edit, "lines_edited_from");
+ SIGNAL_WATCH(text_edit, "caret_changed");
+
+ Array args1;
+ args1.push_back(0);
+ args1.push_back(0);
+ Array lines_edited_args;
+ lines_edited_args.push_back(args1);
+ lines_edited_args.push_back(args1.duplicate());
+
+ SUBCASE("[TextEdit] clear and set text") {
+ // "text_changed" should not be emitted on clear / set.
+ text_edit->clear();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK(text_edit->get_line_count() == 1);
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->set_text("test text");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test text");
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK(text_edit->get_line_count() == 1);
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->clear();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ // Can undo / redo words when editable.
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test text");
+ CHECK(text_edit->get_caret_column() == 9);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->redo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Cannot undo when not-editable but should still clear.
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test text");
+ CHECK(text_edit->get_caret_column() == 9);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Clear.
+ text_edit->set_editable(false);
+
+ Array lines_edited_clear_args;
+ Array new_args = args1.duplicate();
+ new_args[0] = 1;
+ lines_edited_clear_args.push_back(new_args);
+
+ text_edit->clear();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_clear_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->set_editable(true);
+
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK_FALSE("text_set");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("caret_changed");
+
+ // Can still undo set_text.
+ text_edit->set_editable(false);
+
+ text_edit->set_text("test text");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test text");
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->set_editable(true);
+
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Any selections are removed.
+ text_edit->set_text("test text");
+ MessageQueue::get_singleton()->flush();
+ text_edit->select_all();
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test text");
+ CHECK(text_edit->get_caret_column() == 9);
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->set_text("test");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test");
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->select_all();
+ MessageQueue::get_singleton()->flush();
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ CHECK(text_edit->has_selection());
+
+ text_edit->clear();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ }
+
+ SUBCASE("[TextEdit] set and get line") {
+ // Set / Get line is 0 indexed.
+ text_edit->set_line(1, "test");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("text_set");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("caret_changed");
+
+ text_edit->set_line(0, "test");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "test");
+ CHECK(text_edit->get_line(0) == "test");
+ CHECK(text_edit->get_line(1) == "");
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+ SIGNAL_CHECK_FALSE("caret_changed");
+
+ // Setting to a longer line, caret and selections should be preserved.
+ text_edit->select_all();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ text_edit->set_line(0, "test text");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_line(0) == "test text");
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "test");
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Setting to a shorter line, selection and caret should be adjusted. Also works if not editable.
+ text_edit->set_editable(false);
+ text_edit->set_line(0, "te");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_line(0) == "te");
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "te");
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+ text_edit->set_editable(true);
+
+ // Undo / redo should work.
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_line(0) == "test text");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->redo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_line(0) == "te");
+ CHECK_FALSE(text_edit->has_selection()); // Currently not handled.
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Out of range.
+ ERR_PRINT_OFF;
+ text_edit->set_line(-1, "test");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_line(0) == "te");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->set_line(1, "test");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_line(0) == "te");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ ERR_PRINT_ON;
+ }
+
+ SUBCASE("[TextEdit] swap lines") {
+ ((Array)lines_edited_args[1])[1] = 1;
+
+ text_edit->set_text("testing\nswap");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "testing\nswap");
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ ((Array)lines_edited_args[1])[1] = 0;
+ Array swap_args;
+ swap_args.push_back(1);
+ swap_args.push_back(1);
+ lines_edited_args.push_back(swap_args);
+ lines_edited_args.push_back(swap_args);
+
+ // Order does not matter. Should also work if not editable.
+ text_edit->set_editable(false);
+ text_edit->swap_lines(1, 0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "swap\ntesting");
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+ text_edit->set_editable(true);
+
+ lines_edited_args.reverse();
+
+ // Single undo/redo action
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "testing\nswap");
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ lines_edited_args.reverse();
+
+ text_edit->redo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "swap\ntesting");
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Out of range.
+ ERR_PRINT_OFF;
+ text_edit->swap_lines(-1, 0);
+ CHECK(text_edit->get_text() == "swap\ntesting");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->swap_lines(0, -1);
+ CHECK(text_edit->get_text() == "swap\ntesting");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->swap_lines(2, 0);
+ CHECK(text_edit->get_text() == "swap\ntesting");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->swap_lines(0, 2);
+ CHECK(text_edit->get_text() == "swap\ntesting");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+ ERR_PRINT_ON;
+ }
+
+ SUBCASE("[TextEdit] insert line at") {
+ ((Array)lines_edited_args[1])[1] = 1;
+
+ text_edit->set_text("testing\nswap");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "testing\nswap");
+ SIGNAL_CHECK("text_set", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+
+ text_edit->select_all();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selection_from_line() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ // insert before should move caret and selecion, and works when not editable.
+ text_edit->set_editable(false);
+ lines_edited_args.remove_at(0);
+ text_edit->insert_line_at(0, "new");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "new\ntesting\nswap");
+ CHECK(text_edit->get_caret_line() == 2);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(2).size() - 1);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selection_from_line() == 1);
+ CHECK(text_edit->get_selection_to_line() == 2);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+ text_edit->set_editable(true);
+
+ // can undo/redo as single action
+ ((Array)lines_edited_args[0])[0] = 1;
+ ((Array)lines_edited_args[0])[1] = 0;
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "testing\nswap");
+ CHECK_FALSE(text_edit->has_selection()); // Not currently handled.
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ ((Array)lines_edited_args[0])[0] = 0;
+ ((Array)lines_edited_args[0])[1] = 1;
+ text_edit->redo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "new\ntesting\nswap");
+ CHECK_FALSE(text_edit->has_selection()); // Not currently handled.
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Adding inside selection extends selection.
+ text_edit->select_all();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selection_from_line() == 0);
+ CHECK(text_edit->get_selection_to_line() == 2);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ ((Array)lines_edited_args[0])[0] = 2;
+ ((Array)lines_edited_args[0])[1] = 3;
+ text_edit->insert_line_at(2, "after");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "new\ntesting\nafter\nswap");
+ CHECK(text_edit->get_caret_line() == 3);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(3).size() - 1);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selection_from_line() == 0);
+ CHECK(text_edit->get_selection_to_line() == 3);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ // Out of range.
+ ERR_PRINT_OFF;
+ text_edit->insert_line_at(-1, "after");
+ CHECK(text_edit->get_text() == "new\ntesting\nafter\nswap");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->insert_line_at(4, "after");
+ CHECK(text_edit->get_text() == "new\ntesting\nafter\nswap");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("text_set");
+ ERR_PRINT_ON;
+ }
+
+ SUBCASE("[TextEdit] insert line at caret") {
+ lines_edited_args.pop_back();
+ ((Array)lines_edited_args[0])[1] = 1;
+
+ text_edit->insert_text_at_caret("testing\nswap");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "testing\nswap");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(1).size() - 1);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->set_caret_line(0, false);
+ text_edit->set_caret_column(2);
+ SIGNAL_DISCARD("caret_changed");
+
+ ((Array)lines_edited_args[0])[1] = 0;
+ text_edit->insert_text_at_caret("mid");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "temidsting\nswap");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 5);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->select(0, 0, 0, text_edit->get_line(0).length());
+ CHECK(text_edit->has_selection());
+ lines_edited_args.push_back(args1.duplicate());
+
+ text_edit->set_editable(false);
+ text_edit->insert_text_at_caret("new line");
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "new line\nswap");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).size() - 1);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+ text_edit->set_editable(true);
+
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "temidsting\nswap");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 10);
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+
+ text_edit->redo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "new line\nswap");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_set");
+ }
+
+ SIGNAL_UNWATCH(text_edit, "text_set");
+ SIGNAL_UNWATCH(text_edit, "text_changed");
+ SIGNAL_UNWATCH(text_edit, "lines_edited_from");
+ SIGNAL_UNWATCH(text_edit, "caret_changed");
+ }
+
+ SUBCASE("[TextEdit] indent level") {
+ CHECK(text_edit->get_indent_level(0) == 0);
+ CHECK(text_edit->get_first_non_whitespace_column(0) == 0);
+
+ text_edit->set_line(0, "a");
+ CHECK(text_edit->get_indent_level(0) == 0);
+ CHECK(text_edit->get_first_non_whitespace_column(0) == 0);
+
+ text_edit->set_line(0, "\t");
+ CHECK(text_edit->get_indent_level(0) == 4);
+ CHECK(text_edit->get_first_non_whitespace_column(0) == 1);
+
+ text_edit->set_tab_size(8);
+ CHECK(text_edit->get_indent_level(0) == 8);
+
+ text_edit->set_line(0, "\t a");
+ CHECK(text_edit->get_first_non_whitespace_column(0) == 2);
+ CHECK(text_edit->get_indent_level(0) == 9);
+ }
+
+ SUBCASE("[TextEdit] selection") {
+ SIGNAL_WATCH(text_edit, "text_set");
+ SIGNAL_WATCH(text_edit, "text_changed");
+ SIGNAL_WATCH(text_edit, "lines_edited_from");
+ SIGNAL_WATCH(text_edit, "caret_changed");
+
+ Array args1;
+ args1.push_back(0);
+ args1.push_back(0);
+ Array lines_edited_args;
+ lines_edited_args.push_back(args1);
+ lines_edited_args.push_back(args1.duplicate());
+
+ SUBCASE("[TextEdit] select all") {
+ text_edit->select_all();
+ CHECK_FALSE(text_edit->has_selection());
+ ERR_PRINT_OFF;
+ CHECK(text_edit->get_selection_from_line() == -1);
+ CHECK(text_edit->get_selection_from_column() == -1);
+ CHECK(text_edit->get_selection_to_line() == -1);
+ CHECK(text_edit->get_selection_to_column() == -1);
+ CHECK(text_edit->get_selected_text() == "");
+ ERR_PRINT_ON;
+
+ text_edit->set_text("test\nselection");
+ SEND_GUI_ACTION(text_edit, "ui_text_select_all");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_selected_text() == "test\nselection");
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selection_from_line() == 0);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ CHECK(text_edit->get_selection_to_column() == 9);
+ CHECK(text_edit->get_selection_mode() == TextEdit::SelectionMode::SELECTION_MODE_SHIFT);
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 9);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(0);
+ text_edit->set_selecting_enabled(false);
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+
+ text_edit->select_all();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ }
+
+ SUBCASE("[TextEdit] select word under caret") {
+ text_edit->set_text("test test");
+ text_edit->set_caret_column(0);
+ text_edit->select_word_under_caret();
+ CHECK(text_edit->get_selected_text() == "test");
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selection_from_line() == 0);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 0);
+ CHECK(text_edit->get_selection_to_column() == 4);
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 4);
+
+ text_edit->select_word_under_caret();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_select_word_under_caret");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "test");
+ CHECK(text_edit->get_selection_from_line() == 0);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 0);
+ CHECK(text_edit->get_selection_to_column() == 4);
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 4);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ text_edit->set_selecting_enabled(false);
+ text_edit->select_word_under_caret();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 4);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ text_edit->set_selecting_enabled(true);
+
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(5);
+ text_edit->select_word_under_caret();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+
+ text_edit->select_word_under_caret();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 5);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ }
+
+ SUBCASE("[TextEdit] deselect on focus loss") {
+ text_edit->set_text("test");
+
+ text_edit->set_deselect_on_focus_loss_enabled(true);
+ CHECK(text_edit->is_deselect_on_focus_loss_enabled());
+
+ text_edit->grab_focus();
+ text_edit->select_all();
+ CHECK(text_edit->has_focus());
+ CHECK(text_edit->has_selection());
+
+ text_edit->release_focus();
+ CHECK_FALSE(text_edit->has_focus());
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->set_deselect_on_focus_loss_enabled(false);
+ CHECK_FALSE(text_edit->is_deselect_on_focus_loss_enabled());
+
+ text_edit->grab_focus();
+ text_edit->select_all();
+ CHECK(text_edit->has_focus());
+ CHECK(text_edit->has_selection());
+
+ text_edit->release_focus();
+ CHECK_FALSE(text_edit->has_focus());
+ CHECK(text_edit->has_selection());
+
+ text_edit->set_deselect_on_focus_loss_enabled(true);
+ CHECK_FALSE(text_edit->has_selection());
+ }
+
+ SUBCASE("[TextEdit] key select") {
+ text_edit->set_text("test");
+
+ text_edit->grab_focus();
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::SHIFT)
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "t");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::SHIFT | KeyModifierMask::ALT)
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::SHIFT | KeyModifierMask::CMD)
+#endif
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "test");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::SHIFT)
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "tes");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::SHIFT | KeyModifierMask::ALT)
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::SHIFT | KeyModifierMask::CMD)
+#endif
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::SHIFT)
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "t");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT)
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::SHIFT)
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "t");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT)
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+
+ text_edit->set_selecting_enabled(false);
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::SHIFT)
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "");
+ text_edit->set_selecting_enabled(true);
+ }
+
+ SUBCASE("[TextEdit] mouse drag select") {
+ /* Set size for mouse input. */
+ text_edit->set_size(Size2(200, 200));
+
+ text_edit->set_text("this is some text\nfor selection");
+ text_edit->grab_focus();
+ MessageQueue::get_singleton()->flush();
+
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 1), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ SEND_GUI_MOUSE_MOTION_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 7), MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "for s");
+ CHECK(text_edit->get_selection_mode() == TextEdit::SELECTION_MODE_POINTER);
+ CHECK(text_edit->get_selection_from_line() == 1);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ CHECK(text_edit->get_selection_to_column() == 5);
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 5);
+
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 9), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->set_selecting_enabled(false);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 1), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ SEND_GUI_MOUSE_MOTION_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 7), MouseButton::MASK_LEFT, Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 5);
+ text_edit->set_selecting_enabled(true);
+ }
+
+ SUBCASE("[TextEdit] mouse word select") {
+ /* Set size for mouse input. */
+ text_edit->set_size(Size2(200, 200));
+
+ text_edit->set_text("this is some text\nfor selection");
+ MessageQueue::get_singleton()->flush();
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_DOUBLE_CLICK(text_edit, text_edit->get_pos_at_line_column(0, 2), Key::NONE);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "for");
+ CHECK(text_edit->get_selection_mode() == TextEdit::SELECTION_MODE_WORD);
+ CHECK(text_edit->get_selection_from_line() == 1);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ CHECK(text_edit->get_selection_to_column() == 3);
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 3);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ SEND_GUI_MOUSE_MOTION_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 7), MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "for selection");
+ CHECK(text_edit->get_selection_mode() == TextEdit::SELECTION_MODE_WORD);
+ CHECK(text_edit->get_selection_from_line() == 1);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ CHECK(text_edit->get_selection_to_column() == 13);
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 13);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+
+ Point2i line_0 = text_edit->get_pos_at_line_column(0, 0);
+ line_0.y /= 2;
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, line_0, MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->set_selecting_enabled(false);
+ SEND_GUI_DOUBLE_CLICK(text_edit, text_edit->get_pos_at_line_column(0, 2), Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 3);
+ text_edit->set_selecting_enabled(true);
+ }
+
+ SUBCASE("[TextEdit] mouse line select") {
+ /* Set size for mouse input. */
+ text_edit->set_size(Size2(200, 200));
+
+ text_edit->set_text("this is some text\nfor selection");
+ MessageQueue::get_singleton()->flush();
+
+ SEND_GUI_DOUBLE_CLICK(text_edit, text_edit->get_pos_at_line_column(0, 2), Key::NONE);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 2), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "for selection");
+ CHECK(text_edit->get_selection_mode() == TextEdit::SELECTION_MODE_LINE);
+ CHECK(text_edit->get_selection_from_line() == 1);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ CHECK(text_edit->get_selection_to_column() == 13);
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+
+ Point2i line_0 = text_edit->get_pos_at_line_column(0, 0);
+ line_0.y /= 2;
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, line_0, MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->set_selecting_enabled(false);
+ SEND_GUI_DOUBLE_CLICK(text_edit, text_edit->get_pos_at_line_column(0, 2), Key::NONE);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 2), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ text_edit->set_selecting_enabled(true);
+ }
+
+ SUBCASE("[TextEdit] mouse shift click select") {
+ /* Set size for mouse input. */
+ text_edit->set_size(Size2(200, 200));
+
+ text_edit->set_text("this is some text\nfor selection");
+ MessageQueue::get_singleton()->flush();
+
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 0), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 7), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE | KeyModifierMask::SHIFT);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "for s");
+ CHECK(text_edit->get_selection_mode() == TextEdit::SELECTION_MODE_POINTER);
+ CHECK(text_edit->get_selection_from_line() == 1);
+ CHECK(text_edit->get_selection_from_column() == 0);
+ CHECK(text_edit->get_selection_to_line() == 1);
+ CHECK(text_edit->get_selection_to_column() == 5);
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 5);
+
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 9), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->set_selecting_enabled(false);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 0), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 7), MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE | KeyModifierMask::SHIFT);
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 5);
+ text_edit->set_selecting_enabled(true);
+ }
+
+ SUBCASE("[TextEdit] select and deselect") {
+ text_edit->set_text("this is some text\nfor selection");
+ MessageQueue::get_singleton()->flush();
+
+ text_edit->select(-1, -1, 500, 500);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "this is some text\nfor selection");
+
+ text_edit->deselect();
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->select(500, 500, -1, -1);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "this is some text\nfor selection");
+
+ text_edit->deselect();
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->select(0, 4, 0, 8);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == " is ");
+
+ text_edit->deselect();
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->select(0, 8, 0, 4);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == " is ");
+
+ text_edit->set_selecting_enabled(false);
+ CHECK_FALSE(text_edit->has_selection());
+ text_edit->select(0, 8, 0, 4);
+ CHECK_FALSE(text_edit->has_selection());
+ text_edit->set_selecting_enabled(true);
+
+ text_edit->select(0, 8, 0, 4);
+ CHECK(text_edit->has_selection());
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK_FALSE(text_edit->has_selection());
+
+ text_edit->delete_selection();
+ CHECK(text_edit->get_text() == "this is some text\nfor selection");
+
+ text_edit->select(0, 8, 0, 4);
+ CHECK(text_edit->has_selection());
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace");
+ CHECK(text_edit->get_text() == "thissome text\nfor selection");
+
+ text_edit->undo();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "this is some text\nfor selection");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+
+ text_edit->redo();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "thissome text\nfor selection");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+
+ text_edit->undo();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "this is some text\nfor selection");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+
+ text_edit->select(0, 8, 0, 4);
+ CHECK(text_edit->has_selection());
+
+ text_edit->delete_selection();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "thissome text\nfor selection");
+
+ text_edit->undo();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "this is some text\nfor selection");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+
+ text_edit->redo();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "thissome text\nfor selection");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+
+ text_edit->undo();
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "this is some text\nfor selection");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 8);
+
+ text_edit->set_editable(false);
+ text_edit->delete_selection();
+ text_edit->set_editable(false);
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "thissome text\nfor selection");
+
+ text_edit->undo();
+ CHECK_FALSE(text_edit->has_selection());
+ CHECK(text_edit->get_text() == "thissome text\nfor selection");
+ }
+
+ // Add readonly test?
+ SUBCASE("[TextEdit] text drag") {
+ TextEdit *target_text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(target_text_edit);
+ text_edit->get_viewport()->set_embedding_subwindows(true); // Bypass display server for drop handling.
+
+ target_text_edit->set_size(Size2(200, 200));
+ target_text_edit->set_position(Point2(400, 0));
+
+ text_edit->set_size(Size2(200, 200));
+
+ CHECK_FALSE(text_edit->is_mouse_over_selection());
+ text_edit->set_text("drag me");
+ text_edit->select_all();
+ text_edit->grab_click_focus();
+ MessageQueue::get_singleton()->flush();
+
+ Point2i line_0 = text_edit->get_pos_at_line_column(0, 0);
+ line_0.y /= 2;
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, line_0, MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(text_edit->is_mouse_over_selection());
+ SEND_GUI_MOUSE_MOTION_EVENT(text_edit, text_edit->get_pos_at_line_column(0, 7), MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(text_edit->get_viewport()->gui_is_dragging());
+ CHECK(text_edit->get_viewport()->gui_get_drag_data() == "drag me");
+
+ line_0 = target_text_edit->get_pos_at_line_column(0, 0);
+ line_0.y /= 2;
+ line_0.x += 401; // As empty add one.
+ SEND_GUI_MOUSE_MOTION_EVENT(target_text_edit, line_0, MouseButton::MASK_LEFT, Key::NONE);
+ CHECK(text_edit->get_viewport()->gui_is_dragging());
+
+ SEND_GUI_MOUSE_BUTTON_RELEASED_EVENT(target_text_edit, line_0, MouseButton::LEFT, MouseButton::MASK_LEFT, Key::NONE);
+
+ CHECK_FALSE(text_edit->get_viewport()->gui_is_dragging());
+ CHECK(text_edit->get_text() == "");
+ CHECK(target_text_edit->get_text() == "drag me");
+
+ memdelete(target_text_edit);
+ }
+
+ SIGNAL_UNWATCH(text_edit, "text_set");
+ SIGNAL_UNWATCH(text_edit, "text_changed");
+ SIGNAL_UNWATCH(text_edit, "lines_edited_from");
+ SIGNAL_UNWATCH(text_edit, "caret_changed");
+ }
+
+ SUBCASE("[TextEdit] overridable actions") {
+ SIGNAL_WATCH(text_edit, "text_set");
+ SIGNAL_WATCH(text_edit, "text_changed");
+ SIGNAL_WATCH(text_edit, "lines_edited_from");
+ SIGNAL_WATCH(text_edit, "caret_changed");
+
+ Array args1;
+ args1.push_back(0);
+ args1.push_back(0);
+ Array lines_edited_args;
+ lines_edited_args.push_back(args1);
+
+ SUBCASE("[TextEdit] backspace") {
+ text_edit->set_text("this is\nsome\n");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->backspace();
+ MessageQueue::get_singleton()->flush();
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ text_edit->set_caret_line(2);
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+ SIGNAL_DISCARD("caret_changed");
+
+ ((Array)lines_edited_args[0])[0] = 2;
+ ((Array)lines_edited_args[0])[1] = 1;
+ text_edit->backspace();
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_text() == "this is\nsome");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 4);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[0] = 1;
+ text_edit->backspace();
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_text() == "this is\nsom");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 3);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->end_complex_operation();
+ text_edit->select(1, 0, 1, 3);
+ text_edit->backspace();
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_text() == "this is\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_editable(false);
+ text_edit->backspace();
+ text_edit->set_editable(true);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_text() == "this is\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "this is\nsom");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 3);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ }
+
+ SUBCASE("[TextEdit] cut") {
+ text_edit->set_text("this is\nsome\n");
+ text_edit->set_caret_line(0);
+ text_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");
+
+ ERR_PRINT_OFF;
+ text_edit->cut();
+ MessageQueue::get_singleton()->flush();
+ ERR_PRINT_ON; // Can't check display server content.
+
+ ((Array)lines_edited_args[0])[0] = 1;
+ CHECK(text_edit->get_text() == "some\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 4);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[0] = 0;
+ ((Array)lines_edited_args[0])[1] = 1;
+ text_edit->undo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "this is\nsome\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[0] = 1;
+ ((Array)lines_edited_args[0])[1] = 0;
+ text_edit->redo();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "some\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_text("this is\nsome\n");
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ ((Array)lines_edited_args[0])[0] = 0;
+ text_edit->select(0, 5, 0, 7);
+ ERR_PRINT_OFF;
+ SEND_GUI_ACTION(text_edit, "ui_cut");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ MessageQueue::get_singleton()->flush();
+ ERR_PRINT_ON; // Can't check display server content.
+ CHECK(text_edit->get_text() == "this \nsome\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 5);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_editable(false);
+ text_edit->cut();
+ MessageQueue::get_singleton()->flush();
+ text_edit->set_editable(true);
+ CHECK(text_edit->get_text() == "this \nsome\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 5);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] copy") {
+ // TODO: Cannot test need display server support.
+ }
+
+ SUBCASE("[TextEdit] paste") {
+ // TODO: Cannot test need display server support.
+ }
+
+ SUBCASE("[TextEdit] paste primary") {
+ // TODO: Cannot test need display server support.
+ }
+
+ SIGNAL_UNWATCH(text_edit, "text_set");
+ SIGNAL_UNWATCH(text_edit, "text_changed");
+ SIGNAL_UNWATCH(text_edit, "lines_edited_from");
+ SIGNAL_UNWATCH(text_edit, "caret_changed");
+ }
+
+ // Add undo / redo tests?
+ SUBCASE("[TextEdit] input") {
+ SIGNAL_WATCH(text_edit, "text_set");
+ SIGNAL_WATCH(text_edit, "text_changed");
+ SIGNAL_WATCH(text_edit, "lines_edited_from");
+ SIGNAL_WATCH(text_edit, "caret_changed");
+
+ Array args1;
+ args1.push_back(0);
+ args1.push_back(0);
+ Array lines_edited_args;
+ lines_edited_args.push_back(args1);
+
+ SUBCASE("[TextEdit] ui_text_newline_above") {
+ text_edit->set_text("this is some test text.");
+ text_edit->select(0, 0, 0, 4);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ ((Array)lines_edited_args[0])[1] = 1;
+ SEND_GUI_ACTION(text_edit, "ui_text_newline_above");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_caret_line(1);
+ text_edit->set_caret_column(4);
+ text_edit->select(0, 0, 0, 4);
+ MessageQueue::get_singleton()->flush();
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_newline_above");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 4);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_newline_above");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ }
+
+ SUBCASE("[TextEdit] ui_text_newline_blank") {
+ text_edit->set_text("this is some test text.");
+ text_edit->select(0, 0, 0, 4);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ ((Array)lines_edited_args[0])[1] = 1;
+ SEND_GUI_ACTION(text_edit, "ui_text_newline_blank");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text.\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_newline_blank");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text.\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+ }
+
+ SUBCASE("[TextEdit] ui_text_newline") {
+ text_edit->set_text("this is some test text.");
+ text_edit->select(0, 0, 0, 4);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ lines_edited_args.push_back(lines_edited_args[0].duplicate());
+ ((Array)lines_edited_args[1])[1] = 1;
+ SEND_GUI_ACTION(text_edit, "ui_text_newline");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_newline");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+ }
+
+ SUBCASE("[TextEdit] ui_text_backspace_all_to_left") {
+ text_edit->set_text("\nthis is some test text.");
+ text_edit->select(1, 0, 1, 4);
+ text_edit->set_caret_line(1);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ Ref<InputEvent> tmpevent = InputEventKey::create_reference(Key::BACKSPACE | KeyModifierMask::ALT | KeyModifierMask::CMD);
+ InputMap::get_singleton()->action_add_event("ui_text_backspace_all_to_left", tmpevent);
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ // With selection should be a normal backsapce.
+ ((Array)lines_edited_args[0])[0] = 1;
+ ((Array)lines_edited_args[0])[1] = 1;
+
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_all_to_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[1] = 0;
+
+ // Start of line should also be a normal backspace.
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_all_to_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_all_to_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).length());
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ ((Array)lines_edited_args[0])[0] = 0;
+
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_all_to_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ InputMap::get_singleton()->action_erase_event("ui_text_backspace_all_to_left", tmpevent);
+ }
+
+ SUBCASE("[TextEdit] ui_text_backspace_word") {
+ text_edit->set_text("\nthis is some test text.");
+ text_edit->select(1, 0, 1, 4);
+ text_edit->set_caret_line(1);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ // With selection should be a normal backsapce.
+ ((Array)lines_edited_args[0])[0] = 1;
+ ((Array)lines_edited_args[0])[1] = 1;
+
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ text_edit->end_complex_operation();
+
+ ((Array)lines_edited_args[0])[1] = 0;
+
+ // Start of line should also be a normal backspace.
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ ((Array)lines_edited_args[0])[0] = 0;
+
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test ");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 14);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ }
+
+ SUBCASE("[TextEdit] ui_text_backspace") {
+ text_edit->set_text("\nthis is some test text.");
+ text_edit->select(1, 0, 1, 4);
+ text_edit->set_caret_line(1);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ // With selection should be a normal backsapce.
+ ((Array)lines_edited_args[0])[0] = 1;
+ ((Array)lines_edited_args[0])[1] = 1;
+
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[1] = 0;
+
+ // Start of line should also be a normal backspace.
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).length());
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ ((Array)lines_edited_args[0])[0] = 0;
+
+ SEND_GUI_ACTION(text_edit, "ui_text_backspace");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 18);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ }
+
+ SUBCASE("[TextEdit] ui_text_delete_all_to_right") {
+ Ref<InputEvent> tmpevent = InputEventKey::create_reference(Key::BACKSPACE | KeyModifierMask::ALT | KeyModifierMask::CMD);
+ InputMap::get_singleton()->action_add_event("ui_text_delete_all_to_right", tmpevent);
+
+ text_edit->set_text("this is some test text.\n");
+ text_edit->select(0, 0, 0, 4);
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ // With selection should be a normal delete.
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_all_to_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ // End of line should not do anything.
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_all_to_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).length());
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_all_to_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " is some test text.\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_all_to_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ InputMap::get_singleton()->action_erase_event("ui_text_delete_all_to_right", tmpevent);
+ }
+
+ SUBCASE("[TextEdit] ui_text_delete_word") {
+ text_edit->set_caret_mid_grapheme_enabled(true);
+ CHECK(text_edit->is_caret_mid_grapheme_enabled());
+
+ text_edit->set_text("this ffi some test text.\n");
+ text_edit->select(0, 0, 0, 4);
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ // With selection should be a normal delete.
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " ffi some test text.\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ // With selection should be a normal delete.
+ ((Array)lines_edited_args[0])[0] = 1;
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " ffi some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).length());
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[0] = 0;
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " ffi some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete_word");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ }
+
+ SUBCASE("[TextEdit] ui_text_delete") {
+ text_edit->set_caret_mid_grapheme_enabled(true);
+ CHECK(text_edit->is_caret_mid_grapheme_enabled());
+
+ text_edit->set_text("this ffi some test text.\n");
+ text_edit->select(0, 0, 0, 4);
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(4);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ // With selection should be a normal delete.
+ SEND_GUI_ACTION(text_edit, "ui_text_delete");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " ffi some test text.\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ // With selection should be a normal delete.
+ ((Array)lines_edited_args[0])[0] = 1;
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " ffi some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).length());
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ ((Array)lines_edited_args[0])[0] = 0;
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ text_edit->set_editable(false);
+ SEND_GUI_ACTION(text_edit, "ui_text_delete");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " ffi some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "ffi some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "fi some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_caret_mid_grapheme_enabled(false);
+ CHECK_FALSE(text_edit->is_caret_mid_grapheme_enabled());
+
+ text_edit->undo();
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_text() == "ffi some test text.");
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_delete");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == " some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_word_left") {
+ text_edit->set_text("\nthis is some test text.");
+ text_edit->set_caret_line(1);
+ text_edit->set_caret_column(7);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::ALT | KeyModifierMask::SHIFT);
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#endif
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 5);
+ CHECK(text_edit->get_selected_text() == "is");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_word_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_word_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_left") {
+ text_edit->set_text("\nthis is some test text.");
+ text_edit->set_caret_line(1);
+ text_edit->set_caret_column(7);
+ text_edit->select(1, 2, 1, 7);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 2);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::SHIFT);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 1);
+ CHECK(text_edit->get_selected_text() == "h");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 1);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "\nthis is some test text.");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_word_right") {
+ text_edit->set_text("this is some test text\n");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(13);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::ALT | KeyModifierMask::SHIFT);
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#endif
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 17);
+ CHECK(text_edit->get_selected_text() == "test");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_word_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 22);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_word_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_right") {
+ text_edit->set_text("this is some test text\n");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(16);
+ text_edit->select(0, 16, 0, 20);
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 20);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::SHIFT);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 21);
+ CHECK(text_edit->get_selected_text() == "x");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 21);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 22);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some test text\n");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_up") {
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+
+ text_edit->set_size(Size2(110, 100));
+ text_edit->set_text("this is some\nother test\nlines\ngo here");
+ text_edit->set_caret_line(4);
+ text_edit->set_caret_column(7);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->is_line_wrapped(0));
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::UP | KeyModifierMask::SHIFT);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->get_caret_line() == 2);
+ CHECK(text_edit->get_caret_column() == 5);
+ CHECK(text_edit->get_selected_text() == "\ngo here");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 8);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 12);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_caret_column(12, false);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 7);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_down") {
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+
+ text_edit->set_size(Size2(110, 100));
+ text_edit->set_text("go here\nlines\nother test\nthis is some");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(7);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->is_line_wrapped(3));
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::DOWN | KeyModifierMask::SHIFT);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_caret_column() == 5);
+ CHECK(text_edit->get_selected_text() == "\nlines");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->get_caret_line() == 2);
+ CHECK(text_edit->get_caret_column() == 8);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->get_caret_line() == 3);
+ CHECK(text_edit->get_caret_column() == 7);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_caret_column(7, false);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->get_caret_line() == 3);
+ CHECK(text_edit->get_caret_column() == 12);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_document_start") {
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+
+ text_edit->set_size(Size2(110, 100));
+ text_edit->set_text("this is some\nother test\nlines\ngo here");
+ text_edit->set_caret_line(4);
+ text_edit->set_caret_column(7);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->is_line_wrapped(0));
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::UP | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::HOME | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#endif
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK(text_edit->get_selected_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_document_start");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "this is some\nother test\nlines\ngo here");
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_document_end") {
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+
+ text_edit->set_size(Size2(110, 100));
+ text_edit->set_text("go here\nlines\nother test\nthis is some");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->is_line_wrapped(3));
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::DOWN | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::END | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#endif
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->get_caret_line() == 3);
+ CHECK(text_edit->get_caret_column() == 12);
+ CHECK(text_edit->get_selected_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_document_end");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "go here\nlines\nother test\nthis is some");
+ CHECK(text_edit->get_caret_line() == 3);
+ CHECK(text_edit->get_caret_column() == 12);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_line_start") {
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+
+ text_edit->set_size(Size2(110, 100));
+ text_edit->set_text(" this is some");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(text_edit->get_line(0).length());
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->is_line_wrapped(0));
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::LEFT | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::HOME | KeyModifierMask::SHIFT);
+#endif
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 10);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == "some");
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_line_start");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 2);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_line_start");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 0);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_line_start");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 2);
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] ui_text_caret_line_end") {
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+
+ text_edit->set_size(Size2(110, 100));
+ text_edit->set_text(" this is some");
+ text_edit->set_caret_line(0);
+ text_edit->set_caret_column(0);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->is_line_wrapped(0));
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+#ifdef OSX_ENABLED
+ SEND_GUI_KEY_EVENT(text_edit, Key::RIGHT | KeyModifierMask::CMD | KeyModifierMask::SHIFT);
+#else
+ SEND_GUI_KEY_EVENT(text_edit, Key::END | KeyModifierMask::SHIFT);
+#endif
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == 9);
+ CHECK(text_edit->has_selection());
+ CHECK(text_edit->get_selected_text() == " this is");
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_line_end");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 0);
+ CHECK(text_edit->get_caret_column() == text_edit->get_line(0).length());
+ CHECK_FALSE(text_edit->has_selection());
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ }
+
+ SUBCASE("[TextEdit] unicode") {
+ text_edit->insert_text_at_caret("a");
+ MessageQueue::get_singleton()->flush();
+
+ SIGNAL_DISCARD("text_set");
+ SIGNAL_DISCARD("text_changed");
+ SIGNAL_DISCARD("lines_edited_from");
+ SIGNAL_DISCARD("caret_changed");
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::A);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "aA");
+ CHECK(text_edit->get_caret_column() == 2);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->set_editable(false);
+ SEND_GUI_KEY_EVENT(text_edit, Key::A);
+ CHECK_FALSE(text_edit->get_viewport()->is_input_handled()); // Should this be handled?
+ CHECK(text_edit->get_text() == "aA");
+ CHECK(text_edit->get_caret_column() == 2);
+ SIGNAL_CHECK_FALSE("caret_changed");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ text_edit->set_editable(true);
+
+ lines_edited_args.push_back(lines_edited_args[0].duplicate());
+
+ text_edit->select(0, 0, 0, 1);
+ SEND_GUI_KEY_EVENT(text_edit, Key::B);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "BA");
+ CHECK(text_edit->get_caret_column() == 1);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_toggle_insert_mode");
+ CHECK(text_edit->is_overtype_mode_enabled());
+
+ SEND_GUI_KEY_EVENT(text_edit, Key::B);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "BB");
+ CHECK(text_edit->get_caret_column() == 2);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+
+ text_edit->select(0, 0, 0, 1);
+ SEND_GUI_KEY_EVENT(text_edit, Key::A);
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_text() == "AB");
+ CHECK(text_edit->get_caret_column() == 1);
+ SIGNAL_CHECK("caret_changed", empty_singal_args);
+ SIGNAL_CHECK("text_changed", empty_singal_args);
+ SIGNAL_CHECK("lines_edited_from", lines_edited_args);
+ text_edit->set_overtype_mode_enabled(false);
+ CHECK_FALSE(text_edit->is_overtype_mode_enabled());
+ }
+
+ SIGNAL_UNWATCH(text_edit, "text_set");
+ SIGNAL_UNWATCH(text_edit, "text_changed");
+ SIGNAL_UNWATCH(text_edit, "lines_edited_from");
+ SIGNAL_UNWATCH(text_edit, "caret_changed");
+ }
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] context menu") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ text_edit->get_viewport()->set_embedding_subwindows(true); // Bypass display server for drop handling.
+
+ text_edit->set_size(Size2(800, 200));
+ text_edit->set_line(0, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec varius mattis leo, sed porta ex lacinia bibendum. Nunc bibendum pellentesque.");
+ MessageQueue::get_singleton()->flush();
+
+ text_edit->set_context_menu_enabled(false);
+ CHECK_FALSE(text_edit->is_context_menu_enabled());
+
+ CHECK_FALSE(text_edit->is_menu_visible());
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(600, 10), MouseButton::RIGHT, MouseButton::MASK_RIGHT, Key::NONE);
+ CHECK_FALSE(text_edit->is_menu_visible());
+
+ text_edit->set_context_menu_enabled(true);
+ CHECK(text_edit->is_context_menu_enabled());
+
+ CHECK_FALSE(text_edit->is_menu_visible());
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(700, 10), MouseButton::RIGHT, MouseButton::MASK_RIGHT, Key::NONE);
+ CHECK(text_edit->is_menu_visible());
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] versioning") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ // Action undo / redo states are tested in the action test e.g selection_delete.
+ CHECK_FALSE(text_edit->has_undo());
+ CHECK_FALSE(text_edit->has_redo());
+ CHECK(text_edit->get_version() == 0);
+ CHECK(text_edit->get_saved_version() == 0);
+
+ text_edit->begin_complex_operation();
+ text_edit->begin_complex_operation();
+ text_edit->begin_complex_operation();
+
+ text_edit->insert_text_at_caret("test");
+ CHECK(text_edit->get_version() == 1);
+ CHECK(text_edit->get_saved_version() == 0);
+ CHECK(text_edit->has_undo());
+ CHECK_FALSE(text_edit->has_redo());
+
+ text_edit->end_complex_operation();
+
+ // Can undo and redo mid op.
+ text_edit->insert_text_at_caret(" nested");
+ CHECK(text_edit->get_version() == 2);
+ CHECK(text_edit->get_saved_version() == 0);
+ CHECK(text_edit->has_undo());
+ CHECK_FALSE(text_edit->has_redo());
+ text_edit->undo();
+
+ CHECK(text_edit->has_redo());
+ text_edit->redo();
+
+ text_edit->end_complex_operation();
+
+ text_edit->insert_text_at_caret(" ops");
+ CHECK(text_edit->get_version() == 3);
+ CHECK(text_edit->get_saved_version() == 0);
+ CHECK(text_edit->has_undo());
+ CHECK_FALSE(text_edit->has_redo());
+
+ text_edit->end_complex_operation();
+
+ text_edit->tag_saved_version();
+ CHECK(text_edit->get_saved_version() == 3);
+
+ text_edit->undo();
+ CHECK(text_edit->get_line(0) == "");
+ CHECK(text_edit->get_version() == 0);
+ CHECK(text_edit->get_saved_version() == 3);
+ CHECK_FALSE(text_edit->has_undo());
+ CHECK(text_edit->has_redo());
+
+ text_edit->redo();
+ CHECK(text_edit->get_line(0) == "test nested ops");
+ CHECK(text_edit->get_version() == 3);
+ CHECK(text_edit->get_saved_version() == 3);
+ CHECK(text_edit->has_undo());
+ CHECK_FALSE(text_edit->has_redo());
+
+ text_edit->clear_undo_history();
+ CHECK_FALSE(text_edit->has_undo());
+ CHECK_FALSE(text_edit->has_redo());
+ CHECK(text_edit->get_version() == 3); // Should this be cleared?
+ CHECK(text_edit->get_saved_version() == 0);
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] search") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ text_edit->set_text("hay needle, hay\nHAY NEEDLE, HAY");
+ int length = text_edit->get_line(1).length();
+
+ CHECK(text_edit->search("test", 0, 0, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("test", TextEdit::SEARCH_MATCH_CASE, 0, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("test", TextEdit::SEARCH_WHOLE_WORDS, 0, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("test", TextEdit::SEARCH_BACKWARDS, 0, 0) == Point2i(-1, -1));
+
+ CHECK(text_edit->search("test", 0, 1, length) == Point2i(-1, -1));
+ CHECK(text_edit->search("test", TextEdit::SEARCH_MATCH_CASE, 1, length) == Point2i(-1, -1));
+ CHECK(text_edit->search("test", TextEdit::SEARCH_WHOLE_WORDS, 1, length) == Point2i(-1, -1));
+ CHECK(text_edit->search("test", TextEdit::SEARCH_BACKWARDS, 1, length) == Point2i(-1, -1));
+
+ CHECK(text_edit->search("needle", 0, 0, 0) == Point2i(4, 0));
+ CHECK(text_edit->search("needle", 0, 1, length) == Point2i(4, 0));
+ CHECK(text_edit->search("needle", 0, 0, 5) == Point2i(4, 1));
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_BACKWARDS, 0, 0) == Point2i(4, 1));
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_BACKWARDS, 1, 5) == Point2i(4, 1));
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_BACKWARDS, 1, 3) == Point2i(4, 0));
+
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_MATCH_CASE, 0, 0) == Point2i(4, 0));
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_MATCH_CASE | TextEdit::SEARCH_BACKWARDS, 0, 0) == Point2i(4, 0));
+
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_WHOLE_WORDS | TextEdit::SEARCH_MATCH_CASE, 0, 0) == Point2i(4, 0));
+ CHECK(text_edit->search("needle", TextEdit::SEARCH_WHOLE_WORDS | TextEdit::SEARCH_MATCH_CASE | TextEdit::SEARCH_BACKWARDS, 0, 0) == Point2i(4, 0));
+
+ CHECK(text_edit->search("need", TextEdit::SEARCH_MATCH_CASE, 0, 0) == Point2i(4, 0));
+ CHECK(text_edit->search("need", TextEdit::SEARCH_MATCH_CASE | TextEdit::SEARCH_BACKWARDS, 0, 0) == Point2i(4, 0));
+
+ CHECK(text_edit->search("need", TextEdit::SEARCH_WHOLE_WORDS | TextEdit::SEARCH_MATCH_CASE, 0, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("need", TextEdit::SEARCH_WHOLE_WORDS | TextEdit::SEARCH_MATCH_CASE | TextEdit::SEARCH_BACKWARDS, 0, 0) == Point2i(-1, -1));
+
+ ERR_PRINT_OFF;
+ CHECK(text_edit->search("", 0, 0, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("needle", 0, -1, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("needle", 0, 0, -1) == Point2i(-1, -1));
+ CHECK(text_edit->search("needle", 0, 100, 0) == Point2i(-1, -1));
+ CHECK(text_edit->search("needle", 0, 0, 100) == Point2i(-1, -1));
+ ERR_PRINT_ON;
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] mouse") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ text_edit->set_size(Size2(800, 200));
+ text_edit->set_line(0, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec varius mattis leo, sed porta ex lacinia bibendum. Nunc bibendum pellentesque.");
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_word_at_pos(text_edit->get_pos_at_line_column(0, 1)) == "Lorem");
+ CHECK(text_edit->get_word_at_pos(text_edit->get_pos_at_line_column(0, 9)) == "ipsum");
+
+ ERR_PRINT_OFF;
+ CHECK(text_edit->get_pos_at_line_column(0, -1) == Point2i(-1, -1));
+ CHECK(text_edit->get_pos_at_line_column(-1, 0) == Point2i(-1, -1));
+ CHECK(text_edit->get_pos_at_line_column(-1, -1) == Point2i(-1, -1));
+
+ CHECK(text_edit->get_pos_at_line_column(0, 500) == Point2i(-1, -1));
+ CHECK(text_edit->get_pos_at_line_column(2, 0) == Point2i(-1, -1));
+ CHECK(text_edit->get_pos_at_line_column(2, 500) == Point2i(-1, -1));
+
+ // Out of view.
+ CHECK(text_edit->get_pos_at_line_column(0, text_edit->get_line(0).length() - 1) == Point2i(-1, -1));
+ ERR_PRINT_ON;
+
+ // Add method to get drawn column count?
+ Point2i start_pos = text_edit->get_pos_at_line_column(0, 0);
+ Point2i end_pos = text_edit->get_pos_at_line_column(0, 105);
+
+ CHECK(text_edit->get_line_column_at_pos(Point2i(start_pos.x, start_pos.y)) == Point2i(0, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x, end_pos.y)) == Point2i(104, 0));
+
+ // Should this return Point2i(-1, -1) if its also < 0 not just > vis_lines.
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x - 100, end_pos.y), false) == Point2i(90, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x, end_pos.y + 100), false) == Point2i(-1, -1));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x - 100, end_pos.y + 100), false) == Point2i(-1, -1));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x, end_pos.y - 100), false) == Point2i(104, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x - 100, end_pos.y - 100), false) == Point2i(90, 0));
+
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x - 100, end_pos.y)) == Point2i(90, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x, end_pos.y + 100)) == Point2i(141, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x - 100, end_pos.y + 100)) == Point2i(141, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x, end_pos.y - 100)) == Point2i(104, 0));
+ CHECK(text_edit->get_line_column_at_pos(Point2i(end_pos.x - 100, end_pos.y - 100)) == Point2i(90, 0));
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] caret") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ text_edit->set_size(Size2(800, 200));
+ text_edit->grab_focus();
+ text_edit->set_line(0, "ffi");
+
+ text_edit->set_caret_mid_grapheme_enabled(true);
+ CHECK(text_edit->is_caret_mid_grapheme_enabled());
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_caret_column() == 1);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_caret_column() == 2);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_caret_column() == 3);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_caret_column() == 2);
+
+ text_edit->set_caret_mid_grapheme_enabled(false);
+ CHECK_FALSE(text_edit->is_caret_mid_grapheme_enabled());
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_caret_column() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_right");
+ CHECK(text_edit->get_caret_column() == 3);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_left");
+ CHECK(text_edit->get_caret_column() == 0);
+
+ text_edit->set_line(0, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec varius mattis leo, sed porta ex lacinia bibendum. Nunc bibendum pellentesque.");
+ for (int i = 0; i < 3; i++) {
+ text_edit->insert_line_at(0, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec varius mattis leo, sed porta ex lacinia bibendum. Nunc bibendum pellentesque.");
+ }
+ MessageQueue::get_singleton()->flush();
+
+ text_edit->set_caret_blink_enabled(false);
+ CHECK_FALSE(text_edit->is_caret_blink_enabled());
+
+ text_edit->set_caret_blink_enabled(true);
+ CHECK(text_edit->is_caret_blink_enabled());
+
+ text_edit->set_caret_blink_speed(10);
+ CHECK(text_edit->get_caret_blink_speed() == 10);
+
+ ERR_PRINT_OFF;
+ text_edit->set_caret_blink_speed(-1);
+ CHECK(text_edit->get_caret_blink_speed() == 10);
+
+ text_edit->set_caret_blink_speed(0);
+ CHECK(text_edit->get_caret_blink_speed() == 10);
+ ERR_PRINT_ON;
+
+ text_edit->set_caret_type(TextEdit::CaretType::CARET_TYPE_LINE);
+ CHECK(text_edit->get_caret_type() == TextEdit::CaretType::CARET_TYPE_LINE);
+
+ text_edit->set_caret_type(TextEdit::CaretType::CARET_TYPE_BLOCK);
+ CHECK(text_edit->get_caret_type() == TextEdit::CaretType::CARET_TYPE_BLOCK);
+
+ text_edit->set_caret_type(TextEdit::CaretType::CARET_TYPE_LINE);
+ CHECK(text_edit->get_caret_type() == TextEdit::CaretType::CARET_TYPE_LINE);
+
+ int caret_col = text_edit->get_caret_column();
+ text_edit->set_move_caret_on_right_click_enabled(false);
+ CHECK_FALSE(text_edit->is_move_caret_on_right_click_enabled());
+
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(100, 1), MouseButton::RIGHT, MouseButton::MASK_RIGHT, Key::NONE);
+ CHECK(text_edit->get_caret_column() == caret_col);
+
+ text_edit->set_move_caret_on_right_click_enabled(true);
+ CHECK(text_edit->is_move_caret_on_right_click_enabled());
+
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(100, 1), MouseButton::RIGHT, MouseButton::MASK_RIGHT, Key::NONE);
+ CHECK(text_edit->get_caret_column() != caret_col);
+
+ text_edit->set_move_caret_on_right_click_enabled(false);
+ CHECK_FALSE(text_edit->is_move_caret_on_right_click_enabled());
+
+ text_edit->set_caret_column(0);
+ CHECK(text_edit->get_word_under_caret() == "Lorem");
+
+ text_edit->set_caret_column(4);
+ CHECK(text_edit->get_word_under_caret() == "Lorem");
+
+ // Should this work?
+ text_edit->set_caret_column(5);
+ CHECK(text_edit->get_word_under_caret() == "");
+
+ text_edit->set_caret_column(6);
+ CHECK(text_edit->get_word_under_caret() == "");
+
+ text_edit->set_caret_line(1);
+ CHECK(text_edit->get_caret_line() == 1);
+
+ text_edit->set_caret_line(-1);
+ CHECK(text_edit->get_caret_line() == 0);
+ text_edit->set_caret_line(100);
+ CHECK(text_edit->get_caret_line() == 3);
+
+ text_edit->set_caret_column(-1);
+ CHECK(text_edit->get_caret_column() == 0);
+ text_edit->set_caret_column(10000000);
+ CHECK(text_edit->get_caret_column() == 141);
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] line wrapping") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+ text_edit->grab_focus();
+
+ // Set size for boundry.
+ text_edit->set_size(Size2(800, 200));
+ text_edit->set_line(0, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec varius mattis leo, sed porta ex lacinia bibendum. Nunc bibendum pellentesque.");
+ CHECK_FALSE(text_edit->is_line_wrapped(0));
+ CHECK(text_edit->get_line_wrap_count(0) == 0);
+ CHECK(text_edit->get_line_wrap_index_at_column(0, 130) == 0);
+ CHECK(text_edit->get_line_wrapped_text(0).size() == 1);
+
+ SIGNAL_WATCH(text_edit, "text_set");
+ SIGNAL_WATCH(text_edit, "text_changed");
+ SIGNAL_WATCH(text_edit, "lines_edited_from");
+ SIGNAL_WATCH(text_edit, "caret_changed");
+
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+ SIGNAL_CHECK_FALSE("text_set");
+ SIGNAL_CHECK_FALSE("text_changed");
+ SIGNAL_CHECK_FALSE("lines_edited_from");
+ SIGNAL_CHECK_FALSE("caret_changed");
+
+ CHECK(text_edit->is_line_wrapped(0));
+ CHECK(text_edit->get_line_wrap_count(0) == 1);
+ CHECK(text_edit->get_line_wrap_index_at_column(0, 130) == 1);
+ CHECK(text_edit->get_line_wrapped_text(0).size() == 2);
+
+ SIGNAL_UNWATCH(text_edit, "text_set");
+ SIGNAL_UNWATCH(text_edit, "text_changed");
+ SIGNAL_UNWATCH(text_edit, "lines_edited_from");
+ SIGNAL_UNWATCH(text_edit, "caret_changed");
+
+ ERR_PRINT_OFF;
+ CHECK_FALSE(text_edit->is_line_wrapped(-1));
+ CHECK_FALSE(text_edit->is_line_wrapped(1));
+ CHECK(text_edit->get_line_wrap_count(-1) == 0);
+ CHECK(text_edit->get_line_wrap_count(1) == 0);
+ CHECK(text_edit->get_line_wrap_index_at_column(-1, 0) == 0);
+ CHECK(text_edit->get_line_wrap_index_at_column(0, -1) == 0);
+ CHECK(text_edit->get_line_wrap_index_at_column(1, 0) == 0);
+ CHECK(text_edit->get_line_wrap_index_at_column(0, 10000) == 0);
+ CHECK(text_edit->get_line_wrapped_text(-1).size() == 0);
+ CHECK(text_edit->get_line_wrapped_text(1).size() == 0);
+ ERR_PRINT_ON;
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] viewport") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ // No subcases here for performance.
+ text_edit->set_size(Size2(800, 600));
+ for (int i = 0; i < 50; i++) {
+ text_edit->insert_line_at(0, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Donec varius mattis leo, sed porta ex lacinia bibendum. Nunc bibendum pellentesque.");
+ }
+ MessageQueue::get_singleton()->flush();
+
+ const int visible_lines = text_edit->get_visible_line_count();
+ const int total_visible_lines = text_edit->get_total_visible_line_count();
+ CHECK(total_visible_lines == 51);
+
+ // First visible line.
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ text_edit->set_line_as_first_visible(visible_lines);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_v_scroll() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ ERR_PRINT_OFF;
+ text_edit->set_line_as_first_visible(-1);
+ text_edit->set_line_as_first_visible(500);
+ text_edit->set_line_as_first_visible(0, -1);
+ text_edit->set_line_as_first_visible(0, 500);
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ ERR_PRINT_ON;
+
+ // Wrap.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() > total_visible_lines);
+
+ text_edit->set_line_as_first_visible(5, 1);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 5);
+ CHECK(text_edit->get_v_scroll() == 11);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 6);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 1);
+
+ // Reset.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_NONE);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() == total_visible_lines);
+ text_edit->set_line_as_first_visible(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Last visible line.
+ text_edit->set_line_as_last_visible(visible_lines * 2);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_v_scroll() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ ERR_PRINT_OFF;
+ text_edit->set_line_as_last_visible(-1);
+ text_edit->set_line_as_last_visible(500);
+ text_edit->set_line_as_last_visible(0, -1);
+ text_edit->set_line_as_last_visible(0, 500);
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ ERR_PRINT_ON;
+
+ // Wrap.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() > total_visible_lines);
+
+ text_edit->set_line_as_last_visible(visible_lines + 5, 1);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 16);
+ CHECK(text_edit->get_v_scroll() == 32.0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines + 5);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Reset.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_NONE);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() == total_visible_lines);
+ text_edit->set_line_as_first_visible(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Center.
+ text_edit->set_line_as_center_visible(visible_lines + (visible_lines / 2));
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_v_scroll() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ ERR_PRINT_OFF;
+ text_edit->set_line_as_last_visible(-1);
+ text_edit->set_line_as_last_visible(500);
+ text_edit->set_line_as_last_visible(0, -1);
+ text_edit->set_line_as_last_visible(0, 500);
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ ERR_PRINT_ON;
+
+ // Wrap.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() > total_visible_lines);
+
+ text_edit->set_line_as_center_visible(visible_lines + (visible_lines / 2) + 5, 1);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines + (visible_lines / 2));
+ CHECK(text_edit->get_v_scroll() == (visible_lines * 3));
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 1);
+
+ // Scroll past eof.
+ int line_count = text_edit->get_line_count();
+ text_edit->set_scroll_past_end_of_file_enabled(true);
+ MessageQueue::get_singleton()->flush();
+ text_edit->set_line_as_center_visible(line_count - 1);
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_first_visible_line() == (visible_lines * 2) + 3);
+ CHECK(text_edit->get_v_scroll() == (visible_lines * 4) + 6);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) + 8);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ text_edit->set_scroll_past_end_of_file_enabled(false);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == (visible_lines * 2) + 3);
+ CHECK(text_edit->get_v_scroll() == (visible_lines * 4) - 4);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) + 8);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Reset.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_NONE);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() == total_visible_lines);
+ text_edit->set_line_as_first_visible(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Auto adjust - todo: horizontal scroll.
+ // Below.
+ MessageQueue::get_singleton()->flush();
+ CHECK_FALSE(text_edit->is_caret_visible());
+ text_edit->set_caret_line(visible_lines + 5, false);
+ CHECK_FALSE(text_edit->is_caret_visible());
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->is_caret_visible());
+ CHECK(text_edit->get_first_visible_line() == 5);
+ CHECK(text_edit->get_v_scroll() == 5);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines - 1) + 5);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ text_edit->center_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines - 5);
+ CHECK(text_edit->get_v_scroll() == visible_lines - 5);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 6);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Caret visible, do nothing.
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines - 5);
+ CHECK(text_edit->get_v_scroll() == visible_lines - 5);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 6);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Above.
+ text_edit->set_caret_line(1, false);
+ MessageQueue::get_singleton()->flush();
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->is_caret_visible());
+ CHECK(text_edit->get_first_visible_line() == 1);
+ CHECK(text_edit->get_v_scroll() == 1);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ text_edit->set_line_as_first_visible(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Wrap
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_BOUNDARY);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() > total_visible_lines);
+
+ text_edit->set_caret_line(visible_lines + 5, false, true, 1);
+ MessageQueue::get_singleton()->flush();
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+
+ CHECK(text_edit->get_first_visible_line() == (visible_lines / 2) + 4);
+ CHECK(text_edit->get_v_scroll() == (visible_lines + (visible_lines / 2)) - 1);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines) + 3);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 1);
+ CHECK(text_edit->get_caret_wrap_index() == 1);
+
+ text_edit->center_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_v_scroll() == (visible_lines * 2) + 1);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 11);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 1);
+
+ // Caret visible, do nothing.
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == visible_lines);
+ CHECK(text_edit->get_v_scroll() == (visible_lines * 2) + 1);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 11);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 1);
+
+ // Above.
+ text_edit->set_caret_line(1, false, true, 1);
+ MessageQueue::get_singleton()->flush();
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->is_caret_visible());
+ CHECK(text_edit->get_first_visible_line() == 1);
+ CHECK(text_edit->get_v_scroll() == 3);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines / 2) + 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 1);
+ CHECK(text_edit->get_caret_wrap_index() == 1);
+
+ text_edit->set_line_as_first_visible(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->is_caret_visible());
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 11);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ text_edit->adjust_viewport_to_caret();
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 11);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ // Reset.
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_NONE);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_total_visible_line_count() == total_visible_lines);
+ text_edit->set_line_as_first_visible(0);
+ MessageQueue::get_singleton()->flush();
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ // Smooth scroll.
+ text_edit->set_v_scroll_speed(10);
+ CHECK(text_edit->get_v_scroll_speed() == 10);
+ ERR_PRINT_OFF;
+ text_edit->set_v_scroll_speed(-1);
+ CHECK(text_edit->get_v_scroll_speed() == 10);
+
+ text_edit->set_v_scroll_speed(0);
+ CHECK(text_edit->get_v_scroll_speed() == 10);
+
+ text_edit->set_v_scroll_speed(1);
+ CHECK(text_edit->get_v_scroll_speed() == 1);
+ ERR_PRINT_ON;
+
+ // Scroll.
+ int v_scroll = text_edit->get_v_scroll();
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(10, 10), MouseButton::WHEEL_DOWN, MouseButton::WHEEL_DOWN, Key::NONE);
+ CHECK(text_edit->get_v_scroll() > v_scroll);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(10, 10), MouseButton::WHEEL_UP, MouseButton::WHEEL_UP, Key::NONE);
+ CHECK(text_edit->get_v_scroll() == v_scroll);
+
+ // smooth scroll speed.
+ text_edit->set_smooth_scroll_enabled(true);
+
+ v_scroll = text_edit->get_v_scroll();
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(10, 10), MouseButton::WHEEL_DOWN, MouseButton::WHEEL_DOWN, Key::NONE);
+ text_edit->notification(TextEdit::NOTIFICATION_INTERNAL_PHYSICS_PROCESS);
+ CHECK(text_edit->get_v_scroll() > v_scroll);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(10, 10), MouseButton::WHEEL_UP, MouseButton::WHEEL_UP, Key::NONE);
+ text_edit->notification(TextEdit::NOTIFICATION_INTERNAL_PHYSICS_PROCESS);
+ CHECK(text_edit->get_v_scroll() == v_scroll);
+
+ v_scroll = text_edit->get_v_scroll();
+ text_edit->set_v_scroll_speed(10000);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(10, 10), MouseButton::WHEEL_DOWN, MouseButton::WHEEL_DOWN, Key::NONE);
+ text_edit->notification(TextEdit::NOTIFICATION_INTERNAL_PHYSICS_PROCESS);
+ CHECK(text_edit->get_v_scroll() > v_scroll);
+ SEND_GUI_MOUSE_BUTTON_EVENT(text_edit, Point2i(10, 10), MouseButton::WHEEL_UP, MouseButton::WHEEL_UP, Key::NONE);
+ text_edit->notification(TextEdit::NOTIFICATION_INTERNAL_PHYSICS_PROCESS);
+ CHECK(text_edit->get_v_scroll() == v_scroll);
+
+ ERR_PRINT_OFF;
+ CHECK(text_edit->get_scroll_pos_for_line(-1) == 0);
+ CHECK(text_edit->get_scroll_pos_for_line(1000) == 0);
+ CHECK(text_edit->get_scroll_pos_for_line(1, -1) == 0);
+ CHECK(text_edit->get_scroll_pos_for_line(1, 100) == 0);
+ ERR_PRINT_ON;
+
+ text_edit->set_h_scroll(-100);
+ CHECK(text_edit->get_h_scroll() == 0);
+
+ text_edit->set_h_scroll(10000000);
+ CHECK(text_edit->get_h_scroll() == 313);
+
+ text_edit->set_h_scroll(-100);
+ CHECK(text_edit->get_h_scroll() == 0);
+
+ text_edit->set_smooth_scroll_enabled(false);
+
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+
+ text_edit->grab_focus();
+ SEND_GUI_ACTION(text_edit, "ui_text_scroll_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_first_visible_line() == 1);
+ CHECK(text_edit->get_v_scroll() == 1);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_scroll_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ // Page down, similar to VSCode, to end of page then scroll.
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 21);
+ CHECK(text_edit->get_first_visible_line() == 0);
+ CHECK(text_edit->get_v_scroll() == 0);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 41);
+ CHECK(text_edit->get_first_visible_line() == 20);
+ CHECK(text_edit->get_v_scroll() == 20);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines - 1) * 2);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 21);
+ CHECK(text_edit->get_first_visible_line() == 20);
+ CHECK(text_edit->get_v_scroll() == 20);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines - 1) * 2);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 1);
+ CHECK(text_edit->get_first_visible_line() == 1);
+ CHECK(text_edit->get_v_scroll() == 1);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ text_edit->set_line_wrapping_mode(TextEdit::LineWrappingMode::LINE_WRAPPING_NONE);
+ MessageQueue::get_singleton()->flush();
+
+ text_edit->grab_focus();
+ SEND_GUI_ACTION(text_edit, "ui_text_scroll_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 2);
+ CHECK(text_edit->get_first_visible_line() == 2);
+ CHECK(text_edit->get_v_scroll() == 2);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines + 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_scroll_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 2);
+ CHECK(text_edit->get_first_visible_line() == 1);
+ CHECK(text_edit->get_v_scroll() == 1);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ // Page down, similar to VSCode, to end of page then scroll.
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 22);
+ CHECK(text_edit->get_first_visible_line() == 1);
+ CHECK(text_edit->get_v_scroll() == 1);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_down");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 42);
+ CHECK(text_edit->get_first_visible_line() == 21);
+ CHECK(text_edit->get_v_scroll() == 21);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 22);
+ CHECK(text_edit->get_first_visible_line() == 21);
+ CHECK(text_edit->get_v_scroll() == 21);
+ CHECK(text_edit->get_last_full_visible_line() == (visible_lines * 2) - 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ SEND_GUI_ACTION(text_edit, "ui_text_caret_page_up");
+ CHECK(text_edit->get_viewport()->is_input_handled());
+ CHECK(text_edit->get_caret_line() == 2);
+ CHECK(text_edit->get_first_visible_line() == 2);
+ CHECK(text_edit->get_v_scroll() == 2);
+ CHECK(text_edit->get_last_full_visible_line() == visible_lines + 1);
+ CHECK(text_edit->get_last_full_visible_line_wrap_index() == 0);
+ CHECK(text_edit->get_caret_wrap_index() == 0);
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] setter getters") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ SUBCASE("[TextEdit] set and get placeholder") {
+ text_edit->set_placeholder("test\nplaceholder");
+ CHECK(text_edit->get_placeholder() == "test\nplaceholder");
+
+ CHECK(text_edit->get_text() == "");
+ CHECK(text_edit->get_line_count() == 1);
+ CHECK(text_edit->get_last_full_visible_line() == 0);
+ }
+
+ SUBCASE("[TextEdit] highlight current line") {
+ text_edit->set_highlight_current_line(true);
+ CHECK(text_edit->is_highlight_current_line_enabled());
+ text_edit->set_highlight_current_line(false);
+ CHECK_FALSE(text_edit->is_highlight_current_line_enabled());
+ }
+
+ SUBCASE("[TextEdit] highlight all occurrences") {
+ text_edit->set_highlight_all_occurrences(true);
+ CHECK(text_edit->is_highlight_all_occurrences_enabled());
+ text_edit->set_highlight_all_occurrences(false);
+ CHECK_FALSE(text_edit->is_highlight_all_occurrences_enabled());
+ }
+
+ SUBCASE("[TextEdit] draw control chars") {
+ text_edit->set_draw_control_chars(true);
+ CHECK(text_edit->get_draw_control_chars());
+ text_edit->set_draw_control_chars(false);
+ CHECK_FALSE(text_edit->get_draw_control_chars());
+ }
+
+ SUBCASE("[TextEdit] draw tabs") {
+ text_edit->set_draw_tabs(true);
+ CHECK(text_edit->is_drawing_tabs());
+ text_edit->set_draw_tabs(false);
+ CHECK_FALSE(text_edit->is_drawing_tabs());
+ }
+
+ SUBCASE("[TextEdit] draw spaces") {
+ text_edit->set_draw_spaces(true);
+ CHECK(text_edit->is_drawing_spaces());
+ text_edit->set_draw_spaces(false);
+ CHECK_FALSE(text_edit->is_drawing_spaces());
+ }
+
+ SUBCASE("[TextEdit] draw minimao") {
+ text_edit->set_draw_minimap(true);
+ CHECK(text_edit->is_drawing_minimap());
+ text_edit->set_draw_minimap(false);
+ CHECK_FALSE(text_edit->is_drawing_minimap());
+ }
+
+ SUBCASE("[TextEdit] minimap width") {
+ text_edit->set_minimap_width(-1);
+ CHECK(text_edit->get_minimap_width() == -1);
+ text_edit->set_minimap_width(1000);
+ CHECK(text_edit->get_minimap_width() == 1000);
+ }
+
+ SUBCASE("[TextEdit] line color background") {
+ ERR_PRINT_OFF;
+ text_edit->set_line_background_color(-1, Color("#ff0000"));
+ text_edit->set_line_background_color(0, Color("#00ff00"));
+ text_edit->set_line_background_color(1, Color("#0000ff"));
+
+ CHECK(text_edit->get_line_background_color(-1) == Color());
+ CHECK(text_edit->get_line_background_color(0) == Color("#00ff00"));
+ CHECK(text_edit->get_line_background_color(1) == Color());
+ ERR_PRINT_ON;
+
+ text_edit->set_line_background_color(0, Color("#ffff00"));
+ CHECK(text_edit->get_line_background_color(0) == Color("#ffff00"));
+ }
+
+ memdelete(text_edit);
+}
+
+TEST_CASE("[SceneTree][TextEdit] gutters") {
+ TextEdit *text_edit = memnew(TextEdit);
+ SceneTree::get_singleton()->get_root()->add_child(text_edit);
+
+ Array empty_singal_args;
+ empty_singal_args.push_back(Array());
+
+ SIGNAL_WATCH(text_edit, "gutter_clicked");
+ SIGNAL_WATCH(text_edit, "gutter_added");
+ SIGNAL_WATCH(text_edit, "gutter_removed");
+
+ SUBCASE("[TextEdit] gutter add and remove") {
+ text_edit->add_gutter();
+ CHECK(text_edit->get_gutter_count() == 1);
+ SIGNAL_CHECK("gutter_added", empty_singal_args);
+
+ text_edit->set_gutter_name(0, "test_gutter");
+ CHECK(text_edit->get_gutter_name(0) == "test_gutter");
+
+ text_edit->set_gutter_width(0, 10);
+ CHECK(text_edit->get_gutter_width(0) == 10);
+ CHECK(text_edit->get_total_gutter_width() > 10);
+ CHECK(text_edit->get_total_gutter_width() < 20);
+
+ text_edit->add_gutter(-100);
+ text_edit->set_gutter_width(1, 10);
+ CHECK(text_edit->get_total_gutter_width() > 20);
+ CHECK(text_edit->get_total_gutter_width() < 30);
+ CHECK(text_edit->get_gutter_count() == 2);
+ CHECK(text_edit->get_gutter_name(0) == "test_gutter");
+ SIGNAL_CHECK("gutter_added", empty_singal_args);
+
+ text_edit->set_gutter_draw(1, false);
+ CHECK(text_edit->get_total_gutter_width() > 10);
+ CHECK(text_edit->get_total_gutter_width() < 20);
+
+ text_edit->add_gutter(100);
+ CHECK(text_edit->get_gutter_count() == 3);
+ CHECK(text_edit->get_gutter_name(0) == "test_gutter");
+ SIGNAL_CHECK("gutter_added", empty_singal_args);
+
+ text_edit->add_gutter(0);
+ CHECK(text_edit->get_gutter_count() == 4);
+ CHECK(text_edit->get_gutter_name(1) == "test_gutter");
+ SIGNAL_CHECK("gutter_added", empty_singal_args);
+
+ text_edit->remove_gutter(2);
+ CHECK(text_edit->get_gutter_name(1) == "test_gutter");
+ CHECK(text_edit->get_gutter_count() == 3);
+ SIGNAL_CHECK("gutter_removed", empty_singal_args);
+
+ text_edit->remove_gutter(0);
+ CHECK(text_edit->get_gutter_name(0) == "test_gutter");
+ CHECK(text_edit->get_gutter_count() == 2);
+ SIGNAL_CHECK("gutter_removed", empty_singal_args);
+
+ ERR_PRINT_OFF;
+ text_edit->remove_gutter(-1);
+ SIGNAL_CHECK_FALSE("gutter_removed");
+
+ text_edit->remove_gutter(100);
+ SIGNAL_CHECK_FALSE("gutter_removed");
+
+ CHECK(text_edit->get_gutter_name(-1) == "");
+ CHECK(text_edit->get_gutter_name(100) == "");
+ ERR_PRINT_ON;
+ }
+
+ SUBCASE("[TextEdit] gutter data") {
+ text_edit->add_gutter();
+ CHECK(text_edit->get_gutter_count() == 1);
+ SIGNAL_CHECK("gutter_added", empty_singal_args);
+
+ text_edit->set_gutter_name(0, "test_gutter");
+ CHECK(text_edit->get_gutter_name(0) == "test_gutter");
+
+ text_edit->set_gutter_width(0, 10);
+ CHECK(text_edit->get_gutter_width(0) == 10);
+
+ text_edit->set_gutter_clickable(0, true);
+ CHECK(text_edit->is_gutter_clickable(0));
+
+ text_edit->set_gutter_overwritable(0, true);
+ CHECK(text_edit->is_gutter_overwritable(0));
+
+ text_edit->set_gutter_type(0, TextEdit::GutterType::GUTTER_TYPE_CUSTOM);
+ CHECK(text_edit->get_gutter_type(0) == TextEdit::GutterType::GUTTER_TYPE_CUSTOM);
+
+ text_edit->set_text("test\ntext");
+
+ ERR_PRINT_OFF;
+ text_edit->set_line_gutter_metadata(1, 0, "test");
+ text_edit->set_line_gutter_metadata(0, -1, "test");
+ text_edit->set_line_gutter_metadata(0, 2, "test");
+ text_edit->set_line_gutter_metadata(2, 0, "test");
+ text_edit->set_line_gutter_metadata(-1, 0, "test");
+
+ CHECK(text_edit->get_line_gutter_metadata(1, 0) == "test");
+ CHECK(text_edit->get_line_gutter_metadata(0, -1) == "");
+ CHECK(text_edit->get_line_gutter_metadata(0, 2) == "");
+ CHECK(text_edit->get_line_gutter_metadata(2, 0) == "");
+ CHECK(text_edit->get_line_gutter_metadata(-1, 0) == "");
+
+ text_edit->set_line_gutter_text(1, 0, "test");
+ text_edit->set_line_gutter_text(0, -1, "test");
+ text_edit->set_line_gutter_text(0, 2, "test");
+ text_edit->set_line_gutter_text(2, 0, "test");
+ text_edit->set_line_gutter_text(-1, 0, "test");
+
+ CHECK(text_edit->get_line_gutter_text(1, 0) == "test");
+ CHECK(text_edit->get_line_gutter_text(0, -1) == "");
+ CHECK(text_edit->get_line_gutter_text(0, 2) == "");
+ CHECK(text_edit->get_line_gutter_text(2, 0) == "");
+ CHECK(text_edit->get_line_gutter_text(-1, 0) == "");
+
+ text_edit->set_line_gutter_item_color(1, 0, Color(1, 0, 0));
+ text_edit->set_line_gutter_item_color(0, -1, Color(1, 0, 0));
+ text_edit->set_line_gutter_item_color(0, 2, Color(1, 0, 0));
+ text_edit->set_line_gutter_item_color(2, 0, Color(1, 0, 0));
+ text_edit->set_line_gutter_item_color(-1, 0, Color(1, 0, 0));
+
+ CHECK(text_edit->get_line_gutter_item_color(1, 0) == Color(1, 0, 0));
+ CHECK(text_edit->get_line_gutter_item_color(0, -1) == Color());
+ CHECK(text_edit->get_line_gutter_item_color(0, 2) == Color());
+ CHECK(text_edit->get_line_gutter_item_color(2, 0) == Color());
+ CHECK(text_edit->get_line_gutter_item_color(-1, 0) == Color());
+
+ text_edit->set_line_gutter_clickable(1, 0, true);
+ text_edit->set_line_gutter_clickable(0, -1, true);
+ text_edit->set_line_gutter_clickable(0, 2, true);
+ text_edit->set_line_gutter_clickable(2, 0, true);
+ text_edit->set_line_gutter_clickable(-1, 0, true);
+
+ CHECK(text_edit->is_line_gutter_clickable(1, 0) == true);
+ CHECK(text_edit->is_line_gutter_clickable(0, -1) == false);
+ CHECK(text_edit->is_line_gutter_clickable(0, 2) == false);
+ CHECK(text_edit->is_line_gutter_clickable(2, 0) == false);
+ CHECK(text_edit->is_line_gutter_clickable(-1, 0) == false);
+ ERR_PRINT_ON;
+
+ // Merging tested via CodeEdit gutters.
+ }
+
+ SIGNAL_UNWATCH(text_edit, "gutter_clicked");
+ SIGNAL_UNWATCH(text_edit, "gutter_added");
+ SIGNAL_UNWATCH(text_edit, "gutter_removed");
+ memdelete(text_edit);
+}
+
+} // namespace TestTextEdit
+
+#endif // TEST_TEXT_EDIT_H
diff --git a/tests/servers/test_physics_2d.cpp b/tests/servers/test_physics_2d.cpp
deleted file mode 100644
index 138412ec09..0000000000
--- a/tests/servers/test_physics_2d.cpp
+++ /dev/null
@@ -1,374 +0,0 @@
-/*************************************************************************/
-/* test_physics_2d.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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. */
-/*************************************************************************/
-
-#include "test_physics_2d.h"
-
-#include "core/os/main_loop.h"
-#include "servers/physics_server_2d.h"
-#include "servers/rendering_server.h"
-
-static const unsigned char convex_png[] = {
- 0x89, 0x50, 0x4e, 0x47, 0xd, 0xa, 0x1a, 0xa, 0x0, 0x0, 0x0, 0xd, 0x49, 0x48, 0x44, 0x52, 0x0, 0x0, 0x0, 0x40, 0x0, 0x0, 0x0, 0x40, 0x8, 0x6, 0x0, 0x0, 0x0, 0xaa, 0x69, 0x71, 0xde, 0x0, 0x0, 0x0, 0x1, 0x73, 0x52, 0x47, 0x42, 0x0, 0xae, 0xce, 0x1c, 0xe9, 0x0, 0x0, 0x0, 0x6, 0x62, 0x4b, 0x47, 0x44, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xf9, 0x43, 0xbb, 0x7f, 0x0, 0x0, 0x0, 0x9, 0x70, 0x48, 0x59, 0x73, 0x0, 0x0, 0xb, 0x13, 0x0, 0x0, 0xb, 0x13, 0x1, 0x0, 0x9a, 0x9c, 0x18, 0x0, 0x0, 0x0, 0x7, 0x74, 0x49, 0x4d, 0x45, 0x7, 0xdb, 0x6, 0xa, 0x3, 0x13, 0x31, 0x66, 0xa7, 0xac, 0x79, 0x0, 0x0, 0x4, 0xef, 0x49, 0x44, 0x41, 0x54, 0x78, 0xda, 0xed, 0x9b, 0xdd, 0x4e, 0x2a, 0x57, 0x14, 0xc7, 0xf7, 0x1e, 0xc0, 0x19, 0x38, 0x32, 0x80, 0xa, 0x6a, 0xda, 0x18, 0xa3, 0xc6, 0x47, 0x50, 0x7b, 0xa1, 0xd9, 0x36, 0x27, 0x7e, 0x44, 0xed, 0x45, 0x4d, 0x93, 0x3e, 0x40, 0x1f, 0x64, 0x90, 0xf4, 0x1, 0xbc, 0xf0, 0xc2, 0x9c, 0x57, 0x30, 0x4d, 0xbc, 0xa8, 0x6d, 0xc, 0x69, 0x26, 0xb5, 0x68, 0x8b, 0x35, 0x7e, 0x20, 0xb4, 0xf5, 0x14, 0xbf, 0x51, 0x3c, 0x52, 0xe, 0xc, 0xe, 0xc8, 0xf0, 0xb1, 0x7a, 0x51, 0x3d, 0xb1, 0x9e, 0x19, 0x1c, 0x54, 0x70, 0x1c, 0xdc, 0x9, 0x17, 0x64, 0x8, 0xc9, 0xff, 0xb7, 0xd6, 0x7f, 0xcd, 0x3f, 0x2b, 0xd9, 0x8, 0xbd, 0x9c, 0xda, 0x3e, 0xf8, 0x31, 0xff, 0xc, 0x0, 0x8, 0x42, 0x88, 0x9c, 0x9f, 0x9f, 0xbf, 0xa, 0x87, 0xc3, 0xad, 0x7d, 0x7d, 0x7d, 0x7f, 0x23, 0x84, 0x78, 0x8c, 0x31, 0xaf, 0x55, 0x0, 0xc6, 0xc7, 0x14, 0x1e, 0x8f, 0xc7, 0xbf, 0x38, 0x3c, 0x3c, 0x6c, 0x9b, 0x9f, 0x9f, 0x6f, 0xb8, 0x82, 0x9b, 0xee, 0xe8, 0xe8, 0xf8, 0x12, 0x0, 0xbe, 0xd3, 0x2a, 0x8, 0xfc, 0x50, 0xd1, 0xf9, 0x7c, 0x9e, 0x8a, 0x46, 0xa3, 0x5f, 0x9d, 0x9e, 0x9e, 0x7e, 0xb2, 0xb0, 0xb0, 0x60, 0xe5, 0x79, 0x1e, 0xf1, 0xfc, 0x7f, 0x3a, 0x9, 0x21, 0x88, 0x10, 0x82, 0x26, 0x26, 0x26, 0xde, 0x77, 0x75, 0x75, 0x85, 0x59, 0x96, 0xfd, 0x5e, 0x6b, 0x20, 0xf0, 0x7d, 0x85, 0x4b, 0x92, 0xf4, 0xfa, 0xe0, 0xe0, 0xe0, 0xd3, 0xb9, 0xb9, 0xb9, 0x46, 0x49, 0x92, 0xea, 0x6f, 0xa, 0xbf, 0x7d, 0x8, 0x21, 0x68, 0x70, 0x70, 0xb0, 0x38, 0x39, 0x39, 0x79, 0xd6, 0xd9, 0xd9, 0xb9, 0xcf, 0x30, 0xcc, 0xa2, 0xd6, 0xad, 0x21, 0x2b, 0x1c, 0x0, 0x38, 0x41, 0x10, 0xfc, 0xdb, 0xdb, 0xdb, 0x27, 0x1e, 0x8f, 0x27, 0x4b, 0x8, 0x1, 0x84, 0x90, 0xea, 0xf, 0x21, 0x4, 0x3c, 0x1e, 0x4f, 0x76, 0x67, 0x67, 0x67, 0x3f, 0x9f, 0xcf, 0xff, 0x7c, 0x5, 0xf3, 0xd9, 0x0, 0xe0, 0x2, 0x81, 0xc0, 0xa9, 0xdb, 0xed, 0x2e, 0x94, 0x2b, 0x5c, 0xe, 0xc4, 0xca, 0xca, 0x8a, 0x18, 0x8d, 0x46, 0x3, 0x0, 0xc0, 0x69, 0x1e, 0x4, 0x0, 0x90, 0x48, 0x24, 0x12, 0xe4, 0x38, 0xee, 0x41, 0xc2, 0x6f, 0x43, 0xe0, 0x38, 0xe, 0xfc, 0x7e, 0xbf, 0x10, 0x8b, 0xc5, 0xd6, 0x35, 0xd, 0x22, 0x9b, 0xcd, 0x7a, 0x96, 0x97, 0x97, 0x33, 0xf, 0xad, 0x7c, 0x29, 0x10, 0x9b, 0x9b, 0x9b, 0xef, 0x2e, 0x2e, 0x2e, 0x7e, 0xd5, 0x1c, 0x8, 0x0, 0x20, 0xe1, 0x70, 0x38, 0xfc, 0x98, 0xd5, 0x57, 0x2, 0xe1, 0x76, 0xbb, 0xf3, 0xa1, 0x50, 0xe8, 0x38, 0x9b, 0xcd, 0xfe, 0xa2, 0x9, 0x8, 0x0, 0x40, 0x2e, 0x2f, 0x2f, 0x7d, 0x4b, 0x4b, 0x4b, 0xb9, 0x4a, 0x54, 0x5f, 0x9, 0xc4, 0xd2, 0xd2, 0x92, 0xb4, 0xb7, 0xb7, 0xf7, 0x36, 0x97, 0xcb, 0x4d, 0x3d, 0x29, 0x8, 0x0, 0xe0, 0x42, 0xa1, 0xd0, 0x71, 0xb5, 0xc4, 0xdf, 0xb6, 0xc5, 0x93, 0xe, 0x4a, 0x0, 0x20, 0xa9, 0x54, 0xea, 0x37, 0xb7, 0xdb, 0x5d, 0xa8, 0xa6, 0x78, 0x39, 0x10, 0x6b, 0x6b, 0x6b, 0xf1, 0x64, 0x32, 0xb9, 0x5a, 0x55, 0x10, 0x0, 0xc0, 0x6d, 0x6c, 0x6c, 0x9c, 0x57, 0xbb, 0xfa, 0x25, 0x40, 0x14, 0x3, 0x81, 0x40, 0x34, 0x93, 0xc9, 0x2c, 0x57, 0x1c, 0x4, 0x0, 0x90, 0x58, 0x2c, 0xb6, 0x5e, 0xe9, 0xc1, 0x77, 0x1f, 0x10, 0x53, 0x53, 0x53, 0x52, 0xc5, 0x83, 0x14, 0x0, 0x70, 0x7e, 0xbf, 0x5f, 0xd0, 0x42, 0xf5, 0x95, 0x40, 0xf8, 0x7c, 0xbe, 0xcb, 0xa3, 0xa3, 0xa3, 0x3f, 0x1e, 0xbd, 0x1b, 0x0, 0x80, 0x1c, 0x1f, 0x1f, 0x87, 0xb4, 0x56, 0xfd, 0xaa, 0x5, 0x29, 0x51, 0x14, 0xbf, 0xf5, 0xf9, 0x7c, 0x97, 0x5a, 0xad, 0xbe, 0x12, 0x88, 0xf5, 0xf5, 0xf5, 0xd8, 0x83, 0x83, 0x54, 0xb5, 0x42, 0x8f, 0x66, 0x83, 0x94, 0xd6, 0xbd, 0x5f, 0xce, 0x7c, 0x38, 0x3c, 0x3c, 0xfc, 0xb3, 0x50, 0x28, 0xb8, 0xcb, 0x2, 0x1, 0x0, 0xdc, 0xf4, 0xf4, 0xf4, 0xfe, 0x73, 0x15, 0x2f, 0x17, 0xa4, 0x22, 0x91, 0x48, 0x50, 0xb5, 0x2d, 0x0, 0x80, 0x9b, 0x99, 0x99, 0x79, 0xfb, 0xdc, 0x1, 0xc8, 0x5, 0xa9, 0x44, 0x22, 0xf1, 0xfb, 0x9d, 0x10, 0x0, 0x80, 0x9b, 0x9d, 0x9d, 0xd, 0xea, 0x5, 0xc0, 0xad, 0xfd, 0x43, 0x1a, 0x0, 0xb8, 0xdb, 0x9a, 0xa9, 0x8f, 0xb6, 0xa4, 0x46, 0xa3, 0xa4, 0xb7, 0xd5, 0x37, 0xcf, 0xf3, 0x68, 0x75, 0x75, 0xf5, 0x4c, 0xee, 0x99, 0x1c, 0x80, 0x9c, 0x1e, 0xf7, 0xff, 0x16, 0x8b, 0x45, 0x50, 0x5, 0xa0, 0xb7, 0xb7, 0xb7, 0x85, 0x10, 0xa2, 0x2b, 0xf1, 0x84, 0x10, 0xd4, 0xdf, 0xdf, 0x6f, 0x57, 0x3, 0x80, 0x37, 0x18, 0xc, 0x5, 0x3d, 0x2, 0xa0, 0x69, 0x3a, 0x8b, 0x10, 0xe2, 0x4b, 0x2, 0xc0, 0x18, 0xf3, 0xc1, 0x60, 0x70, 0x47, 0x8f, 0x16, 0x38, 0x3a, 0x3a, 0x5a, 0x93, 0x5b, 0xc3, 0x7f, 0x64, 0x81, 0xba, 0xba, 0x3a, 0x49, 0x8f, 0x0, 0x1a, 0x1a, 0x1a, 0xd4, 0xcd, 0x0, 0x93, 0xc9, 0xa4, 0xcb, 0x21, 0xe8, 0x74, 0x3a, 0xd5, 0x1, 0xa0, 0x69, 0x5a, 0x77, 0x1d, 0x80, 0x31, 0x2e, 0x38, 0x9d, 0x4e, 0xb1, 0x66, 0x1, 0x30, 0xc, 0x23, 0x28, 0x3d, 0x93, 0x9b, 0x1, 0xb9, 0x9a, 0x6, 0x60, 0x36, 0x9b, 0x75, 0xd7, 0x1, 0x4a, 0x21, 0xa8, 0x26, 0x0, 0x94, 0xa, 0x41, 0xb2, 0x0, 0x18, 0x86, 0xc9, 0xe9, 0xd, 0x80, 0x52, 0x8, 0x92, 0x5, 0x60, 0xb1, 0x58, 0x74, 0x67, 0x1, 0xa5, 0x10, 0xa4, 0x4, 0x40, 0x77, 0x43, 0xd0, 0xe1, 0x70, 0xa8, 0x9f, 0x1, 0x14, 0x45, 0x1, 0x45, 0x51, 0x79, 0x3d, 0x1, 0x68, 0x6e, 0x6e, 0x4e, 0xaa, 0x6, 0x80, 0x10, 0x42, 0x6, 0x83, 0x41, 0x37, 0x36, 0x28, 0x15, 0x82, 0x6a, 0x2, 0x0, 0x4d, 0xd3, 0xa9, 0x52, 0xcf, 0x95, 0x0, 0xe8, 0x66, 0xe, 0x98, 0xcd, 0x66, 0xa1, 0x6c, 0x0, 0x7a, 0x5a, 0x8b, 0x59, 0x2c, 0x96, 0x64, 0xcd, 0x2, 0xb8, 0x2b, 0x4, 0xe9, 0xde, 0x2, 0x77, 0x85, 0xa0, 0x9a, 0xb0, 0x40, 0xa9, 0x10, 0xa4, 0x8, 0xc0, 0x64, 0x32, 0xe9, 0x6, 0x40, 0xa9, 0x10, 0x54, 0xaa, 0x3, 0x74, 0xf3, 0x16, 0x70, 0xb9, 0x5c, 0xe5, 0x3, 0xe8, 0xe9, 0xe9, 0x69, 0xd5, 0xc3, 0x66, 0x18, 0x63, 0x5c, 0x68, 0x6a, 0x6a, 0x12, 0xcb, 0x5, 0xa0, 0x9b, 0xd5, 0x38, 0x4d, 0xd3, 0x29, 0x8a, 0xa2, 0xa0, 0x2c, 0x0, 0x18, 0x63, 0x3e, 0x14, 0xa, 0xfd, 0x55, 0xb, 0x21, 0x48, 0xd1, 0x2, 0x7a, 0x59, 0x8d, 0xdf, 0x1b, 0x80, 0x1e, 0x56, 0xe3, 0x84, 0x10, 0x34, 0x30, 0x30, 0x60, 0xbb, 0xeb, 0x77, 0x46, 0x5, 0xef, 0x48, 0xcf, 0x4d, 0xec, 0x8d, 0x99, 0x5, 0xf5, 0xf5, 0xf5, 0xef, 0x46, 0x47, 0x47, 0xb, 0x2e, 0x97, 0xeb, 0xbc, 0x54, 0x8, 0x52, 0x4, 0xc0, 0x30, 0x8c, 0xf4, 0x5c, 0x4, 0x9b, 0x4c, 0xa6, 0xf4, 0xf8, 0xf8, 0xb8, 0xc8, 0xb2, 0x6c, 0x32, 0x9d, 0x4e, 0xff, 0xd4, 0xdd, 0xdd, 0x7d, 0x66, 0x34, 0x1a, 0x8b, 0xd7, 0x3, 0xfd, 0xae, 0x5b, 0x29, 0xb2, 0x57, 0x66, 0xb6, 0xb6, 0xb6, 0xde, 0xc4, 0xe3, 0xf1, 0x6f, 0xae, 0xaf, 0xc1, 0x28, 0x5d, 0x85, 0x79, 0x2, 0xc1, 0x60, 0xb5, 0x5a, 0xa3, 0xa3, 0xa3, 0xa3, 0x45, 0xab, 0xd5, 0x9a, 0x2a, 0x16, 0x8b, 0x8b, 0x6d, 0x6d, 0x6d, 0xef, 0xd5, 0x8a, 0x55, 0xd, 0x20, 0x91, 0x48, 0xbc, 0x3e, 0x38, 0x38, 0xf8, 0xda, 0x6e, 0xb7, 0xf7, 0x5f, 0x5c, 0x5c, 0xd4, 0x7b, 0xbd, 0xde, 0xbc, 0x20, 0x8, 0xcd, 0x85, 0x42, 0x81, 0xfe, 0xf0, 0xae, 0xac, 0x10, 0x98, 0x9b, 0xd5, 0xc5, 0x18, 0x17, 0x59, 0x96, 0x3d, 0x1d, 0x19, 0x19, 0x1, 0x96, 0x65, 0x5, 0x8a, 0xa2, 0x7e, 0x6c, 0x69, 0x69, 0x49, 0x3d, 0x44, 0xb0, 0x2a, 0x0, 0x1f, 0xcc, 0x74, 0x75, 0x41, 0xea, 0xfa, 0x7b, 0x32, 0x99, 0x64, 0x76, 0x77, 0x77, 0x5d, 0xe, 0x87, 0xa3, 0x5f, 0x14, 0xc5, 0x57, 0x57, 0x60, 0x5a, 0x8b, 0xc5, 0xa2, 0xf1, 0xbe, 0x50, 0x6e, 0xa, 0x66, 0x18, 0x26, 0x31, 0x36, 0x36, 0x96, 0x65, 0x59, 0x36, 0x29, 0x49, 0x92, 0xb7, 0xbd, 0xbd, 0xfd, 0x9f, 0x72, 0xda, 0xf9, 0xd1, 0x1, 0xa8, 0x1, 0x93, 0xcf, 0xe7, 0xa9, 0x93, 0x93, 0x13, 0x1b, 0x4d, 0xd3, 0x9f, 0xb, 0x82, 0x60, 0xf5, 0x7a, 0xbd, 0xd9, 0x54, 0x2a, 0xe5, 0xcc, 0x64, 0x32, 0xe, 0xb9, 0x6e, 0xb9, 0x16, 0x8c, 0x31, 0x2e, 0xda, 0x6c, 0xb6, 0xc8, 0xd0, 0xd0, 0x10, 0x65, 0xb3, 0xd9, 0x92, 0x95, 0xa8, 0x6e, 0xc5, 0x0, 0xa8, 0xe9, 0x96, 0x68, 0x34, 0x6a, 0xdd, 0xdf, 0xdf, 0x6f, 0x76, 0xb9, 0x5c, 0x9f, 0x89, 0xa2, 0x58, 0xbf, 0xb8, 0xb8, 0x8, 0x26, 0x93, 0x29, 0x3b, 0x3c, 0x3c, 0x8c, 0xed, 0x76, 0x7b, 0xd2, 0x68, 0x34, 0xfe, 0xd0, 0xd8, 0xd8, 0x98, 0xae, 0xb6, 0xe0, 0x8a, 0x1, 0x50, 0xb, 0xe6, 0xa9, 0x5, 0xbf, 0x9c, 0x97, 0xf3, 0xff, 0xf3, 0x2f, 0x6a, 0x82, 0x7f, 0xf6, 0x4e, 0xca, 0x1b, 0xf5, 0x0, 0x0, 0x0, 0x0, 0x49, 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82
-};
-
-class TestPhysics2DMainLoop : public MainLoop {
- GDCLASS(TestPhysics2DMainLoop, MainLoop);
-
- RID circle_img;
- RID circle_shape;
- RID space;
- RID canvas;
- RID ray;
- RID ray_query;
- Transform2D view_xform;
-
- Vector2 ray_from, ray_to;
-
- struct BodyShapeData {
- RID image;
- RID shape;
- };
-
- BodyShapeData body_shape_data[8];
-
- void _create_body_shape_data() {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer2D *ps = PhysicsServer2D::get_singleton();
-
- // SEGMENT
-
- {
- Vector<uint8_t> pixels;
- pixels.resize(32 * 2 * 2);
- for (int i = 0; i < 2; i++) {
- for (int j = 0; j < 32; j++) {
- pixels.set(i * 32 * 2 + j * 2 + 0, (j == 0) ? 255 : 0);
- pixels.set(i * 32 * 2 + j * 2 + 1, 255);
- }
- }
-
- Ref<Image> image = memnew(Image(32, 2, 0, Image::FORMAT_LA8, pixels));
-
- body_shape_data[PhysicsServer2D::SHAPE_SEGMENT].image = vs->texture_2d_create(image);
-
- RID segment_shape = ps->segment_shape_create();
- Rect2 sg(Point2(-16, 0), Point2(16, 0));
- ps->shape_set_data(segment_shape, sg);
-
- body_shape_data[PhysicsServer2D::SHAPE_SEGMENT].shape = segment_shape;
- }
-
- // CIRCLE
-
- {
- Vector<uint8_t> pixels;
- pixels.resize(32 * 32 * 2);
- for (int i = 0; i < 32; i++) {
- for (int j = 0; j < 32; j++) {
- bool black = Vector2(i - 16, j - 16).length_squared() < 16 * 16;
-
- pixels.set(i * 32 * 2 + j * 2 + 0, (i == 16 || j == 16) ? 255 : 0);
- pixels.set(i * 32 * 2 + j * 2 + 1, black ? 255 : 0);
- }
- }
-
- Ref<Image> image = memnew(Image(32, 32, 0, Image::FORMAT_LA8, pixels));
-
- body_shape_data[PhysicsServer2D::SHAPE_CIRCLE].image = vs->texture_2d_create(image);
-
- RID circle_shape = ps->circle_shape_create();
- ps->shape_set_data(circle_shape, 16);
-
- body_shape_data[PhysicsServer2D::SHAPE_CIRCLE].shape = circle_shape;
- }
-
- // BOX
-
- {
- Vector<uint8_t> pixels;
- pixels.resize(32 * 32 * 2);
- for (int i = 0; i < 32; i++) {
- for (int j = 0; j < 32; j++) {
- bool black = i > 0 && i < 31 && j > 0 && j < 31;
-
- pixels.set(i * 32 * 2 + j * 2 + 0, black ? 0 : 255);
- pixels.set(i * 32 * 2 + j * 2 + 1, 255);
- }
- }
-
- Ref<Image> image = memnew(Image(32, 32, 0, Image::FORMAT_LA8, pixels));
-
- body_shape_data[PhysicsServer2D::SHAPE_RECTANGLE].image = vs->texture_2d_create(image);
-
- RID rectangle_shape = ps->rectangle_shape_create();
- ps->shape_set_data(rectangle_shape, Vector2(16, 16));
-
- body_shape_data[PhysicsServer2D::SHAPE_RECTANGLE].shape = rectangle_shape;
- }
-
- // CAPSULE
-
- {
- Vector<uint8_t> pixels;
- pixels.resize(32 * 64 * 2);
- for (int i = 0; i < 64; i++) {
- for (int j = 0; j < 32; j++) {
- int si = i > 48 ? i - 32 : (i < 16 ? i : 16);
- bool black = Vector2(si - 16, j - 16).length_squared() < 16 * 16;
-
- pixels.set(i * 32 * 2 + j * 2 + 0, (i == 16 || j == 16 || i == 48) ? 255 : 0);
- pixels.set(i * 32 * 2 + j * 2 + 1, black ? 255 : 0);
- }
- }
-
- Ref<Image> image = memnew(Image(32, 64, 0, Image::FORMAT_LA8, pixels));
-
- body_shape_data[PhysicsServer2D::SHAPE_CAPSULE].image = vs->texture_2d_create(image);
-
- RID capsule_shape = ps->capsule_shape_create();
- ps->shape_set_data(capsule_shape, Vector2(16, 32));
-
- body_shape_data[PhysicsServer2D::SHAPE_CAPSULE].shape = capsule_shape;
- }
-
- // CONVEX
-
- {
- Ref<Image> image = memnew(Image(convex_png));
-
- body_shape_data[PhysicsServer2D::SHAPE_CONVEX_POLYGON].image = vs->texture_2d_create(image);
-
- RID convex_polygon_shape = ps->convex_polygon_shape_create();
-
- Vector<Vector2> arr;
- Point2 sb(32, 32);
- arr.push_back(Point2(20, 3) - sb);
- arr.push_back(Point2(58, 23) - sb);
- arr.push_back(Point2(55, 54) - sb);
- arr.push_back(Point2(27, 60) - sb);
- arr.push_back(Point2(5, 56) - sb);
- arr.push_back(Point2(4, 20) - sb);
- arr.push_back(Point2(11, 7) - sb);
- ps->shape_set_data(convex_polygon_shape, arr);
-
- body_shape_data[PhysicsServer2D::SHAPE_CONVEX_POLYGON].shape = convex_polygon_shape;
- }
- }
-
- void _do_ray_query() {
- // FIXME: Do something?
- }
-
-protected:
- void input_event(const Ref<InputEvent> &p_event) {
- Ref<InputEventMouseButton> mb = p_event;
-
- if (mb.is_valid()) {
- if (mb->is_pressed()) {
- Point2 p = mb->get_position();
-
- if (mb->get_button_index() == MouseButton::LEFT) {
- ray_to = p;
- _do_ray_query();
- } else if (mb->get_button_index() == MouseButton::RIGHT) {
- ray_from = p;
- _do_ray_query();
- }
- }
- }
-
- Ref<InputEventMouseMotion> mm = p_event;
-
- if (mm.is_valid()) {
- Point2 p = mm->get_position();
-
- if ((mm->get_button_mask() & MouseButton::MASK_LEFT) != MouseButton::NONE) {
- ray_to = p;
- _do_ray_query();
- } else if ((mm->get_button_mask() & MouseButton::MASK_RIGHT) != MouseButton::NONE) {
- ray_from = p;
- _do_ray_query();
- }
- }
- }
-
- RID _add_body(PhysicsServer2D::ShapeType p_shape, const Transform2D &p_xform) {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer2D *ps = PhysicsServer2D::get_singleton();
-
- RID body = ps->body_create();
- ps->body_add_shape(body, body_shape_data[p_shape].shape);
- ps->body_set_space(body, space);
- ps->body_set_continuous_collision_detection_mode(body, PhysicsServer2D::CCD_MODE_CAST_SHAPE);
- ps->body_set_state(body, PhysicsServer2D::BODY_STATE_TRANSFORM, p_xform);
-
- RID sprite = vs->canvas_item_create();
- vs->canvas_item_set_parent(sprite, canvas);
- vs->canvas_item_set_transform(sprite, p_xform);
- Size2 imgsize(5, 5);
- vs->canvas_item_add_texture_rect(sprite, Rect2(-imgsize / 2.0, imgsize), body_shape_data[p_shape].image);
-
- ps->body_set_force_integration_callback(body, callable_mp(this, &TestPhysics2DMainLoop::_body_moved), sprite);
-
- return body;
- }
-
- void _add_world_boundary(const Vector2 &p_normal, real_t p_d) {
- PhysicsServer2D *ps = PhysicsServer2D::get_singleton();
-
- Array arr;
- arr.push_back(p_normal);
- arr.push_back(p_d);
-
- RID world_boundary = ps->world_boundary_shape_create();
- ps->shape_set_data(world_boundary, arr);
-
- RID plane_body = ps->body_create();
- ps->body_set_mode(plane_body, PhysicsServer2D::BODY_MODE_STATIC);
- ps->body_set_space(plane_body, space);
- ps->body_add_shape(plane_body, world_boundary);
- }
-
- void _add_concave(const Vector<Vector2> &p_points, const Transform2D &p_xform = Transform2D()) {
- PhysicsServer2D *ps = PhysicsServer2D::get_singleton();
- RenderingServer *vs = RenderingServer::get_singleton();
-
- RID concave = ps->concave_polygon_shape_create();
- ps->shape_set_data(concave, p_points);
- RID body = ps->body_create();
- ps->body_set_mode(body, PhysicsServer2D::BODY_MODE_STATIC);
- ps->body_set_space(body, space);
- ps->body_add_shape(body, concave);
- ps->body_set_state(body, PhysicsServer2D::BODY_STATE_TRANSFORM, p_xform);
-
- RID sprite = vs->canvas_item_create();
- vs->canvas_item_set_parent(sprite, canvas);
- vs->canvas_item_set_transform(sprite, p_xform);
- for (int i = 0; i < p_points.size(); i += 2) {
- vs->canvas_item_add_line(sprite, p_points[i], p_points[i + 1], Color(0, 0, 0), 2);
- }
- }
-
- void _body_moved(Object *p_state, RID p_sprite) {
- PhysicsDirectBodyState2D *state = (PhysicsDirectBodyState2D *)p_state;
- RenderingServer::get_singleton()->canvas_item_set_transform(p_sprite, state->get_transform());
- }
-
- void _ray_query_callback(const RID &p_rid, ObjectID p_id, int p_shape, const Vector2 &p_point, const Vector2 &p_normal) {
- Vector2 ray_end;
-
- if (p_rid.is_valid()) {
- ray_end = p_point;
- } else {
- ray_end = ray_to;
- }
-
- RenderingServer *vs = RenderingServer::get_singleton();
-
- vs->canvas_item_clear(ray);
- vs->canvas_item_add_line(ray, ray_from, ray_end, p_rid.is_valid() ? Color(0, 1, 0.4) : Color(1, 0.4, 0), 2);
- if (p_rid.is_valid()) {
- vs->canvas_item_add_line(ray, ray_end, ray_end + p_normal * 20, p_rid.is_valid() ? Color(0, 1, 0.4) : Color(1, 0.4, 0), 2);
- }
- }
-
- static void _bind_methods() {
- ClassDB::bind_method(D_METHOD("_ray_query_callback"), &TestPhysics2DMainLoop::_ray_query_callback);
- }
-
-public:
- virtual void initialize() override {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer2D *ps = PhysicsServer2D::get_singleton();
-
- space = ps->space_create();
- ps->space_set_active(space, true);
- ps->set_active(true);
- ps->area_set_param(space, PhysicsServer2D::AREA_PARAM_GRAVITY_VECTOR, Vector2(0, 1));
- ps->area_set_param(space, PhysicsServer2D::AREA_PARAM_GRAVITY, 980);
-
- {
- RID vp = vs->viewport_create();
- canvas = vs->canvas_create();
-
- Size2i screen_size = DisplayServer::get_singleton()->window_get_size();
- vs->viewport_attach_canvas(vp, canvas);
- vs->viewport_set_size(vp, screen_size.x, screen_size.y);
- vs->viewport_attach_to_screen(vp, Rect2(Vector2(), screen_size));
- vs->viewport_set_active(vp, true);
- vs->viewport_set_canvas_transform(vp, canvas, view_xform);
- }
-
- ray = vs->canvas_item_create();
- vs->canvas_item_set_parent(ray, canvas);
-
- for (int i = 0; i < 32; i++) {
- PhysicsServer2D::ShapeType types[4] = {
- PhysicsServer2D::SHAPE_CIRCLE,
- PhysicsServer2D::SHAPE_CAPSULE,
- PhysicsServer2D::SHAPE_RECTANGLE,
- PhysicsServer2D::SHAPE_CONVEX_POLYGON,
-
- };
-
- PhysicsServer2D::ShapeType type = types[i % 4];
- _add_body(type, Transform2D(i * 0.8, Point2(152 + i * 40, 100 - 40 * i)));
- }
-
- Point2 prev;
-
- Vector<Point2> parr;
- for (int i = 0; i < 30; i++) {
- Point2 p(i * 60, Math::randf() * 70 + 340);
- if (i > 0) {
- parr.push_back(prev);
- parr.push_back(p);
- }
- prev = p;
- }
-
- _add_concave(parr);
- }
-
- virtual bool process(double p_time) override {
- return false;
- }
- virtual void finalize() override {
- }
-
- TestPhysics2DMainLoop() {}
-};
-
-namespace TestPhysics2D {
-
-MainLoop *test() {
- return memnew(TestPhysics2DMainLoop);
-}
-} // namespace TestPhysics2D
diff --git a/tests/servers/test_physics_2d.h b/tests/servers/test_physics_2d.h
deleted file mode 100644
index b6c47574cd..0000000000
--- a/tests/servers/test_physics_2d.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_physics_2d.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_PHYSICS_2D_H
-#define TEST_PHYSICS_2D_H
-
-class MainLoop;
-
-namespace TestPhysics2D {
-
-MainLoop *test();
-}
-
-#endif // TEST_PHYSICS_2D_H
diff --git a/tests/servers/test_physics_3d.cpp b/tests/servers/test_physics_3d.cpp
deleted file mode 100644
index 3d38b9d901..0000000000
--- a/tests/servers/test_physics_3d.cpp
+++ /dev/null
@@ -1,410 +0,0 @@
-/*************************************************************************/
-/* test_physics_3d.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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. */
-/*************************************************************************/
-
-#include "test_physics_3d.h"
-
-#include "core/math/convex_hull.h"
-#include "core/math/geometry_3d.h"
-#include "core/os/main_loop.h"
-#include "servers/physics_server_3d.h"
-#include "servers/rendering_server.h"
-
-class TestPhysics3DMainLoop : public MainLoop {
- GDCLASS(TestPhysics3DMainLoop, MainLoop);
-
- enum {
- LINK_COUNT = 20,
- };
-
- RID test_cube;
-
- RID plane;
- RID sphere;
- RID light;
- RID camera;
- RID mover;
- RID scenario;
- RID space;
-
- RID character;
-
- real_t ofs_x, ofs_y;
-
- Point2 joy_direction;
-
- List<RID> bodies;
- Map<PhysicsServer3D::ShapeType, RID> type_shape_map;
- Map<PhysicsServer3D::ShapeType, RID> type_mesh_map;
-
- void body_changed_transform(Object *p_state, RID p_visual_instance) {
- PhysicsDirectBodyState3D *state = (PhysicsDirectBodyState3D *)p_state;
- RenderingServer *vs = RenderingServer::get_singleton();
- Transform3D t = state->get_transform();
- vs->instance_set_transform(p_visual_instance, t);
- }
-
- bool quit;
-
-protected:
- RID create_body(PhysicsServer3D::ShapeType p_shape, PhysicsServer3D::BodyMode p_body, const Transform3D p_location, bool p_active_default = true, const Transform3D &p_shape_xform = Transform3D()) {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
-
- RID mesh_instance = vs->instance_create2(type_mesh_map[p_shape], scenario);
- RID body = ps->body_create();
- ps->body_set_mode(body, p_body);
- ps->body_set_state(body, PhysicsServer3D::BODY_STATE_SLEEPING, !p_active_default);
- ps->body_set_space(body, space);
- ps->body_set_param(body, PhysicsServer3D::BODY_PARAM_BOUNCE, 0.0);
- //todo set space
- ps->body_add_shape(body, type_shape_map[p_shape]);
- ps->body_set_force_integration_callback(body, callable_mp(this, &TestPhysics3DMainLoop::body_changed_transform), mesh_instance);
-
- ps->body_set_state(body, PhysicsServer3D::BODY_STATE_TRANSFORM, p_location);
- bodies.push_back(body);
-
- if (p_body == PhysicsServer3D::BODY_MODE_STATIC) {
- vs->instance_set_transform(mesh_instance, p_location);
- }
- return body;
- }
-
- RID create_world_boundary(const Plane &p_plane) {
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
-
- RID world_boundary_shape = ps->shape_create(PhysicsServer3D::SHAPE_WORLD_BOUNDARY);
- ps->shape_set_data(world_boundary_shape, p_plane);
-
- RID b = ps->body_create();
- ps->body_set_mode(b, PhysicsServer3D::BODY_MODE_STATIC);
-
- ps->body_set_space(b, space);
- ps->body_add_shape(b, world_boundary_shape);
- return b;
- }
-
- void configure_body(RID p_body, real_t p_mass, real_t p_friction, real_t p_bounce) {
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
- ps->body_set_param(p_body, PhysicsServer3D::BODY_PARAM_MASS, p_mass);
- ps->body_set_param(p_body, PhysicsServer3D::BODY_PARAM_FRICTION, p_friction);
- ps->body_set_param(p_body, PhysicsServer3D::BODY_PARAM_BOUNCE, p_bounce);
- }
-
- void initialize_shapes() {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
-
- /* SPHERE SHAPE */
- RID sphere_mesh = vs->make_sphere_mesh(10, 20, 0.5);
- type_mesh_map[PhysicsServer3D::SHAPE_SPHERE] = sphere_mesh;
-
- RID sphere_shape = ps->shape_create(PhysicsServer3D::SHAPE_SPHERE);
- ps->shape_set_data(sphere_shape, 0.5);
- type_shape_map[PhysicsServer3D::SHAPE_SPHERE] = sphere_shape;
-
- /* BOX SHAPE */
-
- Vector<Plane> box_planes = Geometry3D::build_box_planes(Vector3(0.5, 0.5, 0.5));
- RID box_mesh = vs->mesh_create();
- Geometry3D::MeshData box_data = Geometry3D::build_convex_mesh(box_planes);
- vs->mesh_add_surface_from_mesh_data(box_mesh, box_data);
- type_mesh_map[PhysicsServer3D::SHAPE_BOX] = box_mesh;
-
- RID box_shape = ps->shape_create(PhysicsServer3D::SHAPE_BOX);
- ps->shape_set_data(box_shape, Vector3(0.5, 0.5, 0.5));
- type_shape_map[PhysicsServer3D::SHAPE_BOX] = box_shape;
-
- /* CAPSULE SHAPE */
-
- Vector<Plane> capsule_planes = Geometry3D::build_capsule_planes(0.5, 0.7, 12, Vector3::AXIS_Z);
-
- RID capsule_mesh = vs->mesh_create();
- Geometry3D::MeshData capsule_data = Geometry3D::build_convex_mesh(capsule_planes);
- vs->mesh_add_surface_from_mesh_data(capsule_mesh, capsule_data);
-
- type_mesh_map[PhysicsServer3D::SHAPE_CAPSULE] = capsule_mesh;
-
- RID capsule_shape = ps->shape_create(PhysicsServer3D::SHAPE_CAPSULE);
- Dictionary capsule_params;
- capsule_params["radius"] = 0.5;
- capsule_params["height"] = 1.4;
- ps->shape_set_data(capsule_shape, capsule_params);
- type_shape_map[PhysicsServer3D::SHAPE_CAPSULE] = capsule_shape;
-
- /* CONVEX SHAPE */
-
- Vector<Plane> convex_planes = Geometry3D::build_cylinder_planes(0.5, 0.7, 5, Vector3::AXIS_Z);
-
- RID convex_mesh = vs->mesh_create();
- Geometry3D::MeshData convex_data = Geometry3D::build_convex_mesh(convex_planes);
- ConvexHullComputer::convex_hull(convex_data.vertices, convex_data);
- vs->mesh_add_surface_from_mesh_data(convex_mesh, convex_data);
-
- type_mesh_map[PhysicsServer3D::SHAPE_CONVEX_POLYGON] = convex_mesh;
-
- RID convex_shape = ps->shape_create(PhysicsServer3D::SHAPE_CONVEX_POLYGON);
- ps->shape_set_data(convex_shape, convex_data.vertices);
- type_shape_map[PhysicsServer3D::SHAPE_CONVEX_POLYGON] = convex_shape;
- }
-
- void make_trimesh(Vector<Vector3> p_faces, const Transform3D &p_xform = Transform3D()) {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
- RID trimesh_shape = ps->shape_create(PhysicsServer3D::SHAPE_CONCAVE_POLYGON);
- Dictionary trimesh_params;
- trimesh_params["faces"] = p_faces;
- trimesh_params["backface_collision"] = false;
- ps->shape_set_data(trimesh_shape, trimesh_params);
- Vector<Vector3> normals; // for drawing
- for (int i = 0; i < p_faces.size() / 3; i++) {
- Plane p(p_faces[i * 3 + 0], p_faces[i * 3 + 1], p_faces[i * 3 + 2]);
- normals.push_back(p.normal);
- normals.push_back(p.normal);
- normals.push_back(p.normal);
- }
-
- RID trimesh_mesh = vs->mesh_create();
- Array d;
- d.resize(RS::ARRAY_MAX);
- d[RS::ARRAY_VERTEX] = p_faces;
- d[RS::ARRAY_NORMAL] = normals;
- vs->mesh_add_surface_from_arrays(trimesh_mesh, RS::PRIMITIVE_TRIANGLES, d);
-
- RID triins = vs->instance_create2(trimesh_mesh, scenario);
-
- RID tribody = ps->body_create();
- ps->body_set_mode(tribody, PhysicsServer3D::BODY_MODE_STATIC);
- ps->body_set_space(tribody, space);
- //todo set space
- ps->body_add_shape(tribody, trimesh_shape);
- Transform3D tritrans = p_xform;
- ps->body_set_state(tribody, PhysicsServer3D::BODY_STATE_TRANSFORM, tritrans);
- vs->instance_set_transform(triins, tritrans);
- }
-
- void make_grid(int p_width, int p_height, real_t p_cellsize, real_t p_cellheight, const Transform3D &p_xform = Transform3D()) {
- Vector<Vector<real_t>> grid;
-
- grid.resize(p_width);
-
- for (int i = 0; i < p_width; i++) {
- grid.write[i].resize(p_height);
-
- for (int j = 0; j < p_height; j++) {
- grid.write[i].write[j] = 1.0 + Math::random(-p_cellheight, p_cellheight);
- }
- }
-
- Vector<Vector3> faces;
-
- for (int i = 1; i < p_width; i++) {
- for (int j = 1; j < p_height; j++) {
-#define MAKE_VERTEX(m_x, m_z) \
- faces.push_back(Vector3((m_x - p_width / 2) * p_cellsize, grid[m_x][m_z], (m_z - p_height / 2) * p_cellsize))
-
- MAKE_VERTEX(i, j - 1);
- MAKE_VERTEX(i, j);
- MAKE_VERTEX(i - 1, j);
-
- MAKE_VERTEX(i - 1, j - 1);
- MAKE_VERTEX(i, j - 1);
- MAKE_VERTEX(i - 1, j);
- }
- }
-
- make_trimesh(faces, p_xform);
- }
-
-public:
- virtual void input_event(const Ref<InputEvent> &p_event) {
- Ref<InputEventMouseMotion> mm = p_event;
- if (mm.is_valid() && (mm->get_button_mask() & MouseButton::MASK_MIDDLE) != MouseButton::NONE) {
- ofs_y -= mm->get_relative().y / 200.0;
- ofs_x += mm->get_relative().x / 200.0;
- }
-
- if (mm.is_valid() && (mm->get_button_mask() & MouseButton::MASK_LEFT) != MouseButton::NONE) {
- real_t y = -mm->get_relative().y / 20.0;
- real_t x = mm->get_relative().x / 20.0;
-
- if (mover.is_valid()) {
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
- Transform3D t = ps->body_get_state(mover, PhysicsServer3D::BODY_STATE_TRANSFORM);
- t.origin += Vector3(x, y, 0);
-
- ps->body_set_state(mover, PhysicsServer3D::BODY_STATE_TRANSFORM, t);
- }
- }
- }
-
- virtual void request_quit() {
- quit = true;
- }
- virtual void initialize() override {
- ofs_x = ofs_y = 0;
- initialize_shapes();
-
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
- space = ps->space_create();
- ps->space_set_active(space, true);
-
- RenderingServer *vs = RenderingServer::get_singleton();
-
- /* LIGHT */
- RID lightaux = vs->directional_light_create();
- scenario = vs->scenario_create();
- vs->light_set_shadow(lightaux, true);
- light = vs->instance_create2(lightaux, scenario);
- Transform3D t;
- t.rotate(Vector3(1.0, 0, 0), 0.6);
- vs->instance_set_transform(light, t);
-
- /* CAMERA */
-
- camera = vs->camera_create();
-
- RID viewport = vs->viewport_create();
- Size2i screen_size = DisplayServer::get_singleton()->window_get_size();
- vs->viewport_set_size(viewport, screen_size.x, screen_size.y);
- vs->viewport_attach_to_screen(viewport, Rect2(Vector2(), screen_size));
- vs->viewport_set_active(viewport, true);
- vs->viewport_attach_camera(viewport, camera);
- vs->viewport_set_scenario(viewport, scenario);
-
- vs->camera_set_perspective(camera, 60, 0.1, 40.0);
- vs->camera_set_transform(camera, Transform3D(Basis(), Vector3(0, 9, 12)));
-
- Transform3D gxf;
- gxf.basis.scale(Vector3(1.4, 0.4, 1.4));
- gxf.origin = Vector3(-2, 1, -2);
- make_grid(5, 5, 2.5, 1, gxf);
- test_fall();
- quit = false;
- }
- virtual bool physics_process(double p_time) override {
- if (mover.is_valid()) {
- static real_t joy_speed = 10;
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
- Transform3D t = ps->body_get_state(mover, PhysicsServer3D::BODY_STATE_TRANSFORM);
- t.origin += Vector3(joy_speed * joy_direction.x * p_time, -joy_speed * joy_direction.y * p_time, 0);
- ps->body_set_state(mover, PhysicsServer3D::BODY_STATE_TRANSFORM, t);
- };
-
- Transform3D cameratr;
- cameratr.rotate(Vector3(0, 1, 0), ofs_x);
- cameratr.rotate(Vector3(1, 0, 0), -ofs_y);
- cameratr.translate(Vector3(0, 2, 8));
- RenderingServer *vs = RenderingServer::get_singleton();
- vs->camera_set_transform(camera, cameratr);
-
- return quit;
- }
- virtual void finalize() override {
- }
-
- void test_joint() {
- }
-
- void test_hinge() {
- }
-
- void test_character() {
- RenderingServer *vs = RenderingServer::get_singleton();
- PhysicsServer3D *ps = PhysicsServer3D::get_singleton();
-
- Vector<Plane> capsule_planes = Geometry3D::build_capsule_planes(0.5, 1, 12, 5, Vector3::AXIS_Y);
-
- RID capsule_mesh = vs->mesh_create();
- Geometry3D::MeshData capsule_data = Geometry3D::build_convex_mesh(capsule_planes);
- vs->mesh_add_surface_from_mesh_data(capsule_mesh, capsule_data);
- type_mesh_map[PhysicsServer3D::SHAPE_CAPSULE] = capsule_mesh;
-
- RID capsule_shape = ps->shape_create(PhysicsServer3D::SHAPE_CAPSULE);
- Dictionary capsule_params;
- capsule_params["radius"] = 0.5;
- capsule_params["height"] = 1;
- Transform3D shape_xform;
- shape_xform.rotate(Vector3(1, 0, 0), Math_PI / 2.0);
- //shape_xform.origin=Vector3(1,1,1);
- ps->shape_set_data(capsule_shape, capsule_params);
-
- RID mesh_instance = vs->instance_create2(capsule_mesh, scenario);
- character = ps->body_create();
- ps->body_set_mode(character, PhysicsServer3D::BODY_MODE_DYNAMIC_LINEAR);
- ps->body_set_space(character, space);
- //todo add space
- ps->body_add_shape(character, capsule_shape);
- ps->body_set_force_integration_callback(character, callable_mp(this, &TestPhysics3DMainLoop::body_changed_transform), mesh_instance);
-
- ps->body_set_state(character, PhysicsServer3D::BODY_STATE_TRANSFORM, Transform3D(Basis(), Vector3(-2, 5, -2)));
- bodies.push_back(character);
- }
-
- void test_fall() {
- for (int i = 0; i < 35; i++) {
- static const PhysicsServer3D::ShapeType shape_idx[] = {
- PhysicsServer3D::SHAPE_CAPSULE,
- PhysicsServer3D::SHAPE_BOX,
- PhysicsServer3D::SHAPE_SPHERE,
- PhysicsServer3D::SHAPE_CONVEX_POLYGON
- };
-
- PhysicsServer3D::ShapeType type = shape_idx[i % 4];
-
- Transform3D t;
-
- t.origin = Vector3(0.0 * i, 3.5 + 1.1 * i, 0.7 + 0.0 * i);
- t.basis.rotate(Vector3(0.2, -1, 0), Math_PI / 2 * 0.6);
-
- create_body(type, PhysicsServer3D::BODY_MODE_DYNAMIC, t);
- }
-
- create_world_boundary(Plane(Vector3(0, 1, 0), -1));
- }
-
- void test_activate() {
- create_body(PhysicsServer3D::SHAPE_BOX, PhysicsServer3D::BODY_MODE_DYNAMIC, Transform3D(Basis(), Vector3(0, 2, 0)), true);
- create_world_boundary(Plane(Vector3(0, 1, 0), -1));
- }
-
- virtual bool process(double p_time) override {
- return false;
- }
-
- TestPhysics3DMainLoop() {
- }
-};
-
-namespace TestPhysics3D {
-
-MainLoop *test() {
- return memnew(TestPhysics3DMainLoop);
-}
-} // namespace TestPhysics3D
diff --git a/tests/servers/test_physics_3d.h b/tests/servers/test_physics_3d.h
deleted file mode 100644
index f618d0fb4f..0000000000
--- a/tests/servers/test_physics_3d.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_physics_3d.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_PHYSICS_H
-#define TEST_PHYSICS_H
-
-class MainLoop;
-
-namespace TestPhysics3D {
-
-MainLoop *test();
-}
-
-#endif
diff --git a/tests/servers/test_render.cpp b/tests/servers/test_render.cpp
deleted file mode 100644
index 44403e3724..0000000000
--- a/tests/servers/test_render.cpp
+++ /dev/null
@@ -1,232 +0,0 @@
-/*************************************************************************/
-/* test_render.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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. */
-/*************************************************************************/
-
-#include "test_render.h"
-
-#include "core/math/convex_hull.h"
-#include "core/os/main_loop.h"
-#include "servers/rendering_server.h"
-
-#define OBJECT_COUNT 50
-
-namespace TestRender {
-
-class TestMainLoop : public MainLoop {
- RID test_cube;
- RID instance;
- RID camera;
- RID viewport;
- RID light;
- RID scenario;
-
- struct InstanceInfo {
- RID instance;
- Transform3D base;
- Vector3 rot_axis;
- };
-
- List<InstanceInfo> instances;
-
- float ofs;
- bool quit;
-
-protected:
-public:
- virtual void input_event(const Ref<InputEvent> &p_event) {
- if (p_event->is_pressed()) {
- quit = true;
- }
- }
-
- virtual void init() {
- print_line("INITIALIZING TEST RENDER");
- RenderingServer *vs = RenderingServer::get_singleton();
- test_cube = vs->get_test_cube();
- scenario = vs->scenario_create();
-
- Vector<Vector3> vts;
-
- /*
- Vector<Plane> sp = Geometry3D::build_sphere_planes(2,5,5);
- Geometry3D::MeshData md2 = Geometry3D::build_convex_mesh(sp);
- vts=md2.vertices;
-*/
- /*
-
- static const int s = 20;
- for(int i=0;i<s;i++) {
- Basis rot(Vector3(0,1,0),i*Math_PI/s);
-
- for(int j=0;j<s;j++) {
- Vector3 v;
- v.x=Math::sin(j*Math_PI*2/s);
- v.y=Math::cos(j*Math_PI*2/s);
-
- vts.push_back( rot.xform(v*2 ) );
- }
- }*/
- /*for(int i=0;i<100;i++) {
- vts.push_back( Vector3(Math::randf()*2-1.0,Math::randf()*2-1.0,Math::randf()*2-1.0).normalized()*2);
- }*/
- /*
- vts.push_back(Vector3(0,0,1));
- vts.push_back(Vector3(0,0,-1));
- vts.push_back(Vector3(0,1,0));
- vts.push_back(Vector3(0,-1,0));
- vts.push_back(Vector3(1,0,0));
- vts.push_back(Vector3(-1,0,0));*/
-
- vts.push_back(Vector3(1, 1, 1));
- vts.push_back(Vector3(1, -1, 1));
- vts.push_back(Vector3(-1, 1, 1));
- vts.push_back(Vector3(-1, -1, 1));
- vts.push_back(Vector3(1, 1, -1));
- vts.push_back(Vector3(1, -1, -1));
- vts.push_back(Vector3(-1, 1, -1));
- vts.push_back(Vector3(-1, -1, -1));
-
- Geometry3D::MeshData md;
- Error err = ConvexHullComputer::convex_hull(vts, md);
- print_line("ERR: " + itos(err));
- test_cube = vs->mesh_create();
- vs->mesh_add_surface_from_mesh_data(test_cube, md);
- //vs->scenario_set_debug(scenario,RS::SCENARIO_DEBUG_WIREFRAME);
-
- /*
- RID sm = vs->shader_create();
- //vs->shader_set_fragment_code(sm,"OUT_ALPHA=mod(TIME,1);");
- //vs->shader_set_vertex_code(sm,"OUT_VERTEX=IN_VERTEX*mod(TIME,1);");
- vs->shader_set_fragment_code(sm,"OUT_DIFFUSE=vec3(1,0,1);OUT_GLOW=abs(sin(TIME));");
- RID tcmat = vs->mesh_surface_get_material(test_cube,0);
- vs->material_set_shader(tcmat,sm);
- */
-
- List<String> cmdline = OS::get_singleton()->get_cmdline_args();
- int object_count = OBJECT_COUNT;
- if (cmdline.size() > 0 && cmdline[cmdline.size() - 1].to_int()) {
- object_count = cmdline[cmdline.size() - 1].to_int();
- };
-
- for (int i = 0; i < object_count; i++) {
- InstanceInfo ii;
-
- ii.instance = vs->instance_create2(test_cube, scenario);
-
- ii.base.translate(Math::random(-20, 20), Math::random(-20, 20), Math::random(-20, 18));
- ii.base.rotate(Vector3(0, 1, 0), Math::randf() * Math_PI);
- ii.base.rotate(Vector3(1, 0, 0), Math::randf() * Math_PI);
- vs->instance_set_transform(ii.instance, ii.base);
-
- ii.rot_axis = Vector3(Math::random(-1, 1), Math::random(-1, 1), Math::random(-1, 1)).normalized();
-
- instances.push_back(ii);
- }
-
- camera = vs->camera_create();
-
- // vs->camera_set_perspective( camera, 60.0,0.1, 100.0 );
-
- viewport = vs->viewport_create();
- Size2i screen_size = DisplayServer::get_singleton()->window_get_size();
- vs->viewport_set_size(viewport, screen_size.x, screen_size.y);
- vs->viewport_attach_to_screen(viewport, Rect2(Vector2(), screen_size));
- vs->viewport_set_active(viewport, true);
- vs->viewport_attach_camera(viewport, camera);
- vs->viewport_set_scenario(viewport, scenario);
- vs->camera_set_transform(camera, Transform3D(Basis(), Vector3(0, 3, 30)));
- vs->camera_set_perspective(camera, 60, 0.1, 1000);
-
- /*
- RID lightaux = vs->light_create( RenderingServer::LIGHT_OMNI );
- vs->light_set_var( lightaux, RenderingServer::LIGHT_VAR_RADIUS, 80 );
- vs->light_set_var( lightaux, RenderingServer::LIGHT_VAR_ATTENUATION, 1 );
- vs->light_set_var( lightaux, RenderingServer::LIGHT_VAR_ENERGY, 1.5 );
- light = vs->instance_create( lightaux );
- */
- RID lightaux;
-
- lightaux = vs->directional_light_create();
- //vs->light_set_color( lightaux, RenderingServer::LIGHT_COLOR_AMBIENT, Color(0.0,0.0,0.0) );
- vs->light_set_color(lightaux, Color(1.0, 1.0, 1.0));
- //vs->light_set_shadow( lightaux, true );
- light = vs->instance_create2(lightaux, scenario);
- Transform3D lla;
- //lla.set_look_at(Vector3(),Vector3(1, -1, 1));
- lla.basis = Basis::looking_at(Vector3(0.0, -0.836026, -0.548690));
-
- vs->instance_set_transform(light, lla);
-
- lightaux = vs->omni_light_create();
- //vs->light_set_color( lightaux, RenderingServer::LIGHT_COLOR_AMBIENT, Color(0.0,0.0,1.0) );
- vs->light_set_color(lightaux, Color(1.0, 1.0, 0.0));
- vs->light_set_param(lightaux, RenderingServer::LIGHT_PARAM_RANGE, 4);
- vs->light_set_param(lightaux, RenderingServer::LIGHT_PARAM_ENERGY, 8);
- //vs->light_set_shadow( lightaux, true );
- //light = vs->instance_create( lightaux );
-
- ofs = 0;
- quit = false;
- }
- virtual bool iteration(double p_time) {
- RenderingServer *vs = RenderingServer::get_singleton();
- //Transform3D t;
- //t.rotate(Vector3(0, 1, 0), ofs);
- //t.translate(Vector3(0,0,20 ));
- //vs->camera_set_transform(camera, t);
-
- ofs += p_time * 0.05;
-
- //return quit;
-
- for (const InstanceInfo &E : instances) {
- Transform3D pre(Basis(E.rot_axis, ofs), Vector3());
- vs->instance_set_transform(E.instance, pre * E.base);
- /*
- if( !E->next() ) {
- vs->free( E.instance );
- instances.erase(E );
- }*/
- }
-
- return quit;
- }
-
- virtual bool idle(double p_time) {
- return quit;
- }
-
- virtual void finish() {
- }
-};
-
-MainLoop *test() {
- return memnew(TestMainLoop);
-}
-} // namespace TestRender
diff --git a/tests/servers/test_render.h b/tests/servers/test_render.h
deleted file mode 100644
index d5a3e01ee5..0000000000
--- a/tests/servers/test_render.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_render.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_RENDER_H
-#define TEST_RENDER_H
-
-class MainLoop;
-
-namespace TestRender {
-
-MainLoop *test();
-}
-
-#endif // TEST_RENDER_H
diff --git a/tests/servers/test_shader_lang.cpp b/tests/servers/test_shader_lang.cpp
deleted file mode 100644
index 06e28212d2..0000000000
--- a/tests/servers/test_shader_lang.cpp
+++ /dev/null
@@ -1,364 +0,0 @@
-/*************************************************************************/
-/* test_shader_lang.cpp */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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. */
-/*************************************************************************/
-
-#include "test_shader_lang.h"
-
-#include "core/os/main_loop.h"
-#include "core/os/os.h"
-#include "servers/rendering/shader_language.h"
-
-typedef ShaderLanguage SL;
-
-namespace TestShaderLang {
-
-static String _mktab(int p_level) {
- String tb;
- for (int i = 0; i < p_level; i++) {
- tb += "\t";
- }
-
- return tb;
-}
-
-static String _typestr(SL::DataType p_type) {
- return ShaderLanguage::get_datatype_name(p_type);
-}
-
-static String _prestr(SL::DataPrecision p_pres) {
- switch (p_pres) {
- case SL::PRECISION_LOWP:
- return "lowp ";
- case SL::PRECISION_MEDIUMP:
- return "mediump ";
- case SL::PRECISION_HIGHP:
- return "highp ";
- case SL::PRECISION_DEFAULT:
- return "";
- }
- return "";
-}
-
-static String _opstr(SL::Operator p_op) {
- return ShaderLanguage::get_operator_text(p_op);
-}
-
-static String get_constant_text(SL::DataType p_type, const Vector<SL::ConstantNode::Value> &p_values) {
- switch (p_type) {
- case SL::TYPE_BOOL:
- return p_values[0].boolean ? "true" : "false";
- case SL::TYPE_BVEC2:
- return String() + "bvec2(" + (p_values[0].boolean ? "true" : "false") + (p_values[1].boolean ? "true" : "false") + ")";
- case SL::TYPE_BVEC3:
- return String() + "bvec3(" + (p_values[0].boolean ? "true" : "false") + "," + (p_values[1].boolean ? "true" : "false") + "," + (p_values[2].boolean ? "true" : "false") + ")";
- case SL::TYPE_BVEC4:
- return String() + "bvec4(" + (p_values[0].boolean ? "true" : "false") + "," + (p_values[1].boolean ? "true" : "false") + "," + (p_values[2].boolean ? "true" : "false") + "," + (p_values[3].boolean ? "true" : "false") + ")";
- case SL::TYPE_INT:
- return rtos(p_values[0].sint);
- case SL::TYPE_IVEC2:
- return String() + "ivec2(" + rtos(p_values[0].sint) + "," + rtos(p_values[1].sint) + ")";
- case SL::TYPE_IVEC3:
- return String() + "ivec3(" + rtos(p_values[0].sint) + "," + rtos(p_values[1].sint) + "," + rtos(p_values[2].sint) + ")";
- case SL::TYPE_IVEC4:
- return String() + "ivec4(" + rtos(p_values[0].sint) + "," + rtos(p_values[1].sint) + "," + rtos(p_values[2].sint) + "," + rtos(p_values[3].sint) + ")";
- case SL::TYPE_UINT:
- return rtos(p_values[0].real);
- case SL::TYPE_UVEC2:
- return String() + "uvec2(" + rtos(p_values[0].real) + "," + rtos(p_values[1].real) + ")";
- case SL::TYPE_UVEC3:
- return String() + "uvec3(" + rtos(p_values[0].real) + "," + rtos(p_values[1].real) + "," + rtos(p_values[2].real) + ")";
- case SL::TYPE_UVEC4:
- return String() + "uvec4(" + rtos(p_values[0].real) + "," + rtos(p_values[1].real) + "," + rtos(p_values[2].real) + "," + rtos(p_values[3].real) + ")";
- case SL::TYPE_FLOAT:
- return rtos(p_values[0].real);
- case SL::TYPE_VEC2:
- return String() + "vec2(" + rtos(p_values[0].real) + "," + rtos(p_values[1].real) + ")";
- case SL::TYPE_VEC3:
- return String() + "vec3(" + rtos(p_values[0].real) + "," + rtos(p_values[1].real) + "," + rtos(p_values[2].real) + ")";
- case SL::TYPE_VEC4:
- return String() + "vec4(" + rtos(p_values[0].real) + "," + rtos(p_values[1].real) + "," + rtos(p_values[2].real) + "," + rtos(p_values[3].real) + ")";
- default:
- ERR_FAIL_V(String());
- }
-}
-
-static String dump_node_code(SL::Node *p_node, int p_level) {
- String code;
-
- switch (p_node->type) {
- case SL::Node::TYPE_SHADER: {
- SL::ShaderNode *pnode = (SL::ShaderNode *)p_node;
-
- for (const KeyValue<StringName, SL::ShaderNode::Uniform> &E : pnode->uniforms) {
- String ucode = "uniform ";
- ucode += _prestr(E.value.precision);
- ucode += _typestr(E.value.type);
- ucode += " " + String(E.key);
- if (E.value.array_size > 0) {
- ucode += "[";
- ucode += itos(E.value.array_size);
- ucode += "]";
- } else {
- if (E.value.default_value.size()) {
- ucode += " = " + get_constant_text(E.value.type, E.value.default_value);
- }
-
- static const char *hint_name[SL::ShaderNode::Uniform::HINT_MAX] = {
- "",
- "color",
- "range",
- "albedo",
- "normal",
- "black",
- "white"
- };
-
- if (E.value.hint) {
- ucode += " : " + String(hint_name[E.value.hint]);
- }
- }
-
- code += ucode + "\n";
- }
-
- for (const KeyValue<StringName, SL::ShaderNode::Varying> &E : pnode->varyings) {
- String vcode = "varying ";
- vcode += _prestr(E.value.precision);
- vcode += _typestr(E.value.type);
- vcode += " " + String(E.key);
-
- code += vcode + "\n";
- }
- for (int i = 0; i < pnode->functions.size(); i++) {
- SL::FunctionNode *fnode = pnode->functions[i].function;
-
- String header;
- header = _typestr(fnode->return_type) + " " + fnode->name + "(";
- for (int j = 0; j < fnode->arguments.size(); j++) {
- if (j > 0) {
- header += ", ";
- }
- header += _prestr(fnode->arguments[j].precision) + _typestr(fnode->arguments[j].type) + " " + fnode->arguments[j].name;
- }
-
- header += ")\n";
- code += header;
- code += dump_node_code(fnode->body, p_level + 1);
- }
-
- //code+=dump_node_code(pnode->body,p_level);
- } break;
- case SL::Node::TYPE_STRUCT: {
- } break;
- case SL::Node::TYPE_FUNCTION: {
- } break;
- case SL::Node::TYPE_BLOCK: {
- SL::BlockNode *bnode = (SL::BlockNode *)p_node;
-
- //variables
- code += _mktab(p_level - 1) + "{\n";
- for (const KeyValue<StringName, SL::BlockNode::Variable> &E : bnode->variables) {
- code += _mktab(p_level) + _prestr(E.value.precision) + _typestr(E.value.type) + " " + E.key + ";\n";
- }
-
- for (int i = 0; i < bnode->statements.size(); i++) {
- String scode = dump_node_code(bnode->statements[i], p_level);
-
- if (bnode->statements[i]->type == SL::Node::TYPE_CONTROL_FLOW) {
- code += scode; //use directly
- } else {
- code += _mktab(p_level) + scode + ";\n";
- }
- }
- code += _mktab(p_level - 1) + "}\n";
-
- } break;
- case SL::Node::TYPE_VARIABLE: {
- SL::VariableNode *vnode = (SL::VariableNode *)p_node;
- code = vnode->name;
-
- } break;
- case SL::Node::TYPE_VARIABLE_DECLARATION: {
- // FIXME: Implement
- } break;
- case SL::Node::TYPE_ARRAY: {
- SL::ArrayNode *vnode = (SL::ArrayNode *)p_node;
- code = vnode->name;
- } break;
- case SL::Node::TYPE_ARRAY_CONSTRUCT: {
- // FIXME: Implement
- } break;
- case SL::Node::TYPE_CONSTANT: {
- SL::ConstantNode *cnode = (SL::ConstantNode *)p_node;
- return get_constant_text(cnode->datatype, cnode->values);
-
- } break;
- case SL::Node::TYPE_OPERATOR: {
- SL::OperatorNode *onode = (SL::OperatorNode *)p_node;
-
- switch (onode->op) {
- case SL::OP_ASSIGN:
- case SL::OP_ASSIGN_ADD:
- case SL::OP_ASSIGN_SUB:
- case SL::OP_ASSIGN_MUL:
- case SL::OP_ASSIGN_DIV:
- case SL::OP_ASSIGN_SHIFT_LEFT:
- case SL::OP_ASSIGN_SHIFT_RIGHT:
- case SL::OP_ASSIGN_MOD:
- case SL::OP_ASSIGN_BIT_AND:
- case SL::OP_ASSIGN_BIT_OR:
- case SL::OP_ASSIGN_BIT_XOR:
- code = dump_node_code(onode->arguments[0], p_level) + _opstr(onode->op) + dump_node_code(onode->arguments[1], p_level);
- break;
- case SL::OP_BIT_INVERT:
- case SL::OP_NEGATE:
- case SL::OP_NOT:
- case SL::OP_DECREMENT:
- case SL::OP_INCREMENT:
- code = _opstr(onode->op) + dump_node_code(onode->arguments[0], p_level);
- break;
- case SL::OP_POST_DECREMENT:
- case SL::OP_POST_INCREMENT:
- code = dump_node_code(onode->arguments[0], p_level) + _opstr(onode->op);
- break;
- case SL::OP_CALL:
- case SL::OP_CONSTRUCT:
- code = dump_node_code(onode->arguments[0], p_level) + "(";
- for (int i = 1; i < onode->arguments.size(); i++) {
- if (i > 1) {
- code += ", ";
- }
- code += dump_node_code(onode->arguments[i], p_level);
- }
- code += ")";
- break;
- case SL::OP_EMPTY:
- break;
- default: {
- code = "(" + dump_node_code(onode->arguments[0], p_level) + _opstr(onode->op) + dump_node_code(onode->arguments[1], p_level) + ")";
- break;
- }
- }
-
- } break;
- case SL::Node::TYPE_CONTROL_FLOW: {
- SL::ControlFlowNode *cfnode = (SL::ControlFlowNode *)p_node;
- if (cfnode->flow_op == SL::FLOW_OP_IF) {
- code += _mktab(p_level) + "if (" + dump_node_code(cfnode->expressions[0], p_level) + ")\n";
- code += dump_node_code(cfnode->blocks[0], p_level + 1);
- if (cfnode->blocks.size() == 2) {
- code += _mktab(p_level) + "else\n";
- code += dump_node_code(cfnode->blocks[1], p_level + 1);
- }
-
- } else if (cfnode->flow_op == SL::FLOW_OP_RETURN) {
- if (cfnode->blocks.size()) {
- code = "return " + dump_node_code(cfnode->blocks[0], p_level);
- } else {
- code = "return";
- }
- }
-
- } break;
- case SL::Node::TYPE_MEMBER: {
- SL::MemberNode *mnode = (SL::MemberNode *)p_node;
- code = dump_node_code(mnode->owner, p_level) + "." + mnode->name;
-
- } break;
- }
-
- return code;
-}
-
-static Error recreate_code(void *p_str, SL::ShaderNode *p_program) {
- String *str = (String *)p_str;
-
- *str = dump_node_code(p_program, 0);
-
- return OK;
-}
-
-MainLoop *test() {
- List<String> cmdlargs = OS::get_singleton()->get_cmdline_args();
-
- if (cmdlargs.is_empty()) {
- //try editor!
- print_line("usage: godot -test shader_lang <shader>");
- return nullptr;
- }
-
- String test = cmdlargs.back()->get();
-
- FileAccess *fa = FileAccess::open(test, FileAccess::READ);
-
- if (!fa) {
- ERR_FAIL_V(nullptr);
- }
-
- String code;
-
- while (true) {
- char32_t c = fa->get_8();
- if (fa->eof_reached()) {
- break;
- }
- code += c;
- }
-
- SL sl;
- print_line("tokens:\n\n" + sl.token_debug(code));
-
- Map<StringName, SL::FunctionInfo> dt;
- dt["fragment"].built_ins["ALBEDO"] = SL::TYPE_VEC3;
- dt["fragment"].can_discard = true;
-
- Vector<SL::ModeInfo> rm;
- rm.push_back({ "popo" });
- Set<String> types;
- types.insert("spatial");
-
- ShaderLanguage::ShaderCompileInfo info;
- info.functions = dt;
- info.render_modes = rm;
- info.shader_types = types;
-
- Error err = sl.compile(code, info);
-
- if (err) {
- print_line("Error at line: " + rtos(sl.get_error_line()) + ": " + sl.get_error_text());
- return nullptr;
- } else {
- String code2;
- recreate_code(&code2, sl.get_shader());
- print_line("code:\n\n" + code2);
- }
-
- return nullptr;
-}
-} // namespace TestShaderLang
diff --git a/tests/servers/test_shader_lang.h b/tests/servers/test_shader_lang.h
deleted file mode 100644
index 31e1bfbeea..0000000000
--- a/tests/servers/test_shader_lang.h
+++ /dev/null
@@ -1,41 +0,0 @@
-/*************************************************************************/
-/* test_shader_lang.h */
-/*************************************************************************/
-/* This file is part of: */
-/* GODOT ENGINE */
-/* https://godotengine.org */
-/*************************************************************************/
-/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
-/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
-/* */
-/* 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_SHADER_LANG_H
-#define TEST_SHADER_LANG_H
-
-class MainLoop;
-
-namespace TestShaderLang {
-
-MainLoop *test();
-}
-
-#endif // TEST_SHADER_LANG_H
diff --git a/tests/servers/test_text_server.h b/tests/servers/test_text_server.h
index 0a64237285..d7de94516f 100644
--- a/tests/servers/test_text_server.h
+++ b/tests/servers/test_text_server.h
@@ -46,10 +46,14 @@ TEST_SUITE("[[TextServer]") {
Ref<TextServer> ts = TextServerManager::get_singleton()->get_interface(i);
TEST_FAIL_COND(ts.is_null(), "Invalid TS interface.");
+ if (!ts->has_feature(TextServer::FEATURE_FONT_DYNAMIC)) {
+ continue;
+ }
+
RID font = ts->create_font();
ts->font_set_data_ptr(font, _font_NotoSans_Regular, _font_NotoSans_Regular_size);
TEST_FAIL_COND(font == RID(), "Loading font failed.");
- ts->free(font);
+ ts->free_rid(font);
}
}
@@ -58,12 +62,16 @@ TEST_SUITE("[[TextServer]") {
Ref<TextServer> ts = TextServerManager::get_singleton()->get_interface(i);
TEST_FAIL_COND(ts.is_null(), "Invalid TS interface.");
+ if (!ts->has_feature(TextServer::FEATURE_FONT_DYNAMIC) || !ts->has_feature(TextServer::FEATURE_SIMPLE_LAYOUT)) {
+ continue;
+ }
+
RID font1 = ts->create_font();
ts->font_set_data_ptr(font1, _font_NotoSans_Regular, _font_NotoSans_Regular_size);
RID font2 = ts->create_font();
ts->font_set_data_ptr(font2, _font_NotoSansThaiUI_Regular, _font_NotoSansThaiUI_Regular_size);
- Vector<RID> font;
+ Array font;
font.push_back(font1);
font.push_back(font2);
@@ -93,10 +101,10 @@ TEST_SUITE("[[TextServer]") {
TEST_FAIL_COND(glyphs[j].font_size != 16, "Incorrect glyph font size.");
}
- ts->free(ctx);
+ ts->free_rid(ctx);
for (int j = 0; j < font.size(); j++) {
- ts->free(font[j]);
+ ts->free_rid(font[j]);
}
font.clear();
}
@@ -107,7 +115,7 @@ TEST_SUITE("[[TextServer]") {
Ref<TextServer> ts = TextServerManager::get_singleton()->get_interface(i);
TEST_FAIL_COND(ts.is_null(), "Invalid TS interface.");
- if (!ts->has_feature(TextServer::FEATURE_BIDI_LAYOUT)) {
+ if (!ts->has_feature(TextServer::FEATURE_FONT_DYNAMIC) || !ts->has_feature(TextServer::FEATURE_BIDI_LAYOUT)) {
continue;
}
@@ -116,7 +124,7 @@ TEST_SUITE("[[TextServer]") {
RID font2 = ts->create_font();
ts->font_set_data_ptr(font2, _font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size);
- Vector<RID> font;
+ Array font;
font.push_back(font1);
font.push_back(font2);
@@ -145,10 +153,10 @@ TEST_SUITE("[[TextServer]") {
}
}
- ts->free(ctx);
+ ts->free_rid(ctx);
for (int j = 0; j < font.size(); j++) {
- ts->free(font[j]);
+ ts->free_rid(font[j]);
}
font.clear();
}
@@ -159,6 +167,10 @@ TEST_SUITE("[[TextServer]") {
Ref<TextServer> ts = TextServerManager::get_singleton()->get_interface(i);
TEST_FAIL_COND(ts.is_null(), "Invalid TS interface.");
+ if (!ts->has_feature(TextServer::FEATURE_FONT_DYNAMIC) || !ts->has_feature(TextServer::FEATURE_SIMPLE_LAYOUT)) {
+ continue;
+ }
+
RID font1 = ts->create_font();
ts->font_set_data_ptr(font1, _font_NotoSans_Regular, _font_NotoSans_Regular_size);
RID font2 = ts->create_font();
@@ -166,7 +178,7 @@ TEST_SUITE("[[TextServer]") {
RID font3 = ts->create_font();
ts->font_set_data_ptr(font3, _font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size);
- Vector<RID> font;
+ Array font;
font.push_back(font1);
font.push_back(font2);
font.push_back(font3);
@@ -198,7 +210,7 @@ TEST_SUITE("[[TextServer]") {
TEST_FAIL_COND((soft || space || hard || virt || elo), "Invalid glyph flags.");
}
}
- ts->free(ctx);
+ ts->free_rid(ctx);
}
{
@@ -241,7 +253,7 @@ TEST_SUITE("[[TextServer]") {
}
}
}
- ts->free(ctx);
+ ts->free_rid(ctx);
}
{
@@ -294,7 +306,7 @@ TEST_SUITE("[[TextServer]") {
}
}
- ts->free(ctx);
+ ts->free_rid(ctx);
}
{
@@ -322,7 +334,7 @@ TEST_SUITE("[[TextServer]") {
TEST_FAIL_COND((soft || space || hard || virt || elo), "Invalid glyph flags.");
}
}
- ts->free(ctx);
+ ts->free_rid(ctx);
}
if (ts->has_feature(TextServer::FEATURE_BREAK_ITERATORS)) {
@@ -350,11 +362,11 @@ TEST_SUITE("[[TextServer]") {
TEST_FAIL_COND((soft || space || hard || virt || elo), "Invalid glyph flags.");
}
}
- ts->free(ctx);
+ ts->free_rid(ctx);
}
for (int j = 0; j < font.size(); j++) {
- ts->free(font[j]);
+ ts->free_rid(font[j]);
}
font.clear();
}
@@ -365,6 +377,10 @@ TEST_SUITE("[[TextServer]") {
Ref<TextServer> ts = TextServerManager::get_singleton()->get_interface(i);
TEST_FAIL_COND(ts.is_null(), "Invalid TS interface.");
+ if (!ts->has_feature(TextServer::FEATURE_FONT_DYNAMIC) || !ts->has_feature(TextServer::FEATURE_SIMPLE_LAYOUT)) {
+ continue;
+ }
+
String test_1 = U"test test test";
// 5^ 10^
@@ -373,7 +389,7 @@ TEST_SUITE("[[TextServer]") {
RID font2 = ts->create_font();
ts->font_set_data_ptr(font2, _font_NotoSansThaiUI_Regular, _font_NotoSansThaiUI_Regular_size);
- Vector<RID> font;
+ Array font;
font.push_back(font1);
font.push_back(font2);
@@ -395,10 +411,10 @@ TEST_SUITE("[[TextServer]") {
TEST_FAIL_COND(brks[5] != 14, "Invalid line break position.");
}
- ts->free(ctx);
+ ts->free_rid(ctx);
for (int j = 0; j < font.size(); j++) {
- ts->free(font[j]);
+ ts->free_rid(font[j]);
}
font.clear();
}
@@ -409,12 +425,16 @@ TEST_SUITE("[[TextServer]") {
Ref<TextServer> ts = TextServerManager::get_singleton()->get_interface(i);
TEST_FAIL_COND(ts.is_null(), "Invalid TS interface.");
+ if (!ts->has_feature(TextServer::FEATURE_FONT_DYNAMIC) || !ts->has_feature(TextServer::FEATURE_SIMPLE_LAYOUT)) {
+ continue;
+ }
+
RID font1 = ts->create_font();
ts->font_set_data_ptr(font1, _font_NotoSans_Regular, _font_NotoSans_Regular_size);
RID font2 = ts->create_font();
ts->font_set_data_ptr(font2, _font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size);
- Vector<RID> font;
+ Array font;
font.push_back(font1);
font.push_back(font2);
@@ -438,7 +458,7 @@ TEST_SUITE("[[TextServer]") {
width = ts->shaped_text_fit_to_width(ctx, 100, TextServer::JUSTIFICATION_WORD_BOUND | TextServer::JUSTIFICATION_KASHIDA);
TEST_FAIL_COND((width <= width_old || width > 100), "Invalid fill width.");
- ts->free(ctx);
+ ts->free_rid(ctx);
ctx = ts->create_shaped_text();
TEST_FAIL_COND(ctx == RID(), "Creating text buffer failed.");
@@ -451,7 +471,7 @@ TEST_SUITE("[[TextServer]") {
width = ts->shaped_text_fit_to_width(ctx, 100, TextServer::JUSTIFICATION_WORD_BOUND | TextServer::JUSTIFICATION_KASHIDA);
TEST_FAIL_COND((width <= width_old || width > 100), "Invalid fill width.");
- ts->free(ctx);
+ ts->free_rid(ctx);
}
ctx = ts->create_shaped_text();
@@ -463,10 +483,10 @@ TEST_SUITE("[[TextServer]") {
width = ts->shaped_text_fit_to_width(ctx, 100, TextServer::JUSTIFICATION_WORD_BOUND);
TEST_FAIL_COND((width <= width_old || width > 100), "Invalid fill width.");
- ts->free(ctx);
+ ts->free_rid(ctx);
for (int j = 0; j < font.size(); j++) {
- ts->free(font[j]);
+ ts->free_rid(font[j]);
}
font.clear();
}
diff --git a/tests/test_macros.h b/tests/test_macros.h
index ed8a12f155..6e7a84bfb2 100644
--- a/tests/test_macros.h
+++ b/tests/test_macros.h
@@ -134,8 +134,10 @@ int register_test_command(String p_command, TestFunc p_function);
// Requires Message Queue and InputMap to be setup.
// SEND_GUI_ACTION - takes an object and a input map key. e.g SEND_GUI_ACTION(code_edit, "ui_text_newline").
// SEND_GUI_KEY_EVENT - takes an object and a keycode set. e.g SEND_GUI_KEY_EVENT(code_edit, Key::A | KeyModifierMask::CMD).
-// SEND_GUI_MOUSE_EVENT - takes an object, position, mouse button and mouse mask e.g SEND_GUI_MOUSE_EVENT(code_edit, Vector2(50, 50), MOUSE_BUTTON_NONE, MOUSE_BUTTON_NONE);
-// SEND_GUI_DOUBLE_CLICK - takes an object and a position. e.g SEND_GUI_DOUBLE_CLICK(code_edit, Vector2(50, 50));
+// SEND_GUI_MOUSE_BUTTON_EVENT - takes an object, position, mouse button, mouse mask and modifiers e.g SEND_GUI_MOUSE_BUTTON_EVENT(code_edit, Vector2(50, 50), MOUSE_BUTTON_NONE, MOUSE_BUTTON_NONE, Key::None);
+// SEND_GUI_MOUSE_BUTTON_RELEASED_EVENT - takes an object, position, mouse button, mouse mask and modifiers e.g SEND_GUI_MOUSE_BUTTON_RELEASED_EVENT(code_edit, Vector2(50, 50), MOUSE_BUTTON_NONE, MOUSE_BUTTON_NONE, Key::None);
+// SEND_GUI_MOUSE_MOTION_EVENT - takes an object, position, mouse mask and modifiers e.g SEND_GUI_MOUSE_MOTION_EVENT(code_edit, Vector2(50, 50), MouseButton::MASK_LEFT, KeyModifierMask::CMD);
+// SEND_GUI_DOUBLE_CLICK - takes an object, position and modifiers. e.g SEND_GUI_DOUBLE_CLICK(code_edit, Vector2(50, 50), KeyModifierMask::CMD);
#define SEND_GUI_ACTION(m_object, m_action) \
{ \
@@ -143,7 +145,7 @@ int register_test_command(String p_command, TestFunc p_function);
const List<Ref<InputEvent>>::Element *first_event = events->front(); \
Ref<InputEventKey> event = first_event->get(); \
event->set_pressed(true); \
- m_object->gui_input(event); \
+ m_object->get_viewport()->push_input(event); \
MessageQueue::get_singleton()->flush(); \
}
@@ -151,31 +153,64 @@ int register_test_command(String p_command, TestFunc p_function);
{ \
Ref<InputEventKey> event = InputEventKey::create_reference(m_input); \
event->set_pressed(true); \
- m_object->gui_input(event); \
+ m_object->get_viewport()->push_input(event); \
MessageQueue::get_singleton()->flush(); \
}
-#define _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, m_input, m_mask) \
- Ref<InputEventMouseButton> event; \
- event.instantiate(); \
- event->set_position(m_local_pos); \
- event->set_button_index(m_input); \
- event->set_button_mask(m_mask); \
+#define _UPDATE_EVENT_MODIFERS(m_event, m_modifers) \
+ m_event->set_shift_pressed(((m_modifers)&KeyModifierMask::SHIFT) != Key::NONE); \
+ m_event->set_alt_pressed(((m_modifers)&KeyModifierMask::ALT) != Key::NONE); \
+ m_event->set_ctrl_pressed(((m_modifers)&KeyModifierMask::CTRL) != Key::NONE); \
+ m_event->set_command_pressed(((m_modifers)&KeyModifierMask::CMD) != Key::NONE); \
+ m_event->set_meta_pressed(((m_modifers)&KeyModifierMask::META) != Key::NONE);
+
+#define _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, m_input, m_mask, m_modifers) \
+ Ref<InputEventMouseButton> event; \
+ event.instantiate(); \
+ event->set_position(m_local_pos); \
+ event->set_button_index(m_input); \
+ event->set_button_mask(m_mask); \
+ event->set_factor(1); \
+ _UPDATE_EVENT_MODIFERS(event, m_modifers); \
event->set_pressed(true);
-#define SEND_GUI_MOUSE_EVENT(m_object, m_local_pos, m_input, m_mask) \
- { \
- _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, m_input, m_mask); \
- m_object->get_viewport()->push_input(event); \
- MessageQueue::get_singleton()->flush(); \
+#define SEND_GUI_MOUSE_BUTTON_EVENT(m_object, m_local_pos, m_input, m_mask, m_modifers) \
+ { \
+ _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, m_input, m_mask, m_modifers); \
+ m_object->get_viewport()->push_input(event); \
+ MessageQueue::get_singleton()->flush(); \
}
-#define SEND_GUI_DOUBLE_CLICK(m_object, m_local_pos) \
- { \
- _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, MouseButton::LEFT, MouseButton::LEFT); \
- event->set_double_click(true); \
- m_object->get_viewport()->push_input(event); \
- MessageQueue::get_singleton()->flush(); \
+#define SEND_GUI_MOUSE_BUTTON_RELEASED_EVENT(m_object, m_local_pos, m_input, m_mask, m_modifers) \
+ { \
+ _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, m_input, m_mask, m_modifers); \
+ event->set_pressed(false); \
+ m_object->get_viewport()->push_input(event); \
+ MessageQueue::get_singleton()->flush(); \
+ }
+
+#define SEND_GUI_DOUBLE_CLICK(m_object, m_local_pos, m_modifers) \
+ { \
+ _CREATE_GUI_MOUSE_EVENT(m_object, m_local_pos, MouseButton::LEFT, MouseButton::LEFT, m_modifers); \
+ event->set_double_click(true); \
+ m_object->get_viewport()->push_input(event); \
+ MessageQueue::get_singleton()->flush(); \
+ }
+
+// We toogle _print_error_enabled to prevent display server not supported warnings.
+#define SEND_GUI_MOUSE_MOTION_EVENT(m_object, m_local_pos, m_mask, m_modifers) \
+ { \
+ bool errors_enabled = _print_error_enabled; \
+ _print_error_enabled = false; \
+ Ref<InputEventMouseMotion> event; \
+ event.instantiate(); \
+ event->set_position(m_local_pos); \
+ event->set_button_mask(m_mask); \
+ event->set_relative(Vector2(10, 10)); \
+ _UPDATE_EVENT_MODIFERS(event, m_modifers); \
+ m_object->get_viewport()->push_input(event); \
+ MessageQueue::get_singleton()->flush(); \
+ _print_error_enabled = errors_enabled; \
}
// Utility class / macros for testing signals
diff --git a/tests/test_main.cpp b/tests/test_main.cpp
index 830731abcd..8954b8332e 100644
--- a/tests/test_main.cpp
+++ b/tests/test_main.cpp
@@ -45,7 +45,6 @@
#include "tests/core/math/test_expression.h"
#include "tests/core/math/test_geometry_2d.h"
#include "tests/core/math/test_geometry_3d.h"
-#include "tests/core/math/test_math.h"
#include "tests/core/math/test_random_number_generator.h"
#include "tests/core/math/test_rect2.h"
#include "tests/core/math/test_rect2i.h"
@@ -63,7 +62,6 @@
#include "tests/core/templates/test_list.h"
#include "tests/core/templates/test_local_vector.h"
#include "tests/core/templates/test_lru.h"
-#include "tests/core/templates/test_oa_hash_map.h"
#include "tests/core/templates/test_ordered_hash_map.h"
#include "tests/core/templates/test_paged_array.h"
#include "tests/core/templates/test_vector.h"
@@ -77,12 +75,8 @@
#include "tests/scene/test_code_edit.h"
#include "tests/scene/test_curve.h"
#include "tests/scene/test_gradient.h"
-#include "tests/scene/test_gui.h"
#include "tests/scene/test_path_3d.h"
-#include "tests/servers/test_physics_2d.h"
-#include "tests/servers/test_physics_3d.h"
-#include "tests/servers/test_render.h"
-#include "tests/servers/test_shader_lang.h"
+#include "tests/scene/test_text_edit.h"
#include "tests/servers/test_text_server.h"
#include "tests/test_validate_testing.h"
@@ -182,6 +176,8 @@ struct GodotTestCaseListener : public doctest::IReporter {
GLOBAL_DEF("internationalization/rendering/force_right_to_left_layout_direction", false);
+ memnew(Input);
+
Error err = OK;
OS::get_singleton()->set_has_server_feature_callback(nullptr);
for (int i = 0; i < DisplayServer::get_create_function_count(); i++) {
@@ -206,7 +202,7 @@ struct GodotTestCaseListener : public doctest::IReporter {
memnew(InputMap);
InputMap::get_singleton()->load_default();
- make_default_theme(1.0, Ref<Font>(), TextServer::SUBPIXEL_POSITIONING_AUTO, TextServer::HINTING_LIGHT, true);
+ make_default_theme(1.0, Ref<Font>());
memnew(SceneTree);
SceneTree::get_singleton()->initialize();
@@ -251,6 +247,10 @@ struct GodotTestCaseListener : public doctest::IReporter {
physics_2d_server = nullptr;
}
+ if (Input::get_singleton()) {
+ memdelete(Input::get_singleton());
+ }
+
if (RenderingServer::get_singleton()) {
RenderingServer::get_singleton()->sync();
RenderingServer::get_singleton()->global_variables_clear();