pandemonium_engine_docs/usage/scripting/gdscript/gdscript_styleguide.md

854 lines
19 KiB
Markdown
Raw Normal View History

2023-01-12 20:49:14 +01:00
GDScript style guide
====================
This style guide lists conventions to write elegant GDScript. The goal is to
encourage writing clean, readable code and promote consistency across projects,
discussions, and tutorials. Hopefully, this will also support the development of
auto-formatting tools.
Since GDScript is close to Python, this guide is inspired by Python's
2023-01-12 20:39:50 +01:00
`PEP 8 ( https://www.python.org/dev/peps/pep-0008/ )` programming
style guide.
Style guides aren't meant as hard rulebooks. At times, you may not be able to
apply some of the guidelines below. When that happens, use your best judgment,
and ask fellow developers for insights.
In general, keeping your code consistent in your projects and within your team is
more important than following this guide to a tee.
2023-01-12 20:55:57 +01:00
Note:
2024-03-16 20:56:52 +01:00
Pandemonium's built-in script editor uses a lot of these conventions
by default. Let it help you.
Here is a complete class example based on these guidelines:
2023-01-12 22:32:46 +01:00
```
class_name StateMachine
extends Node
# Hierarchical State machine for the player.
# Initializes states and delegates engine callbacks
# (_physics_process, _unhandled_input) to the state.
signal state_changed(previous, new)
export var initial_state = NodePath()
var is_active = true setget set_is_active
onready var _state = get_node(initial_state) setget set_state
onready var _state_name = _state.name
func _init():
add_to_group("state_machine")
func _ready():
connect("state_changed", self, "_on_state_changed")
_state.enter()
func _unhandled_input(event):
_state.unhandled_input(event)
func _physics_process(delta):
_state.physics_process(delta)
func transition_to(target_state_path, msg={}):
if not has_node(target_state_path):
return
var target_state = get_node(target_state_path)
assert(target_state.is_composite == false)
_state.exit()
self._state = target_state
_state.enter(msg)
Events.emit_signal("player_state_changed", _state.name)
func set_is_active(value):
is_active = value
set_physics_process(value)
set_process_unhandled_input(value)
set_block_signals(not value)
func set_state(value):
_state = value
_state_name = _state.name
func _on_state_changed(previous, new):
print("state changed")
emit_signal("state_changed")
2023-01-12 22:32:46 +01:00
```
.. _formatting:
Formatting
----------
Encoding and special characters
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* Use line feed (**LF**) characters to break lines, not CRLF or CR. *(editor default)*
* Use one line feed character at the end of each file. *(editor default)*
2023-01-12 20:57:31 +01:00
* Use **UTF-8** encoding without a `byte order mark ( https://en.wikipedia.org/wiki/Byte_order_mark )`. *(editor default)*
* Use **Tabs** instead of spaces for indentation. *(editor default)*
Indentation
~~~~~~~~~~~
Each indent level should be one greater than the block containing it.
**Good**:
2023-01-12 22:32:46 +01:00
```
for i in range(10):
print("hello")
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
for i in range(10):
print("hello")
for i in range(10):
print("hello")
2023-01-12 22:32:46 +01:00
```
Use 2 indent levels to distinguish continuation lines from
regular code blocks.
**Good**:
2023-01-12 22:32:46 +01:00
```
effect.interpolate_property(sprite, "transform/scale",
sprite.get_scale(), Vector2(2.0, 2.0), 0.3,
Tween.TRANS_QUAD, Tween.EASE_OUT)
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
effect.interpolate_property(sprite, "transform/scale",
sprite.get_scale(), Vector2(2.0, 2.0), 0.3,
Tween.TRANS_QUAD, Tween.EASE_OUT)
2023-01-12 22:32:46 +01:00
```
Exceptions to this rule are arrays, dictionaries, and enums. Use a single
indentation level to distinguish continuation lines:
**Good**:
2023-01-12 22:32:46 +01:00
```
var party = [
2024-03-16 20:56:52 +01:00
"Pandemonium",
"Godette",
"Steve",
]
var character_dict = {
"Name": "Bob",
"Age": 27,
"Job": "Mechanic",
}
enum Tiles {
TILE_BRICK,
TILE_FLOOR,
TILE_SPIKE,
TILE_TELEPORT,
}
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
var party = [
2024-03-16 20:56:52 +01:00
"Pandemonium",
"Godette",
"Steve",
]
var character_dict = {
"Name": "Bob",
"Age": 27,
"Job": "Mechanic",
}
enum Tiles {
TILE_BRICK,
TILE_FLOOR,
TILE_SPIKE,
TILE_TELEPORT,
}
2023-01-12 22:32:46 +01:00
```
Trailing comma
~~~~~~~~~~~~~~
Use a trailing comma on the last line in arrays, dictionaries, and enums. This
results in easier refactoring and better diffs in version control as the last
line doesn't need to be modified when adding new elements.
**Good**:
2023-01-12 22:32:46 +01:00
```
enum Tiles {
TILE_BRICK,
TILE_FLOOR,
TILE_SPIKE,
TILE_TELEPORT,
}
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
enum Tiles {
TILE_BRICK,
TILE_FLOOR,
TILE_SPIKE,
TILE_TELEPORT
}
2023-01-12 22:32:46 +01:00
```
Trailing commas are unnecessary in single-line lists, so don't add them in this case.
**Good**:
2023-01-12 22:32:46 +01:00
```
enum Tiles {TILE_BRICK, TILE_FLOOR, TILE_SPIKE, TILE_TELEPORT}
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
enum Tiles {TILE_BRICK, TILE_FLOOR, TILE_SPIKE, TILE_TELEPORT,}
2023-01-12 22:32:46 +01:00
```
Blank lines
~~~~~~~~~~~
Surround functions and class definitions with two blank lines:
2023-01-12 22:32:46 +01:00
```
func heal(amount):
health += amount
health = min(health, max_health)
emit_signal("health_changed", health)
func take_damage(amount, effect=null):
health -= amount
health = max(0, health)
emit_signal("health_changed", health)
2023-01-12 22:32:46 +01:00
```
Use one blank line inside functions to separate logical sections.
2023-01-12 20:55:57 +01:00
Note:
We use a single line between classes and function definitions in the class reference and
in short code snippets in this documentation.
Line length
~~~~~~~~~~~
Keep individual lines of code under 100 characters.
If you can, try to keep lines under 80 characters. This helps to read the code
on small displays and with two scripts opened side-by-side in an external text
editor. For example, when looking at a differential revision.
One statement per line
~~~~~~~~~~~~~~~~~~~~~~
Never combine multiple statements on a single line. No, C programmers,
not even with a single line conditional statement.
**Good**:
2023-01-12 22:32:46 +01:00
```
if position.x > width:
position.x = 0
if flag:
print("flagged")
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
if position.x > width: position.x = 0
if flag: print("flagged")
2023-01-12 22:32:46 +01:00
```
The only exception to that rule is the ternary operator:
2023-01-12 22:32:46 +01:00
```
next_state = "fall" if not is_on_floor() else "idle"
2023-01-12 22:32:46 +01:00
```
Format multiline statements for readability
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
2023-01-12 19:43:03 +01:00
When you have particularly long `if` statements or nested ternary expressions,
wrapping them over multiple lines improves readability. Since continuation lines
are still part of the same expression, 2 indent levels should be used instead of one.
GDScript allows wrapping statements using multiple lines using parentheses or
backslashes. Parentheses are favored in this style guide since they make for
easier refactoring. With backslashes, you have to ensure that the last line
never contains a backslash at the end. With parentheses, you don't have to
worry about the last line having a backslash at the end.
2023-01-12 19:43:03 +01:00
When wrapping a conditional expression over multiple lines, the `and`/`or`
keywords should be placed at the beginning of the line continuation, not at the
end of the previous line.
**Good**:
2023-01-12 22:32:46 +01:00
```
var angle_degrees = 135
var quadrant = (
"northeast" if angle_degrees <= 90
else "southeast" if angle_degrees <= 180
else "southwest" if angle_degrees <= 270
else "northwest"
)
var position = Vector2(250, 350)
if (
position.x > 200 and position.x < 400
and position.y > 300 and position.y < 400
):
pass
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
var angle_degrees = 135
var quadrant = "northeast" if angle_degrees <= 90 else "southeast" if angle_degrees <= 180 else "southwest" if angle_degrees <= 270 else "northwest"
var position = Vector2(250, 350)
if position.x > 200 and position.x < 400 and position.y > 300 and position.y < 400:
pass
2023-01-12 22:32:46 +01:00
```
Avoid unnecessary parentheses
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Avoid parentheses in expressions and conditional statements. Unless
necessary for order of operations or wrapping over multiple lines,
they only reduce readability.
**Good**:
2023-01-12 22:32:46 +01:00
```
if is_colliding():
queue_free()
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
if (is_colliding()):
queue_free()
2023-01-12 22:32:46 +01:00
```
Boolean operators
~~~~~~~~~~~~~~~~~
Prefer the plain English versions of boolean operators, as they are the most accessible:
2023-01-12 19:43:03 +01:00
- Use `and` instead of `&&`.
- Use `or` instead of `||`.
You may also use parentheses around boolean operators to clear any ambiguity.
This can make long expressions easier to read.
**Good**:
2023-01-12 22:32:46 +01:00
```
if (foo and bar) or baz:
print("condition is true")
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
if foo && bar || baz:
print("condition is true")
2023-01-12 22:32:46 +01:00
```
Comment spacing
~~~~~~~~~~~~~~~
Regular comments should start with a space, but not code that you comment out.
This helps differentiate text comments from disabled code.
**Good**:
2023-01-12 22:32:46 +01:00
```
# This is a comment.
#print("This is disabled code")
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
#This is a comment.
# print("This is disabled code")
2023-01-12 22:32:46 +01:00
```
2023-01-12 20:55:57 +01:00
Note:
In the script editor, to toggle the selected code commented, press
:kbd:`Ctrl + K`. This feature adds a single # sign at the start
of the selected lines.
Whitespace
~~~~~~~~~~
Always use one space around operators and after commas. Also, avoid extra spaces
in dictionary references and function calls.
**Good**:
2023-01-12 22:32:46 +01:00
```
position.x = 5
position.y = target_position.y + 10
dict["key"] = 5
my_array = [4, 5, 6]
print("foo")
2023-01-12 22:32:46 +01:00
```
**Bad**:
2023-01-12 22:32:46 +01:00
```
position.x=5
position.y = mpos.y+10
dict ["key"] = 5
myarray = [4,5,6]
print ("foo")
2023-01-12 22:32:46 +01:00
```
Don't use spaces to align expressions vertically:
2023-01-12 22:32:46 +01:00
```
x = 100
y = 100
velocity = 500
2023-01-12 22:32:46 +01:00
```
Quotes
~~~~~~
Use double quotes unless single quotes make it possible to escape fewer
characters in a given string. See the examples below:
2023-01-12 22:32:46 +01:00
```
# Normal string.
print("hello world")
# Use double quotes as usual to avoid escapes.
print("hello 'world'")
# Use single quotes as an exception to the rule to avoid escapes.
print('hello "world"')
# Both quote styles would require 2 escapes; prefer double quotes if it's a tie.
print("'hello' \"world\"")
2023-01-12 22:32:46 +01:00
```
Numbers
~~~~~~~
Don't omit the leading or trailing zero in floating-point numbers. Otherwise,
this makes them less readable and harder to distinguish from integers at a
glance.
2023-01-12 22:32:46 +01:00
**Good**:
```
var float_number = 0.234
var other_float_number = 13.0
2023-01-12 22:32:46 +01:00
```
2023-01-12 22:32:46 +01:00
**Bad**:
```
var float_number = .234
var other_float_number = 13.
2023-01-12 22:32:46 +01:00
```
Use lowercase for letters in hexadecimal numbers, as their lower height makes
the number easier to read.
2023-01-12 22:32:46 +01:00
**Good**:
```
var hex_number = 0xfb8c0b
2023-01-12 22:32:46 +01:00
```
2023-01-12 22:32:46 +01:00
**Bad**:
```
var hex_number = 0xFB8C0B
2023-01-12 22:32:46 +01:00
```
Take advantage of GDScript's underscores in literals to make large numbers more
readable.
2023-01-12 22:32:46 +01:00
**Good**:
```
var large_number = 1_234_567_890
var large_hex_number = 0xffff_f8f8_0000
var large_bin_number = 0b1101_0010_1010
# Numbers lower than 1000000 generally don't need separators.
var small_number = 12345
2023-01-12 22:32:46 +01:00
```
2023-01-12 22:32:46 +01:00
**Bad**:
```
var large_number = 1234567890
var large_hex_number = 0xfffff8f80000
var large_bin_number = 0b110100101010
# Numbers lower than 1000000 generally don't need separators.
var small_number = 12_345
2023-01-12 22:32:46 +01:00
```
.. _naming_conventions:
Naming conventions
------------------
2024-03-16 20:56:52 +01:00
These naming conventions follow the Pandemonium Engine style. Breaking these will make
your code clash with the built-in naming conventions, leading to inconsistent
code.
File names
~~~~~~~~~~
Use snake_case for file names. For named classes, convert the PascalCase class
2023-01-12 22:32:46 +01:00
name to snake_case:
```
# This file should be saved as `weapon.gd`.
class_name Weapon
extends Node
2023-01-12 22:32:46 +01:00
```
2023-01-12 22:32:46 +01:00
```
# This file should be saved as `yaml_parser.gd`.
class_name YAMLParser
extends Object
2023-01-12 22:32:46 +01:00
```
2024-03-16 20:56:52 +01:00
This is consistent with how C++ files are named in Pandemonium's source code. This
also avoids case sensitivity issues that can crop up when exporting a project
from Windows to other platforms.
Classes and nodes
~~~~~~~~~~~~~~~~~
Use PascalCase for class and node names:
2023-01-12 22:32:46 +01:00
```
extends KinematicBody
2023-01-12 22:32:46 +01:00
```
Also use PascalCase when loading a class into a constant or a variable:
2023-01-12 22:32:46 +01:00
```
const Weapon = preload("res://weapon.gd")
2023-01-12 22:32:46 +01:00
```
Functions and variables
~~~~~~~~~~~~~~~~~~~~~~~
Use snake\_case to name functions and variables:
2023-01-12 22:32:46 +01:00
```
var particle_effect
func load_level():
2023-01-12 22:32:46 +01:00
```
Prepend a single underscore (\_) to virtual methods functions the user must
override, private functions, and private variables:
2023-01-12 22:32:46 +01:00
```
var _counter = 0
func _recalculate_path():
2023-01-12 22:32:46 +01:00
```
Signals
~~~~~~~
Use the past tense to name signals:
2023-01-12 22:32:46 +01:00
```
signal door_opened
signal score_changed
2023-01-12 22:32:46 +01:00
```
Constants and enums
~~~~~~~~~~~~~~~~~~~
Write constants with CONSTANT\_CASE, that is to say in all caps with an
underscore (\_) to separate words:
2023-01-12 22:32:46 +01:00
```
const MAX_SPEED = 200
2023-01-12 22:32:46 +01:00
```
Use PascalCase for enum *names* and CONSTANT\_CASE for their members, as they
are constants:
2023-01-12 22:32:46 +01:00
```
enum Element {
EARTH,
WATER,
AIR,
FIRE,
}
2023-01-12 22:32:46 +01:00
```
Code order
----------
This first section focuses on code order. For formatting, see
2023-01-12 19:29:11 +01:00
`formatting`. For naming conventions, see `naming_conventions`.
We suggest to organize GDScript code this way:
2023-01-12 22:32:46 +01:00
```
01. tool
02. class_name
03. extends
04. # docstring
05. signals
06. enums
07. constants
08. exported variables
09. public variables
10. private variables
11. onready variables
12. optional built-in virtual _init method
13. built-in virtual _ready method
14. remaining built-in virtual methods
15. public methods
16. private methods
2023-01-12 22:32:46 +01:00
```
We optimized the order to make it easy to read the code from top to bottom, to
help developers reading the code for the first time understand how it works, and
to avoid errors linked to the order of variable declarations.
This code order follows four rules of thumb:
1. Properties and signals come first, followed by methods.
2. Public comes before private.
3. Virtual callbacks come before the class's interface.
2023-01-12 20:57:31 +01:00
4. The object's construction and initialization functions, `init` and
`ready`, come before functions that modify the object at runtime.
Class declaration
~~~~~~~~~~~~~~~~~
2023-01-12 19:43:03 +01:00
If the code is meant to run in the editor, place the `tool` keyword on the
first line of the script.
2023-01-12 19:32:38 +01:00
Follow with the `name` if necessary. You can turn a GDScript file into a
global type in your project using this feature. For more information, see
2023-01-12 19:29:11 +01:00
`doc_gdscript`.
Then, add the `extends` keyword if the class extends a built-in type.
Following that, you should have the class's optional docstring as comments. You
can use that to explain the role of your class to your teammates, how it works,
and how other developers should use it, for example.
2023-01-12 22:32:46 +01:00
```
class_name MyNode
extends Node
# A brief description of the class's role and functionality.
# Longer description.
2023-01-12 22:32:46 +01:00
```
Signals and properties
~~~~~~~~~~~~~~~~~~~~~~
Write signal declarations, followed by properties, that is to say, member
variables, after the docstring.
Enums should come after signals, as you can use them as export hints for other
properties.
Then, write constants, exported variables, public, private, and onready
variables, in that order.
2023-01-12 22:32:46 +01:00
```
signal spawn_player(position)
enum Jobs {KNIGHT, WIZARD, ROGUE, HEALER, SHAMAN}
const MAX_LIVES = 3
export(Jobs) var job = Jobs.KNIGHT
export var max_health = 50
export var attack = 5
var health = max_health setget set_health
var _speed = 300.0
onready var sword = get_node("Sword")
onready var gun = get_node("Gun")
2023-01-12 22:32:46 +01:00
```
2023-01-12 20:55:57 +01:00
Note:
2023-01-12 20:57:31 +01:00
The GDScript compiler evaluates onready variables right before the `ready`
callback. You can use that to cache node dependencies, that is to say, to get
child nodes in the scene that your class relies on. This is what the example
above shows.
Member variables
~~~~~~~~~~~~~~~~
Don't declare member variables if they are only used locally in a method, as it
makes the code more difficult to follow. Instead, declare them as local
variables in the method's body.
Local variables
~~~~~~~~~~~~~~~
Declare local variables as close as possible to their first use. This makes it
easier to follow the code, without having to scroll too much to find where the
variable was declared.
Methods and static functions
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
After the class's properties come the methods.
2023-01-12 20:57:31 +01:00
Start with the `init()` callback method, that the engine will call upon
2024-03-16 20:56:52 +01:00
creating the object in memory. Follow with the `ready()` callback, that Pandemonium
calls when it adds a node to the scene tree.
These functions should come first because they show how the object is
initialized.
2023-01-12 20:57:31 +01:00
Other built-in virtual callbacks, like `unhandled_input()` and
`physics_process`, should come next. These control the object's main loop and
interactions with the game engine.
The rest of the class's interface, public and private methods, come after that,
in that order.
2023-01-12 22:32:46 +01:00
```
func _init():
add_to_group("state_machine")
func _ready():
connect("state_changed", self, "_on_state_changed")
_state.enter()
func _unhandled_input(event):
_state.unhandled_input(event)
func transition_to(target_state_path, msg={}):
if not has_node(target_state_path):
return
var target_state = get_node(target_state_path)
assert(target_state.is_composite == false)
_state.exit()
self._state = target_state
_state.enter(msg)
Events.emit_signal("player_state_changed", _state.name)
func _on_state_changed(previous, new):
print("state changed")
emit_signal("state_changed")
2023-01-12 22:32:46 +01:00
```
Static typing
-------------
2024-03-16 20:56:52 +01:00
Since Pandemonium 3.1, GDScript supports `optional static typing( doc_gdscript_static_typing )`.
Declared types
~~~~~~~~~~~~~~
2023-01-12 20:39:50 +01:00
To declare a variable's type, use `<variable>: <type )`:
2023-01-12 22:32:46 +01:00
```
var health: int = 0
2023-01-12 22:32:46 +01:00
```
2023-01-12 20:39:50 +01:00
To declare the return type of a function, use `-> <type )`:
2023-01-12 22:32:46 +01:00
```
func heal(amount: int) -> void:
2023-01-12 22:32:46 +01:00
```
Inferred types
~~~~~~~~~~~~~~
2023-01-12 19:43:03 +01:00
In most cases you can let the compiler infer the type, using `:=`:
2023-01-12 22:32:46 +01:00
```
var health := 0 # The compiler will use the int type.
2023-01-12 22:32:46 +01:00
```
However, in a few cases when context is missing, the compiler falls back to
2023-01-12 19:43:03 +01:00
the function's return type. For example, `get_node()` cannot infer a type
unless the scene or file of the node is loaded in memory. In this case, you
should set the type explicitly.
**Good**:
2023-01-12 22:32:46 +01:00
```
onready var health_bar: ProgressBar = get_node("UI/LifeBar")
2023-01-12 22:32:46 +01:00
```
2023-01-12 19:43:03 +01:00
Alternatively, you can use the `as` keyword to cast the return type, and
that type will be used to infer the type of the var.
2023-01-12 22:32:46 +01:00
.. rst-class: code-example-good
2023-01-12 22:32:46 +01:00
```
onready var health_bar := get_node("UI/LifeBar") as ProgressBar
# health_bar will be typed as ProgressBar
2023-01-12 22:32:46 +01:00
```
2023-01-12 20:47:54 +01:00
This option is also considered more `type-safe( doc_gdscript_static_typing_safe_lines )` than the first.
**Bad**:
2023-01-12 22:32:46 +01:00
```
# The compiler can't infer the exact type and will use Node
# instead of ProgressBar.
onready var health_bar := get_node("UI/LifeBar")
2023-01-12 22:32:46 +01:00
```