pandemonium_engine_docs/tutorials/best_practices/godot_interfaces.md

296 lines
9.7 KiB
Markdown
Raw Normal View History

2023-01-12 20:49:14 +01:00
Godot interfaces
================
Often one needs scripts that rely on other objects for features. There
are 2 parts to this process:
1. Acquiring a reference to the object that presumably has the features.
2. Accessing the data or logic from the object.
The rest of this tutorial outlines the various ways of doing all this.
Acquiring object references
---------------------------
2023-01-12 19:30:47 +01:00
For all `Object`\s, the most basic way of referencing them
is to get a reference to an existing object from another acquired instance.
2023-01-12 18:31:02 +01:00
gdscript GDScript
2023-01-12 18:31:02 +01:00
```
var obj = node.object # Property access.
var obj = node.get_object() # Method access.
2023-01-12 18:31:02 +01:00
```
2023-01-12 19:30:47 +01:00
The same principle applies for `Reference` objects.
While users often access `Node` and
`Resource` this way, alternative measures are available.
Instead of property or method access, one can get Resources by load
access.
2023-01-12 18:31:02 +01:00
gdscript GDScript
2023-01-12 18:31:02 +01:00
```
var preres = preload(path) # Load resource during scene load
var res = load(path) # Load resource when program reaches statement
# Note that users load scenes and scripts, by convention, with PascalCase
# names (like typenames), often into constants.
const MyScene : = preload("my_scene.tscn") as PackedScene # Static load
const MyScript : = preload("my_script.gd") as Script
# This type's value varies, i.e. it is a variable, so it uses snake_case.
export(Script) var script_type: Script
# If need an "export const var" (which doesn't exist), use a conditional
# setter for a tool script that checks if it's executing in the editor.
tool # Must place at top of file.
# Must configure from the editor, defaults to null.
export(Script) var const_script setget set_const_script
func set_const_script(value):
if Engine.is_editor_hint():
const_script = value
# Warn users if the value hasn't been set.
func _get_configuration_warning():
if not const_script:
return "Must initialize property 'const_script'."
return ""
2023-01-12 18:31:02 +01:00
```
Note the following:
1. There are many ways in which a language can load such resources.
2. When designing how objects will access data, don't forget
that one can pass resources around as references as well.
3. Keep in mind that loading a resource fetches the cached resource
instance maintained by the engine. To get a new object, one must
2023-01-12 19:30:47 +01:00
`duplicate` an existing reference
2023-01-12 19:43:03 +01:00
or instantiate one from scratch with `new()`.
Nodes likewise have an alternative access point: the SceneTree.
2023-01-12 18:31:02 +01:00
gdscript GDScript
2023-01-12 18:31:02 +01:00
```
extends Node
# Slow.
func dynamic_lookup_with_dynamic_nodepath():
print(get_node("Child"))
# Faster. GDScript only.
func dynamic_lookup_with_cached_nodepath():
print($Child)
# Fastest. Doesn't break if node moves later.
# Note that `onready` keyword is GDScript only.
# Other languages must do...
# var child
# func _ready():
# child = get_node("Child")
onready var child = $Child
func lookup_and_cache_for_future_access():
print(child)
# Delegate reference assignment to an external source.
# Con: need to perform a validation check.
# Pro: node makes no requirements of its external structure.
# 'prop' can come from anywhere.
var prop
func call_me_after_prop_is_initialized_by_parent():
# Validate prop in one of three ways.
# Fail with no notification.
if not prop:
return
# Fail with an error message.
if not prop:
printerr("'prop' wasn't initialized")
return
# Fail and terminate.
# Note: Scripts run from a release export template don't
# run `assert` statements.
assert(prop, "'prop' wasn't initialized")
# Use an autoload.
# Dangerous for typical nodes, but useful for true singleton nodes
# that manage their own data and don't interfere with other objects.
func reference_a_global_autoloaded_variable():
print(globals)
print(globals.prop)
print(globals.my_getter())
2023-01-12 18:31:02 +01:00
```
2023-01-12 20:49:14 +01:00
Accessing data or logic from an object
--------------------------------------
Godot's scripting API is duck-typed. This means that if a script executes an
operation, Godot doesn't validate that it supports the operation by **type**.
It instead checks that the object **implements** the individual method.
2023-01-12 19:43:03 +01:00
For example, the `CanvasItem` class has a `visible`
property. All properties exposed to the scripting API are in fact a setter and
getter pair bound to a name. If one tried to access
2023-01-12 19:30:47 +01:00
`CanvasItem.visible`, then Godot would do the
following checks, in order:
- If the object has a script attached, it will attempt to set the property
through the script. This leaves open the opportunity for scripts to override
a property defined on a base object by overriding the setter method for the
property.
- If the script does not have the property, it performs a HashMap lookup in
the ClassDB for the "visible" property against the CanvasItem class and all
of its inherited types. If found, it will call the bound setter or getter.
For more information about HashMaps, see the
2023-01-12 20:47:54 +01:00
`data preferences ( doc_data_preferences )` docs.
- If not found, it does an explicit check to see if the user wants to access
the "script" or "meta" properties.
2023-01-12 19:43:03 +01:00
- If not, it checks for a `_set`/`_get` implementation (depending on type
of access) in the CanvasItem and its inherited types. These methods can
execute logic that gives the impression that the Object has a property. This
2023-01-12 19:43:03 +01:00
is also the case with the `_get_property_list` method.
- Note that this happens even for non-legal symbol names such as in the
2023-01-12 19:30:47 +01:00
case of `TileSet`'s "1/tile_name" property. This
refers to the name of the tile with ID 1, i.e.
2023-01-12 19:30:47 +01:00
`TileSet.tile_get_name(1)`.
As a result, this duck-typed system can locate a property either in the script,
the object's class, or any class that object inherits, but only for things
which extend Object.
Godot provides a variety of options for performing runtime checks on these
accesses:
- A duck-typed property access. These will property check (as described above).
If the operation isn't supported by the object, execution will halt.
2023-01-12 18:31:02 +01:00
gdscript GDScript
2023-01-12 18:31:02 +01:00
```
# All Objects have duck-typed get, set, and call wrapper methods.
get_parent().set("visible", false)
# Using a symbol accessor, rather than a string in the method call,
# will implicitly call the `set` method which, in turn, calls the
# setter method bound to the property through the property lookup
# sequence.
get_parent().visible = false
# Note that if one defines a _set and _get that describe a property's
# existence, but the property isn't recognized in any _get_property_list
# method, then the set() and get() methods will work, but the symbol
# access will claim it can't find the property.
2023-01-12 18:31:02 +01:00
```
- A method check. In the case of
2023-01-12 19:30:47 +01:00
`CanvasItem.visible`, one can
2023-01-12 19:43:03 +01:00
access the methods, `set_visible` and `is_visible` like any other method.
2023-01-12 18:31:02 +01:00
gdscript GDScript
2023-01-12 18:31:02 +01:00
```
var child = get_child(0)
# Dynamic lookup.
child.call("set_visible", false)
# Symbol-based dynamic lookup.
# GDScript aliases this into a 'call' method behind the scenes.
child.set_visible(false)
# Dynamic lookup, checks for method existence first.
if child.has_method("set_visible"):
child.set_visible(false)
# Cast check, followed by dynamic lookup.
# Useful when you make multiple "safe" calls knowing that the class
# implements them all. No need for repeated checks.
# Tricky if one executes a cast check for a user-defined type as it
# forces more dependencies.
if child is CanvasItem:
child.set_visible(false)
child.show_on_top = true
# If one does not wish to fail these checks without notifying users,
# one can use an assert instead. These will trigger runtime errors
# immediately if not true.
assert(child.has_method("set_visible"))
assert(child.is_in_group("offer"))
assert(child is CanvasItem)
# Can also use object labels to imply an interface, i.e. assume it
# implements certain methods.
# There are two types, both of which only exist for Nodes: Names and
# Groups.
# Assuming...
# A "Quest" object exists and 1) that it can "complete" or "fail" and
# that it will have text available before and after each state...
# 1. Use a name.
var quest = $Quest
print(quest.text)
quest.complete() # or quest.fail()
print(quest.text) # implied new text content
# 2. Use a group.
for a_child in get_children():
if a_child.is_in_group("quest"):
print(quest.text)
quest.complete() # or quest.fail()
print(quest.text) # implied new text content
# Note that these interfaces are project-specific conventions the team
# defines (which means documentation! But maybe worth it?).
# Any script that conforms to the documented "interface" of the name or
# group can fill in for it.
2023-01-12 18:31:02 +01:00
```
2023-01-12 19:30:47 +01:00
- Outsource the access to a `FuncRef`. These may be useful
in cases where one needs the max level of freedom from dependencies. In
this case, one relies on an external context to setup the method.
2023-01-12 18:31:02 +01:00
gdscript GDScript
2023-01-12 18:31:02 +01:00
```
# child.gd
extends Node
var fn = null
func my_method():
if fn:
fn.call_func()
# parent.gd
extends Node
onready var child = $Child
func _ready():
child.fn = funcref(self, "print_me")
child.my_method()
func print_me():
print(name)
2023-01-12 18:31:02 +01:00
```
These strategies contribute to Godot's flexible design. Between them, users
have a breadth of tools to meet their specific needs.