| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518 | .. _doc_making_plugins:Making plugins==============About plugins-------------A plugin is a great way to extend the editor with useful tools. It can be madeentirely with GDScript and standard scenes, without even reloading the editor.Unlike modules, you don't need to create C++ code nor recompile the engine.While this makes plugins less powerful, there are still many things you cando with them. Note that a plugin is similar to any scene you can alreadymake, except it is created using a script to add editor functionality.This tutorial will guide you through the creation of two plugins soyou can understand how they work and be able to develop your own. The firstis a custom node that you can add to any scene in the project, and theother is a custom dock added to the editor.Creating a plugin-----------------Before starting, create a new empty project wherever you want. This will serveas a base to develop and test the plugins.The first thing you need for the editor to identify a new plugin is tocreate two files: a ``plugin.cfg`` for configuration and a tool script with thefunctionality. Plugins have a standard path like ``addons/plugin_name`` insidethe project folder. Godot provides a dialog for generating those files andplacing them where they need to be.In the main toolbar, click the ``Project`` dropdown. Then click``Project Settings...``. Go to the ``Plugins`` tab and then clickon the :button:`Create New Plugin` button in the top-right.You will see the dialog appear, like so:.. image:: img/making_plugins-create_plugin_dialog.webpThe placeholder text in each field describes how it affects the plugin'screation of the files and the config file's values.To continue with the example, use the following values:.. tabs:: .. code-tab:: ini GDScript    Plugin Name: My Custom Node    Subfolder: my_custom_node    Description: A custom node made to extend the Godot Engine.    Author: Your Name Here    Version: 1.0.0    Language: GDScript    Script Name: custom_node.gd    Activate now: No .. code-tab:: ini C#    Plugin Name: My Custom Node    Subfolder: MyCustomNode    Description: A custom node made to extend the Godot Engine.    Author: Your Name Here    Version: 1.0.0    Language: C#    Script Name: CustomNode.cs    Activate now: No.. warning::    Unchecking the ``Activate now?`` option in C# is always required because,    like every other C# script, the EditorPlugin script needs to be compiled which    requires building the project. After building the project the plugin can be    enabled in the ``Plugins`` tab of ``Project Settings``.You should end up with a directory structure like this:.. image:: img/making_plugins-my_custom_mode_folder.webp``plugin.cfg`` is an INI file with metadata about your plugin.The name and description help people understand what it does.Your name helps you get properly credited for your work.The version number helps others know if they have an outdated version;if you are unsure on how to come up with the version number, check out `Semantic Versioning <https://semver.org/>`_.The main script file will instruct Godot what your plugin does in the editoronce it is active.The script file~~~~~~~~~~~~~~~Upon creation of the plugin, the dialog will automatically open theEditorPlugin script for you. The script has two requirements that you cannotchange: it must be a ``@tool`` script, or else it will not load properly in theeditor, and it must inherit from :ref:`class_EditorPlugin`... warning::    In addition to the EditorPlugin script, any other GDScript that your plugin uses    must *also* be a tool. Any GDScript without ``@tool`` used by the editor    will act like an empty file!It's important to deal with initialization and clean-up of resources.A good practice is to use the virtual function:ref:`_enter_tree() <class_Node_private_method__enter_tree>` to initialize your plugin and:ref:`_exit_tree() <class_Node_private_method__exit_tree>` to clean it up. Thankfully,the dialog generates these callbacks for you. Your script should look somethinglike this:.. _doc_making_plugins_template_code:.. tabs:: .. code-tab:: gdscript GDScript    @tool    extends EditorPlugin    func _enter_tree():        # Initialization of the plugin goes here.        pass    func _exit_tree():        # Clean-up of the plugin goes here.        pass .. code-tab:: csharp    #if TOOLS    using Godot;    [Tool]    public partial class CustomNode : EditorPlugin    {        public override void _EnterTree()        {            // Initialization of the plugin goes here.        }        public override void _ExitTree()        {            // Clean-up of the plugin goes here.        }    }    #endifThis is a good template to use when creating new plugins.A custom node-------------Sometimes you want a certain behavior in many nodes, such as a custom sceneor control that can be reused. Instancing is helpful in a lot of cases, butsometimes it can be cumbersome, especially if you're using it in manyprojects. A good solution to this is to make a plugin that adds a node with acustom behavior... warning::  Nodes added via an EditorPlugin are "CustomType" nodes. While they work  with any scripting language, they have fewer features than  :ref:`the Script Class system <doc_gdscript_basics_class_name>`. If you  are writing GDScript or NativeScript, we recommend using Script Classes instead.To create a new node type, you can use the function:ref:`add_custom_type() <class_EditorPlugin_method_add_custom_type>` from the:ref:`class_EditorPlugin` class. This function can add new types to the editor(nodes or resources). However, before you can create the type, you need a scriptthat will act as the logic for the type. While that script doesn't have to usethe ``@tool`` annotation, it can be added so the script runs in the editor.For this tutorial, we'll create a button that prints a message whenclicked. For that, we'll need a script that extends from:ref:`class_Button`. It could also extend:ref:`class_BaseButton` if you prefer:.. tabs:: .. code-tab:: gdscript GDScript    @tool    extends Button    func _enter_tree():        pressed.connect(clicked)    func clicked():        print("You clicked me!") .. code-tab:: csharp    using Godot;    [Tool]    public partial class MyButton : Button    {        public override void _EnterTree()        {            Pressed += Clicked;        }        public void Clicked()        {            GD.Print("You clicked me!");        }    }That's it for our basic button. You can save this as ``my_button.gd`` inside theplugin folder. You'll also need a 16×16 icon to show in the scene tree. If youdon't have one, you can grab the default one from the engine and save it in your`addons/my_custom_node` folder as `icon.png`, or use the default Godot logo(`preload("res://icon.svg")`)... tip::    SVG images that are used as custom node icons should have the    **Editor > Scale With Editor Scale** and **Editor > Convert Colors With Editor Theme**    :ref:`import options <doc_importing_images_editor_import_options>` enabled. This allows    icons to follow the editor's scale and theming settings if the icons are designed with    the same color palette as Godot's own icons... image:: img/making_plugins-custom_node_icon.pngNow, we need to add it as a custom type so it shows on the **Create New Node**dialog. For that, change the ``custom_node.gd`` script to the following:.. tabs:: .. code-tab:: gdscript GDScript    @tool    extends EditorPlugin    func _enter_tree():        # Initialization of the plugin goes here.        # Add the new type with a name, a parent type, a script and an icon.        add_custom_type("MyButton", "Button", preload("my_button.gd"), preload("icon.png"))    func _exit_tree():        # Clean-up of the plugin goes here.        # Always remember to remove it from the engine when deactivated.        remove_custom_type("MyButton") .. code-tab:: csharp    #if TOOLS    using Godot;    [Tool]    public partial class CustomNode : EditorPlugin    {        public override void _EnterTree()        {            // Initialization of the plugin goes here.            // Add the new type with a name, a parent type, a script and an icon.            var script = GD.Load<Script>("res://addons/MyCustomNode/MyButton.cs");            var texture = GD.Load<Texture2D>("res://addons/MyCustomNode/Icon.png");            AddCustomType("MyButton", "Button", script, texture);        }        public override void _ExitTree()        {            // Clean-up of the plugin goes here.            // Always remember to remove it from the engine when deactivated.            RemoveCustomType("MyButton");        }    }    #endifWith that done, the plugin should already be available in the plugin list in the**Project Settings**, so activate it as explained in `Checking the results`_.Then try it out by adding your new node:.. image:: img/making_plugins-custom_node_create.webpWhen you add the node, you can see that it already has the script you createdattached to it. Set a text to the button, save and run the scene. When youclick the button, you can see some text in the console:.. image:: img/making_plugins-custom_node_console.webpA custom dock-------------Sometimes, you need to extend the editor and add tools that are always available.An easy way to do it is to add a new dock with a plugin. Docks are just scenesbased on Control, so they are created in a way similar to usual GUI scenes.Creating a custom dock is done just like a custom node. Create a new``plugin.cfg`` file in the ``addons/my_custom_dock`` folder, thenadd the following content to it:.. tabs:: .. code-tab:: gdscript GDScript    [plugin]    name="My Custom Dock"    description="A custom dock made so I can learn how to make plugins."    author="Your Name Here"    version="1.0"    script="custom_dock.gd" .. code-tab:: csharp    [plugin]    name="My Custom Dock"    description="A custom dock made so I can learn how to make plugins."    author="Your Name Here"    version="1.0"    script="CustomDock.cs"Then create the script ``custom_dock.gd`` in the same folder. Fill it with the:ref:`template we've seen before <doc_making_plugins_template_code>` to get agood start.Since we're trying to add a new custom dock, we need to create the contents ofthe dock. This is nothing more than a standard Godot scene: just createa new scene in the editor then edit it.For an editor dock, the root node **must** be a :ref:`Control <class_Control>`or one of its child classes. For this tutorial, you can create a single button.The name of the root node will also be the name that appears on the dock tab,so be sure to give it a short and descriptive name.Also, don't forget to add some text to your button... image:: img/making_plugins-my_custom_dock_scene.webpSave this scene as ``my_dock.tscn``. Now, we need to grab the scene we createdthen add it as a dock in the editor. For this, you can rely on the function:ref:`add_control_to_dock() <class_EditorPlugin_method_add_control_to_dock>` from the:ref:`EditorPlugin <class_EditorPlugin>` class.You need to select a dock position and define the control to add(which is the scene you just created). Don't forget to**remove the dock** when the plugin is deactivated.The script could look like this:.. tabs:: .. code-tab:: gdscript GDScript    @tool    extends EditorPlugin    # A class member to hold the dock during the plugin life cycle.    var dock    func _enter_tree():        # Initialization of the plugin goes here.        # Load the dock scene and instantiate it.        dock = preload("res://addons/my_custom_dock/my_dock.tscn").instantiate()        # Add the loaded scene to the docks.        add_control_to_dock(DOCK_SLOT_LEFT_UL, dock)        # Note that LEFT_UL means the left of the editor, upper-left dock.    func _exit_tree():        # Clean-up of the plugin goes here.        # Remove the dock.        remove_control_from_docks(dock)        # Erase the control from the memory.        dock.free() .. code-tab:: csharp    #if TOOLS    using Godot;    [Tool]    public partial class CustomDock : EditorPlugin    {        private Control _dock;        public override void _EnterTree()        {            _dock = GD.Load<PackedScene>("res://addons/MyCustomDock/MyDock.tscn").Instantiate<Control>();            AddControlToDock(DockSlot.LeftUl, _dock);        }        public override void _ExitTree()        {            // Clean-up of the plugin goes here.            // Remove the dock.            RemoveControlFromDocks(_dock);            // Erase the control from the memory.            _dock.Free();        }    }    #endifNote that, while the dock will initially appear at its specified position,the user can freely change its position and save the resulting layout.Checking the results~~~~~~~~~~~~~~~~~~~~It's now time to check the results of your work. Open the **ProjectSettings** and click on the **Plugins** tab. Your plugin should be the only oneon the list... image:: img/making_plugins-project_settings.webpYou can see the plugin is not enabled.Click the **Enable** checkbox to activate the plugin.The dock should become visible before you even closethe settings window. You should now have a custom dock:.. image:: img/making_plugins-custom_dock.webp.. _doc_making_plugins_autoload:Registering autoloads/singletons in plugins-------------------------------------------It is possible for editor plugins to automatically register:ref:`autoloads <doc_singletons_autoload>` when the plugin is enabled.This also includes unregistering the autoload when the plugin is disabled.This makes setting up plugins faster for users, as they no longer have to manuallyadd autoloads to their project settings if your editor plugin requires the use ofan autoload.Use the following code to register a singleton from an editor plugin:.. tabs:: .. code-tab:: gdscript GDScript    @tool    extends EditorPlugin    # Replace this value with a PascalCase autoload name, as per the GDScript style guide.    const AUTOLOAD_NAME = "SomeAutoload"    func _enable_plugin():        # The autoload can be a scene or script file.        add_autoload_singleton(AUTOLOAD_NAME, "res://addons/my_addon/some_autoload.tscn")    func _disable_plugin():        remove_autoload_singleton(AUTOLOAD_NAME) .. code-tab:: csharp    #if TOOLS    using Godot;    [Tool]    public partial class MyEditorPlugin : EditorPlugin    {        // Replace this value with a PascalCase autoload name.        private const string AutoloadName = "SomeAutoload";        public override void _EnablePlugin()        {            // The autoload can be a scene or script file.            AddAutoloadSingleton(AutoloadName, "res://addons/MyAddon/SomeAutoload.tscn");        }        public override void _DisablePlugin()        {            RemoveAutoloadSingleton(AutoloadName);        }    }    #endifUsing sub-plugins-----------------Often a plugin adds multiple things, for example a custom node and a panel.In those cases it might be easier to have a separate plugin script for each of those features.Sub-plugins can be used for this.First create all plugins and sub plugins as normal plugins:.. image:: img/sub_plugin_creation.webpThen move the sub plugins into the main plugin folder:.. image:: img/sub_plugin_moved.webpGodot will hide sub-plugins from the plugin list, so that a user can't enable or disable them.Instead the main plugin script should enable and disable sub-plugins like this:.. tabs:: .. code-tab:: gdscript GDScript    @tool    extends EditorPlugin    # The main plugin is located at res://addons/my_plugin/    const PLUGIN_NAME = "my_plugin"    func _enable_plugin():        EditorInterface.set_plugin_enabled(PLUGIN_NAME + "/node", true)        EditorInterface.set_plugin_enabled(PLUGIN_NAME + "/panel", true)    func _disable_plugin():        EditorInterface.set_plugin_enabled(PLUGIN_NAME + "/node", false)        EditorInterface.set_plugin_enabled(PLUGIN_NAME + "/panel", false)Going beyond------------Now that you've learned how to make basic plugins, you can extend the editor inseveral ways. Lots of functionality can be added to the editor with GDScript;it is a powerful way to create specialized editors without having to delve intoC++ modules.You can make your own plugins to help yourself and share them in the`Asset Library <https://godotengine.org/asset-library/>`_ so that peoplecan benefit from your work.
 |