前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >编写一个IDEA插件之:自动生成Java代码

编写一个IDEA插件之:自动生成Java代码

作者头像
吴就业
发布2020-10-27 16:17:28
1.8K0
发布2020-10-27 16:17:28
举报
文章被收录于专栏:Java艺术Java艺术

我很喜欢IDEA的一键自动生成代码功能,例如自动生成构造方法、字段的Get/Set方法、ToString方法等等,除此之外,也有一些插件提供自动生成代码的功能,例如我们所熟悉的GsonFormat插件,使用该插件可以为我们快速的解析json字符串生成一个对应的Java类,这在对接一些第三方API时很有帮助。

笔者写过一个运行时根据json自动生成Class的工具包:json-class-generator,与GsonFormat不同的是,该工具使用ASM在运行时解析json结构树生成类的字节码,而GsonFormat生成的是Java源代码。当时写json-class-generator目的是实现一个第三方API自动对接框架,由于该框架涉及到业务,所以没有开源。

虽然json-class-generatorGsonFormat实现的功能不同,但原理相似。

上一篇我们了解到,Java源代码编译后生成的Class文件有固定的结构,而在IDEA中,Java源代码也同样有固定的结构:PSI程序结构。与使用ASM操作字节码修改一个Class文件一样,我们也可以通过编辑一个Java源代码的PSI程序结构的元素修改Java代码。

读懂本篇的前提是你已经对PSI有所了解,如果尚未了解PSI,可以先看下笔者写的上一篇《编写一个IDEA插件之:PSI分析Java源代码》。

自动生成Java源代码

我们模仿IDEA提供的自动生成代码功能,给右键弹出菜单的Generate...菜单添加一个子菜单:GeneratedInvokePayMethod,在插件使用者点击该菜单时自动生成一串代码,并且生成的代码插入到当前光标所在位置。

首先需要编写一个对应GeneratedInvokePayMethod菜单的Action,并实现actionPerformed方法,代码如下。

代码语言:javascript
复制
public class GeneratedInvokePayMethodAction extends AnAction {
    @Override
    public void actionPerformed(@NotNull AnActionEvent event) {
    }
}

actionPerformed方法在菜单被点击时调用,该方法只有一个参数:

  • event:这个参数封装了很多有用的信息,比如我们可以从该参数获取当前文件的PsiFile实例、获取当前光标落在的PsiElement等。

其次,我们需要注册Action,将Action放到右键弹出菜单的GenerateGroup。需要在plugin.xml文件添加如下配置信息:

代码语言:javascript
复制
<actions>
    <action id="xxx.action.GeneratedInvokePayMethodAction" class="com.xxx.plugin.action.GeneratedInvokePayMethodAction"
                text="GeneratedInvokePayMethod">
       <!-- 将action放在哪 -->
       <add-to-group group-id="GenerateGroup" anchor="first"/>
   </action>
</actions>

效果如下图所示。

现在我们继续完成GeneratedInvokePayMethodActionactionPerformed方法。

由于Intellij Platform不允许插件在主线程中进行实时的文件写入,只能通过异步任务来完成写入,因此,我们需要通过WriteCommandAction.runWriteCommandAction来执行一个后台写入操作,如下代码所示。

代码语言:javascript
复制
public class GeneratedInvokePayMethodAction extends AnAction {

    @Override
    public void actionPerformed(@NotNull AnActionEvent event) {
      // 立即执行一个后台任务
      WriteCommandAction.runWriteCommandAction(editor.getProject(), () -> {
           // do ...
      });
    }
}

想要在当前光标所在的位置插入一行代码,那么我们需要做这些事情:

  • 1、先判断当前文件是否是一个Java文件,借助actionPerformedevent参数可取得当前文件的PsiFile实例,判断PsiFile实例的类型是否为PsiJavaFile,如果不是,说明这不是一个Java代码文件,什么也不需要做(或者可以给出对话框提示);
