2020-06-23 19:48:59 +08:00
/*************************************************************************/
/* editor_translation_parser.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
2021-01-02 03:13:46 +08:00
/* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
2020-06-23 19:48:59 +08:00
/* */
/* 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 "editor_translation_parser.h"
2020-11-08 06:33:38 +08:00
# include "core/error/error_macros.h"
2021-06-11 20:51:48 +08:00
# include "core/io/file_access.h"
2020-11-08 06:33:38 +08:00
# include "core/object/script_language.h"
# include "core/templates/set.h"
2020-06-23 19:48:59 +08:00
EditorTranslationParser * EditorTranslationParser : : singleton = nullptr ;
2020-07-23 06:07:35 +08:00
Error EditorTranslationParserPlugin : : parse_file ( const String & p_path , Vector < String > * r_ids , Vector < Vector < String > > * r_ids_ctx_plural ) {
2021-04-05 20:09:59 +08:00
if ( ! get_script_instance ( ) ) {
2020-06-23 19:48:59 +08:00
return ERR_UNAVAILABLE ;
2021-04-05 20:09:59 +08:00
}
2020-06-23 19:48:59 +08:00
2020-07-04 02:24:54 +08:00
if ( get_script_instance ( ) - > has_method ( " parse_file " ) ) {
2020-07-23 06:07:35 +08:00
Array ids ;
Array ids_ctx_plural ;
get_script_instance ( ) - > call ( " parse_file " , p_path , ids , ids_ctx_plural ) ;
// Add user's extracted translatable messages.
for ( int i = 0 ; i < ids . size ( ) ; i + + ) {
r_ids - > append ( ids [ i ] ) ;
}
// Add user's collected translatable messages with context or plurals.
for ( int i = 0 ; i < ids_ctx_plural . size ( ) ; i + + ) {
Array arr = ids_ctx_plural [ i ] ;
2020-08-07 19:17:12 +08:00
ERR_FAIL_COND_V_MSG ( arr . size ( ) ! = 3 , ERR_INVALID_DATA , " Array entries written into `msgids_context_plural` in `parse_file()` method should have the form [ \" message \" , \" context \" , \" plural message \" ] " ) ;
2020-07-23 06:07:35 +08:00
Vector < String > id_ctx_plural ;
id_ctx_plural . push_back ( arr [ 0 ] ) ;
id_ctx_plural . push_back ( arr [ 1 ] ) ;
id_ctx_plural . push_back ( arr [ 2 ] ) ;
r_ids_ctx_plural - > append ( id_ctx_plural ) ;
2020-06-23 19:48:59 +08:00
}
2020-07-04 02:24:54 +08:00
return OK ;
2020-06-23 19:48:59 +08:00
} else {
2020-07-04 02:24:54 +08:00
ERR_PRINT ( " Custom translation parser plugin's \" func parse_file(path, extracted_strings) \" is undefined. " ) ;
return ERR_UNAVAILABLE ;
2020-06-23 19:48:59 +08:00
}
}
void EditorTranslationParserPlugin : : get_recognized_extensions ( List < String > * r_extensions ) const {
2021-04-05 20:09:59 +08:00
if ( ! get_script_instance ( ) ) {
2020-06-23 19:48:59 +08:00
return ;
2021-04-05 20:09:59 +08:00
}
2020-06-23 19:48:59 +08:00
if ( get_script_instance ( ) - > has_method ( " get_recognized_extensions " ) ) {
Array extensions = get_script_instance ( ) - > call ( " get_recognized_extensions " ) ;
for ( int i = 0 ; i < extensions . size ( ) ; i + + ) {
r_extensions - > push_back ( extensions [ i ] ) ;
}
} else {
ERR_PRINT ( " Custom translation parser plugin's \" func get_recognized_extensions() \" is undefined. " ) ;
}
}
void EditorTranslationParserPlugin : : _bind_methods ( ) {
2020-07-23 06:07:35 +08:00
ClassDB : : add_virtual_method ( get_class_static ( ) , MethodInfo ( Variant : : NIL , " parse_file " , PropertyInfo ( Variant : : STRING , " path " ) , PropertyInfo ( Variant : : ARRAY , " msgids " ) , PropertyInfo ( Variant : : ARRAY , " msgids_context_plural " ) ) ) ;
2020-06-23 19:48:59 +08:00
ClassDB : : add_virtual_method ( get_class_static ( ) , MethodInfo ( Variant : : ARRAY , " get_recognized_extensions " ) ) ;
}
/////////////////////////
void EditorTranslationParser : : get_recognized_extensions ( List < String > * r_extensions ) const {
Set < String > extensions ;
List < String > temp ;
for ( int i = 0 ; i < standard_parsers . size ( ) ; i + + ) {
standard_parsers [ i ] - > get_recognized_extensions ( & temp ) ;
}
for ( int i = 0 ; i < custom_parsers . size ( ) ; i + + ) {
custom_parsers [ i ] - > get_recognized_extensions ( & temp ) ;
}
// Remove duplicates.
for ( int i = 0 ; i < temp . size ( ) ; i + + ) {
extensions . insert ( temp [ i ] ) ;
}
2021-04-25 22:40:58 +08:00
for ( Set < String > : : Element * E = extensions . front ( ) ; E ; E = E - > next ( ) ) {
2020-06-23 19:48:59 +08:00
r_extensions - > push_back ( E - > get ( ) ) ;
}
}
bool EditorTranslationParser : : can_parse ( const String & p_extension ) const {
List < String > extensions ;
get_recognized_extensions ( & extensions ) ;
for ( int i = 0 ; i < extensions . size ( ) ; i + + ) {
if ( p_extension = = extensions [ i ] ) {
return true ;
}
}
return false ;
}
Ref < EditorTranslationParserPlugin > EditorTranslationParser : : get_parser ( const String & p_extension ) const {
// Consider user-defined parsers first.
for ( int i = 0 ; i < custom_parsers . size ( ) ; i + + ) {
List < String > temp ;
custom_parsers [ i ] - > get_recognized_extensions ( & temp ) ;
for ( int j = 0 ; j < temp . size ( ) ; j + + ) {
if ( temp [ j ] = = p_extension ) {
return custom_parsers [ i ] ;
}
}
}
for ( int i = 0 ; i < standard_parsers . size ( ) ; i + + ) {
List < String > temp ;
standard_parsers [ i ] - > get_recognized_extensions ( & temp ) ;
for ( int j = 0 ; j < temp . size ( ) ; j + + ) {
if ( temp [ j ] = = p_extension ) {
return standard_parsers [ i ] ;
}
}
}
WARN_PRINT ( " No translation parser available for \" " + p_extension + " \" extension. " ) ;
return nullptr ;
}
void EditorTranslationParser : : add_parser ( const Ref < EditorTranslationParserPlugin > & p_parser , ParserType p_type ) {
if ( p_type = = ParserType : : STANDARD ) {
standard_parsers . push_back ( p_parser ) ;
} else if ( p_type = = ParserType : : CUSTOM ) {
custom_parsers . push_back ( p_parser ) ;
}
}
void EditorTranslationParser : : remove_parser ( const Ref < EditorTranslationParserPlugin > & p_parser , ParserType p_type ) {
if ( p_type = = ParserType : : STANDARD ) {
standard_parsers . erase ( p_parser ) ;
} else if ( p_type = = ParserType : : CUSTOM ) {
custom_parsers . erase ( p_parser ) ;
}
}
2020-07-27 22:58:21 +08:00
void EditorTranslationParser : : clean_parsers ( ) {
standard_parsers . clear ( ) ;
custom_parsers . clear ( ) ;
}
2020-06-23 19:48:59 +08:00
EditorTranslationParser * EditorTranslationParser : : get_singleton ( ) {
if ( ! singleton ) {
singleton = memnew ( EditorTranslationParser ) ;
}
return singleton ;
}
EditorTranslationParser : : EditorTranslationParser ( ) {
}
EditorTranslationParser : : ~ EditorTranslationParser ( ) {
memdelete ( singleton ) ;
singleton = nullptr ;
}