2023-01-05 13:25:55 +01:00
|
|
|
/**************************************************************************/
|
|
|
|
/* script_text_editor.h */
|
|
|
|
/**************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
|
|
|
/* https://godotengine.org */
|
|
|
|
/**************************************************************************/
|
|
|
|
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
|
|
|
|
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* */
|
|
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
|
|
/* a copy of this software and associated documentation files (the */
|
|
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
|
|
/* the following conditions: */
|
|
|
|
/* */
|
|
|
|
/* The above copyright notice and this permission notice shall be */
|
|
|
|
/* included in all copies or substantial portions of the Software. */
|
|
|
|
/* */
|
|
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
|
|
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
|
|
/**************************************************************************/
|
2018-01-05 00:50:27 +01:00
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
#ifndef SCRIPT_TEXT_EDITOR_H
|
|
|
|
#define SCRIPT_TEXT_EDITOR_H
|
|
|
|
|
2022-11-19 12:45:49 +01:00
|
|
|
#include "script_editor_plugin.h"
|
|
|
|
|
|
|
|
#include "editor/code_editor.h"
|
2016-09-29 09:12:45 +02:00
|
|
|
#include "scene/gui/color_picker.h"
|
2019-04-20 13:51:25 +02:00
|
|
|
#include "scene/gui/dialogs.h"
|
|
|
|
#include "scene/gui/tree.h"
|
2022-11-19 12:45:49 +01:00
|
|
|
|
|
|
|
class RichTextLabel;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2019-04-20 13:51:25 +02:00
|
|
|
class ConnectionInfoDialog : public AcceptDialog {
|
|
|
|
GDCLASS(ConnectionInfoDialog, AcceptDialog);
|
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
Label *method = nullptr;
|
|
|
|
Tree *tree = nullptr;
|
2019-04-20 13:51:25 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void ok_pressed() override;
|
2019-04-20 13:51:25 +02:00
|
|
|
|
|
|
|
public:
|
|
|
|
void popup_connections(String p_method, Vector<Node *> p_nodes);
|
|
|
|
|
|
|
|
ConnectionInfoDialog();
|
|
|
|
};
|
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
class ScriptTextEditor : public ScriptEditorBase {
|
2017-01-03 03:03:46 +01:00
|
|
|
GDCLASS(ScriptTextEditor, ScriptEditorBase);
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
CodeTextEditor *code_editor = nullptr;
|
|
|
|
RichTextLabel *warnings_panel = nullptr;
|
2021-05-18 05:09:19 +02:00
|
|
|
RichTextLabel *errors_panel = nullptr;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
|
|
|
Ref<Script> script;
|
2019-11-20 10:09:59 +01:00
|
|
|
bool script_is_valid = false;
|
|
|
|
bool editor_enabled = false;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
|
|
|
Vector<String> functions;
|
2022-01-19 06:31:39 +01:00
|
|
|
List<ScriptLanguage::Warning> warnings;
|
|
|
|
List<ScriptLanguage::ScriptError> errors;
|
2022-05-19 17:00:06 +02:00
|
|
|
HashSet<int> safe_lines;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2019-04-20 13:51:25 +02:00
|
|
|
List<Connection> missing_connections;
|
|
|
|
|
2018-01-12 15:00:41 +01:00
|
|
|
Vector<String> member_keywords;
|
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
HBoxContainer *edit_hb = nullptr;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
MenuButton *edit_menu = nullptr;
|
|
|
|
MenuButton *search_menu = nullptr;
|
|
|
|
MenuButton *goto_menu = nullptr;
|
|
|
|
PopupMenu *bookmarks_menu = nullptr;
|
|
|
|
PopupMenu *breakpoints_menu = nullptr;
|
|
|
|
PopupMenu *highlighter_menu = nullptr;
|
|
|
|
PopupMenu *context_menu = nullptr;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
GotoLineDialog *goto_line_dialog = nullptr;
|
|
|
|
ScriptEditorQuickOpen *quick_open = nullptr;
|
|
|
|
ConnectionInfoDialog *connection_info_dialog = nullptr;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2020-07-29 23:26:49 +02:00
|
|
|
int connection_gutter = -1;
|
|
|
|
void _gutter_clicked(int p_line, int p_gutter);
|
|
|
|
void _update_gutter_indexes();
|
|
|
|
|
2020-07-30 12:41:05 +02:00
|
|
|
int line_number_gutter = -1;
|
|
|
|
Color default_line_number_color = Color(1, 1, 1);
|
|
|
|
Color safe_line_number_color = Color(1, 1, 1);
|
|
|
|
|
2020-09-10 22:25:00 +02:00
|
|
|
Color marked_line_color = Color(1, 1, 1);
|
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
PopupPanel *color_panel = nullptr;
|
|
|
|
ColorPicker *color_picker = nullptr;
|
2019-08-02 22:32:33 +02:00
|
|
|
Vector2 color_position;
|
2016-09-29 09:12:45 +02:00
|
|
|
String color_args;
|
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
bool theme_loaded = false;
|
2017-10-03 01:33:42 +02:00
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
enum {
|
|
|
|
EDIT_UNDO,
|
|
|
|
EDIT_REDO,
|
|
|
|
EDIT_CUT,
|
|
|
|
EDIT_COPY,
|
|
|
|
EDIT_PASTE,
|
|
|
|
EDIT_SELECT_ALL,
|
|
|
|
EDIT_COMPLETE,
|
|
|
|
EDIT_AUTO_INDENT,
|
|
|
|
EDIT_TRIM_TRAILING_WHITESAPCE,
|
2017-04-16 17:47:59 +02:00
|
|
|
EDIT_CONVERT_INDENT_TO_SPACES,
|
|
|
|
EDIT_CONVERT_INDENT_TO_TABS,
|
2016-08-03 00:11:05 +02:00
|
|
|
EDIT_TOGGLE_COMMENT,
|
|
|
|
EDIT_MOVE_LINE_UP,
|
|
|
|
EDIT_MOVE_LINE_DOWN,
|
2022-09-29 04:12:52 +02:00
|
|
|
EDIT_INDENT,
|
|
|
|
EDIT_UNINDENT,
|
2017-08-20 14:31:30 +02:00
|
|
|
EDIT_DELETE_LINE,
|
2021-06-29 12:09:07 +02:00
|
|
|
EDIT_DUPLICATE_SELECTION,
|
2016-09-29 09:12:45 +02:00
|
|
|
EDIT_PICK_COLOR,
|
2017-04-24 20:59:55 +02:00
|
|
|
EDIT_TO_UPPERCASE,
|
|
|
|
EDIT_TO_LOWERCASE,
|
|
|
|
EDIT_CAPITALIZE,
|
2019-08-07 16:31:33 +02:00
|
|
|
EDIT_EVALUATE,
|
2017-12-08 19:17:10 +01:00
|
|
|
EDIT_TOGGLE_FOLD_LINE,
|
2017-11-16 05:00:27 +01:00
|
|
|
EDIT_FOLD_ALL_LINES,
|
2017-11-13 00:12:17 +01:00
|
|
|
EDIT_UNFOLD_ALL_LINES,
|
2016-08-03 00:11:05 +02:00
|
|
|
SEARCH_FIND,
|
|
|
|
SEARCH_FIND_NEXT,
|
|
|
|
SEARCH_FIND_PREV,
|
|
|
|
SEARCH_REPLACE,
|
|
|
|
SEARCH_LOCATE_FUNCTION,
|
|
|
|
SEARCH_GOTO_LINE,
|
2018-02-12 02:36:15 +01:00
|
|
|
SEARCH_IN_FILES,
|
2020-02-09 10:10:58 +01:00
|
|
|
REPLACE_IN_FILES,
|
2019-04-20 01:51:25 +02:00
|
|
|
BOOKMARK_TOGGLE,
|
|
|
|
BOOKMARK_GOTO_NEXT,
|
|
|
|
BOOKMARK_GOTO_PREV,
|
|
|
|
BOOKMARK_REMOVE_ALL,
|
2016-08-03 00:11:05 +02:00
|
|
|
DEBUG_TOGGLE_BREAKPOINT,
|
|
|
|
DEBUG_REMOVE_ALL_BREAKPOINTS,
|
|
|
|
DEBUG_GOTO_NEXT_BREAKPOINT,
|
|
|
|
DEBUG_GOTO_PREV_BREAKPOINT,
|
|
|
|
HELP_CONTEXTUAL,
|
2018-01-02 08:10:49 +01:00
|
|
|
LOOKUP_SYMBOL,
|
2016-08-03 00:11:05 +02:00
|
|
|
};
|
|
|
|
|
2019-11-20 10:09:59 +01:00
|
|
|
void _enable_code_editor();
|
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
protected:
|
2019-07-01 21:00:05 +02:00
|
|
|
void _update_breakpoint_list();
|
|
|
|
void _breakpoint_item_pressed(int p_idx);
|
2016-08-03 00:11:05 +02:00
|
|
|
void _breakpoint_toggled(int p_row);
|
|
|
|
|
2019-05-21 10:07:48 +02:00
|
|
|
void _validate_script(); // No longer virtual.
|
2022-01-19 06:31:39 +01:00
|
|
|
void _update_warnings();
|
|
|
|
void _update_errors();
|
2019-05-21 10:07:48 +02:00
|
|
|
void _update_bookmark_list();
|
|
|
|
void _bookmark_item_pressed(int p_idx);
|
2019-06-13 11:17:20 +02:00
|
|
|
|
2022-03-26 16:48:43 +01:00
|
|
|
static void _code_complete_scripts(void *p_ud, const String &p_code, List<ScriptLanguage::CodeCompletionOption> *r_options, bool &r_force);
|
|
|
|
void _code_complete_script(const String &p_code, List<ScriptLanguage::CodeCompletionOption> *r_options, bool &r_force);
|
2019-06-13 11:17:20 +02:00
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
void _load_theme_settings();
|
2017-10-03 01:33:42 +02:00
|
|
|
void _set_theme_for_script();
|
2021-05-18 05:09:19 +02:00
|
|
|
void _show_errors_panel(bool p_show);
|
2019-02-13 10:12:31 +01:00
|
|
|
void _show_warnings_panel(bool p_show);
|
2021-05-18 05:09:19 +02:00
|
|
|
void _error_clicked(Variant p_line);
|
2018-07-01 18:17:40 +02:00
|
|
|
void _warning_clicked(Variant p_line);
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2020-07-29 23:26:49 +02:00
|
|
|
void _notification(int p_what);
|
2016-08-03 00:11:05 +02:00
|
|
|
static void _bind_methods();
|
|
|
|
|
2022-05-13 15:04:37 +02:00
|
|
|
HashMap<String, Ref<EditorSyntaxHighlighter>> highlighters;
|
2018-04-02 13:41:44 +02:00
|
|
|
void _change_syntax_highlighter(int p_idx);
|
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
void _edit_option(int p_op);
|
2019-02-04 20:17:44 +01:00
|
|
|
void _edit_option_toggle_inline_comment();
|
2019-06-22 20:22:52 +02:00
|
|
|
void _make_context_menu(bool p_selection, bool p_color, bool p_foldable, bool p_open_docs, bool p_goto_definition, Vector2 p_pos);
|
2017-05-20 17:38:03 +02:00
|
|
|
void _text_edit_gui_input(const Ref<InputEvent> &ev);
|
2016-09-29 09:12:45 +02:00
|
|
|
void _color_changed(const Color &p_color);
|
2021-08-17 05:41:46 +02:00
|
|
|
void _prepare_edit_menu();
|
2016-08-03 00:11:05 +02:00
|
|
|
|
|
|
|
void _goto_line(int p_line) { goto_line(p_line); }
|
2016-09-12 15:52:29 +02:00
|
|
|
void _lookup_symbol(const String &p_symbol, int p_row, int p_column);
|
2020-03-01 09:13:41 +01:00
|
|
|
void _validate_symbol(const String &p_symbol);
|
2016-09-11 16:28:01 +02:00
|
|
|
|
2018-05-26 00:49:35 +02:00
|
|
|
void _convert_case(CodeTextEditor::CaseStyle p_case);
|
2017-04-24 20:59:55 +02:00
|
|
|
|
2016-09-11 16:28:01 +02:00
|
|
|
Variant get_drag_data_fw(const Point2 &p_point, Control *p_from);
|
|
|
|
bool can_drop_data_fw(const Point2 &p_point, const Variant &p_data, Control *p_from) const;
|
|
|
|
void drop_data_fw(const Point2 &p_point, const Variant &p_data, Control *p_from);
|
|
|
|
|
2020-03-28 09:12:19 +01:00
|
|
|
String _get_absolute_path(const String &rel_path);
|
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
public:
|
2019-04-20 13:51:25 +02:00
|
|
|
void _update_connected_methods();
|
|
|
|
|
2020-05-03 18:08:15 +02:00
|
|
|
virtual void add_syntax_highlighter(Ref<EditorSyntaxHighlighter> p_highlighter) override;
|
|
|
|
virtual void set_syntax_highlighter(Ref<EditorSyntaxHighlighter> p_highlighter) override;
|
2021-09-14 11:17:47 +02:00
|
|
|
void update_toggle_scripts_button() override;
|
2018-04-02 13:41:44 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void apply_code() override;
|
2022-05-03 01:43:50 +02:00
|
|
|
virtual Ref<Resource> get_edited_resource() const override;
|
|
|
|
virtual void set_edited_resource(const Ref<Resource> &p_res) override;
|
2022-10-23 10:45:44 +02:00
|
|
|
virtual void enable_editor(Control *p_shortcut_context = nullptr) override;
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual Vector<String> get_functions() override;
|
|
|
|
virtual void reload_text() override;
|
|
|
|
virtual String get_name() override;
|
|
|
|
virtual Ref<Texture2D> get_theme_icon() override;
|
|
|
|
virtual bool is_unsaved() override;
|
|
|
|
virtual Variant get_edit_state() override;
|
|
|
|
virtual void set_edit_state(const Variant &p_state) override;
|
2022-09-23 14:46:28 +02:00
|
|
|
virtual Variant get_navigation_state() override;
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void ensure_focus() override;
|
|
|
|
virtual void trim_trailing_whitespace() override;
|
|
|
|
virtual void insert_final_newline() override;
|
|
|
|
virtual void convert_indent_to_spaces() override;
|
|
|
|
virtual void convert_indent_to_tabs() override;
|
|
|
|
virtual void tag_saved_version() override;
|
|
|
|
|
|
|
|
virtual void goto_line(int p_line, bool p_with_error = false) override;
|
2018-02-12 02:36:15 +01:00
|
|
|
void goto_line_selection(int p_line, int p_begin, int p_end);
|
2019-01-04 13:09:01 +01:00
|
|
|
void goto_line_centered(int p_line);
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void set_executing_line(int p_line) override;
|
|
|
|
virtual void clear_executing_line() override;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void reload(bool p_soft) override;
|
2022-08-05 03:41:48 +02:00
|
|
|
virtual PackedInt32Array get_breakpoints() override;
|
2021-07-20 13:24:56 +02:00
|
|
|
virtual void set_breakpoint(int p_line, bool p_enabled) override;
|
|
|
|
virtual void clear_breakpoints() override;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void add_callback(const String &p_function, PackedStringArray p_args) override;
|
|
|
|
virtual void update_settings() override;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual bool show_members_overview() override;
|
2017-05-28 16:20:38 +02:00
|
|
|
|
2021-11-07 18:26:15 +01:00
|
|
|
virtual void set_tooltip_request_func(const Callable &p_toolip_callback) override;
|
2016-08-03 00:11:05 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void set_debugger_active(bool p_active) override;
|
2016-08-07 00:00:54 +02:00
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
Control *get_edit_menu() override;
|
|
|
|
virtual void clear_edit_menu() override;
|
2021-06-03 01:05:41 +02:00
|
|
|
virtual void set_find_replace_bar(FindReplaceBar *p_bar) override;
|
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
static void register_editor();
|
|
|
|
|
2021-05-13 21:50:25 +02:00
|
|
|
virtual Control *get_base_editor() const override;
|
|
|
|
|
2020-07-10 12:34:39 +02:00
|
|
|
virtual void validate() override;
|
2019-05-08 18:49:49 +02:00
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
ScriptTextEditor();
|
2019-07-25 18:30:48 +02:00
|
|
|
~ScriptTextEditor();
|
2016-08-03 00:11:05 +02:00
|
|
|
};
|
|
|
|
|
2022-03-08 15:03:36 +01:00
|
|
|
const int KIND_COUNT = 10;
|
|
|
|
// The order in which to sort code completion options.
|
|
|
|
const ScriptLanguage::CodeCompletionKind KIND_SORT_ORDER[KIND_COUNT] = {
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_VARIABLE,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_MEMBER,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_FUNCTION,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_ENUM,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_SIGNAL,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_CONSTANT,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_CLASS,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_NODE_PATH,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_FILE_PATH,
|
|
|
|
ScriptLanguage::CODE_COMPLETION_KIND_PLAIN_TEXT,
|
|
|
|
};
|
|
|
|
|
|
|
|
// The custom comparer which will sort completion options.
|
|
|
|
struct CodeCompletionOptionCompare {
|
|
|
|
_FORCE_INLINE_ bool operator()(const ScriptLanguage::CodeCompletionOption &l, const ScriptLanguage::CodeCompletionOption &r) const {
|
|
|
|
if (l.location == r.location) {
|
|
|
|
// If locations are same, sort on kind
|
|
|
|
if (l.kind == r.kind) {
|
|
|
|
// If kinds are same, sort alphanumeric
|
|
|
|
return l.display < r.display;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort kinds based on the const sorting array defined above. Lower index = higher priority.
|
|
|
|
int l_index = -1;
|
|
|
|
int r_index = -1;
|
|
|
|
for (int i = 0; i < KIND_COUNT; i++) {
|
|
|
|
const ScriptLanguage::CodeCompletionKind kind = KIND_SORT_ORDER[i];
|
|
|
|
l_index = kind == l.kind ? i : l_index;
|
|
|
|
r_index = kind == r.kind ? i : r_index;
|
|
|
|
|
|
|
|
if (l_index != -1 && r_index != -1) {
|
|
|
|
return l_index < r_index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This return should never be hit unless something goes wrong.
|
|
|
|
// l and r should always have a Kind which is in the sort order array.
|
|
|
|
return l.display < r.display;
|
|
|
|
}
|
|
|
|
|
|
|
|
return l.location < r.location;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-03 00:11:05 +02:00
|
|
|
#endif // SCRIPT_TEXT_EDITOR_H
|