Skip to content

Classes

To expose a class written in Kotlin it needs to extend godot.Object (or any of its subtype) and must be annotated with @RegisterClass.

1
2
3
4
@RegisterClass
class RotatingCube: Node3D() {
    // ...
}

Registration files

For each class you register, a corresponding registration file is generated (a gdj file). These are the files you attach to nodes.

Location

By default, these files are generated into a folder called gdj in the root of your project.
You can however configure this inside your build.gradle.kts:

1
2
3
godot {
    registrationFileBaseDir.set(<folder>)
}

Naming

Classes need to be registered with a unique name as Godot does not support namespaces (or packages in this case) for script classes.

By default, we register your classes with the name you give them. While a simple approach and enough in most cases, this can lead to naming conflicts if you have classes in different packages with the same name. For example:

  • com.package.a.MyClass
  • com.package.b.MyClass

This leads to a conflict on the Godot side as both classes are registered as MyClass.

So you are responsible for making sure that classes have a unique name. We do however provide you with some assistance:

  • We have compile time checks in place which should let the build fail if classes would end up having the same name
  • The @RegisterClass annotation lets you define a custom registration name: @RegisterClass("CustomRegistrationName")
  • Register the class names with the fully qualified name: com.mygame.MyClass will be registered as: com_mygame_MyClass. This can be configured with:
    1
    2
    3
    godot {
        isFqNameRegistrationEnabled.set(true)
    }
    

Class names from other languages

Even with all these checks and helpers in place, we cannot check the names of classes from other languages like GDScript or C#. It's your responsibility to make sure there are no naming conflicts.

Hierarchy

Again: Godot does not have the concept of namespaces. So all classes are registered top level. It does not matter where in the folder hierarchy a script resides in, it still is accessed the same way. Hence, it does not matter if the registration files are all in one directory, or scattered across multiple directories.
By default, the registration files are all generated in a folder hierarchy which resembles your package hierarchy:

  • com.mygame.packageA.ClassA
  • com.mygame.packageB.ClassB
1
2
3
4
5
6
7
[registrationFileBaseDir]/
└── com/
    └── mygame/
        ├── packageA/
        │   └── ClassA.gdj
        └── packageB/
            └── ClassB.gdj

Some do not like this hierarchical structure and especially for small games with not many scripts, this could be undesirable to work with. Hence, we let you turn of the hierarchical generation in your build.gradle.kts:

1
2
3
godot {
    isRegistrationFileHierarchyEnabled.set(false)
}

Which would result in a folder structure like this:

  • com.mygame.packageA.ClassA
  • com.mygame.packageB.ClassB
1
2
3
[registrationFileBaseDir]/
├── ClassA.gdj
└── ClassB.gdj

This could also be useful together with the option isFqNameRegistrationEnabled from the Naming section which would result in:

  • com.mygame.packageA.ClassA
  • com.mygame.packageB.ClassB
1
2
3
[registrationFileBaseDir]/
├── com_mygame_packageA_ClassA.gdj
└── com_mygame_packageA_ClassB.gdj

Lifecycle

If you want to be notified when initialization and destruction of your class happens, use the init block and override the _onDestroy function respectively.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
@RegisterClass
class RotatingCube: Node3D() {
    init {
        println("Initializing RotatingCube!")
    }

    override fun _onDestroy() {
        println("Cleaning up RotatingCube!")
    }
}

Instance checks

Checking if an object is an instance of a particular type can be done via the is operator.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
@RegisterFunction
override fun _ready() {
    val parent = getParent()
    if (parent is CollisionShape) {
        // smart cast works!
        parent.setShape(...)
    } else {
        throw AssertionError("Unexpected parent!")
    }
}

This also works for any type you define.

Info

If you are sure that an object is always an instance of some type, then you can take advantage of Kotlin's contracts feature. This allows you to avoid having nested ifs.

1
2
3
4
5
6
7
@RegisterFunction
override fun _ready() {
    val parent = getParent()
    require(parent is CollisionShape)
    // smart cast works here as well!
    parent.setShape(...)
}

Constructors

Godot requires you to have a default constructor on your classes. You can define additional constructors but you have to register them by annothing them with @RegisterConstructor. Default constructors on the other hand are always registered.

Constructors also can have a maximum of 8 arguments and must have a unique argument count as constructor overloading is not yet supported. This limitation is only for registered constructors.

Instantiate Kotlin script classes in GDScript

Creating an instance using the default constructor can be done by:

1
var instance := YourKotlinClass.new()

While for additional constructors, load must be used:

1
var instance := load("res://gdj/YourClass.gdj").new(oneArg, anotherArg)

Info

The limitation of max 8 arguments for constructors is arbitrary. We decided to introduce this limitation to prevent performance bottlenecks for creating objects as each argument passed to a constructor needs to be unpacked by the binding. The more arguments, the more unpacking is needed which means more overhead.

Customization

You can customize to some extent how your class should be registered in Godot:

The @RegisterClass annotation can take two arguments:

  • className: If set, the class will be registered with the name you provide

Unique class names

If you specify the className in the annotation, you have to make sure that this name is unique! We implemented compilation checks to make sure the compilation fails if more than two classes are registered with the same name, but we cannot check class names from other scripting languages like GDScript or C#! It is also recommended installing our intellij plugin as it shows duplicated registered class names in the editor as an error.

Tool mode

Annotate your class with @Tool to make it a tool class (note that @RegisterClass is required for this annotation to take effect).