前言 大家都经常使用 IDEA 进行开发,肯定会使用一些 IDEA 插件,我之前也写过两个插件,不过已经很久没有更新了,就让它先放着吧! 那小伙伴你是否想亲手写一个插件,或者你是否有一些插件的想法,但是找不到插件。那就自己实现一个吧!

创建项目

使用 Gradle 创建

写插件,先从创建项目开始:
File -> New -> Project...

  • 这里使用 Gradle,其中 Java 已经默认选中,咱们再额外选择 IntelliJ Platform Plugin

IDEA插件开发 - 图1

  • 点击 Next ,然后填写项目名称,路径等选项。

IDEA插件开发 - 图2

  • 项目结构

IDEA插件开发 - 图3
build.gradle 为项目配置文件。
resources/META-INF/plugin.xml 为插件配置文件。

使用 GitHub 模版

  1. 访问 https://github.com/JetBrains/intellij-platform-plugin-template
  2. 点击 Use this template 创建模版。IDEA插件开发 - 图4
  3. Clone 项目到自己本地。

注:模版生成的项目是使用的 Kotlin,所以这里使用的第一种方式创建。

开始开发

修改 build.gradle 配置文件

原内容如下:
IDEA插件开发 - 图5
修改后:

  1. plugins {
  2. id 'java'
  3. id 'org.jetbrains.intellij' version '0.4.15'
  4. }
  5. group 'com.liuzhihang.toolkit'
  6. version '1.0.2'
  7. sourceCompatibility = 1.8
  8. repositories {
  9. mavenLocal()
  10. maven { url "https://maven.aliyun.com/repository/public" }
  11. mavenCentral()
  12. jcenter()
  13. }
  14. dependencies {
  15. testCompile group: 'junit', name: 'junit', version: '4.12'
  16. }
  17. // See https://github.com/JetBrains/gradle-intellij-plugin/
  18. intellij {
  19. version '2019.1.1'
  20. pluginName 'Copy as Json'
  21. updateSinceUntilBuild false
  22. sameSinceUntilBuild false
  23. }
  24. patchPluginXml {
  25. pluginDescription(file(descriptionFile).text)
  26. changeNotes(file(changesFile).text)
  27. }
  1. 修改 repositories 使用阿里云
  2. 修改 patchPluginXml 使用外置文件
  3. 根目录下创建 parts 路径,并创建 changeNotes.htmlpluginDescription.html

    修改 resources/META-INF/plugin.xml 插件信息

    原内容如下:
    IDEA插件开发 - 图6
    修改后:

    1. <idea-plugin>
    2. <id>com.liuzhihang.toolkit.copyasjson</id>
    3. <name>Copy as Json</name>
    4. <vendor email="liuzhihangs@qq.com" url="https://liuzhihang.com">Liu ZhiHang</vendor>
    5. <description><![CDATA[ Description will be added by gradle build]]></description>
    6. <!-- please see http://www.jetbrains.org/intellij/sdk/docs/basics/getting_started/plugin_compatibility.html
    7. on how to target different products -->
    8. <depends>com.intellij.modules.java</depends>
    9. <idea-version since-build="181.00"/>
    10. <extensions defaultExtensionNs="com.intellij">
    11. <!-- Add your extensions here -->
    12. </extensions>
    13. <actions>
    14. </actions>
    15. </idea-plugin>

    plugin.xml 说明:https://jetbrains.org/intellij/sdk/docs/basics/plugin_structure/plugin_configuration_file.html

    创建 Action

  4. 先在 main 下 创建 java 目录,及包路径。

  5. New -> Plugin DevKit -> Action IDEA插件开发 - 图7
  6. 设置 Action 的 idClass Namedescriptiongroup快捷键IDEA插件开发 - 图8
  7. 这时候会发现在 plugin.xml 也插入了 action
    1. <actions>
    2. <action id="Toolkit.Json.CopyAsJsonAction" class="com.liuzhihang.toolkit.action.CopyAsJsonAction"
    3. text="CopyAsJsonAction" description="Copy As Json">
    4. <add-to-group group-id="EditorTabsGroup" anchor="first"/>
    5. <keyboard-shortcut keymap="$default" first-keystroke="shift meta J"/>
    6. </action>
    7. </actions>
    到这里已经结构完全创建完毕了,下面将演示插件 copy-as-json 的内部逻辑。当然这块也可以直接跳过,阅读源码即可。
    源码地址:文末相关资料或公众号发送 copy-as-json 获取。
    插件效果:将 JavaBean 复制为 Json 字符串。
    IDEA插件开发 - 图9

    开发笔记

    首先需要知道一些常用的 API,常用 API 可以阅读官方文档或者关注公众号后面会推送,这里仅介绍一些在这里用到的。
  • 打开 CopyAsJsonAction

