2014-02-10 02:10:30 +01:00
/*************************************************************************/
/* script_editor_plugin.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 14:16:55 +02:00
/* https://godotengine.org */
2014-02-10 02:10:30 +01:00
/*************************************************************************/
2021-01-01 20:13:46 +01:00
/* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
2014-02-10 02:10:30 +01: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. */
/*************************************************************************/
2018-01-05 00:50:27 +01:00
2014-02-10 02:10:30 +01:00
# include "script_editor_plugin.h"
2017-01-16 08:04:19 +01:00
2020-11-07 23:33:38 +01:00
# include "core/config/project_settings.h"
2020-04-28 15:19:37 +02:00
# include "core/input/input.h"
2017-08-26 17:46:49 +02:00
# include "core/io/resource_loader.h"
# include "core/os/file_access.h"
# include "core/os/keyboard.h"
# include "core/os/os.h"
2020-02-07 02:52:05 +01:00
# include "editor/debugger/editor_debugger_node.h"
2017-03-05 16:44:50 +01:00
# include "editor/editor_node.h"
2019-12-24 08:17:23 +01:00
# include "editor/editor_run_script.h"
# include "editor/editor_scale.h"
2017-03-05 14:21:25 +01:00
# include "editor/editor_settings.h"
2019-12-24 08:17:23 +01:00
# include "editor/filesystem_dock.h"
2018-02-12 02:36:15 +01:00
# include "editor/find_in_files.h"
2017-08-26 17:46:49 +02:00
# include "editor/node_dock.h"
2018-10-27 17:24:41 +02:00
# include "editor/plugins/shader_editor_plugin.h"
2020-03-04 02:51:12 +01:00
# include "scene/main/window.h"
2019-10-10 22:19:47 +02:00
# include "scene/scene_string_names.h"
2018-02-12 02:36:15 +01:00
# include "script_text_editor.h"
2020-03-03 14:36:29 +01:00
# include "servers/display_server.h"
2019-08-07 08:41:10 +02:00
# include "text_editor.h"
2017-08-26 17:46:49 +02:00
2020-05-03 18:08:15 +02:00
/*** SYNTAX HIGHLIGHTER ****/
String EditorSyntaxHighlighter : : _get_name ( ) const {
ScriptInstance * si = get_script_instance ( ) ;
if ( si & & si - > has_method ( " _get_name " ) ) {
return si - > call ( " _get_name " ) ;
}
return " Unnamed " ;
}
Array EditorSyntaxHighlighter : : _get_supported_languages ( ) const {
ScriptInstance * si = get_script_instance ( ) ;
if ( si & & si - > has_method ( " _get_supported_languages " ) ) {
return si - > call ( " _get_supported_languages " ) ;
}
return Array ( ) ;
}
Ref < EditorSyntaxHighlighter > EditorSyntaxHighlighter : : _create ( ) const {
Ref < EditorSyntaxHighlighter > syntax_highlighter ;
syntax_highlighter . instance ( ) ;
if ( get_script_instance ( ) ) {
syntax_highlighter - > set_script ( get_script_instance ( ) - > get_script ( ) ) ;
}
return syntax_highlighter ;
}
void EditorSyntaxHighlighter : : _bind_methods ( ) {
ClassDB : : bind_method ( D_METHOD ( " _get_edited_resource " ) , & EditorSyntaxHighlighter : : _get_edited_resource ) ;
BIND_VMETHOD ( MethodInfo ( Variant : : STRING , " _get_name " ) ) ;
BIND_VMETHOD ( MethodInfo ( Variant : : ARRAY , " _get_supported_languages " ) ) ;
BIND_VMETHOD ( MethodInfo ( Variant : : ARRAY , " _get_supported_extentions " ) ) ;
}
////
void EditorStandardSyntaxHighlighter : : _update_cache ( ) {
highlighter - > set_text_edit ( text_edit ) ;
highlighter - > clear_keyword_colors ( ) ;
highlighter - > clear_member_keyword_colors ( ) ;
highlighter - > clear_color_regions ( ) ;
highlighter - > set_symbol_color ( EDITOR_GET ( " text_editor/highlighting/symbol_color " ) ) ;
highlighter - > set_function_color ( EDITOR_GET ( " text_editor/highlighting/function_color " ) ) ;
highlighter - > set_number_color ( EDITOR_GET ( " text_editor/highlighting/number_color " ) ) ;
highlighter - > set_member_variable_color ( EDITOR_GET ( " text_editor/highlighting/member_variable_color " ) ) ;
/* Engine types. */
const Color type_color = EDITOR_GET ( " text_editor/highlighting/engine_type_color " ) ;
List < StringName > types ;
ClassDB : : get_class_list ( & types ) ;
for ( List < StringName > : : Element * E = types . front ( ) ; E ; E = E - > next ( ) ) {
String n = E - > get ( ) ;
if ( n . begins_with ( " _ " ) ) {
n = n . substr ( 1 , n . length ( ) ) ;
}
highlighter - > add_keyword_color ( n , type_color ) ;
}
/* User types. */
const Color usertype_color = EDITOR_GET ( " text_editor/highlighting/user_type_color " ) ;
List < StringName > global_classes ;
ScriptServer : : get_global_class_list ( & global_classes ) ;
for ( List < StringName > : : Element * E = global_classes . front ( ) ; E ; E = E - > next ( ) ) {
highlighter - > add_keyword_color ( E - > get ( ) , usertype_color ) ;
}
/* Autoloads. */
Map < StringName , ProjectSettings : : AutoloadInfo > autoloads = ProjectSettings : : get_singleton ( ) - > get_autoload_list ( ) ;
for ( Map < StringName , ProjectSettings : : AutoloadInfo > : : Element * E = autoloads . front ( ) ; E ; E = E - > next ( ) ) {
const ProjectSettings : : AutoloadInfo & info = E - > value ( ) ;
if ( info . is_singleton ) {
highlighter - > add_keyword_color ( info . name , usertype_color ) ;
}
}
const Ref < Script > script = _get_edited_resource ( ) ;
if ( script . is_valid ( ) ) {
/* Core types. */
const Color basetype_color = EDITOR_GET ( " text_editor/highlighting/base_type_color " ) ;
List < String > core_types ;
script - > get_language ( ) - > get_core_type_words ( & core_types ) ;
for ( List < String > : : Element * E = core_types . front ( ) ; E ; E = E - > next ( ) ) {
highlighter - > add_keyword_color ( E - > get ( ) , basetype_color ) ;
}
/* Reserved words. */
const Color keyword_color = EDITOR_GET ( " text_editor/highlighting/keyword_color " ) ;
2021-04-08 16:12:22 +02:00
const Color control_flow_keyword_color = EDITOR_GET ( " text_editor/highlighting/control_flow_keyword_color " ) ;
2020-05-03 18:08:15 +02:00
List < String > keywords ;
script - > get_language ( ) - > get_reserved_words ( & keywords ) ;
for ( List < String > : : Element * E = keywords . front ( ) ; E ; E = E - > next ( ) ) {
2021-04-08 16:12:22 +02:00
if ( script - > get_language ( ) - > is_control_flow_keyword ( E - > get ( ) ) ) {
highlighter - > add_keyword_color ( E - > get ( ) , control_flow_keyword_color ) ;
} else {
highlighter - > add_keyword_color ( E - > get ( ) , keyword_color ) ;
}
2020-05-03 18:08:15 +02:00
}
/* Member types. */
const Color member_variable_color = EDITOR_GET ( " text_editor/highlighting/member_variable_color " ) ;
StringName instance_base = script - > get_instance_base_type ( ) ;
if ( instance_base ! = StringName ( ) ) {
List < PropertyInfo > plist ;
ClassDB : : get_property_list ( instance_base , & plist ) ;
for ( List < PropertyInfo > : : Element * E = plist . front ( ) ; E ; E = E - > next ( ) ) {
String name = E - > get ( ) . name ;
if ( E - > get ( ) . usage & PROPERTY_USAGE_CATEGORY | | E - > get ( ) . usage & PROPERTY_USAGE_GROUP | | E - > get ( ) . usage & PROPERTY_USAGE_SUBGROUP ) {
continue ;
}
if ( name . find ( " / " ) ! = - 1 ) {
continue ;
}
highlighter - > add_member_keyword_color ( name , member_variable_color ) ;
}
List < String > clist ;
ClassDB : : get_integer_constant_list ( instance_base , & clist ) ;
for ( List < String > : : Element * E = clist . front ( ) ; E ; E = E - > next ( ) ) {
highlighter - > add_member_keyword_color ( E - > get ( ) , member_variable_color ) ;
}
}
/* Comments */
const Color comment_color = EDITOR_GET ( " text_editor/highlighting/comment_color " ) ;
List < String > comments ;
script - > get_language ( ) - > get_comment_delimiters ( & comments ) ;
for ( List < String > : : Element * E = comments . front ( ) ; E ; E = E - > next ( ) ) {
String comment = E - > get ( ) ;
String beg = comment . get_slice ( " " , 0 ) ;
String end = comment . get_slice_count ( " " ) > 1 ? comment . get_slice ( " " , 1 ) : String ( ) ;
highlighter - > add_color_region ( beg , end , comment_color , end = = " " ) ;
}
/* Strings */
const Color string_color = EDITOR_GET ( " text_editor/highlighting/string_color " ) ;
List < String > strings ;
script - > get_language ( ) - > get_string_delimiters ( & strings ) ;
for ( List < String > : : Element * E = strings . front ( ) ; E ; E = E - > next ( ) ) {
String string = E - > get ( ) ;
String beg = string . get_slice ( " " , 0 ) ;
String end = string . get_slice_count ( " " ) > 1 ? string . get_slice ( " " , 1 ) : String ( ) ;
highlighter - > add_color_region ( beg , end , string_color , end = = " " ) ;
}
}
}
Ref < EditorSyntaxHighlighter > EditorStandardSyntaxHighlighter : : _create ( ) const {
Ref < EditorStandardSyntaxHighlighter > syntax_highlighter ;
syntax_highlighter . instance ( ) ;
return syntax_highlighter ;
}
////
Ref < EditorSyntaxHighlighter > EditorPlainTextSyntaxHighlighter : : _create ( ) const {
Ref < EditorPlainTextSyntaxHighlighter > syntax_highlighter ;
syntax_highlighter . instance ( ) ;
return syntax_highlighter ;
}
////////////////////////////////////////////////////////////////////////////////
2015-06-22 05:03:19 +02:00
/*** SCRIPT EDITOR ****/
2014-02-10 02:10:30 +01:00
2016-09-12 15:52:29 +02:00
void ScriptEditorBase : : _bind_methods ( ) {
2021-05-13 21:50:25 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_base_editor " ) , & ScriptEditorBase : : get_base_editor ) ;
2016-09-12 15:52:29 +02:00
ADD_SIGNAL ( MethodInfo ( " name_changed " ) ) ;
2017-08-29 12:00:37 +02:00
ADD_SIGNAL ( MethodInfo ( " edited_script_changed " ) ) ;
2017-12-11 00:56:49 +01:00
ADD_SIGNAL ( MethodInfo ( " request_help " , PropertyInfo ( Variant : : STRING , " topic " ) ) ) ;
2017-03-05 16:44:50 +01:00
ADD_SIGNAL ( MethodInfo ( " request_open_script_at_line " , PropertyInfo ( Variant : : OBJECT , " script " ) , PropertyInfo ( Variant : : INT , " line " ) ) ) ;
2016-09-12 15:52:29 +02:00
ADD_SIGNAL ( MethodInfo ( " request_save_history " ) ) ;
2017-03-05 16:44:50 +01:00
ADD_SIGNAL ( MethodInfo ( " go_to_help " , PropertyInfo ( Variant : : STRING , " what " ) ) ) ;
2019-08-09 04:22:33 +02:00
// TODO: This signal is no use for VisualScript.
2018-02-12 02:36:15 +01:00
ADD_SIGNAL ( MethodInfo ( " search_in_files_requested " , PropertyInfo ( Variant : : STRING , " text " ) ) ) ;
2020-02-09 10:10:58 +01:00
ADD_SIGNAL ( MethodInfo ( " replace_in_files_requested " , PropertyInfo ( Variant : : STRING , " text " ) ) ) ;
2020-03-07 15:29:44 +01:00
BIND_VMETHOD ( MethodInfo ( " add_syntax_highlighter " , PropertyInfo ( Variant : : OBJECT , " highlighter " ) ) ) ;
2016-09-12 15:52:29 +02:00
}
2018-09-06 23:33:44 +02:00
static bool _is_built_in_script ( Script * p_script ) {
2016-07-07 01:35:49 +02:00
String path = p_script - > get_path ( ) ;
2018-09-06 23:33:44 +02:00
return path . find ( " :: " ) ! = - 1 ;
2016-07-07 01:35:49 +02:00
}
2015-06-26 06:14:31 +02:00
class EditorScriptCodeCompletionCache : public ScriptCodeCompletionCache {
struct Cache {
2020-11-24 10:12:55 +01:00
uint64_t time_loaded = 0 ;
2015-06-26 06:14:31 +02:00
RES cache ;
} ;
2017-03-05 16:44:50 +01:00
Map < String , Cache > cached ;
2015-06-26 06:14:31 +02:00
public :
2020-11-24 10:12:55 +01:00
uint64_t max_time_cache = 5 * 60 * 1000 ; //minutes, five
int max_cache_size = 128 ;
2015-06-26 06:14:31 +02:00
void cleanup ( ) {
2017-03-05 16:44:50 +01:00
List < Map < String , Cache > : : Element * > to_clean ;
2015-06-26 06:14:31 +02:00
2017-03-05 16:44:50 +01:00
Map < String , Cache > : : Element * I = cached . front ( ) ;
while ( I ) {
if ( ( OS : : get_singleton ( ) - > get_ticks_msec ( ) - I - > get ( ) . time_loaded ) > max_time_cache ) {
2015-06-26 06:14:31 +02:00
to_clean . push_back ( I ) ;
}
2017-03-05 16:44:50 +01:00
I = I - > next ( ) ;
2015-06-26 06:14:31 +02:00
}
2017-03-05 16:44:50 +01:00
while ( to_clean . front ( ) ) {
2015-06-26 06:14:31 +02:00
cached . erase ( to_clean . front ( ) - > get ( ) ) ;
to_clean . pop_front ( ) ;
}
}
2018-10-02 12:07:44 +02:00
virtual RES get_cached_resource ( const String & p_path ) {
2017-03-05 16:44:50 +01:00
Map < String , Cache > : : Element * E = cached . find ( p_path ) ;
2015-06-26 06:14:31 +02:00
if ( ! E ) {
Cache c ;
2017-03-05 16:44:50 +01:00
c . cache = ResourceLoader : : load ( p_path ) ;
E = cached . insert ( p_path , c ) ;
2015-06-26 06:14:31 +02:00
}
2017-03-05 16:44:50 +01:00
E - > get ( ) . time_loaded = OS : : get_singleton ( ) - > get_ticks_msec ( ) ;
2015-06-26 06:14:31 +02:00
2017-03-05 16:44:50 +01:00
if ( cached . size ( ) > max_cache_size ) {
2015-06-26 06:14:31 +02:00
uint64_t older ;
2017-03-05 16:44:50 +01:00
Map < String , Cache > : : Element * O = cached . front ( ) ;
older = O - > get ( ) . time_loaded ;
Map < String , Cache > : : Element * I = O ;
while ( I ) {
if ( I - > get ( ) . time_loaded < older ) {
2015-06-26 06:14:31 +02:00
older = I - > get ( ) . time_loaded ;
2017-03-05 16:44:50 +01:00
O = I ;
2015-06-26 06:14:31 +02:00
}
2017-03-05 16:44:50 +01:00
I = I - > next ( ) ;
2015-06-26 06:14:31 +02:00
}
2017-12-18 18:46:17 +01:00
if ( O ! = E ) { //should never happen..
2015-06-26 06:14:31 +02:00
cached . erase ( O ) ;
}
}
return E - > get ( ) . cache ;
}
2018-10-02 12:07:44 +02:00
virtual ~ EditorScriptCodeCompletionCache ( ) { }
2015-06-26 06:14:31 +02:00
} ;
2018-10-02 12:07:44 +02:00
void ScriptEditorQuickOpen : : popup_dialog ( const Vector < String > & p_functions , bool p_dontclear ) {
2014-02-10 02:10:30 +01:00
popup_centered_ratio ( 0.6 ) ;
2020-05-14 16:41:43 +02:00
if ( p_dontclear ) {
2014-02-10 02:10:30 +01:00
search_box - > select_all ( ) ;
2020-05-14 16:41:43 +02:00
} else {
2014-02-10 02:10:30 +01:00
search_box - > clear ( ) ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
search_box - > grab_focus ( ) ;
2017-03-05 16:44:50 +01:00
functions = p_functions ;
2014-02-10 02:10:30 +01:00
_update_search ( ) ;
}
2017-03-05 16:44:50 +01:00
void ScriptEditorQuickOpen : : _text_changed ( const String & p_newtext ) {
2014-02-10 02:10:30 +01:00
_update_search ( ) ;
}
2017-05-20 17:38:03 +02:00
void ScriptEditorQuickOpen : : _sbox_input ( const Ref < InputEvent > & p_ie ) {
Ref < InputEventKey > k = p_ie ;
2014-02-10 02:10:30 +01:00
2018-04-05 19:59:35 +02:00
if ( k . is_valid ( ) & & ( k - > get_keycode ( ) = = KEY_UP | |
k - > get_keycode ( ) = = KEY_DOWN | |
k - > get_keycode ( ) = = KEY_PAGEUP | |
k - > get_keycode ( ) = = KEY_PAGEDOWN ) ) {
2017-05-20 17:38:03 +02:00
search_options - > call ( " _gui_input " , k ) ;
2014-02-10 02:10:30 +01:00
search_box - > accept_event ( ) ;
}
}
void ScriptEditorQuickOpen : : _update_search ( ) {
search_options - > clear ( ) ;
TreeItem * root = search_options - > create_item ( ) ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < functions . size ( ) ; i + + ) {
2014-02-10 02:10:30 +01:00
String file = functions [ i ] ;
2017-03-05 16:44:50 +01:00
if ( ( search_box - > get_text ( ) = = " " | | file . findn ( search_box - > get_text ( ) ) ! = - 1 ) ) {
2014-02-10 02:10:30 +01:00
TreeItem * ti = search_options - > create_item ( root ) ;
2017-03-05 16:44:50 +01:00
ti - > set_text ( 0 , file ) ;
2021-03-07 21:07:30 +01:00
if ( root - > get_first_child ( ) = = ti ) {
2014-02-10 02:10:30 +01:00
ti - > select ( 0 ) ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
}
}
2021-03-07 21:07:30 +01:00
get_ok_button ( ) - > set_disabled ( root - > get_first_child ( ) = = nullptr ) ;
2014-02-10 02:10:30 +01:00
}
void ScriptEditorQuickOpen : : _confirmed ( ) {
TreeItem * ti = search_options - > get_selected ( ) ;
2020-05-14 16:41:43 +02:00
if ( ! ti ) {
2014-02-10 02:10:30 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-03-05 16:44:50 +01:00
int line = ti - > get_text ( 0 ) . get_slice ( " : " , 1 ) . to_int ( ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
emit_signal ( " goto_line " , line - 1 ) ;
2014-02-10 02:10:30 +01:00
hide ( ) ;
}
void ScriptEditorQuickOpen : : _notification ( int p_what ) {
2017-07-26 16:02:53 +02:00
switch ( p_what ) {
case NOTIFICATION_ENTER_TREE : {
2020-02-21 18:28:45 +01:00
connect ( " confirmed " , callable_mp ( this , & ScriptEditorQuickOpen : : _confirmed ) ) ;
2018-07-26 13:45:38 +02:00
search_box - > set_clear_button_enabled ( true ) ;
2020-02-22 20:47:50 +01:00
[[fallthrough]] ;
2020-03-13 00:12:34 +01:00
}
2020-03-06 18:00:16 +01:00
case NOTIFICATION_VISIBILITY_CHANGED : {
2020-03-12 13:37:40 +01:00
search_box - > set_right_icon ( search_options - > get_theme_icon ( " Search " , " EditorIcons " ) ) ;
2017-07-26 16:02:53 +02:00
} break ;
2018-11-05 03:45:25 +01:00
case NOTIFICATION_EXIT_TREE : {
2020-02-21 18:28:45 +01:00
disconnect ( " confirmed " , callable_mp ( this , & ScriptEditorQuickOpen : : _confirmed ) ) ;
2018-11-05 03:45:25 +01:00
} break ;
2014-02-10 02:10:30 +01:00
}
}
void ScriptEditorQuickOpen : : _bind_methods ( ) {
2017-03-05 16:44:50 +01:00
ADD_SIGNAL ( MethodInfo ( " goto_line " , PropertyInfo ( Variant : : INT , " line " ) ) ) ;
2014-02-10 02:10:30 +01:00
}
ScriptEditorQuickOpen : : ScriptEditorQuickOpen ( ) {
2017-03-05 16:44:50 +01:00
VBoxContainer * vbc = memnew ( VBoxContainer ) ;
2014-02-10 02:10:30 +01:00
add_child ( vbc ) ;
2017-03-05 16:44:50 +01:00
search_box = memnew ( LineEdit ) ;
vbc - > add_margin_child ( TTR ( " Search: " ) , search_box ) ;
2020-02-21 18:28:45 +01:00
search_box - > connect ( " text_changed " , callable_mp ( this , & ScriptEditorQuickOpen : : _text_changed ) ) ;
search_box - > connect ( " gui_input " , callable_mp ( this , & ScriptEditorQuickOpen : : _sbox_input ) ) ;
2017-03-05 16:44:50 +01:00
search_options = memnew ( Tree ) ;
vbc - > add_margin_child ( TTR ( " Matches: " ) , search_options , true ) ;
2020-12-14 19:37:30 +01:00
get_ok_button ( ) - > set_text ( TTR ( " Open " ) ) ;
get_ok_button ( ) - > set_disabled ( true ) ;
2014-02-10 02:10:30 +01:00
register_text_enter ( search_box ) ;
set_hide_on_ok ( false ) ;
2020-02-21 18:28:45 +01:00
search_options - > connect ( " item_activated " , callable_mp ( this , & ScriptEditorQuickOpen : : _confirmed ) ) ;
2014-02-10 02:10:30 +01:00
search_options - > set_hide_root ( true ) ;
2019-08-09 04:22:33 +02:00
search_options - > set_hide_folding ( true ) ;
2020-03-12 13:37:40 +01:00
search_options - > add_theme_constant_override ( " draw_guides " , 1 ) ;
2014-02-10 02:10:30 +01:00
}
/////////////////////////////////
2020-04-02 01:20:12 +02:00
ScriptEditor * ScriptEditor : : script_editor = nullptr ;
2014-02-10 02:10:30 +01:00
/*** SCRIPT EDITOR ******/
2017-03-05 16:44:50 +01:00
String ScriptEditor : : _get_debug_tooltip ( const String & p_text , Node * _se ) {
2020-02-07 02:52:05 +01:00
String val = EditorDebuggerNode : : get_singleton ( ) - > get_var_value ( p_text ) ;
2017-03-05 16:44:50 +01:00
if ( val ! = String ( ) ) {
return p_text + " : " + val ;
2014-02-10 02:10:30 +01:00
} else {
return String ( ) ;
}
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _breaked ( bool p_breaked , bool p_can_debug ) {
2017-02-20 21:22:50 +01:00
if ( bool ( EditorSettings : : get_singleton ( ) - > get ( " text_editor/external/use_external_editor " ) ) ) {
return ;
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-07 00:00:54 +02:00
if ( ! se ) {
continue ;
}
se - > set_debugger_active ( p_breaked ) ;
}
2014-02-10 02:10:30 +01:00
}
2015-08-10 01:39:59 +02:00
void ScriptEditor : : _script_created ( Ref < Script > p_script ) {
editor - > push_item ( p_script . operator - > ( ) ) ;
}
2014-02-10 02:10:30 +01:00
void ScriptEditor : : _goto_script_line2 ( int p_line ) {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * current = _get_current_editor ( ) ;
2020-05-14 16:41:43 +02:00
if ( current ) {
2018-02-12 02:36:15 +01:00
current - > goto_line ( p_line ) ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _goto_script_line ( REF p_script , int p_line ) {
2018-01-09 17:19:37 +01:00
Ref < Script > script = Object : : cast_to < Script > ( * p_script ) ;
2019-01-22 10:55:55 +01:00
if ( script . is_valid ( ) & & ( script - > has_source_code ( ) | | script - > get_path ( ) . is_resource_file ( ) ) ) {
2018-01-09 17:19:37 +01:00
if ( edit ( p_script , p_line , 0 ) ) {
editor - > push_item ( p_script . ptr ( ) ) ;
2016-08-07 00:00:54 +02:00
2018-02-12 02:36:15 +01:00
ScriptEditorBase * current = _get_current_editor ( ) ;
2019-01-04 13:09:01 +01:00
if ( ScriptTextEditor * script_text_editor = Object : : cast_to < ScriptTextEditor > ( current ) ) {
script_text_editor - > goto_line_centered ( p_line ) ;
} else if ( current ) {
2018-02-12 02:36:15 +01:00
current - > goto_line ( p_line , true ) ;
2019-01-04 13:09:01 +01:00
}
2018-01-09 17:19:37 +01:00
}
}
2014-02-10 02:10:30 +01:00
}
2019-04-22 18:20:27 +02:00
void ScriptEditor : : _set_execution ( REF p_script , int p_line ) {
Ref < Script > script = Object : : cast_to < Script > ( * p_script ) ;
if ( script . is_valid ( ) & & ( script - > has_source_code ( ) | | script - > get_path ( ) . is_resource_file ( ) ) ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2019-04-22 18:20:27 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2019-04-22 18:20:27 +02:00
2020-04-02 01:20:12 +02:00
if ( ( script ! = nullptr & & se - > get_edited_resource ( ) = = p_script ) | | se - > get_edited_resource ( ) - > get_path ( ) = = script - > get_path ( ) ) {
2019-04-22 18:20:27 +02:00
se - > set_executing_line ( p_line ) ;
}
}
}
}
void ScriptEditor : : _clear_execution ( REF p_script ) {
Ref < Script > script = Object : : cast_to < Script > ( * p_script ) ;
if ( script . is_valid ( ) & & ( script - > has_source_code ( ) | | script - > get_path ( ) . is_resource_file ( ) ) ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2019-04-22 18:20:27 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2019-04-22 18:20:27 +02:00
2020-04-02 01:20:12 +02:00
if ( ( script ! = nullptr & & se - > get_edited_resource ( ) = = p_script ) | | se - > get_edited_resource ( ) - > get_path ( ) = = script - > get_path ( ) ) {
2019-04-22 18:20:27 +02:00
se - > clear_executing_line ( ) ;
}
}
}
}
2018-02-12 02:36:15 +01:00
ScriptEditorBase * ScriptEditor : : _get_current_editor ( ) const {
int selected = tab_container - > get_current_tab ( ) ;
2020-05-14 16:41:43 +02:00
if ( selected < 0 | | selected > = tab_container - > get_child_count ( ) ) {
2020-04-02 01:20:12 +02:00
return nullptr ;
2020-05-14 16:41:43 +02:00
}
2018-02-12 02:36:15 +01:00
return Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( selected ) ) ;
}
2015-11-17 13:46:08 +01:00
void ScriptEditor : : _update_history_arrows ( ) {
2017-03-05 16:44:50 +01:00
script_back - > set_disabled ( history_pos < = 0 ) ;
script_forward - > set_disabled ( history_pos > = history . size ( ) - 1 ) ;
2015-11-17 13:46:08 +01:00
}
2016-09-12 15:52:29 +02:00
void ScriptEditor : : _save_history ( ) {
2017-03-05 16:44:50 +01:00
if ( history_pos > = 0 & & history_pos < history . size ( ) & & history [ history_pos ] . control = = tab_container - > get_current_tab_control ( ) ) {
2016-09-12 15:52:29 +02:00
Node * n = tab_container - > get_current_tab_control ( ) ;
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < ScriptEditorBase > ( n ) ) {
2018-07-25 03:11:03 +02:00
history . write [ history_pos ] . state = Object : : cast_to < ScriptEditorBase > ( n ) - > get_edit_state ( ) ;
2016-09-12 15:52:29 +02:00
}
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < EditorHelp > ( n ) ) {
2018-07-25 03:11:03 +02:00
history . write [ history_pos ] . state = Object : : cast_to < EditorHelp > ( n ) - > get_scroll ( ) ;
2016-09-12 15:52:29 +02:00
}
}
2017-03-05 16:44:50 +01:00
history . resize ( history_pos + 1 ) ;
2016-09-12 15:52:29 +02:00
ScriptHistory sh ;
2017-03-05 16:44:50 +01:00
sh . control = tab_container - > get_current_tab_control ( ) ;
sh . state = Variant ( ) ;
2016-09-12 15:52:29 +02:00
history . push_back ( sh ) ;
history_pos + + ;
_update_history_arrows ( ) ;
}
2015-11-17 13:46:08 +01:00
void ScriptEditor : : _go_to_tab ( int p_idx ) {
2019-05-05 06:56:19 +02:00
ScriptEditorBase * current = _get_current_editor ( ) ;
if ( current ) {
if ( current - > is_unsaved ( ) ) {
current - > apply_code ( ) ;
}
}
2017-08-24 22:58:51 +02:00
Control * c = Object : : cast_to < Control > ( tab_container - > get_child ( p_idx ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! c ) {
2015-11-17 13:46:08 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
if ( history_pos > = 0 & & history_pos < history . size ( ) & & history [ history_pos ] . control = = tab_container - > get_current_tab_control ( ) ) {
2015-11-17 13:46:08 +01:00
Node * n = tab_container - > get_current_tab_control ( ) ;
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < ScriptEditorBase > ( n ) ) {
2018-07-25 03:11:03 +02:00
history . write [ history_pos ] . state = Object : : cast_to < ScriptEditorBase > ( n ) - > get_edit_state ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < EditorHelp > ( n ) ) {
2018-07-25 03:11:03 +02:00
history . write [ history_pos ] . state = Object : : cast_to < EditorHelp > ( n ) - > get_scroll ( ) ;
2015-11-17 13:46:08 +01:00
}
}
2017-03-05 16:44:50 +01:00
history . resize ( history_pos + 1 ) ;
2015-11-17 13:46:08 +01:00
ScriptHistory sh ;
2017-03-05 16:44:50 +01:00
sh . control = c ;
sh . state = Variant ( ) ;
2015-11-17 13:46:08 +01:00
history . push_back ( sh ) ;
history_pos + + ;
tab_container - > set_current_tab ( p_idx ) ;
c = tab_container - > get_current_tab_control ( ) ;
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < ScriptEditorBase > ( c ) ) {
script_name_label - > set_text ( Object : : cast_to < ScriptEditorBase > ( c ) - > get_name ( ) ) ;
2020-03-12 13:37:40 +01:00
script_icon - > set_texture ( Object : : cast_to < ScriptEditorBase > ( c ) - > get_theme_icon ( ) ) ;
2020-05-14 16:41:43 +02:00
if ( is_visible_in_tree ( ) ) {
2017-08-24 22:58:51 +02:00
Object : : cast_to < ScriptEditorBase > ( c ) - > ensure_focus ( ) ;
2020-05-14 16:41:43 +02:00
}
2017-07-06 09:18:20 +02:00
2018-05-28 17:52:28 +02:00
Ref < Script > script = Object : : cast_to < ScriptEditorBase > ( c ) - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
if ( script ! = nullptr ) {
2018-05-28 17:52:28 +02:00
notify_script_changed ( script ) ;
}
2019-05-08 18:49:49 +02:00
Object : : cast_to < ScriptEditorBase > ( c ) - > validate ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < EditorHelp > ( c ) ) {
script_name_label - > set_text ( Object : : cast_to < EditorHelp > ( c ) - > get_class ( ) ) ;
2020-03-12 13:37:40 +01:00
script_icon - > set_texture ( get_theme_icon ( " Help " , " EditorIcons " ) ) ;
2020-05-14 16:41:43 +02:00
if ( is_visible_in_tree ( ) ) {
2017-08-24 22:58:51 +02:00
Object : : cast_to < EditorHelp > ( c ) - > set_focused ( ) ;
2020-05-14 16:41:43 +02:00
}
2015-11-17 13:46:08 +01:00
}
2017-03-05 16:44:50 +01:00
c - > set_meta ( " __editor_pass " , + + edit_pass ) ;
2015-11-17 13:46:08 +01:00
_update_history_arrows ( ) ;
2017-03-05 16:44:50 +01:00
_update_script_colors ( ) ;
2017-05-28 16:20:38 +02:00
_update_members_overview ( ) ;
2017-09-14 02:56:37 +02:00
_update_help_overview ( ) ;
2016-08-03 00:11:05 +02:00
_update_selected_editor_menu ( ) ;
2017-05-28 16:20:38 +02:00
_update_members_overview_visibility ( ) ;
2017-09-14 02:56:37 +02:00
_update_help_overview_visibility ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-04-27 17:07:39 +02:00
void ScriptEditor : : _add_recent_script ( String p_path ) {
2020-12-15 13:04:21 +01:00
if ( p_path . is_empty ( ) ) {
2017-04-27 17:07:39 +02:00
return ;
}
2018-01-08 01:31:36 +01:00
Array rc = EditorSettings : : get_singleton ( ) - > get_project_metadata ( " recent_files " , " scripts " , Array ( ) ) ;
if ( rc . find ( p_path ) ! = - 1 ) {
rc . erase ( p_path ) ;
}
rc . push_front ( p_path ) ;
if ( rc . size ( ) > 10 ) {
rc . resize ( 10 ) ;
2017-04-27 17:07:39 +02:00
}
2018-01-08 01:31:36 +01:00
EditorSettings : : get_singleton ( ) - > set_project_metadata ( " recent_files " , " scripts " , rc ) ;
2017-04-27 17:07:39 +02:00
_update_recent_scripts ( ) ;
}
void ScriptEditor : : _update_recent_scripts ( ) {
2018-01-08 01:31:36 +01:00
Array rc = EditorSettings : : get_singleton ( ) - > get_project_metadata ( " recent_files " , " scripts " , Array ( ) ) ;
2017-04-27 17:07:39 +02:00
recent_scripts - > clear ( ) ;
2018-01-08 01:31:36 +01:00
String path ;
for ( int i = 0 ; i < rc . size ( ) ; i + + ) {
path = rc [ i ] ;
recent_scripts - > add_item ( path . replace ( " res:// " , " " ) ) ;
2017-04-27 17:07:39 +02:00
}
recent_scripts - > add_separator ( ) ;
recent_scripts - > add_shortcut ( ED_SHORTCUT ( " script_editor/clear_recent " , TTR ( " Clear Recent Files " ) ) ) ;
2017-10-11 10:24:57 +02:00
recent_scripts - > set_as_minsize ( ) ;
2017-04-27 17:07:39 +02:00
}
void ScriptEditor : : _open_recent_script ( int p_idx ) {
// clear button
if ( p_idx = = recent_scripts - > get_item_count ( ) - 1 ) {
2018-01-08 01:31:36 +01:00
EditorSettings : : get_singleton ( ) - > set_project_metadata ( " recent_files " , " scripts " , Array ( ) ) ;
2017-10-11 10:24:57 +02:00
call_deferred ( " _update_recent_scripts " ) ;
2017-04-27 17:07:39 +02:00
return ;
}
2018-01-08 01:31:36 +01:00
Array rc = EditorSettings : : get_singleton ( ) - > get_project_metadata ( " recent_files " , " scripts " , Array ( ) ) ;
ERR_FAIL_INDEX ( p_idx , rc . size ( ) ) ;
2017-04-27 17:07:39 +02:00
2018-01-08 01:31:36 +01:00
String path = rc [ p_idx ] ;
// if its not on disk its a help file or deleted
if ( FileAccess : : exists ( path ) ) {
2018-05-28 17:52:28 +02:00
List < String > extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " Script " , & extensions ) ;
if ( extensions . find ( path . get_extension ( ) ) ) {
Ref < Script > script = ResourceLoader : : load ( path ) ;
if ( script . is_valid ( ) ) {
edit ( script , true ) ;
return ;
}
2017-04-27 17:07:39 +02:00
}
2018-01-08 01:31:36 +01:00
2018-05-28 17:52:28 +02:00
Error err ;
Ref < TextFile > text_file = _load_text_file ( path , & err ) ;
if ( text_file . is_valid ( ) ) {
edit ( text_file , true ) ;
return ;
}
2018-10-25 02:19:21 +02:00
// if it's a path then it's most likely a deleted file not help
2018-09-09 04:26:05 +02:00
} else if ( path . find ( " :: " ) ! = - 1 ) {
// built-in script
2019-09-04 20:00:09 +02:00
String res_path = path . get_slice ( " :: " , 0 ) ;
if ( ResourceLoader : : get_resource_type ( res_path ) = = " PackedScene " ) {
if ( ! EditorNode : : get_singleton ( ) - > is_scene_open ( res_path ) ) {
EditorNode : : get_singleton ( ) - > load_scene ( res_path ) ;
}
} else {
EditorNode : : get_singleton ( ) - > load_resource ( res_path ) ;
2019-09-04 19:14:47 +02:00
}
2018-09-09 04:26:05 +02:00
Ref < Script > script = ResourceLoader : : load ( path ) ;
if ( script . is_valid ( ) ) {
edit ( script , true ) ;
return ;
}
2018-01-08 01:31:36 +01:00
} else if ( ! path . is_resource_file ( ) ) {
_help_class_open ( path ) ;
return ;
2017-04-27 17:07:39 +02:00
}
2018-01-08 01:31:36 +01:00
rc . remove ( p_idx ) ;
EditorSettings : : get_singleton ( ) - > set_project_metadata ( " recent_files " , " scripts " , rc ) ;
_update_recent_scripts ( ) ;
_show_error_dialog ( path ) ;
}
void ScriptEditor : : _show_error_dialog ( String p_path ) {
error_dialog - > set_text ( vformat ( TTR ( " Can't open '%s'. The file could have been moved or deleted. " ) , p_path ) ) ;
2020-03-06 18:00:16 +01:00
error_dialog - > popup_centered ( ) ;
2017-04-27 17:07:39 +02:00
}
2018-02-06 22:32:03 +01:00
void ScriptEditor : : _close_tab ( int p_idx , bool p_save , bool p_history_back ) {
2016-07-07 01:35:49 +02:00
int selected = p_idx ;
2020-05-14 16:41:43 +02:00
if ( selected < 0 | | selected > = tab_container - > get_child_count ( ) ) {
2014-02-10 02:10:30 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-03-09 00:00:52 +01:00
2015-11-17 13:46:08 +01:00
Node * tselected = tab_container - > get_child ( selected ) ;
2018-05-28 17:52:28 +02:00
2019-11-20 10:09:59 +01:00
ScriptEditorBase * current = Object : : cast_to < ScriptEditorBase > ( tselected ) ;
2015-11-17 13:46:08 +01:00
if ( current ) {
2020-07-08 18:53:18 +02:00
Ref < Script > script = current - > get_edited_resource ( ) ;
2020-07-23 14:21:28 +02:00
if ( p_save & & script . is_valid ( ) ) {
// Do not try to save internal scripts, but prompt to save in-memory
// scripts which are not saved to disk yet (have empty path).
if ( script - > get_path ( ) . find ( " local:// " ) = = - 1 & & script - > get_path ( ) . find ( " :: " ) = = - 1 ) {
2021-05-09 12:29:50 +02:00
save_current_script ( ) ;
2020-07-08 18:53:18 +02:00
}
2017-01-02 19:45:59 +01:00
}
2020-07-23 14:21:28 +02:00
if ( script . is_valid ( ) ) {
2020-12-15 13:04:21 +01:00
if ( ! script - > get_path ( ) . is_empty ( ) ) {
2020-07-23 14:21:28 +02:00
// Only saved scripts can be restored.
previous_scripts . push_back ( script - > get_path ( ) ) ;
}
2018-05-28 17:52:28 +02:00
notify_script_close ( script ) ;
}
2015-11-17 13:46:08 +01:00
}
2014-02-10 02:10:30 +01:00
2017-07-22 11:08:56 +02:00
// roll back to previous tab
2018-02-06 22:32:03 +01:00
if ( p_history_back ) {
_history_back ( ) ;
}
2017-07-22 11:08:56 +02:00
2015-11-17 13:46:08 +01:00
//remove from history
2017-03-05 16:44:50 +01:00
history . resize ( history_pos + 1 ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < history . size ( ) ; i + + ) {
if ( history [ i ] . control = = tselected ) {
2015-11-17 13:46:08 +01:00
history . remove ( i ) ;
i - - ;
history_pos - - ;
}
}
2017-03-05 16:44:50 +01:00
if ( history_pos > = history . size ( ) ) {
history_pos = history . size ( ) - 1 ;
2015-11-17 13:46:08 +01:00
}
2014-02-10 02:10:30 +01:00
int idx = tab_container - > get_current_tab ( ) ;
2018-01-08 19:47:47 +01:00
if ( current ) {
current - > clear_edit_menu ( ) ;
}
2015-11-17 13:46:08 +01:00
memdelete ( tselected ) ;
2020-05-14 16:41:43 +02:00
if ( idx > = tab_container - > get_child_count ( ) ) {
2017-03-05 16:44:50 +01:00
idx = tab_container - > get_child_count ( ) - 1 ;
2020-05-14 16:41:43 +02:00
}
2017-03-05 16:44:50 +01:00
if ( idx > = 0 ) {
if ( history_pos > = 0 ) {
2015-11-17 13:46:08 +01:00
idx = history [ history_pos ] . control - > get_index ( ) ;
}
2014-02-10 02:10:30 +01:00
tab_container - > set_current_tab ( idx ) ;
2019-05-03 01:07:06 +02:00
} else {
_update_selected_editor_menu ( ) ;
2015-06-22 05:03:19 +02:00
}
2015-11-17 13:46:08 +01:00
_update_history_arrows ( ) ;
2015-06-22 05:03:19 +02:00
_update_script_names ( ) ;
2017-05-28 16:20:38 +02:00
_update_members_overview_visibility ( ) ;
2017-09-14 02:56:37 +02:00
_update_help_overview_visibility ( ) ;
2016-07-18 17:30:43 +02:00
_save_layout ( ) ;
2016-07-07 01:35:49 +02:00
}
2021-03-04 01:04:31 +01:00
void ScriptEditor : : _close_current_tab ( bool p_save ) {
_close_tab ( tab_container - > get_current_tab ( ) , p_save ) ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _close_discard_current_tab ( const String & p_str ) {
2017-01-02 19:45:59 +01:00
_close_tab ( tab_container - > get_current_tab ( ) , false ) ;
erase_tab_confirm - > hide ( ) ;
}
2016-07-18 00:18:48 +02:00
void ScriptEditor : : _close_docs_tab ( ) {
int child_count = tab_container - > get_child_count ( ) ;
2017-03-05 16:44:50 +01:00
for ( int i = child_count - 1 ; i > = 0 ; i - - ) {
2017-08-24 22:58:51 +02:00
EditorHelp * se = Object : : cast_to < EditorHelp > ( tab_container - > get_child ( i ) ) ;
2016-07-18 00:18:48 +02:00
2016-08-03 00:11:05 +02:00
if ( se ) {
2018-02-06 22:32:03 +01:00
_close_tab ( i , true , false ) ;
2016-07-18 00:18:48 +02:00
}
}
}
2017-12-07 04:57:41 +01:00
void ScriptEditor : : _copy_script_path ( ) {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * se = _get_current_editor ( ) ;
2019-11-20 10:09:59 +01:00
if ( se ) {
RES script = se - > get_edited_resource ( ) ;
DisplayServer : : get_singleton ( ) - > clipboard_set ( script - > get_path ( ) ) ;
}
2017-12-07 04:57:41 +01:00
}
2017-11-20 06:15:40 +01:00
void ScriptEditor : : _close_other_tabs ( ) {
int child_count = tab_container - > get_child_count ( ) ;
int current_idx = tab_container - > get_current_tab ( ) ;
for ( int i = child_count - 1 ; i > = 0 ; i - - ) {
if ( i = = current_idx ) {
continue ;
}
tab_container - > set_current_tab ( i ) ;
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
if ( se ) {
// Maybe there are unsaved changes
if ( se - > is_unsaved ( ) ) {
_ask_close_current_unsaved_tab ( se ) ;
continue ;
}
}
2021-03-04 01:04:31 +01:00
_close_current_tab ( false ) ;
2017-11-20 06:15:40 +01:00
}
}
2016-08-13 02:56:38 +02:00
void ScriptEditor : : _close_all_tabs ( ) {
int child_count = tab_container - > get_child_count ( ) ;
2017-03-05 16:44:50 +01:00
for ( int i = child_count - 1 ; i > = 0 ; i - - ) {
2016-08-13 02:56:38 +02:00
tab_container - > set_current_tab ( i ) ;
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-13 02:56:38 +02:00
if ( se ) {
// Maybe there are unsaved changes
if ( se - > is_unsaved ( ) ) {
_ask_close_current_unsaved_tab ( se ) ;
continue ;
}
}
2021-03-04 01:04:31 +01:00
_close_current_tab ( false ) ;
2016-08-13 02:56:38 +02:00
}
}
void ScriptEditor : : _ask_close_current_unsaved_tab ( ScriptEditorBase * current ) {
2018-01-06 14:32:21 +01:00
erase_tab_confirm - > set_text ( TTR ( " Close and save changes? " ) + " \n \" " + current - > get_name ( ) + " \" " ) ;
2020-03-06 18:00:16 +01:00
erase_tab_confirm - > popup_centered ( ) ;
2016-08-13 02:56:38 +02:00
}
2016-07-18 00:18:48 +02:00
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _resave_scripts ( const String & p_str ) {
2014-02-10 02:10:30 +01:00
apply_scripts ( ) ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2018-05-28 17:52:28 +02:00
RES script = se - > get_edited_resource ( ) ;
2014-02-10 02:10:30 +01:00
2020-05-14 16:41:43 +02:00
if ( script - > get_path ( ) = = " " | | script - > get_path ( ) . find ( " local:// " ) ! = - 1 | | script - > get_path ( ) . find ( " :: " ) ! = - 1 ) {
2014-02-10 02:10:30 +01:00
continue ; //internal script, who cares
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2016-04-23 20:21:34 +02:00
if ( trim_trailing_whitespace_on_save ) {
2016-08-03 00:11:05 +02:00
se - > trim_trailing_whitespace ( ) ;
2016-04-23 20:21:34 +02:00
}
2017-04-17 15:46:00 +02:00
2019-05-28 23:27:32 +02:00
se - > insert_final_newline ( ) ;
2017-04-17 15:46:00 +02:00
if ( convert_indent_on_save ) {
if ( use_space_indentation ) {
se - > convert_indent_to_spaces ( ) ;
} else {
se - > convert_indent_to_tabs ( ) ;
}
}
2018-05-28 17:52:28 +02:00
Ref < TextFile > text_file = script ;
2020-04-02 01:20:12 +02:00
if ( text_file ! = nullptr ) {
2018-05-28 17:52:28 +02:00
se - > apply_code ( ) ;
_save_text_file ( text_file , text_file - > get_path ( ) ) ;
break ;
} else {
editor - > save_resource ( script ) ;
}
2016-08-03 00:11:05 +02:00
se - > tag_saved_version ( ) ;
2014-02-10 02:10:30 +01:00
}
disk_changed - > hide ( ) ;
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _reload_scripts ( ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-03 00:11:05 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
}
2018-05-28 17:52:28 +02:00
RES edited_res = se - > get_edited_resource ( ) ;
2014-02-10 02:10:30 +01:00
2018-05-28 17:52:28 +02:00
if ( edited_res - > get_path ( ) = = " " | | edited_res - > get_path ( ) . find ( " local:// " ) ! = - 1 | | edited_res - > get_path ( ) . find ( " :: " ) ! = - 1 ) {
2014-02-10 02:10:30 +01:00
continue ; //internal script, who cares
}
2018-05-28 17:52:28 +02:00
uint64_t last_date = edited_res - > get_last_modified_time ( ) ;
uint64_t date = FileAccess : : get_modified_time ( edited_res - > get_path ( ) ) ;
2016-07-09 17:12:03 +02:00
2017-03-05 16:44:50 +01:00
if ( last_date = = date ) {
2016-07-09 17:12:03 +02:00
continue ;
}
2018-05-28 17:52:28 +02:00
Ref < Script > script = edited_res ;
2020-04-02 01:20:12 +02:00
if ( script ! = nullptr ) {
2021-02-11 18:18:45 +01:00
Ref < Script > rel_script = ResourceLoader : : load ( script - > get_path ( ) , script - > get_class ( ) , ResourceFormatLoader : : CACHE_MODE_IGNORE ) ;
2018-05-28 17:52:28 +02:00
ERR_CONTINUE ( ! rel_script . is_valid ( ) ) ;
script - > set_source_code ( rel_script - > get_source_code ( ) ) ;
script - > set_last_modified_time ( rel_script - > get_last_modified_time ( ) ) ;
script - > reload ( ) ;
}
Ref < TextFile > text_file = edited_res ;
2020-04-02 01:20:12 +02:00
if ( text_file ! = nullptr ) {
2018-05-28 17:52:28 +02:00
Error err ;
Ref < TextFile > rel_text_file = _load_text_file ( text_file - > get_path ( ) , & err ) ;
ERR_CONTINUE ( ! rel_text_file . is_valid ( ) ) ;
text_file - > set_text ( rel_text_file - > get_text ( ) ) ;
text_file - > set_last_modified_time ( rel_text_file - > get_last_modified_time ( ) ) ;
}
2016-08-03 00:11:05 +02:00
se - > reload_text ( ) ;
2014-02-10 02:10:30 +01:00
}
disk_changed - > hide ( ) ;
2015-12-09 13:08:41 +01:00
_update_script_names ( ) ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _res_saved_callback ( const Ref < Resource > & p_res ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-03 00:11:05 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
}
2018-05-28 17:52:28 +02:00
RES script = se - > get_edited_resource ( ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
if ( script - > get_path ( ) = = " " | | script - > get_path ( ) . find ( " local:// " ) ! = - 1 | | script - > get_path ( ) . find ( " :: " ) ! = - 1 ) {
2014-02-10 02:10:30 +01:00
continue ; //internal script, who cares
}
2017-03-05 16:44:50 +01:00
if ( script = = p_res ) {
2016-08-03 00:11:05 +02:00
se - > tag_saved_version ( ) ;
2014-02-10 02:10:30 +01:00
}
}
2015-06-22 05:03:19 +02:00
_update_script_names ( ) ;
2016-06-03 17:34:11 +02:00
if ( ! pending_auto_reload & & auto_reload_running_scripts ) {
call_deferred ( " _live_auto_reload_running_scripts " ) ;
2017-03-05 16:44:50 +01:00
pending_auto_reload = true ;
2016-06-03 17:34:11 +02:00
}
}
void ScriptEditor : : _live_auto_reload_running_scripts ( ) {
2017-03-05 16:44:50 +01:00
pending_auto_reload = false ;
2020-02-07 02:52:05 +01:00
EditorDebuggerNode : : get_singleton ( ) - > reload_scripts ( ) ;
2014-02-10 02:10:30 +01:00
}
2018-05-28 17:52:28 +02:00
bool ScriptEditor : : _test_script_times_on_disk ( RES p_for_script ) {
2014-02-10 02:10:30 +01:00
disk_changed_list - > clear ( ) ;
TreeItem * r = disk_changed_list - > create_item ( ) ;
disk_changed_list - > set_hide_root ( true ) ;
2017-03-05 16:44:50 +01:00
bool need_ask = false ;
bool need_reload = false ;
bool use_autoreload = bool ( EDITOR_DEF ( " text_editor/files/auto_reload_scripts_on_external_change " , false ) ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-03 00:11:05 +02:00
if ( se ) {
2018-05-28 17:52:28 +02:00
RES edited_res = se - > get_edited_resource ( ) ;
2020-05-14 16:41:43 +02:00
if ( p_for_script . is_valid ( ) & & edited_res . is_valid ( ) & & p_for_script ! = edited_res ) {
2016-06-20 03:07:07 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2016-06-20 03:07:07 +02:00
2020-05-14 16:41:43 +02:00
if ( edited_res - > get_path ( ) = = " " | | edited_res - > get_path ( ) . find ( " local:// " ) ! = - 1 | | edited_res - > get_path ( ) . find ( " :: " ) ! = - 1 ) {
2015-12-09 13:08:41 +01:00
continue ; //internal script, who cares
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2018-05-28 17:52:28 +02:00
uint64_t last_date = edited_res - > get_last_modified_time ( ) ;
uint64_t date = FileAccess : : get_modified_time ( edited_res - > get_path ( ) ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
if ( last_date ! = date ) {
2015-12-09 13:08:41 +01:00
TreeItem * ti = disk_changed_list - > create_item ( r ) ;
2018-05-28 17:52:28 +02:00
ti - > set_text ( 0 , edited_res - > get_path ( ) . get_file ( ) ) ;
2014-02-10 02:10:30 +01:00
2016-08-03 00:11:05 +02:00
if ( ! use_autoreload | | se - > is_unsaved ( ) ) {
2017-03-05 16:44:50 +01:00
need_ask = true ;
2015-12-09 13:08:41 +01:00
}
2017-03-05 16:44:50 +01:00
need_reload = true ;
2015-12-09 13:08:41 +01:00
}
2014-02-10 02:10:30 +01:00
}
}
2015-12-09 13:08:41 +01:00
if ( need_reload ) {
if ( ! need_ask ) {
2015-01-10 18:55:12 +01:00
script_editor - > _reload_scripts ( ) ;
2017-03-05 16:44:50 +01:00
need_reload = false ;
2015-02-17 03:58:41 +01:00
} else {
2017-03-05 16:44:50 +01:00
disk_changed - > call_deferred ( " popup_centered_ratio " , 0.5 ) ;
2015-02-17 03:58:41 +01:00
}
}
2014-02-10 02:10:30 +01:00
2015-12-09 13:08:41 +01:00
return need_reload ;
2014-02-10 02:10:30 +01:00
}
2016-04-12 16:45:31 +02:00
void ScriptEditor : : _file_dialog_action ( String p_file ) {
switch ( file_dialog_option ) {
2018-08-15 00:41:23 +02:00
case FILE_NEW_TEXTFILE : {
Error err ;
FileAccess * file = FileAccess : : open ( p_file , FileAccess : : WRITE , & err ) ;
if ( err ) {
editor - > show_warning ( TTR ( " Error writing TextFile: " ) + " \n " + p_file , TTR ( " Error! " ) ) ;
break ;
}
file - > close ( ) ;
memdelete ( file ) ;
2020-02-22 20:47:50 +01:00
[[fallthrough]] ;
2018-08-15 00:41:23 +02:00
}
2018-05-28 17:52:28 +02:00
case FILE_OPEN : {
List < String > extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " Script " , & extensions ) ;
if ( extensions . find ( p_file . get_extension ( ) ) ) {
Ref < Script > scr = ResourceLoader : : load ( p_file ) ;
if ( ! scr . is_valid ( ) ) {
2019-07-31 21:06:15 +02:00
editor - > show_warning ( TTR ( " Could not load file at: " ) + " \n \n " + p_file , TTR ( " Error! " ) ) ;
2018-05-28 17:52:28 +02:00
file_dialog_option = - 1 ;
return ;
}
edit ( scr ) ;
file_dialog_option = - 1 ;
return ;
}
Error error ;
Ref < TextFile > text_file = _load_text_file ( p_file , & error ) ;
if ( error ! = OK ) {
2019-07-31 21:06:15 +02:00
editor - > show_warning ( TTR ( " Could not load file at: " ) + " \n \n " + p_file , TTR ( " Error! " ) ) ;
2018-05-28 17:52:28 +02:00
}
if ( text_file . is_valid ( ) ) {
edit ( text_file ) ;
file_dialog_option = - 1 ;
return ;
}
2018-08-15 00:41:23 +02:00
} break ;
2018-05-28 17:52:28 +02:00
case FILE_SAVE_AS : {
ScriptEditorBase * current = _get_current_editor ( ) ;
2019-11-20 10:09:59 +01:00
if ( current ) {
RES resource = current - > get_edited_resource ( ) ;
String path = ProjectSettings : : get_singleton ( ) - > localize_path ( p_file ) ;
Error err = _save_text_file ( resource , path ) ;
2018-05-28 17:52:28 +02:00
2019-11-20 10:09:59 +01:00
if ( err ! = OK ) {
editor - > show_accept ( TTR ( " Error saving file! " ) , TTR ( " OK " ) ) ;
return ;
}
2018-05-28 17:52:28 +02:00
2019-11-20 10:09:59 +01:00
resource - > set_path ( path ) ;
_update_script_names ( ) ;
2018-05-28 17:52:28 +02:00
}
} break ;
2018-06-21 11:10:43 +02:00
case THEME_SAVE_AS : {
2017-03-05 16:44:50 +01:00
if ( ! EditorSettings : : get_singleton ( ) - > save_text_editor_theme_as ( p_file ) ) {
2018-12-08 17:18:04 +01:00
editor - > show_warning ( TTR ( " Error while saving theme. " ) , TTR ( " Error Saving " ) ) ;
2016-04-12 16:45:31 +02:00
}
} break ;
2018-06-21 11:10:43 +02:00
case THEME_IMPORT : {
2017-03-05 16:44:50 +01:00
if ( ! EditorSettings : : get_singleton ( ) - > import_text_editor_theme ( p_file ) ) {
2018-12-08 17:18:04 +01:00
editor - > show_warning ( TTR ( " Error importing theme. " ) , TTR ( " Error Importing " ) ) ;
2016-04-12 16:45:31 +02:00
}
} break ;
}
file_dialog_option = - 1 ;
}
2017-07-06 09:18:20 +02:00
Ref < Script > ScriptEditor : : _get_current_script ( ) {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * current = _get_current_editor ( ) ;
2017-07-06 09:18:20 +02:00
if ( current ) {
2018-05-28 17:52:28 +02:00
Ref < Script > script = current - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
return script ! = nullptr ? script : nullptr ;
2017-07-06 09:18:20 +02:00
} else {
2020-04-02 01:20:12 +02:00
return nullptr ;
2017-07-06 09:18:20 +02:00
}
}
2017-07-19 19:47:43 +02:00
Array ScriptEditor : : _get_open_scripts ( ) const {
2017-07-06 09:18:20 +02:00
Array ret ;
2020-03-17 07:33:00 +01:00
Vector < Ref < Script > > scripts = get_open_scripts ( ) ;
2017-07-06 09:18:20 +02:00
int scrits_amount = scripts . size ( ) ;
for ( int idx_script = 0 ; idx_script < scrits_amount ; idx_script + + ) {
ret . push_back ( scripts [ idx_script ] ) ;
}
return ret ;
}
2019-12-18 20:32:25 +01:00
bool ScriptEditor : : toggle_scripts_panel ( ) {
2019-12-15 14:35:16 +01:00
list_split - > set_visible ( ! list_split - > is_visible ( ) ) ;
return list_split - > is_visible ( ) ;
}
bool ScriptEditor : : is_scripts_panel_toggled ( ) {
return list_split - > is_visible ( ) ;
}
2014-02-10 02:10:30 +01:00
void ScriptEditor : : _menu_option ( int p_option ) {
2019-12-17 14:51:49 +01:00
ScriptEditorBase * current = _get_current_editor ( ) ;
2017-03-05 16:44:50 +01:00
switch ( p_option ) {
2015-08-10 01:39:59 +02:00
case FILE_NEW : {
2020-01-08 23:43:55 +01:00
script_create_dialog - > config ( " Node " , " new_script " , false , false ) ;
2020-01-08 05:15:05 +01:00
script_create_dialog - > popup_centered ( ) ;
2015-08-10 01:39:59 +02:00
} break ;
2018-08-15 00:41:23 +02:00
case FILE_NEW_TEXTFILE : {
2020-03-06 18:00:16 +01:00
file_dialog - > set_file_mode ( EditorFileDialog : : FILE_MODE_SAVE_FILE ) ;
2018-08-15 00:41:23 +02:00
file_dialog - > set_access ( EditorFileDialog : : ACCESS_FILESYSTEM ) ;
file_dialog_option = FILE_NEW_TEXTFILE ;
file_dialog - > clear_filters ( ) ;
2020-07-11 18:45:19 +02:00
file_dialog - > popup_file_dialog ( ) ;
2019-08-12 22:23:00 +02:00
file_dialog - > set_title ( TTR ( " New Text File... " ) ) ;
2018-08-15 00:41:23 +02:00
} break ;
2015-11-17 13:46:08 +01:00
case FILE_OPEN : {
2020-03-06 18:00:16 +01:00
file_dialog - > set_file_mode ( EditorFileDialog : : FILE_MODE_OPEN_FILE ) ;
2018-05-28 17:52:28 +02:00
file_dialog - > set_access ( EditorFileDialog : : ACCESS_FILESYSTEM ) ;
file_dialog_option = FILE_OPEN ;
2014-02-10 02:10:30 +01:00
2018-05-28 17:52:28 +02:00
List < String > extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " Script " , & extensions ) ;
file_dialog - > clear_filters ( ) ;
for ( int i = 0 ; i < extensions . size ( ) ; i + + ) {
file_dialog - > add_filter ( " *. " + extensions [ i ] + " ; " + extensions [ i ] . to_upper ( ) ) ;
}
2020-07-11 18:45:19 +02:00
file_dialog - > popup_file_dialog ( ) ;
2018-05-28 17:52:28 +02:00
file_dialog - > set_title ( TTR ( " Open File " ) ) ;
2015-11-17 13:46:08 +01:00
return ;
2014-02-10 02:10:30 +01:00
} break ;
2019-07-31 21:06:15 +02:00
case FILE_REOPEN_CLOSED : {
2020-12-15 13:04:21 +01:00
if ( previous_scripts . is_empty ( ) ) {
2019-07-31 21:06:15 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2019-07-31 21:06:15 +02:00
String path = previous_scripts . back ( ) - > get ( ) ;
previous_scripts . pop_back ( ) ;
List < String > extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " Script " , & extensions ) ;
bool built_in = ! path . is_resource_file ( ) ;
if ( extensions . find ( path . get_extension ( ) ) | | built_in ) {
if ( built_in ) {
2019-09-04 20:00:09 +02:00
String res_path = path . get_slice ( " :: " , 0 ) ;
if ( ResourceLoader : : get_resource_type ( res_path ) = = " PackedScene " ) {
if ( ! EditorNode : : get_singleton ( ) - > is_scene_open ( res_path ) ) {
EditorNode : : get_singleton ( ) - > load_scene ( res_path ) ;
script_editor - > call_deferred ( " _menu_option " , p_option ) ;
previous_scripts . push_back ( path ) ; //repeat the operation
return ;
}
} else {
EditorNode : : get_singleton ( ) - > load_resource ( res_path ) ;
2019-07-31 21:06:15 +02:00
}
}
Ref < Script > scr = ResourceLoader : : load ( path ) ;
if ( ! scr . is_valid ( ) ) {
editor - > show_warning ( TTR ( " Could not load file at: " ) + " \n \n " + path , TTR ( " Error! " ) ) ;
file_dialog_option = - 1 ;
return ;
}
edit ( scr ) ;
file_dialog_option = - 1 ;
return ;
} else {
Error error ;
Ref < TextFile > text_file = _load_text_file ( path , & error ) ;
2020-05-14 16:41:43 +02:00
if ( error ! = OK ) {
2019-07-31 21:06:15 +02:00
editor - > show_warning ( TTR ( " Could not load file at: " ) + " \n \n " + path , TTR ( " Error! " ) ) ;
2020-05-14 16:41:43 +02:00
}
2019-07-31 21:06:15 +02:00
if ( text_file . is_valid ( ) ) {
edit ( text_file ) ;
file_dialog_option = - 1 ;
return ;
}
}
} break ;
2014-02-10 02:10:30 +01:00
case FILE_SAVE_ALL : {
2020-05-14 16:41:43 +02:00
if ( _test_script_times_on_disk ( ) ) {
2014-02-10 02:10:30 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2016-02-03 01:10:52 +01:00
save_all_scripts ( ) ;
2014-02-10 02:10:30 +01:00
} break ;
2019-05-03 01:07:06 +02:00
case SEARCH_IN_FILES : {
_on_find_in_files_requested ( " " ) ;
} break ;
2020-02-09 10:10:58 +01:00
case REPLACE_IN_FILES : {
_on_replace_in_files_requested ( " " ) ;
} break ;
2015-11-17 13:46:08 +01:00
case SEARCH_HELP : {
2018-10-02 12:07:44 +02:00
help_search_dialog - > popup_dialog ( ) ;
2014-02-10 02:10:30 +01:00
} break ;
2015-11-17 13:46:08 +01:00
case SEARCH_WEBSITE : {
2018-11-05 08:46:27 +01:00
OS : : get_singleton ( ) - > shell_open ( " https://docs.godotengine.org/ " ) ;
2014-02-10 02:10:30 +01:00
} break ;
2015-11-17 13:46:08 +01:00
case WINDOW_NEXT : {
_history_forward ( ) ;
2014-02-10 02:10:30 +01:00
} break ;
2015-11-17 13:46:08 +01:00
case WINDOW_PREV : {
_history_back ( ) ;
2014-02-10 02:10:30 +01:00
} break ;
2017-11-12 03:42:56 +01:00
case WINDOW_SORT : {
_sort_list_on_update = true ;
_update_script_names ( ) ;
} break ;
2017-07-10 14:11:31 +02:00
case TOGGLE_SCRIPTS_PANEL : {
2019-12-17 14:51:49 +01:00
if ( current ) {
2019-12-18 20:32:25 +01:00
ScriptTextEditor * editor = Object : : cast_to < ScriptTextEditor > ( current ) ;
toggle_scripts_panel ( ) ;
2019-12-17 14:51:49 +01:00
if ( editor ) {
2019-12-18 20:32:25 +01:00
editor - > update_toggle_scripts_button ( ) ;
2019-12-17 14:51:49 +01:00
}
} else {
2019-12-18 20:32:25 +01:00
toggle_scripts_panel ( ) ;
2019-12-17 14:51:49 +01:00
}
2017-06-25 11:20:01 +02:00
}
2015-11-17 13:46:08 +01:00
}
2014-02-10 02:10:30 +01:00
2015-11-17 13:46:08 +01:00
if ( current ) {
2017-03-05 16:44:50 +01:00
switch ( p_option ) {
2015-11-17 13:46:08 +01:00
case FILE_SAVE : {
2021-05-09 12:29:50 +02:00
save_current_script ( ) ;
2015-11-17 13:46:08 +01:00
} break ;
case FILE_SAVE_AS : {
2020-05-14 16:41:43 +02:00
if ( trim_trailing_whitespace_on_save ) {
2019-05-28 23:27:32 +02:00
current - > trim_trailing_whitespace ( ) ;
2020-05-14 16:41:43 +02:00
}
2019-05-28 23:27:32 +02:00
current - > insert_final_newline ( ) ;
2017-04-17 15:46:00 +02:00
if ( convert_indent_on_save ) {
if ( use_space_indentation ) {
current - > convert_indent_to_spaces ( ) ;
} else {
current - > convert_indent_to_tabs ( ) ;
}
}
2018-05-28 17:52:28 +02:00
2019-11-20 10:09:59 +01:00
RES resource = current - > get_edited_resource ( ) ;
Ref < TextFile > text_file = resource ;
2020-11-29 04:42:06 +01:00
Ref < Script > script = resource ;
2020-04-02 01:20:12 +02:00
if ( text_file ! = nullptr ) {
2020-03-06 18:00:16 +01:00
file_dialog - > set_file_mode ( EditorFileDialog : : FILE_MODE_SAVE_FILE ) ;
2018-05-28 17:52:28 +02:00
file_dialog - > set_access ( EditorFileDialog : : ACCESS_FILESYSTEM ) ;
file_dialog_option = FILE_SAVE_AS ;
List < String > extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " Script " , & extensions ) ;
file_dialog - > clear_filters ( ) ;
file_dialog - > set_current_dir ( text_file - > get_path ( ) . get_base_dir ( ) ) ;
file_dialog - > set_current_file ( text_file - > get_path ( ) . get_file ( ) ) ;
2020-07-11 18:45:19 +02:00
file_dialog - > popup_file_dialog ( ) ;
2018-05-28 17:52:28 +02:00
file_dialog - > set_title ( TTR ( " Save File As... " ) ) ;
break ;
}
2020-11-29 04:42:06 +01:00
if ( script ! = nullptr ) {
const Vector < DocData : : ClassDoc > & documentations = script - > get_documentation ( ) ;
for ( int j = 0 ; j < documentations . size ( ) ; j + + ) {
const DocData : : ClassDoc & doc = documentations . get ( j ) ;
if ( EditorHelp : : get_doc_data ( ) - > has_doc ( doc . name ) ) {
EditorHelp : : get_doc_data ( ) - > remove_doc ( doc . name ) ;
}
}
}
2019-11-20 10:09:59 +01:00
editor - > push_item ( resource . ptr ( ) ) ;
editor - > save_resource_as ( resource ) ;
2015-11-17 13:46:08 +01:00
2020-11-29 04:42:06 +01:00
if ( script ! = nullptr ) {
const Vector < DocData : : ClassDoc > & documentations = script - > get_documentation ( ) ;
for ( int j = 0 ; j < documentations . size ( ) ; j + + ) {
const DocData : : ClassDoc & doc = documentations . get ( j ) ;
EditorHelp : : get_doc_data ( ) - > add_doc ( doc ) ;
update_doc ( doc . name ) ;
}
}
2015-11-17 13:46:08 +01:00
} break ;
2016-06-09 01:00:02 +02:00
case FILE_TOOL_RELOAD :
case FILE_TOOL_RELOAD_SOFT : {
2017-03-05 16:44:50 +01:00
current - > reload ( p_option = = FILE_TOOL_RELOAD_SOFT ) ;
2016-07-19 00:24:38 +02:00
2016-04-23 20:21:34 +02:00
} break ;
2017-08-28 00:03:19 +02:00
case FILE_RUN : {
2018-05-28 17:52:28 +02:00
Ref < Script > scr = current - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
if ( scr = = nullptr | | scr . is_null ( ) ) {
2019-12-21 07:57:15 +01:00
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Can't obtain the script for running. " ) ) ;
2017-08-28 00:03:19 +02:00
break ;
}
2017-09-03 19:53:17 +02:00
current - > apply_code ( ) ;
Error err = scr - > reload ( false ) ; //hard reload script before running always
if ( err ! = OK ) {
2019-12-21 07:57:15 +01:00
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Script failed reloading, check console for errors. " ) ) ;
2017-09-03 19:53:17 +02:00
return ;
}
2017-08-28 00:03:19 +02:00
if ( ! scr - > is_tool ( ) ) {
2019-12-21 07:57:15 +01:00
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Script is not in tool mode, will not be able to run. " ) ) ;
2017-08-28 00:03:19 +02:00
return ;
}
if ( ! ClassDB : : is_parent_class ( scr - > get_instance_base_type ( ) , " EditorScript " ) ) {
2019-12-21 07:57:15 +01:00
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " To run this script, it must inherit EditorScript and be set to tool mode. " ) ) ;
2017-08-28 00:03:19 +02:00
return ;
}
Ref < EditorScript > es = memnew ( EditorScript ) ;
2020-02-13 20:03:10 +01:00
es - > set_script ( scr ) ;
2017-08-28 00:03:19 +02:00
es - > set_editor ( EditorNode : : get_singleton ( ) ) ;
es - > _run ( ) ;
EditorNode : : get_undo_redo ( ) - > clear_history ( ) ;
} break ;
2016-08-03 00:11:05 +02:00
case FILE_CLOSE : {
if ( current - > is_unsaved ( ) ) {
2016-08-13 02:56:38 +02:00
_ask_close_current_unsaved_tab ( current ) ;
2016-05-27 16:29:04 +02:00
} else {
2021-03-04 01:04:31 +01:00
_close_current_tab ( false ) ;
2016-05-27 16:29:04 +02:00
}
} break ;
2017-12-07 04:57:41 +01:00
case FILE_COPY_PATH : {
_copy_script_path ( ) ;
} break ;
2018-01-02 03:11:54 +01:00
case SHOW_IN_FILE_SYSTEM : {
2019-12-30 02:49:52 +01:00
const RES script = current - > get_edited_resource ( ) ;
const String path = script - > get_path ( ) ;
2020-12-15 13:04:21 +01:00
if ( ! path . is_empty ( ) ) {
2019-12-30 02:49:52 +01:00
FileSystemDock * file_system_dock = EditorNode : : get_singleton ( ) - > get_filesystem_dock ( ) ;
file_system_dock - > navigate_to_path ( path ) ;
// Ensure that the FileSystem dock is visible.
TabContainer * tab_container = ( TabContainer * ) file_system_dock - > get_parent_control ( ) ;
2020-04-06 01:06:10 +02:00
tab_container - > set_current_tab ( file_system_dock - > get_index ( ) ) ;
2019-12-30 02:49:52 +01:00
}
2018-01-02 03:11:54 +01:00
} break ;
2016-08-03 00:11:05 +02:00
case CLOSE_DOCS : {
_close_docs_tab ( ) ;
2016-05-27 16:29:04 +02:00
} break ;
2017-11-20 06:15:40 +01:00
case CLOSE_OTHER_TABS : {
_close_other_tabs ( ) ;
} break ;
2016-08-13 02:56:38 +02:00
case CLOSE_ALL : {
_close_all_tabs ( ) ;
} break ;
2017-11-12 03:42:56 +01:00
case WINDOW_MOVE_UP : {
2017-03-05 16:44:50 +01:00
if ( tab_container - > get_current_tab ( ) > 0 ) {
tab_container - > move_child ( current , tab_container - > get_current_tab ( ) - 1 ) ;
2017-11-12 03:42:56 +01:00
tab_container - > set_current_tab ( tab_container - > get_current_tab ( ) - 1 ) ;
2015-11-17 13:46:08 +01:00
_update_script_names ( ) ;
}
} break ;
2017-11-12 03:42:56 +01:00
case WINDOW_MOVE_DOWN : {
2017-03-05 16:44:50 +01:00
if ( tab_container - > get_current_tab ( ) < tab_container - > get_child_count ( ) - 1 ) {
tab_container - > move_child ( current , tab_container - > get_current_tab ( ) + 1 ) ;
2017-11-12 03:42:56 +01:00
tab_container - > set_current_tab ( tab_container - > get_current_tab ( ) + 1 ) ;
2015-11-17 13:46:08 +01:00
_update_script_names ( ) ;
}
} break ;
default : {
2017-03-05 16:44:50 +01:00
if ( p_option > = WINDOW_SELECT_BASE ) {
tab_container - > set_current_tab ( p_option - WINDOW_SELECT_BASE ) ;
2019-06-03 21:57:06 +02:00
_update_script_names ( ) ;
2015-11-17 13:46:08 +01:00
}
2014-02-10 02:10:30 +01:00
}
2015-11-17 13:46:08 +01:00
}
2017-04-17 02:41:24 +02:00
} else {
2017-08-24 22:58:51 +02:00
EditorHelp * help = Object : : cast_to < EditorHelp > ( tab_container - > get_current_tab_control ( ) ) ;
2017-04-17 02:41:24 +02:00
if ( help ) {
switch ( p_option ) {
case HELP_SEARCH_FIND : {
help - > popup_search ( ) ;
} break ;
case HELP_SEARCH_FIND_NEXT : {
help - > search_again ( ) ;
} break ;
2019-10-09 17:41:49 +02:00
case HELP_SEARCH_FIND_PREVIOUS : {
help - > search_again ( true ) ;
} break ;
2017-04-17 02:41:24 +02:00
case FILE_CLOSE : {
_close_current_tab ( ) ;
} break ;
case CLOSE_DOCS : {
_close_docs_tab ( ) ;
} break ;
2017-11-20 06:15:40 +01:00
case CLOSE_OTHER_TABS : {
_close_other_tabs ( ) ;
} break ;
2017-04-17 02:41:24 +02:00
case CLOSE_ALL : {
_close_all_tabs ( ) ;
} break ;
2017-11-12 03:42:56 +01:00
case WINDOW_MOVE_UP : {
if ( tab_container - > get_current_tab ( ) > 0 ) {
tab_container - > move_child ( help , tab_container - > get_current_tab ( ) - 1 ) ;
tab_container - > set_current_tab ( tab_container - > get_current_tab ( ) - 1 ) ;
_update_script_names ( ) ;
}
} break ;
case WINDOW_MOVE_DOWN : {
if ( tab_container - > get_current_tab ( ) < tab_container - > get_child_count ( ) - 1 ) {
tab_container - > move_child ( help , tab_container - > get_current_tab ( ) + 1 ) ;
tab_container - > set_current_tab ( tab_container - > get_current_tab ( ) + 1 ) ;
_update_script_names ( ) ;
}
} break ;
2017-04-17 02:41:24 +02:00
}
2014-02-10 02:10:30 +01:00
}
}
}
2018-06-21 11:10:43 +02:00
void ScriptEditor : : _theme_option ( int p_option ) {
switch ( p_option ) {
case THEME_IMPORT : {
2020-03-06 18:00:16 +01:00
file_dialog - > set_file_mode ( EditorFileDialog : : FILE_MODE_OPEN_FILE ) ;
2018-06-21 11:10:43 +02:00
file_dialog - > set_access ( EditorFileDialog : : ACCESS_FILESYSTEM ) ;
file_dialog_option = THEME_IMPORT ;
file_dialog - > clear_filters ( ) ;
file_dialog - > add_filter ( " *.tet " ) ;
2020-07-11 18:45:19 +02:00
file_dialog - > popup_file_dialog ( ) ;
2018-06-21 11:10:43 +02:00
file_dialog - > set_title ( TTR ( " Import Theme " ) ) ;
} break ;
case THEME_RELOAD : {
EditorSettings : : get_singleton ( ) - > load_text_editor_theme ( ) ;
} break ;
case THEME_SAVE : {
2019-05-23 17:18:24 +02:00
if ( EditorSettings : : get_singleton ( ) - > is_default_text_editor_theme ( ) ) {
ScriptEditor : : _show_save_theme_as_dialog ( ) ;
} else if ( ! EditorSettings : : get_singleton ( ) - > save_text_editor_theme ( ) ) {
2018-06-21 11:10:43 +02:00
editor - > show_warning ( TTR ( " Error while saving theme " ) , TTR ( " Error saving " ) ) ;
}
} break ;
case THEME_SAVE_AS : {
2019-05-23 17:18:24 +02:00
ScriptEditor : : _show_save_theme_as_dialog ( ) ;
2018-06-21 11:10:43 +02:00
} break ;
}
}
2019-05-23 17:18:24 +02:00
void ScriptEditor : : _show_save_theme_as_dialog ( ) {
2020-03-06 18:00:16 +01:00
file_dialog - > set_file_mode ( EditorFileDialog : : FILE_MODE_SAVE_FILE ) ;
2019-05-23 17:18:24 +02:00
file_dialog - > set_access ( EditorFileDialog : : ACCESS_FILESYSTEM ) ;
file_dialog_option = THEME_SAVE_AS ;
file_dialog - > clear_filters ( ) ;
file_dialog - > add_filter ( " *.tet " ) ;
file_dialog - > set_current_path ( EditorSettings : : get_singleton ( ) - > get_text_editor_themes_dir ( ) . plus_file ( EditorSettings : : get_singleton ( ) - > get ( " text_editor/theme/color_theme " ) ) ) ;
2020-07-11 18:45:19 +02:00
file_dialog - > popup_file_dialog ( ) ;
2019-05-23 17:18:24 +02:00
file_dialog - > set_title ( TTR ( " Save Theme As... " ) ) ;
}
2014-02-10 02:10:30 +01:00
void ScriptEditor : : _tab_changed ( int p_which ) {
ensure_select_current ( ) ;
}
void ScriptEditor : : _notification ( int p_what ) {
2017-07-26 16:02:53 +02:00
switch ( p_what ) {
case NOTIFICATION_ENTER_TREE : {
2020-02-21 18:28:45 +01:00
editor - > connect ( " stop_pressed " , callable_mp ( this , & ScriptEditor : : _editor_stop ) ) ;
editor - > connect ( " script_add_function_request " , callable_mp ( this , & ScriptEditor : : _add_callback ) ) ;
editor - > connect ( " resource_saved " , callable_mp ( this , & ScriptEditor : : _res_saved_callback ) ) ;
2020-07-23 14:21:28 +02:00
editor - > get_filesystem_dock ( ) - > connect ( " file_removed " , callable_mp ( this , & ScriptEditor : : _file_removed ) ) ;
2020-02-21 18:28:45 +01:00
script_list - > connect ( " item_selected " , callable_mp ( this , & ScriptEditor : : _script_selected ) ) ;
2017-11-20 06:15:40 +01:00
2020-02-21 18:28:45 +01:00
members_overview - > connect ( " item_selected " , callable_mp ( this , & ScriptEditor : : _members_overview_selected ) ) ;
help_overview - > connect ( " item_selected " , callable_mp ( this , & ScriptEditor : : _help_overview_selected ) ) ;
script_split - > connect ( " dragged " , callable_mp ( this , & ScriptEditor : : _script_split_dragged ) ) ;
2015-05-05 04:32:40 +02:00
2020-02-21 18:28:45 +01:00
EditorSettings : : get_singleton ( ) - > connect ( " settings_changed " , callable_mp ( this , & ScriptEditor : : _editor_settings_changed ) ) ;
2020-07-23 14:21:28 +02:00
EditorFileSystem : : get_singleton ( ) - > connect ( " filesystem_changed " , callable_mp ( this , & ScriptEditor : : _filesystem_changed ) ) ;
2020-02-22 20:47:50 +01:00
[[fallthrough]] ;
2019-08-08 18:39:53 +02:00
}
2020-09-03 13:22:16 +02:00
case NOTIFICATION_TRANSLATION_CHANGED :
case NOTIFICATION_LAYOUT_DIRECTION_CHANGED :
2019-08-08 18:39:53 +02:00
case NOTIFICATION_THEME_CHANGED : {
2020-03-12 13:37:40 +01:00
help_search - > set_icon ( get_theme_icon ( " HelpSearch " , " EditorIcons " ) ) ;
site_search - > set_icon ( get_theme_icon ( " Instance " , " EditorIcons " ) ) ;
2014-02-10 02:10:30 +01:00
2020-09-03 13:22:16 +02:00
if ( is_layout_rtl ( ) ) {
script_forward - > set_icon ( get_theme_icon ( " Back " , " EditorIcons " ) ) ;
script_back - > set_icon ( get_theme_icon ( " Forward " , " EditorIcons " ) ) ;
} else {
script_forward - > set_icon ( get_theme_icon ( " Forward " , " EditorIcons " ) ) ;
script_back - > set_icon ( get_theme_icon ( " Back " , " EditorIcons " ) ) ;
}
2019-08-08 18:39:53 +02:00
2020-03-12 13:37:40 +01:00
members_overview_alphabeta_sort_button - > set_icon ( get_theme_icon ( " Sort " , " EditorIcons " ) ) ;
2019-08-08 18:39:53 +02:00
2020-03-12 13:37:40 +01:00
filter_scripts - > set_right_icon ( get_theme_icon ( " Search " , " EditorIcons " ) ) ;
filter_methods - > set_right_icon ( get_theme_icon ( " Search " , " EditorIcons " ) ) ;
2019-08-08 18:39:53 +02:00
2020-03-12 13:37:40 +01:00
filename - > add_theme_style_override ( " normal " , editor - > get_gui_base ( ) - > get_theme_stylebox ( " normal " , " LineEdit " ) ) ;
2019-08-08 18:39:53 +02:00
recent_scripts - > set_as_minsize ( ) ;
2021-05-20 07:55:32 +02:00
2021-05-26 18:28:38 +02:00
if ( is_inside_tree ( ) ) {
_update_script_colors ( ) ;
_update_script_names ( ) ;
}
2017-07-26 16:02:53 +02:00
} break ;
2014-02-10 02:10:30 +01:00
2017-07-26 16:02:53 +02:00
case NOTIFICATION_READY : {
2020-02-21 18:28:45 +01:00
get_tree ( ) - > connect ( " tree_changed " , callable_mp ( this , & ScriptEditor : : _tree_changed ) ) ;
editor - > get_inspector_dock ( ) - > connect ( " request_help " , callable_mp ( this , & ScriptEditor : : _help_class_open ) ) ;
editor - > connect ( " request_help_search " , callable_mp ( this , & ScriptEditor : : _help_search ) ) ;
2017-07-26 16:02:53 +02:00
} break ;
2014-02-10 02:10:30 +01:00
2017-07-26 16:02:53 +02:00
case NOTIFICATION_EXIT_TREE : {
2020-02-21 18:28:45 +01:00
editor - > disconnect ( " stop_pressed " , callable_mp ( this , & ScriptEditor : : _editor_stop ) ) ;
2017-07-26 16:02:53 +02:00
} break ;
2014-02-10 02:10:30 +01:00
2020-06-30 01:47:18 +02:00
case NOTIFICATION_WM_WINDOW_FOCUS_IN : {
2017-07-26 16:02:53 +02:00
_test_script_times_on_disk ( ) ;
_update_modified_scripts_for_external_editor ( ) ;
} break ;
2018-02-12 02:36:15 +01:00
case CanvasItem : : NOTIFICATION_VISIBILITY_CHANGED : {
if ( is_visible ( ) ) {
find_in_files_button - > show ( ) ;
} else {
2018-10-27 18:49:05 +02:00
if ( find_in_files - > is_visible_in_tree ( ) ) {
editor - > hide_bottom_panel ( ) ;
}
2018-02-12 02:36:15 +01:00
find_in_files_button - > hide ( ) ;
}
} break ;
2017-07-26 16:02:53 +02:00
default :
break ;
2014-02-10 02:10:30 +01:00
}
}
2016-08-25 22:45:20 +02:00
bool ScriptEditor : : can_take_away_focus ( ) const {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * current = _get_current_editor ( ) ;
2020-05-14 16:41:43 +02:00
if ( current ) {
2018-02-12 02:36:15 +01:00
return current - > can_lose_focus_on_node_selection ( ) ;
2020-05-14 16:41:43 +02:00
} else {
2016-08-25 22:45:20 +02:00
return true ;
2020-05-14 16:41:43 +02:00
}
2016-08-25 22:45:20 +02:00
}
2016-07-07 01:35:49 +02:00
2017-03-05 16:44:50 +01:00
void ScriptEditor : : close_builtin_scripts_from_scene ( const String & p_scene ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-07-07 01:35:49 +02:00
2016-08-03 00:11:05 +02:00
if ( se ) {
2018-05-28 17:52:28 +02:00
Ref < Script > script = se - > get_edited_resource ( ) ;
2020-05-14 16:41:43 +02:00
if ( script = = nullptr | | ! script . is_valid ( ) ) {
2016-07-07 01:35:49 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2016-07-07 01:35:49 +02:00
2017-03-05 16:44:50 +01:00
if ( script - > get_path ( ) . find ( " :: " ) ! = - 1 & & script - > get_path ( ) . begins_with ( p_scene ) ) { //is an internal script and belongs to scene being closed
2016-07-07 01:35:49 +02:00
_close_tab ( i ) ;
i - - ;
}
}
}
}
2015-12-09 13:08:41 +01:00
void ScriptEditor : : edited_scene_changed ( ) {
_update_modified_scripts_for_external_editor ( ) ;
}
2014-02-10 02:10:30 +01:00
2017-07-06 09:18:20 +02:00
void ScriptEditor : : notify_script_close ( const Ref < Script > & p_script ) {
emit_signal ( " script_close " , p_script ) ;
}
void ScriptEditor : : notify_script_changed ( const Ref < Script > & p_script ) {
2017-07-19 19:47:43 +02:00
emit_signal ( " editor_script_changed " , p_script ) ;
2017-07-06 09:18:20 +02:00
}
2014-02-10 02:10:30 +01:00
void ScriptEditor : : get_breakpoints ( List < String > * p_breakpoints ) {
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2018-05-28 17:52:28 +02:00
Ref < Script > script = se - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
if ( script = = nullptr ) {
2018-05-28 17:52:28 +02:00
continue ;
}
2014-02-10 02:10:30 +01:00
String base = script - > get_path ( ) ;
2020-07-23 19:51:17 +02:00
if ( base . begins_with ( " local:// " ) | | base = = " " ) {
continue ;
}
2014-02-10 02:10:30 +01:00
2020-07-26 16:57:23 +02:00
Array bpoints = se - > get_breakpoints ( ) ;
for ( int j = 0 ; j < bpoints . size ( ) ; j + + ) {
p_breakpoints - > push_back ( base + " : " + itos ( ( int ) bpoints [ j ] + 1 ) ) ;
2014-02-10 02:10:30 +01:00
}
}
}
2017-05-28 16:20:38 +02:00
void ScriptEditor : : _members_overview_selected ( int p_idx ) {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * se = _get_current_editor ( ) ;
2017-05-28 16:20:38 +02:00
if ( ! se ) {
return ;
}
2018-02-09 18:35:28 +01:00
// Go to the member's line and reset the cursor column. We can't change scroll_position
// directly until we have gone to the line first, since code might be folded.
2017-12-13 06:40:12 +01:00
se - > goto_line ( members_overview - > get_item_metadata ( p_idx ) ) ;
Dictionary state = se - > get_edit_state ( ) ;
2017-10-08 15:29:27 +02:00
state [ " column " ] = 0 ;
2018-02-09 18:35:28 +01:00
state [ " scroll_position " ] = members_overview - > get_item_metadata ( p_idx ) ;
2017-10-08 15:29:27 +02:00
se - > set_edit_state ( state ) ;
2017-05-28 16:20:38 +02:00
}
2017-09-14 02:56:37 +02:00
void ScriptEditor : : _help_overview_selected ( int p_idx ) {
Node * current = tab_container - > get_child ( tab_container - > get_current_tab ( ) ) ;
EditorHelp * se = Object : : cast_to < EditorHelp > ( current ) ;
if ( ! se ) {
return ;
}
se - > scroll_to_section ( help_overview - > get_item_metadata ( p_idx ) ) ;
}
2015-06-22 05:03:19 +02:00
void ScriptEditor : : _script_selected ( int p_idx ) {
2020-04-28 15:19:37 +02:00
grab_focus_block = ! Input : : get_singleton ( ) - > is_mouse_button_pressed ( 1 ) ; //amazing hack, simply amazing
2015-11-17 13:46:08 +01:00
_go_to_tab ( script_list - > get_item_metadata ( p_idx ) ) ;
2017-03-05 16:44:50 +01:00
grab_focus_block = false ;
2015-06-22 05:03:19 +02:00
}
2014-02-10 02:10:30 +01:00
void ScriptEditor : : ensure_select_current ( ) {
2017-03-05 16:44:50 +01:00
if ( tab_container - > get_child_count ( ) & & tab_container - > get_current_tab ( ) > = 0 ) {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * se = _get_current_editor ( ) ;
2016-08-03 00:11:05 +02:00
if ( se ) {
2019-11-20 10:09:59 +01:00
se - > enable_editor ( ) ;
2020-05-14 16:41:43 +02:00
if ( ! grab_focus_block & & is_visible_in_tree ( ) ) {
2016-08-03 00:11:05 +02:00
se - > ensure_focus ( ) ;
2020-05-14 16:41:43 +02:00
}
2015-11-17 13:46:08 +01:00
}
2015-06-22 05:03:19 +02:00
}
2016-08-03 00:11:05 +02:00
_update_selected_editor_menu ( ) ;
2015-06-22 05:03:19 +02:00
}
2020-03-17 07:33:00 +01:00
void ScriptEditor : : _find_scripts ( Node * p_base , Node * p_current , Set < Ref < Script > > & used ) {
2020-05-14 16:41:43 +02:00
if ( p_current ! = p_base & & p_current - > get_owner ( ) ! = p_base ) {
2015-06-22 05:03:19 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2015-06-22 05:03:19 +02:00
if ( p_current - > get_script_instance ( ) ) {
Ref < Script > scr = p_current - > get_script ( ) ;
2020-05-14 16:41:43 +02:00
if ( scr . is_valid ( ) ) {
2015-06-22 05:03:19 +02:00
used . insert ( scr ) ;
2020-05-14 16:41:43 +02:00
}
2015-06-22 05:03:19 +02:00
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < p_current - > get_child_count ( ) ; i + + ) {
_find_scripts ( p_base , p_current - > get_child ( i ) , used ) ;
2015-06-22 05:03:19 +02:00
}
}
2015-11-17 13:46:08 +01:00
struct _ScriptEditorItemData {
String name ;
2017-02-02 13:31:01 +01:00
String sort_key ;
2019-06-11 20:43:37 +02:00
Ref < Texture2D > icon ;
2020-11-24 10:12:55 +01:00
int index = 0 ;
2015-11-17 13:46:08 +01:00
String tooltip ;
2020-11-24 10:12:55 +01:00
bool used = false ;
int category = 0 ;
Node * ref = nullptr ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
bool operator < ( const _ScriptEditorItemData & id ) const {
2019-05-22 22:38:14 +02:00
if ( category = = id . category ) {
if ( sort_key = = id . sort_key ) {
return index < id . index ;
} else {
2020-04-11 13:17:46 +02:00
return sort_key . naturalnocasecmp_to ( id . sort_key ) < 0 ;
2019-05-22 22:38:14 +02:00
}
} else {
return category < id . category ;
}
2015-11-17 13:46:08 +01:00
}
} ;
2017-05-28 16:20:38 +02:00
void ScriptEditor : : _update_members_overview_visibility ( ) {
2018-02-12 02:36:15 +01:00
ScriptEditorBase * se = _get_current_editor ( ) ;
2017-05-28 16:20:38 +02:00
if ( ! se ) {
2018-06-05 23:18:11 +02:00
members_overview_alphabeta_sort_button - > set_visible ( false ) ;
2017-05-28 16:20:38 +02:00
members_overview - > set_visible ( false ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > set_visible ( false ) ;
2017-05-28 16:20:38 +02:00
return ;
}
if ( members_overview_enabled & & se - > show_members_overview ( ) ) {
2018-06-05 23:18:11 +02:00
members_overview_alphabeta_sort_button - > set_visible ( true ) ;
2017-05-28 16:20:38 +02:00
members_overview - > set_visible ( true ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > set_visible ( true ) ;
2017-05-28 16:20:38 +02:00
} else {
2018-06-05 23:18:11 +02:00
members_overview_alphabeta_sort_button - > set_visible ( false ) ;
2017-05-28 16:20:38 +02:00
members_overview - > set_visible ( false ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > set_visible ( false ) ;
2017-05-28 16:20:38 +02:00
}
}
2018-04-30 14:27:00 +02:00
void ScriptEditor : : _toggle_members_overview_alpha_sort ( bool p_alphabetic_sort ) {
EditorSettings : : get_singleton ( ) - > set ( " text_editor/tools/sort_members_outline_alphabetically " , p_alphabetic_sort ) ;
_update_members_overview ( ) ;
}
2017-05-28 16:20:38 +02:00
void ScriptEditor : : _update_members_overview ( ) {
members_overview - > clear ( ) ;
2018-02-12 02:36:15 +01:00
ScriptEditorBase * se = _get_current_editor ( ) ;
2017-05-28 16:20:38 +02:00
if ( ! se ) {
return ;
}
Vector < String > functions = se - > get_functions ( ) ;
2018-04-30 14:27:00 +02:00
if ( EditorSettings : : get_singleton ( ) - > get ( " text_editor/tools/sort_members_outline_alphabetically " ) ) {
functions . sort ( ) ;
}
2017-05-28 16:20:38 +02:00
for ( int i = 0 ; i < functions . size ( ) ; i + + ) {
2019-06-03 21:57:06 +02:00
String filter = filter_methods - > get_text ( ) ;
String name = functions [ i ] . get_slice ( " : " , 0 ) ;
if ( filter = = " " | | filter . is_subsequence_ofi ( name ) ) {
members_overview - > add_item ( name ) ;
members_overview - > set_item_metadata ( members_overview - > get_item_count ( ) - 1 , functions [ i ] . get_slice ( " : " , 1 ) . to_int ( ) - 1 ) ;
}
2017-05-28 16:20:38 +02:00
}
2018-06-05 23:18:11 +02:00
2018-05-28 17:52:28 +02:00
String path = se - > get_edited_resource ( ) - > get_path ( ) ;
2018-06-05 23:18:11 +02:00
bool built_in = ! path . is_resource_file ( ) ;
String name = built_in ? path . get_file ( ) : se - > get_name ( ) ;
filename - > set_text ( name ) ;
2017-05-28 16:20:38 +02:00
}
2017-09-14 02:56:37 +02:00
void ScriptEditor : : _update_help_overview_visibility ( ) {
int selected = tab_container - > get_current_tab ( ) ;
2017-10-14 19:51:35 +02:00
if ( selected < 0 | | selected > = tab_container - > get_child_count ( ) ) {
help_overview - > set_visible ( false ) ;
2017-09-14 02:56:37 +02:00
return ;
2017-10-14 19:51:35 +02:00
}
2017-09-14 02:56:37 +02:00
Node * current = tab_container - > get_child ( tab_container - > get_current_tab ( ) ) ;
EditorHelp * se = Object : : cast_to < EditorHelp > ( current ) ;
if ( ! se ) {
help_overview - > set_visible ( false ) ;
return ;
}
if ( help_overview_enabled ) {
2018-06-05 23:18:11 +02:00
members_overview_alphabeta_sort_button - > set_visible ( false ) ;
2017-09-14 02:56:37 +02:00
help_overview - > set_visible ( true ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > set_visible ( true ) ;
2018-06-05 23:18:11 +02:00
filename - > set_text ( se - > get_name ( ) ) ;
2017-09-14 02:56:37 +02:00
} else {
help_overview - > set_visible ( false ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > set_visible ( false ) ;
2017-09-14 02:56:37 +02:00
}
}
void ScriptEditor : : _update_help_overview ( ) {
2017-10-14 19:51:35 +02:00
help_overview - > clear ( ) ;
2017-09-14 02:56:37 +02:00
int selected = tab_container - > get_current_tab ( ) ;
2020-05-14 16:41:43 +02:00
if ( selected < 0 | | selected > = tab_container - > get_child_count ( ) ) {
2017-09-14 02:56:37 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-09-14 02:56:37 +02:00
Node * current = tab_container - > get_child ( tab_container - > get_current_tab ( ) ) ;
EditorHelp * se = Object : : cast_to < EditorHelp > ( current ) ;
if ( ! se ) {
return ;
}
2020-03-17 07:33:00 +01:00
Vector < Pair < String , int > > sections = se - > get_sections ( ) ;
2017-09-14 02:56:37 +02:00
for ( int i = 0 ; i < sections . size ( ) ; i + + ) {
help_overview - > add_item ( sections [ i ] . first ) ;
help_overview - > set_item_metadata ( i , sections [ i ] . second ) ;
}
}
2015-11-17 13:46:08 +01:00
void ScriptEditor : : _update_script_colors ( ) {
2019-09-01 13:28:57 +02:00
bool script_temperature_enabled = EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/script_temperature_enabled " ) ;
2015-11-17 13:46:08 +01:00
2019-09-01 13:28:57 +02:00
int hist_size = EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/script_temperature_history_size " ) ;
2020-03-12 13:37:40 +01:00
Color hot_color = get_theme_color ( " accent_color " , " Editor " ) ;
Color cold_color = get_theme_color ( " font_color " , " Editor " ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < script_list - > get_item_count ( ) ; i + + ) {
2015-11-17 13:46:08 +01:00
int c = script_list - > get_item_metadata ( i ) ;
Node * n = tab_container - > get_child ( c ) ;
2020-05-14 16:41:43 +02:00
if ( ! n ) {
2015-11-17 13:46:08 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
script_list - > set_item_custom_bg_color ( i , Color ( 0 , 0 , 0 , 0 ) ) ;
2015-11-17 13:46:08 +01:00
2019-03-23 01:57:28 +01:00
if ( script_temperature_enabled ) {
2016-09-15 16:35:25 +02:00
if ( ! n - > has_meta ( " __editor_pass " ) ) {
continue ;
}
2017-03-05 16:44:50 +01:00
int pass = n - > get_meta ( " __editor_pass " ) ;
2016-09-15 16:35:25 +02:00
int h = edit_pass - pass ;
2017-03-05 16:44:50 +01:00
if ( h > hist_size ) {
2016-09-15 16:35:25 +02:00
continue ;
}
2017-03-05 16:44:50 +01:00
int non_zero_hist_size = ( hist_size = = 0 ) ? 1 : hist_size ;
float v = Math : : ease ( ( edit_pass - pass ) / float ( non_zero_hist_size ) , 0.4 ) ;
2016-09-15 16:35:25 +02:00
2020-03-16 10:07:33 +01:00
script_list - > set_item_custom_fg_color ( i , hot_color . lerp ( cold_color , v ) ) ;
2016-09-07 18:54:20 +02:00
}
2015-11-17 13:46:08 +01:00
}
}
2015-06-22 05:03:19 +02:00
void ScriptEditor : : _update_script_names ( ) {
2020-05-14 16:41:43 +02:00
if ( restoring_layout ) {
2016-07-18 17:30:43 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-07-18 17:30:43 +02:00
2020-03-17 07:33:00 +01:00
Set < Ref < Script > > used ;
2017-03-05 16:44:50 +01:00
Node * edited = EditorNode : : get_singleton ( ) - > get_edited_scene ( ) ;
2015-06-22 05:03:19 +02:00
if ( edited ) {
2017-03-05 16:44:50 +01:00
_find_scripts ( edited , edited , used ) ;
2015-06-22 05:03:19 +02:00
}
script_list - > clear ( ) ;
2019-09-01 13:28:57 +02:00
bool split_script_help = EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/group_help_pages " ) ;
ScriptSortBy sort_by = ( ScriptSortBy ) ( int ) EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/sort_scripts_by " ) ;
ScriptListName display_as = ( ScriptListName ) ( int ) EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/list_script_names_as " ) ;
2015-11-17 13:46:08 +01:00
Vector < _ScriptEditorItemData > sedata ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-03 00:11:05 +02:00
if ( se ) {
2020-03-12 13:37:40 +01:00
Ref < Texture2D > icon = se - > get_theme_icon ( ) ;
2018-05-28 17:52:28 +02:00
String path = se - > get_edited_resource ( ) - > get_path ( ) ;
2020-12-15 13:04:21 +01:00
bool saved = ! path . is_empty ( ) ;
2020-07-23 14:21:28 +02:00
if ( saved ) {
// The script might be deleted, moved, or renamed, so make sure
// to update original path to previously edited resource.
se - > set_meta ( " _edit_res_path " , path ) ;
}
2018-05-16 15:04:37 +02:00
bool built_in = ! path . is_resource_file ( ) ;
2019-01-02 06:54:51 +01:00
String name ;
if ( built_in ) {
name = path . get_file ( ) ;
2020-03-12 21:35:52 +01:00
const String & resource_name = se - > get_edited_resource ( ) - > get_name ( ) ;
2019-01-02 06:54:51 +01:00
if ( resource_name ! = " " ) {
2020-03-12 21:35:52 +01:00
// If the built-in script has a custom resource name defined,
// display the built-in script name as follows: `ResourceName (scene_file.tscn)`
name = vformat ( " %s (%s) " , resource_name , name . substr ( 0 , name . find ( " :: " , 0 ) ) ) ;
2019-01-02 06:54:51 +01:00
}
} else {
name = se - > get_name ( ) ;
}
2015-06-22 05:03:19 +02:00
2015-11-17 13:46:08 +01:00
_ScriptEditorItemData sd ;
2017-03-05 16:44:50 +01:00
sd . icon = icon ;
sd . name = name ;
2020-07-23 14:21:28 +02:00
sd . tooltip = saved ? path : TTR ( " Unsaved file. " ) ;
2017-03-05 16:44:50 +01:00
sd . index = i ;
2018-05-28 17:52:28 +02:00
sd . used = used . has ( se - > get_edited_resource ( ) ) ;
2017-03-05 16:44:50 +01:00
sd . category = 0 ;
2017-11-12 03:42:56 +01:00
sd . ref = se ;
2017-03-05 16:44:50 +01:00
2017-02-02 13:31:01 +01:00
switch ( sort_by ) {
case SORT_BY_NAME : {
2017-03-05 16:44:50 +01:00
sd . sort_key = name . to_lower ( ) ;
2017-02-02 13:31:01 +01:00
} break ;
case SORT_BY_PATH : {
2017-03-05 16:44:50 +01:00
sd . sort_key = path ;
2017-02-02 13:31:01 +01:00
} break ;
2018-02-14 17:32:33 +01:00
case SORT_BY_NONE : {
sd . sort_key = " " ;
} break ;
2017-02-02 13:31:01 +01:00
}
2017-03-05 16:44:50 +01:00
2017-02-02 13:31:01 +01:00
switch ( display_as ) {
case DISPLAY_NAME : {
2017-03-05 16:44:50 +01:00
sd . name = name ;
2017-02-02 13:31:01 +01:00
} break ;
case DISPLAY_DIR_AND_NAME : {
2020-12-15 13:04:21 +01:00
if ( ! path . get_base_dir ( ) . get_file ( ) . is_empty ( ) ) {
2018-09-12 12:29:50 +02:00
sd . name = path . get_base_dir ( ) . get_file ( ) . plus_file ( name ) ;
2017-02-02 13:31:01 +01:00
} else {
2017-03-05 16:44:50 +01:00
sd . name = name ;
2017-02-02 13:31:01 +01:00
}
} break ;
case DISPLAY_FULL_PATH : {
2017-03-05 16:44:50 +01:00
sd . name = path ;
2017-02-02 13:31:01 +01:00
} break ;
}
2020-07-23 14:21:28 +02:00
if ( ! saved ) {
sd . name = se - > get_name ( ) ;
}
2017-02-02 13:31:01 +01:00
2015-11-17 13:46:08 +01:00
sedata . push_back ( sd ) ;
}
2015-06-22 05:03:19 +02:00
2020-05-27 06:49:05 +02:00
Vector < String > disambiguated_script_names ;
Vector < String > full_script_paths ;
for ( int j = 0 ; j < sedata . size ( ) ; j + + ) {
2021-01-11 22:01:21 +01:00
String name = sedata [ j ] . name . replace ( " (*) " , " " ) ;
ScriptListName script_display = ( ScriptListName ) ( int ) EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/list_script_names_as " ) ;
switch ( script_display ) {
case DISPLAY_NAME : {
name = name . get_file ( ) ;
} break ;
case DISPLAY_DIR_AND_NAME : {
name = name . get_base_dir ( ) . get_file ( ) . plus_file ( name . get_file ( ) ) ;
} break ;
default :
break ;
}
disambiguated_script_names . append ( name ) ;
2020-05-27 06:49:05 +02:00
full_script_paths . append ( sedata [ j ] . tooltip ) ;
}
EditorNode : : disambiguate_filenames ( full_script_paths , disambiguated_script_names ) ;
for ( int j = 0 ; j < sedata . size ( ) ; j + + ) {
2020-07-31 22:43:40 +02:00
if ( sedata [ j ] . name . ends_with ( " (*) " ) ) {
sedata . write [ j ] . name = disambiguated_script_names [ j ] + " (*) " ;
} else {
sedata . write [ j ] . name = disambiguated_script_names [ j ] ;
}
2020-05-27 06:49:05 +02:00
}
2017-08-24 22:58:51 +02:00
EditorHelp * eh = Object : : cast_to < EditorHelp > ( tab_container - > get_child ( i ) ) ;
2015-11-17 13:46:08 +01:00
if ( eh ) {
2017-01-03 03:03:46 +01:00
String name = eh - > get_class ( ) ;
2020-03-12 13:37:40 +01:00
Ref < Texture2D > icon = get_theme_icon ( " Help " , " EditorIcons " ) ;
2019-04-09 16:06:37 +02:00
String tooltip = vformat ( TTR ( " %s Class Reference " ) , name ) ;
2015-11-17 13:46:08 +01:00
_ScriptEditorItemData sd ;
2017-03-05 16:44:50 +01:00
sd . icon = icon ;
sd . name = name ;
2017-11-12 03:42:56 +01:00
sd . sort_key = name . to_lower ( ) ;
2017-03-05 16:44:50 +01:00
sd . tooltip = tooltip ;
sd . index = i ;
sd . used = false ;
sd . category = split_script_help ? 1 : 0 ;
2017-11-12 03:42:56 +01:00
sd . ref = eh ;
2015-11-17 13:46:08 +01:00
sedata . push_back ( sd ) ;
}
}
2020-12-15 13:04:21 +01:00
if ( _sort_list_on_update & & ! sedata . is_empty ( ) ) {
2017-11-12 03:42:56 +01:00
sedata . sort ( ) ;
// change actual order of tab_container so that the order can be rearranged by user
int cur_tab = tab_container - > get_current_tab ( ) ;
int prev_tab = tab_container - > get_previous_tab ( ) ;
int new_cur_tab = - 1 ;
int new_prev_tab = - 1 ;
for ( int i = 0 ; i < sedata . size ( ) ; i + + ) {
tab_container - > move_child ( sedata [ i ] . ref , i ) ;
if ( new_prev_tab = = - 1 & & sedata [ i ] . index = = prev_tab ) {
new_prev_tab = i ;
}
if ( new_cur_tab = = - 1 & & sedata [ i ] . index = = cur_tab ) {
new_cur_tab = i ;
}
2020-04-26 19:23:00 +02:00
// Update index of sd entries for sorted order
_ScriptEditorItemData sd = sedata [ i ] ;
sd . index = i ;
sedata . set ( i , sd ) ;
2017-11-12 03:42:56 +01:00
}
2018-09-08 20:33:54 +02:00
tab_container - > set_current_tab ( new_prev_tab ) ;
tab_container - > set_current_tab ( new_cur_tab ) ;
2017-11-12 03:42:56 +01:00
_sort_list_on_update = false ;
}
2015-11-17 13:46:08 +01:00
2019-06-03 21:57:06 +02:00
Vector < _ScriptEditorItemData > sedata_filtered ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < sedata . size ( ) ; i + + ) {
2019-06-03 21:57:06 +02:00
String filter = filter_scripts - > get_text ( ) ;
if ( filter = = " " | | filter . is_subsequence_ofi ( sedata [ i ] . name ) ) {
sedata_filtered . push_back ( sedata [ i ] ) ;
}
}
2015-11-17 13:46:08 +01:00
2019-06-03 21:57:06 +02:00
for ( int i = 0 ; i < sedata_filtered . size ( ) ; i + + ) {
script_list - > add_item ( sedata_filtered [ i ] . name , sedata_filtered [ i ] . icon ) ;
2017-03-05 16:44:50 +01:00
int index = script_list - > get_item_count ( ) - 1 ;
2019-06-03 21:57:06 +02:00
script_list - > set_item_tooltip ( index , sedata_filtered [ i ] . tooltip ) ;
script_list - > set_item_metadata ( index , sedata_filtered [ i ] . index ) ; /* Saving as metadata the script's index in the tab container and not the filtered one */
if ( sedata_filtered [ i ] . used ) {
2017-03-05 16:44:50 +01:00
script_list - > set_item_custom_bg_color ( index , Color ( 88 / 255.0 , 88 / 255.0 , 60 / 255.0 ) ) ;
2015-06-22 05:03:19 +02:00
}
2019-06-03 21:57:06 +02:00
if ( tab_container - > get_current_tab ( ) = = sedata_filtered [ i ] . index ) {
2015-06-22 05:03:19 +02:00
script_list - > select ( index ) ;
2019-06-03 21:57:06 +02:00
script_name_label - > set_text ( sedata_filtered [ i ] . name ) ;
script_icon - > set_texture ( sedata_filtered [ i ] . icon ) ;
2019-11-20 10:09:59 +01:00
ScriptEditorBase * se = _get_current_editor ( ) ;
if ( se ) {
se - > enable_editor ( ) ;
_update_selected_editor_menu ( ) ;
}
2015-06-22 05:03:19 +02:00
}
2014-02-10 02:10:30 +01:00
}
2015-06-22 05:03:19 +02:00
2018-08-26 17:35:33 +02:00
if ( ! waiting_update_names ) {
_update_members_overview ( ) ;
_update_help_overview ( ) ;
} else {
waiting_update_names = false ;
}
2017-10-14 19:51:35 +02:00
_update_members_overview_visibility ( ) ;
_update_help_overview_visibility ( ) ;
2015-11-17 13:46:08 +01:00
_update_script_colors ( ) ;
2019-07-31 21:06:15 +02:00
2020-12-15 13:04:21 +01:00
file_menu - > get_popup ( ) - > set_item_disabled ( file_menu - > get_popup ( ) - > get_item_index ( FILE_REOPEN_CLOSED ) , previous_scripts . is_empty ( ) ) ;
2014-02-10 02:10:30 +01:00
}
2019-04-20 13:51:25 +02:00
void ScriptEditor : : _update_script_connections ( ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
ScriptTextEditor * ste = Object : : cast_to < ScriptTextEditor > ( tab_container - > get_child ( i ) ) ;
if ( ! ste ) {
continue ;
}
ste - > _update_connected_methods ( ) ;
}
}
2018-05-28 17:52:28 +02:00
Ref < TextFile > ScriptEditor : : _load_text_file ( const String & p_path , Error * r_error ) {
if ( r_error ) {
* r_error = ERR_FILE_CANT_OPEN ;
}
String local_path = ProjectSettings : : get_singleton ( ) - > localize_path ( p_path ) ;
String path = ResourceLoader : : path_remap ( local_path ) ;
TextFile * text_file = memnew ( TextFile ) ;
Ref < TextFile > text_res ( text_file ) ;
Error err = text_file - > load_text ( path ) ;
2019-09-25 10:28:50 +02:00
ERR_FAIL_COND_V_MSG ( err ! = OK , RES ( ) , " Cannot load text file ' " + path + " '. " ) ;
2018-05-28 17:52:28 +02:00
text_file - > set_file_path ( local_path ) ;
text_file - > set_path ( local_path , true ) ;
2018-12-02 17:24:23 +01:00
if ( ResourceLoader : : get_timestamp_on_load ( ) ) {
text_file - > set_last_modified_time ( FileAccess : : get_modified_time ( path ) ) ;
}
2018-05-28 17:52:28 +02:00
if ( r_error ) {
* r_error = OK ;
}
return text_res ;
}
Error ScriptEditor : : _save_text_file ( Ref < TextFile > p_text_file , const String & p_path ) {
Ref < TextFile > sqscr = p_text_file ;
ERR_FAIL_COND_V ( sqscr . is_null ( ) , ERR_INVALID_PARAMETER ) ;
String source = sqscr - > get_text ( ) ;
Error err ;
FileAccess * file = FileAccess : : open ( p_path , FileAccess : : WRITE , & err ) ;
2019-09-25 10:28:50 +02:00
ERR_FAIL_COND_V_MSG ( err , err , " Cannot save text file ' " + p_path + " '. " ) ;
2018-05-28 17:52:28 +02:00
file - > store_string ( source ) ;
if ( file - > get_error ( ) ! = OK & & file - > get_error ( ) ! = ERR_FILE_EOF ) {
memdelete ( file ) ;
return ERR_CANT_CREATE ;
}
file - > close ( ) ;
memdelete ( file ) ;
2018-12-02 17:24:23 +01:00
if ( ResourceSaver : : get_timestamp_on_save ( ) ) {
p_text_file - > set_last_modified_time ( FileAccess : : get_modified_time ( p_path ) ) ;
}
2018-05-28 17:52:28 +02:00
_res_saved_callback ( sqscr ) ;
return OK ;
}
bool ScriptEditor : : edit ( const RES & p_resource , int p_line , int p_col , bool p_grab_focus ) {
2020-05-14 16:41:43 +02:00
if ( p_resource . is_null ( ) ) {
2017-04-15 19:48:10 +02:00
return false ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2018-05-28 17:52:28 +02:00
Ref < Script > script = p_resource ;
2020-07-26 23:18:05 +02:00
// Don't open dominant script if using an external editor.
2020-07-27 10:29:22 +02:00
const bool use_external_editor =
EditorSettings : : get_singleton ( ) - > get ( " text_editor/external/use_external_editor " ) | |
2020-07-27 15:05:14 +02:00
( script . is_valid ( ) & & script - > get_language ( ) - > overrides_external_editor ( ) ) ;
2020-07-26 23:18:05 +02:00
const bool open_dominant = EditorSettings : : get_singleton ( ) - > get ( " text_editor/files/open_dominant_script_on_scene_change " ) ;
2016-07-07 01:35:49 +02:00
2020-07-26 23:18:05 +02:00
const bool should_open = ( open_dominant & & ! use_external_editor ) | | ! EditorNode : : get_singleton ( ) - > is_changing_scene ( ) ;
2018-01-14 12:34:00 +01:00
2020-07-27 15:05:14 +02:00
if ( script . is_valid ( ) & & script - > get_language ( ) - > overrides_external_editor ( ) ) {
2018-01-14 12:34:00 +01:00
if ( should_open ) {
2018-05-28 17:52:28 +02:00
Error err = script - > get_language ( ) - > open_in_external_editor ( script , p_line > = 0 ? p_line : 0 , p_col ) ;
2020-05-14 16:41:43 +02:00
if ( err ! = OK ) {
2018-01-14 12:34:00 +01:00
ERR_PRINT ( " Couldn't open script in the overridden external text editor " ) ;
2020-05-14 16:41:43 +02:00
}
2018-01-14 12:34:00 +01:00
}
2017-09-03 21:23:36 +02:00
return false ;
}
2020-07-26 23:18:05 +02:00
if ( use_external_editor & &
( EditorDebuggerNode : : get_singleton ( ) - > get_dump_stack_script ( ) ! = p_resource | | EditorDebuggerNode : : get_singleton ( ) - > get_debug_with_external_editor ( ) ) & &
2018-05-28 17:52:28 +02:00
p_resource - > get_path ( ) . is_resource_file ( ) & &
2020-07-26 23:18:05 +02:00
p_resource - > get_class_name ( ) ! = StringName ( " VisualScript " ) ) {
2017-01-05 23:41:36 +01:00
String path = EditorSettings : : get_singleton ( ) - > get ( " text_editor/external/exec_path " ) ;
String flags = EditorSettings : : get_singleton ( ) - > get ( " text_editor/external/exec_flags " ) ;
2017-04-15 19:48:10 +02:00
2014-02-10 02:10:30 +01:00
List < String > args ;
2019-06-11 09:20:42 +02:00
bool has_file_flag = false ;
String script_path = ProjectSettings : : get_singleton ( ) - > globalize_path ( p_resource - > get_path ( ) ) ;
2017-04-15 19:48:10 +02:00
if ( flags . size ( ) ) {
2018-02-04 11:18:54 +01:00
String project_path = ProjectSettings : : get_singleton ( ) - > get_resource_path ( ) ;
flags = flags . replacen ( " {line} " , itos ( p_line > 0 ? p_line : 0 ) ) ;
flags = flags . replacen ( " {col} " , itos ( p_col ) ) ;
flags = flags . strip_edges ( ) . replace ( " \\ \\ " , " \\ " ) ;
int from = 0 ;
int num_chars = 0 ;
2017-04-15 19:48:10 +02:00
bool inside_quotes = false ;
2018-02-04 11:18:54 +01:00
2017-04-15 19:48:10 +02:00
for ( int i = 0 ; i < flags . size ( ) ; i + + ) {
if ( flags [ i ] = = ' " ' & & ( ! i | | flags [ i - 1 ] ! = ' \\ ' ) ) {
2018-02-04 11:18:54 +01:00
if ( ! inside_quotes ) {
from + + ;
}
2017-04-15 19:48:10 +02:00
inside_quotes = ! inside_quotes ;
2018-02-04 11:18:54 +01:00
2017-04-15 19:48:10 +02:00
} else if ( flags [ i ] = = ' \0 ' | | ( ! inside_quotes & & flags [ i ] = = ' ' ) ) {
2018-02-04 11:18:54 +01:00
String arg = flags . substr ( from , num_chars ) ;
2019-06-11 09:20:42 +02:00
if ( arg . find ( " {file} " ) ! = - 1 ) {
has_file_flag = true ;
}
2018-02-04 11:18:54 +01:00
// do path replacement here, else there will be issues with spaces and quotes
arg = arg . replacen ( " {project} " , project_path ) ;
arg = arg . replacen ( " {file} " , script_path ) ;
args . push_back ( arg ) ;
2017-04-15 19:48:10 +02:00
from = i + 1 ;
2018-02-04 11:18:54 +01:00
num_chars = 0 ;
2017-04-15 19:48:10 +02:00
} else {
2018-02-04 11:18:54 +01:00
num_chars + + ;
2017-04-15 19:48:10 +02:00
}
}
2014-02-10 02:10:30 +01:00
}
2019-06-11 09:20:42 +02:00
// Default to passing script path if no {file} flag is specified.
if ( ! has_file_flag ) {
args . push_back ( script_path ) ;
}
2020-12-18 19:49:13 +01:00
Error err = OS : : get_singleton ( ) - > create_process ( path , args ) ;
2020-05-14 16:41:43 +02:00
if ( err = = OK ) {
2017-04-15 19:48:10 +02:00
return false ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
WARN_PRINT ( " Couldn't open external text editor, using internal " ) ;
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2020-04-02 01:20:12 +02:00
if ( ( script ! = nullptr & & se - > get_edited_resource ( ) = = p_resource ) | | se - > get_edited_resource ( ) - > get_path ( ) = = p_resource - > get_path ( ) ) {
2018-01-14 12:34:00 +01:00
if ( should_open ) {
2019-11-20 10:09:59 +01:00
se - > enable_editor ( ) ;
2017-03-05 16:44:50 +01:00
if ( tab_container - > get_current_tab ( ) ! = i ) {
2015-11-17 13:46:08 +01:00
_go_to_tab ( i ) ;
2019-06-03 21:57:06 +02:00
_update_script_names ( ) ;
2015-06-22 05:03:19 +02:00
}
2020-05-14 16:41:43 +02:00
if ( is_visible_in_tree ( ) ) {
2016-08-03 00:11:05 +02:00
se - > ensure_focus ( ) ;
2020-05-14 16:41:43 +02:00
}
2017-04-15 19:48:10 +02:00
2019-03-17 20:47:04 +01:00
if ( p_line > 0 ) {
2017-04-15 19:48:10 +02:00
se - > goto_line ( p_line - 1 ) ;
2019-03-17 20:47:04 +01:00
}
2015-06-22 05:03:19 +02:00
}
2019-06-15 16:24:03 +02:00
_update_script_names ( ) ;
script_list - > ensure_current_is_visible ( ) ;
2017-04-15 19:48:10 +02:00
return true ;
2014-02-10 02:10:30 +01:00
}
}
// doesn't have it, make a new one
2020-04-02 01:20:12 +02:00
ScriptEditorBase * se = nullptr ;
2016-08-03 00:11:05 +02:00
2017-03-05 16:44:50 +01:00
for ( int i = script_editor_func_count - 1 ; i > = 0 ; i - - ) {
2018-05-28 17:52:28 +02:00
se = script_editor_funcs [ i ] ( p_resource ) ;
2020-05-14 16:41:43 +02:00
if ( se ) {
2016-08-03 00:11:05 +02:00
break ;
2020-05-14 16:41:43 +02:00
}
2016-08-03 00:11:05 +02:00
}
2017-04-15 19:48:10 +02:00
ERR_FAIL_COND_V ( ! se , false ) ;
2016-09-11 15:01:52 +02:00
2019-11-20 10:09:59 +01:00
se - > set_edited_resource ( p_resource ) ;
2019-07-25 18:30:48 +02:00
if ( p_resource - > get_class_name ( ) ! = StringName ( " VisualScript " ) ) {
bool highlighter_set = false ;
2020-03-07 15:29:44 +01:00
for ( int i = 0 ; i < syntax_highlighters . size ( ) ; i + + ) {
2020-05-03 18:08:15 +02:00
Ref < EditorSyntaxHighlighter > highlighter = syntax_highlighters [ i ] - > _create ( ) ;
2020-03-07 15:29:44 +01:00
if ( highlighter . is_null ( ) ) {
continue ;
}
2019-07-25 18:30:48 +02:00
se - > add_syntax_highlighter ( highlighter ) ;
2020-04-02 01:20:12 +02:00
if ( script ! = nullptr & & ! highlighter_set ) {
2020-03-07 12:17:18 +01:00
Array languages = highlighter - > _get_supported_languages ( ) ;
if ( languages . find ( script - > get_language ( ) - > get_name ( ) ) > - 1 ) {
2019-07-25 18:30:48 +02:00
se - > set_syntax_highlighter ( highlighter ) ;
highlighter_set = true ;
}
2018-04-02 13:41:44 +02:00
}
}
}
2017-09-17 15:19:38 +02:00
tab_container - > add_child ( se ) ;
2019-11-20 10:09:59 +01:00
if ( p_grab_focus ) {
se - > enable_editor ( ) ;
}
2020-07-23 14:21:28 +02:00
// If we delete a script within the filesystem, the original resource path
// is lost, so keep it as metadata to figure out the exact tab to delete.
se - > set_meta ( " _edit_res_path " , p_resource - > get_path ( ) ) ;
2017-03-05 16:44:50 +01:00
se - > set_tooltip_request_func ( " _get_debug_tooltip " , this ) ;
2016-08-03 00:11:05 +02:00
if ( se - > get_edit_menu ( ) ) {
se - > get_edit_menu ( ) - > hide ( ) ;
menu_hb - > add_child ( se - > get_edit_menu ( ) ) ;
2017-03-05 16:44:50 +01:00
menu_hb - > move_child ( se - > get_edit_menu ( ) , 1 ) ;
2016-08-03 00:11:05 +02:00
}
2016-08-25 22:45:20 +02:00
if ( p_grab_focus ) {
2017-03-05 16:44:50 +01:00
_go_to_tab ( tab_container - > get_tab_count ( ) - 1 ) ;
2019-11-20 10:09:59 +01:00
_add_recent_script ( p_resource - > get_path ( ) ) ;
2016-08-25 22:45:20 +02:00
}
2015-11-17 13:46:08 +01:00
2018-09-08 20:33:54 +02:00
_sort_list_on_update = true ;
2015-06-22 05:03:19 +02:00
_update_script_names ( ) ;
2016-07-18 17:30:43 +02:00
_save_layout ( ) ;
2020-02-21 18:28:45 +01:00
se - > connect ( " name_changed " , callable_mp ( this , & ScriptEditor : : _update_script_names ) ) ;
se - > connect ( " edited_script_changed " , callable_mp ( this , & ScriptEditor : : _script_changed ) ) ;
se - > connect ( " request_help " , callable_mp ( this , & ScriptEditor : : _help_search ) ) ;
se - > connect ( " request_open_script_at_line " , callable_mp ( this , & ScriptEditor : : _goto_script_line ) ) ;
se - > connect ( " go_to_help " , callable_mp ( this , & ScriptEditor : : _help_class_goto ) ) ;
se - > connect ( " request_save_history " , callable_mp ( this , & ScriptEditor : : _save_history ) ) ;
se - > connect ( " search_in_files_requested " , callable_mp ( this , & ScriptEditor : : _on_find_in_files_requested ) ) ;
se - > connect ( " replace_in_files_requested " , callable_mp ( this , & ScriptEditor : : _on_replace_in_files_requested ) ) ;
2016-06-20 03:07:07 +02:00
//test for modification, maybe the script was not edited but was loaded
2018-05-28 17:52:28 +02:00
_test_script_times_on_disk ( p_resource ) ;
_update_modified_scripts_for_external_editor ( p_resource ) ;
2017-04-15 19:48:10 +02:00
2019-03-17 20:47:04 +01:00
if ( p_line > 0 ) {
2017-04-15 19:48:10 +02:00
se - > goto_line ( p_line - 1 ) ;
2019-03-17 20:47:04 +01:00
}
2017-04-15 19:48:10 +02:00
2018-05-28 17:52:28 +02:00
notify_script_changed ( p_resource ) ;
2017-04-15 19:48:10 +02:00
return true ;
2014-02-10 02:10:30 +01:00
}
2021-05-09 12:29:50 +02:00
void ScriptEditor : : save_current_script ( ) {
ScriptEditorBase * current = _get_current_editor ( ) ;
2021-05-18 19:43:54 +02:00
if ( ! current | | _test_script_times_on_disk ( ) ) {
2021-05-09 12:29:50 +02:00
return ;
}
if ( trim_trailing_whitespace_on_save ) {
current - > trim_trailing_whitespace ( ) ;
}
current - > insert_final_newline ( ) ;
if ( convert_indent_on_save ) {
if ( use_space_indentation ) {
current - > convert_indent_to_spaces ( ) ;
} else {
current - > convert_indent_to_tabs ( ) ;
}
}
RES resource = current - > get_edited_resource ( ) ;
Ref < TextFile > text_file = resource ;
Ref < Script > script = resource ;
if ( text_file ! = nullptr ) {
current - > apply_code ( ) ;
_save_text_file ( text_file , text_file - > get_path ( ) ) ;
return ;
}
if ( script ! = nullptr ) {
const Vector < DocData : : ClassDoc > & documentations = script - > get_documentation ( ) ;
for ( int j = 0 ; j < documentations . size ( ) ; j + + ) {
const DocData : : ClassDoc & doc = documentations . get ( j ) ;
if ( EditorHelp : : get_doc_data ( ) - > has_doc ( doc . name ) ) {
EditorHelp : : get_doc_data ( ) - > remove_doc ( doc . name ) ;
}
}
}
editor - > save_resource ( resource ) ;
if ( script ! = nullptr ) {
const Vector < DocData : : ClassDoc > & documentations = script - > get_documentation ( ) ;
for ( int j = 0 ; j < documentations . size ( ) ; j + + ) {
const DocData : : ClassDoc & doc = documentations . get ( j ) ;
EditorHelp : : get_doc_data ( ) - > add_doc ( doc ) ;
update_doc ( doc . name ) ;
}
}
}
2016-02-03 01:10:52 +01:00
void ScriptEditor : : save_all_scripts ( ) {
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2017-04-17 15:46:00 +02:00
if ( convert_indent_on_save ) {
if ( use_space_indentation ) {
se - > convert_indent_to_spaces ( ) ;
} else {
se - > convert_indent_to_tabs ( ) ;
}
}
2017-04-23 20:19:30 +02:00
if ( trim_trailing_whitespace_on_save ) {
se - > trim_trailing_whitespace ( ) ;
}
2019-05-28 23:27:32 +02:00
se - > insert_final_newline ( ) ;
2020-05-14 16:41:43 +02:00
if ( ! se - > is_unsaved ( ) ) {
2017-04-23 20:19:30 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2017-04-23 20:19:30 +02:00
2018-05-28 17:52:28 +02:00
RES edited_res = se - > get_edited_resource ( ) ;
if ( edited_res . is_valid ( ) ) {
2016-08-03 00:11:05 +02:00
se - > apply_code ( ) ;
2018-05-28 17:52:28 +02:00
}
2016-06-12 02:59:35 +02:00
2018-05-28 17:52:28 +02:00
if ( edited_res - > get_path ( ) ! = " " & & edited_res - > get_path ( ) . find ( " local:// " ) = = - 1 & & edited_res - > get_path ( ) . find ( " :: " ) = = - 1 ) {
Ref < TextFile > text_file = edited_res ;
2020-11-29 04:42:06 +01:00
Ref < Script > script = edited_res ;
2020-04-02 01:20:12 +02:00
if ( text_file ! = nullptr ) {
2018-05-28 17:52:28 +02:00
_save_text_file ( text_file , text_file - > get_path ( ) ) ;
continue ;
}
2020-11-29 04:42:06 +01:00
if ( script ! = nullptr ) {
const Vector < DocData : : ClassDoc > & documentations = script - > get_documentation ( ) ;
for ( int j = 0 ; j < documentations . size ( ) ; j + + ) {
const DocData : : ClassDoc & doc = documentations . get ( j ) ;
if ( EditorHelp : : get_doc_data ( ) - > has_doc ( doc . name ) ) {
EditorHelp : : get_doc_data ( ) - > remove_doc ( doc . name ) ;
}
}
}
2018-05-28 17:52:28 +02:00
editor - > save_resource ( edited_res ) ; //external script, save it
2020-11-29 04:42:06 +01:00
if ( script ! = nullptr ) {
const Vector < DocData : : ClassDoc > & documentations = script - > get_documentation ( ) ;
for ( int j = 0 ; j < documentations . size ( ) ; j + + ) {
const DocData : : ClassDoc & doc = documentations . get ( j ) ;
EditorHelp : : get_doc_data ( ) - > add_doc ( doc ) ;
update_doc ( doc . name ) ;
}
}
2018-05-28 17:52:28 +02:00
}
2014-02-10 02:10:30 +01:00
}
2016-08-07 00:00:54 +02:00
_update_script_names ( ) ;
2018-07-16 00:29:00 +02:00
EditorFileSystem : : get_singleton ( ) - > update_script_classes ( ) ;
2014-02-10 02:10:30 +01:00
}
void ScriptEditor : : apply_scripts ( ) const {
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2016-08-03 00:11:05 +02:00
se - > apply_code ( ) ;
2014-02-10 02:10:30 +01:00
}
}
2017-12-13 18:41:28 +01:00
void ScriptEditor : : open_script_create_dialog ( const String & p_base_name , const String & p_base_path ) {
_menu_option ( FILE_NEW ) ;
script_create_dialog - > config ( p_base_name , p_base_path ) ;
}
2014-02-10 02:10:30 +01:00
void ScriptEditor : : _editor_stop ( ) {
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-07 00:00:54 +02:00
if ( ! se ) {
continue ;
}
se - > set_debugger_active ( false ) ;
}
2014-02-10 02:10:30 +01:00
}
2020-02-17 22:06:54 +01:00
void ScriptEditor : : _add_callback ( Object * p_obj , const String & p_function , const PackedStringArray & p_args ) {
2014-02-10 02:10:30 +01:00
ERR_FAIL_COND ( ! p_obj ) ;
Ref < Script > script = p_obj - > get_script ( ) ;
2017-03-05 16:44:50 +01:00
ERR_FAIL_COND ( ! script . is_valid ( ) ) ;
2014-02-10 02:10:30 +01:00
editor - > push_item ( script . ptr ( ) ) ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
if ( se - > get_edited_resource ( ) ! = script ) {
2014-02-10 02:10:30 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
se - > add_callback ( p_function , p_args ) ;
2014-02-10 02:10:30 +01:00
2015-11-17 13:46:08 +01:00
_go_to_tab ( i ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
script_list - > select ( script_list - > find_metadata ( i ) ) ;
2015-06-22 05:03:19 +02:00
2020-09-29 23:00:34 +02:00
// Save the current script so the changes can be picked up by an external editor.
2021-05-09 12:29:50 +02:00
save_current_script ( ) ;
2020-09-29 23:00:34 +02:00
2014-02-10 02:10:30 +01:00
break ;
}
}
2016-07-18 17:30:43 +02:00
void ScriptEditor : : _save_layout ( ) {
if ( restoring_layout ) {
return ;
}
editor - > save_layout ( ) ;
}
2015-05-05 04:32:40 +02:00
void ScriptEditor : : _editor_settings_changed ( ) {
2017-01-05 23:41:36 +01:00
trim_trailing_whitespace_on_save = EditorSettings : : get_singleton ( ) - > get ( " text_editor/files/trim_trailing_whitespace_on_save " ) ;
2017-04-17 15:24:30 +02:00
convert_indent_on_save = EditorSettings : : get_singleton ( ) - > get ( " text_editor/indent/convert_indent_on_save " ) ;
2017-04-26 13:14:03 +02:00
use_space_indentation = EditorSettings : : get_singleton ( ) - > get ( " text_editor/indent/type " ) ;
2017-04-17 15:24:30 +02:00
2019-09-01 13:28:57 +02:00
members_overview_enabled = EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/show_members_overview " ) ;
2017-09-14 02:56:37 +02:00
help_overview_enabled = EditorSettings : : get_singleton ( ) - > get ( " text_editor/help/show_help_index " ) ;
2017-05-28 16:20:38 +02:00
_update_members_overview_visibility ( ) ;
2017-09-14 02:56:37 +02:00
_update_help_overview_visibility ( ) ;
2017-05-28 16:20:38 +02:00
2019-10-10 22:19:47 +02:00
_update_autosave_timer ( ) ;
2015-05-05 04:32:40 +02:00
2016-04-12 16:45:31 +02:00
if ( current_theme = = " " ) {
2017-01-05 23:41:36 +01:00
current_theme = EditorSettings : : get_singleton ( ) - > get ( " text_editor/theme/color_theme " ) ;
2020-11-05 03:01:55 +01:00
} else if ( current_theme ! = String ( EditorSettings : : get_singleton ( ) - > get ( " text_editor/theme/color_theme " ) ) ) {
2017-01-05 23:41:36 +01:00
current_theme = EditorSettings : : get_singleton ( ) - > get ( " text_editor/theme/color_theme " ) ;
2016-04-12 16:45:31 +02:00
EditorSettings : : get_singleton ( ) - > load_text_editor_theme ( ) ;
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2016-01-01 13:52:01 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2016-01-01 13:52:01 +01:00
2016-08-03 00:11:05 +02:00
se - > update_settings ( ) ;
2016-01-01 13:52:01 +01:00
}
2016-09-12 16:19:30 +02:00
_update_script_colors ( ) ;
2017-02-02 13:31:01 +01:00
_update_script_names ( ) ;
2016-01-01 13:52:01 +01:00
2017-03-05 16:44:50 +01:00
ScriptServer : : set_reload_scripts_on_save ( EDITOR_DEF ( " text_editor/files/auto_reload_and_parse_scripts_on_save " , true ) ) ;
2015-05-05 04:32:40 +02:00
}
2020-07-23 14:21:28 +02:00
void ScriptEditor : : _filesystem_changed ( ) {
_update_script_names ( ) ;
}
void ScriptEditor : : _file_removed ( const String & p_removed_file ) {
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
if ( ! se ) {
continue ;
}
if ( se - > get_meta ( " _edit_res_path " ) = = p_removed_file ) {
// The script is deleted with no undo, so just close the tab.
_close_tab ( i , false , false ) ;
}
}
}
2015-05-05 04:32:40 +02:00
void ScriptEditor : : _autosave_scripts ( ) {
2016-02-03 01:10:52 +01:00
save_all_scripts ( ) ;
2015-05-05 04:32:40 +02:00
}
2019-10-10 22:19:47 +02:00
void ScriptEditor : : _update_autosave_timer ( ) {
if ( ! autosave_timer - > is_inside_tree ( ) ) {
return ;
}
float autosave_time = EditorSettings : : get_singleton ( ) - > get ( " text_editor/files/autosave_interval_secs " ) ;
if ( autosave_time > 0 ) {
autosave_timer - > set_wait_time ( autosave_time ) ;
autosave_timer - > start ( ) ;
} else {
autosave_timer - > stop ( ) ;
}
}
2015-06-22 05:03:19 +02:00
void ScriptEditor : : _tree_changed ( ) {
2020-05-14 16:41:43 +02:00
if ( waiting_update_names ) {
2015-06-22 05:03:19 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2015-06-22 05:03:19 +02:00
2017-03-05 16:44:50 +01:00
waiting_update_names = true ;
2015-06-22 05:03:19 +02:00
call_deferred ( " _update_script_names " ) ;
2019-04-20 13:51:25 +02:00
call_deferred ( " _update_script_connections " ) ;
2015-06-22 05:03:19 +02:00
}
void ScriptEditor : : _script_split_dragged ( float ) {
2016-07-18 17:30:43 +02:00
_save_layout ( ) ;
2015-06-22 05:03:19 +02:00
}
2017-11-12 20:21:09 +01:00
Variant ScriptEditor : : get_drag_data_fw ( const Point2 & p_point , Control * p_from ) {
2020-05-14 16:41:43 +02:00
if ( tab_container - > get_child_count ( ) = = 0 ) {
2019-03-07 16:09:02 +01:00
return Variant ( ) ;
2020-05-14 16:41:43 +02:00
}
2019-03-07 16:09:02 +01:00
2017-11-12 20:21:09 +01:00
Node * cur_node = tab_container - > get_child ( tab_container - > get_current_tab ( ) ) ;
2017-11-12 03:42:56 +01:00
2017-11-12 20:21:09 +01:00
HBoxContainer * drag_preview = memnew ( HBoxContainer ) ;
String preview_name = " " ;
2019-06-11 20:43:37 +02:00
Ref < Texture2D > preview_icon ;
2017-11-12 20:21:09 +01:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( cur_node ) ;
if ( se ) {
preview_name = se - > get_name ( ) ;
2020-03-12 13:37:40 +01:00
preview_icon = se - > get_theme_icon ( ) ;
2017-11-12 20:21:09 +01:00
}
EditorHelp * eh = Object : : cast_to < EditorHelp > ( cur_node ) ;
if ( eh ) {
preview_name = eh - > get_class ( ) ;
2020-03-12 13:37:40 +01:00
preview_icon = get_theme_icon ( " Help " , " EditorIcons " ) ;
2017-11-12 20:21:09 +01:00
}
if ( ! preview_icon . is_null ( ) ) {
TextureRect * tf = memnew ( TextureRect ) ;
tf - > set_texture ( preview_icon ) ;
drag_preview - > add_child ( tf ) ;
}
Label * label = memnew ( Label ( preview_name ) ) ;
drag_preview - > add_child ( label ) ;
set_drag_preview ( drag_preview ) ;
Dictionary drag_data ;
drag_data [ " type " ] = " script_list_element " ; // using a custom type because node caused problems when dragging to scene tree
drag_data [ " script_list_element " ] = cur_node ;
return drag_data ;
2017-11-12 03:42:56 +01:00
}
2017-11-12 20:21:09 +01:00
2017-11-12 03:42:56 +01:00
bool ScriptEditor : : can_drop_data_fw ( const Point2 & p_point , const Variant & p_data , Control * p_from ) const {
2017-11-12 20:21:09 +01:00
Dictionary d = p_data ;
2020-05-14 16:41:43 +02:00
if ( ! d . has ( " type " ) ) {
2017-11-12 20:21:09 +01:00
return false ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
if ( String ( d [ " type " ] ) = = " script_list_element " ) {
Node * node = d [ " script_list_element " ] ;
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( node ) ;
if ( se ) {
return true ;
}
EditorHelp * eh = Object : : cast_to < EditorHelp > ( node ) ;
if ( eh ) {
return true ;
}
}
if ( String ( d [ " type " ] ) = = " nodes " ) {
Array nodes = d [ " nodes " ] ;
2020-05-14 16:41:43 +02:00
if ( nodes . size ( ) = = 0 ) {
2017-11-12 20:21:09 +01:00
return false ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
Node * node = get_node ( ( nodes [ 0 ] ) ) ;
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( node ) ;
if ( se ) {
return true ;
}
EditorHelp * eh = Object : : cast_to < EditorHelp > ( node ) ;
if ( eh ) {
return true ;
}
}
if ( String ( d [ " type " ] ) = = " files " ) {
Vector < String > files = d [ " files " ] ;
2020-05-14 16:41:43 +02:00
if ( files . size ( ) = = 0 ) {
2017-11-12 20:21:09 +01:00
return false ; //weird
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
for ( int i = 0 ; i < files . size ( ) ; i + + ) {
String file = files [ i ] ;
2020-05-14 16:41:43 +02:00
if ( file = = " " | | ! FileAccess : : exists ( file ) ) {
2017-11-12 20:21:09 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
Ref < Script > scr = ResourceLoader : : load ( file ) ;
if ( scr . is_valid ( ) ) {
return true ;
}
}
return true ;
}
2017-11-12 03:42:56 +01:00
return false ;
}
2017-11-12 20:21:09 +01:00
2017-11-12 03:42:56 +01:00
void ScriptEditor : : drop_data_fw ( const Point2 & p_point , const Variant & p_data , Control * p_from ) {
2020-05-14 16:41:43 +02:00
if ( ! can_drop_data_fw ( p_point , p_data , p_from ) ) {
2017-11-12 20:21:09 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
Dictionary d = p_data ;
2020-05-14 16:41:43 +02:00
if ( ! d . has ( " type " ) ) {
2017-11-12 20:21:09 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
if ( String ( d [ " type " ] ) = = " script_list_element " ) {
Node * node = d [ " script_list_element " ] ;
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( node ) ;
EditorHelp * eh = Object : : cast_to < EditorHelp > ( node ) ;
if ( se | | eh ) {
2019-11-25 17:16:29 +01:00
int new_index = 0 ;
if ( script_list - > get_item_count ( ) > 0 ) {
new_index = script_list - > get_item_metadata ( script_list - > get_item_at_position ( p_point ) ) ;
}
2017-11-12 20:21:09 +01:00
tab_container - > move_child ( node , new_index ) ;
tab_container - > set_current_tab ( new_index ) ;
_update_script_names ( ) ;
}
}
if ( String ( d [ " type " ] ) = = " nodes " ) {
Array nodes = d [ " nodes " ] ;
2020-05-14 16:41:43 +02:00
if ( nodes . size ( ) = = 0 ) {
2017-11-12 20:21:09 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
Node * node = get_node ( nodes [ 0 ] ) ;
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( node ) ;
EditorHelp * eh = Object : : cast_to < EditorHelp > ( node ) ;
if ( se | | eh ) {
2019-11-25 17:16:29 +01:00
int new_index = 0 ;
if ( script_list - > get_item_count ( ) > 0 ) {
new_index = script_list - > get_item_metadata ( script_list - > get_item_at_position ( p_point ) ) ;
}
2017-11-12 20:21:09 +01:00
tab_container - > move_child ( node , new_index ) ;
tab_container - > set_current_tab ( new_index ) ;
_update_script_names ( ) ;
}
}
if ( String ( d [ " type " ] ) = = " files " ) {
Vector < String > files = d [ " files " ] ;
2019-11-25 17:16:29 +01:00
int new_index = 0 ;
if ( script_list - > get_item_count ( ) > 0 ) {
new_index = script_list - > get_item_metadata ( script_list - > get_item_at_position ( p_point ) ) ;
}
2017-11-12 20:21:09 +01:00
int num_tabs_before = tab_container - > get_child_count ( ) ;
for ( int i = 0 ; i < files . size ( ) ; i + + ) {
String file = files [ i ] ;
2020-05-14 16:41:43 +02:00
if ( file = = " " | | ! FileAccess : : exists ( file ) ) {
2017-11-12 20:21:09 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2017-11-12 20:21:09 +01:00
Ref < Script > scr = ResourceLoader : : load ( file ) ;
if ( scr . is_valid ( ) ) {
edit ( scr ) ;
if ( tab_container - > get_child_count ( ) > num_tabs_before ) {
tab_container - > move_child ( tab_container - > get_child ( tab_container - > get_child_count ( ) - 1 ) , new_index ) ;
num_tabs_before = tab_container - > get_child_count ( ) ;
2019-06-03 21:57:06 +02:00
} else { /* Maybe script was already open */
2017-11-12 20:21:09 +01:00
tab_container - > move_child ( tab_container - > get_child ( tab_container - > get_current_tab ( ) ) , new_index ) ;
}
}
}
tab_container - > set_current_tab ( new_index ) ;
_update_script_names ( ) ;
}
2017-11-12 03:42:56 +01:00
}
2020-09-17 03:40:00 +02:00
void ScriptEditor : : _unhandled_key_input ( const Ref < InputEvent > & p_event ) {
2021-04-05 08:52:21 +02:00
ERR_FAIL_COND ( p_event . is_null ( ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! is_visible_in_tree ( ) | | ! p_event - > is_pressed ( ) | | p_event - > is_echo ( ) ) {
2017-11-12 03:42:56 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-06-26 02:02:17 +02:00
if ( ED_IS_SHORTCUT ( " script_editor/next_script " , p_event ) ) {
2019-11-25 17:16:29 +01:00
if ( script_list - > get_item_count ( ) > 1 ) {
int next_tab = script_list - > get_current ( ) + 1 ;
next_tab % = script_list - > get_item_count ( ) ;
_go_to_tab ( script_list - > get_item_metadata ( next_tab ) ) ;
_update_script_names ( ) ;
}
2016-06-26 02:02:17 +02:00
}
if ( ED_IS_SHORTCUT ( " script_editor/prev_script " , p_event ) ) {
2019-11-25 17:16:29 +01:00
if ( script_list - > get_item_count ( ) > 1 ) {
int next_tab = script_list - > get_current ( ) - 1 ;
next_tab = next_tab > = 0 ? next_tab : script_list - > get_item_count ( ) - 1 ;
_go_to_tab ( script_list - > get_item_metadata ( next_tab ) ) ;
_update_script_names ( ) ;
}
2016-06-26 02:02:17 +02:00
}
2017-11-12 03:42:56 +01:00
if ( ED_IS_SHORTCUT ( " script_editor/window_move_up " , p_event ) ) {
_menu_option ( WINDOW_MOVE_UP ) ;
}
if ( ED_IS_SHORTCUT ( " script_editor/window_move_down " , p_event ) ) {
_menu_option ( WINDOW_MOVE_DOWN ) ;
}
2016-06-26 02:02:17 +02:00
}
2017-11-11 22:36:46 +01:00
void ScriptEditor : : _script_list_gui_input ( const Ref < InputEvent > & ev ) {
2017-11-12 20:21:09 +01:00
Ref < InputEventMouseButton > mb = ev ;
2018-08-10 10:37:04 +02:00
if ( mb . is_valid ( ) & & mb - > is_pressed ( ) ) {
switch ( mb - > get_button_index ( ) ) {
2021-01-08 04:37:37 +01:00
case MOUSE_BUTTON_MIDDLE : {
2018-08-10 10:37:04 +02:00
// Right-click selects automatically; middle-click does not.
int idx = script_list - > get_item_at_position ( mb - > get_position ( ) , true ) ;
if ( idx > = 0 ) {
script_list - > select ( idx ) ;
_script_selected ( idx ) ;
_menu_option ( FILE_CLOSE ) ;
}
} break ;
2017-11-11 22:36:46 +01:00
2021-01-08 04:37:37 +01:00
case MOUSE_BUTTON_RIGHT : {
2018-08-10 10:37:04 +02:00
_make_script_list_context_menu ( ) ;
} break ;
}
2017-11-11 22:36:46 +01:00
}
}
void ScriptEditor : : _make_script_list_context_menu ( ) {
context_menu - > clear ( ) ;
int selected = tab_container - > get_current_tab ( ) ;
2020-05-14 16:41:43 +02:00
if ( selected < 0 | | selected > = tab_container - > get_child_count ( ) ) {
2017-11-11 22:36:46 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-11-11 22:36:46 +01:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( selected ) ) ;
if ( se ) {
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/save " ) , FILE_SAVE ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/save_as " ) , FILE_SAVE_AS ) ;
2018-01-02 08:10:49 +01:00
}
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/close_file " ) , FILE_CLOSE ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/close_all " ) , CLOSE_ALL ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/close_other_tabs " ) , CLOSE_OTHER_TABS ) ;
context_menu - > add_separator ( ) ;
if ( se ) {
2018-07-25 20:20:11 +02:00
Ref < Script > scr = se - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
if ( scr ! = nullptr ) {
2018-07-25 20:20:11 +02:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/reload_script_soft " ) , FILE_TOOL_RELOAD_SOFT ) ;
if ( ! scr . is_null ( ) & & scr - > is_tool ( ) ) {
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/run_file " ) , FILE_RUN ) ;
2018-01-02 08:10:49 +01:00
context_menu - > add_separator ( ) ;
2018-07-25 20:20:11 +02:00
}
2017-11-11 22:36:46 +01:00
}
2018-01-02 08:10:49 +01:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/copy_path " ) , FILE_COPY_PATH ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/show_in_file_system " ) , SHOW_IN_FILE_SYSTEM ) ;
context_menu - > add_separator ( ) ;
2017-11-11 22:36:46 +01:00
}
2017-11-12 03:42:56 +01:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/window_move_up " ) , WINDOW_MOVE_UP ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/window_move_down " ) , WINDOW_MOVE_DOWN ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/window_sort " ) , WINDOW_SORT ) ;
2017-11-11 22:36:46 +01:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_editor/toggle_scripts_panel " ) , TOGGLE_SCRIPTS_PANEL ) ;
context_menu - > set_position ( get_global_transform ( ) . xform ( get_local_mouse_position ( ) ) ) ;
context_menu - > set_size ( Vector2 ( 1 , 1 ) ) ;
context_menu - > popup ( ) ;
}
2015-06-22 05:03:19 +02:00
void ScriptEditor : : set_window_layout ( Ref < ConfigFile > p_layout ) {
2017-03-05 16:44:50 +01:00
if ( ! bool ( EDITOR_DEF ( " text_editor/files/restore_scripts_on_load " , true ) ) ) {
2015-06-22 05:03:19 +02:00
return ;
}
2020-05-14 16:41:43 +02:00
if ( ! p_layout - > has_section_key ( " ScriptEditor " , " open_scripts " ) & & ! p_layout - > has_section_key ( " ScriptEditor " , " open_help " ) ) {
2015-06-22 05:03:19 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2015-06-22 05:03:19 +02:00
2017-03-05 16:44:50 +01:00
Array scripts = p_layout - > get_value ( " ScriptEditor " , " open_scripts " ) ;
2015-11-17 13:46:08 +01:00
Array helps ;
2020-05-14 16:41:43 +02:00
if ( p_layout - > has_section_key ( " ScriptEditor " , " open_help " ) ) {
2017-03-05 16:44:50 +01:00
helps = p_layout - > get_value ( " ScriptEditor " , " open_help " ) ;
2020-05-14 16:41:43 +02:00
}
2015-06-22 05:03:19 +02:00
2017-03-05 16:44:50 +01:00
restoring_layout = true ;
2015-06-22 05:03:19 +02:00
2018-05-28 17:52:28 +02:00
List < String > extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " Script " , & extensions ) ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < scripts . size ( ) ; i + + ) {
2015-06-22 05:03:19 +02:00
String path = scripts [ i ] ;
2019-04-13 13:43:35 +02:00
Dictionary script_info = scripts [ i ] ;
2020-12-15 13:04:21 +01:00
if ( ! script_info . is_empty ( ) ) {
2019-04-13 13:43:35 +02:00
path = script_info [ " path " ] ;
}
2020-05-14 16:41:43 +02:00
if ( ! FileAccess : : exists ( path ) ) {
2016-07-08 02:27:20 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2018-05-28 17:52:28 +02:00
if ( extensions . find ( path . get_extension ( ) ) ) {
Ref < Script > scr = ResourceLoader : : load ( path ) ;
2019-04-13 13:43:35 +02:00
if ( ! scr . is_valid ( ) ) {
2018-05-28 17:52:28 +02:00
continue ;
}
2019-11-20 10:09:59 +01:00
if ( ! edit ( scr , false ) ) {
2019-04-23 21:52:30 +02:00
continue ;
}
2019-04-13 13:43:35 +02:00
} else {
Error error ;
Ref < TextFile > text_file = _load_text_file ( path , & error ) ;
if ( error ! = OK | | ! text_file . is_valid ( ) ) {
continue ;
}
2019-11-20 10:09:59 +01:00
if ( ! edit ( text_file , false ) ) {
2019-04-23 21:52:30 +02:00
continue ;
}
2018-05-28 17:52:28 +02:00
}
2020-12-15 13:04:21 +01:00
if ( ! script_info . is_empty ( ) ) {
2019-04-23 21:52:30 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( tab_container - > get_tab_count ( ) - 1 ) ) ;
if ( se ) {
se - > set_edit_state ( script_info [ " state " ] ) ;
}
2015-06-22 05:03:19 +02:00
}
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < helps . size ( ) ; i + + ) {
2015-11-17 13:46:08 +01:00
String path = helps [ i ] ;
2017-07-02 12:31:08 +02:00
if ( path = = " " ) { // invalid, skip
continue ;
}
2015-11-17 13:46:08 +01:00
_help_class_open ( path ) ;
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
tab_container - > get_child ( i ) - > set_meta ( " __editor_pass " , Variant ( ) ) ;
2015-11-17 13:46:08 +01:00
}
2017-03-05 16:44:50 +01:00
if ( p_layout - > has_section_key ( " ScriptEditor " , " split_offset " ) ) {
script_split - > set_split_offset ( p_layout - > get_value ( " ScriptEditor " , " split_offset " ) ) ;
2015-06-22 05:03:19 +02:00
}
2017-03-05 16:44:50 +01:00
restoring_layout = false ;
2015-06-22 05:03:19 +02:00
2016-07-18 17:30:43 +02:00
_update_script_names ( ) ;
2015-06-22 05:03:19 +02:00
}
void ScriptEditor : : get_window_layout ( Ref < ConfigFile > p_layout ) {
Array scripts ;
2015-11-17 13:46:08 +01:00
Array helps ;
2015-06-22 05:03:19 +02:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-03 00:11:05 +02:00
if ( se ) {
2018-05-28 17:52:28 +02:00
String path = se - > get_edited_resource ( ) - > get_path ( ) ;
2020-05-14 16:41:43 +02:00
if ( ! path . is_resource_file ( ) ) {
2015-11-17 13:46:08 +01:00
continue ;
2020-05-14 16:41:43 +02:00
}
2015-11-17 13:46:08 +01:00
2019-04-13 13:43:35 +02:00
Dictionary script_info ;
script_info [ " path " ] = path ;
script_info [ " state " ] = se - > get_edit_state ( ) ;
scripts . push_back ( script_info ) ;
2015-11-17 13:46:08 +01:00
}
2017-08-24 22:58:51 +02:00
EditorHelp * eh = Object : : cast_to < EditorHelp > ( tab_container - > get_child ( i ) ) ;
2015-11-17 13:46:08 +01:00
if ( eh ) {
2017-01-03 03:03:46 +01:00
helps . push_back ( eh - > get_class ( ) ) ;
2015-11-17 13:46:08 +01:00
}
2015-06-22 05:03:19 +02:00
}
2017-03-05 16:44:50 +01:00
p_layout - > set_value ( " ScriptEditor " , " open_scripts " , scripts ) ;
p_layout - > set_value ( " ScriptEditor " , " open_help " , helps ) ;
p_layout - > set_value ( " ScriptEditor " , " split_offset " , script_split - > get_split_offset ( ) ) ;
2015-06-22 05:03:19 +02:00
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _help_class_open ( const String & p_class ) {
2020-05-14 16:41:43 +02:00
if ( p_class = = " " ) {
2016-11-25 00:46:55 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
EditorHelp * eh = Object : : cast_to < EditorHelp > ( tab_container - > get_child ( i ) ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
if ( eh & & eh - > get_class ( ) = = p_class ) {
2015-11-17 13:46:08 +01:00
_go_to_tab ( i ) ;
_update_script_names ( ) ;
return ;
}
}
2017-03-05 16:44:50 +01:00
EditorHelp * eh = memnew ( EditorHelp ) ;
2015-11-17 13:46:08 +01:00
eh - > set_name ( p_class ) ;
tab_container - > add_child ( eh ) ;
2017-03-05 16:44:50 +01:00
_go_to_tab ( tab_container - > get_tab_count ( ) - 1 ) ;
eh - > go_to_class ( p_class , 0 ) ;
2020-02-21 18:28:45 +01:00
eh - > connect ( " go_to_help " , callable_mp ( this , & ScriptEditor : : _help_class_goto ) ) ;
2018-01-08 01:31:36 +01:00
_add_recent_script ( p_class ) ;
2018-09-08 20:33:54 +02:00
_sort_list_on_update = true ;
2015-11-17 13:46:08 +01:00
_update_script_names ( ) ;
2016-07-18 17:30:43 +02:00
_save_layout ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _help_class_goto ( const String & p_desc ) {
String cname = p_desc . get_slice ( " : " , 1 ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
EditorHelp * eh = Object : : cast_to < EditorHelp > ( tab_container - > get_child ( i ) ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
if ( eh & & eh - > get_class ( ) = = cname ) {
2015-11-17 13:46:08 +01:00
_go_to_tab ( i ) ;
eh - > go_to_help ( p_desc ) ;
_update_script_names ( ) ;
return ;
}
}
2017-03-05 16:44:50 +01:00
EditorHelp * eh = memnew ( EditorHelp ) ;
2015-11-17 13:46:08 +01:00
eh - > set_name ( cname ) ;
tab_container - > add_child ( eh ) ;
2017-03-05 16:44:50 +01:00
_go_to_tab ( tab_container - > get_tab_count ( ) - 1 ) ;
2015-11-17 13:46:08 +01:00
eh - > go_to_help ( p_desc ) ;
2020-02-21 18:28:45 +01:00
eh - > connect ( " go_to_help " , callable_mp ( this , & ScriptEditor : : _help_class_goto ) ) ;
2018-01-08 01:31:36 +01:00
_add_recent_script ( eh - > get_class ( ) ) ;
2018-09-08 20:33:54 +02:00
_sort_list_on_update = true ;
2015-11-17 13:46:08 +01:00
_update_script_names ( ) ;
2016-07-18 17:30:43 +02:00
_save_layout ( ) ;
2015-11-17 13:46:08 +01:00
}
2020-11-29 03:37:57 +01:00
void ScriptEditor : : update_doc ( const String & p_name ) {
ERR_FAIL_COND ( ! EditorHelp : : get_doc_data ( ) - > has_doc ( p_name ) ) ;
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
EditorHelp * eh = Object : : cast_to < EditorHelp > ( tab_container - > get_child ( i ) ) ;
if ( eh & & eh - > get_class ( ) = = p_name ) {
eh - > update_doc ( ) ;
return ;
}
}
}
2016-08-03 00:11:05 +02:00
void ScriptEditor : : _update_selected_editor_menu ( ) {
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2016-08-03 00:11:05 +02:00
bool current = tab_container - > get_current_tab ( ) = = i ;
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2016-08-03 00:11:05 +02:00
if ( se & & se - > get_edit_menu ( ) ) {
2020-05-14 16:41:43 +02:00
if ( current ) {
2016-08-03 00:11:05 +02:00
se - > get_edit_menu ( ) - > show ( ) ;
2020-05-14 16:41:43 +02:00
} else {
2016-08-03 00:11:05 +02:00
se - > get_edit_menu ( ) - > hide ( ) ;
2020-05-14 16:41:43 +02:00
}
2016-08-03 00:11:05 +02:00
}
}
2017-08-24 22:58:51 +02:00
EditorHelp * eh = Object : : cast_to < EditorHelp > ( tab_container - > get_current_tab_control ( ) ) ;
2019-05-03 01:07:06 +02:00
script_search_menu - > get_popup ( ) - > clear ( ) ;
2016-10-30 16:07:16 +01:00
if ( eh ) {
2019-05-03 01:07:06 +02:00
script_search_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/find " , TTR ( " Find... " ) , KEY_MASK_CMD | KEY_F ) , HELP_SEARCH_FIND ) ;
script_search_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/find_next " , TTR ( " Find Next " ) , KEY_F3 ) , HELP_SEARCH_FIND_NEXT ) ;
2019-10-09 17:41:49 +02:00
script_search_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/find_previous " , TTR ( " Find Previous " ) , KEY_MASK_SHIFT | KEY_F3 ) , HELP_SEARCH_FIND_PREVIOUS ) ;
2019-05-03 01:07:06 +02:00
script_search_menu - > get_popup ( ) - > add_separator ( ) ;
script_search_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/find_in_files " , TTR ( " Find in Files " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_F ) , SEARCH_IN_FILES ) ;
2016-10-30 16:07:16 +01:00
script_search_menu - > show ( ) ;
} else {
2019-05-03 01:07:06 +02:00
if ( tab_container - > get_child_count ( ) = = 0 ) {
script_search_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/find_in_files " , TTR ( " Find in Files " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_F ) , SEARCH_IN_FILES ) ;
script_search_menu - > show ( ) ;
} else {
script_search_menu - > hide ( ) ;
}
2016-10-30 16:07:16 +01:00
}
2016-08-03 00:11:05 +02:00
}
2015-11-17 13:46:08 +01:00
void ScriptEditor : : _update_history_pos ( int p_new_pos ) {
Node * n = tab_container - > get_current_tab_control ( ) ;
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < ScriptEditorBase > ( n ) ) {
2018-07-25 03:11:03 +02:00
history . write [ history_pos ] . state = Object : : cast_to < ScriptEditorBase > ( n ) - > get_edit_state ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < EditorHelp > ( n ) ) {
2018-07-25 03:11:03 +02:00
history . write [ history_pos ] . state = Object : : cast_to < EditorHelp > ( n ) - > get_scroll ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-03-05 16:44:50 +01:00
history_pos = p_new_pos ;
2015-11-17 13:46:08 +01:00
tab_container - > set_current_tab ( history [ history_pos ] . control - > get_index ( ) ) ;
n = history [ history_pos ] . control ;
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < ScriptEditorBase > ( n ) ) {
Object : : cast_to < ScriptEditorBase > ( n ) - > set_edit_state ( history [ history_pos ] . state ) ;
Object : : cast_to < ScriptEditorBase > ( n ) - > ensure_focus ( ) ;
2017-07-06 09:18:20 +02:00
2018-05-28 17:52:28 +02:00
Ref < Script > script = Object : : cast_to < ScriptEditorBase > ( n ) - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
if ( script ! = nullptr ) {
2018-05-28 17:52:28 +02:00
notify_script_changed ( script ) ;
}
2015-11-17 13:46:08 +01:00
}
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < EditorHelp > ( n ) ) {
Object : : cast_to < EditorHelp > ( n ) - > set_scroll ( history [ history_pos ] . state ) ;
Object : : cast_to < EditorHelp > ( n ) - > set_focused ( ) ;
2015-11-17 13:46:08 +01:00
}
2017-03-05 16:44:50 +01:00
n - > set_meta ( " __editor_pass " , + + edit_pass ) ;
2015-11-17 13:46:08 +01:00
_update_script_names ( ) ;
_update_history_arrows ( ) ;
2016-08-03 00:11:05 +02:00
_update_selected_editor_menu ( ) ;
2015-11-17 13:46:08 +01:00
}
void ScriptEditor : : _history_forward ( ) {
2017-03-05 16:44:50 +01:00
if ( history_pos < history . size ( ) - 1 ) {
_update_history_pos ( history_pos + 1 ) ;
2015-11-17 13:46:08 +01:00
}
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _history_back ( ) {
if ( history_pos > 0 ) {
_update_history_pos ( history_pos - 1 ) ;
2015-11-17 13:46:08 +01:00
}
}
2017-07-06 09:18:20 +02:00
2020-03-17 07:33:00 +01:00
Vector < Ref < Script > > ScriptEditor : : get_open_scripts ( ) const {
Vector < Ref < Script > > out_scripts = Vector < Ref < Script > > ( ) ;
2017-07-06 09:18:20 +02:00
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
2017-08-24 22:58:51 +02:00
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
2020-05-14 16:41:43 +02:00
if ( ! se ) {
2017-07-06 09:18:20 +02:00
continue ;
2020-05-14 16:41:43 +02:00
}
2018-05-28 17:52:28 +02:00
Ref < Script > script = se - > get_edited_resource ( ) ;
2020-04-02 01:20:12 +02:00
if ( script ! = nullptr ) {
2018-05-28 17:52:28 +02:00
out_scripts . push_back ( script ) ;
}
2017-07-06 09:18:20 +02:00
}
return out_scripts ;
}
2020-03-07 15:29:44 +01:00
Array ScriptEditor : : _get_open_script_editors ( ) const {
Array script_editors ;
for ( int i = 0 ; i < tab_container - > get_child_count ( ) ; i + + ) {
ScriptEditorBase * se = Object : : cast_to < ScriptEditorBase > ( tab_container - > get_child ( i ) ) ;
if ( ! se ) {
continue ;
}
script_editors . push_back ( se ) ;
}
return script_editors ;
}
2017-03-05 16:44:50 +01:00
void ScriptEditor : : set_scene_root_script ( Ref < Script > p_script ) {
2020-07-26 23:18:05 +02:00
// Don't open dominant script if using an external editor.
2020-07-27 10:29:22 +02:00
const bool use_external_editor =
EditorSettings : : get_singleton ( ) - > get ( " text_editor/external/use_external_editor " ) | |
2020-07-27 15:05:14 +02:00
( p_script . is_valid ( ) & & p_script - > get_language ( ) - > overrides_external_editor ( ) ) ;
2020-07-26 23:18:05 +02:00
const bool open_dominant = EditorSettings : : get_singleton ( ) - > get ( " text_editor/files/open_dominant_script_on_scene_change " ) ;
2015-12-14 12:21:18 +01:00
2020-07-26 23:18:05 +02:00
if ( open_dominant & & ! use_external_editor & & p_script . is_valid ( ) ) {
2015-11-17 13:46:08 +01:00
edit ( p_script ) ;
}
}
2017-04-15 19:48:10 +02:00
bool ScriptEditor : : script_goto_method ( Ref < Script > p_script , const String & p_method ) {
int line = p_script - > get_member_line ( p_method ) ;
2016-06-12 01:01:17 +02:00
2020-05-14 16:41:43 +02:00
if ( line = = - 1 ) {
2017-04-15 19:48:10 +02:00
return false ;
2020-05-14 16:41:43 +02:00
}
2017-04-15 19:48:10 +02:00
return edit ( p_script , line , 0 ) ;
2016-08-03 00:11:05 +02:00
}
2016-06-12 01:01:17 +02:00
2016-08-03 00:11:05 +02:00
void ScriptEditor : : set_live_auto_reload_running_scripts ( bool p_enabled ) {
2017-03-05 16:44:50 +01:00
auto_reload_running_scripts = p_enabled ;
2016-08-03 00:11:05 +02:00
}
2016-06-12 01:01:17 +02:00
2016-08-03 00:11:05 +02:00
void ScriptEditor : : _help_search ( String p_text ) {
2018-10-02 12:07:44 +02:00
help_search_dialog - > popup_dialog ( p_text ) ;
2016-08-03 00:11:05 +02:00
}
2016-06-12 01:01:17 +02:00
2017-03-05 16:44:50 +01:00
void ScriptEditor : : _open_script_request ( const String & p_path ) {
2016-08-25 22:45:20 +02:00
Ref < Script > script = ResourceLoader : : load ( p_path ) ;
if ( script . is_valid ( ) ) {
2017-03-05 16:44:50 +01:00
script_editor - > edit ( script , false ) ;
2018-05-28 17:52:28 +02:00
return ;
}
Error err ;
Ref < TextFile > text_file = script_editor - > _load_text_file ( p_path , & err ) ;
if ( text_file . is_valid ( ) ) {
script_editor - > edit ( text_file , false ) ;
return ;
2016-08-25 22:45:20 +02:00
}
}
2016-06-12 01:01:17 +02:00
2020-05-03 18:08:15 +02:00
void ScriptEditor : : register_syntax_highlighter ( const Ref < EditorSyntaxHighlighter > & p_syntax_highlighter ) {
ERR_FAIL_COND ( p_syntax_highlighter . is_null ( ) ) ;
2020-03-07 15:29:44 +01:00
if ( syntax_highlighters . find ( p_syntax_highlighter ) = = - 1 ) {
syntax_highlighters . push_back ( p_syntax_highlighter ) ;
}
}
2018-04-02 13:41:44 +02:00
2020-05-03 18:08:15 +02:00
void ScriptEditor : : unregister_syntax_highlighter ( const Ref < EditorSyntaxHighlighter > & p_syntax_highlighter ) {
ERR_FAIL_COND ( p_syntax_highlighter . is_null ( ) ) ;
2020-03-07 15:29:44 +01:00
syntax_highlighters . erase ( p_syntax_highlighter ) ;
2018-04-02 13:41:44 +02:00
}
2017-03-05 16:44:50 +01:00
int ScriptEditor : : script_editor_func_count = 0 ;
2016-08-03 00:11:05 +02:00
CreateScriptEditorFunc ScriptEditor : : script_editor_funcs [ ScriptEditor : : SCRIPT_EDITOR_FUNC_MAX ] ;
2016-06-12 01:01:17 +02:00
2016-08-03 00:11:05 +02:00
void ScriptEditor : : register_create_script_editor_function ( CreateScriptEditorFunc p_func ) {
2017-03-05 16:44:50 +01:00
ERR_FAIL_COND ( script_editor_func_count = = SCRIPT_EDITOR_FUNC_MAX ) ;
script_editor_funcs [ script_editor_func_count + + ] = p_func ;
2016-06-03 17:34:11 +02:00
}
2017-08-18 21:43:23 +02:00
void ScriptEditor : : _script_changed ( ) {
NodeDock : : singleton - > update_lists ( ) ;
}
2018-02-12 02:36:15 +01:00
void ScriptEditor : : _on_find_in_files_requested ( String text ) {
2020-02-09 10:10:58 +01:00
find_in_files_dialog - > set_find_in_files_mode ( FindInFilesDialog : : SEARCH_MODE ) ;
find_in_files_dialog - > set_search_text ( text ) ;
2020-03-06 18:00:16 +01:00
find_in_files_dialog - > popup_centered ( ) ;
2020-02-09 10:10:58 +01:00
}
void ScriptEditor : : _on_replace_in_files_requested ( String text ) {
find_in_files_dialog - > set_find_in_files_mode ( FindInFilesDialog : : REPLACE_MODE ) ;
2018-02-12 02:36:15 +01:00
find_in_files_dialog - > set_search_text ( text ) ;
2020-02-09 10:10:58 +01:00
find_in_files_dialog - > set_replace_text ( " " ) ;
2020-03-06 18:00:16 +01:00
find_in_files_dialog - > popup_centered ( ) ;
2018-02-12 02:36:15 +01:00
}
void ScriptEditor : : _on_find_in_files_result_selected ( String fpath , int line_number , int begin , int end ) {
2019-08-08 04:06:33 +02:00
if ( ResourceLoader : : exists ( fpath ) ) {
RES res = ResourceLoader : : load ( fpath ) ;
if ( fpath . get_extension ( ) = = " shader " ) {
ShaderEditorPlugin * shader_editor = Object : : cast_to < ShaderEditorPlugin > ( EditorNode : : get_singleton ( ) - > get_editor_data ( ) . get_editor ( " Shader " ) ) ;
shader_editor - > edit ( res . ptr ( ) ) ;
shader_editor - > make_visible ( true ) ;
shader_editor - > get_shader_editor ( ) - > goto_line_selection ( line_number - 1 , begin , end ) ;
return ;
} else {
Ref < Script > script = res ;
if ( script . is_valid ( ) ) {
edit ( script ) ;
2019-08-07 08:41:10 +02:00
2019-08-08 04:06:33 +02:00
ScriptTextEditor * ste = Object : : cast_to < ScriptTextEditor > ( _get_current_editor ( ) ) ;
if ( ste ) {
ste - > goto_line_selection ( line_number - 1 , begin , end ) ;
}
return ;
2019-08-07 08:41:10 +02:00
}
2019-08-08 04:06:33 +02:00
}
}
2018-02-12 02:36:15 +01:00
2019-08-08 04:06:33 +02:00
// If the file is not a valid resource/script, load it as a text file.
Error err ;
Ref < TextFile > text_file = _load_text_file ( fpath , & err ) ;
if ( text_file . is_valid ( ) ) {
edit ( text_file ) ;
2019-08-07 08:41:10 +02:00
2019-08-08 04:06:33 +02:00
TextEditor * te = Object : : cast_to < TextEditor > ( _get_current_editor ( ) ) ;
if ( te ) {
te - > goto_line_selection ( line_number - 1 , begin , end ) ;
2018-10-27 17:24:41 +02:00
}
2018-02-12 02:36:15 +01:00
}
}
void ScriptEditor : : _start_find_in_files ( bool with_replace ) {
FindInFiles * f = find_in_files - > get_finder ( ) ;
f - > set_search_text ( find_in_files_dialog - > get_search_text ( ) ) ;
f - > set_match_case ( find_in_files_dialog - > is_match_case ( ) ) ;
2019-01-09 06:23:46 +01:00
f - > set_whole_words ( find_in_files_dialog - > is_whole_words ( ) ) ;
2018-02-12 02:36:15 +01:00
f - > set_folder ( find_in_files_dialog - > get_folder ( ) ) ;
f - > set_filter ( find_in_files_dialog - > get_filter ( ) ) ;
find_in_files - > set_with_replace ( with_replace ) ;
2020-02-09 10:10:58 +01:00
find_in_files - > set_replace_text ( find_in_files_dialog - > get_replace_text ( ) ) ;
2018-02-12 02:36:15 +01:00
find_in_files - > start_search ( ) ;
2018-10-27 18:49:05 +02:00
editor - > make_bottom_panel_item_visible ( find_in_files ) ;
2018-02-12 02:36:15 +01:00
}
2020-02-17 22:06:54 +01:00
void ScriptEditor : : _on_find_in_files_modified_files ( PackedStringArray paths ) {
2018-02-12 02:36:15 +01:00
_test_script_times_on_disk ( ) ;
_update_modified_scripts_for_external_editor ( ) ;
}
2019-06-03 21:57:06 +02:00
void ScriptEditor : : _filter_scripts_text_changed ( const String & p_newtext ) {
_update_script_names ( ) ;
}
void ScriptEditor : : _filter_methods_text_changed ( const String & p_newtext ) {
_update_members_overview ( ) ;
}
2015-11-17 13:46:08 +01:00
void ScriptEditor : : _bind_methods ( ) {
2017-01-03 03:03:46 +01:00
ClassDB : : bind_method ( " _close_docs_tab " , & ScriptEditor : : _close_docs_tab ) ;
ClassDB : : bind_method ( " _close_all_tabs " , & ScriptEditor : : _close_all_tabs ) ;
2017-11-20 06:15:40 +01:00
ClassDB : : bind_method ( " _close_other_tabs " , & ScriptEditor : : _close_other_tabs ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( " _goto_script_line2 " , & ScriptEditor : : _goto_script_line2 ) ;
2017-12-07 04:57:41 +01:00
ClassDB : : bind_method ( " _copy_script_path " , & ScriptEditor : : _copy_script_path ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( " _get_debug_tooltip " , & ScriptEditor : : _get_debug_tooltip ) ;
2019-04-20 13:51:25 +02:00
ClassDB : : bind_method ( " _update_script_connections " , & ScriptEditor : : _update_script_connections ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( " _help_class_open " , & ScriptEditor : : _help_class_open ) ;
ClassDB : : bind_method ( " _live_auto_reload_running_scripts " , & ScriptEditor : : _live_auto_reload_running_scripts ) ;
2020-09-17 03:40:00 +02:00
ClassDB : : bind_method ( " _unhandled_key_input " , & ScriptEditor : : _unhandled_key_input ) ;
2018-04-30 14:27:00 +02:00
ClassDB : : bind_method ( " _update_members_overview " , & ScriptEditor : : _update_members_overview ) ;
2017-10-11 10:24:57 +02:00
ClassDB : : bind_method ( " _update_recent_scripts " , & ScriptEditor : : _update_recent_scripts ) ;
2017-07-06 09:18:20 +02:00
2020-03-07 15:29:44 +01:00
ClassDB : : bind_method ( " get_current_editor " , & ScriptEditor : : _get_current_editor ) ;
ClassDB : : bind_method ( " get_open_script_editors " , & ScriptEditor : : _get_open_script_editors ) ;
ClassDB : : bind_method ( D_METHOD ( " register_syntax_highlighter " , " syntax_highlighter " ) , & ScriptEditor : : register_syntax_highlighter ) ;
ClassDB : : bind_method ( D_METHOD ( " unregister_syntax_highlighter " , " syntax_highlighter " ) , & ScriptEditor : : unregister_syntax_highlighter ) ;
2017-12-25 15:19:56 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_drag_data_fw " , " point " , " from " ) , & ScriptEditor : : get_drag_data_fw ) ;
ClassDB : : bind_method ( D_METHOD ( " can_drop_data_fw " , " point " , " data " , " from " ) , & ScriptEditor : : can_drop_data_fw ) ;
ClassDB : : bind_method ( D_METHOD ( " drop_data_fw " , " point " , " data " , " from " ) , & ScriptEditor : : drop_data_fw ) ;
2017-11-12 20:21:09 +01:00
2019-04-09 22:07:13 +02:00
ClassDB : : bind_method ( D_METHOD ( " goto_line " , " line_number " ) , & ScriptEditor : : _goto_script_line2 ) ;
2017-07-06 09:18:20 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_current_script " ) , & ScriptEditor : : _get_current_script ) ;
2017-07-19 19:47:43 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_open_scripts " ) , & ScriptEditor : : _get_open_scripts ) ;
2017-12-13 18:41:28 +01:00
ClassDB : : bind_method ( D_METHOD ( " open_script_create_dialog " , " base_name " , " base_path " ) , & ScriptEditor : : open_script_create_dialog ) ;
2017-07-06 09:18:20 +02:00
2017-08-29 07:15:46 +02:00
ADD_SIGNAL ( MethodInfo ( " editor_script_changed " , PropertyInfo ( Variant : : OBJECT , " script " , PROPERTY_HINT_RESOURCE_TYPE , " Script " ) ) ) ;
ADD_SIGNAL ( MethodInfo ( " script_close " , PropertyInfo ( Variant : : OBJECT , " script " , PROPERTY_HINT_RESOURCE_TYPE , " Script " ) ) ) ;
2015-11-17 13:46:08 +01:00
}
2015-06-22 05:03:19 +02:00
2014-02-10 02:10:30 +01:00
ScriptEditor : : ScriptEditor ( EditorNode * p_editor ) {
2016-04-12 16:45:31 +02:00
current_theme = " " ;
2017-03-05 16:44:50 +01:00
completion_cache = memnew ( EditorScriptCodeCompletionCache ) ;
restoring_layout = false ;
waiting_update_names = false ;
pending_auto_reload = false ;
2018-11-11 13:14:06 +01:00
auto_reload_running_scripts = true ;
2019-09-01 13:28:57 +02:00
members_overview_enabled = EditorSettings : : get_singleton ( ) - > get ( " text_editor/script_list/show_members_overview " ) ;
2018-03-01 21:45:04 +01:00
help_overview_enabled = EditorSettings : : get_singleton ( ) - > get ( " text_editor/help/show_help_index " ) ;
2017-03-05 16:44:50 +01:00
editor = p_editor ;
2014-02-10 02:10:30 +01:00
2017-07-15 06:40:17 +02:00
VBoxContainer * main_container = memnew ( VBoxContainer ) ;
add_child ( main_container ) ;
2017-03-05 16:44:50 +01:00
menu_hb = memnew ( HBoxContainer ) ;
2017-07-15 06:40:17 +02:00
main_container - > add_child ( menu_hb ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
script_split = memnew ( HSplitContainer ) ;
2017-07-15 06:40:17 +02:00
main_container - > add_child ( script_split ) ;
2015-06-22 05:03:19 +02:00
script_split - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2017-05-28 16:20:38 +02:00
list_split = memnew ( VSplitContainer ) ;
script_split - > add_child ( list_split ) ;
list_split - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2019-06-03 21:57:06 +02:00
scripts_vbox = memnew ( VBoxContainer ) ;
scripts_vbox - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
list_split - > add_child ( scripts_vbox ) ;
filter_scripts = memnew ( LineEdit ) ;
filter_scripts - > set_placeholder ( TTR ( " Filter scripts " ) ) ;
filter_scripts - > set_clear_button_enabled ( true ) ;
2020-02-21 18:28:45 +01:00
filter_scripts - > connect ( " text_changed " , callable_mp ( this , & ScriptEditor : : _filter_scripts_text_changed ) ) ;
2019-06-03 21:57:06 +02:00
scripts_vbox - > add_child ( filter_scripts ) ;
2017-03-05 16:44:50 +01:00
script_list = memnew ( ItemList ) ;
2019-06-03 21:57:06 +02:00
scripts_vbox - > add_child ( script_list ) ;
2020-02-05 06:16:54 +01:00
script_list - > set_custom_minimum_size ( Size2 ( 150 , 60 ) * EDSCALE ) ; //need to give a bit of limit to avoid it from disappearing
2017-07-01 02:30:17 +02:00
script_list - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2020-02-18 22:57:40 +01:00
script_split - > set_split_offset ( 70 * EDSCALE ) ;
2017-11-12 20:21:09 +01:00
_sort_list_on_update = true ;
2020-02-21 18:28:45 +01:00
script_list - > connect ( " gui_input " , callable_mp ( this , & ScriptEditor : : _script_list_gui_input ) , varray ( ) , CONNECT_DEFERRED ) ;
2017-11-11 22:36:46 +01:00
script_list - > set_allow_rmb_select ( true ) ;
2017-11-12 03:42:56 +01:00
script_list - > set_drag_forwarding ( this ) ;
2017-11-11 22:36:46 +01:00
context_menu = memnew ( PopupMenu ) ;
add_child ( context_menu ) ;
2020-02-21 18:28:45 +01:00
context_menu - > connect ( " id_pressed " , callable_mp ( this , & ScriptEditor : : _menu_option ) ) ;
2017-11-11 22:36:46 +01:00
2018-06-02 02:28:49 +02:00
overview_vbox = memnew ( VBoxContainer ) ;
overview_vbox - > set_custom_minimum_size ( Size2 ( 0 , 90 ) ) ;
overview_vbox - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2018-04-30 14:27:00 +02:00
2018-06-02 02:28:49 +02:00
list_split - > add_child ( overview_vbox ) ;
buttons_hbox = memnew ( HBoxContainer ) ;
overview_vbox - > add_child ( buttons_hbox ) ;
2018-04-30 14:27:00 +02:00
2018-06-05 23:18:11 +02:00
filename = memnew ( Label ) ;
filename - > set_clip_text ( true ) ;
filename - > set_h_size_flags ( SIZE_EXPAND_FILL ) ;
2020-03-12 13:37:40 +01:00
filename - > add_theme_style_override ( " normal " , EditorNode : : get_singleton ( ) - > get_gui_base ( ) - > get_theme_stylebox ( " normal " , " LineEdit " ) ) ;
2018-06-02 02:28:49 +02:00
buttons_hbox - > add_child ( filename ) ;
2018-06-05 23:18:11 +02:00
2020-06-19 20:49:04 +02:00
members_overview_alphabeta_sort_button = memnew ( Button ) ;
members_overview_alphabeta_sort_button - > set_flat ( true ) ;
2018-05-17 01:44:39 +02:00
members_overview_alphabeta_sort_button - > set_tooltip ( TTR ( " Toggle alphabetical sorting of the method list. " ) ) ;
2018-04-30 14:27:00 +02:00
members_overview_alphabeta_sort_button - > set_toggle_mode ( true ) ;
members_overview_alphabeta_sort_button - > set_pressed ( EditorSettings : : get_singleton ( ) - > get ( " text_editor/tools/sort_members_outline_alphabetically " ) ) ;
2020-02-21 18:28:45 +01:00
members_overview_alphabeta_sort_button - > connect ( " toggled " , callable_mp ( this , & ScriptEditor : : _toggle_members_overview_alpha_sort ) ) ;
2018-04-30 14:27:00 +02:00
2018-06-02 02:28:49 +02:00
buttons_hbox - > add_child ( members_overview_alphabeta_sort_button ) ;
2018-04-30 14:27:00 +02:00
2019-06-03 21:57:06 +02:00
filter_methods = memnew ( LineEdit ) ;
filter_methods - > set_placeholder ( TTR ( " Filter methods " ) ) ;
filter_methods - > set_clear_button_enabled ( true ) ;
2020-02-21 18:28:45 +01:00
filter_methods - > connect ( " text_changed " , callable_mp ( this , & ScriptEditor : : _filter_methods_text_changed ) ) ;
2019-06-03 21:57:06 +02:00
overview_vbox - > add_child ( filter_methods ) ;
2017-05-28 16:20:38 +02:00
members_overview = memnew ( ItemList ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > add_child ( members_overview ) ;
2018-04-30 14:27:00 +02:00
2018-02-14 04:24:57 +01:00
members_overview - > set_allow_reselect ( true ) ;
2020-02-05 06:16:54 +01:00
members_overview - > set_custom_minimum_size ( Size2 ( 0 , 60 ) * EDSCALE ) ; //need to give a bit of limit to avoid it from disappearing
2017-07-01 02:30:17 +02:00
members_overview - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2018-04-30 14:27:00 +02:00
members_overview - > set_allow_rmb_select ( true ) ;
2015-06-22 05:03:19 +02:00
2017-09-14 02:56:37 +02:00
help_overview = memnew ( ItemList ) ;
2018-06-02 02:28:49 +02:00
overview_vbox - > add_child ( help_overview ) ;
2018-02-14 04:24:57 +01:00
help_overview - > set_allow_reselect ( true ) ;
2020-02-05 06:16:54 +01:00
help_overview - > set_custom_minimum_size ( Size2 ( 0 , 60 ) * EDSCALE ) ; //need to give a bit of limit to avoid it from disappearing
2017-09-14 02:56:37 +02:00
help_overview - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2017-03-05 16:44:50 +01:00
tab_container = memnew ( TabContainer ) ;
2015-06-22 05:03:19 +02:00
tab_container - > set_tabs_visible ( false ) ;
2018-10-20 22:05:11 +02:00
tab_container - > set_custom_minimum_size ( Size2 ( 200 , 0 ) * EDSCALE ) ;
2015-06-22 05:03:19 +02:00
script_split - > add_child ( tab_container ) ;
tab_container - > set_h_size_flags ( SIZE_EXPAND_FILL ) ;
2017-11-12 20:21:09 +01:00
2017-11-12 03:42:56 +01:00
ED_SHORTCUT ( " script_editor/window_sort " , TTR ( " Sort " ) ) ;
ED_SHORTCUT ( " script_editor/window_move_up " , TTR ( " Move Up " ) , KEY_MASK_SHIFT | KEY_MASK_ALT | KEY_UP ) ;
ED_SHORTCUT ( " script_editor/window_move_down " , TTR ( " Move Down " ) , KEY_MASK_SHIFT | KEY_MASK_ALT | KEY_DOWN ) ;
ED_SHORTCUT ( " script_editor/next_script " , TTR ( " Next script " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_PERIOD ) ; // these should be KEY_GREATER and KEY_LESS but those don't work
2019-11-25 17:16:29 +01:00
ED_SHORTCUT ( " script_editor/prev_script " , TTR ( " Previous script " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_COMMA ) ;
2020-09-17 03:40:00 +02:00
set_process_unhandled_key_input ( true ) ;
2016-06-26 02:02:17 +02:00
2017-03-05 16:44:50 +01:00
file_menu = memnew ( MenuButton ) ;
2016-05-04 03:25:37 +02:00
file_menu - > set_text ( TTR ( " File " ) ) ;
2018-07-30 00:26:43 +02:00
file_menu - > set_switch_on_hover ( true ) ;
2020-09-17 03:40:00 +02:00
file_menu - > set_shortcut_context ( this ) ;
menu_hb - > add_child ( file_menu ) ;
2020-03-12 13:37:40 +01:00
2018-12-08 17:18:04 +01:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/new " , TTR ( " New Script... " ) ) , FILE_NEW ) ;
2019-08-12 22:23:00 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/new_textfile " , TTR ( " New Text File... " ) ) , FILE_NEW_TEXTFILE ) ;
2018-12-08 17:18:04 +01:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/open " , TTR ( " Open... " ) ) , FILE_OPEN ) ;
2019-07-31 21:06:15 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/reopen_closed_script " , TTR ( " Reopen Closed Script " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_T ) , FILE_REOPEN_CLOSED ) ;
2017-04-27 17:07:39 +02:00
file_menu - > get_popup ( ) - > add_submenu_item ( TTR ( " Open Recent " ) , " RecentScripts " , FILE_OPEN_RECENT ) ;
recent_scripts = memnew ( PopupMenu ) ;
recent_scripts - > set_name ( " RecentScripts " ) ;
file_menu - > get_popup ( ) - > add_child ( recent_scripts ) ;
2020-02-21 18:28:45 +01:00
recent_scripts - > connect ( " id_pressed " , callable_mp ( this , & ScriptEditor : : _open_recent_script ) ) ;
2017-04-27 17:07:39 +02:00
_update_recent_scripts ( ) ;
2015-06-22 05:03:19 +02:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
2017-03-05 16:44:50 +01:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/save " , TTR ( " Save " ) , KEY_MASK_ALT | KEY_MASK_CMD | KEY_S ) , FILE_SAVE ) ;
2018-04-22 19:36:01 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/save_as " , TTR ( " Save As... " ) ) , FILE_SAVE_AS ) ;
2020-06-24 11:02:54 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/save_all " , TTR ( " Save All " ) , KEY_MASK_SHIFT | KEY_MASK_ALT | KEY_S ) , FILE_SAVE_ALL ) ;
2015-06-22 05:03:19 +02:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
2020-05-01 12:17:14 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/reload_script_soft " , TTR ( " Soft Reload Script " ) , KEY_MASK_CMD | KEY_MASK_ALT | KEY_R ) , FILE_TOOL_RELOAD_SOFT ) ;
2017-12-07 04:57:41 +01:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/copy_path " , TTR ( " Copy Script Path " ) ) , FILE_COPY_PATH ) ;
2018-10-26 21:11:36 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/show_in_file_system " , TTR ( " Show in FileSystem " ) ) , SHOW_IN_FILE_SYSTEM ) ;
2016-08-03 00:11:05 +02:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
2018-09-25 22:44:50 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/history_previous " , TTR ( " History Previous " ) , KEY_MASK_ALT | KEY_LEFT ) , WINDOW_PREV ) ;
2017-03-05 16:44:50 +01:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/history_next " , TTR ( " History Next " ) , KEY_MASK_ALT | KEY_RIGHT ) , WINDOW_NEXT ) ;
2015-11-17 13:46:08 +01:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
2018-06-21 11:10:43 +02:00
file_menu - > get_popup ( ) - > add_submenu_item ( TTR ( " Theme " ) , " Theme " , FILE_THEME ) ;
theme_submenu = memnew ( PopupMenu ) ;
theme_submenu - > set_name ( " Theme " ) ;
file_menu - > get_popup ( ) - > add_child ( theme_submenu ) ;
2020-02-21 18:28:45 +01:00
theme_submenu - > connect ( " id_pressed " , callable_mp ( this , & ScriptEditor : : _theme_option ) ) ;
2018-12-08 17:18:04 +01:00
theme_submenu - > add_shortcut ( ED_SHORTCUT ( " script_editor/import_theme " , TTR ( " Import Theme... " ) ) , THEME_IMPORT ) ;
2018-06-21 11:10:43 +02:00
theme_submenu - > add_shortcut ( ED_SHORTCUT ( " script_editor/reload_theme " , TTR ( " Reload Theme " ) ) , THEME_RELOAD ) ;
2019-08-12 22:23:00 +02:00
2018-12-08 17:18:04 +01:00
theme_submenu - > add_separator ( ) ;
2018-06-21 11:10:43 +02:00
theme_submenu - > add_shortcut ( ED_SHORTCUT ( " script_editor/save_theme " , TTR ( " Save Theme " ) ) , THEME_SAVE ) ;
2018-12-08 17:18:04 +01:00
theme_submenu - > add_shortcut ( ED_SHORTCUT ( " script_editor/save_theme_as " , TTR ( " Save Theme As... " ) ) , THEME_SAVE_AS ) ;
2018-06-21 11:10:43 +02:00
2016-04-12 16:45:31 +02:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
2016-07-18 00:18:48 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/close_file " , TTR ( " Close " ) , KEY_MASK_CMD | KEY_W ) , FILE_CLOSE ) ;
2016-08-13 02:56:38 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/close_all " , TTR ( " Close All " ) ) , CLOSE_ALL ) ;
2017-11-20 06:15:40 +01:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/close_other_tabs " , TTR ( " Close Other Tabs " ) ) , CLOSE_OTHER_TABS ) ;
2019-08-12 22:23:00 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/close_docs " , TTR ( " Close Docs " ) ) , CLOSE_DOCS ) ;
2017-07-10 14:11:31 +02:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
2017-08-28 00:03:19 +02:00
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/run_file " , TTR ( " Run " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_X ) , FILE_RUN ) ;
2019-12-17 14:51:49 +01:00
file_menu - > get_popup ( ) - > add_separator ( ) ;
file_menu - > get_popup ( ) - > add_shortcut ( ED_SHORTCUT ( " script_editor/toggle_scripts_panel " , TTR ( " Toggle Scripts Panel " ) , KEY_MASK_CMD | KEY_BACKSLASH ) , TOGGLE_SCRIPTS_PANEL ) ;
2020-02-21 18:28:45 +01:00
file_menu - > get_popup ( ) - > connect ( " id_pressed " , callable_mp ( this , & ScriptEditor : : _menu_option ) ) ;
2016-08-03 00:11:05 +02:00
2017-03-05 16:44:50 +01:00
script_search_menu = memnew ( MenuButton ) ;
2016-05-04 03:25:37 +02:00
script_search_menu - > set_text ( TTR ( " Search " ) ) ;
2018-07-30 00:26:43 +02:00
script_search_menu - > set_switch_on_hover ( true ) ;
2020-09-17 03:40:00 +02:00
script_search_menu - > set_shortcut_context ( this ) ;
2020-02-21 18:28:45 +01:00
script_search_menu - > get_popup ( ) - > connect ( " id_pressed " , callable_mp ( this , & ScriptEditor : : _menu_option ) ) ;
2020-09-17 03:40:00 +02:00
menu_hb - > add_child ( script_search_menu ) ;
2015-11-17 13:46:08 +01:00
2020-02-07 02:52:05 +01:00
MenuButton * debug_menu = memnew ( MenuButton ) ;
2014-02-10 02:10:30 +01:00
menu_hb - > add_child ( debug_menu ) ;
2020-02-07 02:52:05 +01:00
debug_menu - > hide ( ) ; // Handled by EditorDebuggerNode below.
EditorDebuggerNode * debugger = EditorDebuggerNode : : get_singleton ( ) ;
debugger - > set_script_debug_button ( debug_menu ) ;
2020-02-21 18:28:45 +01:00
debugger - > connect ( " goto_script_line " , callable_mp ( this , & ScriptEditor : : _goto_script_line ) ) ;
debugger - > connect ( " set_execution " , callable_mp ( this , & ScriptEditor : : _set_execution ) ) ;
debugger - > connect ( " clear_execution " , callable_mp ( this , & ScriptEditor : : _clear_execution ) ) ;
debugger - > connect ( " breaked " , callable_mp ( this , & ScriptEditor : : _breaked ) ) ;
2014-02-10 02:10:30 +01:00
2015-11-17 13:46:08 +01:00
menu_hb - > add_spacer ( ) ;
2017-03-05 16:44:50 +01:00
script_icon = memnew ( TextureRect ) ;
2015-11-17 13:46:08 +01:00
menu_hb - > add_child ( script_icon ) ;
2017-03-05 16:44:50 +01:00
script_name_label = memnew ( Label ) ;
2015-11-17 13:46:08 +01:00
menu_hb - > add_child ( script_name_label ) ;
script_icon - > hide ( ) ;
script_name_label - > hide ( ) ;
menu_hb - > add_spacer ( ) ;
2020-06-19 20:49:04 +02:00
site_search = memnew ( Button ) ;
site_search - > set_flat ( true ) ;
2017-05-26 16:34:41 +02:00
site_search - > set_text ( TTR ( " Online Docs " ) ) ;
2020-02-21 18:28:45 +01:00
site_search - > connect ( " pressed " , callable_mp ( this , & ScriptEditor : : _menu_option ) , varray ( SEARCH_WEBSITE ) ) ;
2015-11-17 13:46:08 +01:00
menu_hb - > add_child ( site_search ) ;
2019-03-05 18:04:22 +01:00
site_search - > set_tooltip ( TTR ( " Open Godot online documentation. " ) ) ;
2015-11-17 13:46:08 +01:00
2020-06-19 20:49:04 +02:00
help_search = memnew ( Button ) ;
help_search - > set_flat ( true ) ;
2016-05-04 03:25:37 +02:00
help_search - > set_text ( TTR ( " Search Help " ) ) ;
2020-02-21 18:28:45 +01:00
help_search - > connect ( " pressed " , callable_mp ( this , & ScriptEditor : : _menu_option ) , varray ( SEARCH_HELP ) ) ;
2015-11-17 13:46:08 +01:00
menu_hb - > add_child ( help_search ) ;
2016-05-04 03:25:37 +02:00
help_search - > set_tooltip ( TTR ( " Search the reference documentation. " ) ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
menu_hb - > add_child ( memnew ( VSeparator ) ) ;
2015-11-17 13:46:08 +01:00
2020-06-19 20:49:04 +02:00
script_back = memnew ( Button ) ;
script_back - > set_flat ( true ) ;
2020-02-21 18:28:45 +01:00
script_back - > connect ( " pressed " , callable_mp ( this , & ScriptEditor : : _history_back ) ) ;
2015-11-17 13:46:08 +01:00
menu_hb - > add_child ( script_back ) ;
script_back - > set_disabled ( true ) ;
2016-05-04 03:25:37 +02:00
script_back - > set_tooltip ( TTR ( " Go to previous edited document. " ) ) ;
2015-11-17 13:46:08 +01:00
2020-06-19 20:49:04 +02:00
script_forward = memnew ( Button ) ;
script_forward - > set_flat ( true ) ;
2020-02-21 18:28:45 +01:00
script_forward - > connect ( " pressed " , callable_mp ( this , & ScriptEditor : : _history_forward ) ) ;
2015-11-17 13:46:08 +01:00
menu_hb - > add_child ( script_forward ) ;
script_forward - > set_disabled ( true ) ;
2016-05-04 03:25:37 +02:00
script_forward - > set_tooltip ( TTR ( " Go to next edited document. " ) ) ;
2015-11-17 13:46:08 +01:00
2020-02-21 18:28:45 +01:00
tab_container - > connect ( " tab_changed " , callable_mp ( this , & ScriptEditor : : _tab_changed ) ) ;
2015-11-17 13:46:08 +01:00
2017-03-05 16:44:50 +01:00
erase_tab_confirm = memnew ( ConfirmationDialog ) ;
2020-12-14 19:37:30 +01:00
erase_tab_confirm - > get_ok_button ( ) - > set_text ( TTR ( " Save " ) ) ;
2020-07-09 01:02:38 +02:00
erase_tab_confirm - > add_button ( TTR ( " Discard " ) , DisplayServer : : get_singleton ( ) - > get_swap_cancel_ok ( ) , " discard " ) ;
2021-03-04 01:04:31 +01:00
erase_tab_confirm - > connect ( " confirmed " , callable_mp ( this , & ScriptEditor : : _close_current_tab ) , varray ( true ) ) ;
2020-02-21 18:28:45 +01:00
erase_tab_confirm - > connect ( " custom_action " , callable_mp ( this , & ScriptEditor : : _close_discard_current_tab ) ) ;
2017-01-02 19:45:59 +01:00
add_child ( erase_tab_confirm ) ;
2014-02-10 02:10:30 +01:00
2015-08-10 01:39:59 +02:00
script_create_dialog = memnew ( ScriptCreateDialog ) ;
2016-05-04 03:25:37 +02:00
script_create_dialog - > set_title ( TTR ( " Create Script " ) ) ;
2015-08-10 01:39:59 +02:00
add_child ( script_create_dialog ) ;
2020-02-21 18:28:45 +01:00
script_create_dialog - > connect ( " script_created " , callable_mp ( this , & ScriptEditor : : _script_created ) ) ;
2014-02-10 02:10:30 +01:00
2016-04-12 16:45:31 +02:00
file_dialog_option = - 1 ;
2017-03-05 16:44:50 +01:00
file_dialog = memnew ( EditorFileDialog ) ;
2016-04-12 16:45:31 +02:00
add_child ( file_dialog ) ;
2020-02-21 18:28:45 +01:00
file_dialog - > connect ( " file_selected " , callable_mp ( this , & ScriptEditor : : _file_dialog_action ) ) ;
2014-02-10 02:10:30 +01:00
2018-01-08 01:31:36 +01:00
error_dialog = memnew ( AcceptDialog ) ;
add_child ( error_dialog ) ;
2017-03-05 16:44:50 +01:00
disk_changed = memnew ( ConfirmationDialog ) ;
2014-02-10 02:10:30 +01:00
{
2017-03-05 16:44:50 +01:00
VBoxContainer * vbc = memnew ( VBoxContainer ) ;
2014-02-10 02:10:30 +01:00
disk_changed - > add_child ( vbc ) ;
2017-03-05 16:44:50 +01:00
Label * dl = memnew ( Label ) ;
2016-05-21 01:18:35 +02:00
dl - > set_text ( TTR ( " The following files are newer on disk. \n What action should be taken?: " ) ) ;
2014-02-10 02:10:30 +01:00
vbc - > add_child ( dl ) ;
2017-03-05 16:44:50 +01:00
disk_changed_list = memnew ( Tree ) ;
2014-02-10 02:10:30 +01:00
vbc - > add_child ( disk_changed_list ) ;
disk_changed_list - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2020-02-21 18:28:45 +01:00
disk_changed - > connect ( " confirmed " , callable_mp ( this , & ScriptEditor : : _reload_scripts ) ) ;
2020-12-14 19:37:30 +01:00
disk_changed - > get_ok_button ( ) - > set_text ( TTR ( " Reload " ) ) ;
2014-02-10 02:10:30 +01:00
2020-07-09 01:02:38 +02:00
disk_changed - > add_button ( TTR ( " Resave " ) , ! DisplayServer : : get_singleton ( ) - > get_swap_cancel_ok ( ) , " resave " ) ;
2020-02-21 18:28:45 +01:00
disk_changed - > connect ( " custom_action " , callable_mp ( this , & ScriptEditor : : _resave_scripts ) ) ;
2014-02-10 02:10:30 +01:00
}
add_child ( disk_changed ) ;
2017-03-05 16:44:50 +01:00
script_editor = this ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
autosave_timer = memnew ( Timer ) ;
2015-05-05 04:32:40 +02:00
autosave_timer - > set_one_shot ( false ) ;
2020-02-21 23:26:13 +01:00
autosave_timer - > connect ( SceneStringNames : : get_singleton ( ) - > tree_entered , callable_mp ( this , & ScriptEditor : : _update_autosave_timer ) ) ;
2020-02-21 18:28:45 +01:00
autosave_timer - > connect ( " timeout " , callable_mp ( this , & ScriptEditor : : _autosave_scripts ) ) ;
2015-05-05 04:32:40 +02:00
add_child ( autosave_timer ) ;
2017-03-05 16:44:50 +01:00
grab_focus_block = false ;
2015-06-22 05:03:19 +02:00
2017-03-05 16:44:50 +01:00
help_search_dialog = memnew ( EditorHelpSearch ) ;
2015-11-17 13:46:08 +01:00
add_child ( help_search_dialog ) ;
2020-02-21 18:28:45 +01:00
help_search_dialog - > connect ( " go_to_help " , callable_mp ( this , & ScriptEditor : : _help_class_goto ) ) ;
2015-11-17 13:46:08 +01:00
2018-02-12 02:36:15 +01:00
find_in_files_dialog = memnew ( FindInFilesDialog ) ;
2020-02-21 23:26:13 +01:00
find_in_files_dialog - > connect ( FindInFilesDialog : : SIGNAL_FIND_REQUESTED , callable_mp ( this , & ScriptEditor : : _start_find_in_files ) , varray ( false ) ) ;
find_in_files_dialog - > connect ( FindInFilesDialog : : SIGNAL_REPLACE_REQUESTED , callable_mp ( this , & ScriptEditor : : _start_find_in_files ) , varray ( true ) ) ;
2018-02-12 02:36:15 +01:00
add_child ( find_in_files_dialog ) ;
find_in_files = memnew ( FindInFilesPanel ) ;
2018-10-03 22:43:26 +02:00
find_in_files_button = editor - > add_bottom_panel_item ( TTR ( " Search Results " ) , find_in_files ) ;
find_in_files - > set_custom_minimum_size ( Size2 ( 0 , 200 ) * EDSCALE ) ;
2020-02-21 23:26:13 +01:00
find_in_files - > connect ( FindInFilesPanel : : SIGNAL_RESULT_SELECTED , callable_mp ( this , & ScriptEditor : : _on_find_in_files_result_selected ) ) ;
find_in_files - > connect ( FindInFilesPanel : : SIGNAL_FILES_MODIFIED , callable_mp ( this , & ScriptEditor : : _on_find_in_files_modified_files ) ) ;
2018-02-12 02:36:15 +01:00
find_in_files - > hide ( ) ;
find_in_files_button - > hide ( ) ;
2017-03-05 16:44:50 +01:00
history_pos = - 1 ;
2014-05-06 11:43:14 +02:00
2017-03-05 16:44:50 +01:00
edit_pass = 0 ;
2018-03-01 21:45:04 +01:00
trim_trailing_whitespace_on_save = EditorSettings : : get_singleton ( ) - > get ( " text_editor/files/trim_trailing_whitespace_on_save " ) ;
convert_indent_on_save = EditorSettings : : get_singleton ( ) - > get ( " text_editor/indent/convert_indent_on_save " ) ;
use_space_indentation = EditorSettings : : get_singleton ( ) - > get ( " text_editor/indent/type " ) ;
2016-08-25 22:45:20 +02:00
2017-03-05 16:44:50 +01:00
ScriptServer : : edit_request_func = _open_script_request ;
2017-09-06 23:25:57 +02:00
2020-03-12 13:37:40 +01:00
add_theme_style_override ( " panel " , editor - > get_gui_base ( ) - > get_theme_stylebox ( " ScriptEditorPanel " , " EditorStyles " ) ) ;
tab_container - > add_theme_style_override ( " panel " , editor - > get_gui_base ( ) - > get_theme_stylebox ( " ScriptEditor " , " EditorStyles " ) ) ;
2014-02-10 02:10:30 +01:00
}
2015-06-26 06:14:31 +02:00
ScriptEditor : : ~ ScriptEditor ( ) {
memdelete ( completion_cache ) ;
}
2014-02-10 02:10:30 +01:00
void ScriptEditorPlugin : : edit ( Object * p_object ) {
2018-05-28 17:52:28 +02:00
if ( Object : : cast_to < Script > ( p_object ) ) {
2018-09-06 23:33:44 +02:00
Script * p_script = Object : : cast_to < Script > ( p_object ) ;
2019-09-04 20:00:09 +02:00
String res_path = p_script - > get_path ( ) . get_slice ( " :: " , 0 ) ;
2018-09-06 23:33:44 +02:00
2019-09-04 20:00:09 +02:00
if ( _is_built_in_script ( p_script ) ) {
if ( ResourceLoader : : get_resource_type ( res_path ) = = " PackedScene " ) {
if ( ! EditorNode : : get_singleton ( ) - > is_scene_open ( res_path ) ) {
EditorNode : : get_singleton ( ) - > load_scene ( res_path ) ;
}
} else {
EditorNode : : get_singleton ( ) - > load_resource ( res_path ) ;
}
2018-09-06 23:33:44 +02:00
}
2019-09-04 20:00:09 +02:00
script_editor - > edit ( p_script ) ;
2019-05-18 01:27:52 +02:00
} else if ( Object : : cast_to < TextFile > ( p_object ) ) {
2018-05-28 17:52:28 +02:00
script_editor - > edit ( Object : : cast_to < TextFile > ( p_object ) ) ;
}
2014-02-10 02:10:30 +01:00
}
bool ScriptEditorPlugin : : handles ( Object * p_object ) const {
2018-05-28 17:52:28 +02:00
if ( Object : : cast_to < TextFile > ( p_object ) ) {
return true ;
}
2017-08-24 22:58:51 +02:00
if ( Object : : cast_to < Script > ( p_object ) ) {
2018-09-06 23:33:44 +02:00
return true ;
2016-07-07 01:35:49 +02:00
}
2017-01-03 03:03:46 +01:00
return p_object - > is_class ( " Script " ) ;
2014-02-10 02:10:30 +01:00
}
void ScriptEditorPlugin : : make_visible ( bool p_visible ) {
if ( p_visible ) {
script_editor - > show ( ) ;
script_editor - > set_process ( true ) ;
script_editor - > ensure_select_current ( ) ;
} else {
script_editor - > hide ( ) ;
script_editor - > set_process ( false ) ;
}
}
void ScriptEditorPlugin : : selected_notify ( ) {
script_editor - > ensure_select_current ( ) ;
}
void ScriptEditorPlugin : : save_external_data ( ) {
2016-02-03 01:10:52 +01:00
script_editor - > save_all_scripts ( ) ;
2014-02-10 02:10:30 +01:00
}
void ScriptEditorPlugin : : apply_changes ( ) {
script_editor - > apply_scripts ( ) ;
}
void ScriptEditorPlugin : : restore_global_state ( ) {
}
void ScriptEditorPlugin : : save_global_state ( ) {
2015-06-22 05:03:19 +02:00
}
2014-02-10 02:10:30 +01:00
2015-06-22 05:03:19 +02:00
void ScriptEditorPlugin : : set_window_layout ( Ref < ConfigFile > p_layout ) {
script_editor - > set_window_layout ( p_layout ) ;
2014-02-10 02:10:30 +01:00
}
2017-03-05 16:44:50 +01:00
void ScriptEditorPlugin : : get_window_layout ( Ref < ConfigFile > p_layout ) {
2015-06-22 05:03:19 +02:00
script_editor - > get_window_layout ( p_layout ) ;
}
2014-02-10 02:10:30 +01:00
void ScriptEditorPlugin : : get_breakpoints ( List < String > * p_breakpoints ) {
2019-07-01 12:59:42 +02:00
script_editor - > get_breakpoints ( p_breakpoints ) ;
2014-02-10 02:10:30 +01:00
}
2015-12-09 13:08:41 +01:00
void ScriptEditorPlugin : : edited_scene_changed ( ) {
script_editor - > edited_scene_changed ( ) ;
}
2014-02-10 02:10:30 +01:00
ScriptEditorPlugin : : ScriptEditorPlugin ( EditorNode * p_node ) {
2017-03-05 16:44:50 +01:00
editor = p_node ;
script_editor = memnew ( ScriptEditor ( p_node ) ) ;
2020-12-28 19:51:39 +01:00
editor - > get_main_control ( ) - > add_child ( script_editor ) ;
2016-02-08 17:01:54 +01:00
script_editor - > set_v_size_flags ( Control : : SIZE_EXPAND_FILL ) ;
2014-02-10 02:10:30 +01:00
script_editor - > hide ( ) ;
2017-03-05 16:44:50 +01:00
EDITOR_DEF ( " text_editor/files/auto_reload_scripts_on_external_change " , true ) ;
ScriptServer : : set_reload_scripts_on_save ( EDITOR_DEF ( " text_editor/files/auto_reload_and_parse_scripts_on_save " , true ) ) ;
EDITOR_DEF ( " text_editor/files/open_dominant_script_on_scene_change " , true ) ;
EDITOR_DEF ( " text_editor/external/use_external_editor " , false ) ;
EDITOR_DEF ( " text_editor/external/exec_path " , " " ) ;
2019-09-01 13:28:57 +02:00
EDITOR_DEF ( " text_editor/script_list/script_temperature_enabled " , true ) ;
EDITOR_DEF ( " text_editor/script_list/script_temperature_history_size " , 15 ) ;
EDITOR_DEF ( " text_editor/script_list/group_help_pages " , true ) ;
EditorSettings : : get_singleton ( ) - > add_property_hint ( PropertyInfo ( Variant : : INT , " text_editor/script_list/sort_scripts_by " , PROPERTY_HINT_ENUM , " Name,Path,None " ) ) ;
EDITOR_DEF ( " text_editor/script_list/sort_scripts_by " , 0 ) ;
EditorSettings : : get_singleton ( ) - > add_property_hint ( PropertyInfo ( Variant : : INT , " text_editor/script_list/list_script_names_as " , PROPERTY_HINT_ENUM , " Name,Parent Directory And Name,Full Path " ) ) ;
EDITOR_DEF ( " text_editor/script_list/list_script_names_as " , 0 ) ;
2017-03-05 16:44:50 +01:00
EditorSettings : : get_singleton ( ) - > add_property_hint ( PropertyInfo ( Variant : : STRING , " text_editor/external/exec_path " , PROPERTY_HINT_GLOBAL_FILE ) ) ;
2019-06-11 09:20:42 +02:00
EDITOR_DEF ( " text_editor/external/exec_flags " , " {file} " ) ;
EditorSettings : : get_singleton ( ) - > add_property_hint ( PropertyInfo ( Variant : : STRING , " text_editor/external/exec_flags " , PROPERTY_HINT_PLACEHOLDER_TEXT , " Call flags with placeholders: {project}, {file}, {col}, {line}. " ) ) ;
2017-04-27 17:07:39 +02:00
2019-07-31 21:06:15 +02:00
ED_SHORTCUT ( " script_editor/reopen_closed_script " , TTR ( " Reopen Closed Script " ) , KEY_MASK_CMD | KEY_MASK_SHIFT | KEY_T ) ;
2019-08-12 22:23:00 +02:00
ED_SHORTCUT ( " script_editor/clear_recent " , TTR ( " Clear Recent Scripts " ) ) ;
2017-03-05 16:44:50 +01:00
}
ScriptEditorPlugin : : ~ ScriptEditorPlugin ( ) {
2014-02-10 02:10:30 +01:00
}