IntelliJ Action System ========== This tutorial is meant to give general information about the IntelliJ IDEA Action System and lead you through a series of steps which show how to create, register, and customize custom actions and action groups. Action system provides an option to handle certain events in a desired way. Action can either be simply a response to some state, or be bound to UI element and could be invoked on demand. These UI elements include main menu, context menus and toolbars. ---------------- [Source code](https://github.com/JetBrains/intellij-sdk/tree/master/code_samples/register_actions) ---------------- #Working with custom actions An action is technically a class, derived from the [AnAction] (https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) class. To update the state of the action, the method AnAction.update() is periodically called by IDEA. The object of type [AnActionEvent] (https://github.com/JetBrains/intellij-community/blob/ff16ce78a1e0ddb6e67fd1dbc6e6a597e20d483a/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnActionEvent.java) passed to this method carries the information about the current context for the action, and in particular, the specific presentation which needs to be updated. ------------- ##Creating actions To create a new we need to extend [AnAction] (https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) class: ```java public class SimpleAction extends AnAction { } ``` The only method of an inheritor of [AnAction] (https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) which needs to be overridden is ```public void actionPerformed(AnActionEvent anActionEvent);``` , and it should contain a part of code to be executed after the action has been invoked. In this case the action does nothing. ```java public class SimpleAction extends AnAction { @Override public void actionPerformed(AnActionEvent anActionEvent) { } } ``` ------------- ##Registering actions To register a newly created action, attribute should be added to the section of the plugin configuration file [plugin.xml] (https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml). IntelliJ IDEA has an embedded inspection that spots unregistered actions. !["Action never used" inspection](img/action_never_used.png) To register the action and set up it's attributes press ***Alt + Enter*** while the caret is placed on the action's declaration. !["Register action" quick fix](img/action_never_used.png) Fill the "New Action" form to set up action's parameters such as: action's name and description, a UI component the action is bound to, visual position of the menu item the action is bound to, and a shortcut for invoking the action. In our case the action will be available in the Tools Menu, it will be placed on top, and will have no shortcuts. !["Register action" quick fix](img/new_action.png) After filling the "New Action" form and applying the changes ** section of our [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml) file will look like this: ```xml ``` Full list of action's attributes can also be set manually in [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml) configuration file like the following code sample shows: ```xml ``` After performing the steps described above we need to compile and run the plugin to the the newly created action available as a Tools Menu item: !["Register action" quick fix](img/tools_menu_item_action.png) ----------- ##Performing an action In order to make the action do something we need to implement it's ```public void actionPerformed(AnActionEvent anActionEvent);``` method. In the following example action invokes a dialog that shows information about a selected Project View Item and has no icon and any pre-selected default option: ```java @Override public void actionPerformed(AnActionEvent anActionEvent) { Object navigatable = anActionEvent.getData(CommonDataKeys.NAVIGATABLE); if (navigatable != null) { Messages.showDialog(navigatable.toString(), "Selected Element:", new String[]{"OK"}, -1, null); } } ``` ----------- ##Setting up action's visibility and availability To manipulate with action's visibility and availability we need to override it's ```public void update(@NotNull AnActionEvent e);``` Default implementation of this method does nothing. Override this method to provide the ability to dynamically change action's state and(or) presentation depending on the context. ```java public class SimpleAction extends AnAction { @Override public void actionPerformed(AnActionEvent anActionEvent) { //... } @Override public void update(AnActionEvent anActionEvent) { } } ``` The following code sample illustrates how to make the action visible and available only when the following conditions are met: there's a project available and there's an item you can navigate to selected in the project view pane: ```java public class SimpleAction extends AnAction { @Override public void actionPerformed(AnActionEvent anActionEvent) { //... } @Override public void update(AnActionEvent anActionEvent) { final Project project = anActionEvent.getData(CommonDataKeys.PROJECT); if (project == null) return; Object navigatable = anActionEvent.getData(CommonDataKeys.NAVIGATABLE); anActionEvent.getPresentation().setEnabledAndVisible(navigatable != null); } } ``` Parameter anActionEvent carries information on the invocation place and data available. **Note** This method can be called frequently, for instance, if an action is added to a toolbar, it will be updated twice a second. This means that this method is supposed to work really fast, no real work should be done at this phase. For example, checking selection in a tree or a list, is considered valid, but working with a file system is not. If you cannot understand the state of the action fast you should do it in the [AnActionEvent] (https://github.com/JetBrains/intellij-community/blob/ff16ce78a1e0ddb6e67fd1dbc6e6a597e20d483a/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnActionEvent.java) method and notify the user that action cannot be executed if it's the case. ------------- After compiling and running the plugin project and invoking the action, the dialog will pop up: !["Register action" quick fix](img/action_performed.png) ------------- #Groupping actions If some part of the functionality requires to implement several actions or actions are simply too many and overload the menu they can be joined into groups. In this case the group will be available as a top-level menu item, action will be represented as drop-down menu items. ##Creating simple action groups Grouping can be done by extending adding ** attribute to ** [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml) file. ```xml ``` ##Binding action groups to UI component The following sample shows how to place a custom action group on top of the editor popup menu: ```xml ``` ##Adding actions to the group To create an action we need to extend [AnAction.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) class: ```java public class GroupedAction extends AnAction { @Override public void update(AnActionEvent event) { event.getPresentation().setEnabledAndVisible(true); } @Override public void actionPerformed(AnActionEvent event) { //Does nothing } } ``` And then the actions needs to be registered in the newly created group: ```xml ``` After performing the steps described above the action group nad it's content will be available in the editor popup menu: ![Simple Action Group](img/grouped_action.png) ##Working with DefaultActionGroup In some cases we need to implement some specific behaviour of a group of actions dependently on the context. The steps below are meant to show how to make a group of actions available and visible if a certain condition is met and how to set up a group icon dynamically. In our case the condition is: an instance of the editor is available. ###Extending DefaultActionGroup [DefaultActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/platform-api/src/com/intellij/openapi/actionSystem/DefaultActionGroup.java) is a default implementations of [ActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/ActionGroup.java) and used to add children actions and separators between them to a group. This class is used if a set of actions belonging to the group is fixed, which is the majority of all the cases. Firstly, [DefaultActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/platform-api/src/com/intellij/openapi/actionSystem/DefaultActionGroup.java) should be derived: ```java public class CustomDefaultActionGroup extends DefaultActionGroup { @Override public void update(AnActionEvent event) { } } ``` ###Registering action group As in case with the simple action group, the inheritor of [DefaultActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/platform-api/src/com/intellij/openapi/actionSystem/DefaultActionGroup.java) should be declared in [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml) file: ```xml ``` ###Creating an action [AnAction.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) needs to be extended: ```java public class CustomGroupedAction extends AnAction { @Override public void actionPerformed(AnActionEvent anActionEvent) { //Does nothing } } ``` ###Adding actions to the group Action's class should be registered in [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml) : ```xml ``` ###Providing specific behaviour for the group In this case we override ```public void update(AnActionEvent event);``` method to make the group visible as a *Tools* menu item, however, it will be enabled only if there's an instance of the editor available. Also a custom icon is set up: ```java public class CustomDefaultActionGroup extends DefaultActionGroup { @Override public void update(AnActionEvent event) { Editor editor = event.getData(CommonDataKeys.EDITOR); event.getPresentation().setVisible(true); event.getPresentation().setEnabled(editor != null); event.getPresentation().setIcon(AllIcons.General.Error); } } ``` After compiling and running the code sample above, *Tools* menu item should contain an extra group of action with a user-defined icon: ![Default Action Group](img/default_action_group.png) ##Action groups with variable actions set If a set of actions belonging to a custom actions group may vary dependently on the context, we need to work with [ActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/ActionGroup.java). In this case set of actions to be grouped can be dynamically defined. ###Createing variable action group To create a group of actions with a variable actions set we extend [ActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/ActionGroup.java) first: ```java public class BaseActionGroup extends ActionGroup { } ``` ###Registering variable action group To register the group ** attribute needs to be placed in the ** section of [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml): ```xml ``` **Note**: Since the set of actions is defined dynamically no action definitions should be placed in [plugin.xml](https://github.com/JetBrains/intellij-sdk/blob/master/code_samples/register_actions/META-INF/plugin.xml). If ** attribute contains any static action definition an exception will be thrown. For statically defined group of action use [DefaultActionGroup.java](https://github.com/JetBrains/intellij-community/blob/master/platform/platform-api/src/com/intellij/openapi/actionSystem/DefaultActionGroup.java) ###Accessing children actions An array of children actions should be returned by the method ```public AnAction[] getChildren(AnActionEvent anActionEvent);``` of the a created group: ```java public class BaseActionGroup extends ActionGroup { @NotNull @Override public AnAction[] getChildren(AnActionEvent anActionEvent) { return new AnAction[0]; } } ``` ###Adding children actions to the group To make the group contain actions a non-empty array of [AnAction.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) elements should be returned: ```java public class BaseActionGroup extends ActionGroup { @NotNull @Override public AnAction[] getChildren(AnActionEvent anActionEvent) { return new AnAction[]{new MyAction()}; } class MyAction extends AnAction { public MyAction() { super("Dynamically Added Action"); } @Override public void actionPerformed(@NotNull AnActionEvent anActionEvent) { } } } ``` After providing an implementation of [AnAction.java](https://github.com/JetBrains/intellij-community/blob/master/platform/editor-ui-api/src/com/intellij/openapi/actionSystem/AnAction.java) and making it return a non-empty array of action Tools Menu should contain an extra group of action: ![Dynamic Action Group](img/dynamic_action_group.png) ---------------- [Source code](https://github.com/JetBrains/intellij-sdk/tree/master/code_samples/register_actions) ----------------