| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*  dictionary.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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #include "dictionary.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-16 08:04:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | #include "core/templates/hash_map.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-07 19:33:38 -03:00
										 |  |  | #include "core/templates/safe_refcount.h"
 | 
					
						
							|  |  |  | #include "core/variant/variant.h"
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:03:37 -03:00
										 |  |  | // required in this order by VariantInternal, do not remove this comment.
 | 
					
						
							|  |  |  | #include "core/object/class_db.h"
 | 
					
						
							|  |  |  | #include "core/object/object.h"
 | 
					
						
							|  |  |  | #include "core/variant/type_info.h"
 | 
					
						
							|  |  |  | #include "core/variant/variant_internal.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct DictionaryPrivate { | 
					
						
							|  |  |  | 	SafeRefCount refcount; | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 	Variant *read_only = nullptr; // If enabled, a pointer is used to a temporary value that is used to return read-only values.
 | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	HashMap<Variant, Variant, VariantHasher, StringLikeVariantComparator> variant_map; | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | void Dictionary::get_key_list(List<Variant> *p_keys) const { | 
					
						
							| 
									
										
										
										
											2020-12-15 12:04:21 +00:00
										 |  |  | 	if (_p->variant_map.is_empty()) { | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							|  |  |  | 		p_keys->push_back(E.key); | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-19 17:57:44 -03:00
										 |  |  | Variant Dictionary::get_key_at_index(int p_index) const { | 
					
						
							|  |  |  | 	int index = 0; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							| 
									
										
										
										
											2018-05-19 17:57:44 -03:00
										 |  |  | 		if (index == p_index) { | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 			return E.key; | 
					
						
							| 
									
										
										
										
											2018-05-19 17:57:44 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		index++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return Variant(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant Dictionary::get_value_at_index(int p_index) const { | 
					
						
							|  |  |  | 	int index = 0; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							| 
									
										
										
										
											2018-05-19 17:57:44 -03:00
										 |  |  | 		if (index == p_index) { | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 			return E.value; | 
					
						
							| 
									
										
										
										
											2018-05-19 17:57:44 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		index++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return Variant(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | Variant &Dictionary::operator[](const Variant &p_key) { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 	if (unlikely(_p->read_only)) { | 
					
						
							|  |  |  | 		if (p_key.get_type() == Variant::STRING_NAME) { | 
					
						
							|  |  |  | 			const StringName *sn = VariantInternal::get_string_name(&p_key); | 
					
						
							| 
									
										
										
										
											2023-03-10 17:26:34 +01:00
										 |  |  | 			const String &key = sn->operator String(); | 
					
						
							| 
									
										
										
										
											2023-03-11 01:08:23 +01:00
										 |  |  | 			if (likely(_p->variant_map.has(key))) { | 
					
						
							| 
									
										
										
										
											2023-03-10 17:26:34 +01:00
										 |  |  | 				*_p->read_only = _p->variant_map[key]; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				*_p->read_only = Variant(); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-03-11 01:08:23 +01:00
										 |  |  | 		} else if (likely(_p->variant_map.has(p_key))) { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 			*_p->read_only = _p->variant_map[p_key]; | 
					
						
							| 
									
										
										
										
											2023-03-11 01:08:23 +01:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			*_p->read_only = Variant(); | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return *_p->read_only; | 
					
						
							| 
									
										
										
										
											2021-07-01 16:03:37 -03:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 		if (p_key.get_type() == Variant::STRING_NAME) { | 
					
						
							|  |  |  | 			const StringName *sn = VariantInternal::get_string_name(&p_key); | 
					
						
							|  |  |  | 			return _p->variant_map[sn->operator String()]; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			return _p->variant_map[p_key]; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-07-01 16:03:37 -03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Variant &Dictionary::operator[](const Variant &p_key) const { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	// Will not insert key, so no conversion is necessary.
 | 
					
						
							|  |  |  | 	return _p->variant_map[p_key]; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-14 14:29:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | const Variant *Dictionary::getptr(const Variant &p_key) const { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	HashMap<Variant, Variant, VariantHasher, StringLikeVariantComparator>::ConstIterator E(_p->variant_map.find(p_key)); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (!E) { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		return nullptr; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	return &E->value; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | Variant *Dictionary::getptr(const Variant &p_key) { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	HashMap<Variant, Variant, VariantHasher, StringLikeVariantComparator>::Iterator E(_p->variant_map.find(p_key)); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (!E) { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		return nullptr; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 	if (unlikely(_p->read_only != nullptr)) { | 
					
						
							|  |  |  | 		*_p->read_only = E->value; | 
					
						
							|  |  |  | 		return _p->read_only; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return &E->value; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant Dictionary::get_valid(const Variant &p_key) const { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	HashMap<Variant, Variant, VariantHasher, StringLikeVariantComparator>::ConstIterator E(_p->variant_map.find(p_key)); | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (!E) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return Variant(); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	return E->value; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-30 00:04:32 -04:00
										 |  |  | Variant Dictionary::get(const Variant &p_key, const Variant &p_default) const { | 
					
						
							|  |  |  | 	const Variant *result = getptr(p_key); | 
					
						
							|  |  |  | 	if (!result) { | 
					
						
							|  |  |  | 		return p_default; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return *result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | int Dictionary::size() const { | 
					
						
							|  |  |  | 	return _p->variant_map.size(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-14 14:29:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 12:04:21 +00:00
										 |  |  | bool Dictionary::is_empty() const { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return !_p->variant_map.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Dictionary::has(const Variant &p_key) const { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	return _p->variant_map.has(p_key); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-04 18:37:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool Dictionary::has_all(const Array &p_keys) const { | 
					
						
							|  |  |  | 	for (int i = 0; i < p_keys.size(); i++) { | 
					
						
							|  |  |  | 		if (!has(p_keys[i])) { | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 19:08:27 +02:00
										 |  |  | Variant Dictionary::find_key(const Variant &p_value) const { | 
					
						
							|  |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							|  |  |  | 		if (E.value == p_value) { | 
					
						
							|  |  |  | 			return E.key; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return Variant(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 19:19:05 +02:00
										 |  |  | bool Dictionary::erase(const Variant &p_key) { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 	ERR_FAIL_COND_V_MSG(_p->read_only, false, "Dictionary is in read-only state."); | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	return _p->variant_map.erase(p_key); | 
					
						
							| 
									
										
										
										
											2018-07-18 23:07:31 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | bool Dictionary::operator==(const Dictionary &p_dictionary) const { | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 	return recursive_equal(p_dictionary, 0); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-06 16:20:41 -04:00
										 |  |  | bool Dictionary::operator!=(const Dictionary &p_dictionary) const { | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 	return !recursive_equal(p_dictionary, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Dictionary::recursive_equal(const Dictionary &p_dictionary, int recursion_count) const { | 
					
						
							|  |  |  | 	// Cheap checks
 | 
					
						
							|  |  |  | 	if (_p == p_dictionary._p) { | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (_p->variant_map.size() != p_dictionary._p->variant_map.size()) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Heavy O(n) check
 | 
					
						
							|  |  |  | 	if (recursion_count > MAX_RECURSION) { | 
					
						
							|  |  |  | 		ERR_PRINT("Max recursion reached"); | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	recursion_count++; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &this_E : _p->variant_map) { | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 		HashMap<Variant, Variant, VariantHasher, StringLikeVariantComparator>::ConstIterator other_E(p_dictionary._p->variant_map.find(this_E.key)); | 
					
						
							| 
									
										
										
										
											2023-03-07 19:13:00 -08:00
										 |  |  | 		if (!other_E || !this_E.value.hash_compare(other_E->value, recursion_count, false)) { | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2018-10-06 16:20:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void Dictionary::_ref(const Dictionary &p_from) const { | 
					
						
							|  |  |  | 	//make a copy first (thread safe)
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (!p_from._p->refcount.ref()) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return; // couldn't copy
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	//if this is the same, unreference the other one
 | 
					
						
							|  |  |  | 	if (p_from._p == _p) { | 
					
						
							|  |  |  | 		_p->refcount.unref(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (_p) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		_unref(); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_p = p_from._p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Dictionary::clear() { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 	ERR_FAIL_COND_MSG(_p->read_only, "Dictionary is in read-only state."); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_p->variant_map.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-04 18:46:55 +02:00
										 |  |  | void Dictionary::merge(const Dictionary &p_dictionary, bool p_overwrite) { | 
					
						
							|  |  |  | 	for (const KeyValue<Variant, Variant> &E : p_dictionary._p->variant_map) { | 
					
						
							|  |  |  | 		if (p_overwrite || !has(E.key)) { | 
					
						
							|  |  |  | 			this->operator[](E.key) = E.value; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void Dictionary::_unref() const { | 
					
						
							| 
									
										
										
										
											2023-09-09 16:11:33 +02:00
										 |  |  | 	ERR_FAIL_NULL(_p); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (_p->refcount.unref()) { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 		if (_p->read_only) { | 
					
						
							|  |  |  | 			memdelete(_p->read_only); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		memdelete(_p); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 	_p = nullptr; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-14 14:29:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | uint32_t Dictionary::hash() const { | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 	return recursive_hash(0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint32_t Dictionary::recursive_hash(int recursion_count) const { | 
					
						
							|  |  |  | 	if (recursion_count > MAX_RECURSION) { | 
					
						
							|  |  |  | 		ERR_PRINT("Max recursion reached"); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-18 16:20:55 +02:00
										 |  |  | 	uint32_t h = hash_murmur3_one_32(Variant::DICTIONARY); | 
					
						
							| 
									
										
										
										
											2015-12-13 15:20:58 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 	recursion_count++; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							| 
									
										
										
										
											2022-06-18 16:20:55 +02:00
										 |  |  | 		h = hash_murmur3_one_32(E.key.recursive_hash(recursion_count), h); | 
					
						
							|  |  |  | 		h = hash_murmur3_one_32(E.value.recursive_hash(recursion_count), h); | 
					
						
							| 
									
										
										
										
											2015-12-13 15:20:58 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-18 16:20:55 +02:00
										 |  |  | 	return hash_fmix32(h); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Array Dictionary::keys() const { | 
					
						
							| 
									
										
										
										
											2017-07-23 12:23:18 -03:00
										 |  |  | 	Array varr; | 
					
						
							| 
									
										
										
										
											2020-12-15 12:04:21 +00:00
										 |  |  | 	if (_p->variant_map.is_empty()) { | 
					
						
							| 
									
										
										
										
											2017-07-23 12:23:18 -03:00
										 |  |  | 		return varr; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-23 12:23:18 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-10 18:12:20 +02:00
										 |  |  | 	varr.resize(size()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 	int i = 0; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							|  |  |  | 		varr[i] = E.key; | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 		i++; | 
					
						
							| 
									
										
										
										
											2017-07-23 12:23:18 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return varr; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-05 14:20:52 +02:00
										 |  |  | Array Dictionary::values() const { | 
					
						
							|  |  |  | 	Array varr; | 
					
						
							| 
									
										
										
										
											2020-12-15 12:04:21 +00:00
										 |  |  | 	if (_p->variant_map.is_empty()) { | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 		return varr; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-10 18:12:20 +02:00
										 |  |  | 	varr.resize(size()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 	int i = 0; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							|  |  |  | 		varr[i] = E.value; | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 		i++; | 
					
						
							| 
									
										
										
										
											2016-06-05 14:20:52 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-08 14:16:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-05 14:20:52 +02:00
										 |  |  | 	return varr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | const Variant *Dictionary::next(const Variant *p_key) const { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 	if (p_key == nullptr) { | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 		// caller wants to get the first element
 | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 		if (_p->variant_map.begin()) { | 
					
						
							|  |  |  | 			return &_p->variant_map.begin()->key; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		return nullptr; | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-12-05 21:46:47 -05:00
										 |  |  | 	HashMap<Variant, Variant, VariantHasher, StringLikeVariantComparator>::Iterator E = _p->variant_map.find(*p_key); | 
					
						
							| 
									
										
										
										
											2017-11-05 15:27:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 	if (!E) { | 
					
						
							|  |  |  | 		return nullptr; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	++E; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (E) { | 
					
						
							|  |  |  | 		return &E->key; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 	return nullptr; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-09 21:16:08 +02:00
										 |  |  | Dictionary Dictionary::duplicate(bool p_deep) const { | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 	return recursive_duplicate(p_deep, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-22 11:07:48 +02:00
										 |  |  | void Dictionary::make_read_only() { | 
					
						
							|  |  |  | 	if (_p->read_only == nullptr) { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 		_p->read_only = memnew(Variant); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | bool Dictionary::is_read_only() const { | 
					
						
							|  |  |  | 	return _p->read_only != nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | Dictionary Dictionary::recursive_duplicate(bool p_deep, int recursion_count) const { | 
					
						
							| 
									
										
										
										
											2017-01-11 08:53:31 -03:00
										 |  |  | 	Dictionary n; | 
					
						
							| 
									
										
										
										
											2016-08-02 19:11:05 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 	if (recursion_count > MAX_RECURSION) { | 
					
						
							|  |  |  | 		ERR_PRINT("Max recursion reached"); | 
					
						
							|  |  |  | 		return n; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_deep) { | 
					
						
							|  |  |  | 		recursion_count++; | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 		for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							|  |  |  | 			n[E.key.recursive_duplicate(true, recursion_count)] = E.value.recursive_duplicate(true, recursion_count); | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2022-05-08 10:09:19 +02:00
										 |  |  | 		for (const KeyValue<Variant, Variant> &E : _p->variant_map) { | 
					
						
							|  |  |  | 			n[E.key] = E.value; | 
					
						
							| 
									
										
										
										
											2020-02-01 07:04:14 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-02 19:11:05 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void Dictionary::operator=(const Dictionary &p_dictionary) { | 
					
						
							| 
									
										
										
										
											2022-05-16 15:44:43 +02:00
										 |  |  | 	if (this == &p_dictionary) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_ref(p_dictionary); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 01:57:29 +02:00
										 |  |  | const void *Dictionary::id() const { | 
					
						
							| 
									
										
										
										
											2021-12-30 13:14:09 +08:00
										 |  |  | 	return _p; | 
					
						
							| 
									
										
										
										
											2019-04-20 01:57:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | Dictionary::Dictionary(const Dictionary &p_from) { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 	_p = nullptr; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_ref(p_from); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 08:53:31 -03:00
										 |  |  | Dictionary::Dictionary() { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_p = memnew(DictionaryPrivate); | 
					
						
							|  |  |  | 	_p->refcount.init(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-14 14:29:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | Dictionary::~Dictionary() { | 
					
						
							|  |  |  | 	_unref(); | 
					
						
							|  |  |  | } |