summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--editor/editor_properties_array_dict.cpp236
-rw-r--r--editor/editor_properties_array_dict.h26
2 files changed, 145 insertions, 117 deletions
diff --git a/editor/editor_properties_array_dict.cpp b/editor/editor_properties_array_dict.cpp
index 0a9d35fe64..52bdc182b2 100644
--- a/editor/editor_properties_array_dict.cpp
+++ b/editor/editor_properties_array_dict.cpp
@@ -202,22 +202,16 @@ void EditorPropertyArray::_property_changed(const String &p_property, Variant p_
p_value = Variant(); // `EditorResourcePicker` resets to `Ref<Resource>()`. See GH-82716.
}
- int index;
- if (p_property.begins_with("metadata/")) {
- index = p_property.get_slice("/", 2).to_int();
- } else {
- index = p_property.get_slice("/", 1).to_int();
- }
+ int index = p_property.get_slice("/", 1).to_int();
Variant array = object->get_array().duplicate();
array.set(index, p_value);
- object->set_array(array);
- emit_changed(get_edited_property(), array, "", true);
+ emit_changed(get_edited_property(), array, p_name, p_changing);
}
-void EditorPropertyArray::_change_type(Object *p_button, int p_index) {
+void EditorPropertyArray::_change_type(Object *p_button, int p_slot_index) {
Button *button = Object::cast_to<Button>(p_button);
- changing_type_index = p_index;
+ changing_type_index = slots[p_slot_index].index;
Rect2 rect = button->get_screen_rect();
change_type->reset_size();
change_type->set_position(rect.get_end() - Vector2(change_type->get_contents_minimum_size().x, 0));
@@ -244,6 +238,48 @@ void EditorPropertyArray::_object_id_selected(const StringName &p_property, Obje
emit_signal(SNAME("object_id_selected"), p_property, p_id);
}
+void EditorPropertyArray::create_new_property_slot() {
+ int idx = slots.size();
+ HBoxContainer *hbox = memnew(HBoxContainer);
+
+ Button *reorder_button = memnew(Button);
+ reorder_button->set_icon(get_editor_theme_icon(SNAME("TripleBar")));
+ reorder_button->set_default_cursor_shape(Control::CURSOR_MOVE);
+ reorder_button->set_disabled(is_read_only());
+ reorder_button->connect(SNAME("gui_input"), callable_mp(this, &EditorPropertyArray::_reorder_button_gui_input));
+ reorder_button->connect(SNAME("button_up"), callable_mp(this, &EditorPropertyArray::_reorder_button_up));
+ reorder_button->connect(SNAME("button_down"), callable_mp(this, &EditorPropertyArray::_reorder_button_down).bind(idx));
+
+ hbox->add_child(reorder_button);
+ EditorProperty *prop = memnew(EditorPropertyNil);
+ hbox->add_child(prop);
+
+ bool is_untyped_array = object->get_array().get_type() == Variant::ARRAY && subtype == Variant::NIL;
+
+ if (is_untyped_array) {
+ Button *edit_btn = memnew(Button);
+ edit_btn->set_icon(get_editor_theme_icon(SNAME("Edit")));
+ edit_btn->set_disabled(is_read_only());
+ edit_btn->connect("pressed", callable_mp(this, &EditorPropertyArray::_change_type).bind(edit_btn, idx));
+ hbox->add_child(edit_btn);
+ } else {
+ Button *remove_btn = memnew(Button);
+ remove_btn->set_icon(get_editor_theme_icon(SNAME("Remove")));
+ remove_btn->set_disabled(is_read_only());
+ remove_btn->connect("pressed", callable_mp(this, &EditorPropertyArray::_remove_pressed).bind(idx));
+ hbox->add_child(remove_btn);
+ }
+ property_vbox->add_child(hbox);
+
+ Slot slot;
+ slot.prop = prop;
+ slot.object = object;
+ slot.container = hbox;
+ slot.reorder_button = reorder_button;
+ slot.set_index(idx + page_index * page_length);
+ slots.push_back(slot);
+}
+
void EditorPropertyArray::update_property() {
Variant array = get_edited_property_value();
@@ -276,7 +312,6 @@ void EditorPropertyArray::update_property() {
int size = array.call("size");
int max_page = MAX(0, size - 1) / page_length;
page_index = MIN(page_index, max_page);
- int offset = page_index * page_length;
edit->set_text(vformat(TTR("%s (size %s)"), array_type_name, itos(size)));
@@ -326,16 +361,9 @@ void EditorPropertyArray::update_property() {
paginator = memnew(EditorPaginator);
paginator->connect("page_changed", callable_mp(this, &EditorPropertyArray::_page_changed));
vbox->add_child(paginator);
- } else {
- // Bye bye children of the box.
- for (int i = property_vbox->get_child_count() - 1; i >= 0; i--) {
- Node *child = property_vbox->get_child(i);
- if (child == reorder_selected_element_hbox) {
- continue; // Don't remove the property that the user is moving.
- }
- child->queue_free(); // Button still needed after pressed is called.
- property_vbox->remove_child(child);
+ for (int i = 0; i < page_length; i++) {
+ create_new_property_slot();
}
}
@@ -345,81 +373,46 @@ void EditorPropertyArray::update_property() {
paginator->update(page_index, max_page);
paginator->set_visible(max_page > 0);
- int amount = MIN(size - offset, page_length);
- for (int i = 0; i < amount; i++) {
- bool reorder_is_from_current_page = reorder_from_index / page_length == page_index;
- if (reorder_is_from_current_page && i == reorder_from_index % page_length) {
- // Don't duplicate the property that the user is moving.
- continue;
- }
- if (!reorder_is_from_current_page && i == reorder_to_index % page_length) {
- // Don't create the property the moving property will take the place of,
- // e.g. (if page_length == 20) don't create element 20 if dragging an item from
- // the first page to the second page because element 20 would become element 19.
+ for (Slot &slot : slots) {
+ bool slot_visible = &slot != &reorder_slot && slot.index < size;
+ slot.container->set_visible(slot_visible);
+ // If not visible no need to update it
+ if (!slot_visible) {
continue;
}
- HBoxContainer *hbox = memnew(HBoxContainer);
- property_vbox->add_child(hbox);
-
- Button *reorder_button = memnew(Button);
- reorder_button->set_icon(get_editor_theme_icon(SNAME("TripleBar")));
- reorder_button->set_default_cursor_shape(Control::CURSOR_MOVE);
- reorder_button->set_disabled(is_read_only());
- reorder_button->connect("gui_input", callable_mp(this, &EditorPropertyArray::_reorder_button_gui_input));
- reorder_button->connect("button_down", callable_mp(this, &EditorPropertyArray::_reorder_button_down).bind(i + offset));
- reorder_button->connect("button_up", callable_mp(this, &EditorPropertyArray::_reorder_button_up));
- hbox->add_child(reorder_button);
-
- String prop_name = "indices/" + itos(i + offset);
+ int idx = slot.index;
+ Variant::Type value_type = subtype;
- EditorProperty *prop = nullptr;
- Variant value = array.get(i + offset);
- Variant::Type value_type = value.get_type();
-
- if (value_type == Variant::NIL && subtype != Variant::NIL) {
- value_type = subtype;
+ if (value_type == Variant::NIL) {
+ value_type = array.get(idx).get_type();
}
- if (value_type == Variant::OBJECT && Object::cast_to<EncodedObjectAsID>(value)) {
- EditorPropertyObjectID *editor = memnew(EditorPropertyObjectID);
- editor->setup("Object");
- prop = editor;
- } else {
- prop = EditorInspector::instantiate_property_editor(nullptr, value_type, "", subtype_hint, subtype_hint_string, PROPERTY_USAGE_NONE);
- }
-
- prop->set_object_and_property(object.ptr(), prop_name);
- prop->set_label(itos(i + offset));
- prop->set_selectable(false);
- prop->set_use_folding(is_using_folding());
- prop->connect("property_changed", callable_mp(this, &EditorPropertyArray::_property_changed));
- prop->connect("object_id_selected", callable_mp(this, &EditorPropertyArray::_object_id_selected));
- prop->set_h_size_flags(SIZE_EXPAND_FILL);
- prop->set_read_only(is_read_only());
- hbox->add_child(prop);
-
- bool is_untyped_array = array.get_type() == Variant::ARRAY && subtype == Variant::NIL;
-
- if (is_untyped_array) {
- Button *edit_btn = memnew(Button);
- edit_btn->set_icon(get_editor_theme_icon(SNAME("Edit")));
- hbox->add_child(edit_btn);
- edit_btn->set_disabled(is_read_only());
- edit_btn->connect("pressed", callable_mp(this, &EditorPropertyArray::_change_type).bind(edit_btn, i + offset));
- } else {
- Button *remove_btn = memnew(Button);
- remove_btn->set_icon(get_editor_theme_icon(SNAME("Remove")));
- remove_btn->set_disabled(is_read_only());
- remove_btn->connect("pressed", callable_mp(this, &EditorPropertyArray::_remove_pressed).bind(i + offset));
- hbox->add_child(remove_btn);
+ // Check if the editor property needs to be updated.
+ bool value_as_id = Object::cast_to<EncodedObjectAsID>(array.get(idx));
+ if (value_type != slot.type || (value_type == Variant::OBJECT && (value_as_id != slot.as_id))) {
+ slot.as_id = value_as_id;
+ slot.type = value_type;
+ EditorProperty *new_prop = nullptr;
+ if (value_type == Variant::OBJECT && value_as_id) {
+ EditorPropertyObjectID *editor = memnew(EditorPropertyObjectID);
+ editor->setup("Object");
+ new_prop = editor;
+ } else {
+ new_prop = EditorInspector::instantiate_property_editor(nullptr, value_type, "", subtype_hint, subtype_hint_string, PROPERTY_USAGE_NONE);
+ }
+ new_prop->set_selectable(false);
+ new_prop->set_use_folding(is_using_folding());
+ new_prop->connect(SNAME("property_changed"), callable_mp(this, &EditorPropertyArray::_property_changed));
+ new_prop->connect(SNAME("object_id_selected"), callable_mp(this, &EditorPropertyArray::_object_id_selected));
+ new_prop->set_h_size_flags(SIZE_EXPAND_FILL);
+ new_prop->set_read_only(is_read_only());
+ slot.prop->call_deferred("add_sibling", new_prop);
+ slot.prop->call_deferred("queue_free");
+ slot.prop = new_prop;
+ slot.set_index(idx);
}
-
- prop->update_property();
- }
-
- if (reorder_to_index % page_length > 0) {
- property_vbox->move_child(property_vbox->get_child(0), reorder_to_index % page_length);
+ slot.prop->update_property();
}
updating = false;
@@ -430,13 +423,14 @@ void EditorPropertyArray::update_property() {
memdelete(container);
button_add_item = nullptr;
container = nullptr;
+ slots.clear();
}
}
}
-void EditorPropertyArray::_remove_pressed(int p_index) {
+void EditorPropertyArray::_remove_pressed(int p_slot_index) {
Variant array = object->get_array().duplicate();
- array.call("remove_at", p_index);
+ array.call("remove_at", slots[p_slot_index].index);
emit_changed(get_edited_property(), array, "", false);
update_property();
@@ -579,6 +573,27 @@ void EditorPropertyArray::_page_changed(int p_page) {
return;
}
page_index = p_page;
+ int i = p_page * page_length;
+
+ if (reorder_slot.index < 0) {
+ for (Slot &slot : slots) {
+ slot.set_index(i);
+ i++;
+ }
+ } else {
+ int reorder_from_page = reorder_slot.index / page_length;
+ if (reorder_from_page < p_page) {
+ i++;
+ }
+ for (Slot &slot : slots) {
+ if (slot.index != reorder_slot.index) {
+ slot.set_index(i);
+ i++;
+ } else if (i == reorder_slot.index) {
+ i++;
+ }
+ }
+ }
update_property();
}
@@ -620,7 +635,7 @@ void EditorPropertyArray::setup(Variant::Type p_array_type, const String &p_hint
}
void EditorPropertyArray::_reorder_button_gui_input(const Ref<InputEvent> &p_event) {
- if (reorder_from_index < 0 || is_read_only()) {
+ if (reorder_slot.index < 0 || is_read_only()) {
return;
}
@@ -645,26 +660,25 @@ void EditorPropertyArray::_reorder_button_gui_input(const Ref<InputEvent> &p_eve
reorder_mouse_y_delta -= required_y_distance * direction;
reorder_to_index += direction;
+
+ property_vbox->move_child(reorder_slot.container, reorder_to_index % page_length);
+
if ((direction < 0 && reorder_to_index % page_length == page_length - 1) || (direction > 0 && reorder_to_index % page_length == 0)) {
// Automatically move to the next/previous page.
_page_changed(page_index + direction);
}
- property_vbox->move_child(reorder_selected_element_hbox, reorder_to_index % page_length);
// Ensure the moving element is visible.
- InspectorDock::get_inspector_singleton()->ensure_control_visible(reorder_selected_element_hbox);
+ InspectorDock::get_inspector_singleton()->ensure_control_visible(reorder_slot.container);
}
}
}
-void EditorPropertyArray::_reorder_button_down(int p_index) {
+void EditorPropertyArray::_reorder_button_down(int p_slot_index) {
if (is_read_only()) {
return;
}
-
- reorder_from_index = p_index;
- reorder_to_index = p_index;
- reorder_selected_element_hbox = Object::cast_to<HBoxContainer>(property_vbox->get_child(p_index % page_length));
- reorder_selected_button = Object::cast_to<Button>(reorder_selected_element_hbox->get_child(0));
+ reorder_slot = slots[p_slot_index];
+ reorder_to_index = reorder_slot.index;
// Ideally it'd to be able to show the mouse but I had issues with
// Control's `mouse_exit()`/`mouse_entered()` signals not getting called.
Input::get_singleton()->set_mouse_mode(Input::MOUSE_MODE_CAPTURED);
@@ -675,29 +689,27 @@ void EditorPropertyArray::_reorder_button_up() {
return;
}
- if (reorder_from_index != reorder_to_index) {
+ if (reorder_slot.index != reorder_to_index) {
// Move the element.
Variant array = object->get_array().duplicate();
- Variant value_to_move = array.get(reorder_from_index);
- array.call("remove_at", reorder_from_index);
+ property_vbox->move_child(reorder_slot.container, reorder_slot.index % page_length);
+ Variant value_to_move = array.get(reorder_slot.index);
+ array.call("remove_at", reorder_slot.index);
array.call("insert", reorder_to_index, value_to_move);
+ reorder_slot.index = reorder_slot.index % page_length + page_index * page_length;
emit_changed(get_edited_property(), array, "", false);
- update_property();
}
- reorder_from_index = -1;
- reorder_to_index = -1;
- reorder_mouse_y_delta = 0.0f;
-
Input::get_singleton()->set_mouse_mode(Input::MOUSE_MODE_VISIBLE);
- ERR_FAIL_NULL(reorder_selected_button);
- reorder_selected_button->warp_mouse(reorder_selected_button->get_size() / 2.0f);
-
- reorder_selected_element_hbox = nullptr;
- reorder_selected_button = nullptr;
+ ERR_FAIL_NULL(reorder_slot.reorder_button);
+ reorder_slot.reorder_button->warp_mouse(reorder_slot.reorder_button->get_size() / 2.0f);
+ reorder_to_index = -1;
+ reorder_mouse_y_delta = 0.0f;
+ reorder_slot = Slot();
+ _page_changed(page_index);
}
void EditorPropertyArray::_bind_methods() {
diff --git a/editor/editor_properties_array_dict.h b/editor/editor_properties_array_dict.h
index 2ca1931f85..f157da00c2 100644
--- a/editor/editor_properties_array_dict.h
+++ b/editor/editor_properties_array_dict.h
@@ -81,6 +81,23 @@ public:
class EditorPropertyArray : public EditorProperty {
GDCLASS(EditorPropertyArray, EditorProperty);
+ struct Slot {
+ Ref<EditorPropertyArrayObject> object;
+ HBoxContainer *container = nullptr;
+ int index = -1;
+ Variant::Type type = Variant::VARIANT_MAX;
+ bool as_id = false;
+ EditorProperty *prop = nullptr;
+ Button *reorder_button = nullptr;
+
+ void set_index(int p_idx) {
+ String prop_name = "indices/" + itos(p_idx);
+ prop->set_object_and_property(object.ptr(), prop_name);
+ prop->set_label(itos(p_idx));
+ index = p_idx;
+ }
+ };
+
PopupMenu *change_type = nullptr;
int page_length = 20;
@@ -96,13 +113,11 @@ class EditorPropertyArray : public EditorProperty {
Variant::Type subtype;
PropertyHint subtype_hint;
String subtype_hint_string;
+ LocalVector<Slot> slots;
- int reorder_from_index = -1;
+ Slot reorder_slot;
int reorder_to_index = -1;
float reorder_mouse_y_delta = 0.0f;
- HBoxContainer *reorder_selected_element_hbox = nullptr;
- Button *reorder_selected_button = nullptr;
-
void initialize_array(Variant &p_array);
void _page_changed(int p_page);
@@ -110,6 +125,7 @@ class EditorPropertyArray : public EditorProperty {
void _reorder_button_gui_input(const Ref<InputEvent> &p_event);
void _reorder_button_down(int p_index);
void _reorder_button_up();
+ void create_new_property_slot();
protected:
Ref<EditorPropertyArrayObject> object;
@@ -124,7 +140,7 @@ protected:
virtual void _length_changed(double p_page);
virtual void _edit_pressed();
virtual void _property_changed(const String &p_property, Variant p_value, const String &p_name = "", bool p_changing = false);
- virtual void _change_type(Object *p_button, int p_index);
+ virtual void _change_type(Object *p_button, int p_slot_index);
virtual void _change_type_menu(int p_index);
virtual void _object_id_selected(const StringName &p_property, ObjectID p_id);