mirror of
https://github.com/godotengine/godot-docs.git
synced 2026-01-05 22:09:56 +03:00
The previous text implied (at least to me) that there was something special in Godot called an "init assignment value", and that there was special logic determining which kind of assignment was used. In reality, nothing special is happening: the property is initialized, then the constructor may set the value, then exports may as well. I also harmonized the Godot and C# code samples, rather than expecting the user to read both in order. Frankly I almost feel like this would be clearer if the entire section were simply deleted, but I wasn't sure if that would be accepted.
312 lines
11 KiB
ReStructuredText
312 lines
11 KiB
ReStructuredText
.. _doc_godot_notifications:
|
|
|
|
Godot notifications
|
|
===================
|
|
|
|
Every Object in Godot implements a
|
|
:ref:`_notification <class_Object_private_method__notification>` method. Its purpose is to
|
|
allow the Object to respond to a variety of engine-level callbacks that may
|
|
relate to it. For example, if the engine tells a
|
|
:ref:`CanvasItem <class_CanvasItem>` to "draw", it will call
|
|
``_notification(NOTIFICATION_DRAW)``.
|
|
|
|
Some of these notifications, like draw, are useful to override in scripts. So
|
|
much so that Godot exposes many of them with dedicated functions:
|
|
|
|
- ``_ready()``: ``NOTIFICATION_READY``
|
|
|
|
- ``_enter_tree()``: ``NOTIFICATION_ENTER_TREE``
|
|
|
|
- ``_exit_tree()``: ``NOTIFICATION_EXIT_TREE``
|
|
|
|
- ``_process(delta)``: ``NOTIFICATION_PROCESS``
|
|
|
|
- ``_physics_process(delta)``: ``NOTIFICATION_PHYSICS_PROCESS``
|
|
|
|
- ``_draw()``: ``NOTIFICATION_DRAW``
|
|
|
|
What users might *not* realize is that notifications exist for types other
|
|
than Node alone, for example:
|
|
|
|
- :ref:`Object::NOTIFICATION_POSTINITIALIZE <class_Object_constant_NOTIFICATION_POSTINITIALIZE>`:
|
|
a callback that triggers during object initialization. Not accessible to scripts.
|
|
|
|
- :ref:`Object::NOTIFICATION_PREDELETE <class_Object_constant_NOTIFICATION_PREDELETE>`:
|
|
a callback that triggers before the engine deletes an Object, i.e. a
|
|
"destructor".
|
|
|
|
And many of the callbacks that *do* exist in Nodes don't have any dedicated
|
|
methods, but are still quite useful.
|
|
|
|
- :ref:`Node::NOTIFICATION_PARENTED <class_Node_constant_NOTIFICATION_PARENTED>`:
|
|
a callback that triggers anytime one adds a child node to another node.
|
|
|
|
- :ref:`Node::NOTIFICATION_UNPARENTED <class_Node_constant_NOTIFICATION_UNPARENTED>`:
|
|
a callback that triggers anytime one removes a child node from another
|
|
node.
|
|
|
|
One can access all these custom notifications from the universal
|
|
``_notification()`` method.
|
|
|
|
.. note::
|
|
Methods in the documentation labeled as "virtual" are also intended to be
|
|
overridden by scripts.
|
|
|
|
A classic example is the
|
|
:ref:`_init <class_Object_private_method__init>` method in Object. While it has no
|
|
``NOTIFICATION_*`` equivalent, the engine still calls the method. Most languages
|
|
(except C#) rely on it as a constructor.
|
|
|
|
So, in which situation should one use each of these notifications or
|
|
virtual functions?
|
|
|
|
_process vs. _physics_process vs. \*_input
|
|
------------------------------------------
|
|
|
|
Use ``_process()`` when one needs a framerate-dependent delta time between
|
|
frames. If code that updates object data needs to update as often as
|
|
possible, this is the right place. Recurring logic checks and data caching
|
|
often execute here, but it comes down to the frequency at which one needs
|
|
the evaluations to update. If they don't need to execute every frame, then
|
|
implementing a Timer-timeout loop is another option.
|
|
|
|
.. tabs::
|
|
.. code-tab:: gdscript GDScript
|
|
|
|
# Allows for recurring operations that don't trigger script logic
|
|
# every frame (or even every fixed frame).
|
|
func _ready():
|
|
var timer = Timer.new()
|
|
timer.autostart = true
|
|
timer.wait_time = 0.5
|
|
add_child(timer)
|
|
timer.timeout.connect(func():
|
|
print("This block runs every 0.5 seconds")
|
|
)
|
|
|
|
.. code-tab:: csharp
|
|
|
|
using Godot;
|
|
|
|
public partial class MyNode : Node
|
|
{
|
|
// Allows for recurring operations that don't trigger script logic
|
|
// every frame (or even every fixed frame).
|
|
public override void _Ready()
|
|
{
|
|
var timer = new Timer();
|
|
timer.Autostart = true;
|
|
timer.WaitTime = 0.5;
|
|
AddChild(timer);
|
|
timer.Timeout += () => GD.Print("This block runs every 0.5 seconds");
|
|
}
|
|
}
|
|
|
|
Use ``_physics_process()`` when one needs a framerate-independent delta time
|
|
between frames. If code needs consistent updates over time, regardless
|
|
of how fast or slow time advances, this is the right place.
|
|
Recurring kinematic and object transform operations should execute here.
|
|
|
|
While it is possible, to achieve the best performance, one should avoid
|
|
making input checks during these callbacks. ``_process()`` and
|
|
``_physics_process()`` will trigger at every opportunity (they do not "rest" by
|
|
default). In contrast, ``*_input()`` callbacks will trigger only on frames in
|
|
which the engine has actually detected the input.
|
|
|
|
One can check for input actions within the input callbacks just the same.
|
|
If one wants to use delta time, one can fetch it from the related
|
|
delta time methods as needed.
|
|
|
|
.. tabs::
|
|
.. code-tab:: gdscript GDScript
|
|
|
|
# Called every frame, even when the engine detects no input.
|
|
func _process(delta):
|
|
if Input.is_action_just_pressed("ui_select"):
|
|
print(delta)
|
|
|
|
# Called during every input event.
|
|
func _unhandled_input(event):
|
|
match event.get_class():
|
|
"InputEventKey":
|
|
if Input.is_action_just_pressed("ui_accept"):
|
|
print(get_process_delta_time())
|
|
|
|
.. code-tab:: csharp
|
|
|
|
using Godot;
|
|
|
|
public partial class MyNode : Node
|
|
{
|
|
|
|
// Called every frame, even when the engine detects no input.
|
|
public void _Process(double delta)
|
|
{
|
|
if (Input.IsActionJustPressed("ui_select"))
|
|
GD.Print(delta);
|
|
}
|
|
|
|
// Called during every input event. Equally true for _input().
|
|
public void _UnhandledInput(InputEvent @event)
|
|
{
|
|
switch (@event)
|
|
{
|
|
case InputEventKey:
|
|
if (Input.IsActionJustPressed("ui_accept"))
|
|
GD.Print(GetProcessDeltaTime());
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
_init vs. initialization vs. export
|
|
-----------------------------------
|
|
|
|
If the script initializes its own node subtree, without a scene,
|
|
that code should execute in ``_init()``. Other property or SceneTree-independent
|
|
initializations should also run here.
|
|
|
|
.. note::
|
|
The C# equivalent to GDScript's ``_init()`` method is the constructor.
|
|
|
|
``_init()`` triggers before ``_enter_tree()`` or ``_ready()``, but after a script
|
|
creates and initializes its properties. When instantiating a scene, property
|
|
values will set up according to the following sequence:
|
|
|
|
1. **Initial value assignment:** the property is assigned its initialization value,
|
|
or its default value if one is not specified. If a setter exists, it is not used.
|
|
|
|
2. **``_init()`` assignment:** the property's value is replaced by any assignments
|
|
made in ``_init()``, triggering the setter.
|
|
|
|
3. **Exported value assignment:** an exported property's value is again replaced by
|
|
any value set in the Inspector, triggering the setter.
|
|
|
|
.. tabs::
|
|
.. code-tab:: gdscript GDScript
|
|
|
|
# test is initialized to "one", without triggering the setter.
|
|
@export var test: String = "one":
|
|
set(value):
|
|
test = value + "!"
|
|
|
|
func _init():
|
|
# Triggers the setter, changing test's value from "one" to "two!".
|
|
test = "two"
|
|
|
|
# If someone sets test to "three" from the Inspector, it would trigger
|
|
# the setter, changing test's value from "two!" to "three!".
|
|
|
|
.. code-tab:: csharp
|
|
|
|
using Godot;
|
|
|
|
public partial class MyNode : Node
|
|
{
|
|
private string _test = "one";
|
|
|
|
[Export]
|
|
public string Test
|
|
{
|
|
get { return _test; }
|
|
set { _test = $"{value}!"; }
|
|
}
|
|
|
|
public MyNode()
|
|
{
|
|
// Triggers the setter, changing _test's value from "one" to "two!".
|
|
Test = "two";
|
|
}
|
|
|
|
// If someone sets Test to "three" in the Inspector, it would trigger
|
|
// the setter, changing _test's value from "two!" to "three!".
|
|
}
|
|
|
|
As a result, instantiating a script versus a scene may affect both the
|
|
initialization *and* the number of times the engine calls the setter.
|
|
|
|
_ready vs. _enter_tree vs. NOTIFICATION_PARENTED
|
|
------------------------------------------------
|
|
|
|
When instantiating a scene connected to the first executed scene, Godot will
|
|
instantiate nodes down the tree (making ``_init()`` calls) and build the tree
|
|
going downwards from the root. This causes ``_enter_tree()`` calls to cascade
|
|
down the tree. Once the tree is complete, leaf nodes call ``_ready``. A node
|
|
will call this method once all child nodes have finished calling theirs. This
|
|
then causes a reverse cascade going up back to the tree's root.
|
|
|
|
When instantiating a script or a standalone scene, nodes are not
|
|
added to the SceneTree upon creation, so no ``_enter_tree()`` callbacks
|
|
trigger. Instead, only the ``_init()`` call occurs. When the scene is added
|
|
to the SceneTree, the ``_enter_tree()`` and ``_ready()`` calls occur.
|
|
|
|
If one needs to trigger behavior that occurs as nodes parent to another,
|
|
regardless of whether it occurs as part of the main/active scene or not, one
|
|
can use the :ref:`PARENTED <class_Node_constant_NOTIFICATION_PARENTED>` notification.
|
|
For example, here is a snippet that connects a node's method to
|
|
a custom signal on the parent node without failing. Useful on data-centric
|
|
nodes that one might create at runtime.
|
|
|
|
.. tabs::
|
|
.. code-tab:: gdscript GDScript
|
|
|
|
extends Node
|
|
|
|
var parent_cache
|
|
|
|
func connection_check():
|
|
return parent_cache.has_user_signal("interacted_with")
|
|
|
|
func _notification(what):
|
|
match what:
|
|
NOTIFICATION_PARENTED:
|
|
parent_cache = get_parent()
|
|
if connection_check():
|
|
parent_cache.interacted_with.connect(_on_parent_interacted_with)
|
|
NOTIFICATION_UNPARENTED:
|
|
if connection_check():
|
|
parent_cache.interacted_with.disconnect(_on_parent_interacted_with)
|
|
|
|
func _on_parent_interacted_with():
|
|
print("I'm reacting to my parent's interaction!")
|
|
|
|
.. code-tab:: csharp
|
|
|
|
using Godot;
|
|
|
|
public partial class MyNode : Node
|
|
{
|
|
private Node _parentCache;
|
|
|
|
public void ConnectionCheck()
|
|
{
|
|
return _parentCache.HasUserSignal("InteractedWith");
|
|
}
|
|
|
|
public void _Notification(int what)
|
|
{
|
|
switch (what)
|
|
{
|
|
case NotificationParented:
|
|
_parentCache = GetParent();
|
|
if (ConnectionCheck())
|
|
{
|
|
_parentCache.Connect("InteractedWith", Callable.From(OnParentInteractedWith));
|
|
}
|
|
break;
|
|
case NotificationUnparented:
|
|
if (ConnectionCheck())
|
|
{
|
|
_parentCache.Disconnect("InteractedWith", Callable.From(OnParentInteractedWith));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void OnParentInteractedWith()
|
|
{
|
|
GD.Print("I'm reacting to my parent's interaction!");
|
|
}
|
|
}
|