Разработка плагина IntelliJ IDEA: сохранение групп вкладок, сохранение их настойчиво и перезагрузка набора вкладок по запросу пользователя

В настоящий момент я перехожу к написанию плагина IntelliJ. Я хочу иметь возможность сохранять/восстанавливать набор вкладок для переключения между различными сеансами вкладок (сравнимыми с плагинами браузера, такими как Session Manager или Session Buddy).

Поэтому мне нужно в основном три типа действий:

  • Прочитать открытые вкладки (какой файл и редактор используется?)
  • Сохранять эту информацию в качестве вкладок
  • Откройте вкладки выбранного сеанса и закройте все остальные.

Я просмотрел доступные действия: IdeActions.java - кажется, что я не ищу того, что ищу. Но, возможно, я смотрю не туда. Может ли кто-нибудь сказать мне, возможно ли то, что я пытаюсь достичь, и дать мне несколько указателей в правильном направлении?

Update

Я успешно создал плагин, и он доступен в Github: http://alp82.github.com/idea-tabsession/

Он доступен в официальном репозитории плагинов: вкладка сеанса.

Обновление 2

Ниже приведен следующий вопрос о разбитых окнах: Получение и настройка параметров разделенного окна

Ответ 1

Обновление 2017

Я прекращаю поддержку этого плагина, потому что IDEA уже поддерживает эту функцию. Вы можете легко сохранять и загружать контексты, как показано здесь: https://github.com/alp82/idea-tabsession#discontinued

Update

Плагин готов и готов и может быть загружен в IDEA → Настройки → Плагины. Исходный код доступен по адресу: https://github.com/alp82/idea-tabsession

Короткий ответ

Чтобы прочитать, какие вкладки открыты прямо сейчас, используйте кнопки EditorFactory и FileDocumentManager Singletons:

    Editor[] editors = EditorFactory.getInstance().getAllEditors();
    FileDocumentManager fileDocManager = FileDocumentManager.getInstance();
    for(Editor editor : editors) {
        VirtualFile vf = fileDocManager.getFile(editor.getDocument());
        String path = vf.getCanonicalPath();
        System.out.println("path = " + path);
    }

Чтобы открыть вкладки, используйте FileEditorManager singleton (files, являющийся String Array of canonical paths):

    FileEditorManager fileEditorManager = FileEditorManager.getInstance(project);
    for(String path : files) {
        System.out.println("path = " + path);
        VirtualFile vf = LocalFileSystem.getInstance().findFileByPath(path);
        fileEditorManager.openFile(vf, true, true);
    }

Длинный ответ

Необходимые условия

Структура плагинов

После того, как вы создали свой плагин, вам нужно отредактировать файл plugin.xml, расположенный в папке META-INF. Измените id, name и description.

Нам нужен файл конфигурации для постоянного хранения. Создайте файл mystorage.xml в папке src. Теперь пришло время создать необходимые файлы:

SessionComponent.java(создайте его с помощью мастера Add Project Component для автоматического создания необходимых настроек xml):

@State(
    name = "SessionComponent",
    storages = {
        @Storage(id = "default", file = StoragePathMacros.PROJECT_FILE),
        @Storage(id = "dir", file = StoragePathMacros.PROJECT_CONFIG_DIR + "/mystorage.xml", scheme = StorageScheme.DIRECTORY_BASED)
    }
)
public class SessionComponent implements ProjectComponent, PersistentStateComponent<SessionState> {

    Project project;
    SessionState sessionState;

    public SessionComponent(Project project) {
        this.project = project;
        sessionState = new SessionState();
    }

    public void initComponent() {
        // TODO: insert component initialization logic here
    }

    @Override
    public void loadState(SessionState sessionState) {
        System.out.println("load sessionState = " + sessionState);
        this.sessionState = sessionState;
    }

    public void projectOpened() {
        // called when project is opened
    }

    public void projectClosed() {
        // called when project is being closed
    }

    @Nullable
    @Override
    public SessionState getState() {
        System.out.println("save sessionState = " + sessionState);
        return sessionState;
    }

    public void disposeComponent() {
        // TODO: insert component disposal logic here
    }

    @NotNull
    public String getComponentName() {
        return "SessionComponent";
    }

