2017-12-07 04:36:34 +08:00
/*************************************************************************/
2014-02-10 09:10:30 +08:00
/* object.h */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 20:16:55 +08:00
/* https://godotengine.org */
2014-02-10 09:10:30 +08:00
/*************************************************************************/
2022-01-04 04:27:34 +08:00
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
2014-02-10 09:10:30 +08: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-01-05 07:50:27 +08:00
2014-02-10 09:10:30 +08:00
# ifndef OBJECT_H
# define OBJECT_H
2021-06-19 23:58:49 +08:00
# include "core/extension/gdnative_interface.h"
2022-03-09 21:58:40 +08:00
# include "core/object/message_queue.h"
2020-11-08 06:33:38 +08:00
# include "core/object/object_id.h"
2018-09-12 00:13:45 +08:00
# include "core/os/rw_lock.h"
2020-11-08 06:33:38 +08:00
# include "core/os/spin_lock.h"
# include "core/templates/hash_map.h"
# include "core/templates/list.h"
# include "core/templates/map.h"
2021-02-11 02:22:13 +08:00
# include "core/templates/safe_refcount.h"
2020-11-08 06:33:38 +08:00
# include "core/templates/set.h"
# include "core/templates/vmap.h"
# include "core/variant/callable_bind.h"
# include "core/variant/variant.h"
2014-02-10 09:10:30 +08:00
enum PropertyHint {
PROPERTY_HINT_NONE , ///< no hint provided.
Fix editor suffixes and degrees conversion
* Functions to convert to/from degrees are all gone. Conversion is done by the editor.
* Use PROPERTY_HINT_ANGLE instead of PROPERTY_HINT_RANGE to edit radian angles in degrees.
* Added possibility to add suffixes to range properties, use "min,max[,step][,suffix:<something>]" example "0,100,1,suffix:m"
* In general, can add suffixes for EditorSpinSlider
Not covered by this PR, will have to be addressed by future ones:
* Ability to switch radians/degrees in the inspector for angle properties (if actually wanted).
* Animations previously made will most likely break, need to add a way to make old ones compatible.
* Only added a "px" suffix to 2D position and a "m" one to 3D position, someone needs to go through the rest of the engine and add all remaining suffixes.
* Likely also need to track down usage of EditorSpinSlider outside properties to add suffixes to it too.
2021-06-30 03:42:12 +08:00
PROPERTY_HINT_RANGE , ///< hint_text = "min,max[,step][,or_greater][,or_lesser][,noslider][,radians][,degrees][,exp][,suffix:<keyword>] range.
2014-02-10 09:10:30 +08:00
PROPERTY_HINT_ENUM , ///< hint_text= "val1,val2,val3,etc"
2021-07-05 04:42:23 +08:00
PROPERTY_HINT_ENUM_SUGGESTION , ///< hint_text= "val1,val2,val3,etc"
2018-05-16 04:12:35 +08:00
PROPERTY_HINT_EXP_EASING , /// exponential easing function (Math::ease) use "attenuation" hint string to revert (flip h), "full" to also include in/out. (ie: "attenuation,inout")
2014-02-10 09:10:30 +08:00
PROPERTY_HINT_LENGTH , ///< hint_text= "length" (as integer)
PROPERTY_HINT_KEY_ACCEL , ///< hint_text= "length" (as integer)
PROPERTY_HINT_FLAGS , ///< hint_text= "flag1,flag2,etc" (as bit flags)
2017-01-11 09:20:57 +08:00
PROPERTY_HINT_LAYERS_2D_RENDER ,
PROPERTY_HINT_LAYERS_2D_PHYSICS ,
2021-03-09 03:56:33 +08:00
PROPERTY_HINT_LAYERS_2D_NAVIGATION ,
2017-01-11 09:20:57 +08:00
PROPERTY_HINT_LAYERS_3D_RENDER ,
PROPERTY_HINT_LAYERS_3D_PHYSICS ,
2021-03-09 03:56:33 +08:00
PROPERTY_HINT_LAYERS_3D_NAVIGATION ,
2015-08-30 08:09:11 +08:00
PROPERTY_HINT_FILE , ///< a file path must be passed, hint_text (optionally) is a filter "*.png,*.wav,*.doc,"
2017-09-22 11:58:29 +08:00
PROPERTY_HINT_DIR , ///< a directory path must be passed
2014-02-10 09:10:30 +08:00
PROPERTY_HINT_GLOBAL_FILE , ///< a file path must be passed, hint_text (optionally) is a filter "*.png,*.wav,*.doc,"
2017-09-22 11:58:29 +08:00
PROPERTY_HINT_GLOBAL_DIR , ///< a directory path must be passed
2014-02-10 09:10:30 +08:00
PROPERTY_HINT_RESOURCE_TYPE , ///< a resource object type
2015-08-30 08:09:11 +08:00
PROPERTY_HINT_MULTILINE_TEXT , ///< used for string properties that can contain multiple lines
2018-08-17 09:50:12 +08:00
PROPERTY_HINT_PLACEHOLDER_TEXT , ///< used to set a placeholder text for string properties
2014-02-10 09:10:30 +08:00
PROPERTY_HINT_COLOR_NO_ALPHA , ///< used for ignoring alpha component when editing a color
PROPERTY_HINT_IMAGE_COMPRESS_LOSSY ,
PROPERTY_HINT_IMAGE_COMPRESS_LOSSLESS ,
2016-05-23 06:28:37 +08:00
PROPERTY_HINT_OBJECT_ID ,
2016-08-03 06:11:05 +08:00
PROPERTY_HINT_TYPE_STRING , ///< a type string, the hint is the base type to choose
PROPERTY_HINT_NODE_PATH_TO_EDITED_NODE , ///< so something else can provide this (used in scripts)
2016-08-24 06:29:07 +08:00
PROPERTY_HINT_METHOD_OF_VARIANT_TYPE , ///< a method of a type
PROPERTY_HINT_METHOD_OF_BASE_TYPE , ///< a method of a base type
PROPERTY_HINT_METHOD_OF_INSTANCE , ///< a method of an instance
PROPERTY_HINT_METHOD_OF_SCRIPT , ///< a method of a script & base
PROPERTY_HINT_PROPERTY_OF_VARIANT_TYPE , ///< a property of a type
PROPERTY_HINT_PROPERTY_OF_BASE_TYPE , ///< a property of a base type
PROPERTY_HINT_PROPERTY_OF_INSTANCE , ///< a property of an instance
PROPERTY_HINT_PROPERTY_OF_SCRIPT , ///< a property of a script & base
2017-08-18 21:59:31 +08:00
PROPERTY_HINT_OBJECT_TOO_BIG , ///< object is too big to send
2018-06-28 07:50:25 +08:00
PROPERTY_HINT_NODE_PATH_VALID_TYPES ,
2019-04-20 02:54:33 +08:00
PROPERTY_HINT_SAVE_FILE , ///< a file path must be passed, hint_text (optionally) is a filter "*.png,*.wav,*.doc,". This opens a save dialog
2020-02-13 01:24:06 +08:00
PROPERTY_HINT_INT_IS_OBJECTID ,
2020-04-21 06:06:00 +08:00
PROPERTY_HINT_ARRAY_TYPE ,
2021-08-24 01:53:27 +08:00
PROPERTY_HINT_INT_IS_POINTER ,
2021-09-23 19:08:50 +08:00
PROPERTY_HINT_LOCALE_ID ,
2022-03-04 21:04:59 +08:00
PROPERTY_HINT_LOCALIZABLE_STRING ,
2014-02-10 09:10:30 +08:00
PROPERTY_HINT_MAX ,
2018-01-14 01:13:15 +08:00
// When updating PropertyHint, also sync the hardcoded list in VisualScriptEditorVariableEdit
2014-02-10 09:10:30 +08:00
} ;
enum PropertyUsageFlags {
2021-06-18 07:10:18 +08:00
PROPERTY_USAGE_NONE = 0 ,
2017-03-05 23:44:50 +08:00
PROPERTY_USAGE_STORAGE = 1 ,
PROPERTY_USAGE_EDITOR = 2 ,
PROPERTY_USAGE_NETWORK = 4 ,
PROPERTY_USAGE_EDITOR_HELPER = 8 ,
PROPERTY_USAGE_CHECKABLE = 16 , //used for editing global variables
PROPERTY_USAGE_CHECKED = 32 , //used for editing global variables
PROPERTY_USAGE_INTERNATIONALIZED = 64 , //hint for internationalized strings
PROPERTY_USAGE_GROUP = 128 , //used for grouping props in the editor
PROPERTY_USAGE_CATEGORY = 256 ,
2020-04-08 09:51:52 +08:00
PROPERTY_USAGE_SUBGROUP = 512 ,
2017-03-05 23:44:50 +08:00
PROPERTY_USAGE_NO_INSTANCE_STATE = 2048 ,
PROPERTY_USAGE_RESTART_IF_CHANGED = 4096 ,
PROPERTY_USAGE_SCRIPT_VARIABLE = 8192 ,
PROPERTY_USAGE_STORE_IF_NULL = 16384 ,
PROPERTY_USAGE_ANIMATE_AS_TRIGGER = 32768 ,
PROPERTY_USAGE_UPDATE_ALL_IF_MODIFIED = 65536 ,
2017-08-06 20:32:52 +08:00
PROPERTY_USAGE_SCRIPT_DEFAULT_VALUE = 1 < < 17 ,
2017-08-24 06:10:32 +08:00
PROPERTY_USAGE_CLASS_IS_ENUM = 1 < < 18 ,
PROPERTY_USAGE_NIL_IS_VARIANT = 1 < < 19 ,
2017-12-07 04:16:25 +08:00
PROPERTY_USAGE_INTERNAL = 1 < < 20 ,
2018-01-12 02:50:33 +08:00
PROPERTY_USAGE_DO_NOT_SHARE_ON_DUPLICATE = 1 < < 21 , // If the object is duplicated also this property will be duplicated
2018-09-29 07:32:40 +08:00
PROPERTY_USAGE_HIGH_END_GFX = 1 < < 22 ,
2019-01-15 02:46:56 +08:00
PROPERTY_USAGE_NODE_PATH_FROM_SCENE_ROOT = 1 < < 23 ,
2019-02-22 07:49:42 +08:00
PROPERTY_USAGE_RESOURCE_NOT_PERSISTENT = 1 < < 24 ,
2019-07-25 15:11:41 +08:00
PROPERTY_USAGE_KEYING_INCREMENTS = 1 < < 25 , // Used in inspector to increment property when keyed in animation player
2020-02-28 19:27:04 +08:00
PROPERTY_USAGE_DEFERRED_SET_RESOURCE = 1 < < 26 , // when loading, the resource for this property can be set at the end of loading
2020-06-12 19:16:14 +08:00
PROPERTY_USAGE_EDITOR_INSTANTIATE_OBJECT = 1 < < 27 , // For Object properties, instantiate them when creating in editor.
2021-02-18 00:44:49 +08:00
PROPERTY_USAGE_EDITOR_BASIC_SETTING = 1 < < 28 , //for project or editor settings, show when basic settings are selected
2021-08-31 16:48:45 +08:00
PROPERTY_USAGE_READ_ONLY = 1 < < 29 , // Mark a property as read-only in the inspector.
PROPERTY_USAGE_ARRAY = 1 < < 30 , // Used in the inspector to group properties as elements of an array.
2017-03-05 23:44:50 +08:00
PROPERTY_USAGE_DEFAULT = PROPERTY_USAGE_STORAGE | PROPERTY_USAGE_EDITOR | PROPERTY_USAGE_NETWORK ,
PROPERTY_USAGE_DEFAULT_INTL = PROPERTY_USAGE_STORAGE | PROPERTY_USAGE_EDITOR | PROPERTY_USAGE_NETWORK | PROPERTY_USAGE_INTERNATIONALIZED ,
2021-11-04 06:06:17 +08:00
PROPERTY_USAGE_NO_EDITOR = PROPERTY_USAGE_STORAGE | PROPERTY_USAGE_NETWORK ,
2014-02-10 09:10:30 +08:00
} ;
2021-08-13 22:46:14 +08:00
# define ADD_SIGNAL(m_signal) ::ClassDB::add_signal(get_class_static(), m_signal)
# define ADD_PROPERTY(m_property, m_setter, m_getter) ::ClassDB::add_property(get_class_static(), m_property, _scs_create(m_setter), _scs_create(m_getter))
# define ADD_PROPERTYI(m_property, m_setter, m_getter, m_index) ::ClassDB::add_property(get_class_static(), m_property, _scs_create(m_setter), _scs_create(m_getter), m_index)
# define ADD_PROPERTY_DEFAULT(m_property, m_default) ::ClassDB::set_property_default_value(get_class_static(), m_property, m_default)
# define ADD_GROUP(m_name, m_prefix) ::ClassDB::add_property_group(get_class_static(), m_name, m_prefix)
2021-11-09 04:53:41 +08:00
# define ADD_GROUP_INDENT(m_name, m_prefix, m_depth) ::ClassDB::add_property_group(get_class_static(), m_name, m_prefix, m_depth)
2021-08-13 22:46:14 +08:00
# define ADD_SUBGROUP(m_name, m_prefix) ::ClassDB::add_property_subgroup(get_class_static(), m_name, m_prefix)
2021-11-09 04:53:41 +08:00
# define ADD_SUBGROUP_INDENT(m_name, m_prefix, m_depth) ::ClassDB::add_property_subgroup(get_class_static(), m_name, m_prefix, m_depth)
2021-08-13 02:26:47 +08:00
# define ADD_LINKED_PROPERTY(m_property, m_linked_property) ::ClassDB::add_linked_property(get_class_static(), m_property, m_linked_property)
2014-02-10 09:10:30 +08:00
2021-08-31 16:48:45 +08:00
# define ADD_ARRAY_COUNT(m_label, m_count_property, m_count_property_setter, m_count_property_getter, m_prefix) ClassDB::add_property_array_count(get_class_static(), m_label, m_count_property, _scs_create(m_count_property_setter), _scs_create(m_count_property_getter), m_prefix)
# define ADD_ARRAY_COUNT_WITH_USAGE_FLAGS(m_label, m_count_property, m_count_property_setter, m_count_property_getter, m_prefix, m_property_usage_flags) ClassDB::add_property_array_count(get_class_static(), m_label, m_count_property, _scs_create(m_count_property_setter), _scs_create(m_count_property_getter), m_prefix, m_property_usage_flags)
# define ADD_ARRAY(m_array_path, m_prefix) ClassDB::add_property_array(get_class_static(), m_array_path, m_prefix)
2014-02-10 09:10:30 +08:00
struct PropertyInfo {
2020-05-12 23:01:17 +08:00
Variant : : Type type = Variant : : NIL ;
2014-02-10 09:10:30 +08:00
String name ;
2021-08-13 22:46:14 +08:00
StringName class_name ; // For classes
2020-05-12 23:01:17 +08:00
PropertyHint hint = PROPERTY_HINT_NONE ;
2015-08-30 08:09:11 +08:00
String hint_string ;
2020-05-12 23:01:17 +08:00
uint32_t usage = PROPERTY_USAGE_DEFAULT ;
2014-02-10 09:10:30 +08:00
2021-08-13 02:26:47 +08:00
# ifdef TOOLS_ENABLED
Vector < String > linked_properties ;
# endif
2021-07-01 09:24:34 +08:00
_FORCE_INLINE_ PropertyInfo added_usage ( uint32_t p_fl ) const {
2017-03-05 23:44:50 +08:00
PropertyInfo pi = * this ;
pi . usage | = p_fl ;
return pi ;
}
2016-08-26 04:45:20 +08:00
operator Dictionary ( ) const ;
2017-03-05 23:44:50 +08:00
static PropertyInfo from_dict ( const Dictionary & p_dict ) ;
2016-08-26 04:45:20 +08:00
2020-05-12 23:01:17 +08:00
PropertyInfo ( ) { }
2017-09-15 00:54:37 +08:00
2021-07-01 09:24:34 +08:00
PropertyInfo ( const Variant : : Type p_type , const String p_name , const PropertyHint p_hint = PROPERTY_HINT_NONE , const String & p_hint_string = " " , const uint32_t p_usage = PROPERTY_USAGE_DEFAULT , const StringName & p_class_name = StringName ( ) ) :
2017-12-07 04:36:34 +08:00
type ( p_type ) ,
name ( p_name ) ,
hint ( p_hint ) ,
hint_string ( p_hint_string ) ,
usage ( p_usage ) {
2017-08-24 06:10:32 +08:00
if ( hint = = PROPERTY_HINT_RESOURCE_TYPE ) {
class_name = hint_string ;
} else {
class_name = p_class_name ;
}
}
2017-09-15 00:54:37 +08:00
2017-12-07 04:36:34 +08:00
PropertyInfo ( const StringName & p_class_name ) :
type ( Variant : : OBJECT ) ,
2020-05-12 23:01:17 +08:00
class_name ( p_class_name ) { }
2017-08-24 06:10:32 +08:00
2019-07-06 04:30:01 +08:00
bool operator = = ( const PropertyInfo & p_info ) const {
return ( ( type = = p_info . type ) & &
( name = = p_info . name ) & &
( class_name = = p_info . class_name ) & &
( hint = = p_info . hint ) & &
( hint_string = = p_info . hint_string ) & &
( usage = = p_info . usage ) ) ;
}
2017-03-05 23:44:50 +08:00
bool operator < ( const PropertyInfo & p_info ) const {
return name < p_info . name ;
2015-05-01 08:53:41 +08:00
}
2014-02-10 09:10:30 +08:00
} ;
2017-03-05 23:44:50 +08:00
Array convert_property_list ( const List < PropertyInfo > * p_list ) ;
2014-02-10 09:10:30 +08:00
struct MethodInfo {
String name ;
PropertyInfo return_val ;
2020-05-12 23:01:17 +08:00
uint32_t flags ; // NOLINT - prevent clang-tidy to assign method_bind.h constant here, it should stay in .cpp.
int id = 0 ;
2018-09-28 23:17:38 +08:00
List < PropertyInfo > arguments ;
Vector < Variant > default_arguments ;
2015-08-30 08:09:11 +08:00
2017-08-27 03:41:25 +08:00
inline bool operator = = ( const MethodInfo & p_method ) const { return id = = p_method . id ; }
2017-03-05 23:44:50 +08:00
inline bool operator < ( const MethodInfo & p_method ) const { return id = = p_method . id ? ( name < p_method . name ) : ( id < p_method . id ) ; }
2015-08-30 08:09:11 +08:00
2016-08-26 04:45:20 +08:00
operator Dictionary ( ) const ;
2017-03-05 23:44:50 +08:00
static MethodInfo from_dict ( const Dictionary & p_dict ) ;
2020-05-12 23:01:17 +08:00
2014-02-10 09:10:30 +08:00
MethodInfo ( ) ;
2017-03-05 23:44:50 +08:00
MethodInfo ( const String & p_name ) ;
MethodInfo ( const String & p_name , const PropertyInfo & p_param1 ) ;
MethodInfo ( const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 ) ;
MethodInfo ( const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 ) ;
MethodInfo ( const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 , const PropertyInfo & p_param4 ) ;
MethodInfo ( const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 , const PropertyInfo & p_param4 , const PropertyInfo & p_param5 ) ;
2014-02-10 09:10:30 +08:00
MethodInfo ( Variant : : Type ret ) ;
2017-03-05 23:44:50 +08:00
MethodInfo ( Variant : : Type ret , const String & p_name ) ;
MethodInfo ( Variant : : Type ret , const String & p_name , const PropertyInfo & p_param1 ) ;
MethodInfo ( Variant : : Type ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 ) ;
MethodInfo ( Variant : : Type ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 ) ;
MethodInfo ( Variant : : Type ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 , const PropertyInfo & p_param4 ) ;
MethodInfo ( Variant : : Type ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 , const PropertyInfo & p_param4 , const PropertyInfo & p_param5 ) ;
2017-08-29 13:15:46 +08:00
MethodInfo ( const PropertyInfo & p_ret , const String & p_name ) ;
MethodInfo ( const PropertyInfo & p_ret , const String & p_name , const PropertyInfo & p_param1 ) ;
MethodInfo ( const PropertyInfo & p_ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 ) ;
MethodInfo ( const PropertyInfo & p_ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 ) ;
MethodInfo ( const PropertyInfo & p_ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 , const PropertyInfo & p_param4 ) ;
MethodInfo ( const PropertyInfo & p_ret , const String & p_name , const PropertyInfo & p_param1 , const PropertyInfo & p_param2 , const PropertyInfo & p_param3 , const PropertyInfo & p_param4 , const PropertyInfo & p_param5 ) ;
2014-02-10 09:10:30 +08:00
} ;
2022-02-16 20:56:32 +08:00
// API used to extend in GDNative and other C compatible compiled languages.
2021-06-05 01:33:48 +08:00
class MethodBind ;
struct ObjectNativeExtension {
ObjectNativeExtension * parent = nullptr ;
2021-06-19 23:58:49 +08:00
List < ObjectNativeExtension * > children ;
2021-06-05 01:33:48 +08:00
StringName parent_class_name ;
StringName class_name ;
bool editor_class = false ;
2021-06-19 23:58:49 +08:00
GDNativeExtensionClassSet set ;
GDNativeExtensionClassGet get ;
GDNativeExtensionClassGetPropertyList get_property_list ;
GDNativeExtensionClassFreePropertyList free_property_list ;
GDNativeExtensionClassNotification notification ;
GDNativeExtensionClassToString to_string ;
GDNativeExtensionClassReference reference ;
GDNativeExtensionClassReference unreference ;
2022-03-10 15:17:38 +08:00
GDNativeExtensionClassGetRID get_rid ;
2021-06-05 01:33:48 +08:00
_FORCE_INLINE_ bool is_class ( const String & p_class ) const {
const ObjectNativeExtension * e = this ;
while ( e ) {
if ( p_class = = e - > class_name . operator String ( ) ) {
return true ;
}
e = e - > parent ;
}
return false ;
}
2021-06-19 23:58:49 +08:00
void * class_userdata = nullptr ;
GDNativeExtensionClassCreateInstance create_instance ;
GDNativeExtensionClassFreeInstance free_instance ;
GDNativeExtensionClassGetVirtual get_virtual ;
2021-06-05 01:33:48 +08:00
} ;
2022-03-10 15:17:38 +08:00
# define GDVIRTUAL_CALL(m_name, ...) _gdvirtual_##m_name##_call<false>(__VA_ARGS__)
# define GDVIRTUAL_CALL_PTR(m_obj, m_name, ...) m_obj->_gdvirtual_##m_name##_call<false>(__VA_ARGS__)
# define GDVIRTUAL_REQUIRED_CALL(m_name, ...) _gdvirtual_##m_name##_call<true>(__VA_ARGS__)
# define GDVIRTUAL_REQUIRED_CALL_PTR(m_obj, m_name, ...) m_obj->_gdvirtual_##m_name##_call<true>(__VA_ARGS__)
2021-08-22 09:52:44 +08:00
# ifdef DEBUG_METHODS_ENABLED
# define GDVIRTUAL_BIND(m_name, ...) ::ClassDB::add_virtual_method(get_class_static(), _gdvirtual_##m_name##_get_method_info(), true, sarray(__VA_ARGS__));
# else
# define GDVIRTUAL_BIND(m_name, ...)
# endif
2021-08-26 01:44:01 +08:00
# define GDVIRTUAL_IS_OVERRIDDEN(m_name) _gdvirtual_##m_name##_overridden()
# define GDVIRTUAL_IS_OVERRIDDEN_PTR(m_obj, m_name) m_obj->_gdvirtual_##m_name##_overridden()
2021-06-19 23:58:49 +08:00
2014-02-10 09:10:30 +08:00
/*
2022-02-16 20:56:32 +08:00
* The following is an incomprehensible blob of hacks and workarounds to
* compensate for many of the fallacies in C + + . As a plus , this macro pretty
* much alone defines the object model .
*/
2014-02-10 09:10:30 +08:00
2017-03-05 23:44:50 +08:00
# define REVERSE_GET_PROPERTY_LIST \
public : \
_FORCE_INLINE_ bool _is_gpl_reversed ( ) const { return true ; } ; \
\
2014-02-10 09:10:30 +08:00
private :
2017-03-05 23:44:50 +08:00
# define UNREVERSE_GET_PROPERTY_LIST \
public : \
_FORCE_INLINE_ bool _is_gpl_reversed ( ) const { return false ; } ; \
\
2014-02-10 09:10:30 +08:00
private :
2020-07-10 18:34:39 +08:00
# define GDCLASS(m_class, m_inherits) \
private : \
void operator = ( const m_class & p_rval ) { } \
mutable StringName _class_name ; \
2021-08-13 22:46:14 +08:00
friend class : : ClassDB ; \
2020-07-10 18:34:39 +08:00
\
public : \
virtual String get_class ( ) const override { \
2021-06-05 01:33:48 +08:00
if ( _get_extension ( ) ) { \
return _get_extension ( ) - > class_name . operator String ( ) ; \
} \
2020-07-10 18:34:39 +08:00
return String ( # m_class ) ; \
} \
virtual const StringName * _get_class_namev ( ) const override { \
2021-06-05 01:33:48 +08:00
if ( _get_extension ( ) ) { \
return & _get_extension ( ) - > class_name ; \
} \
2020-07-10 18:34:39 +08:00
if ( ! _class_name ) { \
_class_name = get_class_static ( ) ; \
} \
return & _class_name ; \
} \
static _FORCE_INLINE_ void * get_class_ptr_static ( ) { \
static int ptr ; \
return & ptr ; \
} \
static _FORCE_INLINE_ String get_class_static ( ) { \
return String ( # m_class ) ; \
} \
static _FORCE_INLINE_ String get_parent_class_static ( ) { \
return m_inherits : : get_class_static ( ) ; \
} \
static void get_inheritance_list_static ( List < String > * p_inheritance_list ) { \
m_inherits : : get_inheritance_list_static ( p_inheritance_list ) ; \
p_inheritance_list - > push_back ( String ( # m_class ) ) ; \
} \
static String get_category_static ( ) { \
String category = m_inherits : : get_category_static ( ) ; \
if ( _get_category ! = m_inherits : : _get_category ) { \
2021-12-09 17:42:46 +08:00
if ( ! category . is_empty ( ) ) { \
2020-07-10 18:34:39 +08:00
category + = " / " ; \
} \
category + = _get_category ( ) ; \
} \
return category ; \
} \
2021-06-05 01:33:48 +08:00
virtual bool is_class ( const String & p_class ) const override { \
if ( _get_extension ( ) & & _get_extension ( ) - > is_class ( p_class ) ) { \
return true ; \
} \
return ( p_class = = ( # m_class ) ) ? true : m_inherits : : is_class ( p_class ) ; \
} \
2020-07-10 18:34:39 +08:00
virtual bool is_class_ptr ( void * p_ptr ) const override { return ( p_ptr = = get_class_ptr_static ( ) ) ? true : m_inherits : : is_class_ptr ( p_ptr ) ; } \
\
static void get_valid_parents_static ( List < String > * p_parents ) { \
if ( m_class : : _get_valid_parents_static ! = m_inherits : : _get_valid_parents_static ) { \
m_class : : _get_valid_parents_static ( p_parents ) ; \
} \
\
m_inherits : : get_valid_parents_static ( p_parents ) ; \
} \
\
protected : \
_FORCE_INLINE_ static void ( * _get_bind_methods ( ) ) ( ) { \
return & m_class : : _bind_methods ; \
} \
\
public : \
static void initialize_class ( ) { \
static bool initialized = false ; \
if ( initialized ) { \
return ; \
} \
m_inherits : : initialize_class ( ) ; \
2021-08-13 22:46:14 +08:00
: : ClassDB : : _add_class < m_class > ( ) ; \
2020-07-10 18:34:39 +08:00
if ( m_class : : _get_bind_methods ( ) ! = m_inherits : : _get_bind_methods ( ) ) { \
_bind_methods ( ) ; \
} \
initialized = true ; \
} \
\
protected : \
virtual void _initialize_classv ( ) override { \
initialize_class ( ) ; \
} \
_FORCE_INLINE_ bool ( Object : : * _get_get ( ) const ) ( const StringName & p_name , Variant & ) const { \
2021-10-28 22:01:30 +08:00
return ( bool ( Object : : * ) ( const StringName & , Variant & ) const ) & m_class : : _get ; \
2020-07-10 18:34:39 +08:00
} \
virtual bool _getv ( const StringName & p_name , Variant & r_ret ) const override { \
if ( m_class : : _get_get ( ) ! = m_inherits : : _get_get ( ) ) { \
if ( _get ( p_name , r_ret ) ) { \
return true ; \
} \
} \
return m_inherits : : _getv ( p_name , r_ret ) ; \
} \
_FORCE_INLINE_ bool ( Object : : * _get_set ( ) const ) ( const StringName & p_name , const Variant & p_property ) { \
2021-10-28 22:01:30 +08:00
return ( bool ( Object : : * ) ( const StringName & , const Variant & ) ) & m_class : : _set ; \
2020-07-10 18:34:39 +08:00
} \
virtual bool _setv ( const StringName & p_name , const Variant & p_property ) override { \
if ( m_inherits : : _setv ( p_name , p_property ) ) { \
return true ; \
} \
if ( m_class : : _get_set ( ) ! = m_inherits : : _get_set ( ) ) { \
return _set ( p_name , p_property ) ; \
} \
return false ; \
} \
_FORCE_INLINE_ void ( Object : : * _get_get_property_list ( ) const ) ( List < PropertyInfo > * p_list ) const { \
2021-10-28 22:01:30 +08:00
return ( void ( Object : : * ) ( List < PropertyInfo > * ) const ) & m_class : : _get_property_list ; \
2020-07-10 18:34:39 +08:00
} \
virtual void _get_property_listv ( List < PropertyInfo > * p_list , bool p_reversed ) const override { \
if ( ! p_reversed ) { \
m_inherits : : _get_property_listv ( p_list , p_reversed ) ; \
} \
p_list - > push_back ( PropertyInfo ( Variant : : NIL , get_class_static ( ) , PROPERTY_HINT_NONE , String ( ) , PROPERTY_USAGE_CATEGORY ) ) ; \
if ( ! _is_gpl_reversed ( ) ) { \
2021-08-13 22:46:14 +08:00
: : ClassDB : : get_property_list ( # m_class , p_list , true , this ) ; \
2020-07-10 18:34:39 +08:00
} \
if ( m_class : : _get_get_property_list ( ) ! = m_inherits : : _get_get_property_list ( ) ) { \
_get_property_list ( p_list ) ; \
} \
if ( _is_gpl_reversed ( ) ) { \
2021-08-13 22:46:14 +08:00
: : ClassDB : : get_property_list ( # m_class , p_list , true , this ) ; \
2020-07-10 18:34:39 +08:00
} \
if ( p_reversed ) { \
m_inherits : : _get_property_listv ( p_list , p_reversed ) ; \
} \
} \
_FORCE_INLINE_ void ( Object : : * _get_notification ( ) const ) ( int ) { \
2021-10-28 22:01:30 +08:00
return ( void ( Object : : * ) ( int ) ) & m_class : : _notification ; \
2020-07-10 18:34:39 +08:00
} \
virtual void _notificationv ( int p_notification , bool p_reversed ) override { \
if ( ! p_reversed ) { \
m_inherits : : _notificationv ( p_notification , p_reversed ) ; \
} \
if ( m_class : : _get_notification ( ) ! = m_inherits : : _get_notification ( ) ) { \
_notification ( p_notification ) ; \
} \
if ( p_reversed ) { \
m_inherits : : _notificationv ( p_notification , p_reversed ) ; \
} \
} \
\
2014-02-10 09:10:30 +08:00
private :
2017-03-05 23:44:50 +08:00
# define OBJ_CATEGORY(m_category) \
protected : \
_FORCE_INLINE_ static String _get_category ( ) { return m_category ; } \
\
2014-02-10 09:10:30 +08:00
private :
2020-07-10 18:34:39 +08:00
# define OBJ_SAVE_TYPE(m_class) \
public : \
virtual String get_save_class ( ) const override { return # m_class ; } \
\
2014-02-10 09:10:30 +08:00
private :
class ScriptInstance ;
2015-08-30 08:09:11 +08:00
class Object {
2014-02-10 09:10:30 +08:00
public :
enum ConnectFlags {
2017-03-05 23:44:50 +08:00
CONNECT_DEFERRED = 1 ,
CONNECT_PERSIST = 2 , // hint for scene to save this connection
2018-08-21 00:38:18 +08:00
CONNECT_ONESHOT = 4 ,
CONNECT_REFERENCE_COUNTED = 8 ,
2014-02-10 09:10:30 +08:00
} ;
struct Connection {
2020-02-20 03:27:19 +08:00
: : Signal signal ;
Callable callable ;
2020-05-12 23:01:17 +08:00
uint32_t flags = 0 ;
2014-02-10 09:10:30 +08:00
Vector < Variant > binds ;
2017-03-05 23:44:50 +08:00
bool operator < ( const Connection & p_conn ) const ;
2014-02-10 09:10:30 +08:00
operator Variant ( ) const ;
2020-05-12 23:01:17 +08:00
Connection ( ) { }
2017-03-05 23:44:50 +08:00
Connection ( const Variant & p_variant ) ;
2014-02-10 09:10:30 +08:00
} ;
2017-03-05 23:44:50 +08:00
2014-02-10 09:10:30 +08:00
private :
2014-04-05 23:39:30 +08:00
# ifdef DEBUG_ENABLED
2018-10-01 22:46:50 +08:00
friend struct _ObjectDebugLock ;
2014-04-05 23:39:30 +08:00
# endif
2017-03-05 23:44:50 +08:00
friend bool predelete_handler ( Object * ) ;
friend void postinitialize_handler ( Object * ) ;
2014-02-10 09:10:30 +08:00
2021-06-05 01:33:48 +08:00
ObjectNativeExtension * _extension = nullptr ;
2021-06-19 23:58:49 +08:00
GDExtensionClassInstancePtr _extension_instance = nullptr ;
2021-06-05 01:33:48 +08:00
2020-02-20 03:27:19 +08:00
struct SignalData {
2014-02-10 09:10:30 +08:00
struct Slot {
2020-05-12 23:01:17 +08:00
int reference_count = 0 ;
2014-02-10 09:10:30 +08:00
Connection conn ;
2020-05-12 23:01:17 +08:00
List < Connection > : : Element * cE = nullptr ;
2014-02-10 09:10:30 +08:00
} ;
MethodInfo user ;
2020-02-20 03:27:19 +08:00
VMap < Callable , Slot > slot_map ;
2014-02-10 09:10:30 +08:00
} ;
2020-02-20 03:27:19 +08:00
HashMap < StringName , SignalData > signal_map ;
2014-02-10 09:10:30 +08:00
List < Connection > connections ;
2014-04-05 23:39:30 +08:00
# ifdef DEBUG_ENABLED
SafeRefCount _lock_index ;
# endif
2020-05-12 23:01:17 +08:00
bool _block_signals = false ;
int _predelete_ok = 0 ;
2019-05-09 17:21:49 +08:00
ObjectID _instance_id ;
2014-02-10 09:10:30 +08:00
bool _predelete ( ) ;
void _postinitialize ( ) ;
2020-05-12 23:01:17 +08:00
bool _can_translate = true ;
bool _emitting = false ;
2014-02-10 09:10:30 +08:00
# ifdef TOOLS_ENABLED
2020-05-12 23:01:17 +08:00
bool _edited = false ;
uint32_t _edited_version = 0 ;
2017-06-26 04:30:28 +08:00
Set < String > editor_section_folding ;
2014-02-10 09:10:30 +08:00
# endif
2020-05-12 23:01:17 +08:00
ScriptInstance * script_instance = nullptr ;
2022-02-16 20:56:32 +08:00
Variant script ; // Reference does not exist yet, store it in a Variant.
2014-02-10 09:10:30 +08:00
Dictionary metadata ;
2017-01-03 10:03:46 +08:00
mutable StringName _class_name ;
2020-05-12 23:01:17 +08:00
mutable const StringName * _class_ptr = nullptr ;
2014-02-10 09:10:30 +08:00
2017-08-12 03:10:05 +08:00
void _add_user_signal ( const String & p_name , const Array & p_args = Array ( ) ) ;
2017-03-05 23:44:50 +08:00
bool _has_user_signal ( const StringName & p_name ) const ;
2020-02-20 03:27:19 +08:00
Variant _emit_signal ( const Variant * * p_args , int p_argcount , Callable : : CallError & r_error ) ;
2014-02-10 09:10:30 +08:00
Array _get_signal_list ( ) const ;
2017-03-05 23:44:50 +08:00
Array _get_signal_connection_list ( const String & p_signal ) const ;
2017-06-15 21:31:57 +08:00
Array _get_incoming_connections ( ) const ;
2017-03-05 23:44:50 +08:00
void _set_bind ( const String & p_set , const Variant & p_value ) ;
Variant _get_bind ( const String & p_name ) const ;
2017-05-31 04:20:15 +08:00
void _set_indexed_bind ( const NodePath & p_name , const Variant & p_value ) ;
Variant _get_indexed_bind ( const NodePath & p_name ) const ;
2014-02-10 09:10:30 +08:00
2020-02-14 03:03:10 +08:00
_FORCE_INLINE_ void _construct_object ( bool p_reference ) ;
2021-06-05 00:03:15 +08:00
friend class RefCounted ;
2020-02-14 03:03:10 +08:00
bool type_is_reference = false ;
2021-07-09 03:16:02 +08:00
std : : mutex _instance_binding_mutex ;
struct InstanceBinding {
void * binding ;
void * token ;
GDNativeInstanceBindingFreeCallback free_callback = nullptr ;
GDNativeInstanceBindingReferenceCallback reference_callback = nullptr ;
} ;
InstanceBinding * _instance_bindings = nullptr ;
uint32_t _instance_binding_count = 0 ;
2020-05-12 23:01:17 +08:00
2020-02-14 03:03:10 +08:00
Object ( bool p_reference ) ;
2018-02-22 22:34:08 +08:00
2015-08-30 08:09:11 +08:00
protected :
2021-07-09 03:16:02 +08:00
_FORCE_INLINE_ bool _instance_binding_reference ( bool p_reference ) {
bool can_die = true ;
if ( _instance_bindings ) {
_instance_binding_mutex . lock ( ) ;
for ( uint32_t i = 0 ; i < _instance_binding_count ; i + + ) {
if ( _instance_bindings [ i ] . reference_callback ) {
if ( ! _instance_bindings [ i ] . reference_callback ( _instance_bindings [ i ] . token , _instance_bindings [ i ] . binding , p_reference ) ) {
can_die = false ;
}
}
}
_instance_binding_mutex . unlock ( ) ;
}
return can_die ;
}
2022-02-16 20:56:32 +08:00
2021-06-19 23:58:49 +08:00
friend class NativeExtensionMethodBind ;
2021-06-05 01:33:48 +08:00
_ALWAYS_INLINE_ const ObjectNativeExtension * _get_extension ( ) const { return _extension ; }
2021-06-19 23:58:49 +08:00
_ALWAYS_INLINE_ GDExtensionClassInstancePtr _get_extension_instance ( ) const { return _extension_instance ; }
2017-01-03 10:03:46 +08:00
virtual void _initialize_classv ( ) { initialize_class ( ) ; }
2017-03-05 23:44:50 +08:00
virtual bool _setv ( const StringName & p_name , const Variant & p_property ) { return false ; } ;
virtual bool _getv ( const StringName & p_name , Variant & r_property ) const { return false ; } ;
virtual void _get_property_listv ( List < PropertyInfo > * p_list , bool p_reversed ) const { } ;
2020-05-12 23:01:17 +08:00
virtual void _notificationv ( int p_notification , bool p_reversed ) { }
2015-08-30 08:09:11 +08:00
2014-02-10 09:10:30 +08:00
static String _get_category ( ) { return " " ; }
static void _bind_methods ( ) ;
2017-03-05 23:44:50 +08:00
bool _set ( const StringName & p_name , const Variant & p_property ) { return false ; } ;
bool _get ( const StringName & p_name , Variant & r_property ) const { return false ; } ;
2014-02-10 09:10:30 +08:00
void _get_property_list ( List < PropertyInfo > * p_list ) const { } ;
2020-05-12 23:01:17 +08:00
void _notification ( int p_notification ) { }
2015-08-30 08:09:11 +08:00
2014-02-10 09:10:30 +08:00
_FORCE_INLINE_ static void ( * _get_bind_methods ( ) ) ( ) {
return & Object : : _bind_methods ;
}
2018-10-01 04:25:55 +08:00
_FORCE_INLINE_ bool ( Object : : * _get_get ( ) const ) ( const StringName & p_name , Variant & r_ret ) const {
2014-02-10 09:10:30 +08:00
return & Object : : _get ;
}
2018-10-01 04:25:55 +08:00
_FORCE_INLINE_ bool ( Object : : * _get_set ( ) const ) ( const StringName & p_name , const Variant & p_property ) {
2017-03-05 23:44:50 +08:00
return & Object : : _set ;
2014-02-10 09:10:30 +08:00
}
2018-10-01 04:25:55 +08:00
_FORCE_INLINE_ void ( Object : : * _get_get_property_list ( ) const ) ( List < PropertyInfo > * p_list ) const {
2017-03-05 23:44:50 +08:00
return & Object : : _get_property_list ;
2015-08-30 08:09:11 +08:00
}
2018-10-01 04:25:55 +08:00
_FORCE_INLINE_ void ( Object : : * _get_notification ( ) const ) ( int ) {
2017-03-05 23:44:50 +08:00
return & Object : : _notification ;
2015-08-30 08:09:11 +08:00
}
2014-02-10 09:10:30 +08:00
static void get_valid_parents_static ( List < String > * p_parents ) ;
static void _get_valid_parents_static ( List < String > * p_parents ) ;
2015-08-30 08:09:11 +08:00
2020-02-20 03:27:19 +08:00
Variant _call_bind ( const Variant * * p_args , int p_argcount , Callable : : CallError & r_error ) ;
Variant _call_deferred_bind ( const Variant * * p_args , int p_argcount , Callable : : CallError & r_error ) ;
2014-02-10 09:10:30 +08:00
2017-03-05 23:44:50 +08:00
virtual const StringName * _get_class_namev ( ) const {
2020-05-14 22:41:43 +08:00
if ( ! _class_name ) {
2017-03-05 23:44:50 +08:00
_class_name = get_class_static ( ) ;
2020-05-14 22:41:43 +08:00
}
2017-01-03 10:03:46 +08:00
return & _class_name ;
2015-06-29 11:29:49 +08:00
}
2014-02-10 09:10:30 +08:00
2021-07-23 04:37:17 +08:00
Vector < StringName > _get_meta_list_bind ( ) const ;
2014-02-10 09:10:30 +08:00
Array _get_property_list_bind ( ) const ;
2015-05-25 12:46:45 +08:00
Array _get_method_list_bind ( ) const ;
2014-02-10 09:10:30 +08:00
2015-06-22 11:03:19 +08:00
void _clear_internal_resource_paths ( const Variant & p_var ) ;
2017-03-05 23:44:50 +08:00
friend class ClassDB ;
virtual void _validate_property ( PropertyInfo & property ) const ;
2016-05-15 10:48:23 +08:00
2020-02-20 03:27:19 +08:00
void _disconnect ( const StringName & p_signal , const Callable & p_callable , bool p_force = false ) ;
2018-08-21 03:35:36 +08:00
2022-02-16 20:56:32 +08:00
public : // Should be protected, but bug in clang++.
2017-01-03 10:03:46 +08:00
static void initialize_class ( ) ;
2020-05-12 23:01:17 +08:00
_FORCE_INLINE_ static void register_custom_data_to_otdb ( ) { }
2014-02-10 09:10:30 +08:00
public :
2021-02-11 04:18:45 +08:00
void notify_property_list_changed ( ) ;
2017-03-05 23:44:50 +08:00
static void * get_class_ptr_static ( ) {
2014-02-10 09:10:30 +08:00
static int ptr ;
return & ptr ;
}
bool _is_gpl_reversed ( ) const { return false ; }
2019-05-09 17:21:49 +08:00
_FORCE_INLINE_ ObjectID get_instance_id ( ) const { return _instance_id ; }
2014-02-10 09:10:30 +08:00
2017-08-25 03:58:13 +08:00
template < class T >
static T * cast_to ( Object * p_object ) {
# ifndef NO_SAFE_CAST
return dynamic_cast < T * > ( p_object ) ;
# else
2020-05-14 22:41:43 +08:00
if ( ! p_object ) {
2020-04-02 07:20:12 +08:00
return nullptr ;
2020-05-14 22:41:43 +08:00
}
if ( p_object - > is_class_ptr ( T : : get_class_ptr_static ( ) ) ) {
2017-08-26 23:10:58 +08:00
return static_cast < T * > ( p_object ) ;
2020-05-14 22:41:43 +08:00
} else {
2020-04-02 07:20:12 +08:00
return nullptr ;
2020-05-14 22:41:43 +08:00
}
2017-08-25 03:58:13 +08:00
# endif
}
template < class T >
static const T * cast_to ( const Object * p_object ) {
# ifndef NO_SAFE_CAST
return dynamic_cast < const T * > ( p_object ) ;
# else
2020-05-14 22:41:43 +08:00
if ( ! p_object ) {
2020-04-02 07:20:12 +08:00
return nullptr ;
2020-05-14 22:41:43 +08:00
}
if ( p_object - > is_class_ptr ( T : : get_class_ptr_static ( ) ) ) {
2017-08-25 03:58:13 +08:00
return static_cast < const T * > ( p_object ) ;
2020-05-14 22:41:43 +08:00
} else {
2020-04-02 07:20:12 +08:00
return nullptr ;
2020-05-14 22:41:43 +08:00
}
2017-08-25 03:58:13 +08:00
# endif
}
2014-02-10 09:10:30 +08:00
enum {
2017-03-05 23:44:50 +08:00
NOTIFICATION_POSTINITIALIZE = 0 ,
NOTIFICATION_PREDELETE = 1
2014-02-10 09:10:30 +08:00
} ;
2015-08-30 08:09:11 +08:00
2014-02-10 09:10:30 +08:00
/* TYPE API */
2017-03-05 23:44:50 +08:00
static void get_inheritance_list_static ( List < String > * p_inheritance_list ) { p_inheritance_list - > push_back ( " Object " ) ; }
2014-02-10 09:10:30 +08:00
2017-01-03 10:03:46 +08:00
static String get_class_static ( ) { return " Object " ; }
static String get_parent_class_static ( ) { return String ( ) ; }
2014-02-10 09:10:30 +08:00
static String get_category_static ( ) { return String ( ) ; }
2021-06-05 01:33:48 +08:00
virtual String get_class ( ) const {
2022-01-28 00:34:33 +08:00
if ( _extension ) {
2021-06-05 01:33:48 +08:00
return _extension - > class_name . operator String ( ) ;
2022-01-28 00:34:33 +08:00
}
2021-06-05 01:33:48 +08:00
return " Object " ;
}
2017-01-03 10:03:46 +08:00
virtual String get_save_class ( ) const { return get_class ( ) ; } //class stored when saving
2015-06-29 11:29:49 +08:00
2021-06-05 01:33:48 +08:00
virtual bool is_class ( const String & p_class ) const {
if ( _extension & & _extension - > is_class ( p_class ) ) {
return true ;
}
return ( p_class = = " Object " ) ;
}
2017-03-05 23:44:50 +08:00
virtual bool is_class_ptr ( void * p_ptr ) const { return get_class_ptr_static ( ) = = p_ptr ; }
2015-06-29 11:29:49 +08:00
2017-03-05 23:44:50 +08:00
_FORCE_INLINE_ const StringName & get_class_name ( ) const {
2021-06-05 01:33:48 +08:00
if ( _extension ) {
return _extension - > class_name ;
}
2017-01-03 10:03:46 +08:00
if ( ! _class_ptr ) {
return * _get_class_namev ( ) ;
2015-06-29 11:29:49 +08:00
} else {
2017-01-03 10:03:46 +08:00
return * _class_ptr ;
2015-06-29 11:29:49 +08:00
}
}
2015-08-30 08:09:11 +08:00
2014-02-10 09:10:30 +08:00
/* IAPI */
2020-04-02 07:20:12 +08:00
void set ( const StringName & p_name , const Variant & p_value , bool * r_valid = nullptr ) ;
Variant get ( const StringName & p_name , bool * r_valid = nullptr ) const ;
void set_indexed ( const Vector < StringName > & p_names , const Variant & p_value , bool * r_valid = nullptr ) ;
Variant get_indexed ( const Vector < StringName > & p_names , bool * r_valid = nullptr ) const ;
2014-02-10 09:10:30 +08:00
2017-03-05 23:44:50 +08:00
void get_property_list ( List < PropertyInfo > * p_list , bool p_reversed = false ) const ;
2015-08-30 08:09:11 +08:00
2017-03-05 23:44:50 +08:00
bool has_method ( const StringName & p_method ) const ;
2014-02-10 09:10:30 +08:00
void get_method_list ( List < MethodInfo > * p_list ) const ;
2017-03-05 23:44:50 +08:00
Variant callv ( const StringName & p_method , const Array & p_args ) ;
2022-03-09 21:58:40 +08:00
virtual Variant callp ( const StringName & p_method , const Variant * * p_args , int p_argcount , Callable : : CallError & r_error ) ;
template < typename . . . VarArgs >
Variant call ( const StringName & p_method , VarArgs . . . p_args ) {
Variant args [ sizeof . . . ( p_args ) + 1 ] = { p_args . . . , Variant ( ) } ; // +1 makes sure zero sized arrays are also supported.
const Variant * argptrs [ sizeof . . . ( p_args ) + 1 ] ;
for ( uint32_t i = 0 ; i < sizeof . . . ( p_args ) ; i + + ) {
argptrs [ i ] = & args [ i ] ;
}
Callable : : CallError cerr ;
return callp ( p_method , sizeof . . . ( p_args ) = = 0 ? nullptr : ( const Variant * * ) argptrs , sizeof . . . ( p_args ) , cerr ) ;
}
2014-02-10 09:10:30 +08:00
2017-03-05 23:44:50 +08:00
void notification ( int p_notification , bool p_reversed = false ) ;
2020-09-15 23:14:45 +08:00
virtual String to_string ( ) ;
2014-02-10 09:10:30 +08:00
2022-02-16 20:56:32 +08:00
// Used mainly by script, get and set all INCLUDING string.
2020-04-02 07:20:12 +08:00
virtual Variant getvar ( const Variant & p_key , bool * r_valid = nullptr ) const ;
virtual void setvar ( const Variant & p_key , const Variant & p_value , bool * r_valid = nullptr ) ;
2014-02-10 09:10:30 +08:00
/* SCRIPT */
2015-08-30 08:09:11 +08:00
2020-02-14 03:03:10 +08:00
void set_script ( const Variant & p_script ) ;
Variant get_script ( ) const ;
2014-02-10 09:10:30 +08:00
2021-07-23 04:37:17 +08:00
bool has_meta ( const StringName & p_name ) const ;
void set_meta ( const StringName & p_name , const Variant & p_value ) ;
void remove_meta ( const StringName & p_name ) ;
Variant get_meta ( const StringName & p_name ) const ;
void get_meta_list ( List < StringName > * p_list ) const ;
2014-02-10 09:10:30 +08:00
# ifdef TOOLS_ENABLED
void set_edited ( bool p_edited ) ;
bool is_edited ( ) const ;
2022-02-16 20:56:32 +08:00
// This function is used to check when something changed beyond a point, it's used mainly for generating previews.
uint32_t get_edited_version ( ) const ;
2014-02-10 09:10:30 +08:00
# endif
void set_script_instance ( ScriptInstance * p_instance ) ;
2017-03-05 23:44:50 +08:00
_FORCE_INLINE_ ScriptInstance * get_script_instance ( ) const { return script_instance ; }
2014-02-10 09:10:30 +08:00
2022-02-16 20:56:32 +08:00
// Some script languages can't control instance creation, so this function eases the process.
void set_script_and_instance ( const Variant & p_script , ScriptInstance * p_instance ) ;
2017-07-23 03:57:56 +08:00
2017-03-05 23:44:50 +08:00
void add_user_signal ( const MethodInfo & p_signal ) ;
2022-03-09 21:58:40 +08:00
template < typename . . . VarArgs >
Error emit_signal ( const StringName & p_name , VarArgs . . . p_args ) {
Variant args [ sizeof . . . ( p_args ) + 1 ] = { p_args . . . , Variant ( ) } ; // +1 makes sure zero sized arrays are also supported.
const Variant * argptrs [ sizeof . . . ( p_args ) + 1 ] ;
for ( uint32_t i = 0 ; i < sizeof . . . ( p_args ) ; i + + ) {
argptrs [ i ] = & args [ i ] ;
}
return emit_signalp ( p_name , sizeof . . . ( p_args ) = = 0 ? nullptr : ( const Variant * * ) argptrs , sizeof . . . ( p_args ) ) ;
}
Error emit_signalp ( const StringName & p_name , const Variant * * p_args , int p_argcount ) ;
2019-11-10 13:59:44 +08:00
bool has_signal ( const StringName & p_name ) const ;
2017-03-05 23:44:50 +08:00
void get_signal_list ( List < MethodInfo > * p_signals ) const ;
void get_signal_connection_list ( const StringName & p_signal , List < Connection > * p_connections ) const ;
2015-05-11 02:45:33 +08:00
void get_all_signal_connections ( List < Connection > * p_connections ) const ;
2019-08-17 04:30:31 +08:00
int get_persistent_signal_connection_count ( ) const ;
2016-06-07 13:39:40 +08:00
void get_signals_connected_to_this ( List < Connection > * p_connections ) const ;
2014-02-10 09:10:30 +08:00
2020-02-20 03:27:19 +08:00
Error connect ( const StringName & p_signal , const Callable & p_callable , const Vector < Variant > & p_binds = Vector < Variant > ( ) , uint32_t p_flags = 0 ) ;
void disconnect ( const StringName & p_signal , const Callable & p_callable ) ;
bool is_connected ( const StringName & p_signal , const Callable & p_callable ) const ;
2014-02-10 09:10:30 +08:00
2022-03-09 21:58:40 +08:00
template < typename . . . VarArgs >
void call_deferred ( const StringName & p_name , VarArgs . . . p_args ) {
MessageQueue : : get_singleton ( ) - > push_call ( this , p_name , p_args . . . ) ;
}
2018-11-16 19:49:26 +08:00
void set_deferred ( const StringName & p_property , const Variant & p_value ) ;
2014-02-10 09:10:30 +08:00
void set_block_signals ( bool p_block ) ;
bool is_blocking_signals ( ) const ;
2020-04-02 07:20:12 +08:00
Variant : : Type get_static_property_type ( const StringName & p_property , bool * r_valid = nullptr ) const ;
Variant : : Type get_static_property_type_indexed ( const Vector < StringName > & p_path , bool * r_valid = nullptr ) const ;
2015-12-06 01:18:22 +08:00
2014-02-10 09:10:30 +08:00
virtual void get_translatable_strings ( List < String > * p_strings ) const ;
2017-03-05 23:44:50 +08:00
virtual void get_argument_options ( const StringName & p_function , int p_idx , List < String > * r_options ) const ;
2014-02-10 09:10:30 +08:00
2022-02-16 20:56:32 +08:00
// Translate message (internationalization).
String tr ( const StringName & p_message , const StringName & p_context = " " ) const ;
2020-07-16 16:52:06 +08:00
String tr_n ( const StringName & p_message , const StringName & p_message_plural , int p_n , const StringName & p_context = " " ) const ;
2014-02-10 09:10:30 +08:00
2022-02-16 20:56:32 +08:00
bool _is_queued_for_deletion = false ; // Set to true by SceneTree::queue_delete().
2015-08-30 08:09:11 +08:00
bool is_queued_for_deletion ( ) const ;
2015-03-29 01:34:28 +08:00
2017-08-19 04:29:15 +08:00
_FORCE_INLINE_ void set_message_translation ( bool p_enable ) { _can_translate = p_enable ; }
2014-02-10 09:10:30 +08:00
_FORCE_INLINE_ bool can_translate_messages ( ) const { return _can_translate ; }
2015-06-22 11:03:19 +08:00
2017-06-26 04:30:28 +08:00
# ifdef TOOLS_ENABLED
void editor_set_section_unfold ( const String & p_section , bool p_unfolded ) ;
bool editor_is_section_unfolded ( const String & p_section ) ;
2018-10-30 03:36:31 +08:00
const Set < String > & editor_get_section_folding ( ) const { return editor_section_folding ; }
void editor_clear_section_folding ( ) { editor_section_folding . clear ( ) ; }
2017-06-26 04:30:28 +08:00
# endif
2021-07-15 22:41:57 +08:00
// Used by script languages to store binding data.
2021-07-09 03:16:02 +08:00
void * get_instance_binding ( void * p_token , const GDNativeInstanceBindingCallbacks * p_callbacks ) ;
2021-07-15 22:41:57 +08:00
// Used on creation by binding only.
void set_instance_binding ( void * p_token , void * p_binding , const GDNativeInstanceBindingCallbacks * p_callbacks ) ;
2021-08-16 23:16:36 +08:00
bool has_instance_binding ( void * p_token ) ;
2017-07-16 23:39:23 +08:00
2015-06-22 11:03:19 +08:00
void clear_internal_resource_paths ( ) ;
2021-06-05 00:03:15 +08:00
_ALWAYS_INLINE_ bool is_ref_counted ( ) const { return type_is_reference ; }
2020-05-12 23:01:17 +08:00
2015-08-30 08:09:11 +08:00
Object ( ) ;
2014-02-10 09:10:30 +08:00
virtual ~ Object ( ) ;
} ;
bool predelete_handler ( Object * p_object ) ;
void postinitialize_handler ( Object * p_object ) ;
class ObjectDB {
2022-02-16 20:56:32 +08:00
// This needs to add up to 63, 1 bit is for reference.
2020-02-14 03:03:10 +08:00
# define OBJECTDB_VALIDATOR_BITS 39
# define OBJECTDB_VALIDATOR_MASK ((uint64_t(1) << OBJECTDB_VALIDATOR_BITS) - 1)
# define OBJECTDB_SLOT_MAX_COUNT_BITS 24
# define OBJECTDB_SLOT_MAX_COUNT_MASK ((uint64_t(1) << OBJECTDB_SLOT_MAX_COUNT_BITS) - 1)
# define OBJECTDB_REFERENCE_BIT (uint64_t(1) << (OBJECTDB_SLOT_MAX_COUNT_BITS + OBJECTDB_VALIDATOR_BITS))
2022-02-16 20:56:32 +08:00
struct ObjectSlot { // 128 bits per slot.
2020-02-14 03:03:10 +08:00
uint64_t validator : OBJECTDB_VALIDATOR_BITS ;
uint64_t next_free : OBJECTDB_SLOT_MAX_COUNT_BITS ;
2021-06-05 00:03:15 +08:00
uint64_t is_ref_counted : 1 ;
2020-02-14 03:03:10 +08:00
Object * object ;
2014-02-10 09:10:30 +08:00
} ;
2020-02-14 03:03:10 +08:00
static SpinLock spin_lock ;
static uint32_t slot_count ;
static uint32_t slot_max ;
static ObjectSlot * object_slots ;
static uint64_t validator_counter ;
2014-02-10 09:10:30 +08:00
2017-03-05 23:44:50 +08:00
friend class Object ;
friend void unregister_core_types ( ) ;
2014-02-10 09:10:30 +08:00
static void cleanup ( ) ;
2020-02-14 03:03:10 +08:00
2017-06-09 11:23:50 +08:00
static ObjectID add_instance ( Object * p_object ) ;
2014-02-10 09:10:30 +08:00
static void remove_instance ( Object * p_object ) ;
2020-02-14 03:03:10 +08:00
2017-03-05 23:44:50 +08:00
friend void register_core_types ( ) ;
2017-01-08 05:25:37 +08:00
static void setup ( ) ;
2015-08-30 08:09:11 +08:00
public :
2014-02-10 09:10:30 +08:00
typedef void ( * DebugFunc ) ( Object * p_obj ) ;
2020-02-14 03:03:10 +08:00
_ALWAYS_INLINE_ static Object * get_instance ( ObjectID p_instance_id ) {
uint64_t id = p_instance_id ;
uint32_t slot = id & OBJECTDB_SLOT_MAX_COUNT_MASK ;
2022-02-16 20:56:32 +08:00
ERR_FAIL_COND_V ( slot > = slot_max , nullptr ) ; // This should never happen unless RID is corrupted.
2020-02-14 03:03:10 +08:00
spin_lock . lock ( ) ;
2014-02-10 09:10:30 +08:00
2020-02-14 03:03:10 +08:00
uint64_t validator = ( id > > OBJECTDB_SLOT_MAX_COUNT_BITS ) & OBJECTDB_VALIDATOR_MASK ;
2014-02-10 09:10:30 +08:00
2020-02-14 03:03:10 +08:00
if ( unlikely ( object_slots [ slot ] . validator ! = validator ) ) {
spin_lock . unlock ( ) ;
return nullptr ;
}
Object * object = object_slots [ slot ] . object ;
2019-07-29 04:19:44 +08:00
2020-02-14 03:03:10 +08:00
spin_lock . unlock ( ) ;
2019-07-29 04:19:44 +08:00
2020-02-14 03:03:10 +08:00
return object ;
2014-02-10 09:10:30 +08:00
}
2020-02-14 03:03:10 +08:00
static void debug_objects ( DebugFunc p_func ) ;
static int get_object_count ( ) ;
2014-02-10 09:10:30 +08:00
} ;
2020-03-25 18:10:34 +08:00
# endif // OBJECT_H