[Из песочницы] Создание плагина для Intellij IDEA. Заметки и мелкие советы

Полгода назад или около того я загорелся-таки идей написать свой плагин для Intellij IDEA. Согласно задумке, он должен был считать, сколько времени разработчик потратил на проект (всего, за день, за сеанс) и отображать результат на диаграмме. Никакой магии, но такая функция здорово помогла бы мне рассчитывать время работы.

550c431301474e55be9c302426770c55.png
Пролистав каталог существующих плагинов, я все же нашел один подобный. Но никаких диаграмм он не рисовал, открывался через два выпадающих списка — что жуть, как неудобно, — да и на вид оказался довольно скудным.

87c49e4668574b258e456a556e9013f8.PNG

Короче, я еще больше укрепился в вере, что мой плагин был бы полезен, и приступил к работе. Сразу хочу сказать, что материалов по разработке плагинов Intellij IDEA не просто мало, а их почти что нет.

Но кое-что все же раскопать удалось.

Полезные ссылки


Небольшой гайд от JetBrains
Форум разработчиков плагинов для Intellij Platform
Коротко и понятно о том, где взять исходный код IDE
Пост одного польского разработчика
Очень подробный цикл статей на Хабре от Lucyfer
Java API Examples

Мелкие советы


Я решил не рассказывать о таких вещах, как настройка среды для разработки плагина, конфигурационном файле и др., так как все немногие статьи в рунете посвящены именно этому. За время создания плагина я много раз спотыкался о неочевидные (мне, во всяком случае) особенности Intellij IDEA. Знай я о них раньше, дело пошло бы намного быстрее. Поэтому поделюсь некоторыми моментами, которые могут сэкономить вам порядочно времени.

Добавляем свой CustomStatusBarWidget


Для удобства я решил дополнить Status Bar небольшим элементом, который отображал бы сколько времени было потрачено за текущий сеанс работы в среде, не отвлекая при этом разработчика от редактора. Интерфейс StatusBar содержит перегруженный метод addWidget(), принимающий CustomStatusBarWidget параметр:

public interface StatusBar extends StatusBarInfo, Disposable {
...
    void addWidget(@NotNull StatusBarWidget var1);

    void addWidget(@NotNull StatusBarWidget var1, @NotNull String var2);

    void addWidget(@NotNull StatusBarWidget var1, @NotNull Disposable var2);

    void addWidget(@NotNull StatusBarWidget var1, @NotNull String var2, @NotNull Disposable var3);
...
 }


Получить же сам StatusBar проекта можно следующим образом:

statusBar = WindowManager.getInstance().getStatusBar(currentProject);


Главной проблемой для меня стала имплементация CustomStatusBarWidget, а также расположение его по отношению к другим виджетам (я не знал их названий, так что найти их в исходном коде среды не удалось).

Спасение пришло отсюда. То есть, виджет можно добавить относительно других следующим образом:

statusBar.addWidget(myWidget1,"before " + IdeMessagePanel.FATAL_ERROR);
statusBar.addWidget(myWidget2, "after Encoding");
statusBar.addWidget(myWidget3,"after InsertOverwrite");
statusBar.addWidget(myWidget4,"after Position");


Имплементируем CustomStatusBarWidget:

    class Widget implements CustomStatusBarWidget{
        
        private JLabel myLabel = new JLabel("00:00:00");

        @Override
        public JComponent getComponent() {
           return myLabel;
        }

        @NotNull
        @Override
        public String ID() {
            return null;
        }

        @Nullable
        @Override
        public WidgetPresentation getPresentation(@NotNull PlatformType platformType) {
            return null;
        }

        @Override
        public void install(@NotNull StatusBar statusBar) {

        }

        @Override
        public void dispose() {

        }
    }


Добавим наш виджет на StatusBar, дополним JLabel иконкой…

4cce96ac65144a98bf078fcba08bfecf.PNG

Получение ToolBar'a действий из Java кода


Подробно про саму систему действий в плагинах Intellij IDEA можно почитать тут.
Получение компонента при помощи ActionManager.

    private JComponent createActionToolBar(AnAction ...actions){
        DefaultActionGroup actionGroup = new DefaultActionGroup();
        for(AnAction anAction : actions){
            actionGroup.add(anAction);
        }
        ActionToolbar toolbar = ActionManager.getInstance().createActionToolbar("Tempore.MainPanel", actionGroup, false);
        return toolbar.getComponent();
    }


Адрес проекта, в котором работает плагин


Путь к проекту можно получить с помощью экземпляра класса Project. Например, путь к папке .idea проекта:

String path = currentProject.getProjectFile().getParent().getPath();


Получение компонентов из метода actionPerformed()


Из параметра AnActionEvent можно получить доступ к компонентам:

Project currentProject = DataKeys.PROJECT.getData(actionEvent.getDataContext());
VirtualFile currentFile = DataKeys.VIRTUAL_FILE.getData(actionEvent.getDataContext());
Editor editor = DataKeys.EDITOR.getData(actionEvent.getDataContext());
StatusBar statusBar = WindowManager.getInstance().getStatusBar(DataKeys.PROJECT.getData(actionEvent.getDataContext()));
                  .


Всплывающие сообщения


Простое сообщение с информацией или сообщение об ошибке можно добавить на StatusBar следующим образом:

                                    JBPopupFactory.getInstance()
                                            .createHtmlTextBalloonBuilder("You have been working for two hours! Recommend to have a break ", MessageType.INFO, null)
                                            .setFadeoutTime(7500)
                                            .createBalloon()
                                            .show(RelativePoint.getCenterOf(statusBar.getComponent()),
                                                    Balloon.Position.atRight);



Выглядит вот так:
48319cc254b44da280d14da1db9ee3ae.png

Совместимость со средами Intellij Platform


Так как подсчет времени работы над проектом полезен не только в Intellij IDEA, я решил сделать плагин совместимым с другими средами Intellij Platform. Для этого нужно добавить тег в файле plugin.xml:

<idea-plugin version="0.5.5b">
  ...
  <depends>com.intellij.modules.lang</depends>
  ...
</idea-plugin>


Теперь помимо Intellij IDEA плагин подходит еще и для RubyMine, WebStorm, PhpStorm, PyCharm и AppCode.

Итоги


Примерно за месяц, мне удалось создать плагин, который следит за тем, сколько времени вы тратите на свои проекты, советует сделать перерыв, если вы пишете код нон-стоп несколько часов и рисует диаграмму потраченных часов за каждый день.
Так же он останавливается, если активность в среде прекратилась больше, чем на 5 минут и запускается, когда пользователь снова принялся за работу.

f741aba82fe2436fa464b25b6e32492a.PNG

При желании, сам плагин можно скачать тут.

Надеюсь, кому-то эта статья поможет разобраться с некоторыми аспектами разработки и избавит от бессонных ночей за чтением API.

© Habrahabr.ru