    public int saveCurrentTabs() {
        Editor[] editors = getOpenEditors();
        FileEditorManager fileEditorManager = FileEditorManager.getInstance(project);
        VirtualFile[] selectedFiles = fileEditorManager.getSelectedFiles();

        FileDocumentManager fileDocManager = FileDocumentManager.getInstance();
        sessionState.files = new String[editors.length];
        int i = 0;
        for(Editor editor : editors) {
            VirtualFile vf = fileDocManager.getFile(editor.getDocument());
            String path = vf.getCanonicalPath();
            System.out.println("path = " + path);
            if(path.equals(selectedFiles[0].getCanonicalPath())) {
                sessionState.focusedFile = path;
            }
            sessionState.files[i] = path;
            i++;
        }

        return editors.length;
    }

    public int loadSession() {
        closeCurrentTabs();
        FileEditorManager fileEditorManager = FileEditorManager.getInstance(project);
        for(String path : sessionState.files) {
            System.out.println("path = " + path);
            VirtualFile vf = LocalFileSystem.getInstance().findFileByPath(path);
            fileEditorManager.openFile(vf, true, true);
        }

        return sessionState.files.length;
    }

    public void closeCurrentTabs() {
        Editor[] editors = getOpenEditors();
        FileEditorManager fileEditorManager = FileEditorManager.getInstance(project);
        FileDocumentManager fileDocManager = FileDocumentManager.getInstance();
        for(Editor editor : editors) {
            System.out.println("editor = " + editor);
            VirtualFile vf = fileDocManager.getFile(editor.getDocument());
            fileEditorManager.closeFile(vf);
        }
    }

    public void showMessage(String htmlText) {
        StatusBar statusBar = WindowManager.getInstance().getStatusBar(project);
        JBPopupFactory.getInstance()
            .createHtmlTextBalloonBuilder(htmlText, MessageType.INFO, null)
            .setFadeoutTime(7500)
            .createBalloon()
            .show(RelativePoint.getCenterOf(statusBar.getComponent()), Balloon.Position.atRight);
    }

    private Editor[] getOpenEditors() {
        return EditorFactory.getInstance().getAllEditors();
    }

}

Нам также нужен класс хранения:

public class SessionState {
    public String[] files = new String[0];
    public String focusedFile = "";

    public String toString() {
        String result = "";
        for (String file : files) {
            result += file + ", ";
        }
        result += "selected: " + focusedFile;
        return result;
    }
}

Класс компонента должен иметь запись в вашем файле plugin.xml, подобную этой:

<project-components>
  <component>
    <implementation-class>my.package.SessionComponent</implementation-class>
  </component>
</project-components>

Класс компонента предлагает все необходимые функции, но никогда не используется. Поэтому нам нужны действия для выполнения загрузки и сохранения:

Save.java:

public class Save extends AnAction {

    public Save() {
        super();
    }

    public void actionPerformed(AnActionEvent event) {
        Project project = event.getData(PlatformDataKeys.PROJECT);
        SessionComponent sessionComponent = project.getComponent(SessionComponent.class);

        int tabCount = sessionComponent.saveCurrentTabs();
        String htmlText = "Saved " + String.valueOf(tabCount) + " tabs";
        sessionComponent.showMessage(htmlText);
    }

}

Load.java:

public class Load extends AnAction {

    public Load() {
        super();
    }

    public void actionPerformed(AnActionEvent event) {
        Project project = event.getData(PlatformDataKeys.PROJECT);
        SessionComponent sessionComponent = project.getComponent(SessionComponent.class);

        int tabCount = sessionComponent.loadSession();
        String htmlText = "Loaded " + String.valueOf(tabCount) + " tabs";
        sessionComponent.showMessage(htmlText);
    }

}

Aaand... Action!

Последнее, что нам нужно, - это пользовательский интерфейс для выбора этих действий. Просто поставьте это в свой plugin.xml:

  <actions>
    <!-- Add your actions here -->
      <group id="MyPlugin.SampleMenu" text="_Sample Menu" description="Sample menu">
          <add-to-group group-id="MainMenu" anchor="last"  />
          <action id="MyPlugin.Save" class="my.package.Save" text="_Save" description="A test menu item" />
          <action id="MyPlugin.Load" class="my.package.Load" text="_Load" description="A test menu item" />
      </group>
  </actions>

Развертывание плагинов

Базовая функциональность готова. Я добавлю поддержку нескольких сеансов и некоторых других опрятных вещей перед развертыванием этого плагина и выпуском его в сообщество с открытым исходным кодом. Ссылка будет размещена здесь, когда он будет в сети.