| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004 | /*************************************************************************//*  editor_properties_array_dict.cpp                                     *//*************************************************************************//*                       This file is part of:                           *//*                           GODOT ENGINE                                *//*                      https://godotengine.org                          *//*************************************************************************//* Copyright (c) 2007-2019 Juan Linietsky, Ariel Manzur.                 *//* Copyright (c) 2014-2019 Godot Engine contributors (cf. AUTHORS.md)    *//*                                                                       *//* 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.                *//*************************************************************************/#include "editor_properties_array_dict.h"#include "core/io/marshalls.h"#include "editor/editor_scale.h"#include "editor_properties.h"bool EditorPropertyArrayObject::_set(const StringName &p_name, const Variant &p_value) {	String pn = p_name;	if (pn.begins_with("indices")) {		int idx = pn.get_slicec('/', 1).to_int();		array.set(idx, p_value);		return true;	}	return false;}bool EditorPropertyArrayObject::_get(const StringName &p_name, Variant &r_ret) const {	String pn = p_name;	if (pn.begins_with("indices")) {		int idx = pn.get_slicec('/', 1).to_int();		bool valid;		r_ret = array.get(idx, &valid);		if (r_ret.get_type() == Variant::OBJECT && Object::cast_to<EncodedObjectAsID>(r_ret)) {			r_ret = Object::cast_to<EncodedObjectAsID>(r_ret)->get_object_id();		}		return valid;	}	return false;}void EditorPropertyArrayObject::set_array(const Variant &p_array) {	array = p_array;}Variant EditorPropertyArrayObject::get_array() {	return array;}EditorPropertyArrayObject::EditorPropertyArrayObject() {}///////////////////bool EditorPropertyDictionaryObject::_set(const StringName &p_name, const Variant &p_value) {	String pn = p_name;	if (pn == "new_item_key") {		new_item_key = p_value;		return true;	}	if (pn == "new_item_value") {		new_item_value = p_value;		return true;	}	if (pn.begins_with("indices")) {		int idx = pn.get_slicec('/', 1).to_int();		Variant key = dict.get_key_at_index(idx);		dict[key] = p_value;		return true;	}	return false;}bool EditorPropertyDictionaryObject::_get(const StringName &p_name, Variant &r_ret) const {	String pn = p_name;	if (pn == "new_item_key") {		r_ret = new_item_key;		return true;	}	if (pn == "new_item_value") {		r_ret = new_item_value;		return true;	}	if (pn.begins_with("indices")) {		int idx = pn.get_slicec('/', 1).to_int();		Variant key = dict.get_key_at_index(idx);		r_ret = dict[key];		if (r_ret.get_type() == Variant::OBJECT && Object::cast_to<EncodedObjectAsID>(r_ret)) {			r_ret = Object::cast_to<EncodedObjectAsID>(r_ret)->get_object_id();		}		return true;	}	return false;}void EditorPropertyDictionaryObject::set_dict(const Dictionary &p_dict) {	dict = p_dict;}Dictionary EditorPropertyDictionaryObject::get_dict() {	return dict;}void EditorPropertyDictionaryObject::set_new_item_key(const Variant &p_new_item) {	new_item_key = p_new_item;}Variant EditorPropertyDictionaryObject::get_new_item_key() {	return new_item_key;}void EditorPropertyDictionaryObject::set_new_item_value(const Variant &p_new_item) {	new_item_value = p_new_item;}Variant EditorPropertyDictionaryObject::get_new_item_value() {	return new_item_value;}EditorPropertyDictionaryObject::EditorPropertyDictionaryObject() {}///////////////////// ARRAY ///////////////////////////void EditorPropertyArray::_property_changed(const String &p_prop, Variant p_value, const String &p_name, bool changing) {	if (p_prop.begins_with("indices")) {		int idx = p_prop.get_slice("/", 1).to_int();		Variant array = object->get_array();		array.set(idx, p_value);		emit_changed(get_edited_property(), array, "", true);		if (array.get_type() == Variant::ARRAY) {			array = array.call("duplicate"); //dupe, so undo/redo works better		}		object->set_array(array);	}}void EditorPropertyArray::_change_type(Object *p_button, int p_index) {	Button *button = Object::cast_to<Button>(p_button);	Rect2 rect = button->get_global_rect();	change_type->set_as_minsize();	change_type->set_global_position(rect.position + rect.size - Vector2(change_type->get_combined_minimum_size().x, 0));	change_type->popup();	changing_type_idx = p_index;}void EditorPropertyArray::_change_type_menu(int p_index) {	Variant value;	Variant::CallError ce;	value = Variant::construct(Variant::Type(p_index), NULL, 0, ce);	Variant array = object->get_array();	array.set(changing_type_idx, value);	emit_changed(get_edited_property(), array, "", true);	if (array.get_type() == Variant::ARRAY) {		array = array.call("duplicate"); //dupe, so undo/redo works better	}	object->set_array(array);	update_property();}void EditorPropertyArray::_object_id_selected(const String &p_property, ObjectID p_id) {	emit_signal("object_id_selected", p_property, p_id);}void EditorPropertyArray::update_property() {	Variant array = get_edited_object()->get(get_edited_property());	String arrtype = "";	switch (array_type) {		case Variant::ARRAY: {			arrtype = "Array";		} break;		// arrays		case Variant::POOL_BYTE_ARRAY: {			arrtype = "PoolByteArray";		} break;		case Variant::POOL_INT_ARRAY: {			arrtype = "PoolIntArray";		} break;		case Variant::POOL_REAL_ARRAY: {			arrtype = "PoolFloatArray";		} break;		case Variant::POOL_STRING_ARRAY: {			arrtype = "PoolStringArray";		} break;		case Variant::POOL_VECTOR2_ARRAY: {			arrtype = "PoolVector2Array";		} break;		case Variant::POOL_VECTOR3_ARRAY: {			arrtype = "PoolVector3Array";		} break;		case Variant::POOL_COLOR_ARRAY: {			arrtype = "PoolColorArray";		} break;		default: {}	}	if (array.get_type() == Variant::NIL) {		edit->set_text(String("(Nil) ") + arrtype);		edit->set_pressed(false);		if (vbox) {			memdelete(vbox);		}		return;	}	edit->set_text(arrtype + " (size " + itos(array.call("size")) + ")");#ifdef TOOLS_ENABLED	bool unfolded = get_edited_object()->editor_is_section_unfolded(get_edited_property());	if (edit->is_pressed() != unfolded) {		edit->set_pressed(unfolded);	}	if (unfolded) {		updating = true;		if (!vbox) {			vbox = memnew(VBoxContainer);			add_child(vbox);			set_bottom_editor(vbox);			HBoxContainer *hbc = memnew(HBoxContainer);			vbox->add_child(hbc);			Label *label = memnew(Label(TTR("Size: ")));			label->set_h_size_flags(SIZE_EXPAND_FILL);			hbc->add_child(label);			length = memnew(EditorSpinSlider);			length->set_step(1);			length->set_max(1000000);			length->set_h_size_flags(SIZE_EXPAND_FILL);			hbc->add_child(length);			length->connect("value_changed", this, "_length_changed");			page_hb = memnew(HBoxContainer);			vbox->add_child(page_hb);			label = memnew(Label(TTR("Page: ")));			label->set_h_size_flags(SIZE_EXPAND_FILL);			page_hb->add_child(label);			page = memnew(EditorSpinSlider);			page->set_step(1);			page_hb->add_child(page);			page->set_h_size_flags(SIZE_EXPAND_FILL);			page->connect("value_changed", this, "_page_changed");		} else {			//bye bye children of the box			while (vbox->get_child_count() > 2) {				memdelete(vbox->get_child(2));			}		}		int len = array.call("size");		length->set_value(len);		int pages = MAX(0, len - 1) / page_len + 1;		page->set_max(pages);		page_idx = MIN(page_idx, pages - 1);		page->set_value(page_idx);		page_hb->set_visible(pages > 1);		int offset = page_idx * page_len;		int amount = MIN(len - offset, page_len);		if (array.get_type() == Variant::ARRAY) {			array = array.call("duplicate");		}		object->set_array(array);		for (int i = 0; i < amount; i++) {			String prop_name = "indices/" + itos(i + offset);			EditorProperty *prop = NULL;			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::OBJECT && Object::cast_to<EncodedObjectAsID>(value)) {				EditorPropertyObjectID *editor = memnew(EditorPropertyObjectID);				editor->setup("Object");				prop = editor;			} else {				prop = EditorInspector::instantiate_property_editor(NULL, value_type, "", subtype_hint, subtype_hint_string, 0);			}			prop->set_object_and_property(object.ptr(), prop_name);			prop->set_label(itos(i + offset));			prop->set_selectable(false);			prop->connect("property_changed", this, "_property_changed");			prop->connect("object_id_selected", this, "_object_id_selected");			if (array.get_type() == Variant::ARRAY) {				HBoxContainer *hb = memnew(HBoxContainer);				vbox->add_child(hb);				hb->add_child(prop);				prop->set_h_size_flags(SIZE_EXPAND_FILL);				if (subtype == Variant::NIL) {					Button *edit = memnew(Button);					edit->set_icon(get_icon("Edit", "EditorIcons"));					hb->add_child(edit);					edit->connect("pressed", this, "_change_type", varray(edit, i + offset));				}			} else {				vbox->add_child(prop);			}			prop->update_property();		}		updating = false;	} else {		if (vbox) {			set_bottom_editor(NULL);			memdelete(vbox);			vbox = NULL;		}	}#endif}void EditorPropertyArray::_notification(int p_what) {	if (p_what == NOTIFICATION_ENTER_TREE || p_what == NOTIFICATION_THEME_CHANGED) {	}}void EditorPropertyArray::_edit_pressed() {	Variant array = get_edited_object()->get(get_edited_property());	if (!array.is_array()) {		Variant::CallError ce;		array = Variant::construct(array_type, NULL, 0, ce);		get_edited_object()->set(get_edited_property(), array);	}	get_edited_object()->editor_set_section_unfold(get_edited_property(), edit->is_pressed());	update_property();}void EditorPropertyArray::_page_changed(double p_page) {	if (updating)		return;	page_idx = p_page;	update_property();}void EditorPropertyArray::_length_changed(double p_page) {	if (updating)		return;	Variant array = object->get_array();	int previous_size = array.call("size");	array.call("resize", int(p_page));	if (array.get_type() == Variant::ARRAY) {		if (subtype != Variant::NIL) {			int size = array.call("size");			for (int i = previous_size; i < size; i++) {				if (array.get(i).get_type() == Variant::NIL) {					Variant::CallError ce;					array.set(i, Variant::construct(subtype, NULL, 0, ce));				}			}		}		array = array.call("duplicate"); //dupe, so undo/redo works better	} else {		int size = array.call("size");		// Pool*Array don't initialize their elements, have to do it manually		for (int i = previous_size; i < size; i++) {			Variant::CallError ce;			array.set(i, Variant::construct(array.get(i).get_type(), NULL, 0, ce));		}	}	emit_changed(get_edited_property(), array, "", false);	object->set_array(array);	update_property();}void EditorPropertyArray::setup(Variant::Type p_array_type, const String &p_hint_string) {	array_type = p_array_type;	if (array_type == Variant::ARRAY && !p_hint_string.empty()) {		int hint_subtype_seperator = p_hint_string.find(":");		if (hint_subtype_seperator >= 0) {			String subtype_string = p_hint_string.substr(0, hint_subtype_seperator);			int slash_pos = subtype_string.find("/");			if (slash_pos >= 0) {				subtype_hint = PropertyHint(subtype_string.substr(slash_pos + 1, subtype_string.size() - slash_pos - 1).to_int());				subtype_string = subtype_string.substr(0, slash_pos);			}			subtype_hint_string = p_hint_string.substr(hint_subtype_seperator + 1, p_hint_string.size() - hint_subtype_seperator - 1);			subtype = Variant::Type(subtype_string.to_int());		}	}}void EditorPropertyArray::_bind_methods() {	ClassDB::bind_method("_edit_pressed", &EditorPropertyArray::_edit_pressed);	ClassDB::bind_method("_page_changed", &EditorPropertyArray::_page_changed);	ClassDB::bind_method("_length_changed", &EditorPropertyArray::_length_changed);	ClassDB::bind_method("_property_changed", &EditorPropertyArray::_property_changed, DEFVAL(String()), DEFVAL(false));	ClassDB::bind_method("_change_type", &EditorPropertyArray::_change_type);	ClassDB::bind_method("_change_type_menu", &EditorPropertyArray::_change_type_menu);	ClassDB::bind_method("_object_id_selected", &EditorPropertyArray::_object_id_selected);}EditorPropertyArray::EditorPropertyArray() {	object.instance();	page_idx = 0;	page_len = 10;	edit = memnew(Button);	edit->set_flat(true);	edit->set_h_size_flags(SIZE_EXPAND_FILL);	edit->set_clip_text(true);	edit->connect("pressed", this, "_edit_pressed");	edit->set_toggle_mode(true);	add_child(edit);	add_focusable(edit);	vbox = NULL;	page = NULL;	length = NULL;	updating = false;	change_type = memnew(PopupMenu);	add_child(change_type);	change_type->connect("id_pressed", this, "_change_type_menu");	changing_type_idx = -1;	for (int i = 0; i < Variant::VARIANT_MAX; i++) {		String type = Variant::get_type_name(Variant::Type(i));		change_type->add_item(type, i);	}	changing_type_idx = -1;	subtype = Variant::NIL;	subtype_hint = PROPERTY_HINT_NONE;	subtype_hint_string = "";}///////////////////// DICTIONARY ///////////////////////////void EditorPropertyDictionary::_property_changed(const String &p_prop, Variant p_value, const String &p_name, bool changing) {	if (p_prop == "new_item_key") {		object->set_new_item_key(p_value);	} else if (p_prop == "new_item_value") {		object->set_new_item_value(p_value);	} else if (p_prop.begins_with("indices")) {		int idx = p_prop.get_slice("/", 1).to_int();		Dictionary dict = object->get_dict();		Variant key = dict.get_key_at_index(idx);		dict[key] = p_value;		emit_changed(get_edited_property(), dict, "", true);		dict = dict.duplicate(); //dupe, so undo/redo works better		object->set_dict(dict);	}}void EditorPropertyDictionary::_change_type(Object *p_button, int p_index) {	Button *button = Object::cast_to<Button>(p_button);	Rect2 rect = button->get_global_rect();	change_type->set_as_minsize();	change_type->set_global_position(rect.position + rect.size - Vector2(change_type->get_combined_minimum_size().x, 0));	change_type->popup();	changing_type_idx = p_index;}void EditorPropertyDictionary::_add_key_value() {	// Do not allow nil as valid key. I experienced errors with this	if (object->get_new_item_key().get_type() == Variant::NIL) {		return;	}	Dictionary dict = object->get_dict();	dict[object->get_new_item_key()] = object->get_new_item_value();	object->set_new_item_key(Variant());	object->set_new_item_value(Variant());	emit_changed(get_edited_property(), dict, "", false);	dict = dict.duplicate(); //dupe, so undo/redo works better	object->set_dict(dict);	update_property();}void EditorPropertyDictionary::_change_type_menu(int p_index) {	if (changing_type_idx < 0) {		Variant value;		Variant::CallError ce;		value = Variant::construct(Variant::Type(p_index), NULL, 0, ce);		if (changing_type_idx == -1) {			object->set_new_item_key(value);		} else {			object->set_new_item_value(value);		}		update_property();		return;	}	Dictionary dict = object->get_dict();	if (p_index < Variant::VARIANT_MAX) {		Variant value;		Variant::CallError ce;		value = Variant::construct(Variant::Type(p_index), NULL, 0, ce);		Variant key = dict.get_key_at_index(changing_type_idx);		dict[key] = value;	} else {		Variant key = dict.get_key_at_index(changing_type_idx);		dict.erase(key);	}	emit_changed(get_edited_property(), dict, "", false);	dict = dict.duplicate(); //dupe, so undo/redo works better	object->set_dict(dict);	update_property();}void EditorPropertyDictionary::update_property() {	Variant updated_val = get_edited_object()->get(get_edited_property());	if (updated_val.get_type() == Variant::NIL) {		edit->set_text("Dictionary (Nil)"); //This provides symmetry with the array property.		edit->set_pressed(false);		if (vbox) {			memdelete(vbox);		}		return;	}	Dictionary dict = updated_val;	edit->set_text("Dictionary (size " + itos(dict.size()) + ")");#ifdef TOOLS_ENABLED	bool unfolded = get_edited_object()->editor_is_section_unfolded(get_edited_property());	if (edit->is_pressed() != unfolded) {		edit->set_pressed(unfolded);	}	if (unfolded) {		updating = true;		if (!vbox) {			vbox = memnew(VBoxContainer);			add_child(vbox);			set_bottom_editor(vbox);			page_hb = memnew(HBoxContainer);			vbox->add_child(page_hb);			Label *label = memnew(Label(TTR("Page: ")));			label->set_h_size_flags(SIZE_EXPAND_FILL);			page_hb->add_child(label);			page = memnew(EditorSpinSlider);			page->set_step(1);			page_hb->add_child(page);			page->set_h_size_flags(SIZE_EXPAND_FILL);			page->connect("value_changed", this, "_page_changed");		} else {			// Queue childs for deletion, delete immediately might cause errors.			for (int i = 1; i < vbox->get_child_count(); i++) {				vbox->get_child(i)->queue_delete();			}		}		int len = dict.size();		int pages = MAX(0, len - 1) / page_len + 1;		page->set_max(pages);		page_idx = MIN(page_idx, pages - 1);		page->set_value(page_idx);		page_hb->set_visible(pages > 1);		int offset = page_idx * page_len;		int amount = MIN(len - offset, page_len);		dict = dict.duplicate();		object->set_dict(dict);		VBoxContainer *add_vbox = NULL;		for (int i = 0; i < amount + 2; i++) {			String prop_name;			Variant key;			Variant value;			if (i < amount) {				prop_name = "indices/" + itos(i + offset);				key = dict.get_key_at_index(i + offset);				value = dict.get_value_at_index(i + offset);			} else if (i == amount) {				prop_name = "new_item_key";				value = object->get_new_item_key();			} else if (i == amount + 1) {				prop_name = "new_item_value";				value = object->get_new_item_value();			}			EditorProperty *prop = NULL;			switch (value.get_type()) {				case Variant::NIL: {					prop = memnew(EditorPropertyNil);				} break;				// atomic types				case Variant::BOOL: {					prop = memnew(EditorPropertyCheck);				} break;				case Variant::INT: {					EditorPropertyInteger *editor = memnew(EditorPropertyInteger);					editor->setup(-100000, 100000, 1, true, true);					prop = editor;				} break;				case Variant::REAL: {					EditorPropertyFloat *editor = memnew(EditorPropertyFloat);					editor->setup(-100000, 100000, 0.001, true, false, true, true);					prop = editor;				} break;				case Variant::STRING: {					prop = memnew(EditorPropertyText);				} break;				// math types				case Variant::VECTOR2: {					EditorPropertyVector2 *editor = memnew(EditorPropertyVector2);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::RECT2: {					EditorPropertyRect2 *editor = memnew(EditorPropertyRect2);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::VECTOR3: {					EditorPropertyVector3 *editor = memnew(EditorPropertyVector3);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::TRANSFORM2D: {					EditorPropertyTransform2D *editor = memnew(EditorPropertyTransform2D);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::PLANE: {					EditorPropertyPlane *editor = memnew(EditorPropertyPlane);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::QUAT: {					EditorPropertyQuat *editor = memnew(EditorPropertyQuat);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::AABB: {					EditorPropertyAABB *editor = memnew(EditorPropertyAABB);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::BASIS: {					EditorPropertyBasis *editor = memnew(EditorPropertyBasis);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				case Variant::TRANSFORM: {					EditorPropertyTransform *editor = memnew(EditorPropertyTransform);					editor->setup(-100000, 100000, 0.001, true);					prop = editor;				} break;				// misc types				case Variant::COLOR: {					prop = memnew(EditorPropertyColor);				} break;				case Variant::NODE_PATH: {					prop = memnew(EditorPropertyNodePath);				} break;				case Variant::_RID: {					prop = memnew(EditorPropertyNil);				} break;				case Variant::OBJECT: {					if (Object::cast_to<EncodedObjectAsID>(value)) {						EditorPropertyObjectID *editor = memnew(EditorPropertyObjectID);						editor->setup("Object");						prop = editor;					} else {						EditorPropertyResource *editor = memnew(EditorPropertyResource);						editor->setup("Resource");						prop = editor;					}				} break;				case Variant::DICTIONARY: {					prop = memnew(EditorPropertyDictionary);				} break;				case Variant::ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::ARRAY);					prop = editor;				} break;				// arrays				case Variant::POOL_BYTE_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_BYTE_ARRAY);					prop = editor;				} break;				case Variant::POOL_INT_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_INT_ARRAY);					prop = editor;				} break;				case Variant::POOL_REAL_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_REAL_ARRAY);					prop = editor;				} break;				case Variant::POOL_STRING_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_STRING_ARRAY);					prop = editor;				} break;				case Variant::POOL_VECTOR2_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_VECTOR2_ARRAY);					prop = editor;				} break;				case Variant::POOL_VECTOR3_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_VECTOR3_ARRAY);					prop = editor;				} break;				case Variant::POOL_COLOR_ARRAY: {					EditorPropertyArray *editor = memnew(EditorPropertyArray);					editor->setup(Variant::POOL_COLOR_ARRAY);					prop = editor;				} break;				default: {}			}			if (i == amount) {				PanelContainer *pc = memnew(PanelContainer);				vbox->add_child(pc);				Ref<StyleBoxFlat> flat;				flat.instance();				for (int j = 0; j < 4; j++) {					flat->set_default_margin(Margin(j), 2 * EDSCALE);				}				flat->set_bg_color(get_color("prop_subsection", "Editor"));				pc->add_style_override("panel", flat);				add_vbox = memnew(VBoxContainer);				pc->add_child(add_vbox);			}			prop->set_object_and_property(object.ptr(), prop_name);			int change_index = 0;			if (i < amount) {				String cs = key.get_construct_string();				prop->set_label(key.get_construct_string());				prop->set_tooltip(cs);				change_index = i + offset;			} else if (i == amount) {				prop->set_label(TTR("New Key:"));				change_index = -1;			} else if (i == amount + 1) {				prop->set_label(TTR("New Value:"));				change_index = -2;			}			prop->set_selectable(false);			prop->connect("property_changed", this, "_property_changed");			prop->connect("object_id_selected", this, "_object_id_selected");			HBoxContainer *hb = memnew(HBoxContainer);			if (add_vbox) {				add_vbox->add_child(hb);			} else {				vbox->add_child(hb);			}			hb->add_child(prop);			prop->set_h_size_flags(SIZE_EXPAND_FILL);			Button *edit = memnew(Button);			edit->set_icon(get_icon("Edit", "EditorIcons"));			hb->add_child(edit);			edit->connect("pressed", this, "_change_type", varray(edit, change_index));			prop->update_property();			if (i == amount + 1) {				Button *butt_add_item = memnew(Button);				butt_add_item->set_text(TTR("Add Key/Value Pair"));				butt_add_item->connect("pressed", this, "_add_key_value");				add_vbox->add_child(butt_add_item);			}		}		updating = false;	} else {		if (vbox) {			set_bottom_editor(NULL);			memdelete(vbox);			vbox = NULL;		}	}#endif}void EditorPropertyDictionary::_object_id_selected(const String &p_property, ObjectID p_id) {	emit_signal("object_id_selected", p_property, p_id);}void EditorPropertyDictionary::_notification(int p_what) {	if (p_what == NOTIFICATION_ENTER_TREE || p_what == NOTIFICATION_THEME_CHANGED) {	}}void EditorPropertyDictionary::_edit_pressed() {	Variant prop_val = get_edited_object()->get(get_edited_property());	if (prop_val.get_type() == Variant::NIL) {		Variant::CallError ce;		prop_val = Variant::construct(Variant::DICTIONARY, NULL, 0, ce);		get_edited_object()->set(get_edited_property(), prop_val);	}	get_edited_object()->editor_set_section_unfold(get_edited_property(), edit->is_pressed());	update_property();}void EditorPropertyDictionary::_page_changed(double p_page) {	if (updating)		return;	page_idx = p_page;	update_property();}void EditorPropertyDictionary::_bind_methods() {	ClassDB::bind_method("_edit_pressed", &EditorPropertyDictionary::_edit_pressed);	ClassDB::bind_method("_page_changed", &EditorPropertyDictionary::_page_changed);	ClassDB::bind_method("_property_changed", &EditorPropertyDictionary::_property_changed, DEFVAL(String()), DEFVAL(false));	ClassDB::bind_method("_change_type", &EditorPropertyDictionary::_change_type);	ClassDB::bind_method("_change_type_menu", &EditorPropertyDictionary::_change_type_menu);	ClassDB::bind_method("_add_key_value", &EditorPropertyDictionary::_add_key_value);	ClassDB::bind_method("_object_id_selected", &EditorPropertyDictionary::_object_id_selected);}EditorPropertyDictionary::EditorPropertyDictionary() {	object.instance();	page_idx = 0;	page_len = 10;	edit = memnew(Button);	edit->set_flat(true);	edit->set_h_size_flags(SIZE_EXPAND_FILL);	edit->set_clip_text(true);	edit->connect("pressed", this, "_edit_pressed");	edit->set_toggle_mode(true);	add_child(edit);	add_focusable(edit);	vbox = NULL;	page = NULL;	updating = false;	change_type = memnew(PopupMenu);	add_child(change_type);	change_type->connect("id_pressed", this, "_change_type_menu");	changing_type_idx = -1;	for (int i = 0; i < Variant::VARIANT_MAX; i++) {		String type = Variant::get_type_name(Variant::Type(i));		change_type->add_item(type, i);	}	change_type->add_separator();	change_type->add_item(TTR("Remove Item"), Variant::VARIANT_MAX);	changing_type_idx = -1;}
 |