2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* script_text_editor.cpp */
/**************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/**************************************************************************/
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
/* */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the */
/* "Software"), to deal in the Software without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of the Software, and to */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions: */
/* */
/* The above copyright notice and this permission notice shall be */
/* included in all copies or substantial portions of the Software. */
/* */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/**************************************************************************/
2018-01-05 00:50:27 +01:00
2016-08-02 19:11:05 -03:00
# include "script_text_editor.h"
2017-01-16 08:04:19 +01:00
2022-02-12 02:46:22 +01:00
# include "core/config/project_settings.h"
2024-12-14 21:25:05 +03:00
# include "core/io/json.h"
2019-08-07 16:31:33 +02:00
# include "core/math/expression.h"
2018-09-11 18:13:45 +02:00
# include "core/os/keyboard.h"
2020-02-07 02:52:05 +01:00
# include "editor/debugger/editor_debugger_node.h"
2021-05-29 18:58:16 +05:30
# include "editor/editor_command_palette.h"
2024-12-14 21:25:05 +03:00
# include "editor/editor_help.h"
2017-03-05 16:44:50 +01:00
# include "editor/editor_node.h"
2017-03-05 14:21:25 +01:00
# include "editor/editor_settings.h"
2023-08-13 02:33:39 +02:00
# include "editor/editor_string_names.h"
2023-07-09 23:15:54 +02:00
# include "editor/gui/editor_toaster.h"
2024-12-18 14:42:10 +01:00
# include "editor/plugins/editor_context_menu_plugin.h"
2024-01-15 13:14:55 +01:00
# include "editor/themes/editor_scale.h"
2024-11-22 01:09:24 +01:00
# include "scene/gui/menu_button.h"
2023-04-12 21:02:28 +02:00
# include "scene/gui/rich_text_label.h"
2022-11-19 12:45:49 +01:00
# include "scene/gui/split_container.h"
2016-08-02 19:11:05 -03:00
2019-04-20 12:51:25 +01:00
void ConnectionInfoDialog : : ok_pressed ( ) {
}
2024-02-15 13:25:58 -03:00
void ConnectionInfoDialog : : popup_connections ( const String & p_method , const Vector < Node * > & p_nodes ) {
2019-04-20 12:51:25 +01:00
method - > set_text ( p_method ) ;
tree - > clear ( ) ;
TreeItem * root = tree - > create_item ( ) ;
for ( int i = 0 ; i < p_nodes . size ( ) ; i + + ) {
List < Connection > all_connections ;
p_nodes [ i ] - > get_signals_connected_to_this ( & all_connections ) ;
2021-07-24 15:46:25 +02:00
for ( const Connection & connection : all_connections ) {
2020-02-19 16:27:19 -03:00
if ( connection . callable . get_method ( ) ! = p_method ) {
2019-04-20 12:51:25 +01:00
continue ;
}
TreeItem * node_item = tree - > create_item ( root ) ;
2020-02-19 16:27:19 -03:00
node_item - > set_text ( 0 , Object : : cast_to < Node > ( connection . signal . get_object ( ) ) - > get_name ( ) ) ;
node_item - > set_icon ( 0 , EditorNode : : get_singleton ( ) - > get_object_icon ( connection . signal . get_object ( ) , " Node " ) ) ;
2019-04-20 12:51:25 +01:00
node_item - > set_selectable ( 0 , false ) ;
node_item - > set_editable ( 0 , false ) ;
2020-02-19 16:27:19 -03:00
node_item - > set_text ( 1 , connection . signal . get_name ( ) ) ;
2020-03-06 14:00:16 -03:00
Control * p = Object : : cast_to < Control > ( get_parent ( ) ) ;
2023-08-13 02:33:39 +02:00
node_item - > set_icon ( 1 , p - > get_editor_theme_icon ( SNAME ( " Slot " ) ) ) ;
2019-04-20 12:51:25 +01:00
node_item - > set_selectable ( 1 , false ) ;
node_item - > set_editable ( 1 , false ) ;
2020-02-19 16:27:19 -03:00
node_item - > set_text ( 2 , Object : : cast_to < Node > ( connection . callable . get_object ( ) ) - > get_name ( ) ) ;
node_item - > set_icon ( 2 , EditorNode : : get_singleton ( ) - > get_object_icon ( connection . callable . get_object ( ) , " Node " ) ) ;
2019-04-20 12:51:25 +01:00
node_item - > set_selectable ( 2 , false ) ;
node_item - > set_editable ( 2 , false ) ;
}
}
2020-02-15 16:36:10 +01:00
popup_centered ( Size2 ( 600 , 300 ) * EDSCALE ) ;
2019-04-20 12:51:25 +01:00
}
ConnectionInfoDialog : : ConnectionInfoDialog ( ) {
set_title ( TTR ( " Connections to method: " ) ) ;
VBoxContainer * vbc = memnew ( VBoxContainer ) ;
2020-12-22 16:24:29 +00:00
vbc - > set_anchor_and_offset ( SIDE_LEFT , Control : : ANCHOR_BEGIN , 8 * EDSCALE ) ;
vbc - > set_anchor_and_offset ( SIDE_TOP , Control : : ANCHOR_BEGIN , 8 * EDSCALE ) ;
vbc - > set_anchor_and_offset ( SIDE_RIGHT , Control : : ANCHOR_END , - 8 * EDSCALE ) ;
vbc - > set_anchor_and_offset ( SIDE_BOTTOM , Control : : ANCHOR_END , - 8 * EDSCALE ) ;
2019-04-20 12:51:25 +01:00
add_child ( vbc ) ;
method = memnew ( Label ) ;
2021-11-24 20:58:47 -06:00
method - > set_horizontal_alignment ( HORIZONTAL_ALIGNMENT_CENTER ) ;
2019-04-20 12:51:25 +01:00
vbc - > add_child ( method ) ;
tree = memnew ( Tree ) ;
2024-03-17 16:28:18 +08:00
tree - > set_auto_translate_mode ( AUTO_TRANSLATE_MODE_DISABLED ) ;
2019-04-20 12:51:25 +01:00
tree - > set_columns ( 3 ) ;
tree - > set_hide_root ( true ) ;
tree - > set_column_titles_visible ( true ) ;
tree - > set_column_title ( 0 , TTR ( " Source " ) ) ;
tree - > set_column_title ( 1 , TTR ( " Signal " ) ) ;
tree - > set_column_title ( 2 , TTR ( " Target " ) ) ;
vbc - > add_child ( tree ) ;
2020-03-06 14:00:16 -03:00
tree - > set_v_size_flags ( Control : : SIZE_EXPAND_FILL ) ;
2019-04-20 12:51:25 +01:00
tree - > set_allow_rmb_select ( true ) ;
}
////////////////////////////////////////////////////////////////////////////////
2017-03-05 16:44:50 +01:00
Vector < String > ScriptTextEditor : : get_functions ( ) {
2020-07-24 15:50:35 +01:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2016-08-02 19:11:05 -03:00
String text = te - > get_text ( ) ;
List < String > fnc ;
2021-05-18 13:09:19 +10:00
if ( script - > get_language ( ) - > validate ( text , script - > get_path ( ) , & fnc ) ) {
2016-08-02 19:11:05 -03:00
//if valid rewrite functions to latest
functions . clear ( ) ;
2021-07-24 15:46:25 +02:00
for ( const String & E : fnc ) {
2021-07-15 23:45:57 -04:00
functions . push_back ( E ) ;
2016-08-02 19:11:05 -03:00
}
}
return functions ;
}
void ScriptTextEditor : : apply_code ( ) {
2020-05-14 16:41:43 +02:00
if ( script . is_null ( ) ) {
2016-08-02 19:11:05 -03:00
return ;
2020-05-14 16:41:43 +02:00
}
2020-07-24 15:50:35 +01:00
script - > set_source_code ( code_editor - > get_text_editor ( ) - > get_text ( ) ) ;
2016-08-02 19:11:05 -03:00
script - > update_exports ( ) ;
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > get_syntax_highlighter ( ) - > update_cache ( ) ;
2016-08-02 19:11:05 -03:00
}
2022-05-03 01:43:50 +02:00
Ref < Resource > ScriptTextEditor : : get_edited_resource ( ) const {
2016-08-02 19:11:05 -03:00
return script ;
}
2022-05-03 01:43:50 +02:00
void ScriptTextEditor : : set_edited_resource ( const Ref < Resource > & p_res ) {
2019-11-20 10:09:59 +01:00
ERR_FAIL_COND ( script . is_valid ( ) ) ;
ERR_FAIL_COND ( p_res . is_null ( ) ) ;
2018-05-28 16:52:28 +01:00
script = p_res ;
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > set_text ( script - > get_source_code ( ) ) ;
code_editor - > get_text_editor ( ) - > clear_undo_history ( ) ;
code_editor - > get_text_editor ( ) - > tag_saved_version ( ) ;
2018-05-28 16:52:28 +01:00
2021-07-17 18:22:52 -03:00
emit_signal ( SNAME ( " name_changed " ) ) ;
2018-05-28 16:52:28 +01:00
code_editor - > update_line_and_column ( ) ;
2019-11-20 10:09:59 +01:00
}
2022-10-23 18:45:44 +10:00
void ScriptTextEditor : : enable_editor ( Control * p_shortcut_context ) {
2019-11-20 10:09:59 +01:00
if ( editor_enabled ) {
return ;
}
editor_enabled = true ;
_enable_code_editor ( ) ;
2019-04-16 02:14:58 +02:00
_validate_script ( ) ;
2022-10-23 18:45:44 +10:00
if ( p_shortcut_context ) {
for ( int i = 0 ; i < edit_hb - > get_child_count ( ) ; + + i ) {
Control * c = cast_to < Control > ( edit_hb - > get_child ( i ) ) ;
if ( c ) {
c - > set_shortcut_context ( p_shortcut_context ) ;
}
}
}
2018-05-28 16:52:28 +01:00
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : _load_theme_settings ( ) {
2020-07-24 15:50:35 +01:00
CodeEdit * text_edit = code_editor - > get_text_editor ( ) ;
2016-08-02 19:11:05 -03:00
2021-08-15 18:14:46 +01:00
Color updated_marked_line_color = EDITOR_GET ( " text_editor/theme/highlighting/mark_color " ) ;
Color updated_safe_line_number_color = EDITOR_GET ( " text_editor/theme/highlighting/safe_line_number_color " ) ;
2023-03-12 17:48:37 +01:00
Color updated_folded_code_region_color = EDITOR_GET ( " text_editor/theme/highlighting/folded_code_region_color " ) ;
2020-09-10 21:25:00 +01:00
bool safe_line_number_color_updated = updated_safe_line_number_color ! = safe_line_number_color ;
bool marked_line_color_updated = updated_marked_line_color ! = marked_line_color ;
2023-03-12 17:48:37 +01:00
bool folded_code_region_color_updated = updated_folded_code_region_color ! = folded_code_region_color ;
if ( safe_line_number_color_updated | | marked_line_color_updated | | folded_code_region_color_updated ) {
2020-07-30 11:41:05 +01:00
safe_line_number_color = updated_safe_line_number_color ;
for ( int i = 0 ; i < text_edit - > get_line_count ( ) ; i + + ) {
2020-09-10 21:25:00 +01:00
if ( marked_line_color_updated & & text_edit - > get_line_background_color ( i ) = = marked_line_color ) {
text_edit - > set_line_background_color ( i , updated_marked_line_color ) ;
}
if ( safe_line_number_color_updated & & text_edit - > get_line_gutter_item_color ( i , line_number_gutter ) ! = default_line_number_color ) {
2020-07-30 11:41:05 +01:00
text_edit - > set_line_gutter_item_color ( i , line_number_gutter , safe_line_number_color ) ;
}
2023-03-12 17:48:37 +01:00
if ( folded_code_region_color_updated & & text_edit - > get_line_background_color ( i ) = = folded_code_region_color ) {
text_edit - > set_line_background_color ( i , updated_folded_code_region_color ) ;
}
2020-07-30 11:41:05 +01:00
}
2020-09-10 21:25:00 +01:00
marked_line_color = updated_marked_line_color ;
2023-03-12 17:48:37 +01:00
folded_code_region_color = updated_folded_code_region_color ;
2020-07-30 11:41:05 +01:00
}
2017-10-02 18:33:42 -05:00
theme_loaded = true ;
2024-08-25 14:15:10 +02:00
if ( script . is_valid ( ) ) {
2017-10-02 18:33:42 -05:00
_set_theme_for_script ( ) ;
2020-05-14 16:41:43 +02:00
}
2017-10-02 18:33:42 -05:00
}
void ScriptTextEditor : : _set_theme_for_script ( ) {
2020-05-14 16:41:43 +02:00
if ( ! theme_loaded ) {
2017-10-02 18:33:42 -05:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-10-02 18:33:42 -05:00
2020-07-24 15:50:35 +01:00
CodeEdit * text_edit = code_editor - > get_text_editor ( ) ;
2020-05-03 17:08:15 +01:00
text_edit - > get_syntax_highlighter ( ) - > update_cache ( ) ;
2017-10-02 18:33:42 -05:00
2020-09-10 21:25:40 +01:00
List < String > strings ;
script - > get_language ( ) - > get_string_delimiters ( & strings ) ;
text_edit - > clear_string_delimiters ( ) ;
2021-07-24 15:46:25 +02:00
for ( const String & string : strings ) {
2020-09-10 21:25:40 +01:00
String beg = string . get_slice ( " " , 0 ) ;
String end = string . get_slice_count ( " " ) > 1 ? string . get_slice ( " " , 1 ) : String ( ) ;
2021-07-01 17:40:59 +01:00
if ( ! text_edit - > has_string_delimiter ( beg ) ) {
2021-12-09 03:42:46 -06:00
text_edit - > add_string_delimiter ( beg , end , end . is_empty ( ) ) ;
2021-07-01 17:40:59 +01:00
}
2021-06-28 17:14:44 +01:00
if ( ! end . is_empty ( ) & & ! text_edit - > has_auto_brace_completion_open_key ( beg ) ) {
text_edit - > add_auto_brace_completion_pair ( beg , end ) ;
}
2020-09-10 21:25:40 +01:00
}
2023-02-05 12:01:01 +03:00
text_edit - > clear_comment_delimiters ( ) ;
2020-09-10 21:25:40 +01:00
List < String > comments ;
script - > get_language ( ) - > get_comment_delimiters ( & comments ) ;
2021-07-24 15:46:25 +02:00
for ( const String & comment : comments ) {
2020-09-10 21:25:40 +01:00
String beg = comment . get_slice ( " " , 0 ) ;
String end = comment . get_slice_count ( " " ) > 1 ? comment . get_slice ( " " , 1 ) : String ( ) ;
2021-12-09 03:42:46 -06:00
text_edit - > add_comment_delimiter ( beg , end , end . is_empty ( ) ) ;
2021-06-28 17:14:44 +01:00
if ( ! end . is_empty ( ) & & ! text_edit - > has_auto_brace_completion_open_key ( beg ) ) {
text_edit - > add_auto_brace_completion_pair ( beg , end ) ;
}
2020-09-10 21:25:40 +01:00
}
2023-02-05 12:01:01 +03:00
List < String > doc_comments ;
script - > get_language ( ) - > get_doc_comment_delimiters ( & doc_comments ) ;
for ( const String & doc_comment : doc_comments ) {
String beg = doc_comment . get_slice ( " " , 0 ) ;
String end = doc_comment . get_slice_count ( " " ) > 1 ? doc_comment . get_slice ( " " , 1 ) : String ( ) ;
text_edit - > add_comment_delimiter ( beg , end , end . is_empty ( ) ) ;
if ( ! end . is_empty ( ) & & ! text_edit - > has_auto_brace_completion_open_key ( beg ) ) {
text_edit - > add_auto_brace_completion_pair ( beg , end ) ;
}
}
2016-08-02 19:11:05 -03:00
}
2021-05-18 13:09:19 +10:00
void ScriptTextEditor : : _show_errors_panel ( bool p_show ) {
errors_panel - > set_visible ( p_show ) ;
}
2019-02-13 15:12:31 +06:00
void ScriptTextEditor : : _show_warnings_panel ( bool p_show ) {
warnings_panel - > set_visible ( p_show ) ;
2018-07-01 13:17:40 -03:00
}
2024-02-15 13:25:58 -03:00
void ScriptTextEditor : : _warning_clicked ( const Variant & p_line ) {
2018-07-01 13:17:40 -03:00
if ( p_line . get_type ( ) = = Variant : : INT ) {
2021-05-12 12:45:30 +02:00
goto_line_centered ( p_line . operator int64_t ( ) ) ;
2022-04-26 05:13:24 +02:00
} else if ( p_line . get_type ( ) = = Variant : : DICTIONARY ) {
Dictionary meta = p_line . operator Dictionary ( ) ;
const int line = meta [ " line " ] . operator int64_t ( ) - 1 ;
2023-01-18 20:12:33 +03:00
const String code = meta [ " code " ] . operator String ( ) ;
const String quote_style = EDITOR_GET ( " text_editor/completion/use_single_quotes " ) ? " ' " : " \" " ;
2022-04-26 05:13:24 +02:00
CodeEdit * text_editor = code_editor - > get_text_editor ( ) ;
String prev_line = line > 0 ? text_editor - > get_line ( line - 1 ) : " " ;
if ( prev_line . contains ( " @warning_ignore " ) ) {
2024-11-16 18:52:15 +01:00
const int closing_bracket_idx = prev_line . find_char ( ' ) ' ) ;
2023-01-18 20:12:33 +03:00
const String text_to_insert = " , " + code . quote ( quote_style ) ;
2024-01-22 18:27:19 -05:00
text_editor - > insert_text ( text_to_insert , line - 1 , closing_bracket_idx ) ;
2022-04-26 05:13:24 +02:00
} else {
const int indent = text_editor - > get_indent_level ( line ) / text_editor - > get_indent_size ( ) ;
String annotation_indent ;
if ( ! text_editor - > is_indent_using_spaces ( ) ) {
annotation_indent = String ( " \t " ) . repeat ( indent ) ;
} else {
annotation_indent = String ( " " ) . repeat ( text_editor - > get_indent_size ( ) * indent ) ;
}
2023-01-18 20:12:33 +03:00
text_editor - > insert_line_at ( line , annotation_indent + " @warning_ignore( " + code . quote ( quote_style ) + " ) " ) ;
2022-04-26 05:13:24 +02:00
}
_validate_script ( ) ;
2018-07-01 13:17:40 -03:00
}
}
2024-02-15 13:25:58 -03:00
void ScriptTextEditor : : _error_clicked ( const Variant & p_line ) {
2021-05-18 13:09:19 +10:00
if ( p_line . get_type ( ) = = Variant : : INT ) {
2024-07-05 19:27:19 -04:00
goto_line_centered ( p_line . operator int64_t ( ) ) ;
2023-07-24 15:49:39 +02:00
} else if ( p_line . get_type ( ) = = Variant : : DICTIONARY ) {
Dictionary meta = p_line . operator Dictionary ( ) ;
const String path = meta [ " path " ] . operator String ( ) ;
const int line = meta [ " line " ] . operator int64_t ( ) ;
const int column = meta [ " column " ] . operator int64_t ( ) ;
if ( path . is_empty ( ) ) {
2024-07-05 19:27:19 -04:00
goto_line_centered ( line , column ) ;
2023-07-24 15:49:39 +02:00
} else {
Ref < Resource > scr = ResourceLoader : : load ( path ) ;
2024-08-25 14:15:10 +02:00
if ( scr . is_null ( ) ) {
2023-07-24 15:49:39 +02:00
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Could not load file at: " ) + " \n \n " + path , TTR ( " Error! " ) ) ;
} else {
2024-04-21 22:14:18 +03:00
int corrected_column = column ;
const String line_text = code_editor - > get_text_editor ( ) - > get_line ( line ) ;
const int indent_size = code_editor - > get_text_editor ( ) - > get_indent_size ( ) ;
if ( indent_size > 1 ) {
const int tab_count = line_text . length ( ) - line_text . lstrip ( " \t " ) . length ( ) ;
corrected_column - = tab_count * ( indent_size - 1 ) ;
}
ScriptEditor : : get_singleton ( ) - > edit ( scr , line , corrected_column ) ;
2023-07-24 15:49:39 +02:00
}
}
2021-05-18 13:09:19 +10:00
}
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : reload_text ( ) {
2017-03-05 16:44:50 +01:00
ERR_FAIL_COND ( script . is_null ( ) ) ;
2016-08-02 19:11:05 -03:00
2020-07-24 15:50:35 +01:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2021-07-08 18:35:56 +01:00
int column = te - > get_caret_column ( ) ;
int row = te - > get_caret_line ( ) ;
2016-08-02 19:11:05 -03:00
int h = te - > get_h_scroll ( ) ;
int v = te - > get_v_scroll ( ) ;
te - > set_text ( script - > get_source_code ( ) ) ;
2021-07-08 18:35:56 +01:00
te - > set_caret_line ( row ) ;
te - > set_caret_column ( column ) ;
2016-08-02 19:11:05 -03:00
te - > set_h_scroll ( h ) ;
te - > set_v_scroll ( v ) ;
te - > tag_saved_version ( ) ;
code_editor - > update_line_and_column ( ) ;
2023-11-14 21:14:18 +02:00
if ( editor_enabled ) {
_validate_script ( ) ;
}
2016-08-02 19:11:05 -03:00
}
2024-02-15 13:25:58 -03:00
void ScriptTextEditor : : add_callback ( const String & p_function , const PackedStringArray & p_args ) {
2024-02-04 19:08:41 +01:00
ScriptLanguage * language = script - > get_language ( ) ;
if ( ! language - > can_make_function ( ) ) {
return ;
}
2024-01-22 18:27:19 -05:00
code_editor - > get_text_editor ( ) - > begin_complex_operation ( ) ;
code_editor - > get_text_editor ( ) - > remove_secondary_carets ( ) ;
code_editor - > get_text_editor ( ) - > deselect ( ) ;
2020-07-24 15:50:35 +01:00
String code = code_editor - > get_text_editor ( ) - > get_text ( ) ;
2024-02-04 19:08:41 +01:00
int pos = language - > find_function ( p_function , code ) ;
2017-03-05 16:44:50 +01:00
if ( pos = = - 1 ) {
2024-01-22 18:27:19 -05:00
// Function does not exist, create it at the end of the file.
int last_line = code_editor - > get_text_editor ( ) - > get_line_count ( ) - 1 ;
2024-02-04 19:08:41 +01:00
String func = language - > make_function ( " " , p_function , p_args ) ;
2024-01-22 18:27:19 -05:00
code_editor - > get_text_editor ( ) - > insert_text ( " \n \n " + func , last_line , code_editor - > get_text_editor ( ) - > get_line ( last_line ) . length ( ) ) ;
pos = last_line + 3 ;
2016-08-02 19:11:05 -03:00
}
2024-01-22 18:27:19 -05:00
// Put caret on the line after the function, after the indent.
int indent_column = 1 ;
if ( EDITOR_GET ( " text_editor/behavior/indent/type " ) ) {
indent_column = EDITOR_GET ( " text_editor/behavior/indent/size " ) ;
}
code_editor - > get_text_editor ( ) - > set_caret_line ( pos , true , true , - 1 ) ;
code_editor - > get_text_editor ( ) - > set_caret_column ( indent_column ) ;
code_editor - > get_text_editor ( ) - > end_complex_operation ( ) ;
2016-08-02 19:11:05 -03:00
}
2017-05-28 15:20:38 +01:00
bool ScriptTextEditor : : show_members_overview ( ) {
return true ;
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : update_settings ( ) {
2022-10-18 16:43:37 +02:00
code_editor - > get_text_editor ( ) - > set_gutter_draw ( connection_gutter , EDITOR_GET ( " text_editor/appearance/gutters/show_info_gutter " ) ) ;
2016-10-10 10:38:12 +02:00
code_editor - > update_editor_settings ( ) ;
2016-08-02 19:11:05 -03:00
}
2017-03-05 16:44:50 +01:00
bool ScriptTextEditor : : is_unsaved ( ) {
2020-07-23 15:21:28 +03:00
const bool unsaved =
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > get_version ( ) ! = code_editor - > get_text_editor ( ) - > get_saved_version ( ) | |
2020-12-15 12:04:21 +00:00
script - > get_path ( ) . is_empty ( ) ; // In memory.
2020-07-23 15:21:28 +03:00
return unsaved ;
2016-08-02 19:11:05 -03:00
}
Variant ScriptTextEditor : : get_edit_state ( ) {
2018-05-25 23:49:35 +01:00
return code_editor - > get_edit_state ( ) ;
}
2016-08-02 19:11:05 -03:00
2018-05-25 23:49:35 +01:00
void ScriptTextEditor : : set_edit_state ( const Variant & p_state ) {
code_editor - > set_edit_state ( p_state ) ;
2019-04-13 12:43:35 +01:00
Dictionary state = p_state ;
if ( state . has ( " syntax_highlighter " ) ) {
int idx = highlighter_menu - > get_item_idx_from_text ( state [ " syntax_highlighter " ] ) ;
if ( idx > = 0 ) {
_change_syntax_highlighter ( idx ) ;
}
}
2019-11-20 10:09:59 +01:00
if ( editor_enabled ) {
2022-09-05 22:13:03 -07:00
# ifndef ANDROID_ENABLED
2019-11-20 10:09:59 +01:00
ensure_focus ( ) ;
2022-09-05 22:13:03 -07:00
# endif
2019-11-20 10:09:59 +01:00
}
2016-08-02 19:11:05 -03:00
}
2022-09-23 14:46:28 +02:00
Variant ScriptTextEditor : : get_navigation_state ( ) {
return code_editor - > get_navigation_state ( ) ;
}
2022-10-22 23:54:37 +02:00
Variant ScriptTextEditor : : get_previous_state ( ) {
return code_editor - > get_previous_state ( ) ;
}
void ScriptTextEditor : : store_previous_state ( ) {
return code_editor - > store_previous_state ( ) ;
}
2018-05-25 23:49:35 +01:00
void ScriptTextEditor : : _convert_case ( CodeTextEditor : : CaseStyle p_case ) {
code_editor - > convert_case ( p_case ) ;
2017-04-24 19:59:55 +01:00
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : trim_trailing_whitespace ( ) {
2018-05-25 23:49:35 +01:00
code_editor - > trim_trailing_whitespace ( ) ;
2016-08-02 19:11:05 -03:00
}
2024-01-12 00:32:13 -05:00
void ScriptTextEditor : : trim_final_newlines ( ) {
code_editor - > trim_final_newlines ( ) ;
}
2019-05-28 18:27:32 -03:00
void ScriptTextEditor : : insert_final_newline ( ) {
code_editor - > insert_final_newline ( ) ;
}
2023-05-01 21:41:50 +01:00
void ScriptTextEditor : : convert_indent ( ) {
code_editor - > get_text_editor ( ) - > convert_indent ( ) ;
2017-04-16 16:47:59 +01:00
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : tag_saved_version ( ) {
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > tag_saved_version ( ) ;
2016-08-02 19:11:05 -03:00
}
2024-07-05 19:27:19 -04:00
void ScriptTextEditor : : goto_line ( int p_line , int p_column ) {
code_editor - > goto_line ( p_line , p_column ) ;
2018-02-12 02:36:15 +01:00
}
2018-05-25 23:49:35 +01:00
void ScriptTextEditor : : goto_line_selection ( int p_line , int p_begin , int p_end ) {
code_editor - > goto_line_selection ( p_line , p_begin , p_end ) ;
2016-08-02 19:11:05 -03:00
}
2024-07-05 19:27:19 -04:00
void ScriptTextEditor : : goto_line_centered ( int p_line , int p_column ) {
code_editor - > goto_line_centered ( p_line , p_column ) ;
2019-01-04 13:09:01 +01:00
}
2019-04-22 17:20:27 +01:00
void ScriptTextEditor : : set_executing_line ( int p_line ) {
code_editor - > set_executing_line ( p_line ) ;
}
void ScriptTextEditor : : clear_executing_line ( ) {
code_editor - > clear_executing_line ( ) ;
}
2018-05-25 23:49:35 +01:00
void ScriptTextEditor : : ensure_focus ( ) {
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > grab_focus ( ) ;
2016-08-02 19:11:05 -03:00
}
2017-03-05 16:44:50 +01:00
String ScriptTextEditor : : get_name ( ) {
2016-08-02 19:11:05 -03:00
String name ;
2021-11-06 02:15:19 +01:00
name = script - > get_path ( ) . get_file ( ) ;
if ( name . is_empty ( ) ) {
// This appears for newly created built-in scripts before saving the scene.
name = TTR ( " [unsaved] " ) ;
} else if ( script - > is_built_in ( ) ) {
const String & script_name = script - > get_name ( ) ;
2021-12-09 03:42:46 -06:00
if ( ! script_name . is_empty ( ) ) {
2021-11-06 02:15:19 +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) " , script_name , name . get_slice ( " :: " , 0 ) ) ;
2016-08-02 19:11:05 -03:00
}
2021-11-06 02:15:19 +01:00
}
if ( is_unsaved ( ) ) {
name + = " (*) " ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
return name ;
}
2020-03-12 09:37:40 -03:00
Ref < Texture2D > ScriptTextEditor : : get_theme_icon ( ) {
2021-12-02 13:01:49 +01:00
if ( get_parent_control ( ) ) {
String icon_name = script - > get_class ( ) ;
if ( script - > is_built_in ( ) ) {
icon_name + = " Internal " ;
}
2023-08-13 02:33:39 +02:00
if ( get_parent_control ( ) - > has_theme_icon ( icon_name , EditorStringName ( EditorIcons ) ) ) {
return get_parent_control ( ) - > get_editor_theme_icon ( icon_name ) ;
} else if ( get_parent_control ( ) - > has_theme_icon ( script - > get_class ( ) , EditorStringName ( EditorIcons ) ) ) {
return get_parent_control ( ) - > get_editor_theme_icon ( script - > get_class ( ) ) ;
2021-12-02 13:01:49 +01:00
}
2016-08-02 19:11:05 -03:00
}
2019-06-11 15:43:37 -03:00
return Ref < Texture2D > ( ) ;
2016-08-02 19:11:05 -03:00
}
void ScriptTextEditor : : _validate_script ( ) {
2020-07-24 15:50:35 +01:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2016-08-02 19:11:05 -03:00
String text = te - > get_text ( ) ;
List < String > fnc ;
2022-01-19 08:31:39 +03:00
warnings . clear ( ) ;
errors . clear ( ) ;
2023-07-24 15:49:39 +02:00
depended_errors . clear ( ) ;
2022-01-19 08:31:39 +03:00
safe_lines . clear ( ) ;
2016-08-02 19:11:05 -03:00
2021-05-18 13:09:19 +10:00
if ( ! script - > get_language ( ) - > validate ( text , script - > get_path ( ) , & fnc , & errors , & warnings , & safe_lines ) ) {
2023-12-29 18:05:58 -08:00
List < ScriptLanguage : : ScriptError > : : Element * E = errors . front ( ) ;
while ( E ) {
List < ScriptLanguage : : ScriptError > : : Element * next_E = E - > next ( ) ;
2023-08-30 17:43:16 +08:00
if ( ( E - > get ( ) . path . is_empty ( ) & & ! script - > get_path ( ) . is_empty ( ) ) | | E - > get ( ) . path ! = script - > get_path ( ) ) {
2023-07-24 15:49:39 +02:00
depended_errors [ E - > get ( ) . path ] . push_back ( E - > get ( ) ) ;
E - > erase ( ) ;
}
2023-12-29 18:05:58 -08:00
E = next_E ;
2023-07-24 15:49:39 +02:00
}
2023-08-30 17:43:16 +08:00
if ( errors . size ( ) > 0 ) {
// TRANSLATORS: Script error pointing to a line and column number.
2024-04-15 15:18:34 +02:00
String error_text = vformat ( TTR ( " Error at (%d, %d): " ) , errors . front ( ) - > get ( ) . line , errors . front ( ) - > get ( ) . column ) + " " + errors . front ( ) - > get ( ) . message ;
2023-08-30 17:43:16 +08:00
code_editor - > set_error ( error_text ) ;
2024-04-15 15:18:34 +02:00
code_editor - > set_error_pos ( errors . front ( ) - > get ( ) . line - 1 , errors . front ( ) - > get ( ) . column - 1 ) ;
2023-08-30 17:43:16 +08:00
}
2019-11-02 13:38:25 +00:00
script_is_valid = false ;
2016-08-02 19:11:05 -03:00
} else {
code_editor - > set_error ( " " ) ;
if ( ! script - > is_tool ( ) ) {
script - > set_source_code ( text ) ;
script - > update_exports ( ) ;
2020-05-03 17:08:15 +01:00
te - > get_syntax_highlighter ( ) - > update_cache ( ) ;
2016-08-02 19:11:05 -03:00
}
functions . clear ( ) ;
2021-07-24 15:46:25 +02:00
for ( const String & E : fnc ) {
2021-07-15 23:45:57 -04:00
functions . push_back ( E ) ;
2016-08-02 19:11:05 -03:00
}
2019-11-02 13:38:25 +00:00
script_is_valid = true ;
2016-08-02 19:11:05 -03:00
}
2019-04-20 12:51:25 +01:00
_update_connected_methods ( ) ;
2022-01-19 08:31:39 +03:00
_update_warnings ( ) ;
_update_errors ( ) ;
2016-08-02 19:11:05 -03:00
2022-01-19 08:31:39 +03:00
emit_signal ( SNAME ( " name_changed " ) ) ;
emit_signal ( SNAME ( " edited_script_changed " ) ) ;
}
void ScriptTextEditor : : _update_warnings ( ) {
2019-06-11 19:48:35 -03:00
int warning_nb = warnings . size ( ) ;
2018-07-01 13:17:40 -03:00
warnings_panel - > clear ( ) ;
2019-04-20 12:51:25 +01:00
2021-10-04 18:03:26 +10:00
bool has_connections_table = false ;
2019-06-11 19:48:35 -03:00
// Add missing connections.
if ( GLOBAL_GET ( " debug/gdscript/warnings/enable " ) . booleanize ( ) ) {
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base & & missing_connections . size ( ) > 0 ) {
2021-10-04 18:03:26 +10:00
has_connections_table = true ;
2019-06-11 19:48:35 -03:00
warnings_panel - > push_table ( 1 ) ;
2021-07-24 15:46:25 +02:00
for ( const Connection & connection : missing_connections ) {
2019-06-11 19:48:35 -03:00
String base_path = base - > get_name ( ) ;
2020-02-19 16:27:19 -03:00
String source_path = base = = connection . signal . get_object ( ) ? base_path : base_path + " / " + base - > get_path_to ( Object : : cast_to < Node > ( connection . signal . get_object ( ) ) ) ;
String target_path = base = = connection . callable . get_object ( ) ? base_path : base_path + " / " + base - > get_path_to ( Object : : cast_to < Node > ( connection . callable . get_object ( ) ) ) ;
2019-06-11 19:48:35 -03:00
warnings_panel - > push_cell ( ) ;
2023-08-13 02:33:39 +02:00
warnings_panel - > push_color ( warnings_panel - > get_theme_color ( SNAME ( " warning_color " ) , EditorStringName ( Editor ) ) ) ;
2020-02-19 16:27:19 -03:00
warnings_panel - > add_text ( vformat ( TTR ( " Missing connected method '%s' for signal '%s' from node '%s' to node '%s'. " ) , connection . callable . get_method ( ) , connection . signal . get_name ( ) , source_path , target_path ) ) ;
2019-06-11 19:48:35 -03:00
warnings_panel - > pop ( ) ; // Color.
warnings_panel - > pop ( ) ; // Cell.
}
warnings_panel - > pop ( ) ; // Table.
2019-04-20 12:51:25 +01:00
2019-06-11 19:48:35 -03:00
warning_nb + = missing_connections . size ( ) ;
2019-04-20 12:51:25 +01:00
}
}
2021-05-18 13:09:19 +10:00
code_editor - > set_warning_count ( warning_nb ) ;
2019-06-11 19:48:35 -03:00
2021-10-04 18:03:26 +10:00
if ( has_connections_table ) {
warnings_panel - > add_newline ( ) ;
}
2019-06-11 19:48:35 -03:00
// Add script warnings.
2022-04-26 05:13:24 +02:00
warnings_panel - > push_table ( 3 ) ;
2021-07-24 15:46:25 +02:00
for ( const ScriptLanguage : : Warning & w : warnings ) {
2022-04-26 05:13:24 +02:00
Dictionary ignore_meta ;
ignore_meta [ " line " ] = w . start_line ;
ignore_meta [ " code " ] = w . string_code . to_lower ( ) ;
warnings_panel - > push_cell ( ) ;
warnings_panel - > push_meta ( ignore_meta ) ;
warnings_panel - > push_color (
2023-08-13 02:33:39 +02:00
warnings_panel - > get_theme_color ( SNAME ( " accent_color " ) , EditorStringName ( Editor ) ) . lerp ( warnings_panel - > get_theme_color ( SNAME ( " mono_color " ) , EditorStringName ( Editor ) ) , 0.5f ) ) ;
2022-04-26 05:13:24 +02:00
warnings_panel - > add_text ( TTR ( " [Ignore] " ) ) ;
warnings_panel - > pop ( ) ; // Color.
warnings_panel - > pop ( ) ; // Meta ignore.
warnings_panel - > pop ( ) ; // Cell.
2018-07-01 13:17:40 -03:00
warnings_panel - > push_cell ( ) ;
2020-06-11 19:31:28 -03:00
warnings_panel - > push_meta ( w . start_line - 1 ) ;
2023-08-13 02:33:39 +02:00
warnings_panel - > push_color ( warnings_panel - > get_theme_color ( SNAME ( " warning_color " ) , EditorStringName ( Editor ) ) ) ;
2024-03-23 18:39:01 +08:00
warnings_panel - > add_text ( vformat ( TTR ( " Line %d (%s): " ) , w . start_line , w . string_code ) ) ;
2019-06-11 19:48:35 -03:00
warnings_panel - > pop ( ) ; // Color.
warnings_panel - > pop ( ) ; // Meta goto.
warnings_panel - > pop ( ) ; // Cell.
2018-07-01 13:17:40 -03:00
warnings_panel - > push_cell ( ) ;
warnings_panel - > add_text ( w . message ) ;
2023-01-07 14:01:10 +06:00
warnings_panel - > add_newline ( ) ;
2019-06-11 19:48:35 -03:00
warnings_panel - > pop ( ) ; // Cell.
2018-07-01 13:17:40 -03:00
}
2019-06-11 19:48:35 -03:00
warnings_panel - > pop ( ) ; // Table.
2022-01-19 08:31:39 +03:00
}
void ScriptTextEditor : : _update_errors ( ) {
code_editor - > set_error_count ( errors . size ( ) ) ;
2018-07-01 13:17:40 -03:00
2021-05-18 13:09:19 +10:00
errors_panel - > clear ( ) ;
errors_panel - > push_table ( 2 ) ;
2021-07-24 15:46:25 +02:00
for ( const ScriptLanguage : : ScriptError & err : errors ) {
2023-07-24 15:49:39 +02:00
Dictionary click_meta ;
click_meta [ " line " ] = err . line ;
click_meta [ " column " ] = err . column ;
2021-05-18 13:09:19 +10:00
errors_panel - > push_cell ( ) ;
errors_panel - > push_meta ( err . line - 1 ) ;
2023-08-13 02:33:39 +02:00
errors_panel - > push_color ( warnings_panel - > get_theme_color ( SNAME ( " error_color " ) , EditorStringName ( Editor ) ) ) ;
2024-03-23 18:39:01 +08:00
errors_panel - > add_text ( vformat ( TTR ( " Line %d: " ) , err . line ) ) ;
2021-05-18 13:09:19 +10:00
errors_panel - > pop ( ) ; // Color.
errors_panel - > pop ( ) ; // Meta goto.
errors_panel - > pop ( ) ; // Cell.
errors_panel - > push_cell ( ) ;
errors_panel - > add_text ( err . message ) ;
2023-01-07 14:01:10 +06:00
errors_panel - > add_newline ( ) ;
2021-05-18 13:09:19 +10:00
errors_panel - > pop ( ) ; // Cell.
}
errors_panel - > pop ( ) ; // Table
2023-07-24 15:49:39 +02:00
for ( const KeyValue < String , List < ScriptLanguage : : ScriptError > > & KV : depended_errors ) {
Dictionary click_meta ;
click_meta [ " path " ] = KV . key ;
click_meta [ " line " ] = 1 ;
errors_panel - > add_newline ( ) ;
errors_panel - > add_newline ( ) ;
errors_panel - > push_meta ( click_meta ) ;
errors_panel - > add_text ( vformat ( R " (%s:) " , KV . key ) ) ;
errors_panel - > pop ( ) ; // Meta goto.
errors_panel - > add_newline ( ) ;
errors_panel - > push_indent ( 1 ) ;
errors_panel - > push_table ( 2 ) ;
String filename = KV . key . get_file ( ) ;
for ( const ScriptLanguage : : ScriptError & err : KV . value ) {
click_meta [ " line " ] = err . line ;
click_meta [ " column " ] = err . column ;
errors_panel - > push_cell ( ) ;
errors_panel - > push_meta ( click_meta ) ;
2023-08-13 02:33:39 +02:00
errors_panel - > push_color ( errors_panel - > get_theme_color ( SNAME ( " error_color " ) , EditorStringName ( Editor ) ) ) ;
2024-03-23 18:39:01 +08:00
errors_panel - > add_text ( vformat ( TTR ( " Line %d: " ) , err . line ) ) ;
2023-07-24 15:49:39 +02:00
errors_panel - > pop ( ) ; // Color.
errors_panel - > pop ( ) ; // Meta goto.
errors_panel - > pop ( ) ; // Cell.
errors_panel - > push_cell ( ) ;
errors_panel - > add_text ( err . message ) ;
errors_panel - > pop ( ) ; // Cell.
}
errors_panel - > pop ( ) ; // Table
errors_panel - > pop ( ) ; // Indent.
}
2022-01-19 08:31:39 +03:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2022-03-06 21:39:19 +01:00
bool highlight_safe = EDITOR_GET ( " text_editor/appearance/gutters/highlight_type_safe_lines " ) ;
2018-06-05 13:50:21 -03:00
bool last_is_safe = false ;
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < te - > get_line_count ( ) ; i + + ) {
2021-05-18 13:09:19 +10:00
if ( errors . is_empty ( ) ) {
2023-03-12 17:48:37 +01:00
bool is_folded_code_region = te - > is_line_code_region_start ( i ) & & te - > is_line_folded ( i ) ;
te - > set_line_background_color ( i , is_folded_code_region ? folded_code_region_color : Color ( 0 , 0 , 0 , 0 ) ) ;
2021-05-18 13:09:19 +10:00
} else {
2021-07-24 15:46:25 +02:00
for ( const ScriptLanguage : : ScriptError & E : errors ) {
2021-07-15 23:45:57 -04:00
bool error_line = i = = E . line - 1 ;
2021-05-18 13:09:19 +10:00
te - > set_line_background_color ( i , error_line ? marked_line_color : Color ( 0 , 0 , 0 , 0 ) ) ;
if ( error_line ) {
break ;
}
}
}
2018-06-05 13:50:21 -03:00
if ( highlight_safe ) {
if ( safe_lines . has ( i + 1 ) ) {
2020-07-30 11:41:05 +01:00
te - > set_line_gutter_item_color ( i , line_number_gutter , safe_line_number_color ) ;
2018-06-05 13:50:21 -03:00
last_is_safe = true ;
2021-06-15 15:05:01 +01:00
} else if ( last_is_safe & & ( te - > is_in_comment ( i ) ! = - 1 | | te - > get_line ( i ) . strip_edges ( ) . is_empty ( ) ) ) {
2020-07-30 11:41:05 +01:00
te - > set_line_gutter_item_color ( i , line_number_gutter , safe_line_number_color ) ;
2018-06-05 13:50:21 -03:00
} else {
2020-07-30 11:41:05 +01:00
te - > set_line_gutter_item_color ( i , line_number_gutter , default_line_number_color ) ;
2018-06-05 13:50:21 -03:00
last_is_safe = false ;
}
} else {
2021-05-18 13:09:19 +10:00
te - > set_line_gutter_item_color ( i , 1 , default_line_number_color ) ;
2018-06-05 13:50:21 -03:00
}
2016-08-02 19:11:05 -03:00
}
}
2019-05-21 05:07:48 -03:00
void ScriptTextEditor : : _update_bookmark_list ( ) {
2019-07-01 16:00:05 -03:00
bookmarks_menu - > clear ( ) ;
2021-11-20 11:04:57 +03:00
bookmarks_menu - > reset_size ( ) ;
2019-05-21 05:07:48 -03:00
2019-07-01 16:00:05 -03:00
bookmarks_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_bookmark " ) , BOOKMARK_TOGGLE ) ;
bookmarks_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/remove_all_bookmarks " ) , BOOKMARK_REMOVE_ALL ) ;
bookmarks_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/goto_next_bookmark " ) , BOOKMARK_GOTO_NEXT ) ;
bookmarks_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/goto_previous_bookmark " ) , BOOKMARK_GOTO_PREV ) ;
2019-05-21 05:07:48 -03:00
2022-08-05 03:41:48 +02:00
PackedInt32Array bookmark_list = code_editor - > get_text_editor ( ) - > get_bookmarked_lines ( ) ;
2019-05-21 05:07:48 -03:00
if ( bookmark_list . size ( ) = = 0 ) {
return ;
}
2019-07-01 16:00:05 -03:00
bookmarks_menu - > add_separator ( ) ;
2019-05-21 05:07:48 -03:00
for ( int i = 0 ; i < bookmark_list . size ( ) ; i + + ) {
2019-12-03 17:19:57 +01:00
// Strip edges to remove spaces or tabs.
// Also replace any tabs by spaces, since we can't print tabs in the menu.
2020-07-24 15:50:35 +01:00
String line = code_editor - > get_text_editor ( ) - > get_line ( bookmark_list [ i ] ) . replace ( " \t " , " " ) . strip_edges ( ) ;
2019-12-03 17:19:57 +01:00
2019-05-21 05:07:48 -03:00
// Limit the size of the line if too big.
if ( line . length ( ) > 50 ) {
line = line . substr ( 0 , 50 ) ;
}
2019-12-03 17:19:57 +01:00
bookmarks_menu - > add_item ( String : : num ( ( int ) bookmark_list [ i ] + 1 ) + " - ` " + line + " ` " ) ;
2022-03-12 01:06:45 +01:00
bookmarks_menu - > set_item_metadata ( - 1 , bookmark_list [ i ] ) ;
2019-05-21 05:07:48 -03:00
}
}
void ScriptTextEditor : : _bookmark_item_pressed ( int p_idx ) {
if ( p_idx < 4 ) { // Any item before the separator.
2019-07-01 16:00:05 -03:00
_edit_option ( bookmarks_menu - > get_item_id ( p_idx ) ) ;
2019-05-21 05:07:48 -03:00
} else {
2021-03-24 12:29:14 -04:00
code_editor - > goto_line_centered ( bookmarks_menu - > get_item_metadata ( p_idx ) ) ;
2019-05-21 05:07:48 -03:00
}
}
2019-04-20 12:51:25 +01:00
static Vector < Node * > _find_all_node_for_script ( Node * p_base , Node * p_current , const Ref < Script > & p_script ) {
Vector < Node * > nodes ;
if ( p_current - > get_owner ( ) ! = p_base & & p_base ! = p_current ) {
return nodes ;
}
Ref < Script > c = p_current - > get_script ( ) ;
if ( c = = p_script ) {
nodes . push_back ( p_current ) ;
}
for ( int i = 0 ; i < p_current - > get_child_count ( ) ; i + + ) {
Vector < Node * > found = _find_all_node_for_script ( p_base , p_current - > get_child ( i ) , p_script ) ;
nodes . append_array ( found ) ;
}
return nodes ;
}
2017-03-05 16:44:50 +01:00
static Node * _find_node_for_script ( Node * p_base , Node * p_current , const Ref < Script > & p_script ) {
2020-05-14 16:41:43 +02:00
if ( p_current - > get_owner ( ) ! = p_base & & p_base ! = p_current ) {
2020-04-02 01:20:12 +02:00
return nullptr ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
Ref < Script > c = p_current - > get_script ( ) ;
2020-05-14 16:41:43 +02:00
if ( c = = p_script ) {
2016-08-02 19:11:05 -03:00
return p_current ;
2020-05-14 16:41:43 +02:00
}
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < p_current - > get_child_count ( ) ; i + + ) {
Node * found = _find_node_for_script ( p_base , p_current - > get_child ( i ) , p_script ) ;
2020-05-14 16:41:43 +02:00
if ( found ) {
2016-08-02 19:11:05 -03:00
return found ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
}
2020-04-02 01:20:12 +02:00
return nullptr ;
2016-08-02 19:11:05 -03:00
}
2022-05-19 17:00:06 +02:00
static void _find_changed_scripts_for_external_editor ( Node * p_base , Node * p_current , HashSet < Ref < Script > > & r_scripts ) {
2020-05-14 16:41:43 +02:00
if ( p_current - > get_owner ( ) ! = p_base & & p_base ! = p_current ) {
2016-08-02 19:11:05 -03:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
Ref < Script > c = p_current - > get_script ( ) ;
2020-05-14 16:41:43 +02:00
if ( c . is_valid ( ) ) {
2016-08-02 19:11:05 -03:00
r_scripts . insert ( c ) ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < p_current - > get_child_count ( ) ; i + + ) {
_find_changed_scripts_for_external_editor ( p_base , p_current - > get_child ( i ) , r_scripts ) ;
2016-08-02 19:11:05 -03:00
}
}
void ScriptEditor : : _update_modified_scripts_for_external_editor ( Ref < Script > p_for_script ) {
2023-12-22 23:27:13 +01:00
bool use_external_editor = bool ( EDITOR_GET ( " text_editor/external/use_external_editor " ) ) ;
2016-08-02 19:11:05 -03:00
2023-09-28 11:40:18 +02:00
ERR_FAIL_NULL ( get_tree ( ) ) ;
2017-06-25 17:20:01 +08:00
2022-05-19 17:00:06 +02:00
HashSet < Ref < Script > > scripts ;
2016-08-02 19:11:05 -03:00
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base ) {
2017-03-05 16:44:50 +01:00
_find_changed_scripts_for_external_editor ( base , base , scripts ) ;
2016-08-02 19:11:05 -03:00
}
2022-05-18 17:43:40 -06:00
for ( const Ref < Script > & E : scripts ) {
2022-09-29 12:53:28 +03:00
Ref < Script > scr = E ;
2016-08-02 19:11:05 -03:00
2023-12-22 23:27:13 +01:00
if ( ! use_external_editor & & ! scr - > get_language ( ) - > overrides_external_editor ( ) ) {
continue ; // We're not using an external editor for this script.
}
2022-09-29 12:53:28 +03:00
if ( p_for_script . is_valid ( ) & & p_for_script ! = scr ) {
2016-08-02 19:11:05 -03:00
continue ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
2022-09-29 12:53:28 +03:00
if ( scr - > is_built_in ( ) ) {
2016-08-02 19:11:05 -03:00
continue ; //internal script, who cares, though weird
}
2022-09-29 12:53:28 +03:00
uint64_t last_date = scr - > get_last_modified_time ( ) ;
uint64_t date = FileAccess : : get_modified_time ( scr - > get_path ( ) ) ;
2016-08-02 19:11:05 -03:00
2017-03-05 16:44:50 +01:00
if ( last_date ! = date ) {
2022-09-29 12:53:28 +03:00
Ref < Script > rel_scr = ResourceLoader : : load ( scr - > get_path ( ) , scr - > get_class ( ) , ResourceFormatLoader : : CACHE_MODE_IGNORE ) ;
2024-08-25 14:15:10 +02:00
ERR_CONTINUE ( rel_scr . is_null ( ) ) ;
2022-09-29 12:53:28 +03:00
scr - > set_source_code ( rel_scr - > get_source_code ( ) ) ;
scr - > set_last_modified_time ( rel_scr - > get_last_modified_time ( ) ) ;
scr - > update_exports ( ) ;
2021-08-17 17:32:23 +09:00
2023-12-29 23:06:54 -08:00
trigger_live_script_reload ( scr - > get_path ( ) ) ;
2016-08-02 19:11:05 -03:00
}
}
}
2022-03-26 16:48:43 +01:00
void ScriptTextEditor : : _code_complete_scripts ( void * p_ud , const String & p_code , List < ScriptLanguage : : CodeCompletionOption > * r_options , bool & r_force ) {
2016-08-02 19:11:05 -03:00
ScriptTextEditor * ste = ( ScriptTextEditor * ) p_ud ;
2017-08-24 00:06:56 -03:00
ste - > _code_complete_script ( p_code , r_options , r_force ) ;
2016-08-02 19:11:05 -03:00
}
2022-03-26 16:48:43 +01:00
void ScriptTextEditor : : _code_complete_script ( const String & p_code , List < ScriptLanguage : : CodeCompletionOption > * r_options , bool & r_force ) {
2020-05-14 16:41:43 +02:00
if ( color_panel - > is_visible ( ) ) {
2020-05-10 12:56:01 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base ) {
2017-03-05 16:44:50 +01:00
base = _find_node_for_script ( base , base , script ) ;
2016-08-02 19:11:05 -03:00
}
String hint ;
2019-04-16 22:27:13 +02:00
Error err = script - > get_language ( ) - > complete_code ( p_code , script - > get_path ( ) , base , r_options , r_force , hint ) ;
2022-03-09 00:03:36 +10:00
2019-10-02 11:21:48 +02:00
if ( err = = OK ) {
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > set_code_hint ( hint ) ;
2016-08-02 19:11:05 -03:00
}
}
2019-07-01 16:00:05 -03:00
void ScriptTextEditor : : _update_breakpoint_list ( ) {
breakpoints_menu - > clear ( ) ;
2021-11-20 11:04:57 +03:00
breakpoints_menu - > reset_size ( ) ;
2019-07-01 16:00:05 -03:00
breakpoints_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_breakpoint " ) , DEBUG_TOGGLE_BREAKPOINT ) ;
breakpoints_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/remove_all_breakpoints " ) , DEBUG_REMOVE_ALL_BREAKPOINTS ) ;
breakpoints_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/goto_next_breakpoint " ) , DEBUG_GOTO_NEXT_BREAKPOINT ) ;
breakpoints_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/goto_previous_breakpoint " ) , DEBUG_GOTO_PREV_BREAKPOINT ) ;
2022-08-05 03:41:48 +02:00
PackedInt32Array breakpoint_list = code_editor - > get_text_editor ( ) - > get_breakpointed_lines ( ) ;
2019-07-01 16:00:05 -03:00
if ( breakpoint_list . size ( ) = = 0 ) {
return ;
}
breakpoints_menu - > add_separator ( ) ;
for ( int i = 0 ; i < breakpoint_list . size ( ) ; i + + ) {
2019-12-03 17:19:57 +01:00
// Strip edges to remove spaces or tabs.
// Also replace any tabs by spaces, since we can't print tabs in the menu.
2020-07-24 15:50:35 +01:00
String line = code_editor - > get_text_editor ( ) - > get_line ( breakpoint_list [ i ] ) . replace ( " \t " , " " ) . strip_edges ( ) ;
2019-12-03 17:19:57 +01:00
2019-07-01 16:00:05 -03:00
// Limit the size of the line if too big.
if ( line . length ( ) > 50 ) {
line = line . substr ( 0 , 50 ) ;
}
2019-12-03 17:19:57 +01:00
breakpoints_menu - > add_item ( String : : num ( ( int ) breakpoint_list [ i ] + 1 ) + " - ` " + line + " ` " ) ;
2022-03-12 01:06:45 +01:00
breakpoints_menu - > set_item_metadata ( - 1 , breakpoint_list [ i ] ) ;
2019-07-01 16:00:05 -03:00
}
}
void ScriptTextEditor : : _breakpoint_item_pressed ( int p_idx ) {
if ( p_idx < 4 ) { // Any item before the separator.
_edit_option ( breakpoints_menu - > get_item_id ( p_idx ) ) ;
} else {
2024-07-05 19:27:19 -04:00
code_editor - > goto_line_centered ( breakpoints_menu - > get_item_metadata ( p_idx ) ) ;
2019-07-01 16:00:05 -03:00
}
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : _breakpoint_toggled ( int p_row ) {
2020-07-26 15:57:23 +01:00
EditorDebuggerNode : : get_singleton ( ) - > set_breakpoint ( script - > get_path ( ) , p_row + 1 , code_editor - > get_text_editor ( ) - > is_line_breakpointed ( p_row ) ) ;
2016-08-02 19:11:05 -03:00
}
2022-10-22 23:54:37 +02:00
void ScriptTextEditor : : _on_caret_moved ( ) {
2024-02-22 15:32:12 -05:00
if ( code_editor - > is_previewing_navigation_change ( ) ) {
return ;
}
2022-10-22 23:54:37 +02:00
int current_line = code_editor - > get_text_editor ( ) - > get_caret_line ( ) ;
if ( ABS ( current_line - previous_line ) > = 10 ) {
Dictionary nav_state = get_navigation_state ( ) ;
nav_state [ " row " ] = previous_line ;
nav_state [ " scroll_position " ] = - 1 ;
emit_signal ( SNAME ( " request_save_previous_state " ) , nav_state ) ;
store_previous_state ( ) ;
}
previous_line = current_line ;
}
2017-03-05 16:44:50 +01:00
void ScriptTextEditor : : _lookup_symbol ( const String & p_symbol , int p_row , int p_column ) {
2016-09-12 10:52:29 -03:00
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base ) {
2017-03-05 16:44:50 +01:00
base = _find_node_for_script ( base , base , script ) ;
2016-09-12 10:52:29 -03:00
}
ScriptLanguage : : LookupResult result ;
2023-02-13 07:33:10 +03:00
String code_text = code_editor - > get_text_editor ( ) - > get_text_with_cursor_char ( p_row , p_column ) ;
Error lc_error = script - > get_language ( ) - > lookup_code ( code_text , p_symbol , script - > get_path ( ) , base , result ) ;
2019-03-11 12:25:54 -03:00
if ( ScriptServer : : is_global_class ( p_symbol ) ) {
EditorNode : : get_singleton ( ) - > load_resource ( ScriptServer : : get_global_class_path ( p_symbol ) ) ;
2023-09-19 15:23:28 +02:00
} else if ( p_symbol . is_resource_file ( ) | | p_symbol . begins_with ( " uid:// " ) ) {
String symbol = p_symbol ;
if ( symbol . begins_with ( " uid:// " ) ) {
2024-11-14 23:22:59 +01:00
symbol = ResourceUID : : uid_to_path ( symbol ) ;
2023-09-19 15:23:28 +02:00
}
2017-08-02 19:20:27 +02:00
List < String > scene_extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " PackedScene " , & scene_extensions ) ;
2023-09-19 15:23:28 +02:00
if ( scene_extensions . find ( symbol . get_extension ( ) ) ) {
EditorNode : : get_singleton ( ) - > load_scene ( symbol ) ;
2017-08-02 19:20:27 +02:00
} else {
2023-09-19 15:23:28 +02:00
EditorNode : : get_singleton ( ) - > load_resource ( symbol ) ;
2017-08-02 19:20:27 +02:00
}
2023-02-13 07:33:10 +03:00
} else if ( lc_error = = OK ) {
2016-09-12 10:52:29 -03:00
_goto_line ( p_row ) ;
2024-12-21 21:10:09 +03:00
if ( ! result . class_name . is_empty ( ) & & EditorHelp : : get_doc_data ( ) - > class_list . has ( result . class_name ) & & ! EditorHelp : : get_doc_data ( ) - > class_list [ result . class_name ] . is_script_doc ) {
2024-12-14 21:25:05 +03:00
switch ( result . type ) {
case ScriptLanguage : : LOOKUP_RESULT_CLASS : {
emit_signal ( SNAME ( " go_to_help " ) , " class_name: " + result . class_name ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_CONSTANT : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_integer_constant ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
2017-03-05 16:44:50 +01:00
cname = ClassDB : : get_parent_class ( cname ) ;
2016-09-12 10:52:29 -03:00
}
2024-12-14 21:25:05 +03:00
emit_signal ( SNAME ( " go_to_help " ) , " class_constant: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_PROPERTY : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_property ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
2017-03-05 16:44:50 +01:00
cname = ClassDB : : get_parent_class ( cname ) ;
2016-09-12 10:52:29 -03:00
}
2024-12-14 21:25:05 +03:00
emit_signal ( SNAME ( " go_to_help " ) , " class_property: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_METHOD : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_method ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
2022-05-12 16:07:05 +03:00
cname = ClassDB : : get_parent_class ( cname ) ;
}
2024-12-14 21:25:05 +03:00
emit_signal ( SNAME ( " go_to_help " ) , " class_method: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_SIGNAL : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_signal ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
2018-03-30 16:20:24 +02:00
cname = ClassDB : : get_parent_class ( cname ) ;
}
2024-12-14 21:25:05 +03:00
emit_signal ( SNAME ( " go_to_help " ) , " class_signal: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_ENUM : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_enum ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
cname = ClassDB : : get_parent_class ( cname ) ;
}
emit_signal ( SNAME ( " go_to_help " ) , " class_enum: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_ANNOTATION : {
emit_signal ( SNAME ( " go_to_help " ) , " class_annotation: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_TBD_GLOBALSCOPE : { // Deprecated.
emit_signal ( SNAME ( " go_to_help " ) , " class_global: " + result . class_name + " : " + result . class_member ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_SCRIPT_LOCATION :
case ScriptLanguage : : LOOKUP_RESULT_LOCAL_CONSTANT :
case ScriptLanguage : : LOOKUP_RESULT_LOCAL_VARIABLE :
case ScriptLanguage : : LOOKUP_RESULT_MAX : {
// Nothing to do.
} break ;
}
} else if ( result . location > = 0 ) {
if ( result . script . is_valid ( ) ) {
emit_signal ( SNAME ( " request_open_script_at_line " ) , result . script , result . location - 1 ) ;
} else {
emit_signal ( SNAME ( " request_save_history " ) ) ;
goto_line_centered ( result . location - 1 ) ;
2022-03-26 16:48:43 +01:00
}
2016-09-12 10:52:29 -03:00
}
2020-06-17 20:45:08 -03:00
} else if ( ProjectSettings : : get_singleton ( ) - > has_autoload ( p_symbol ) ) {
// Check for Autoload scenes.
const ProjectSettings : : AutoloadInfo & info = ProjectSettings : : get_singleton ( ) - > get_autoload ( p_symbol ) ;
if ( info . is_singleton ) {
EditorNode : : get_singleton ( ) - > load_scene ( info . path ) ;
2020-03-28 13:42:19 +05:30
}
2021-08-29 19:43:47 -04:00
} else if ( p_symbol . is_relative_path ( ) ) {
2020-03-28 13:42:19 +05:30
// Every symbol other than absolute path is relative path so keep this condition at last.
String path = _get_absolute_path ( p_symbol ) ;
if ( FileAccess : : exists ( path ) ) {
List < String > scene_extensions ;
ResourceLoader : : get_recognized_extensions_for_type ( " PackedScene " , & scene_extensions ) ;
if ( scene_extensions . find ( path . get_extension ( ) ) ) {
EditorNode : : get_singleton ( ) - > load_scene ( path ) ;
} else {
EditorNode : : get_singleton ( ) - > load_resource ( path ) ;
}
}
2016-09-12 10:52:29 -03:00
}
}
2016-08-02 19:11:05 -03:00
2020-03-01 13:43:41 +05:30
void ScriptTextEditor : : _validate_symbol ( const String & p_symbol ) {
2020-07-24 15:50:35 +01:00
CodeEdit * text_edit = code_editor - > get_text_editor ( ) ;
2020-03-01 13:43:41 +05:30
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base ) {
base = _find_node_for_script ( base , base , script ) ;
}
ScriptLanguage : : LookupResult result ;
2023-02-13 07:33:10 +03:00
String lc_text = code_editor - > get_text_editor ( ) - > get_text_for_symbol_lookup ( ) ;
Error lc_error = script - > get_language ( ) - > lookup_code ( lc_text , p_symbol , script - > get_path ( ) , base , result ) ;
bool is_singleton = ProjectSettings : : get_singleton ( ) - > has_autoload ( p_symbol ) & & ProjectSettings : : get_singleton ( ) - > get_autoload ( p_symbol ) . is_singleton ;
2023-09-19 15:23:28 +02:00
if ( lc_error = = OK | | is_singleton | | ScriptServer : : is_global_class ( p_symbol ) | | p_symbol . is_resource_file ( ) | | p_symbol . begins_with ( " uid:// " ) ) {
2021-07-01 17:10:54 +01:00
text_edit - > set_symbol_lookup_word_as_valid ( true ) ;
2021-08-29 19:43:47 -04:00
} else if ( p_symbol . is_relative_path ( ) ) {
2020-03-28 13:42:19 +05:30
String path = _get_absolute_path ( p_symbol ) ;
if ( FileAccess : : exists ( path ) ) {
2021-07-01 17:10:54 +01:00
text_edit - > set_symbol_lookup_word_as_valid ( true ) ;
2020-03-28 13:42:19 +05:30
} else {
2021-07-01 17:10:54 +01:00
text_edit - > set_symbol_lookup_word_as_valid ( false ) ;
2020-03-28 13:42:19 +05:30
}
2020-03-01 13:43:41 +05:30
} else {
2021-07-01 17:10:54 +01:00
text_edit - > set_symbol_lookup_word_as_valid ( false ) ;
2020-03-01 13:43:41 +05:30
}
}
2024-12-14 21:25:05 +03:00
void ScriptTextEditor : : _show_symbol_tooltip ( const String & p_symbol , int p_row , int p_column ) {
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base ) {
base = _find_node_for_script ( base , base , script ) ;
}
ScriptLanguage : : LookupResult result ;
2025-01-05 09:57:21 -05:00
String doc_symbol ;
2024-12-14 21:25:05 +03:00
const String code_text = code_editor - > get_text_editor ( ) - > get_text_with_cursor_char ( p_row , p_column ) ;
const Error lc_error = script - > get_language ( ) - > lookup_code ( code_text , p_symbol , script - > get_path ( ) , base , result ) ;
2025-01-05 09:57:21 -05:00
if ( lc_error = = OK ) {
switch ( result . type ) {
case ScriptLanguage : : LOOKUP_RESULT_CLASS : {
doc_symbol = " class| " + result . class_name + " | " ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_CONSTANT : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_integer_constant ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
cname = ClassDB : : get_parent_class ( cname ) ;
2024-12-14 21:25:05 +03:00
}
2025-01-05 09:57:21 -05:00
doc_symbol = " constant| " + result . class_name + " | " + result . class_member ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_PROPERTY : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_property ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
cname = ClassDB : : get_parent_class ( cname ) ;
2024-12-14 21:25:05 +03:00
}
2025-01-05 09:57:21 -05:00
doc_symbol = " property| " + result . class_name + " | " + result . class_member ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_METHOD : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_method ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
cname = ClassDB : : get_parent_class ( cname ) ;
2024-12-14 21:25:05 +03:00
}
2025-01-05 09:57:21 -05:00
doc_symbol = " method| " + result . class_name + " | " + result . class_member ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_SIGNAL : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_signal ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
cname = ClassDB : : get_parent_class ( cname ) ;
2024-12-14 21:25:05 +03:00
}
2025-01-05 09:57:21 -05:00
doc_symbol = " signal| " + result . class_name + " | " + result . class_member ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_ENUM : {
StringName cname = result . class_name ;
while ( ClassDB : : class_exists ( cname ) ) {
if ( ClassDB : : has_enum ( cname , result . class_member , true ) ) {
result . class_name = cname ;
break ;
}
cname = ClassDB : : get_parent_class ( cname ) ;
2024-12-14 21:25:05 +03:00
}
2025-01-05 09:57:21 -05:00
doc_symbol = " enum| " + result . class_name + " | " + result . class_member ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_CLASS_ANNOTATION : {
doc_symbol = " annotation| " + result . class_name + " | " + result . class_member ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_LOCAL_CONSTANT :
case ScriptLanguage : : LOOKUP_RESULT_LOCAL_VARIABLE : {
const String item_type = ( result . type = = ScriptLanguage : : LOOKUP_RESULT_LOCAL_CONSTANT ) ? " local_constant " : " local_variable " ;
Dictionary item_data ;
item_data [ " description " ] = result . description ;
item_data [ " is_deprecated " ] = result . is_deprecated ;
item_data [ " deprecated_message " ] = result . deprecated_message ;
item_data [ " is_experimental " ] = result . is_experimental ;
item_data [ " experimental_message " ] = result . experimental_message ;
item_data [ " doc_type " ] = result . doc_type ;
item_data [ " enumeration " ] = result . enumeration ;
item_data [ " is_bitfield " ] = result . is_bitfield ;
item_data [ " value " ] = result . value ;
doc_symbol = item_type + " || " + p_symbol + " | " + JSON : : stringify ( item_data ) ;
} break ;
case ScriptLanguage : : LOOKUP_RESULT_SCRIPT_LOCATION :
case ScriptLanguage : : LOOKUP_RESULT_CLASS_TBD_GLOBALSCOPE : // Deprecated.
case ScriptLanguage : : LOOKUP_RESULT_MAX : {
// Nothing to do.
} break ;
}
}
String debug_value = EditorDebuggerNode : : get_singleton ( ) - > get_var_value ( p_symbol ) ;
if ( ! debug_value . is_empty ( ) ) {
constexpr int DISPLAY_LIMIT = 1024 ;
if ( debug_value . size ( ) > DISPLAY_LIMIT ) {
debug_value = debug_value . left ( DISPLAY_LIMIT ) + " ... " + TTR ( " (truncated) " ) ;
}
debug_value = debug_value . replace ( " [ " , " [lb] " ) ;
if ( doc_symbol . is_empty ( ) ) {
debug_value = p_symbol + " : " + debug_value ;
} else {
debug_value = TTR ( " Current value: " ) + debug_value ;
}
2024-12-14 21:25:05 +03:00
}
2025-01-05 09:57:21 -05:00
if ( ! doc_symbol . is_empty ( ) | | ! debug_value . is_empty ( ) ) {
EditorHelpBitTooltip : : show_tooltip ( code_editor - > get_text_editor ( ) , doc_symbol , debug_value , true ) ;
2024-12-14 21:25:05 +03:00
}
}
2020-03-28 13:42:19 +05:30
String ScriptTextEditor : : _get_absolute_path ( const String & rel_path ) {
String base_path = script - > get_path ( ) . get_base_dir ( ) ;
2022-08-29 19:34:01 -05:00
String path = base_path . path_join ( rel_path ) ;
2020-03-28 13:42:19 +05:30
return path . replace ( " /// " , " // " ) . simplify_path ( ) ;
}
2019-12-18 22:32:25 +03:00
void ScriptTextEditor : : update_toggle_scripts_button ( ) {
2021-09-14 12:17:47 +03:00
code_editor - > update_toggle_scripts_button ( ) ;
2019-12-18 22:32:25 +03:00
}
2019-04-20 12:51:25 +01:00
void ScriptTextEditor : : _update_connected_methods ( ) {
2020-07-24 15:50:35 +01:00
CodeEdit * text_edit = code_editor - > get_text_editor ( ) ;
2021-07-10 11:41:38 +01:00
text_edit - > set_gutter_width ( connection_gutter , text_edit - > get_line_height ( ) ) ;
2020-07-29 22:26:49 +01:00
for ( int i = 0 ; i < text_edit - > get_line_count ( ) ; i + + ) {
2022-09-09 16:36:07 +02:00
text_edit - > set_line_gutter_metadata ( i , connection_gutter , Dictionary ( ) ) ;
2020-07-29 22:26:49 +01:00
text_edit - > set_line_gutter_icon ( i , connection_gutter , nullptr ) ;
text_edit - > set_line_gutter_clickable ( i , connection_gutter , false ) ;
}
2019-04-20 12:51:25 +01:00
missing_connections . clear ( ) ;
2019-11-02 13:38:25 +00:00
if ( ! script_is_valid ) {
2019-10-27 11:07:19 +00:00
return ;
}
2019-04-20 12:51:25 +01:00
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( ! base ) {
return ;
}
2022-09-09 16:36:07 +02:00
// Add connection icons to methods.
2019-04-20 12:51:25 +01:00
Vector < Node * > nodes = _find_all_node_for_script ( base , base , script ) ;
2022-05-19 17:00:06 +02:00
HashSet < StringName > methods_found ;
2019-04-20 12:51:25 +01:00
for ( int i = 0 ; i < nodes . size ( ) ; i + + ) {
2022-09-29 12:53:28 +03:00
List < Connection > signal_connections ;
nodes [ i ] - > get_signals_connected_to_this ( & signal_connections ) ;
2019-04-20 12:51:25 +01:00
2022-09-29 12:53:28 +03:00
for ( const Connection & connection : signal_connections ) {
2019-04-20 12:51:25 +01:00
if ( ! ( connection . flags & CONNECT_PERSIST ) ) {
continue ;
}
2019-06-10 19:35:22 -03:00
// As deleted nodes are still accessible via the undo/redo system, check if they're still on the tree.
2020-02-19 16:27:19 -03:00
Node * source = Object : : cast_to < Node > ( connection . signal . get_object ( ) ) ;
2019-06-10 19:35:22 -03:00
if ( source & & ! source - > is_inside_tree ( ) ) {
continue ;
}
2022-02-06 19:02:53 +01:00
const StringName method = connection . callable . get_method ( ) ;
if ( methods_found . has ( method ) ) {
2019-10-27 11:07:19 +00:00
continue ;
}
2022-02-06 19:02:53 +01:00
if ( ! ClassDB : : has_method ( script - > get_instance_base_type ( ) , method ) ) {
2019-10-27 11:07:19 +00:00
int line = - 1 ;
2019-11-02 13:38:25 +00:00
for ( int j = 0 ; j < functions . size ( ) ; j + + ) {
String name = functions [ j ] . get_slice ( " : " , 0 ) ;
2022-02-06 19:02:53 +01:00
if ( name = = method ) {
2022-09-09 16:36:07 +02:00
Dictionary line_meta ;
line_meta [ " type " ] = " connection " ;
line_meta [ " method " ] = method ;
2020-07-29 22:26:49 +01:00
line = functions [ j ] . get_slice ( " : " , 1 ) . to_int ( ) - 1 ;
2022-09-09 16:36:07 +02:00
text_edit - > set_line_gutter_metadata ( line , connection_gutter , line_meta ) ;
2023-08-13 02:33:39 +02:00
text_edit - > set_line_gutter_icon ( line , connection_gutter , get_parent_control ( ) - > get_editor_theme_icon ( SNAME ( " Slot " ) ) ) ;
2020-07-29 22:26:49 +01:00
text_edit - > set_line_gutter_clickable ( line , connection_gutter , true ) ;
2022-02-06 19:02:53 +01:00
methods_found . insert ( method ) ;
2019-11-02 13:38:25 +00:00
break ;
}
}
if ( line > = 0 ) {
2019-10-27 11:07:19 +00:00
continue ;
}
2019-06-13 12:15:13 -03:00
2019-10-27 11:07:19 +00:00
// There is a chance that the method is inherited from another script.
bool found_inherited_function = false ;
Ref < Script > inherited_script = script - > get_base_script ( ) ;
2024-08-25 14:15:10 +02:00
while ( inherited_script . is_valid ( ) ) {
2022-02-06 19:02:53 +01:00
if ( inherited_script - > has_method ( method ) ) {
2019-10-27 11:07:19 +00:00
found_inherited_function = true ;
break ;
2019-06-11 19:48:35 -03:00
}
2019-10-27 11:07:19 +00:00
inherited_script = inherited_script - > get_base_script ( ) ;
}
if ( ! found_inherited_function ) {
missing_connections . push_back ( connection ) ;
2019-06-11 19:48:35 -03:00
}
2019-04-20 12:51:25 +01:00
}
}
}
2022-09-09 16:36:07 +02:00
// Add override icons to methods.
methods_found . clear ( ) ;
for ( int i = 0 ; i < functions . size ( ) ; i + + ) {
2024-03-16 00:09:03 +01:00
String raw_name = functions [ i ] . get_slice ( " : " , 0 ) ;
StringName name = StringName ( raw_name ) ;
2022-09-09 16:36:07 +02:00
if ( methods_found . has ( name ) ) {
continue ;
}
2024-03-26 20:54:14 +01:00
// Account for inner classes by stripping the class names from the method,
// starting from the right since our inner class might be inside of another inner class.
2024-11-16 18:52:15 +01:00
int pos = raw_name . rfind_char ( ' . ' ) ;
2024-03-26 20:54:14 +01:00
if ( pos ! = - 1 ) {
name = raw_name . substr ( pos + 1 ) ;
2024-03-16 00:09:03 +01:00
}
2022-09-09 16:36:07 +02:00
String found_base_class ;
StringName base_class = script - > get_instance_base_type ( ) ;
Ref < Script > inherited_script = script - > get_base_script ( ) ;
2024-08-25 14:15:10 +02:00
while ( inherited_script . is_valid ( ) ) {
2022-09-09 16:36:07 +02:00
if ( inherited_script - > has_method ( name ) ) {
found_base_class = " script: " + inherited_script - > get_path ( ) ;
break ;
}
base_class = inherited_script - > get_instance_base_type ( ) ;
inherited_script = inherited_script - > get_base_script ( ) ;
}
2022-09-30 02:11:52 +02:00
if ( found_base_class . is_empty ( ) ) {
while ( base_class ) {
List < MethodInfo > methods ;
ClassDB : : get_method_list ( base_class , & methods , true ) ;
2024-04-15 15:18:34 +02:00
for ( const MethodInfo & mi : methods ) {
if ( mi . name = = name ) {
2022-09-30 02:11:52 +02:00
found_base_class = " builtin: " + base_class ;
break ;
}
}
ClassDB : : ClassInfo * base_class_ptr = ClassDB : : classes . getptr ( base_class ) - > inherits_ptr ;
if ( base_class_ptr = = nullptr ) {
2022-09-09 16:36:07 +02:00
break ;
}
2022-09-30 02:11:52 +02:00
base_class = base_class_ptr - > name ;
2022-09-09 16:36:07 +02:00
}
}
if ( ! found_base_class . is_empty ( ) ) {
int line = functions [ i ] . get_slice ( " : " , 1 ) . to_int ( ) - 1 ;
Dictionary line_meta = text_edit - > get_line_gutter_metadata ( line , connection_gutter ) ;
if ( line_meta . is_empty ( ) ) {
// Add override icon to gutter.
line_meta [ " type " ] = " inherits " ;
line_meta [ " method " ] = name ;
line_meta [ " base_class " ] = found_base_class ;
2023-08-13 02:33:39 +02:00
text_edit - > set_line_gutter_icon ( line , connection_gutter , get_parent_control ( ) - > get_editor_theme_icon ( SNAME ( " MethodOverride " ) ) ) ;
2022-09-09 16:36:07 +02:00
text_edit - > set_line_gutter_clickable ( line , connection_gutter , true ) ;
} else {
// If method is also connected to signal, then merge icons and keep the click behavior of the slot.
2023-08-13 02:33:39 +02:00
text_edit - > set_line_gutter_icon ( line , connection_gutter , get_parent_control ( ) - > get_editor_theme_icon ( SNAME ( " MethodOverrideAndSlot " ) ) ) ;
2022-09-09 16:36:07 +02:00
}
2024-03-16 00:09:03 +01:00
methods_found . insert ( StringName ( raw_name ) ) ;
2022-09-09 16:36:07 +02:00
}
}
2019-04-20 12:51:25 +01:00
}
2020-07-29 22:26:49 +01:00
void ScriptTextEditor : : _update_gutter_indexes ( ) {
for ( int i = 0 ; i < code_editor - > get_text_editor ( ) - > get_gutter_count ( ) ; i + + ) {
if ( code_editor - > get_text_editor ( ) - > get_gutter_name ( i ) = = " connection_gutter " ) {
connection_gutter = i ;
2020-07-30 11:41:05 +01:00
continue ;
}
if ( code_editor - > get_text_editor ( ) - > get_gutter_name ( i ) = = " line_numbers " ) {
line_number_gutter = i ;
continue ;
2020-07-29 22:26:49 +01:00
}
}
}
void ScriptTextEditor : : _gutter_clicked ( int p_line , int p_gutter ) {
if ( p_gutter ! = connection_gutter ) {
return ;
}
2022-09-09 16:36:07 +02:00
Dictionary meta = code_editor - > get_text_editor ( ) - > get_line_gutter_metadata ( p_line , p_gutter ) ;
String type = meta . get ( " type " , " " ) ;
if ( type . is_empty ( ) ) {
2020-07-29 22:26:49 +01:00
return ;
}
2022-09-09 16:36:07 +02:00
// All types currently need a method name.
String method = meta . get ( " method " , " " ) ;
if ( method . is_empty ( ) ) {
2019-04-20 12:51:25 +01:00
return ;
}
2022-09-09 16:36:07 +02:00
if ( type = = " connection " ) {
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( ! base ) {
return ;
}
Vector < Node * > nodes = _find_all_node_for_script ( base , base , script ) ;
connection_info_dialog - > popup_connections ( method , nodes ) ;
} else if ( type = = " inherits " ) {
String base_class_raw = meta [ " base_class " ] ;
PackedStringArray base_class_split = base_class_raw . split ( " : " , true , 1 ) ;
if ( base_class_split [ 0 ] = = " script " ) {
// Go to function declaration.
Ref < Script > base_script = ResourceLoader : : load ( base_class_split [ 1 ] ) ;
2024-08-25 14:15:10 +02:00
ERR_FAIL_COND ( base_script . is_null ( ) ) ;
2022-09-09 16:36:07 +02:00
emit_signal ( SNAME ( " go_to_method " ) , base_script , method ) ;
} else if ( base_class_split [ 0 ] = = " builtin " ) {
// Open method documentation.
emit_signal ( SNAME ( " go_to_help " ) , " class_method: " + base_class_split [ 1 ] + " : " + method ) ;
}
}
2019-04-20 12:51:25 +01:00
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : _edit_option ( int p_op ) {
2020-07-24 15:50:35 +01:00
CodeEdit * tx = code_editor - > get_text_editor ( ) ;
2024-01-22 12:34:12 -05:00
tx - > apply_ime ( ) ;
2017-11-15 23:00:27 -05:00
2017-03-05 16:44:50 +01:00
switch ( p_op ) {
2016-08-02 19:11:05 -03:00
case EDIT_UNDO : {
2017-11-15 23:00:27 -05:00
tx - > undo ( ) ;
2023-12-18 15:46:56 +01:00
callable_mp ( ( Control * ) tx , & Control : : grab_focus ) . call_deferred ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_REDO : {
2017-11-15 23:00:27 -05:00
tx - > redo ( ) ;
2023-12-18 15:46:56 +01:00
callable_mp ( ( Control * ) tx , & Control : : grab_focus ) . call_deferred ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_CUT : {
2017-11-15 23:00:27 -05:00
tx - > cut ( ) ;
2023-12-18 15:46:56 +01:00
callable_mp ( ( Control * ) tx , & Control : : grab_focus ) . call_deferred ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_COPY : {
2017-11-15 23:00:27 -05:00
tx - > copy ( ) ;
2023-12-18 15:46:56 +01:00
callable_mp ( ( Control * ) tx , & Control : : grab_focus ) . call_deferred ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_PASTE : {
2017-11-15 23:00:27 -05:00
tx - > paste ( ) ;
2023-12-18 15:46:56 +01:00
callable_mp ( ( Control * ) tx , & Control : : grab_focus ) . call_deferred ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_SELECT_ALL : {
2017-11-15 23:00:27 -05:00
tx - > select_all ( ) ;
2023-12-18 15:46:56 +01:00
callable_mp ( ( Control * ) tx , & Control : : grab_focus ) . call_deferred ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_MOVE_LINE_UP : {
2024-01-22 18:27:19 -05:00
code_editor - > get_text_editor ( ) - > move_lines_up ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_MOVE_LINE_DOWN : {
2024-01-22 18:27:19 -05:00
code_editor - > get_text_editor ( ) - > move_lines_down ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
2022-09-29 04:12:52 +02:00
case EDIT_INDENT : {
2018-05-28 16:52:28 +01:00
Ref < Script > scr = script ;
2020-05-14 16:41:43 +02:00
if ( scr . is_null ( ) ) {
2016-08-02 19:11:05 -03:00
return ;
2020-05-14 16:41:43 +02:00
}
2022-09-29 04:12:52 +02:00
tx - > indent_lines ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
2022-09-29 04:12:52 +02:00
case EDIT_UNINDENT : {
2018-05-28 16:52:28 +01:00
Ref < Script > scr = script ;
2020-05-14 16:41:43 +02:00
if ( scr . is_null ( ) ) {
2016-08-02 19:11:05 -03:00
return ;
2020-05-14 16:41:43 +02:00
}
2022-09-29 04:12:52 +02:00
tx - > unindent_lines ( ) ;
2017-08-20 13:31:30 +01:00
} break ;
case EDIT_DELETE_LINE : {
2024-01-22 18:27:19 -05:00
code_editor - > get_text_editor ( ) - > delete_lines ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
2021-06-29 12:09:07 +02:00
case EDIT_DUPLICATE_SELECTION : {
2024-01-22 18:27:19 -05:00
code_editor - > get_text_editor ( ) - > duplicate_selection ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
2022-09-28 17:09:45 +02:00
case EDIT_DUPLICATE_LINES : {
code_editor - > get_text_editor ( ) - > duplicate_lines ( ) ;
} break ;
2017-12-08 19:17:10 +01:00
case EDIT_TOGGLE_FOLD_LINE : {
2024-01-22 18:27:19 -05:00
tx - > toggle_foldable_lines_at_carets ( ) ;
2017-11-12 18:12:17 -05:00
} break ;
2017-11-15 23:00:27 -05:00
case EDIT_FOLD_ALL_LINES : {
tx - > fold_all_lines ( ) ;
2022-08-13 23:21:24 +02:00
tx - > queue_redraw ( ) ;
2017-11-15 23:00:27 -05:00
} break ;
2017-11-12 18:12:17 -05:00
case EDIT_UNFOLD_ALL_LINES : {
2021-07-09 13:07:10 +01:00
tx - > unfold_all_lines ( ) ;
2022-08-13 23:21:24 +02:00
tx - > queue_redraw ( ) ;
2017-11-12 18:12:17 -05:00
} break ;
2023-03-12 17:48:37 +01:00
case EDIT_CREATE_CODE_REGION : {
tx - > create_code_region ( ) ;
} break ;
2016-08-02 19:11:05 -03:00
case EDIT_TOGGLE_COMMENT : {
2019-02-04 20:17:44 +01:00
_edit_option_toggle_inline_comment ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_COMPLETE : {
2020-09-13 21:14:20 +01:00
tx - > request_code_completion ( true ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_AUTO_INDENT : {
2017-11-15 23:00:27 -05:00
String text = tx - > get_text ( ) ;
2018-05-28 16:52:28 +01:00
Ref < Script > scr = script ;
2020-05-14 16:41:43 +02:00
if ( scr . is_null ( ) ) {
2016-08-02 19:11:05 -03:00
return ;
2020-05-14 16:41:43 +02:00
}
2017-08-22 19:52:17 +02:00
2017-11-15 23:00:27 -05:00
tx - > begin_complex_operation ( ) ;
2024-01-22 18:27:19 -05:00
tx - > begin_multicaret_edit ( ) ;
int begin = tx - > get_line_count ( ) - 1 , end = 0 ;
2021-07-09 12:42:55 +01:00
if ( tx - > has_selection ( ) ) {
2024-01-22 18:27:19 -05:00
// Auto indent all lines that have a caret or selection on it.
Vector < Point2i > line_ranges = tx - > get_line_ranges_from_carets ( ) ;
for ( Point2i line_range : line_ranges ) {
scr - > get_language ( ) - > auto_indent_code ( text , line_range . x , line_range . y ) ;
if ( line_range . x < begin ) {
begin = line_range . x ;
}
if ( line_range . y > end ) {
end = line_range . y ;
}
2017-08-22 19:52:17 +02:00
}
2016-08-02 19:11:05 -03:00
} else {
2024-01-22 18:27:19 -05:00
// Auto indent entire text.
2017-03-05 16:44:50 +01:00
begin = 0 ;
2017-11-15 23:00:27 -05:00
end = tx - > get_line_count ( ) - 1 ;
2024-01-22 18:27:19 -05:00
scr - > get_language ( ) - > auto_indent_code ( text , begin , end ) ;
2016-08-02 19:11:05 -03:00
}
2024-01-22 18:27:19 -05:00
// Apply auto indented code.
2017-08-22 19:52:17 +02:00
Vector < String > lines = text . split ( " \n " ) ;
for ( int i = begin ; i < = end ; + + i ) {
2017-11-15 23:00:27 -05:00
tx - > set_line ( i , lines [ i ] ) ;
2017-08-22 19:52:17 +02:00
}
2024-01-22 18:27:19 -05:00
tx - > end_multicaret_edit ( ) ;
2017-11-15 23:00:27 -05:00
tx - > end_complex_operation ( ) ;
2016-08-02 19:11:05 -03:00
} break ;
case EDIT_TRIM_TRAILING_WHITESAPCE : {
trim_trailing_whitespace ( ) ;
} break ;
2024-01-12 00:32:13 -05:00
case EDIT_TRIM_FINAL_NEWLINES : {
trim_final_newlines ( ) ;
} break ;
2017-04-16 16:47:59 +01:00
case EDIT_CONVERT_INDENT_TO_SPACES : {
2024-02-17 20:16:58 -06:00
code_editor - > set_indent_using_spaces ( true ) ;
2023-05-01 21:41:50 +01:00
convert_indent ( ) ;
2017-04-16 16:47:59 +01:00
} break ;
case EDIT_CONVERT_INDENT_TO_TABS : {
2024-02-17 20:16:58 -06:00
code_editor - > set_indent_using_spaces ( false ) ;
2023-05-01 21:41:50 +01:00
convert_indent ( ) ;
2017-04-16 16:47:59 +01:00
} break ;
2016-09-29 09:12:45 +02:00
case EDIT_PICK_COLOR : {
color_panel - > popup ( ) ;
} break ;
2017-04-24 19:59:55 +01:00
case EDIT_TO_UPPERCASE : {
2018-05-25 23:49:35 +01:00
_convert_case ( CodeTextEditor : : UPPER ) ;
2017-04-24 19:59:55 +01:00
} break ;
case EDIT_TO_LOWERCASE : {
2018-05-25 23:49:35 +01:00
_convert_case ( CodeTextEditor : : LOWER ) ;
2017-04-24 19:59:55 +01:00
} break ;
case EDIT_CAPITALIZE : {
2018-05-25 23:49:35 +01:00
_convert_case ( CodeTextEditor : : CAPITALIZE ) ;
2017-04-24 19:59:55 +01:00
} break ;
2019-08-07 16:31:33 +02:00
case EDIT_EVALUATE : {
Expression expression ;
2022-12-05 19:05:02 +01:00
tx - > begin_complex_operation ( ) ;
for ( int caret_idx = 0 ; caret_idx < tx - > get_caret_count ( ) ; caret_idx + + ) {
Vector < String > lines = tx - > get_selected_text ( caret_idx ) . split ( " \n " ) ;
PackedStringArray results ;
for ( int i = 0 ; i < lines . size ( ) ; i + + ) {
2023-11-18 17:40:56 -05:00
const String & line = lines [ i ] ;
2022-12-05 19:05:02 +01:00
String whitespace = line . substr ( 0 , line . size ( ) - line . strip_edges ( true , false ) . size ( ) ) ; // Extract the whitespace at the beginning.
if ( expression . parse ( line ) = = OK ) {
Variant result = expression . execute ( Array ( ) , Variant ( ) , false , true ) ;
if ( expression . get_error_text ( ) . is_empty ( ) ) {
results . push_back ( whitespace + result . get_construct_string ( ) ) ;
} else {
results . push_back ( line ) ;
}
2019-08-07 16:31:33 +02:00
} else {
2020-02-17 18:06:54 -03:00
results . push_back ( line ) ;
2019-08-07 16:31:33 +02:00
}
}
2022-12-05 19:05:02 +01:00
tx - > insert_text_at_caret ( String ( " \n " ) . join ( results ) , caret_idx ) ;
2019-08-07 16:31:33 +02:00
}
2022-12-05 19:05:02 +01:00
tx - > end_complex_operation ( ) ;
2019-08-07 16:31:33 +02:00
} break ;
2023-04-19 23:46:22 +02:00
case EDIT_TOGGLE_WORD_WRAP : {
TextEdit : : LineWrappingMode wrap = code_editor - > get_text_editor ( ) - > get_line_wrapping_mode ( ) ;
code_editor - > get_text_editor ( ) - > set_line_wrapping_mode ( wrap = = TextEdit : : LINE_WRAPPING_BOUNDARY ? TextEdit : : LINE_WRAPPING_NONE : TextEdit : : LINE_WRAPPING_BOUNDARY ) ;
} break ;
2016-08-02 19:11:05 -03:00
case SEARCH_FIND : {
code_editor - > get_find_replace_bar ( ) - > popup_search ( ) ;
} break ;
case SEARCH_FIND_NEXT : {
code_editor - > get_find_replace_bar ( ) - > search_next ( ) ;
} break ;
case SEARCH_FIND_PREV : {
code_editor - > get_find_replace_bar ( ) - > search_prev ( ) ;
} break ;
case SEARCH_REPLACE : {
code_editor - > get_find_replace_bar ( ) - > popup_replace ( ) ;
} break ;
2018-02-12 02:36:15 +01:00
case SEARCH_IN_FILES : {
2022-12-05 19:05:02 +01:00
String selected_text = tx - > get_selected_text ( ) ;
2018-02-12 02:36:15 +01:00
// Yep, because it doesn't make sense to instance this dialog for every single script open...
2019-05-21 05:07:48 -03:00
// So this will be delegated to the ScriptEditor.
2021-07-17 18:22:52 -03:00
emit_signal ( SNAME ( " search_in_files_requested " ) , selected_text ) ;
2018-02-12 02:36:15 +01:00
} break ;
2020-02-09 10:10:58 +01:00
case REPLACE_IN_FILES : {
2022-12-05 19:05:02 +01:00
String selected_text = tx - > get_selected_text ( ) ;
2020-02-09 10:10:58 +01:00
2021-07-17 18:22:52 -03:00
emit_signal ( SNAME ( " replace_in_files_requested " ) , selected_text ) ;
2020-02-09 10:10:58 +01:00
} break ;
2016-08-02 19:11:05 -03:00
case SEARCH_LOCATE_FUNCTION : {
2018-10-02 12:07:44 +02:00
quick_open - > popup_dialog ( get_functions ( ) ) ;
2018-09-25 22:44:50 +02:00
quick_open - > set_title ( TTR ( " Go to Function " ) ) ;
2016-08-02 19:11:05 -03:00
} break ;
case SEARCH_GOTO_LINE : {
2024-02-22 15:32:12 -05:00
goto_line_popup - > popup_find_line ( code_editor ) ;
2016-08-02 19:11:05 -03:00
} break ;
2019-04-20 01:51:25 +02:00
case BOOKMARK_TOGGLE : {
code_editor - > toggle_bookmark ( ) ;
} break ;
case BOOKMARK_GOTO_NEXT : {
code_editor - > goto_next_bookmark ( ) ;
} break ;
case BOOKMARK_GOTO_PREV : {
code_editor - > goto_prev_bookmark ( ) ;
} break ;
case BOOKMARK_REMOVE_ALL : {
code_editor - > remove_all_bookmarks ( ) ;
} break ;
2016-08-02 19:11:05 -03:00
case DEBUG_TOGGLE_BREAKPOINT : {
2024-01-22 18:27:19 -05:00
Vector < int > sorted_carets = tx - > get_sorted_carets ( ) ;
2023-02-01 15:35:35 +02:00
int last_line = - 1 ;
2024-01-22 18:27:19 -05:00
for ( const int & c : sorted_carets ) {
int from = tx - > get_selection_from_line ( c ) ;
2023-02-01 15:35:35 +02:00
from + = from = = last_line ? 1 : 0 ;
2024-01-22 18:27:19 -05:00
int to = tx - > get_selection_to_line ( c ) ;
2023-02-01 15:35:35 +02:00
if ( to < from ) {
continue ;
}
// Check first if there's any lines with breakpoints in the selection.
bool selection_has_breakpoints = false ;
for ( int line = from ; line < = to ; line + + ) {
if ( tx - > is_line_breakpointed ( line ) ) {
selection_has_breakpoints = true ;
break ;
}
}
// Set breakpoint on caret or remove all bookmarks from the selection.
if ( ! selection_has_breakpoints ) {
if ( tx - > get_caret_line ( c ) ! = last_line ) {
tx - > set_line_as_breakpoint ( tx - > get_caret_line ( c ) , true ) ;
}
} else {
for ( int line = from ; line < = to ; line + + ) {
tx - > set_line_as_breakpoint ( line , false ) ;
}
}
last_line = to ;
2022-12-05 19:05:02 +01:00
}
2016-08-02 19:11:05 -03:00
} break ;
case DEBUG_REMOVE_ALL_BREAKPOINTS : {
2022-08-05 03:41:48 +02:00
PackedInt32Array bpoints = tx - > get_breakpointed_lines ( ) ;
2016-08-02 19:11:05 -03:00
2020-07-26 15:57:23 +01:00
for ( int i = 0 ; i < bpoints . size ( ) ; i + + ) {
int line = bpoints [ i ] ;
bool dobreak = ! tx - > is_line_breakpointed ( line ) ;
2017-11-15 23:00:27 -05:00
tx - > set_line_as_breakpoint ( line , dobreak ) ;
2020-02-07 02:52:05 +01:00
EditorDebuggerNode : : get_singleton ( ) - > set_breakpoint ( script - > get_path ( ) , line + 1 , dobreak ) ;
2016-08-02 19:11:05 -03:00
}
2019-04-05 14:06:16 +02:00
} break ;
2016-08-02 19:11:05 -03:00
case DEBUG_GOTO_NEXT_BREAKPOINT : {
2022-08-05 03:41:48 +02:00
PackedInt32Array bpoints = tx - > get_breakpointed_lines ( ) ;
2016-08-02 19:11:05 -03:00
if ( bpoints . size ( ) < = 0 ) {
return ;
}
2022-12-05 19:05:02 +01:00
int current_line = tx - > get_caret_line ( ) ;
int bpoint_idx = 0 ;
if ( current_line < ( int ) bpoints [ bpoints . size ( ) - 1 ] ) {
while ( bpoint_idx < bpoints . size ( ) & & bpoints [ bpoint_idx ] < = current_line ) {
bpoint_idx + + ;
2016-08-02 19:11:05 -03:00
}
}
2022-12-05 19:05:02 +01:00
code_editor - > goto_line_centered ( bpoints [ bpoint_idx ] ) ;
2016-08-02 19:11:05 -03:00
} break ;
case DEBUG_GOTO_PREV_BREAKPOINT : {
2022-08-05 03:41:48 +02:00
PackedInt32Array bpoints = tx - > get_breakpointed_lines ( ) ;
2016-08-02 19:11:05 -03:00
if ( bpoints . size ( ) < = 0 ) {
return ;
}
2022-12-05 19:05:02 +01:00
int current_line = tx - > get_caret_line ( ) ;
int bpoint_idx = bpoints . size ( ) - 1 ;
if ( current_line > ( int ) bpoints [ 0 ] ) {
while ( bpoint_idx > = 0 & & bpoints [ bpoint_idx ] > = current_line ) {
bpoint_idx - - ;
2016-08-02 19:11:05 -03:00
}
}
2022-12-05 19:05:02 +01:00
code_editor - > goto_line_centered ( bpoints [ bpoint_idx ] ) ;
2016-08-02 19:11:05 -03:00
} break ;
case HELP_CONTEXTUAL : {
2022-06-08 22:41:38 +01:00
String text = tx - > get_selected_text ( 0 ) ;
2021-12-09 03:42:46 -06:00
if ( text . is_empty ( ) ) {
2022-06-08 22:41:38 +01:00
text = tx - > get_word_under_caret ( 0 ) ;
2020-05-14 16:41:43 +02:00
}
2021-12-09 03:42:46 -06:00
if ( ! text . is_empty ( ) ) {
2021-07-17 18:22:52 -03:00
emit_signal ( SNAME ( " request_help " ) , text ) ;
2016-08-02 19:11:05 -03:00
}
} break ;
2018-01-02 02:10:49 -05:00
case LOOKUP_SYMBOL : {
2022-06-08 22:41:38 +01:00
String text = tx - > get_word_under_caret ( 0 ) ;
2021-12-09 03:42:46 -06:00
if ( text . is_empty ( ) ) {
2022-06-08 22:41:38 +01:00
text = tx - > get_selected_text ( 0 ) ;
2020-05-14 16:41:43 +02:00
}
2021-12-09 03:42:46 -06:00
if ( ! text . is_empty ( ) ) {
2022-06-08 22:41:38 +01:00
_lookup_symbol ( text , tx - > get_caret_line ( 0 ) , tx - > get_caret_column ( 0 ) ) ;
2018-01-02 02:10:49 -05:00
}
} break ;
2024-12-18 14:42:10 +01:00
default : {
if ( p_op > = EditorContextMenuPlugin : : BASE_ID ) {
EditorContextMenuPluginManager : : get_singleton ( ) - > activate_custom_option ( EditorContextMenuPlugin : : CONTEXT_SLOT_SCRIPT_EDITOR_CODE , p_op , tx ) ;
}
}
2016-08-02 19:11:05 -03:00
}
}
2019-02-04 20:17:44 +01:00
void ScriptTextEditor : : _edit_option_toggle_inline_comment ( ) {
2020-05-14 16:41:43 +02:00
if ( script . is_null ( ) ) {
2019-02-04 20:17:44 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2019-02-04 20:17:44 +01:00
String delimiter = " # " ;
List < String > comment_delimiters ;
script - > get_language ( ) - > get_comment_delimiters ( & comment_delimiters ) ;
2021-07-24 15:46:25 +02:00
for ( const String & script_delimiter : comment_delimiters ) {
2024-12-05 17:56:08 +01:00
if ( ! script_delimiter . contains_char ( ' ' ) ) {
2019-02-04 20:17:44 +01:00
delimiter = script_delimiter ;
break ;
}
}
code_editor - > toggle_inline_comment ( delimiter ) ;
}
2020-05-03 17:08:15 +01:00
void ScriptTextEditor : : add_syntax_highlighter ( Ref < EditorSyntaxHighlighter > p_highlighter ) {
2019-11-20 10:09:59 +01:00
ERR_FAIL_COND ( p_highlighter . is_null ( ) ) ;
2020-03-07 11:17:18 +00:00
highlighters [ p_highlighter - > _get_name ( ) ] = p_highlighter ;
highlighter_menu - > add_radio_check_item ( p_highlighter - > _get_name ( ) ) ;
2018-04-02 12:41:44 +01:00
}
2020-05-03 17:08:15 +01:00
void ScriptTextEditor : : set_syntax_highlighter ( Ref < EditorSyntaxHighlighter > p_highlighter ) {
2019-11-20 10:09:59 +01:00
ERR_FAIL_COND ( p_highlighter . is_null ( ) ) ;
2022-05-13 15:04:37 +02:00
HashMap < String , Ref < EditorSyntaxHighlighter > > : : Iterator el = highlighters . begin ( ) ;
while ( el ) {
int highlighter_index = highlighter_menu - > get_item_idx_from_text ( el - > key ) ;
highlighter_menu - > set_item_checked ( highlighter_index , el - > value = = p_highlighter ) ;
+ + el ;
2019-11-20 10:09:59 +01:00
}
2020-07-24 15:50:35 +01:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2020-05-03 17:08:15 +01:00
p_highlighter - > _set_edited_resource ( script ) ;
te - > set_syntax_highlighter ( p_highlighter ) ;
2018-04-02 12:41:44 +01:00
}
void ScriptTextEditor : : _change_syntax_highlighter ( int p_idx ) {
2018-06-01 21:32:19 +02:00
set_syntax_highlighter ( highlighters [ highlighter_menu - > get_item_text ( p_idx ) ] ) ;
2018-04-02 12:41:44 +01:00
}
2020-07-29 22:26:49 +01:00
void ScriptTextEditor : : _notification ( int p_what ) {
switch ( p_what ) {
2022-08-29 11:04:31 +02:00
case NOTIFICATION_THEME_CHANGED :
2022-03-08 13:20:09 -05:00
if ( ! editor_enabled ) {
break ;
}
2022-01-19 08:31:39 +03:00
if ( is_visible_in_tree ( ) ) {
_update_warnings ( ) ;
_update_errors ( ) ;
}
2022-08-29 11:04:31 +02:00
[[fallthrough]] ;
case NOTIFICATION_ENTER_TREE : {
code_editor - > get_text_editor ( ) - > set_gutter_width ( connection_gutter , code_editor - > get_text_editor ( ) - > get_line_height ( ) ) ;
2020-07-29 22:26:49 +01:00
} break ;
}
}
2016-08-02 19:11:05 -03:00
Control * ScriptTextEditor : : get_edit_menu ( ) {
return edit_hb ;
}
2017-06-30 21:30:17 -03:00
void ScriptTextEditor : : clear_edit_menu ( ) {
2022-07-19 19:09:10 +08:00
if ( editor_enabled ) {
memdelete ( edit_hb ) ;
}
2017-06-30 21:30:17 -03:00
}
2021-06-03 01:05:41 +02:00
void ScriptTextEditor : : set_find_replace_bar ( FindReplaceBar * p_bar ) {
code_editor - > set_find_replace_bar ( p_bar ) ;
}
2016-08-02 19:11:05 -03:00
void ScriptTextEditor : : reload ( bool p_soft ) {
2020-07-24 15:50:35 +01:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2018-05-28 16:52:28 +01:00
Ref < Script > scr = script ;
2020-05-14 16:41:43 +02:00
if ( scr . is_null ( ) ) {
2016-08-02 19:11:05 -03:00
return ;
2020-05-14 16:41:43 +02:00
}
2016-08-02 19:11:05 -03:00
scr - > set_source_code ( te - > get_text ( ) ) ;
2023-11-23 16:19:24 +01:00
bool soft = p_soft | | ClassDB : : is_parent_class ( scr - > get_instance_base_type ( ) , " EditorPlugin " ) ; // Always soft-reload editor plugins.
2016-08-02 19:11:05 -03:00
2017-03-05 16:44:50 +01:00
scr - > get_language ( ) - > reload_tool_script ( scr , soft ) ;
2016-08-02 19:11:05 -03:00
}
2022-08-05 03:41:48 +02:00
PackedInt32Array ScriptTextEditor : : get_breakpoints ( ) {
2020-07-26 15:57:23 +01:00
return code_editor - > get_text_editor ( ) - > get_breakpointed_lines ( ) ;
2016-08-02 19:11:05 -03:00
}
2021-07-20 12:24:56 +01:00
void ScriptTextEditor : : set_breakpoint ( int p_line , bool p_enabled ) {
code_editor - > get_text_editor ( ) - > set_line_as_breakpoint ( p_line , p_enabled ) ;
}
void ScriptTextEditor : : clear_breakpoints ( ) {
code_editor - > get_text_editor ( ) - > clear_breakpointed_lines ( ) ;
}
2021-11-07 17:26:15 +00:00
void ScriptTextEditor : : set_tooltip_request_func ( const Callable & p_toolip_callback ) {
Variant args [ 1 ] = { this } ;
const Variant * argp [ ] = { & args [ 0 ] } ;
2022-07-28 22:56:41 +02:00
code_editor - > get_text_editor ( ) - > set_tooltip_request_func ( p_toolip_callback . bindp ( argp , 1 ) ) ;
2016-08-02 19:11:05 -03:00
}
2016-08-06 19:00:54 -03:00
void ScriptTextEditor : : set_debugger_active ( bool p_active ) {
}
2021-05-14 02:50:25 +07:00
Control * ScriptTextEditor : : get_base_editor ( ) const {
return code_editor - > get_text_editor ( ) ;
}
2024-02-17 20:16:58 -06:00
CodeTextEditor * ScriptTextEditor : : get_code_editor ( ) const {
return code_editor ;
}
2017-03-05 16:44:50 +01:00
Variant ScriptTextEditor : : get_drag_data_fw ( const Point2 & p_point , Control * p_from ) {
2016-09-11 11:28:01 -03:00
return Variant ( ) ;
}
2017-03-05 16:44:50 +01:00
bool ScriptTextEditor : : can_drop_data_fw ( const Point2 & p_point , const Variant & p_data , Control * p_from ) const {
2016-09-11 11:28:01 -03:00
Dictionary d = p_data ;
2021-10-28 15:19:35 +02:00
if ( d . has ( " type " ) & &
( String ( d [ " type " ] ) = = " resource " | |
String ( d [ " type " ] ) = = " files " | |
String ( d [ " type " ] ) = = " nodes " | |
String ( d [ " type " ] ) = = " obj_property " | |
String ( d [ " type " ] ) = = " files_and_dirs " ) ) {
2017-03-05 16:44:50 +01:00
return true ;
2016-09-11 11:28:01 -03:00
}
return false ;
}
2017-03-05 16:44:50 +01:00
static Node * _find_script_node ( Node * p_edited_scene , Node * p_current_node , const Ref < Script > & script ) {
2022-09-21 15:46:30 +02:00
// Check scripts only for the nodes belonging to the edited scene.
if ( p_current_node = = p_edited_scene | | p_current_node - > get_owner ( ) = = p_edited_scene ) {
Ref < Script > scr = p_current_node - > get_script ( ) ;
if ( scr . is_valid ( ) & & scr = = script ) {
return p_current_node ;
}
2020-05-14 16:41:43 +02:00
}
2016-09-11 11:28:01 -03:00
2022-09-21 15:46:30 +02:00
// Traverse all children, even the ones not owned by the edited scene as they
// can still have child nodes added within the edited scene and thus owned by
// it (e.g. nodes added to subscene's root or to its editable children).
2017-03-05 16:44:50 +01:00
for ( int i = 0 ; i < p_current_node - > get_child_count ( ) ; i + + ) {
Node * n = _find_script_node ( p_edited_scene , p_current_node - > get_child ( i ) , script ) ;
2020-05-14 16:41:43 +02:00
if ( n ) {
2016-09-11 11:28:01 -03:00
return n ;
2020-05-14 16:41:43 +02:00
}
2016-09-11 11:28:01 -03:00
}
2020-04-02 01:20:12 +02:00
return nullptr ;
2016-09-11 11:28:01 -03:00
}
2022-10-11 19:25:31 -04:00
static String _quote_drop_data ( const String & str ) {
// This function prepares a string for being "dropped" into the script editor.
// The string can be a resource path, node path or property name.
const bool using_single_quotes = EDITOR_GET ( " text_editor/completion/use_single_quotes " ) ;
String escaped = str . c_escape ( ) ;
// If string is double quoted, there is no need to escape single quotes.
// We can revert the extra escaping added in c_escape().
if ( ! using_single_quotes ) {
escaped = escaped . replace ( " \\ ' " , " \' " ) ;
}
return escaped . quote ( using_single_quotes ? " ' " : " \" " ) ;
}
2021-08-13 21:24:02 +02:00
2023-09-15 17:00:18 -03:00
static String _get_dropped_resource_line ( const Ref < Resource > & p_resource , bool p_create_field ) {
const String & path = p_resource - > get_path ( ) ;
const bool is_script = ClassDB : : is_parent_class ( p_resource - > get_class ( ) , " Script " ) ;
if ( ! p_create_field ) {
return vformat ( " preload(%s) " , _quote_drop_data(path)) ;
}
String variable_name = p_resource - > get_name ( ) ;
if ( variable_name . is_empty ( ) ) {
variable_name = path . get_file ( ) . get_basename ( ) ;
}
if ( is_script ) {
2024-09-22 19:31:58 +08:00
variable_name = variable_name . to_pascal_case ( ) . validate_unicode_identifier ( ) ;
2023-09-15 17:00:18 -03:00
} else {
2024-09-22 19:31:58 +08:00
variable_name = variable_name . to_snake_case ( ) . to_upper ( ) . validate_unicode_identifier ( ) ;
2023-09-15 17:00:18 -03:00
}
return vformat ( " const %s = preload(%s) " , variable_name, _quote_drop_data(path)) ;
}
2022-10-11 19:25:31 -04:00
void ScriptTextEditor : : drop_data_fw ( const Point2 & p_point , const Variant & p_data , Control * p_from ) {
2016-09-11 11:28:01 -03:00
Dictionary d = p_data ;
2020-07-24 15:50:35 +01:00
CodeEdit * te = code_editor - > get_text_editor ( ) ;
2021-07-10 11:41:38 +01:00
Point2i pos = te - > get_line_column_at_pos ( p_point ) ;
2024-07-04 19:29:15 -04:00
int drop_at_line = pos . y ;
int drop_at_column = pos . x ;
int selection_index = te - > get_selection_at_line_column ( drop_at_line , drop_at_column ) ;
bool line_will_be_empty = false ;
if ( selection_index > = 0 ) {
// Dropped on a selection, it will be replaced.
drop_at_line = te - > get_selection_from_line ( selection_index ) ;
drop_at_column = te - > get_selection_from_column ( selection_index ) ;
line_will_be_empty = drop_at_column < = te - > get_first_non_whitespace_column ( drop_at_line ) & & te - > get_selection_to_column ( selection_index ) = = te - > get_line ( te - > get_selection_to_line ( selection_index ) ) . length ( ) ;
}
String text_to_drop ;
2018-02-03 21:33:32 -05:00
2023-09-15 17:00:18 -03:00
const bool drop_modifier_pressed = Input : : get_singleton ( ) - > is_key_pressed ( Key : : CMD_OR_CTRL ) ;
2024-07-04 19:29:15 -04:00
const String & line = te - > get_line ( drop_at_line ) ;
const bool is_empty_line = line_will_be_empty | | line . is_empty ( ) | | te - > get_first_non_whitespace_column ( drop_at_line ) = = line . length ( ) ;
2023-09-15 17:00:18 -03:00
2024-08-31 11:35:59 +02:00
const String type = d . get ( " type " , " " ) ;
if ( type = = " resource " ) {
2023-09-15 17:00:18 -03:00
Ref < Resource > resource = d [ " resource " ] ;
if ( resource . is_null ( ) ) {
2016-09-11 11:28:01 -03:00
return ;
}
2023-09-15 17:00:18 -03:00
const String & path = resource - > get_path ( ) ;
if ( path . is_empty ( ) | | path . ends_with ( " :: " ) ) {
String warning = TTR ( " The resource does not have a valid path because it has not been saved. \n Please save the scene or resource that contains this resource and try again. " ) ;
EditorToaster : : get_singleton ( ) - > popup_str ( warning , EditorToaster : : SEVERITY_ERROR ) ;
2016-09-11 11:28:01 -03:00
return ;
}
2023-09-15 17:00:18 -03:00
if ( drop_modifier_pressed ) {
if ( resource - > is_built_in ( ) ) {
String warning = TTR ( " Preloading internal resources is not supported. " ) ;
EditorToaster : : get_singleton ( ) - > popup_str ( warning , EditorToaster : : SEVERITY_ERROR ) ;
} else {
text_to_drop = _get_dropped_resource_line ( resource , is_empty_line ) ;
}
} else {
text_to_drop = _quote_drop_data ( path ) ;
}
2016-09-11 11:28:01 -03:00
}
2024-08-31 11:35:59 +02:00
if ( type = = " files " | | type = = " files_and_dirs " ) {
const PackedStringArray files = d [ " files " ] ;
PackedStringArray parts ;
2021-05-27 00:29:36 +02:00
2024-08-31 11:35:59 +02:00
for ( const String & path : files ) {
2023-09-15 17:00:18 -03:00
if ( drop_modifier_pressed & & ResourceLoader : : exists ( path ) ) {
Ref < Resource > resource = ResourceLoader : : load ( path ) ;
2024-04-14 15:26:53 +02:00
if ( resource . is_null ( ) ) {
// Resource exists, but failed to load. We need only path and name, so we can use a dummy Resource instead.
resource . instantiate ( ) ;
resource - > set_path_cache ( path ) ;
}
2024-08-31 11:35:59 +02:00
parts . append ( _get_dropped_resource_line ( resource , is_empty_line ) ) ;
2021-05-27 00:29:36 +02:00
} else {
2024-08-31 11:35:59 +02:00
parts . append ( _quote_drop_data ( path ) ) ;
2020-05-14 16:41:43 +02:00
}
2016-09-11 11:28:01 -03:00
}
2024-08-31 11:35:59 +02:00
text_to_drop = String ( is_empty_line ? " \n " : " , " ) . join ( parts ) ;
2016-09-11 11:28:01 -03:00
}
2024-08-31 11:35:59 +02:00
if ( type = = " nodes " ) {
2022-09-21 02:48:01 +02:00
Node * scene_root = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( ! scene_root ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Can't drop nodes without an open scene. " ) ) ;
return ;
}
2016-09-11 11:28:01 -03:00
2023-07-09 23:15:54 +02:00
if ( ! ClassDB : : is_parent_class ( script - > get_instance_base_type ( ) , " Node " ) ) {
EditorToaster : : get_singleton ( ) - > popup_str ( vformat ( TTR ( " Can't drop nodes because script '%s' does not inherit Node. " ) , get_name ( ) ) , EditorToaster : : SEVERITY_WARNING ) ;
return ;
}
2022-09-21 02:48:01 +02:00
Node * sn = _find_script_node ( scene_root , scene_root , script ) ;
2016-09-11 11:28:01 -03:00
if ( ! sn ) {
2023-07-09 23:15:54 +02:00
sn = scene_root ;
2016-09-11 11:28:01 -03:00
}
Array nodes = d [ " nodes " ] ;
2023-09-15 17:00:18 -03:00
if ( drop_modifier_pressed ) {
const bool use_type = EDITOR_GET ( " text_editor/completion/add_type_hints " ) ;
2022-05-02 21:04:17 +08:00
for ( int i = 0 ; i < nodes . size ( ) ; i + + ) {
NodePath np = nodes [ i ] ;
Node * node = get_node ( np ) ;
if ( ! node ) {
continue ;
}
2022-05-26 12:56:39 -03:00
bool is_unique = false ;
2022-05-18 12:19:56 +02:00
String path ;
if ( node - > is_unique_name_in_owner ( ) ) {
2022-05-26 12:56:39 -03:00
path = node - > get_name ( ) ;
is_unique = true ;
2022-05-18 12:19:56 +02:00
} else {
path = sn - > get_path_to ( node ) ;
}
2022-05-02 21:04:17 +08:00
for ( const String & segment : path . split ( " / " ) ) {
2024-09-22 19:31:58 +08:00
if ( ! segment . is_valid_unicode_identifier ( ) ) {
2022-10-11 19:25:31 -04:00
path = _quote_drop_data ( path ) ;
2022-05-02 21:04:17 +08:00
break ;
}
}
2024-09-22 19:31:58 +08:00
String variable_name = String ( node - > get_name ( ) ) . to_snake_case ( ) . validate_unicode_identifier ( ) ;
2022-05-02 21:04:17 +08:00
if ( use_type ) {
2023-07-08 06:23:01 -05:00
StringName class_name = node - > get_class_name ( ) ;
Ref < Script > node_script = node - > get_script ( ) ;
if ( node_script . is_valid ( ) ) {
StringName global_node_script_name = node_script - > get_global_name ( ) ;
if ( global_node_script_name ! = StringName ( ) ) {
class_name = global_node_script_name ;
}
}
text_to_drop + = vformat ( " @onready var %s: %s = %c%s \n " , variable_name , class_name , is_unique ? ' % ' : ' $ ' , path ) ;
2022-05-02 21:04:17 +08:00
} else {
2023-07-08 06:23:01 -05:00
text_to_drop + = vformat ( " @onready var %s = %c%s \n " , variable_name , is_unique ? ' % ' : ' $ ' , path ) ;
2022-05-02 21:04:17 +08:00
}
2016-09-11 11:28:01 -03:00
}
2022-05-02 21:04:17 +08:00
} else {
for ( int i = 0 ; i < nodes . size ( ) ; i + + ) {
if ( i > 0 ) {
2022-05-17 14:56:16 +02:00
text_to_drop + = " , " ;
2022-05-02 21:04:17 +08:00
}
2016-09-11 11:28:01 -03:00
2022-05-02 21:04:17 +08:00
NodePath np = nodes [ i ] ;
Node * node = get_node ( np ) ;
if ( ! node ) {
continue ;
}
2022-05-26 12:56:39 -03:00
bool is_unique = false ;
2022-05-18 12:19:56 +02:00
String path ;
if ( node - > is_unique_name_in_owner ( ) ) {
2022-05-26 12:56:39 -03:00
path = node - > get_name ( ) ;
is_unique = true ;
2022-05-18 12:19:56 +02:00
} else {
path = sn - > get_path_to ( node ) ;
}
2022-05-26 12:56:39 -03:00
2022-05-17 14:56:16 +02:00
for ( const String & segment : path . split ( " / " ) ) {
2024-08-23 14:30:51 +08:00
if ( ! segment . is_valid_ascii_identifier ( ) ) {
2022-10-11 19:25:31 -04:00
path = _quote_drop_data ( path ) ;
2022-05-17 14:56:16 +02:00
break ;
}
}
2022-05-26 12:56:39 -03:00
text_to_drop + = ( is_unique ? " % " : " $ " ) + path ;
2022-05-02 21:04:17 +08:00
}
2016-09-11 11:28:01 -03:00
}
}
2021-08-13 23:01:40 +08:00
2024-08-31 11:35:59 +02:00
if ( type = = " obj_property " ) {
2022-10-06 22:08:01 +02:00
bool add_literal = EDITOR_GET ( " text_editor/completion/add_node_path_literals " ) ;
2024-07-04 19:29:15 -04:00
text_to_drop = add_literal ? " ^ " : " " ;
2022-10-11 19:25:31 -04:00
// It is unclear whether properties may contain single or double quotes.
// Assume here that double-quotes may not exist. We are escaping single-quotes if necessary.
2022-10-06 22:08:01 +02:00
text_to_drop + = _quote_drop_data ( String ( d [ " property " ] ) ) ;
2024-07-04 19:29:15 -04:00
}
if ( text_to_drop . is_empty ( ) ) {
return ;
}
2021-08-13 23:01:40 +08:00
2024-07-04 19:29:15 -04:00
// Remove drag caret before any actions so it is not included in undo.
te - > remove_drag_caret ( ) ;
te - > begin_complex_operation ( ) ;
if ( selection_index > = 0 ) {
te - > delete_selection ( selection_index ) ;
2021-08-13 23:01:40 +08:00
}
2024-07-04 19:29:15 -04:00
te - > remove_secondary_carets ( ) ;
te - > deselect ( ) ;
te - > set_caret_line ( drop_at_line ) ;
te - > set_caret_column ( drop_at_column ) ;
te - > insert_text_at_caret ( text_to_drop ) ;
te - > end_complex_operation ( ) ;
te - > grab_focus ( ) ;
2016-09-11 11:28:01 -03:00
}
2017-05-20 12:38:03 -03:00
void ScriptTextEditor : : _text_edit_gui_input ( const Ref < InputEvent > & ev ) {
Ref < InputEventMouseButton > mb = ev ;
2019-06-22 20:22:52 +02:00
Ref < InputEventKey > k = ev ;
Point2 local_pos ;
bool create_menu = false ;
2017-05-20 12:38:03 -03:00
2020-07-24 15:50:35 +01:00
CodeEdit * tx = code_editor - > get_text_editor ( ) ;
2021-08-13 16:31:57 -05:00
if ( mb . is_valid ( ) & & mb - > get_button_index ( ) = = MouseButton : : RIGHT & & mb - > is_pressed ( ) ) {
2019-06-22 20:22:52 +02:00
local_pos = mb - > get_global_position ( ) - tx - > get_global_position ( ) ;
create_menu = true ;
2021-07-08 18:35:56 +01:00
} else if ( k . is_valid ( ) & & k - > is_action ( " ui_menu " , true ) ) {
2022-06-08 22:41:38 +01:00
tx - > adjust_viewport_to_caret ( 0 ) ;
local_pos = tx - > get_caret_draw_pos ( 0 ) ;
2019-06-22 20:22:52 +02:00
create_menu = true ;
}
2017-12-20 02:36:47 +01:00
2019-06-22 20:22:52 +02:00
if ( create_menu ) {
2024-01-22 12:34:12 -05:00
tx - > apply_ime ( ) ;
2021-07-10 11:41:38 +01:00
Point2i pos = tx - > get_line_column_at_pos ( local_pos ) ;
2024-01-22 18:26:16 -05:00
int mouse_line = pos . y ;
int mouse_column = pos . x ;
2017-12-20 02:36:47 +01:00
2022-10-18 16:43:37 +02:00
tx - > set_move_caret_on_right_click_enabled ( EDITOR_GET ( " text_editor/behavior/navigation/move_caret_on_right_click " ) ) ;
2024-01-22 18:26:16 -05:00
int selection_clicked = - 1 ;
2021-07-08 18:35:56 +01:00
if ( tx - > is_move_caret_on_right_click_enabled ( ) ) {
2024-01-22 18:26:16 -05:00
selection_clicked = tx - > get_selection_at_line_column ( mouse_line , mouse_column , true ) ;
if ( selection_clicked < 0 ) {
2023-04-30 18:28:21 +02:00
tx - > deselect ( ) ;
tx - > remove_secondary_carets ( ) ;
2024-01-22 18:26:16 -05:00
selection_clicked = 0 ;
tx - > set_caret_line ( mouse_line , false , false , - 1 ) ;
tx - > set_caret_column ( mouse_column ) ;
2019-06-22 20:22:52 +02:00
}
}
2017-12-20 02:36:47 +01:00
2019-06-22 20:22:52 +02:00
String word_at_pos = tx - > get_word_at_pos ( local_pos ) ;
2021-12-09 03:42:46 -06:00
if ( word_at_pos . is_empty ( ) ) {
2024-01-22 18:26:16 -05:00
word_at_pos = tx - > get_word_under_caret ( selection_clicked ) ;
2020-05-14 16:41:43 +02:00
}
2021-12-09 03:42:46 -06:00
if ( word_at_pos . is_empty ( ) ) {
2024-01-22 18:26:16 -05:00
word_at_pos = tx - > get_selected_text ( selection_clicked ) ;
2020-05-14 16:41:43 +02:00
}
2018-01-02 02:10:49 -05:00
2019-06-22 20:22:52 +02:00
bool has_color = ( word_at_pos = = " Color " ) ;
2024-01-22 18:26:16 -05:00
bool foldable = tx - > can_fold_line ( mouse_line ) | | tx - > is_line_folded ( mouse_line ) ;
2019-06-22 20:22:52 +02:00
bool open_docs = false ;
bool goto_definition = false ;
2018-01-02 02:10:49 -05:00
2023-02-13 07:33:10 +03:00
if ( ScriptServer : : is_global_class ( word_at_pos ) | | word_at_pos . is_resource_file ( ) ) {
2019-06-22 20:22:52 +02:00
open_docs = true ;
} else {
Node * base = get_tree ( ) - > get_edited_scene_root ( ) ;
if ( base ) {
base = _find_node_for_script ( base , base , script ) ;
}
ScriptLanguage : : LookupResult result ;
2022-12-05 19:05:02 +01:00
if ( script - > get_language ( ) - > lookup_code ( tx - > get_text_for_symbol_lookup ( ) , word_at_pos , script - > get_path ( ) , base , result ) = = OK ) {
2018-01-02 02:10:49 -05:00
open_docs = true ;
}
2019-06-22 20:22:52 +02:00
}
2018-01-02 02:10:49 -05:00
2019-06-22 20:22:52 +02:00
if ( has_color ) {
2024-01-22 18:26:16 -05:00
String line = tx - > get_line ( mouse_line ) ;
color_position . x = mouse_line ;
color_position . y = mouse_column ;
2019-06-22 20:22:52 +02:00
2022-08-04 20:11:07 +09:00
int begin = - 1 ;
int end = - 1 ;
enum EXPRESSION_PATTERNS {
NOT_PARSED ,
RGBA_PARAMETER , // Color(float,float,float) or Color(float,float,float,float)
COLOR_NAME , // Color.COLOR_NAME
} expression_pattern = NOT_PARSED ;
2024-01-22 18:26:16 -05:00
for ( int i = mouse_column ; i < line . length ( ) ; i + + ) {
2019-06-22 20:22:52 +02:00
if ( line [ i ] = = ' ( ' ) {
2022-08-04 20:11:07 +09:00
if ( expression_pattern = = NOT_PARSED ) {
begin = i ;
expression_pattern = RGBA_PARAMETER ;
} else {
// Method call or '(' appearing twice.
expression_pattern = NOT_PARSED ;
break ;
}
} else if ( expression_pattern = = RGBA_PARAMETER & & line [ i ] = = ' ) ' & & end < 0 ) {
end = i + 1 ;
break ;
} else if ( expression_pattern = = NOT_PARSED & & line [ i ] = = ' . ' ) {
2019-06-22 20:22:52 +02:00
begin = i ;
2022-08-04 20:11:07 +09:00
expression_pattern = COLOR_NAME ;
} else if ( expression_pattern = = COLOR_NAME & & end < 0 & & ( line [ i ] = = ' ' | | line [ i ] = = ' \t ' ) ) {
// Including '.' and spaces.
2019-06-22 20:22:52 +02:00
continue ;
2022-08-04 20:11:07 +09:00
} else if ( expression_pattern = = COLOR_NAME & & ! ( line [ i ] = = ' _ ' | | ( ' A ' < = line [ i ] & & line [ i ] < = ' Z ' ) ) ) {
end = i ;
2019-06-22 20:22:52 +02:00
break ;
2018-01-02 02:10:49 -05:00
}
2019-06-22 20:22:52 +02:00
}
2022-08-04 20:11:07 +09:00
switch ( expression_pattern ) {
case RGBA_PARAMETER : {
color_args = line . substr ( begin , end - begin ) ;
String stripped = color_args . replace ( " " , " " ) . replace ( " \t " , " " ) . replace ( " ( " , " " ) . replace ( " ) " , " " ) ;
PackedFloat64Array color = stripped . split_floats ( " , " ) ;
if ( color . size ( ) > 2 ) {
float alpha = color . size ( ) > 3 ? color [ 3 ] : 1.0f ;
color_picker - > set_pick_color ( Color ( color [ 0 ] , color [ 1 ] , color [ 2 ] , alpha ) ) ;
}
} break ;
case COLOR_NAME : {
if ( end < 0 ) {
end = line . length ( ) ;
}
color_args = line . substr ( begin , end - begin ) ;
const String color_name = color_args . replace ( " " , " " ) . replace ( " \t " , " " ) . replace ( " . " , " " ) ;
const int color_index = Color : : find_named_color ( color_name ) ;
if ( 0 < = color_index ) {
const Color color_constant = Color : : get_named_color ( color_index ) ;
color_picker - > set_pick_color ( color_constant ) ;
} else {
has_color = false ;
}
} break ;
default :
has_color = false ;
break ;
}
if ( has_color ) {
2021-08-31 17:43:35 +02:00
color_panel - > set_position ( get_screen_position ( ) + local_pos ) ;
2016-09-29 09:12:45 +02:00
}
}
2021-07-09 12:42:55 +01:00
_make_context_menu ( tx - > has_selection ( ) , has_color , foldable , open_docs , goto_definition , local_pos ) ;
2016-09-29 09:12:45 +02:00
}
}
2017-03-05 16:44:50 +01:00
void ScriptTextEditor : : _color_changed ( const Color & p_color ) {
2016-09-29 09:12:45 +02:00
String new_args ;
2024-03-18 12:51:32 +08:00
const int decimals = 3 ;
2016-09-29 09:12:45 +02:00
if ( p_color . a = = 1.0f ) {
2024-03-18 12:51:32 +08:00
new_args = String ( " ( " + String : : num ( p_color . r , decimals ) + " , " + String : : num ( p_color . g , decimals ) + " , " + String : : num ( p_color . b , decimals ) + " ) " ) ;
2016-09-29 09:12:45 +02:00
} else {
2024-03-18 12:51:32 +08:00
new_args = String ( " ( " + String : : num ( p_color . r , decimals ) + " , " + String : : num ( p_color . g , decimals ) + " , " + String : : num ( p_color . b , decimals ) + " , " + String : : num ( p_color . a , decimals ) + " ) " ) ;
2016-09-29 09:12:45 +02:00
}
2020-07-24 15:50:35 +01:00
String line = code_editor - > get_text_editor ( ) - > get_line ( color_position . x ) ;
2021-11-11 09:08:08 +01:00
String line_with_replaced_args = line . replace ( color_args , new_args ) ;
2019-08-02 22:32:33 +02:00
2016-09-29 09:12:45 +02:00
color_args = new_args ;
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > begin_complex_operation ( ) ;
code_editor - > get_text_editor ( ) - > set_line ( color_position . x , line_with_replaced_args ) ;
code_editor - > get_text_editor ( ) - > end_complex_operation ( ) ;
2016-09-29 09:12:45 +02:00
}
2021-08-17 11:41:46 +08:00
void ScriptTextEditor : : _prepare_edit_menu ( ) {
const CodeEdit * tx = code_editor - > get_text_editor ( ) ;
PopupMenu * popup = edit_menu - > get_popup ( ) ;
popup - > set_item_disabled ( popup - > get_item_index ( EDIT_UNDO ) , ! tx - > has_undo ( ) ) ;
popup - > set_item_disabled ( popup - > get_item_index ( EDIT_REDO ) , ! tx - > has_redo ( ) ) ;
}
2019-06-22 20:22:52 +02:00
void ScriptTextEditor : : _make_context_menu ( bool p_selection , bool p_color , bool p_foldable , bool p_open_docs , bool p_goto_definition , Vector2 p_pos ) {
2016-09-29 09:12:45 +02:00
context_menu - > clear ( ) ;
2020-12-07 21:32:00 +10:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " ui_undo " ) , EDIT_UNDO ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " ui_redo " ) , EDIT_REDO ) ;
2016-09-29 09:12:45 +02:00
2019-08-12 22:23:00 +02:00
context_menu - > add_separator ( ) ;
2020-12-07 21:32:00 +10:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " ui_cut " ) , EDIT_CUT ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " ui_copy " ) , EDIT_COPY ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " ui_paste " ) , EDIT_PASTE ) ;
2019-08-12 22:23:00 +02:00
2016-09-29 09:12:45 +02:00
context_menu - > add_separator ( ) ;
2020-12-07 21:32:00 +10:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " ui_text_select_all " ) , EDIT_SELECT_ALL ) ;
2019-08-12 22:23:00 +02:00
2017-12-14 10:10:53 +01:00
context_menu - > add_separator ( ) ;
2022-09-29 04:12:52 +02:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/indent " ) , EDIT_INDENT ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/unindent " ) , EDIT_UNINDENT ) ;
2017-12-14 10:10:53 +01:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_comment " ) , EDIT_TOGGLE_COMMENT ) ;
2019-04-20 01:51:25 +02:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_bookmark " ) , BOOKMARK_TOGGLE ) ;
2016-09-29 09:12:45 +02:00
if ( p_selection ) {
context_menu - > add_separator ( ) ;
2017-12-07 16:09:02 +08:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/convert_to_uppercase " ) , EDIT_TO_UPPERCASE ) ;
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/convert_to_lowercase " ) , EDIT_TO_LOWERCASE ) ;
2019-08-07 16:31:33 +02:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/evaluate_selection " ) , EDIT_EVALUATE ) ;
2023-03-12 17:48:37 +01:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/create_code_region " ) , EDIT_CREATE_CODE_REGION ) ;
2016-09-29 09:12:45 +02:00
}
2020-05-14 16:41:43 +02:00
if ( p_foldable ) {
2017-12-08 19:17:10 +01:00
context_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_fold_line " ) , EDIT_TOGGLE_FOLD_LINE ) ;
2020-05-14 16:41:43 +02:00
}
2017-12-08 19:17:10 +01:00
2018-01-02 02:10:49 -05:00
if ( p_color | | p_open_docs | | p_goto_definition ) {
2016-09-29 09:12:45 +02:00
context_menu - > add_separator ( ) ;
2020-05-14 16:41:43 +02:00
if ( p_open_docs ) {
2018-01-02 02:10:49 -05:00
context_menu - > add_item ( TTR ( " Lookup Symbol " ) , LOOKUP_SYMBOL ) ;
2020-05-14 16:41:43 +02:00
}
if ( p_color ) {
2018-01-02 02:10:49 -05:00
context_menu - > add_item ( TTR ( " Pick Color " ) , EDIT_PICK_COLOR ) ;
2020-05-14 16:41:43 +02:00
}
2016-09-29 09:12:45 +02:00
}
2018-01-02 02:10:49 -05:00
2024-12-18 14:42:10 +01:00
const PackedStringArray paths = { code_editor - > get_text_editor ( ) - > get_path ( ) } ;
EditorContextMenuPluginManager : : get_singleton ( ) - > add_options_from_plugins ( context_menu , EditorContextMenuPlugin : : CONTEXT_SLOT_SCRIPT_EDITOR_CODE , paths ) ;
2021-08-17 11:41:46 +08:00
const CodeEdit * tx = code_editor - > get_text_editor ( ) ;
context_menu - > set_item_disabled ( context_menu - > get_item_index ( EDIT_UNDO ) , ! tx - > has_undo ( ) ) ;
context_menu - > set_item_disabled ( context_menu - > get_item_index ( EDIT_REDO ) , ! tx - > has_redo ( ) ) ;
2021-08-31 17:43:35 +02:00
context_menu - > set_position ( get_screen_position ( ) + p_pos ) ;
2021-11-20 11:04:57 +03:00
context_menu - > reset_size ( ) ;
2016-09-29 09:12:45 +02:00
context_menu - > popup ( ) ;
}
2016-09-11 11:28:01 -03:00
2019-11-20 10:09:59 +01:00
void ScriptTextEditor : : _enable_code_editor ( ) {
ERR_FAIL_COND ( code_editor - > get_parent ( ) ) ;
2017-10-02 18:33:42 -05:00
2018-07-01 13:17:40 -03:00
VSplitContainer * editor_box = memnew ( VSplitContainer ) ;
add_child ( editor_box ) ;
2022-03-18 19:02:57 -05:00
editor_box - > set_anchors_and_offsets_preset ( Control : : PRESET_FULL_RECT ) ;
2018-07-01 13:17:40 -03:00
editor_box - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
editor_box - > add_child ( code_editor ) ;
2021-05-18 13:09:19 +10:00
code_editor - > connect ( " show_errors_panel " , callable_mp ( this , & ScriptTextEditor : : _show_errors_panel ) ) ;
2019-11-20 10:09:59 +01:00
code_editor - > connect ( " show_warnings_panel " , callable_mp ( this , & ScriptTextEditor : : _show_warnings_panel ) ) ;
2020-02-21 18:28:45 +01:00
code_editor - > connect ( " validate_script " , callable_mp ( this , & ScriptTextEditor : : _validate_script ) ) ;
code_editor - > connect ( " load_theme_settings " , callable_mp ( this , & ScriptTextEditor : : _load_theme_settings ) ) ;
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > connect ( " symbol_lookup " , callable_mp ( this , & ScriptTextEditor : : _lookup_symbol ) ) ;
2024-12-14 21:25:05 +03:00
code_editor - > get_text_editor ( ) - > connect ( " symbol_hovered " , callable_mp ( this , & ScriptTextEditor : : _show_symbol_tooltip ) ) ;
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > connect ( " symbol_validate " , callable_mp ( this , & ScriptTextEditor : : _validate_symbol ) ) ;
2020-07-29 22:26:49 +01:00
code_editor - > get_text_editor ( ) - > connect ( " gutter_added " , callable_mp ( this , & ScriptTextEditor : : _update_gutter_indexes ) ) ;
code_editor - > get_text_editor ( ) - > connect ( " gutter_removed " , callable_mp ( this , & ScriptTextEditor : : _update_gutter_indexes ) ) ;
code_editor - > get_text_editor ( ) - > connect ( " gutter_clicked " , callable_mp ( this , & ScriptTextEditor : : _gutter_clicked ) ) ;
2024-05-13 16:56:03 +02:00
code_editor - > get_text_editor ( ) - > connect ( SceneStringName ( gui_input ) , callable_mp ( this , & ScriptTextEditor : : _text_edit_gui_input ) ) ;
2019-12-17 11:16:17 +03:00
code_editor - > show_toggle_scripts_button ( ) ;
2020-07-30 11:41:05 +01:00
_update_gutter_indexes ( ) ;
2018-07-01 13:17:40 -03:00
editor_box - > add_child ( warnings_panel ) ;
2020-05-24 14:30:09 +02:00
warnings_panel - > add_theme_font_override (
2023-09-13 13:14:07 +02:00
" normal_font " , EditorNode : : get_singleton ( ) - > get_editor_theme ( ) - > get_font ( SNAME ( " main " ) , EditorStringName ( EditorFonts ) ) ) ;
2020-09-03 14:22:16 +03:00
warnings_panel - > add_theme_font_size_override (
2023-09-13 13:14:07 +02:00
" normal_font_size " , EditorNode : : get_singleton ( ) - > get_editor_theme ( ) - > get_font_size ( SNAME ( " main_size " ) , EditorStringName ( EditorFonts ) ) ) ;
2020-02-21 18:28:45 +01:00
warnings_panel - > connect ( " meta_clicked " , callable_mp ( this , & ScriptTextEditor : : _warning_clicked ) ) ;
2016-09-12 10:52:29 -03:00
2021-05-18 13:09:19 +10:00
editor_box - > add_child ( errors_panel ) ;
errors_panel - > add_theme_font_override (
2023-09-13 13:14:07 +02:00
" normal_font " , EditorNode : : get_singleton ( ) - > get_editor_theme ( ) - > get_font ( SNAME ( " main " ) , EditorStringName ( EditorFonts ) ) ) ;
2021-05-18 13:09:19 +10:00
errors_panel - > add_theme_font_size_override (
2023-09-13 13:14:07 +02:00
" normal_font_size " , EditorNode : : get_singleton ( ) - > get_editor_theme ( ) - > get_font_size ( SNAME ( " main_size " ) , EditorStringName ( EditorFonts ) ) ) ;
2021-05-18 13:09:19 +10:00
errors_panel - > connect ( " meta_clicked " , callable_mp ( this , & ScriptTextEditor : : _error_clicked ) ) ;
2016-09-29 09:12:45 +02:00
add_child ( context_menu ) ;
2024-05-14 14:13:31 +02:00
context_menu - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2016-09-29 09:12:45 +02:00
add_child ( color_panel ) ;
2019-11-20 10:09:59 +01:00
2016-09-29 09:12:45 +02:00
color_picker = memnew ( ColorPicker ) ;
2019-08-02 22:32:33 +02:00
color_picker - > set_deferred_mode ( true ) ;
2020-02-21 18:28:45 +01:00
color_picker - > connect ( " color_changed " , callable_mp ( this , & ScriptTextEditor : : _color_changed ) ) ;
2022-07-28 22:56:41 +02:00
color_panel - > connect ( " about_to_popup " , callable_mp ( EditorNode : : get_singleton ( ) , & EditorNode : : setup_color_picker ) . bind ( color_picker ) ) ;
2016-09-12 10:52:29 -03:00
2019-11-20 10:09:59 +01:00
color_panel - > add_child ( color_picker ) ;
quick_open = memnew ( ScriptEditorQuickOpen ) ;
quick_open - > connect ( " goto_line " , callable_mp ( this , & ScriptTextEditor : : _goto_line ) ) ;
add_child ( quick_open ) ;
2016-08-02 19:11:05 -03:00
2024-02-22 15:32:12 -05:00
goto_line_popup = memnew ( GotoLinePopup ) ;
add_child ( goto_line_popup ) ;
2019-11-20 10:09:59 +01:00
add_child ( connection_info_dialog ) ;
edit_hb - > add_child ( edit_menu ) ;
2021-08-17 11:41:46 +08:00
edit_menu - > connect ( " about_to_popup " , callable_mp ( this , & ScriptTextEditor : : _prepare_edit_menu ) ) ;
2020-12-07 21:32:00 +10:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_undo " ) , EDIT_UNDO ) ;
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_redo " ) , EDIT_REDO ) ;
2016-08-02 19:11:05 -03:00
edit_menu - > get_popup ( ) - > add_separator ( ) ;
2020-12-07 21:32:00 +10:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_cut " ) , EDIT_CUT ) ;
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_copy " ) , EDIT_COPY ) ;
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_paste " ) , EDIT_PASTE ) ;
2016-08-02 19:11:05 -03:00
edit_menu - > get_popup ( ) - > add_separator ( ) ;
2020-12-07 21:32:00 +10:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_text_select_all " ) , EDIT_SELECT_ALL ) ;
2022-08-11 17:15:15 +02:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/duplicate_selection " ) , EDIT_DUPLICATE_SELECTION ) ;
2022-09-28 17:09:45 +02:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/duplicate_lines " ) , EDIT_DUPLICATE_LINES ) ;
2022-08-11 17:15:15 +02:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/evaluate_selection " ) , EDIT_EVALUATE ) ;
2023-04-19 23:46:22 +02:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_word_wrap " ) , EDIT_TOGGLE_WORD_WRAP ) ;
2016-08-02 19:11:05 -03:00
edit_menu - > get_popup ( ) - > add_separator ( ) ;
2022-08-11 17:15:15 +02:00
{
PopupMenu * sub_menu = memnew ( PopupMenu ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/move_up " ) , EDIT_MOVE_LINE_UP ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/move_down " ) , EDIT_MOVE_LINE_DOWN ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/indent " ) , EDIT_INDENT ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/unindent " ) , EDIT_UNINDENT ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/delete_line " ) , EDIT_DELETE_LINE ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_comment " ) , EDIT_TOGGLE_COMMENT ) ;
2024-05-14 14:13:31 +02:00
sub_menu - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2023-11-28 17:33:04 +01:00
edit_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Line " ) , sub_menu ) ;
2022-08-11 17:15:15 +02:00
}
{
PopupMenu * sub_menu = memnew ( PopupMenu ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/toggle_fold_line " ) , EDIT_TOGGLE_FOLD_LINE ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/fold_all_lines " ) , EDIT_FOLD_ALL_LINES ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/unfold_all_lines " ) , EDIT_UNFOLD_ALL_LINES ) ;
2023-03-12 17:48:37 +01:00
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/create_code_region " ) , EDIT_CREATE_CODE_REGION ) ;
2024-05-14 14:13:31 +02:00
sub_menu - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2023-11-28 17:33:04 +01:00
edit_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Folding " ) , sub_menu ) ;
2022-08-11 17:15:15 +02:00
}
2016-08-02 19:11:05 -03:00
edit_menu - > get_popup ( ) - > add_separator ( ) ;
2020-12-07 21:32:00 +10:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " ui_text_completion_query " ) , EDIT_COMPLETE ) ;
2016-08-02 19:11:05 -03:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/trim_trailing_whitespace " ) , EDIT_TRIM_TRAILING_WHITESAPCE ) ;
2024-01-12 00:32:13 -05:00
edit_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/trim_final_newlines " ) , EDIT_TRIM_FINAL_NEWLINES ) ;
2022-08-11 17:15:15 +02:00
{
PopupMenu * sub_menu = memnew ( PopupMenu ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/convert_indent_to_spaces " ) , EDIT_CONVERT_INDENT_TO_SPACES ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/convert_indent_to_tabs " ) , EDIT_CONVERT_INDENT_TO_TABS ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/auto_indent " ) , EDIT_AUTO_INDENT ) ;
2024-05-14 14:13:31 +02:00
sub_menu - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2023-11-28 17:33:04 +01:00
edit_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Indentation " ) , sub_menu ) ;
2022-08-11 17:15:15 +02:00
}
2024-05-14 14:13:31 +02:00
edit_menu - > get_popup ( ) - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2016-08-02 19:11:05 -03:00
edit_menu - > get_popup ( ) - > add_separator ( ) ;
2022-08-11 17:15:15 +02:00
{
PopupMenu * sub_menu = memnew ( PopupMenu ) ;
2024-05-21 22:11:50 +02:00
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/convert_to_uppercase " ) , EDIT_TO_UPPERCASE ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/convert_to_lowercase " ) , EDIT_TO_LOWERCASE ) ;
sub_menu - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/capitalize " ) , EDIT_CAPITALIZE ) ;
2024-05-14 14:13:31 +02:00
sub_menu - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2023-11-28 17:33:04 +01:00
edit_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Convert Case " ) , sub_menu ) ;
2022-08-11 17:15:15 +02:00
}
2023-11-28 17:33:04 +01:00
edit_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Syntax Highlighter " ) , highlighter_menu ) ;
2024-05-14 14:13:31 +02:00
highlighter_menu - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _change_syntax_highlighter ) ) ;
2018-06-01 21:32:19 +02:00
2022-08-11 17:15:15 +02:00
edit_hb - > add_child ( search_menu ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/find " ) , SEARCH_FIND ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/find_next " ) , SEARCH_FIND_NEXT ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/find_previous " ) , SEARCH_FIND_PREV ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/replace " ) , SEARCH_REPLACE ) ;
search_menu - > get_popup ( ) - > add_separator ( ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/find_in_files " ) , SEARCH_IN_FILES ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/replace_in_files " ) , REPLACE_IN_FILES ) ;
search_menu - > get_popup ( ) - > add_separator ( ) ;
search_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/contextual_help " ) , HELP_CONTEXTUAL ) ;
2024-05-14 14:13:31 +02:00
search_menu - > get_popup ( ) - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2022-08-11 17:15:15 +02:00
2019-11-20 10:09:59 +01:00
_load_theme_settings ( ) ;
edit_hb - > add_child ( goto_menu ) ;
goto_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/goto_function " ) , SEARCH_LOCATE_FUNCTION ) ;
goto_menu - > get_popup ( ) - > add_shortcut ( ED_GET_SHORTCUT ( " script_text_editor/goto_line " ) , SEARCH_GOTO_LINE ) ;
goto_menu - > get_popup ( ) - > add_separator ( ) ;
2023-11-28 17:33:04 +01:00
goto_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Bookmarks " ) , bookmarks_menu ) ;
2019-11-20 10:09:59 +01:00
_update_bookmark_list ( ) ;
bookmarks_menu - > connect ( " about_to_popup " , callable_mp ( this , & ScriptTextEditor : : _update_bookmark_list ) ) ;
bookmarks_menu - > connect ( " index_pressed " , callable_mp ( this , & ScriptTextEditor : : _bookmark_item_pressed ) ) ;
2023-11-28 17:33:04 +01:00
goto_menu - > get_popup ( ) - > add_submenu_node_item ( TTR ( " Breakpoints " ) , breakpoints_menu ) ;
2019-11-20 10:09:59 +01:00
_update_breakpoint_list ( ) ;
breakpoints_menu - > connect ( " about_to_popup " , callable_mp ( this , & ScriptTextEditor : : _update_breakpoint_list ) ) ;
breakpoints_menu - > connect ( " index_pressed " , callable_mp ( this , & ScriptTextEditor : : _breakpoint_item_pressed ) ) ;
2024-05-14 14:13:31 +02:00
goto_menu - > get_popup ( ) - > connect ( SceneStringName ( id_pressed ) , callable_mp ( this , & ScriptTextEditor : : _edit_option ) ) ;
2019-11-20 10:09:59 +01:00
}
ScriptTextEditor : : ScriptTextEditor ( ) {
code_editor = memnew ( CodeTextEditor ) ;
2022-07-31 23:07:10 +02:00
code_editor - > set_toggle_list_control ( ScriptEditor : : get_singleton ( ) - > get_left_list_split ( ) ) ;
2022-02-08 10:14:58 +01:00
code_editor - > add_theme_constant_override ( " separation " , 2 ) ;
2022-03-18 19:02:57 -05:00
code_editor - > set_anchors_and_offsets_preset ( Control : : PRESET_FULL_RECT ) ;
2019-11-20 10:09:59 +01:00
code_editor - > set_code_complete_func ( _code_complete_scripts , this ) ;
code_editor - > set_v_size_flags ( SIZE_EXPAND_FILL ) ;
2020-07-30 12:22:09 +01:00
code_editor - > get_text_editor ( ) - > set_draw_breakpoints_gutter ( true ) ;
code_editor - > get_text_editor ( ) - > set_draw_executing_lines_gutter ( true ) ;
2021-10-07 19:46:44 +01:00
code_editor - > get_text_editor ( ) - > connect ( " breakpoint_toggled " , callable_mp ( this , & ScriptTextEditor : : _breakpoint_toggled ) ) ;
2022-10-22 23:54:37 +02:00
code_editor - > get_text_editor ( ) - > connect ( " caret_changed " , callable_mp ( this , & ScriptTextEditor : : _on_caret_moved ) ) ;
2024-02-22 15:32:12 -05:00
code_editor - > connect ( " navigation_preview_ended " , callable_mp ( this , & ScriptTextEditor : : _on_caret_moved ) ) ;
2020-07-30 12:22:09 +01:00
2020-07-29 22:26:49 +01:00
connection_gutter = 1 ;
code_editor - > get_text_editor ( ) - > add_gutter ( connection_gutter ) ;
code_editor - > get_text_editor ( ) - > set_gutter_name ( connection_gutter , " connection_gutter " ) ;
code_editor - > get_text_editor ( ) - > set_gutter_draw ( connection_gutter , false ) ;
code_editor - > get_text_editor ( ) - > set_gutter_overwritable ( connection_gutter , true ) ;
2021-07-19 11:16:00 +03:00
code_editor - > get_text_editor ( ) - > set_gutter_type ( connection_gutter , TextEdit : : GUTTER_TYPE_ICON ) ;
2020-07-29 22:26:49 +01:00
2019-11-20 10:09:59 +01:00
warnings_panel = memnew ( RichTextLabel ) ;
warnings_panel - > set_custom_minimum_size ( Size2 ( 0 , 100 * EDSCALE ) ) ;
warnings_panel - > set_h_size_flags ( SIZE_EXPAND_FILL ) ;
warnings_panel - > set_meta_underline ( true ) ;
warnings_panel - > set_selection_enabled ( true ) ;
2023-02-28 16:19:48 +03:00
warnings_panel - > set_context_menu_enabled ( true ) ;
2019-11-20 10:09:59 +01:00
warnings_panel - > set_focus_mode ( FOCUS_CLICK ) ;
warnings_panel - > hide ( ) ;
2021-05-18 13:09:19 +10:00
errors_panel = memnew ( RichTextLabel ) ;
errors_panel - > set_custom_minimum_size ( Size2 ( 0 , 100 * EDSCALE ) ) ;
errors_panel - > set_h_size_flags ( SIZE_EXPAND_FILL ) ;
errors_panel - > set_meta_underline ( true ) ;
errors_panel - > set_selection_enabled ( true ) ;
2023-02-28 16:19:48 +03:00
errors_panel - > set_context_menu_enabled ( true ) ;
2021-05-18 13:09:19 +10:00
errors_panel - > set_focus_mode ( FOCUS_CLICK ) ;
errors_panel - > hide ( ) ;
2019-11-20 10:09:59 +01:00
update_settings ( ) ;
2021-07-01 17:10:54 +01:00
code_editor - > get_text_editor ( ) - > set_symbol_lookup_on_click_enabled ( true ) ;
2024-12-14 21:25:05 +03:00
code_editor - > get_text_editor ( ) - > set_symbol_tooltip_on_hover_enabled ( true ) ;
2020-07-24 15:50:35 +01:00
code_editor - > get_text_editor ( ) - > set_context_menu_enabled ( false ) ;
2019-11-20 10:09:59 +01:00
context_menu = memnew ( PopupMenu ) ;
color_panel = memnew ( PopupPanel ) ;
edit_hb = memnew ( HBoxContainer ) ;
edit_menu = memnew ( MenuButton ) ;
edit_menu - > set_text ( TTR ( " Edit " ) ) ;
edit_menu - > set_switch_on_hover ( true ) ;
2020-09-17 11:40:00 +10:00
edit_menu - > set_shortcut_context ( this ) ;
2019-11-20 10:09:59 +01:00
highlighter_menu = memnew ( PopupMenu ) ;
2020-05-03 17:08:15 +01:00
Ref < EditorPlainTextSyntaxHighlighter > plain_highlighter ;
2021-06-17 16:03:09 -06:00
plain_highlighter . instantiate ( ) ;
2020-05-03 17:08:15 +01:00
add_syntax_highlighter ( plain_highlighter ) ;
Ref < EditorStandardSyntaxHighlighter > highlighter ;
2021-06-17 16:03:09 -06:00
highlighter . instantiate ( ) ;
2020-05-03 17:08:15 +01:00
add_syntax_highlighter ( highlighter ) ;
set_syntax_highlighter ( highlighter ) ;
2017-03-05 16:44:50 +01:00
search_menu = memnew ( MenuButton ) ;
2016-08-02 19:11:05 -03:00
search_menu - > set_text ( TTR ( " Search " ) ) ;
2018-07-29 19:26:43 -03:00
search_menu - > set_switch_on_hover ( true ) ;
2020-09-17 11:40:00 +10:00
search_menu - > set_shortcut_context ( this ) ;
2020-03-12 09:37:40 -03:00
2019-11-20 10:09:59 +01:00
goto_menu = memnew ( MenuButton ) ;
2019-07-01 16:00:05 -03:00
goto_menu - > set_text ( TTR ( " Go To " ) ) ;
goto_menu - > set_switch_on_hover ( true ) ;
2020-09-17 11:40:00 +10:00
goto_menu - > set_shortcut_context ( this ) ;
2019-07-01 16:00:05 -03:00
bookmarks_menu = memnew ( PopupMenu ) ;
breakpoints_menu = memnew ( PopupMenu ) ;
2016-09-11 11:28:01 -03:00
2019-04-20 12:51:25 +01:00
connection_info_dialog = memnew ( ConnectionInfoDialog ) ;
2023-01-22 17:07:55 +01:00
SET_DRAG_FORWARDING_GCD ( code_editor - > get_text_editor ( ) , ScriptTextEditor ) ;
2016-08-02 19:11:05 -03:00
}
2019-07-25 17:30:48 +01:00
ScriptTextEditor : : ~ ScriptTextEditor ( ) {
highlighters . clear ( ) ;
2019-11-20 10:09:59 +01:00
if ( ! editor_enabled ) {
memdelete ( code_editor ) ;
memdelete ( warnings_panel ) ;
2021-05-18 13:09:19 +10:00
memdelete ( errors_panel ) ;
2019-11-20 10:09:59 +01:00
memdelete ( context_menu ) ;
memdelete ( color_panel ) ;
memdelete ( edit_hb ) ;
memdelete ( edit_menu ) ;
memdelete ( highlighter_menu ) ;
memdelete ( search_menu ) ;
memdelete ( goto_menu ) ;
memdelete ( bookmarks_menu ) ;
memdelete ( breakpoints_menu ) ;
memdelete ( connection_info_dialog ) ;
}
2019-07-25 17:30:48 +01:00
}
2022-05-03 01:43:50 +02:00
static ScriptEditorBase * create_editor ( const Ref < Resource > & p_resource ) {
2018-05-28 16:52:28 +01:00
if ( Object : : cast_to < Script > ( * p_resource ) ) {
return memnew ( ScriptTextEditor ) ;
}
2020-04-02 01:20:12 +02:00
return nullptr ;
2016-08-02 19:11:05 -03:00
}
void ScriptTextEditor : : register_editor ( ) {
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/move_up " , TTRC ( " Move Up " ) , KeyModifierMask : : ALT | Key : : UP ) ;
ED_SHORTCUT ( " script_text_editor/move_down " , TTRC ( " Move Down " ) , KeyModifierMask : : ALT | Key : : DOWN ) ;
ED_SHORTCUT ( " script_text_editor/delete_line " , TTRC ( " Delete Line " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : K ) ;
2017-01-04 17:37:45 -03:00
2019-08-08 22:36:51 -03:00
// Leave these at zero, same can be accomplished with tab/shift-tab, including selection.
// The next/previous in history shortcut in this case makes a lot more sense.
2017-01-04 17:37:45 -03:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/indent " , TTRC ( " Indent " ) , Key : : NONE ) ;
ED_SHORTCUT ( " script_text_editor/unindent " , TTRC ( " Unindent " ) , KeyModifierMask : : SHIFT | Key : : TAB ) ;
ED_SHORTCUT_ARRAY ( " script_text_editor/toggle_comment " , TTRC ( " Toggle Comment " ) , { int32_t ( KeyModifierMask : : CMD_OR_CTRL | Key : : K ) , int32_t ( KeyModifierMask : : CMD_OR_CTRL | Key : : SLASH ) , int32_t ( KeyModifierMask : : CMD_OR_CTRL | Key : : KP_DIVIDE ) , int32_t ( KeyModifierMask : : CMD_OR_CTRL | Key : : NUMBERSIGN ) } ) ;
ED_SHORTCUT ( " script_text_editor/toggle_fold_line " , TTRC ( " Fold/Unfold Line " ) , KeyModifierMask : : ALT | Key : : F ) ;
2023-02-16 11:28:04 +02:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/toggle_fold_line " , " macos " , KeyModifierMask : : CTRL | KeyModifierMask : : META | Key : : F ) ;
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/fold_all_lines " , TTRC ( " Fold All Lines " ) , Key : : NONE ) ;
ED_SHORTCUT ( " script_text_editor/create_code_region " , TTRC ( " Create Code Region " ) , KeyModifierMask : : ALT | Key : : R ) ;
ED_SHORTCUT ( " script_text_editor/unfold_all_lines " , TTRC ( " Unfold All Lines " ) , Key : : NONE ) ;
ED_SHORTCUT ( " script_text_editor/duplicate_selection " , TTRC ( " Duplicate Selection " ) , KeyModifierMask : : SHIFT | KeyModifierMask : : CTRL | Key : : D ) ;
2022-09-02 12:37:48 +03:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/duplicate_selection " , " macos " , KeyModifierMask : : SHIFT | KeyModifierMask : : META | Key : : C ) ;
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/duplicate_lines " , TTRC ( " Duplicate Lines " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : ALT | Key : : DOWN ) ;
2022-09-28 17:09:45 +02:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/duplicate_lines " , " macos " , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : DOWN ) ;
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/evaluate_selection " , TTRC ( " Evaluate Selection " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : E ) ;
ED_SHORTCUT ( " script_text_editor/toggle_word_wrap " , TTRC ( " Toggle Word Wrap " ) , KeyModifierMask : : ALT | Key : : Z ) ;
ED_SHORTCUT ( " script_text_editor/trim_trailing_whitespace " , TTRC ( " Trim Trailing Whitespace " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : ALT | Key : : T ) ;
ED_SHORTCUT ( " script_text_editor/trim_final_newlines " , TTRC ( " Trim Final Newlines " ) , Key : : NONE ) ;
ED_SHORTCUT ( " script_text_editor/convert_indent_to_spaces " , TTRC ( " Convert Indent to Spaces " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : Y ) ;
ED_SHORTCUT ( " script_text_editor/convert_indent_to_tabs " , TTRC ( " Convert Indent to Tabs " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : I ) ;
ED_SHORTCUT ( " script_text_editor/auto_indent " , TTRC ( " Auto Indent " ) , KeyModifierMask : : CMD_OR_CTRL | Key : : I ) ;
2016-08-02 19:11:05 -03:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT_AND_COMMAND ( " script_text_editor/find " , TTRC ( " Find... " ) , KeyModifierMask : : CMD_OR_CTRL | Key : : F ) ;
2021-08-08 13:52:00 +10:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/find_next " , TTRC ( " Find Next " ) , Key : : F3 ) ;
2022-09-02 12:37:48 +03:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/find_next " , " macos " , KeyModifierMask : : META | Key : : G ) ;
2021-08-08 13:52:00 +10:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/find_previous " , TTRC ( " Find Previous " ) , KeyModifierMask : : SHIFT | Key : : F3 ) ;
2022-09-02 12:37:48 +03:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/find_previous " , " macos " , KeyModifierMask : : META | KeyModifierMask : : SHIFT | Key : : G ) ;
2021-08-08 13:52:00 +10:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT_AND_COMMAND ( " script_text_editor/replace " , TTRC ( " Replace... " ) , KeyModifierMask : : CTRL | Key : : R ) ;
2022-09-02 12:37:48 +03:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/replace " , " macos " , KeyModifierMask : : ALT | KeyModifierMask : : META | Key : : F ) ;
2016-08-02 19:11:05 -03:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/find_in_files " , TTRC ( " Find in Files... " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : F ) ;
ED_SHORTCUT ( " script_text_editor/replace_in_files " , TTRC ( " Replace in Files... " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : R ) ;
2018-02-12 02:36:15 +01:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/contextual_help " , TTRC ( " Contextual Help " ) , KeyModifierMask : : ALT | Key : : F1 ) ;
2021-08-13 16:31:57 -05:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/contextual_help " , " macos " , KeyModifierMask : : ALT | KeyModifierMask : : SHIFT | Key : : SPACE ) ;
2019-08-08 22:36:51 -03:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/toggle_bookmark " , TTRC ( " Toggle Bookmark " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : ALT | Key : : B ) ;
2023-11-29 21:36:56 -05:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/goto_next_bookmark " , TTRC ( " Go to Next Bookmark " ) , KeyModifierMask : : CMD_OR_CTRL | Key : : B ) ;
2023-11-29 21:36:56 -05:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/goto_next_bookmark " , " macos " , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | KeyModifierMask : : ALT | Key : : B ) ;
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/goto_previous_bookmark " , TTRC ( " Go to Previous Bookmark " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : B ) ;
ED_SHORTCUT ( " script_text_editor/remove_all_bookmarks " , TTRC ( " Remove All Bookmarks " ) , Key : : NONE ) ;
2019-08-08 22:36:51 -03:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/goto_function " , TTRC ( " Go to Function... " ) , KeyModifierMask : : ALT | KeyModifierMask : : CTRL | Key : : F ) ;
2022-09-02 12:37:48 +03:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/goto_function " , " macos " , KeyModifierMask : : CTRL | KeyModifierMask : : META | Key : : J ) ;
2021-08-08 13:52:00 +10:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/goto_line " , TTRC ( " Go to Line... " ) , KeyModifierMask : : CMD_OR_CTRL | Key : : L ) ;
2016-08-02 19:11:05 -03:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/toggle_breakpoint " , TTRC ( " Toggle Breakpoint " ) , Key : : F9 ) ;
2022-09-02 12:37:48 +03:00
ED_SHORTCUT_OVERRIDE ( " script_text_editor/toggle_breakpoint " , " macos " , KeyModifierMask : : META | KeyModifierMask : : SHIFT | Key : : B ) ;
2021-08-08 13:52:00 +10:00
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/remove_all_breakpoints " , TTRC ( " Remove All Breakpoints " ) , KeyModifierMask : : CMD_OR_CTRL | KeyModifierMask : : SHIFT | Key : : F9 ) ;
2024-08-27 22:44:56 +04:00
// Using Control for these shortcuts even on macOS because Command+Comma is taken for opening Editor Settings.
2024-10-18 19:07:44 +08:00
ED_SHORTCUT ( " script_text_editor/goto_next_breakpoint " , TTRC ( " Go to Next Breakpoint " ) , KeyModifierMask : : CTRL | Key : : PERIOD ) ;
ED_SHORTCUT ( " script_text_editor/goto_previous_breakpoint " , TTRC ( " Go to Previous Breakpoint " ) , KeyModifierMask : : CTRL | Key : : COMMA ) ;
2016-08-02 19:11:05 -03:00
ScriptEditor : : register_create_script_editor_function ( create_editor ) ;
}
2019-05-08 18:49:49 +02:00
void ScriptTextEditor : : validate ( ) {
2024-01-28 21:51:39 +01:00
code_editor - > validate_script ( ) ;
2019-05-08 18:49:49 +02:00
}