2022-03-18 17:46:08 +01:00
|
|
|
|
.. Intention: give the user a first taste of signals. We should write more
|
|
|
|
|
documentation in the scripting/ section.
|
|
|
|
|
.. Note: GDScript snippets use one line return instead of two because they're
|
|
|
|
|
really short.
|
|
|
|
|
|
|
|
|
|
.. meta::
|
|
|
|
|
:keywords: Signal
|
|
|
|
|
|
2023-01-12 20:49:14 +01:00
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
Using signals
|
2022-03-18 17:46:08 +01:00
|
|
|
|
=============
|
|
|
|
|
|
|
|
|
|
In this lesson, we will look at signals. They are messages that nodes emit when
|
|
|
|
|
something specific happens to them, like a button being pressed. Other nodes can
|
|
|
|
|
connect to that signal and call a function when the event occurs.
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
Signals are a delegation mechanism built into Godot that allows one game object to
|
2022-03-18 17:46:08 +01:00
|
|
|
|
react to a change in another without them referencing one another. Using signals
|
|
|
|
|
limits `coupling
|
2023-01-12 20:57:31 +01:00
|
|
|
|
( https://en.wikipedia.org/wiki/Coupling_(computer_programming) )` and keeps your
|
2022-03-18 17:46:08 +01:00
|
|
|
|
code flexible.
|
|
|
|
|
|
|
|
|
|
For example, you might have a life bar on the screen that represents the
|
|
|
|
|
player’s health. When the player takes damage or uses a healing potion, you want
|
|
|
|
|
the bar to reflect the change. To do so, in Godot, you would use signals.
|
|
|
|
|
|
2023-01-12 20:55:57 +01:00
|
|
|
|
Note:
|
|
|
|
|
As mentioned in the introduction, signals are Godot's version of the
|
2022-03-18 17:46:08 +01:00
|
|
|
|
observer pattern. You can learn more about it here:
|
|
|
|
|
https://gameprogrammingpatterns.com/observer.html
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
We will now use a signal to make our Godot icon from the previous lesson
|
2023-01-12 19:29:11 +01:00
|
|
|
|
(`doc_scripting_player_input`) move and stop by pressing a button.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
.. Example
|
|
|
|
|
|
|
|
|
|
Scene setup
|
|
|
|
|
-----------
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
To add a button to our game, we will create a new "main" scene which will
|
2023-01-12 19:43:03 +01:00
|
|
|
|
include both a button and the `Sprite.tscn` scene that we scripted in previous
|
2022-09-10 12:15:58 +02:00
|
|
|
|
lessons.
|
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
Create a new scene by going to the menu Scene -> New Scene.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_01_new_scene.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
In the Scene dock, click the 2D Scene button. This will add a Node2D as our
|
|
|
|
|
root.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_02_2d_scene.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
|
In the FileSystem dock, click and drag the `Sprite.tscn` file you saved
|
2022-03-18 17:46:08 +01:00
|
|
|
|
previously onto the Node2D to instantiate it.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_03_dragging_scene.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
We want to add another node as a sibling of the Sprite. To do so, right-click on
|
|
|
|
|
Node2D and select Add Child Node.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_04_add_child_node.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Search for the Button node type and add it.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_05_add_button.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
The node is small by default. Click and drag on the bottom-right handle of the
|
|
|
|
|
Button in the viewport to resize it.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_06_drag_button.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
If you don't see the handles, ensure the select tool is active in the toolbar.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_07_select_tool.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Click and drag on the button itself to move it closer to the sprite.
|
|
|
|
|
|
|
|
|
|
You can also write a label on the Button by editing its Text property in the
|
2022-09-10 12:15:58 +02:00
|
|
|
|
Inspector. Enter "Toggle motion".
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_08_toggle_motion_text.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Your scene tree and viewport should look like this.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_09_scene_setup.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
Save your newly created scene. You can then run it with :kbd:`F6`.
|
|
|
|
|
At the moment, the button will be visible, but nothing will happen if you
|
|
|
|
|
press it.
|
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
Connecting a signal in the editor
|
|
|
|
|
---------------------------------
|
|
|
|
|
|
|
|
|
|
Here, we want to connect the Button's "pressed" signal to our Sprite, and we
|
|
|
|
|
want to call a new function that will toggle its motion on and off. We need to
|
|
|
|
|
have a script attached to the Sprite node, which we do from the previous lesson.
|
|
|
|
|
|
|
|
|
|
You can connect signals in the Node dock. Select the Button node and, on the
|
|
|
|
|
right side of the editor, click on the tab named "Node" next to the Inspector.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_10_node_dock.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
The dock displays a list of signals available on the selected node.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_11_pressed_signals.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Double-click the "pressed" signal to open the node connection window.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_12_node_connection.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
There, you can connect the signal to the Sprite node. The node needs a receiver
|
|
|
|
|
method, a function that Godot will call when the Button emits the signal. The
|
|
|
|
|
editor generates one for you. By convention, we name these callback methods
|
|
|
|
|
"_on_NodeName_signal_name". Here, it'll be "_on_Button_pressed".
|
|
|
|
|
|
2023-01-12 20:55:57 +01:00
|
|
|
|
Note:
|
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
When connecting signals via the editor's Node dock, you can use two
|
|
|
|
|
modes. The simple one only allows you to connect to nodes that have a
|
|
|
|
|
script attached to them and creates a new callback function on them.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_advanced_connection_window.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
The advanced view lets you connect to any node and any built-in
|
|
|
|
|
function, add arguments to the callback, and set options. You can
|
2022-09-10 12:15:58 +02:00
|
|
|
|
toggle the mode in the window's bottom-right by clicking the Advanced
|
2022-03-18 17:46:08 +01:00
|
|
|
|
button.
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
Click the Connect button to complete the signal connection and jump to the
|
2022-03-18 17:46:08 +01:00
|
|
|
|
Script workspace. You should see the new method with a connection icon in the
|
|
|
|
|
left margin.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_13_signals_connection_icon.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
If you click the icon, a window pops up and displays information about the
|
|
|
|
|
connection. This feature is only available when connecting nodes in the editor.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_14_signals_connection_info.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
|
Let's replace the line with the `pass` keyword with code that'll toggle the
|
2022-03-18 17:46:08 +01:00
|
|
|
|
node's motion.
|
|
|
|
|
|
2023-01-12 20:57:31 +01:00
|
|
|
|
Our Sprite moves thanks to code in the `process()` function. Godot provides a
|
2023-01-12 19:29:11 +01:00
|
|
|
|
method to toggle processing on and off: `Node.set_process()
|
2023-01-12 20:47:54 +01:00
|
|
|
|
( Node_method_set_process )`. Another method of the Node class,
|
2023-01-12 19:43:03 +01:00
|
|
|
|
`is_processing()`, returns `true` if idle processing is active. We can use
|
|
|
|
|
the `not` keyword to invert the value.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func _on_Button_pressed():
|
|
|
|
|
set_process(not is_processing())
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
This function will toggle processing and, in turn, the icon's motion on and off
|
|
|
|
|
upon pressing the button.
|
|
|
|
|
|
2023-01-12 20:57:31 +01:00
|
|
|
|
Before trying the game, we need to simplify our `process()` function to move
|
2022-03-18 17:46:08 +01:00
|
|
|
|
the node automatically and not wait for user input. Replace it with the
|
|
|
|
|
following code, which we saw two lessons ago:
|
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-09-10 12:15:58 +02:00
|
|
|
|
func _process(delta):
|
2022-03-18 17:46:08 +01:00
|
|
|
|
rotation += angular_speed * delta
|
|
|
|
|
var velocity = Vector2.UP.rotated(rotation) * speed
|
|
|
|
|
position += velocity * delta
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
|
Your complete `Sprite.gd` code should look like the following.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
extends Sprite
|
|
|
|
|
|
|
|
|
|
var speed = 400
|
|
|
|
|
var angular_speed = PI
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
|
|
|
|
|
func _process(delta):
|
2022-03-18 17:46:08 +01:00
|
|
|
|
rotation += angular_speed * delta
|
|
|
|
|
var velocity = Vector2.UP.rotated(rotation) * speed
|
|
|
|
|
position += velocity * delta
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func _on_Button_pressed():
|
|
|
|
|
set_process(not is_processing())
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Run the scene now and click the button to see the sprite start and stop.
|
|
|
|
|
|
|
|
|
|
Connecting a signal via code
|
|
|
|
|
----------------------------
|
|
|
|
|
|
|
|
|
|
You can connect signals via code instead of using the editor. This is necessary
|
|
|
|
|
when you create nodes or instantiate scenes inside of a script.
|
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
|
Let's use a different node here. Godot has a `Timer` node
|
2022-03-18 17:46:08 +01:00
|
|
|
|
that's useful to implement skill cooldown times, weapon reloading, and more.
|
|
|
|
|
|
|
|
|
|
Head back to the 2D workspace. You can either click the "2D" text at the top of
|
|
|
|
|
the window or press :kbd:`Ctrl + F1` (:kbd:`Alt + 1` on macOS).
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
In the Scene dock, right-click on the Sprite node and add a new child node.
|
|
|
|
|
Search for Timer and add the corresponding node. Your scene should now look like
|
|
|
|
|
this.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_15_scene_tree.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
With the Timer node selected, go to the Inspector and check the **Autostart**
|
|
|
|
|
property.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_18_timer_autostart.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Click the script icon next to Sprite to jump back to the scripting workspace.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_16_click_script.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
We need to do two operations to connect the nodes via code:
|
|
|
|
|
|
|
|
|
|
1. Get a reference to the Timer from the Sprite.
|
2023-01-12 19:43:03 +01:00
|
|
|
|
2. Call the Timer's `connect()` method.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 20:55:57 +01:00
|
|
|
|
Note:
|
|
|
|
|
To connect to a signal via code, you need to call the `connect()`
|
2022-03-18 17:46:08 +01:00
|
|
|
|
method of the node you want to listen to. In this case, we want to
|
|
|
|
|
listen to the Timer's "timeout" signal.
|
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
We want to connect the signal when the scene is instantiated, and we can do that
|
2023-01-12 19:30:47 +01:00
|
|
|
|
using the `Node._ready()` built-in function,
|
2022-09-10 12:15:58 +02:00
|
|
|
|
which is called automatically by the engine when a node is fully instantiated.
|
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
To get a reference to a node relative to the current one, we use the method
|
2023-01-12 19:30:47 +01:00
|
|
|
|
`Node.get_node()`. We can store the reference
|
2022-03-18 17:46:08 +01:00
|
|
|
|
in a variable.
|
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func _ready():
|
|
|
|
|
var timer = get_node("Timer")
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
|
The function `get_node()` looks at the Sprite's children and gets nodes by
|
2022-03-18 17:46:08 +01:00
|
|
|
|
their name. For example, if you renamed the Timer node to "BlinkingTimer" in the
|
2023-01-12 19:43:03 +01:00
|
|
|
|
editor, you would have to change the call to `get_node("BlinkingTimer")`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
.. add seealso to a page that explains node features.
|
|
|
|
|
|
2023-01-12 20:57:31 +01:00
|
|
|
|
We can now connect the Timer to the Sprite in the `ready()` function.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func _ready():
|
|
|
|
|
var timer = get_node("Timer")
|
|
|
|
|
timer.connect("timeout", self, "_on_Timer_timeout")
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
The line reads like so: we connect the Timer's "timeout" signal to the node to
|
2023-01-12 19:43:03 +01:00
|
|
|
|
which the script is attached (`self`). When the Timer emits "timeout", we want
|
2022-03-18 17:46:08 +01:00
|
|
|
|
to call the function "_on_Timer_timeout", that we need to define. Let's add it
|
|
|
|
|
at the bottom of our script and use it to toggle our sprite's visibility.
|
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func _on_Timer_timeout():
|
|
|
|
|
visible = not visible
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
|
The `visible` property is a boolean that controls the visibility of our node.
|
|
|
|
|
The line `visible = not visible` toggles the value. If `visible` is
|
|
|
|
|
`true`, it becomes `false`, and vice-versa.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2022-09-10 12:15:58 +02:00
|
|
|
|
If you run the scene now, you will see that the sprite blinks on and off, at one
|
|
|
|
|
second intervals.
|
|
|
|
|
|
|
|
|
|
Complete script
|
|
|
|
|
---------------
|
|
|
|
|
|
|
|
|
|
That's it for our little moving and blinking Godot icon demo!
|
2023-01-12 19:43:03 +01:00
|
|
|
|
Here is the complete `Sprite.gd` file for reference.
|
2022-09-10 12:15:58 +02:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-09-10 12:15:58 +02:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-09-10 12:15:58 +02:00
|
|
|
|
extends Sprite
|
|
|
|
|
|
|
|
|
|
var speed = 400
|
|
|
|
|
var angular_speed = PI
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func _ready():
|
|
|
|
|
var timer = get_node("Timer")
|
|
|
|
|
timer.connect("timeout", self, "_on_Timer_timeout")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func _process(delta):
|
|
|
|
|
rotation += angular_speed * delta
|
|
|
|
|
var velocity = Vector2.UP.rotated(rotation) * speed
|
|
|
|
|
position += velocity * delta
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func _on_Button_pressed():
|
|
|
|
|
set_process(not is_processing())
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func _on_Timer_timeout():
|
|
|
|
|
visible = not visible
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-09-10 12:15:58 +02:00
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
Custom signals
|
|
|
|
|
--------------
|
|
|
|
|
|
2023-01-12 20:55:57 +01:00
|
|
|
|
Note:
|
|
|
|
|
This section is a reference on how to define and use your own signals,
|
2022-09-10 12:15:58 +02:00
|
|
|
|
and does not build upon the project created in previous lessons.
|
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
You can define custom signals in a script. Say, for example, that you want to
|
|
|
|
|
show a game over screen when the player's health reaches zero. To do so, you
|
|
|
|
|
could define a signal named "died" or "health_depleted" when their health
|
|
|
|
|
reaches 0.
|
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
extends Node2D
|
|
|
|
|
|
|
|
|
|
signal health_depleted
|
|
|
|
|
|
|
|
|
|
var health = 10
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 20:55:57 +01:00
|
|
|
|
Note:
|
|
|
|
|
As signals represent events that just occurred, we generally use an
|
2022-03-18 17:46:08 +01:00
|
|
|
|
action verb in the past tense in their names.
|
|
|
|
|
|
|
|
|
|
Your signals work the same way as built-in ones: they appear in the Node tab and
|
|
|
|
|
you can connect to them like any other.
|
|
|
|
|
|
2023-01-12 20:16:00 +01:00
|
|
|
|
![](img/signals_17_custom_signal.png)
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
|
To emit a signal in your scripts, call `emit_signal()`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func take_damage(amount):
|
|
|
|
|
health -= amount
|
|
|
|
|
if health <= 0:
|
|
|
|
|
emit_signal("health_depleted")
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
A signal can optionally declare one or more arguments. Specify the argument
|
|
|
|
|
names between parentheses:
|
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
extends Node
|
|
|
|
|
|
|
|
|
|
signal health_changed(old_value, new_value)
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 20:55:57 +01:00
|
|
|
|
Note:
|
|
|
|
|
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
The signal arguments show up in the editor's node dock, and Godot can use
|
|
|
|
|
them to generate callback functions for you. However, you can still emit any
|
|
|
|
|
number of arguments when you emit signals. So it's up to you to emit the
|
|
|
|
|
correct values.
|
|
|
|
|
|
|
|
|
|
To emit values along with the signal, add them as extra arguments to the
|
2023-01-12 19:43:03 +01:00
|
|
|
|
`emit_signal()` function:
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
func take_damage(amount):
|
|
|
|
|
var old_health = health
|
|
|
|
|
health -= amount
|
|
|
|
|
emit_signal("health_changed", old_health, health)
|
2023-01-12 18:31:02 +01:00
|
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
|
|
Summary
|
|
|
|
|
-------
|
|
|
|
|
|
|
|
|
|
Any node in Godot emits signals when something specific happens to them, like a
|
|
|
|
|
button being pressed. Other nodes can connect to individual signals and react to
|
|
|
|
|
selected events.
|
|
|
|
|
|
|
|
|
|
Signals have many uses. With them, you can react to a node entering or exiting
|
|
|
|
|
the game world, to a collision, to a character entering or leaving an area, to
|
|
|
|
|
an element of the interface changing size, and much more.
|
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
|
For example, an `Area2D` representing a coin emits a
|
2023-01-12 19:43:03 +01:00
|
|
|
|
`body_entered` signal whenever the player's physics body enters its collision
|
2022-03-18 17:46:08 +01:00
|
|
|
|
shape, allowing you to know when the player collected it.
|
|
|
|
|
|
2023-01-12 19:29:11 +01:00
|
|
|
|
In the next section, `doc_your_first_2d_game`, you'll create a complete 2D
|
2022-03-18 17:46:08 +01:00
|
|
|
|
game and put everything you learned so far into practice.
|