2018-05-16 19:19:33 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/* editor_inspector.h */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
|
|
|
/* https://godotengine.org */
|
|
|
|
/*************************************************************************/
|
2020-01-01 11:16:22 +01:00
|
|
|
/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md). */
|
2018-05-16 19:19:33 +02:00
|
|
|
/* */
|
|
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
|
|
/* a copy of this software and associated documentation files (the */
|
|
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
|
|
/* the following conditions: */
|
|
|
|
/* */
|
|
|
|
/* The above copyright notice and this permission notice shall be */
|
|
|
|
/* included in all copies or substantial portions of the Software. */
|
|
|
|
/* */
|
|
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
|
|
/*************************************************************************/
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
#ifndef EDITOR_INSPECTOR_H
|
|
|
|
#define EDITOR_INSPECTOR_H
|
|
|
|
|
2018-05-17 23:02:16 +02:00
|
|
|
#include "scene/gui/box_container.h"
|
|
|
|
#include "scene/gui/line_edit.h"
|
2018-05-15 22:12:35 +02:00
|
|
|
#include "scene/gui/scroll_container.h"
|
|
|
|
|
2018-05-17 23:02:16 +02:00
|
|
|
class UndoRedo;
|
|
|
|
|
2018-11-25 14:46:26 +01:00
|
|
|
class EditorPropertyRevert {
|
|
|
|
public:
|
|
|
|
static bool may_node_be_in_instance(Node *p_node);
|
|
|
|
static bool get_instanced_node_original_property(Node *p_node, const StringName &p_prop, Variant &value);
|
|
|
|
static bool is_node_property_different(Node *p_node, const Variant &p_current, const Variant &p_orig);
|
|
|
|
|
|
|
|
static bool can_property_revert(Object *p_object, const StringName &p_property);
|
|
|
|
};
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
class EditorProperty : public Container {
|
|
|
|
|
2019-03-19 19:35:57 +01:00
|
|
|
GDCLASS(EditorProperty, Container);
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
private:
|
|
|
|
String label;
|
|
|
|
int text_size;
|
|
|
|
friend class EditorInspector;
|
|
|
|
Object *object;
|
|
|
|
StringName property;
|
|
|
|
|
|
|
|
int property_usage;
|
|
|
|
|
|
|
|
bool read_only;
|
|
|
|
bool checkable;
|
|
|
|
bool checked;
|
|
|
|
bool draw_red;
|
|
|
|
bool keying;
|
|
|
|
|
2018-07-19 00:37:17 +02:00
|
|
|
Rect2 right_child_rect;
|
|
|
|
Rect2 bottom_child_rect;
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
Rect2 keying_rect;
|
|
|
|
bool keying_hover;
|
|
|
|
Rect2 revert_rect;
|
|
|
|
bool revert_hover;
|
|
|
|
Rect2 check_rect;
|
|
|
|
bool check_hover;
|
|
|
|
|
|
|
|
bool can_revert;
|
|
|
|
|
2018-05-17 23:02:16 +02:00
|
|
|
bool use_folding;
|
2018-08-07 17:19:19 +02:00
|
|
|
bool draw_top_bg;
|
2018-05-17 23:02:16 +02:00
|
|
|
|
2018-11-08 15:30:02 +01:00
|
|
|
bool _is_property_different(const Variant &p_current, const Variant &p_orig);
|
2018-05-15 22:12:35 +02:00
|
|
|
bool _get_instanced_node_original_property(const StringName &p_prop, Variant &value);
|
|
|
|
void _focusable_focused(int p_index);
|
|
|
|
|
2018-05-19 21:09:38 +02:00
|
|
|
bool selectable;
|
2018-05-15 22:12:35 +02:00
|
|
|
bool selected;
|
|
|
|
int selected_focusable;
|
|
|
|
|
2018-07-14 23:15:42 +02:00
|
|
|
float split_ratio;
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
Vector<Control *> focusables;
|
|
|
|
Control *label_reference;
|
2018-05-17 23:02:16 +02:00
|
|
|
Control *bottom_editor;
|
2018-05-15 22:12:35 +02:00
|
|
|
|
2018-07-20 23:14:33 +02:00
|
|
|
mutable String tooltip_text;
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
protected:
|
|
|
|
void _notification(int p_what);
|
|
|
|
static void _bind_methods();
|
|
|
|
|
|
|
|
void _gui_input(const Ref<InputEvent> &p_event);
|
|
|
|
|
|
|
|
public:
|
2019-01-18 17:01:24 +01:00
|
|
|
void emit_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field = StringName(), bool p_changing = false);
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
virtual Size2 get_minimum_size() const;
|
|
|
|
|
|
|
|
void set_label(const String &p_label);
|
|
|
|
String get_label() const;
|
|
|
|
|
|
|
|
void set_read_only(bool p_read_only);
|
|
|
|
bool is_read_only() const;
|
|
|
|
|
|
|
|
Object *get_edited_object();
|
|
|
|
StringName get_edited_property();
|
|
|
|
|
|
|
|
virtual void update_property();
|
|
|
|
void update_reload_status();
|
|
|
|
|
|
|
|
virtual bool use_keying_next() const;
|
|
|
|
|
|
|
|
void set_checkable(bool p_checkable);
|
|
|
|
bool is_checkable() const;
|
|
|
|
|
|
|
|
void set_checked(bool p_checked);
|
|
|
|
bool is_checked() const;
|
|
|
|
|
|
|
|
void set_draw_red(bool p_draw_red);
|
|
|
|
bool is_draw_red() const;
|
|
|
|
|
|
|
|
void set_keying(bool p_keying);
|
|
|
|
bool is_keying() const;
|
|
|
|
|
|
|
|
void add_focusable(Control *p_control);
|
|
|
|
void select(int p_focusable = -1);
|
|
|
|
void deselect();
|
|
|
|
bool is_selected() const;
|
|
|
|
|
|
|
|
void set_label_reference(Control *p_control);
|
2019-06-26 15:08:25 +02:00
|
|
|
void set_bottom_editor(Control *p_control);
|
2018-05-17 23:02:16 +02:00
|
|
|
|
|
|
|
void set_use_folding(bool p_use_folding);
|
|
|
|
bool is_using_folding() const;
|
|
|
|
|
|
|
|
virtual void expand_all_folding();
|
|
|
|
virtual void collapse_all_folding();
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
virtual Variant get_drag_data(const Point2 &p_point);
|
|
|
|
|
2018-05-19 21:09:38 +02:00
|
|
|
void set_selectable(bool p_selectable);
|
|
|
|
bool is_selectable() const;
|
|
|
|
|
2018-07-14 23:15:42 +02:00
|
|
|
void set_name_split_ratio(float p_ratio);
|
|
|
|
float get_name_split_ratio() const;
|
|
|
|
|
2018-05-19 21:09:38 +02:00
|
|
|
void set_object_and_property(Object *p_object, const StringName &p_property);
|
2018-07-20 23:14:33 +02:00
|
|
|
virtual Control *make_custom_tooltip(const String &p_text) const;
|
|
|
|
|
|
|
|
String get_tooltip_text() const;
|
|
|
|
|
2018-08-07 17:19:19 +02:00
|
|
|
void set_draw_top_bg(bool p_draw) { draw_top_bg = p_draw; }
|
|
|
|
|
2018-10-29 21:38:51 +01:00
|
|
|
bool can_revert_to_default() const { return can_revert; }
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
EditorProperty();
|
|
|
|
};
|
|
|
|
|
|
|
|
class EditorInspectorPlugin : public Reference {
|
2019-03-19 19:35:57 +01:00
|
|
|
GDCLASS(EditorInspectorPlugin, Reference);
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
friend class EditorInspector;
|
|
|
|
struct AddedEditor {
|
|
|
|
Control *property_editor;
|
|
|
|
Vector<String> properties;
|
|
|
|
String label;
|
|
|
|
};
|
|
|
|
|
|
|
|
List<AddedEditor> added_editors;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
static void _bind_methods();
|
|
|
|
|
|
|
|
public:
|
|
|
|
void add_custom_control(Control *control);
|
|
|
|
void add_property_editor(const String &p_for_property, Control *p_prop);
|
|
|
|
void add_property_editor_for_multiple_properties(const String &p_label, const Vector<String> &p_properties, Control *p_prop);
|
|
|
|
|
|
|
|
virtual bool can_handle(Object *p_object);
|
|
|
|
virtual void parse_begin(Object *p_object);
|
2018-05-17 23:02:16 +02:00
|
|
|
virtual void parse_category(Object *p_object, const String &p_parse_category);
|
2018-05-15 22:12:35 +02:00
|
|
|
virtual bool parse_property(Object *p_object, Variant::Type p_type, const String &p_path, PropertyHint p_hint, const String &p_hint_text, int p_usage);
|
|
|
|
virtual void parse_end();
|
|
|
|
};
|
|
|
|
|
|
|
|
class EditorInspectorCategory : public Control {
|
|
|
|
GDCLASS(EditorInspectorCategory, Control);
|
|
|
|
|
|
|
|
friend class EditorInspector;
|
2019-06-11 20:43:37 +02:00
|
|
|
Ref<Texture2D> icon;
|
2018-05-15 22:12:35 +02:00
|
|
|
String label;
|
|
|
|
Color bg_color;
|
2018-07-20 23:14:33 +02:00
|
|
|
mutable String tooltip_text;
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
protected:
|
|
|
|
void _notification(int p_what);
|
2018-07-20 23:14:33 +02:00
|
|
|
static void _bind_methods();
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
public:
|
|
|
|
virtual Size2 get_minimum_size() const;
|
2018-07-20 23:14:33 +02:00
|
|
|
virtual Control *make_custom_tooltip(const String &p_text) const;
|
|
|
|
|
|
|
|
String get_tooltip_text() const;
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
EditorInspectorCategory();
|
|
|
|
};
|
|
|
|
|
|
|
|
class EditorInspectorSection : public Container {
|
|
|
|
GDCLASS(EditorInspectorSection, Container);
|
|
|
|
|
|
|
|
String label;
|
|
|
|
String section;
|
|
|
|
Object *object;
|
|
|
|
VBoxContainer *vbox;
|
2018-07-19 00:37:17 +02:00
|
|
|
bool vbox_added; //optimization
|
2018-05-15 22:12:35 +02:00
|
|
|
Color bg_color;
|
|
|
|
bool foldable;
|
|
|
|
|
2018-07-19 00:37:17 +02:00
|
|
|
void _test_unfold();
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
protected:
|
|
|
|
void _notification(int p_what);
|
|
|
|
static void _bind_methods();
|
|
|
|
void _gui_input(const Ref<InputEvent> &p_event);
|
|
|
|
|
|
|
|
public:
|
|
|
|
virtual Size2 get_minimum_size() const;
|
|
|
|
|
|
|
|
void setup(const String &p_section, const String &p_label, Object *p_object, const Color &p_bg_color, bool p_foldable);
|
|
|
|
VBoxContainer *get_vbox();
|
|
|
|
void unfold();
|
|
|
|
void fold();
|
|
|
|
|
|
|
|
Object *get_edited_object();
|
|
|
|
|
|
|
|
EditorInspectorSection();
|
2018-07-19 00:37:17 +02:00
|
|
|
~EditorInspectorSection();
|
2018-05-15 22:12:35 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class EditorInspector : public ScrollContainer {
|
|
|
|
GDCLASS(EditorInspector, ScrollContainer);
|
|
|
|
|
|
|
|
UndoRedo *undo_redo;
|
|
|
|
enum {
|
|
|
|
MAX_PLUGINS = 1024
|
|
|
|
};
|
|
|
|
static Ref<EditorInspectorPlugin> inspector_plugins[MAX_PLUGINS];
|
|
|
|
static int inspector_plugin_count;
|
|
|
|
|
|
|
|
VBoxContainer *main_vbox;
|
|
|
|
|
|
|
|
//map use to cache the instanced editors
|
|
|
|
Map<StringName, List<EditorProperty *> > editor_property_map;
|
|
|
|
List<EditorInspectorSection *> sections;
|
|
|
|
Set<StringName> pending;
|
|
|
|
|
|
|
|
void _clear();
|
|
|
|
Object *object;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
LineEdit *search_box;
|
|
|
|
bool show_categories;
|
2019-07-22 12:03:57 +02:00
|
|
|
bool hide_script;
|
2018-05-15 22:12:35 +02:00
|
|
|
bool use_doc_hints;
|
|
|
|
bool capitalize_paths;
|
|
|
|
bool use_filter;
|
|
|
|
bool autoclear;
|
|
|
|
bool use_folding;
|
|
|
|
int changing;
|
|
|
|
bool update_all_pending;
|
|
|
|
bool read_only;
|
|
|
|
bool keying;
|
2019-01-25 19:14:56 +01:00
|
|
|
bool sub_inspector;
|
2018-05-15 22:12:35 +02:00
|
|
|
|
2018-07-05 01:08:45 +02:00
|
|
|
float refresh_countdown;
|
2018-05-15 22:12:35 +02:00
|
|
|
bool update_tree_pending;
|
|
|
|
StringName _prop_edited;
|
|
|
|
StringName property_selected;
|
|
|
|
int property_focusable;
|
2018-07-19 00:37:17 +02:00
|
|
|
int update_scroll_request;
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
Map<StringName, Map<StringName, String> > descr_cache;
|
|
|
|
Map<StringName, String> class_descr_cache;
|
2018-07-19 23:58:15 +02:00
|
|
|
Set<StringName> restart_request_props;
|
2018-05-15 22:12:35 +02:00
|
|
|
|
2018-07-19 00:37:17 +02:00
|
|
|
Map<ObjectID, int> scroll_cache;
|
|
|
|
|
2018-07-19 23:58:15 +02:00
|
|
|
String property_prefix; //used for sectioned inspector
|
|
|
|
String object_class;
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
void _edit_set(const String &p_name, const Variant &p_value, bool p_refresh_all, const String &p_changed_field);
|
|
|
|
|
2018-09-02 19:16:33 +02:00
|
|
|
void _property_changed(const String &p_path, const Variant &p_value, const String &p_name = "", bool changing = false);
|
2019-01-27 17:37:24 +01:00
|
|
|
void _property_changed_update_all(const String &p_path, const Variant &p_value, const String &p_name = "", bool p_changing = false);
|
2018-05-15 22:12:35 +02:00
|
|
|
void _multiple_properties_changed(Vector<String> p_paths, Array p_values);
|
2018-11-08 21:46:34 +01:00
|
|
|
void _property_keyed(const String &p_path, bool p_advance);
|
|
|
|
void _property_keyed_with_value(const String &p_path, const Variant &p_value, bool p_advance);
|
2018-05-17 23:02:16 +02:00
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
void _property_checked(const String &p_path, bool p_checked);
|
|
|
|
|
|
|
|
void _resource_selected(const String &p_path, RES p_resource);
|
|
|
|
void _property_selected(const String &p_path, int p_focusable);
|
|
|
|
void _object_id_selected(const String &p_path, ObjectID p_id);
|
|
|
|
|
|
|
|
void _node_removed(Node *p_node);
|
|
|
|
|
|
|
|
void _changed_callback(Object *p_changed, const char *p_prop);
|
2019-06-26 15:08:25 +02:00
|
|
|
void _edit_request_change(Object *p_object, const String &p_prop);
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
void _filter_changed(const String &p_text);
|
2018-05-17 23:02:16 +02:00
|
|
|
void _parse_added_editors(VBoxContainer *current_vbox, Ref<EditorInspectorPlugin> ped);
|
2018-05-15 22:12:35 +02:00
|
|
|
|
2018-07-19 00:37:17 +02:00
|
|
|
void _vscroll_changed(double);
|
|
|
|
|
2019-04-09 00:18:03 +02:00
|
|
|
void _feature_profile_changed();
|
|
|
|
|
|
|
|
bool _is_property_disabled_by_feature_profile(const StringName &p_property);
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
protected:
|
|
|
|
static void _bind_methods();
|
|
|
|
void _notification(int p_what);
|
|
|
|
|
|
|
|
public:
|
|
|
|
static void add_inspector_plugin(const Ref<EditorInspectorPlugin> &p_plugin);
|
|
|
|
static void remove_inspector_plugin(const Ref<EditorInspectorPlugin> &p_plugin);
|
|
|
|
static void cleanup_plugins();
|
|
|
|
|
2018-07-14 23:15:42 +02:00
|
|
|
static EditorProperty *instantiate_property_editor(Object *p_object, Variant::Type p_type, const String &p_path, PropertyHint p_hint, const String &p_hint_text, int p_usage);
|
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
void set_undo_redo(UndoRedo *p_undo_redo);
|
|
|
|
|
|
|
|
String get_selected_path() const;
|
|
|
|
|
|
|
|
void update_tree();
|
|
|
|
void update_property(const String &p_prop);
|
|
|
|
|
|
|
|
void refresh();
|
|
|
|
|
|
|
|
void edit(Object *p_object);
|
2018-05-17 23:02:16 +02:00
|
|
|
Object *get_edited_object();
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
void set_keying(bool p_active);
|
|
|
|
void set_read_only(bool p_read_only);
|
|
|
|
|
|
|
|
bool is_capitalize_paths_enabled() const;
|
|
|
|
void set_enable_capitalize_paths(bool p_capitalize);
|
|
|
|
void set_autoclear(bool p_enable);
|
|
|
|
|
|
|
|
void set_show_categories(bool p_show);
|
|
|
|
void set_use_doc_hints(bool p_enable);
|
2019-07-22 12:03:57 +02:00
|
|
|
void set_hide_script(bool p_hide);
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
void set_use_filter(bool p_use);
|
|
|
|
void register_text_enter(Node *p_line_edit);
|
|
|
|
|
|
|
|
void set_use_folding(bool p_enable);
|
2018-06-14 20:36:38 +02:00
|
|
|
bool is_using_folding();
|
2018-05-15 22:12:35 +02:00
|
|
|
|
|
|
|
void collapse_all_folding();
|
|
|
|
void expand_all_folding();
|
|
|
|
|
|
|
|
void set_scroll_offset(int p_offset);
|
|
|
|
int get_scroll_offset() const;
|
|
|
|
|
2018-07-19 23:58:15 +02:00
|
|
|
void set_property_prefix(const String &p_prefix);
|
|
|
|
String get_property_prefix() const;
|
|
|
|
|
|
|
|
void set_object_class(const String &p_class);
|
|
|
|
String get_object_class() const;
|
|
|
|
|
2019-01-25 19:14:56 +01:00
|
|
|
void set_sub_inspector(bool p_enable);
|
2018-07-19 00:37:17 +02:00
|
|
|
|
2018-05-15 22:12:35 +02:00
|
|
|
EditorInspector();
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif // INSPECTOR_H
|