GDevelop editor and games engines are designed so that all objects, behaviors, effects, actions, conditions and expressions are provided by extensions. These extensions are composed of two parts:
- the declaration of the extension, traditionally done in a file called
JsExtension.js
. - the implementation of the extension for the game engine (also called the "Runtime"), containing the functions corresponding to the actions/conditions/expressions and the classes used for the objects or behaviors. The implementation is traditionally in files called
extensionnametools.js
,objectnameruntimeobject.js
orobjectnameruntimebehavior.js
.
Note that some GDevelop extensions are declared in C++, in files called
JsExtension.cpp
. If you want to edit them, refer to the paragraph about them at the end.
To modify extensions, you need to have the development version of GDevelop running. Make sure to have Git and Node.js installed. Yarn is optional.
git clone https://github.com/4ian/GD.git
cd GD/newIDE/app
npm install #or yarn
Refer to the GDevelop IDE Readme for more information about the installation.
-
First, run GDevelop with Electron.
When GDevelop is started, the developer console should be opened. Search for the message
Loaded x JS extensions.
that indicates the loading of extensions. -
You can now open an extension contained in the folder Extensions at the root of the repository. For example, you can open Extensions/FacebookInstantGames. Edit the JsExtension.js file or a runtime file. Any changes will be automatically imported in the editor.
Verify that changes are imported in the console: you should see a message starting by
GDJS Runtime update
. If you deactivated the automatic import in the preferences or want to import manually your changes, runimport-GDJS-Runtime.js
script:cd scripts node import-GDJS-Runtime.js # This copy extensions declaration and runtime into GDevelop.
-
Finally, verify that the changes are applied:
- If you modified the declaration (
JsExtension.js
), reload GDevelop by pressing Ctrl+R (or Cmd+R on macOS) in the developer console. - If you modified a Runtime file, relaunch a preview. Open the developer console if you want to check for any errors.
- If you modified the declaration (
-
You can now iterate and relaunch the script to develop the extension! 🚀
⚠️ Always check the developer console after reloading GDevelop. If there is any error signaled, click on it to see what went wrong. You may have done a syntax error or mis-used an API.
ℹ️ Implement your extension in file called
extensionnametools.js
(for general functions),objectnameruntimeobject.js
(for objects) orbehaviornameruntimebehavior.js
(for behaviors). See then the next section for declaring these files and the content of the extension to the IDE.
Check the GDJS game engine documentation here. It's also a good idea to check the Runtime folder of GDJS to see directly how the game engine is done when needed. Files for the game engine should mostly be written in JavaScript "ES5 flavor" (i.e: usual, classic good old JavaScript) (click to learn more).
See examples in examplejsextensiontools.js.
Read about gdjs.RuntimeScene
, the class representing a scene being played, as lots of events can need it.
See examples in dummyruntimebehavior.js (or dummywithshareddataruntimebehavior.js for an example with shared data between behaviors).
You'll be interested in the constructor (to initialize things), onDeActivate
(called when behavior is deactivated), doStepPreEvents
and doStepPostEvents
(to run logic before/after the events at each frame).
Read about gdjs.RuntimeBehavior
, the base class inherited by all behaviors, to see everything that is available.
See example in dummyruntimeobject.js (the object itself) and dummyruntimeobject-pixi-renderer.js (the renderer, using PIXI.js).
You'll be interested in the constructor (to initialize things), update
(called every frame) and the other methods. In the PIXI renderer, check the constructor (where PIXI objects are created). Other methods depend on the renderer.
Read about gdjs.RuntimeObject
, the base class inherited by all objects.
See lots of examples in Effects (the extension containing lots of effects) and light-night-pixi-filter.js (an example of a custom filter for PixiJS).
You'll have to store the code for your PixiJS filter in the file, and then call gdjs.PixiFiltersTools.registerFilterCreator
to tell the game engine how to create and update the filter. Don't forget to then declare the effect (see next section).
ℹ️ Declaration must be done in a file called
JsExtension.js
. Your extension must be in Extensions folder, in its own directory.
The API to declare extensions is almost 100% equivalent to the way extensions are declared in C++, so most links will redirect to this documentation.
Use extension.setExtensionInformation
to declare basic information about your extension.
👉 See an example in the example extension JsExtension.js file.
Use addAction
, addCondition
, addExpression
or addStrExpression
to declare actions, conditions or expressions.
- Chain calls to
addParameter
to declare the parameters of your action/condition/expression. - Call
getCodeExtraInformation()
and then functions likesetFunctionName
andsetIncludeFile
to declare the JavaScript function to be called and the file to be included.
You can call these functions on the
extension
object, or on the objects returned byextension.addObject
(for objects) orextension.addBehavior
(for behaviors). See below.
⚠️ Always double check that you've not forgotten an argument. Such errors/mismatchs can create silent bugs that could make GDevelop instable or crash while being used.
👉 See an example in the example extension JsExtension.js file.
Add a behavior using addBehavior
. The last two parameters are the gd.Behavior
and the gd.BehaviorsSharedData
object representing the behavior and its (optional) shared data
- For the behavior, create a
new gd.BehaviorJsImplementation()
and defineinitializeContent
,updateProperty
andgetProperties
. - For the shared data (which are properties shared between all behaviors of the same type), if you don't have the need for it, just pass
new gd.BehaviorsSharedData()
. If you need shared data, create anew gd.BehaviorSharedDataJsImplementation()
and defineinitializeContent
,updateProperty
andgetProperties
.
⚠️ Like other functions to declare extensions, make sure that you've not forgotten to declare a function and that all arguments are correct.
👉 See an example in the example extension JsExtension.js file. Learn more about properties here.
Add an object using addObject
. The last parameter is the gd.Object
representing the object:
- Create a
new gd.ObjectJsImplementation()
and defineupdateProperty
andgetProperties
(for the object properties) andupdateInitialInstanceProperty
andgetInitialInstanceProperties
(for the optional properties that are attached to each instance).
👉 See an example in the example extension JsExtension.js file. Learn more about properties here.
ℹ️ After doing this, you can actually see your object in GDevelop! Read the next sections to see how to add an editor and a renderer for instances on the scene editor.
To add an editor to your object, implement the function registerEditorConfigurations
in your extension module. For now, only a default editor, displaying the object properties, is supported:
registerEditorConfigurations: function(objectsEditorService) {
objectsEditorService.registerEditorConfiguration(
"MyDummyExtension::DummyObject", // Replace by your extension and object type names.
objectsEditorService.getDefaultObjectJsImplementationPropertiesEditor({
helpPagePath : "/extensions/extend-gdevelop" // The link to the help page for your object, in GDevelop wiki.
})
);
}
👉 See an example in the example extension JsExtension.js file.
Finally, to have the instances of your object displayed properly on the scene editor, implement the function registerInstanceRenderers
in your extension module. The function is passed an object called objectsRenderingService
, containing RenderedInstance, the "base class" for instance renderers, and PIXI, which give you access to Pixi.js rendering engine, used in the editor to render the scene.
👉 See an example in the example extension JsExtension.js file.
Add an effect using addEffect
in your JsExtension.js file.
👉 See an example in the Effects extension JsExtension.js file. Learn more about properties here.
ℹ️ Don't forget to use
addIncludeFile
to set the file containing your effect implementation in JavaScript.
👋 Declaring events is not yet exposed to JavaScript extensions. Your help is welcome to expose this feature!
If you want to start a new extension:
- Choose a unique and descriptive name. Create a folder with this name in Extensions.
- Create a file in it named JsExtension.js and copy the content of the JsExtension.js of another extension.
- Change the extension information (
extension.setExtensionInformation
). The first argument is the extension internal name and should be the same name as your folder for consistency. - Remove all the actions/conditions/expressions declarations and tests, run
node import-GDJS-Runtime.js
and reload GDevelop to verify that your extension is loaded. - Create a file called for example yourextensionnametools.js in the same directory.
- Add back the declarations in your extension. Use
setIncludeFile
when declaring your actions/conditions/expressions and set the name of the js file that you've created, prefixed by the path from the root folder. For example:.setIncludeFile("Extensions/FacebookInstantGames/facebookinstantgamestools.js")
If you have ideas or are creating a new extension, your contribution is welcome!
-
To submit your extension, you have first to create a Fork on GitHub (use the Fork button on the top right), then create a Pull Request.
-
Check the the roadmap for ideas and features planned.
-
A few enhancements are also possible to exploit the full potential of extensions:
- Add support for events
- Document how to add custom icons
- Add a button to reload extensions without reloading GDevelop IDE entirely.
Some extensions are still declared in C++ for being compatible with GDevelop 4. Check the sources in Extensions folder and install GDevelop.js. You'll then be able to make changes in C++ source files and have this reflected in the editor.