pandemonium_engine_docs/05_engine_development/cpp/object_class.md

326 lines
9.3 KiB
Markdown
Raw Normal View History

2023-01-12 20:49:14 +01:00
Object class
============
2023-01-12 20:55:57 +01:00
See also:
2024-03-16 20:56:52 +01:00
This page describes the C++ implementation of objects in Pandemonium.
2023-01-12 19:30:47 +01:00
Looking for the Object class reference? `Have a look here.`
General definition
------------------
2024-03-16 20:56:52 +01:00
`Object` is the base class for almost everything. Most classes in Pandemonium
inherit directly or indirectly from it. Objects provide reflection and
editable properties, and declaring them is a matter of using a single
macro like this.
2023-01-12 22:00:14 +01:00
```
class CustomObject : public Object {
GDCLASS(CustomObject, Object); // this is required to inherit
};
2023-01-12 22:00:14 +01:00
```
This makes Objects gain a lot of functionality, like for example
2023-01-12 22:00:14 +01:00
```
obj = memnew(CustomObject);
print_line("Object class: ", obj->get_class()); // print object class
obj2 = Object::cast_to<OtherClass>(obj); // converting between classes, this also works without RTTI enabled.
2023-01-12 22:00:14 +01:00
```
References:
~~~~~~~~~~~
- `core/object.h ( https://github.com/Relintai/pandemonium_engine/blob/3.x/core/object.h )`
Registering an Object
---------------------
ClassDB is a static class that holds the entire list of registered
classes that inherit from Object, as well as dynamic bindings to all
their methods properties and integer constants.
Classes are registered by calling:
2023-01-12 22:00:14 +01:00
```
ClassDB::register_class<MyCustomClass>()
2023-01-12 22:00:14 +01:00
```
Registering it will allow the class to be instanced by scripts, code, or
creating them again when deserializing.
Registering as virtual is the same but it can't be instanced.
2023-01-12 22:00:14 +01:00
```
ClassDB::register_virtual_class<MyCustomClass>()
2023-01-12 22:00:14 +01:00
```
Object-derived classes can override the static function
2023-01-12 19:43:03 +01:00
`static void _bind_methods()`. When one class is registered, this
static function is called to register all the object methods,
properties, constants, etc. It's only called once. If an Object derived
class is instanced but has not been registered, it will be registered as
virtual automatically.
2023-01-12 20:57:31 +01:00
Inside `bind_methods`, there are a couple of things that can be done.
Registering functions is one:
2023-01-12 22:00:14 +01:00
```
ClassDB::bind_method(D_METHOD("methodname", "arg1name", "arg2name"), &MyCustomMethod);
2023-01-12 22:00:14 +01:00
```
Default values for arguments can be passed in reverse order:
2023-01-12 22:00:14 +01:00
```
ClassDB::bind_method(D_METHOD("methodname", "arg1name", "arg2name"), &MyCustomType::method, DEFVAL(-1)); // default value for arg2name
2023-01-12 22:00:14 +01:00
```
2023-01-12 19:43:03 +01:00
`D_METHOD` is a macro that converts "methodname" to a StringName for more
efficiency. Argument names are used for introspection, but when
compiling on release, the macro ignores them, so the strings are unused
and optimized away.
2023-01-12 20:57:31 +01:00
Check `bind_methods` of Control or Object for more examples.
If just adding modules and functionality that is not expected to be
2023-01-12 19:43:03 +01:00
documented as thoroughly, the `D_METHOD()` macro can safely be ignored and a
string passing the name can be passed for brevity.
References:
~~~~~~~~~~~
- `core/class_db.h ( https://github.com/Relintai/pandemonium_engine/blob/3.x/core/class_db.h )`
Constants
---------
Classes often have enums such as:
2023-01-12 22:00:14 +01:00
```
enum SomeMode {
MODE_FIRST,
MODE_SECOND
};
2023-01-12 22:00:14 +01:00
```
For these to work when binding to methods, the enum must be declared
convertible to int, for this a macro is provided:
2023-01-12 22:00:14 +01:00
```
VARIANT_ENUM_CAST(MyClass::SomeMode); // now functions that take SomeMode can be bound.
2023-01-12 22:00:14 +01:00
```
2023-01-12 20:57:31 +01:00
The constants can also be bound inside `bind_methods`, by using:
2023-01-12 22:00:14 +01:00
```
BIND_CONSTANT(MODE_FIRST);
BIND_CONSTANT(MODE_SECOND);
2023-01-12 22:00:14 +01:00
```
Properties (set/get)
--------------------
Objects export properties, properties are useful for the following:
- Serializing and deserializing the object.
- Creating a list of editable values for the Object derived class.
Properties are usually defined by the PropertyInfo() class. Usually
constructed as:
2023-01-12 22:00:14 +01:00
```
PropertyInfo(type, name, hint, hint_string, usage_flags)
2023-01-12 22:00:14 +01:00
```
For example:
2023-01-12 22:00:14 +01:00
```
PropertyInfo(Variant::INT, "amount", PROPERTY_HINT_RANGE, "0,49,1", PROPERTY_USAGE_EDITOR)
2023-01-12 22:00:14 +01:00
```
This is an integer property, named "amount", hint is a range, range goes
from 0 to 49 in steps of 1 (integers). It is only usable for the editor
(edit value visually) but won't be serialized.
Another example:
2023-01-12 22:00:14 +01:00
```
PropertyInfo(Variant::STRING, "modes", PROPERTY_HINT_ENUM, "Enabled,Disabled,Turbo")
2023-01-12 22:00:14 +01:00
```
This is a string property, can take any string but the editor will only
allow the defined hint ones. Since no usage flags were specified, the
default ones are PROPERTY_USAGE_STORAGE and PROPERTY_USAGE_EDITOR.
There are plenty of hints and usage flags available in object.h, give them a
check.
Properties can also work like C# properties and be accessed from script
using indexing, but this usage is generally discouraged, as using
functions is preferred for legibility. Many properties are also bound
with categories, such as "animation/frame" which also make indexing
impossible unless using operator [].
2023-01-12 20:57:31 +01:00
From `bind_methods()`, properties can be created and bound as long as
set/get functions exist. Example:
2023-01-12 22:00:14 +01:00
```
ADD_PROPERTY(PropertyInfo(Variant::INT, "amount"), "set_amount", "get_amount")
2023-01-12 22:00:14 +01:00
```
This creates the property using the setter and the getter.
2023-01-12 20:49:14 +01:00
2023-01-12 20:57:31 +01:00
Binding properties using `set`/`get`/`get_property_list`
-----------------------------------------------------------------
An additional method of creating properties exists when more flexibility
is desired (i.e. adding or removing properties on context).
The following functions can be overridden in an Object derived class,
they are NOT virtual, DO NOT make them virtual, they are called for
every override and the previous ones are not invalidated (multilevel
call).
2023-01-12 22:00:14 +01:00
```
protected:
void _get_property_list(List<PropertyInfo> *r_props) const; // return list of properties
bool _get(const StringName &p_property, Variant &r_value) const; // return true if property was found
bool _set(const StringName &p_property, const Variant &p_value); // return true if property was found
2023-01-12 22:00:14 +01:00
```
2023-01-12 19:43:03 +01:00
This is also a little less efficient since `p_property` must be
compared against the desired names in serial order.
Dynamic casting
---------------
2024-03-16 20:56:52 +01:00
Pandemonium provides dynamic casting between Object-derived classes, for
example:
2023-01-12 22:00:14 +01:00
```
void somefunc(Object *some_obj) {
Button *button = Object::cast_to<Button>(some_obj);
}
2023-01-12 22:00:14 +01:00
```
If cast fails, NULL is returned. This system uses RTTI, but it also
works fine (although a bit slower) when RTTI is disabled. This is useful
on platforms where a small binary size is ideal, such as HTML5 or
consoles (with low memory footprint).
Signals
-------
Objects can have a set of signals defined (similar to Delegates in other
languages). Connecting to them is rather easy:
2023-01-12 22:00:14 +01:00
```
obj->connect(<signal>, target_instance, target_method)
// for example:
obj->connect("enter_tree", this, "_node_entered_tree")
2023-01-12 22:00:14 +01:00
```
2023-01-12 20:57:31 +01:00
The method `node_entered_tree` must be registered to the class using
2023-01-12 19:43:03 +01:00
`ClassDB::bind_method` (explained before).
2023-01-12 20:57:31 +01:00
Adding signals to a class is done in `bind_methods`, using the
2023-01-12 19:43:03 +01:00
`ADD_SIGNAL` macro, for example:
2023-01-12 22:00:14 +01:00
```
ADD_SIGNAL(MethodInfo("been_killed"))
2023-01-12 22:00:14 +01:00
```
Notifications
-------------
2024-03-16 20:56:52 +01:00
All objects in Pandemonium have a `notification`
method that allows it to respond to engine level callbacks that may relate to it.
2024-03-16 20:56:52 +01:00
More information can be found on the `doc_pandemonium_notifications` page.
References
----------
2023-01-12 19:30:47 +01:00
`Reference` inherits from Object and holds a
reference count. It is the base for reference counted object types.
Declaring them must be done using Ref<> template. For example:
2023-01-12 22:00:14 +01:00
```
class MyReference: public Reference {
GDCLASS(MyReference, Reference);
};
Ref<MyReference> myref(memnew(MyReference));
2023-01-12 22:00:14 +01:00
```
2023-01-12 19:43:03 +01:00
`myref` is reference counted. It will be freed when no more Ref<>
templates point to it.
References:
~~~~~~~~~~~
- `core/reference.h ( https://github.com/Relintai/pandemonium_engine/blob/3.x/core/reference.h )`
Resources:
----------
2023-01-12 19:30:47 +01:00
`Resource` inherits from Reference, so all resources
are reference counted. Resources can optionally contain a path, which
2023-01-12 19:43:03 +01:00
reference a file on disk. This can be set with `resource.set_path(path)`.
This is normally done by the resource loader though. No two different
resources can have the same path, attempt to do so will result in an error.
Resources without a path are fine too.
References:
~~~~~~~~~~~
- `core/resource.h ( https://github.com/Relintai/pandemonium_engine/blob/3.x/core/resource.h )`
Resource loading
----------------
Resources can be loaded with the ResourceLoader API, like this:
2023-01-12 22:00:14 +01:00
```
Ref<Resource> res = ResourceLoader::load("res://someresource.res")
2023-01-12 22:00:14 +01:00
```
If a reference to that resource has been loaded previously and is in
memory, the resource loader will return that reference. This means that
there can be only one resource loaded from a file referenced on disk at
the same time.
- resourceinteractiveloader (TODO)
References:
~~~~~~~~~~~
- `core/io/resource_loader.h ( https://github.com/Relintai/pandemonium_engine/blob/3.x/core/io/resource_loader.h )`
Resource saving
---------------
Saving a resource can be done with the resource saver API:
2023-01-12 22:00:14 +01:00
```
ResourceSaver::save("res://someresource.res", instance)
2023-01-12 22:00:14 +01:00
```
Instance will be saved. Sub resources that have a path to a file will be
saved as a reference to that resource. Sub resources without a path will
be bundled with the saved resource and assigned sub-IDs, like
2023-01-12 19:43:03 +01:00
`res://someresource.res::1`. This also helps to cache them when loaded.
References:
~~~~~~~~~~~
- `core/io/resource_saver.h ( https://github.com/Relintai/pandemonium_engine/blob/3.x/core/io/resource_saver.h )`