代码语言:javascript
复制
// AnActionEvent event
PsiFile psiFile = event.getData(LangDataKeys.PSI_FILE);
  • 2、通过第一步获取的PsiFile,查找当前光标所在位置的PsiElement实例;
代码语言:javascript
复制
PsiElement element = psiFile.findElementAt(editor.getCaretModel().getOffset());

其中editor.getCaretModel().getOffset()为获取当前光标位置;

另外,可以使用AnActionEvent#getData方法获取当前光标所在的PsiElement,代码如下:

代码语言:javascript
复制
// AnActionEvent event
PsiElement psiElement = event.getData(LangDataKeys.PSI_ELEMENT);

但这种方式不适用于当前场景,如果将光标放在一行代码的;后面,那么该方法就会返回null值。

  • 3、根据光标所在的PsiElement,获取该PsiElement所在方法的PsiCodeBlock(一个方法只有一个PsiCodeBlock);
代码语言:javascript
复制
PsiElement codeBlock = element;
while (!(codeBlock instanceof PsiCodeBlock)) {
     codeBlock = codeBlock.getParent();
}
  • 4、创建新的PsiElement,该PsiElement就是需要自动生成的代码;例如创建一个表达式元素(PsiExpression),可使用PsiElementFactory#createExpressionFromText方法创建,代码如下。
代码语言:javascript
复制
PsiElement newElement = PsiElementFactory.getInstance(element.getProject())
        .createExpressionFromText("Invocation<Object> invocation = Invocation.<Object>builder()\n" +
                                    "                .scope(scope)\n" +
                                    "                .service(payType)" +
                                    "                .operate(\"" + method + "\")\n" +
                                    "                .body(merchantNo)\n" +
                                    "                .build()", element.getContext());

PsiElementFactory使用工厂模式生产PsiElement,提供了大量的API,例如创建字段的createField、创建方法的createMethod、创建类的createClass,创建关键字的createKeyword

  • 5、最后,将新创建的PsiElement添加到光标所在PsiElement的后面;
代码语言:javascript
复制
// 参数1:新增的PsiElement
// 参数2:位置参照的PsiElement
codeBlock.addAfter(newElement, element);

完整示例代码如下。

代码语言:javascript
复制
public class GeneratedInvokePayMethodAction extends AnAction {

    @Override
    public void actionPerformed(@NotNull AnActionEvent event) {
        WriteCommandAction.runWriteCommandAction(editor.getProject(), () -> {
           PsiFile psiFile = event.getData(LangDataKeys.PSI_FILE);
           // 查找当前光标停留在的元素
           PsiElement element = psiFile.findElementAt(editor.getCaretModel().getOffset());
           // 获取当前方法的PsiCodeBlock元素
           PsiElement codeBlock = element;
           while (!(codeBlock instanceof PsiCodeBlock)) {
                codeBlock = codeBlock.getParent();
           }
           // 使用PsiElementFactory创建表达式元素
           PsiElement newElement = PsiElementFactory.getInstance(element.getProject())
                 .createExpressionFromText("Invocation<Object> invocation = Invocation.<Object>builder()\n" +
                                    "                .scope(scope)\n" +
                                    "                .service(payType)" +
                                    "                .operate(\"" + method + "\")\n" +
                                    "                .body(merchantNo)\n" +
                                    "                .build()", element.getContext());
           // 将新创建的表达式元素插入到光标停留在的元素的后面
           codeBlock.addAfter(newElement, element);
      });
    }
}

后记

实际要实现一个插件可能没有那么简单,例如本篇没有介绍到的UI部分,笔者省略了一些步骤:当点击菜单时,先弹出一个Dialog,提供一些选项,在完成选项点击ok后再生成代码。

编写插件UI其实与开发Android应用编辑UI布局类似,如果你开发过Android应用,那么也就不难理解。

[Java艺术] 微信号:javaskill

一个只推送原创文章的技术公众号,分享Java后端相关技术。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2020-10-11,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java艺术 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 自动生成Java源代码
  • 后记
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档