pandemonium_engine_docs/usage/scripting/gdscript/static_typing.md

392 lines
11 KiB
Markdown
Raw Normal View History

2023-01-12 20:49:14 +01:00
Static typing in GDScript
=========================
In this guide, you will learn:
- **How to use types in GDScript**
- That **static types can help you avoid bugs**
Where and how you use this new language feature is entirely up to you:
you can use it only in some sensitive GDScript files, use it everywhere,
or write code like you always did!
Static types can be used on variables, constants, functions, parameters,
and return types.
2023-01-12 20:55:57 +01:00
Note:
2024-03-16 20:56:52 +01:00
Typed GDScript is available since Pandemonium 3.1.
A brief look at static typing
-----------------------------
2024-03-16 20:56:52 +01:00
With typed GDScript, Pandemonium can detect even more errors as you write
code! It gives you and your teammates more information as you're
working, as the arguments' types show up when you call a method.
2023-01-12 19:43:03 +01:00
Imagine you're programming an inventory system. You code an `Item`
node, then an `Inventory`. To add items to the inventory, the people
who work with your code should always pass an `Item` to the
`Inventory.add` method. With types, you can enforce this:
2023-01-12 22:32:46 +01:00
```
# In 'Item.gd'.
class_name Item
# In 'Inventory.gd'.
class_name Inventory
func add(reference: Item, amount: int = 1):
var item = find_item(reference)
if not item:
item = _instance_item_from_db(reference)
item.amount += amount
2023-01-12 22:32:46 +01:00
```
Another significant advantage of typed GDScript is the new **warning
2024-03-16 20:56:52 +01:00
system**. From version 3.1, Pandemonium gives you warnings about your code as
you write it: the engine identifies sections of your code that may lead
to issues at runtime, but lets you decide whether or not you want to
leave the code as it is. More on that in a moment.
Static types also give you better code completion options. Below, you
can see the difference between a dynamic and a static typed completion
2023-01-12 19:43:03 +01:00
options for a class called `PlayerController`.
You've probably stored a node in a variable before, and typed a dot to
be left with no autocomplete suggestions:
2023-01-12 20:16:00 +01:00
.. figure:: img/typed_gdscript_code_completion_dynamic.png)
:alt: code completion options for dynamic
2024-03-16 20:56:52 +01:00
This is due to dynamic code. Pandemonium cannot know what node or value type
you're passing to the function. If you write the type explicitly
however, you will get all public methods and variables from the node:
2023-01-12 20:16:00 +01:00
.. figure:: img/typed_gdscript_code_completion_typed.png)
:alt: code completion options for typed
In the future, typed GDScript will also increase code performance:
Just-In-Time compilation and other compiler improvements are already
on the roadmap!
Overall, typed programming gives you a more structured experience. It
helps prevent errors and improves the self-documenting aspect of your
scripts. This is especially helpful when you're working in a team or on
a long-term project: studies have shown that developers spend most of
their time reading other people's code, or scripts they wrote in the
past and forgot about. The clearer and the more structured the code, the
faster it is to understand, the faster you can move forward.
How to use static typing
------------------------
To define the type of a variable or a constant, write a colon after the
2023-01-12 19:43:03 +01:00
variable's name, followed by its type. E.g. `var health: int`. This
forces the variable's type to always stay the same:
2023-01-12 22:32:46 +01:00
```
var damage: float = 10.5
const MOVE_SPEED: float = 50.0
2023-01-12 22:32:46 +01:00
```
2024-03-16 20:56:52 +01:00
Pandemonium will try to infer types if you write a colon, but you omit the
type:
2023-01-12 22:32:46 +01:00
```
var life_points := 4
var damage := 10.5
var motion := Vector2()
2023-01-12 22:32:46 +01:00
```
Currently you can use three types of… types:
2023-01-12 20:47:54 +01:00
1. `Built-in ( doc_gdscript_builtin_types )`
2023-01-12 19:43:03 +01:00
2. Core classes and nodes (`Object`, `Node`, `Area2D`,
`Camera2D`, etc.)
2023-01-12 20:47:54 +01:00
3. Your own custom classes. Look at the new `name ( doc_gdscript_basics_class_name )`
feature to register types in the editor.
2023-01-12 20:55:57 +01:00
Note:
2024-03-16 20:56:52 +01:00
You don't need to write type hints for constants, as Pandemonium sets it automatically from the assigned value. But you can still do so to make the intent of your code clearer.
Custom variable types
~~~~~~~~~~~~~~~~~~~~~
You can use any class, including your custom classes, as types. There
are two ways to use them in scripts. The first method is to preload the
script you want to use as a type in a constant:
2023-01-12 22:32:46 +01:00
```
const Rifle = preload("res://player/weapons/Rifle.gd")
var my_rifle: Rifle
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
The second method is to use the `name` keyword when you create.
For the example above, your Rifle.gd would look like this:
2023-01-12 22:32:46 +01:00
```
extends Node2D
class_name Rifle
2023-01-12 22:32:46 +01:00
```
2024-03-16 20:56:52 +01:00
If you use `name`, Pandemonium registers the Rifle type globally in
the editor, and you can use it anywhere, without having to preload it
into a constant:
2023-01-12 22:32:46 +01:00
```
var my_rifle: Rifle
2023-01-12 22:32:46 +01:00
```
Variable casting
~~~~~~~~~~~~~~~~
Type casting is a key concept in typed languages.
Casting is the conversion of a value from one type to another.
2023-01-12 19:43:03 +01:00
Imagine an Enemy in your game, that `extends Area2D`. You want it to
collide with the Player, a `KinematicBody2D` with a script called
`PlayerController` attached to it. You use the `on_body_entered`
signal to detect the collision. With typed code, the body you detect is
2023-01-12 19:43:03 +01:00
going to be a generic `PhysicsBody2D`, and not your
2023-01-12 20:57:31 +01:00
`PlayerController` on the `on_body_entered` callback.
2023-01-12 19:43:03 +01:00
You can check if this `PhysicsBody2D` is your Player with the `as`
casting keyword, and using the colon `:` again to force the variable
to use this type. This forces the variable to stick to the
2023-01-12 19:43:03 +01:00
`PlayerController` type:
2023-01-12 22:32:46 +01:00
```
func _on_body_entered(body: PhysicsBody2D) -> void:
var player := body as PlayerController
if not player:
return
player.damage()
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
As we're dealing with a custom type, if the `body` doesn't extend
`PlayerController`, the `player`\ variable will be set to `null`.
We can use this to check if the body is the player or not. We will also
get full autocompletion on the player variable thanks to that cast.
2023-01-12 20:55:57 +01:00
Note:
2024-03-16 20:56:52 +01:00
If you try to cast with a built-in type and it fails, Pandemonium will throw an error.
2023-01-12 20:49:14 +01:00
Safe lines
^^^^^^^^^^
You can also use casting to ensure safe lines. Safe lines are a new
2024-03-16 20:56:52 +01:00
tool in Pandemonium 3.1 to tell you when ambiguous lines of code are
type-safe. As you can mix and match typed and dynamic code, at times,
2024-03-16 20:56:52 +01:00
Pandemonium doesn't have enough information to know if an instruction will trigger
an error or not at runtime.
This happens when you get a child node. Let's take a timer for example:
2023-01-12 19:43:03 +01:00
with dynamic code, you can get the node with `$Timer`. GDScript
2023-01-12 20:39:50 +01:00
supports `duck-typing ( https://stackoverflow.com/a/4205163/8125343 )`,
2023-01-12 19:43:03 +01:00
so even if your timer is of type `Timer`, it is also a `Node` and an
`Object`, two classes it extends. With dynamic GDScript, you also
don't care about the node's type as long as it has the methods you need
to call.
2024-03-16 20:56:52 +01:00
You can use casting to tell Pandemonium the type you expect when you get a
2023-01-12 19:43:03 +01:00
node: `($Timer as Timer)`, `($Player as KinematicBody2D)`, etc.
2024-03-16 20:56:52 +01:00
Pandemonium will ensure the type works and if so, the line number will turn
green at the left of the script editor.
2023-01-12 20:16:00 +01:00
.. figure:: img/typed_gdscript_safe_unsafe_line.png)
:alt: Unsafe vs Safe Line
Unsafe line (line 7) vs Safe Lines (line 6 and 8)
2023-01-12 20:55:57 +01:00
Note:
You can turn off safe lines or change their color in the editor settings.
Define the return type of a function with the arrow ->
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
To define the return type of a function, write a dash and a right angle
2023-01-12 20:39:50 +01:00
bracket `- )` after its declaration, followed by the return type:
2023-01-12 22:32:46 +01:00
```
func _process(delta: float) -> void:
pass
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
The type `void` means the function does not return anything. You can
use any type, as with variables:
2023-01-12 22:32:46 +01:00
```
func hit(damage: float) -> bool:
health_points -= damage
return health_points <= 0
2023-01-12 22:32:46 +01:00
```
You can also use your own nodes as return types:
2023-01-12 22:32:46 +01:00
```
# Inventory.gd
# Adds an item to the inventory and returns it.
func add(reference: Item, amount: int) -> Item:
var item: Item = find_item(reference)
if not item:
item = ItemDatabase.get_instance(reference)
item.amount += amount
return item
2023-01-12 22:32:46 +01:00
```
Typed or dynamic: stick to one style
------------------------------------
Typed GDScript and dynamic GDScript can coexist in the same project. But
it's recommended to stick to either style for consistency in your codebase,
and for your peers. It's easier for everyone to work together if you
follow the same guidelines, and faster to read and understand other
people's code.
Typed code takes a little more writing, but you get the benefits we
discussed above. Here's an example of the same, empty script, in a
dynamic style:
2023-01-12 22:32:46 +01:00
```
extends Node
func _ready():
pass
func _process(delta):
pass
2023-01-12 22:32:46 +01:00
```
And with static typing:
2023-01-12 22:32:46 +01:00
```
extends Node
func _ready() -> void:
pass
func _process(delta: float) -> void:
pass
2023-01-12 22:32:46 +01:00
```
As you can see, you can also use types with the engine's virtual
methods. Signal callbacks, like any methods, can also use types. Here's
2023-01-12 19:43:03 +01:00
a `body_entered` signal in a dynamic style:
2023-01-12 22:32:46 +01:00
```
func _on_Area2D_body_entered(body):
pass
2023-01-12 22:32:46 +01:00
```
And the same callback, with type hints:
2023-01-12 22:32:46 +01:00
```
func _on_area_entered(area: CollisionObject2D) -> void:
pass
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
You're free to replace, e.g. the `CollisionObject2D`, with your own type,
to cast parameters automatically:
2023-01-12 22:32:46 +01:00
```
func _on_area_entered(bullet: Bullet) -> void:
if not bullet:
return
take_damage(bullet.damage)
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
The `bullet` variable could hold any `CollisionObject2D` here, but
we make sure it is our `Bullet`, a node we created for our project. If
it's anything else, like an `Area2D`, or any node that doesn't extend
`Bullet`, the `bullet` variable will be `null`.
Warning system
--------------
2023-01-12 20:55:57 +01:00
Note:
Documentation about the GDScript warning system has been moved to
2023-01-12 19:29:11 +01:00
`doc_gdscript_warning_system`.
Cases where you can't specify types
-----------------------------------
To wrap up this introduction, let's cover a few cases where you can't
use type hints. All the examples below **will trigger errors**.
You can't use Enums as types:
2023-01-12 22:32:46 +01:00
```
enum MoveDirection {UP, DOWN, LEFT, RIGHT}
var current_direction: MoveDirection
2023-01-12 22:32:46 +01:00
```
You can't specify the type of individual members in an array. This will
give you an error:
2023-01-12 22:32:46 +01:00
```
var enemies: Array = [$Goblin: Enemy, $Zombie: Enemy]
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
You can't force the assignment of types in a `for` loop, as each
element the `for` keyword loops over already has a different type. So you
**cannot** write:
2023-01-12 22:32:46 +01:00
```
var names = ["John", "Marta", "Samantha", "Jimmy"]
for name: String in names:
pass
2023-01-12 22:32:46 +01:00
```
Two scripts can't depend on each other in a cyclic fashion:
2023-01-12 22:32:46 +01:00
```
# Player.gd
extends Area2D
class_name Player
var rifle: Rifle
2023-01-12 22:32:46 +01:00
```
2023-01-12 22:32:46 +01:00
```
# Rifle.gd
extends Area2D
class_name Rifle
var player: Player
2023-01-12 22:32:46 +01:00
```
Summary
-------
2024-03-16 20:56:52 +01:00
Typed GDScript is a powerful tool. Available as of version 3.1 of Pandemonium, it
helps you write more structured code, avoid common errors, and
create scalable systems. In the future, static types will also bring you
a nice performance boost thanks to upcoming compiler optimizations.