| 
									
										
										
										
											2021-05-17 02:02:17 +02:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*  test_translation.h                                                    */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*                         This file is part of:                          */ | 
					
						
							|  |  |  | /*                             GODOT ENGINE                               */ | 
					
						
							|  |  |  | /*                        https://godotengine.org                         */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | 
					
						
							|  |  |  | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* Permission is hereby granted, free of charge, to any person obtaining  */ | 
					
						
							|  |  |  | /* a copy of this software and associated documentation files (the        */ | 
					
						
							|  |  |  | /* "Software"), to deal in the Software without restriction, including    */ | 
					
						
							|  |  |  | /* without limitation the rights to use, copy, modify, merge, publish,    */ | 
					
						
							|  |  |  | /* distribute, sublicense, and/or sell copies of the Software, and to     */ | 
					
						
							|  |  |  | /* permit persons to whom the Software is furnished to do so, subject to  */ | 
					
						
							|  |  |  | /* the following conditions:                                              */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* The above copyright notice and this permission notice shall be         */ | 
					
						
							|  |  |  | /* included in all copies or substantial portions of the Software.        */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,        */ | 
					
						
							|  |  |  | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF     */ | 
					
						
							|  |  |  | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ | 
					
						
							|  |  |  | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY   */ | 
					
						
							|  |  |  | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,   */ | 
					
						
							|  |  |  | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE      */ | 
					
						
							|  |  |  | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                 */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef TEST_TRANSLATION_H
 | 
					
						
							|  |  |  | #define TEST_TRANSLATION_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "core/string/optimized_translation.h"
 | 
					
						
							|  |  |  | #include "core/string/translation.h"
 | 
					
						
							|  |  |  | #include "core/string/translation_po.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | #ifdef TOOLS_ENABLED
 | 
					
						
							|  |  |  | #include "editor/import/resource_importer_csv_translation.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-06 16:05:33 -05:00
										 |  |  | #include "tests/test_macros.h"
 | 
					
						
							|  |  |  | #include "tests/test_utils.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-17 02:02:17 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace TestTranslation { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST_CASE("[Translation] Messages") { | 
					
						
							|  |  |  | 	Ref<Translation> translation = memnew(Translation); | 
					
						
							|  |  |  | 	translation->set_locale("fr"); | 
					
						
							|  |  |  | 	translation->add_message("Hello", "Bonjour"); | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello") == "Bonjour"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	translation->erase_message("Hello"); | 
					
						
							|  |  |  | 	// The message no longer exists, so it returns an empty string instead.
 | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello") == ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	List<StringName> messages; | 
					
						
							|  |  |  | 	translation->get_message_list(&messages); | 
					
						
							|  |  |  | 	CHECK(translation->get_message_count() == 0); | 
					
						
							|  |  |  | 	CHECK(messages.size() == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	translation->add_message("Hello2", "Bonjour2"); | 
					
						
							|  |  |  | 	translation->add_message("Hello3", "Bonjour3"); | 
					
						
							|  |  |  | 	messages.clear(); | 
					
						
							|  |  |  | 	translation->get_message_list(&messages); | 
					
						
							|  |  |  | 	CHECK(translation->get_message_count() == 2); | 
					
						
							|  |  |  | 	CHECK(messages.size() == 2); | 
					
						
							| 
									
										
										
										
											2021-05-17 12:29:24 +02:00
										 |  |  | 	// Messages are stored in a Map, don't assume ordering.
 | 
					
						
							|  |  |  | 	CHECK(messages.find("Hello2")); | 
					
						
							|  |  |  | 	CHECK(messages.find("Hello3")); | 
					
						
							| 
									
										
										
										
											2021-05-17 02:02:17 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST_CASE("[TranslationPO] Messages with context") { | 
					
						
							|  |  |  | 	Ref<TranslationPO> translation = memnew(TranslationPO); | 
					
						
							|  |  |  | 	translation->set_locale("fr"); | 
					
						
							|  |  |  | 	translation->add_message("Hello", "Bonjour"); | 
					
						
							|  |  |  | 	translation->add_message("Hello", "Salut", "friendly"); | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello") == "Bonjour"); | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello", "friendly") == "Salut"); | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello", "nonexistent_context") == ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Only remove the message for the default context, not the "friendly" context.
 | 
					
						
							|  |  |  | 	translation->erase_message("Hello"); | 
					
						
							|  |  |  | 	// The message no longer exists, so it returns an empty string instead.
 | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello") == ""); | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello", "friendly") == "Salut"); | 
					
						
							|  |  |  | 	CHECK(translation->get_message("Hello", "nonexistent_context") == ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	List<StringName> messages; | 
					
						
							|  |  |  | 	translation->get_message_list(&messages); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// `get_message_count()` takes all contexts into account.
 | 
					
						
							|  |  |  | 	CHECK(translation->get_message_count() == 1); | 
					
						
							|  |  |  | 	// Only the default context is taken into account.
 | 
					
						
							|  |  |  | 	// Since "Hello" is now only present in a non-default context, it is not counted in the list of messages.
 | 
					
						
							|  |  |  | 	CHECK(messages.size() == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	translation->add_message("Hello2", "Bonjour2"); | 
					
						
							|  |  |  | 	translation->add_message("Hello2", "Salut2", "friendly"); | 
					
						
							|  |  |  | 	translation->add_message("Hello3", "Bonjour3"); | 
					
						
							|  |  |  | 	messages.clear(); | 
					
						
							|  |  |  | 	translation->get_message_list(&messages); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// `get_message_count()` takes all contexts into account.
 | 
					
						
							|  |  |  | 	CHECK(translation->get_message_count() == 4); | 
					
						
							|  |  |  | 	// Only the default context is taken into account.
 | 
					
						
							|  |  |  | 	CHECK(messages.size() == 2); | 
					
						
							| 
									
										
										
										
											2021-05-17 12:29:24 +02:00
										 |  |  | 	// Messages are stored in a Map, don't assume ordering.
 | 
					
						
							|  |  |  | 	CHECK(messages.find("Hello2")); | 
					
						
							|  |  |  | 	CHECK(messages.find("Hello3")); | 
					
						
							| 
									
										
										
										
											2021-05-17 02:02:17 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST_CASE("[TranslationPO] Plural messages") { | 
					
						
							|  |  |  | 	Ref<TranslationPO> translation = memnew(TranslationPO); | 
					
						
							|  |  |  | 	translation->set_locale("fr"); | 
					
						
							|  |  |  | 	translation->set_plural_rule("Plural-Forms: nplurals=2; plural=(n >= 2);"); | 
					
						
							|  |  |  | 	CHECK(translation->get_plural_forms() == 2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PackedStringArray plurals; | 
					
						
							|  |  |  | 	plurals.push_back("Il y a %d pomme"); | 
					
						
							|  |  |  | 	plurals.push_back("Il y a %d pommes"); | 
					
						
							|  |  |  | 	translation->add_plural_message("There are %d apples", plurals); | 
					
						
							|  |  |  | 	ERR_PRINT_OFF; | 
					
						
							|  |  |  | 	// This is invalid, as the number passed to `get_plural_message()` may not be negative.
 | 
					
						
							|  |  |  | 	CHECK(vformat(translation->get_plural_message("There are %d apples", "", -1), -1) == ""); | 
					
						
							|  |  |  | 	ERR_PRINT_ON; | 
					
						
							|  |  |  | 	CHECK(vformat(translation->get_plural_message("There are %d apples", "", 0), 0) == "Il y a 0 pomme"); | 
					
						
							|  |  |  | 	CHECK(vformat(translation->get_plural_message("There are %d apples", "", 1), 1) == "Il y a 1 pomme"); | 
					
						
							|  |  |  | 	CHECK(vformat(translation->get_plural_message("There are %d apples", "", 2), 2) == "Il y a 2 pommes"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST_CASE("[OptimizedTranslation] Generate from Translation and read messages") { | 
					
						
							|  |  |  | 	Ref<Translation> translation = memnew(Translation); | 
					
						
							|  |  |  | 	translation->set_locale("fr"); | 
					
						
							|  |  |  | 	translation->add_message("Hello", "Bonjour"); | 
					
						
							|  |  |  | 	translation->add_message("Hello2", "Bonjour2"); | 
					
						
							|  |  |  | 	translation->add_message("Hello3", "Bonjour3"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<OptimizedTranslation> optimized_translation = memnew(OptimizedTranslation); | 
					
						
							|  |  |  | 	optimized_translation->generate(translation); | 
					
						
							|  |  |  | 	CHECK(optimized_translation->get_message("Hello") == "Bonjour"); | 
					
						
							|  |  |  | 	CHECK(optimized_translation->get_message("Hello2") == "Bonjour2"); | 
					
						
							|  |  |  | 	CHECK(optimized_translation->get_message("Hello3") == "Bonjour3"); | 
					
						
							|  |  |  | 	CHECK(optimized_translation->get_message("DoesNotExist") == ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	List<StringName> messages; | 
					
						
							|  |  |  | 	// `get_message_list()` can't return the list of messages stored in an OptimizedTranslation.
 | 
					
						
							|  |  |  | 	optimized_translation->get_message_list(&messages); | 
					
						
							|  |  |  | 	CHECK(optimized_translation->get_message_count() == 0); | 
					
						
							|  |  |  | 	CHECK(messages.size() == 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | #ifdef TOOLS_ENABLED
 | 
					
						
							| 
									
										
										
										
											2021-03-19 19:50:53 +01:00
										 |  |  | TEST_CASE("[TranslationCSV] CSV import") { | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | 	Ref<ResourceImporterCSVTranslation> import_csv_translation = memnew(ResourceImporterCSVTranslation); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 	HashMap<StringName, Variant> options; | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | 	options["compress"] = false; | 
					
						
							|  |  |  | 	options["delimiter"] = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	List<String> gen_files; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Error result = import_csv_translation->import(TestUtils::get_data_path("translations.csv"), | 
					
						
							|  |  |  | 			"", options, nullptr, &gen_files); | 
					
						
							|  |  |  | 	CHECK(result == OK); | 
					
						
							| 
									
										
										
										
											2021-03-19 19:50:53 +01:00
										 |  |  | 	CHECK(gen_files.size() == 4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TranslationServer *ts = TranslationServer::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (const String &file : gen_files) { | 
					
						
							|  |  |  | 		Ref<Translation> translation = ResourceLoader::load(file); | 
					
						
							|  |  |  | 		CHECK(translation.is_valid()); | 
					
						
							| 
									
										
										
										
											2021-03-19 19:50:53 +01:00
										 |  |  | 		ts->add_translation(translation); | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 19:50:53 +01:00
										 |  |  | 	ts->set_locale("en"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// `tr` can be called on any Object, we reuse TranslationServer for convenience.
 | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_MORNING") == "Good Morning"); | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_EVENING") == "Good Evening"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ts->set_locale("de"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_MORNING") == "Guten Morgen"); | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_EVENING") == "Good Evening"); // Left blank in CSV, should source from 'en'.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ts->set_locale("ja"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_MORNING") == String::utf8("おはよう")); | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_EVENING") == String::utf8("こんばんは")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* FIXME: This passes, but triggers a chain reaction that makes test_viewport
 | 
					
						
							|  |  |  | 	 * and test_text_edit explode in a billion glittery Unicode particles. | 
					
						
							|  |  |  | 	ts->set_locale("fa"); | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 19:50:53 +01:00
										 |  |  | 	CHECK(ts->tr("GOOD_MORNING") == String::utf8("صبح بخیر")); | 
					
						
							|  |  |  | 	CHECK(ts->tr("GOOD_EVENING") == String::utf8("عصر بخیر")); | 
					
						
							|  |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2021-09-29 10:38:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | #endif // TOOLS_ENABLED
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 02:02:17 +02:00
										 |  |  | } // namespace TestTranslation
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // TEST_TRANSLATION_H
 |