该类继承并需要实现 actionPerformed 方法。在 actionPerformed 方法中可以通过以下三个方法获取到项目相关信息:

  1. // 获取项目
  2. Project project = e.getData(PlatformDataKeys.PROJECT);
  3. // 获取Psi文件
  4. PsiFile psiFile = e.getData(CommonDataKeys.PSI_FILE);
  5. // 获取当前编辑的文件
  6. Editor editor = e.getData(CommonDataKeys.EDITOR);
  • 获取到当前编辑的文件

    1. @Nullable
    2. public static PsiClass getTargetClass(@NotNull Editor editor, @NotNull PsiFile file) {
    3. int offset = editor.getCaretModel().getOffset();
    4. PsiElement element = file.findElementAt(offset);
    5. if (element != null) {
    6. // 当前类
    7. PsiClass target = PsiTreeUtil.getParentOfType(element, PsiClass.class);
    8. return target instanceof SyntheticElement ? null : target;
    9. }
    10. return null;
    11. }
  • 从当前编辑的文件里面获取到字段

将当前编辑的 JavaBean 中的字段提取,并转换为 Map。

  1. Map<String, Object> fieldsMap = getFields(selectedClass);

getFields 方法篇幅较长,请参考源码。

  • 将字段转化成 Json 字符串,并格式化

使用 Gson 将 Map 转换为 Json 字符串,并格式化。其中格式化自定义了缩进。
见代码:com.liuzhihang.toolkit.utils.GsonFormatUtil

  1. Gson gson = new GsonBuilder().create();
  2. String json = GsonFormatUtil.gsonFormat(gson, fieldsMap);
  3. // 使用自定义缩进格式 String json = new GsonBuilder().setPrettyPrinting().create().toJson(fieldsMap);
  4. StringSelection selection = new StringSelection(json);
  • 将 Json 字符串拷贝到剪贴板

    1. StringSelection selection = new StringSelection(json);
    2. Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
    3. clipboard.setContents(selection, selection);
  • 发出提示 success

    1. String message = "Convert " + selectedClass.getName() + " to JSON success, copied to clipboard.";
    2. Notification success = NOTIFICATION_GROUP.createNotification(message, NotificationType.INFORMATION);
    3. Notifications.Bus.notify(success, project);

    测试运行

    右侧 Gradle -> 选择 intellij -> 点击 runlde
    IDEA插件开发 - 图10

    打包

    右侧 Gradle -> 选择 intellij -> 点击 buildPlugin
    IDEA插件开发 - 图11
    此时在项目路径下会生成插件,把这个插件包发给小兄弟安装使用就行了。
    IDEA插件开发 - 图12

    上传到 IDEA 插件库

    访问 https://plugins.jetbrains.com/ 创建账号,将插件包上传到仓库即可。当然也有其他的方式,这块就没有研究了。

    总结

    通过上面的方式已经简单开发一个插件了,要问这个插件有什么用?
    其实就是在写文档,或者接口调用的时候,直接将 Java Bean 复制为 Json 串,省过一个一个敲,然后手写 Json 了。
    IDEA插件开发 - 图13
    IDEA插件开发 - 图14

    相关资料

    [1] IntelliJ Platform SDK DevGuide:https://jetbrains.org/intellij/sdk/docs/intro/intellij_platform.html
    [2] JetBrains Plugins Repository:https://plugins.jetbrains.com/
    [3] Toolkit: https://github.com/liuzhihang/toolkit
    [4] copy-as-json:https://github.com/liuzhihang/copy-as-json
    [5] copy-as-json 插件地址:https://plugins.jetbrains.com/plugin/13606-copy-as-json