idea插件开发入门实战


文章目录

  • idea插件开发入门实战
  • 前言
  • 一、书写第一个Hello World
  • 二、IDEA插件开发进阶
    • 1.基于java文件的规则校验
    • 2.基于XML文件规则的校验
    • 3.基于java文件的代码自动生成方法
    • 4.基于XML文件的读写方法
    • 5.XML跳转Java文件

前言

建议先看一遍官方文档再开始

IDEA插件开发官方文档


一、书写第一个Hello World

新建项目


完事之后先修改plugin.xml配置文件。

<idea-plugin><id>com.yishu.plugin</id><name>插件入门实战</name><version>1.0</version><vendor email="1165883867@qq.com" url="http://www.xxx.com">顺便写啥</vendor><description>描述。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。</description><change-notes>啦啦啦。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。</change-notes><idea-version since-build="173.0"/><extensions defaultExtensionNs="com.intellij"><!-- Add your extensions here --></extensions><actions><!-- Add your actions here --></actions></idea-plugin>

再新建第一个action


这里新建一个java文件也是可以的,但是这样需要自己配置xml,不推荐。

此时xml文件会多出一行配置

开始编写第一个hello world

import com.intellij.openapi.actionSystem.AnAction;import com.intellij.openapi.actionSystem.AnActionEvent;import com.intellij.openapi.actionSystem.PlatformDataKeys;import com.intellij.openapi.project.Project;import com.intellij.openapi.ui.Messages;public class FirstAction extends AnAction {@Overridepublic void actionPerformed(AnActionEvent e) {Project project = e.getData(PlatformDataKeys.PROJECT);Messages.showMessageDialog(project,"Hello,World","First Action", Messages.getInformationIcon());}}

点击运行

点击idea help菜单


恭喜你入门成功。

二、IDEA插件开发进阶

1.基于java文件的规则校验

IDEA默认有对于JAVA的一些校验规则,如果我们想自己定义一些规则,如命名方式什么的,让不符合规则的内容爆红线,应该怎么做呢?
方法如下。

代码:

import com.intellij.codeInspection.AbstractBaseJavaLocalInspectionTool;import com.intellij.codeInspection.ProblemsHolder;import com.intellij.psi.JavaElementVisitor;import com.intellij.psi.PsiElementVisitor;import com.intellij.psi.PsiField;import org.jetbrains.annotations.NotNull;public class JavaInspection extends AbstractBaseJavaLocalInspectionTool {@Overridepublic PsiElementVisitor buildVisitor(@NotNull final ProblemsHolder holder, final boolean isOnTheFly){JavaElementVisitor javaElementVisitor = new JavaElementVisitor() {@Overridepublic void visitField(PsiField field) {super.visitField(field);if (field.getName().equals("name")){holder.registerProblem(field,"命名非法");}}};return javaElementVisitor;}}

XML需要新增配置

<extensions defaultExtensionNs="com.intellij"><localInspectionid="inspection.JavaInspection"language="JAVA"shortName="JavaInspection"displayName="JavaInspection"groupName="JavaInspection"enabledByDefault="true"level="ERROR"implementationClass="com.yishu.plugin.inspection.JavaInspection"/></extensions>

language 这里表示作用于JAVA文件
此时我们重新运行,新建查看效果。


如果想实现对方法,类名,包名等的校验,只需要重写相应的方法就行了。这里的方法有很多

2.基于XML文件规则的校验

和Java类似,这里我示范一个对于标签属性的校验。
直接贴代码和配置:

import com.intellij.codeInspection.AbstractBaseJavaLocalInspectionTool;import com.intellij.codeInspection.ProblemsHolder;import com.intellij.psi.PsiElementVisitor;import com.intellij.psi.XmlElementVisitor;import com.intellij.psi.xml.XmlAttribute;import org.jetbrains.annotations.NotNull;public class XMLInspection extends AbstractBaseJavaLocalInspectionTool {@Overridepublic PsiElementVisitor buildVisitor(@NotNull final ProblemsHolder holder, final boolean isOnTheFly){XmlElementVisitor xml = new XmlElementVisitor(){@Overridepublic void visitXmlAttribute(XmlAttribute attribute) {super.visitXmlAttribute(attribute);if (attribute.getName().equals("name") && attribute.getValue().equals("亦疏")){holder.registerProblem(attribute,"描述哔巴拉巴拉");}}};return xml;}}

配置:

<localInspectionid="inspection.XMLInspection"language="XML"shortName="XMLInspection"displayName="XMLInspection"groupName="XMLInspection"enabledByDefault="true"level="ERROR"implementationClass="com.yishu.plugin.inspection.XMLInspection"/>

效果

3.基于java文件的代码自动生成方法

代码:

import com.intellij.openapi.actionSystem.AnAction;import com.intellij.openapi.actionSystem.AnActionEvent;import com.intellij.openapi.actionSystem.CommonDataKeys;import com.intellij.openapi.command.WriteCommandAction;import com.intellij.psi.*;import org.jetbrains.annotations.NotNull;public class GeneratedJavaCode extends AnAction {@Overridepublic void actionPerformed(@NotNull AnActionEvent event) {WriteCommandAction.runWriteCommandAction(event.getProject(), () -> {PsiJavaFile file = (PsiJavaFile)event.getData(CommonDataKeys.PSI_FILE);PsiElementFactory factory = PsiElementFactory.SERVICE.getInstance(event.getProject());file.add(factory.createField("age",PsiType.INT));});}}

配置:

<action id="action.GeneratedJavaCode" class="com.yishu.plugin.action.GeneratedJavaCode"text="生成java代码"><!-- 将action放在哪 --><add-to-group group-id="GenerateGroup" anchor="first"/></action>

效果


这里的age没标记位置,就简单写了个例子,需要自己通过addAfter等方法定位要生成的地方。就不详细说了

4.基于XML文件的读写方法

XML文件比较独特,这里展开讲讲。
建议先看官网介绍
https://plugins.jetbrains.com/docs/intellij/xml-dom-api.html#abstract


官网有提供两种方法读写XML文件,第一种是上面那样的,但是我觉得有点蠢,还不好复用也容易写错。
我们用第二种。
先把示例的XML文件拿到

<root><foo name="Test"><bar name="name">123</bar><bar>456</bar></foo></root>

定义interface

Root

import com.intellij.util.xml.DomElement;import com.intellij.util.xml.SubTag;public interface Root extends DomElement {@SubTag("foo")Foo getFoo();}

Foo

public interface Foo extends DomElement {@Attribute("name")GenericAttributeValue<String> getName();@SubTagList("bar")List<Bar> getBars();Bar addBar();}

Bar

public interface Bar extends DomElement {String getValue();void setValue(String s);@Attribute("name")GenericAttributeValue<String> getName();}

注解记得加上,因为子标签不是唯一的时候,不加注解会获取不到

注册DomFileDescription:

public class XmlDescription extends DomFileDescription<Root> {public XmlDescription() {super(Root.class, "root", "");}}

XML配置


读取和生成逻辑的代码

import com.intellij.openapi.actionSystem.AnAction;import com.intellij.openapi.actionSystem.AnActionEvent;import com.intellij.openapi.actionSystem.LangDataKeys;import com.intellij.openapi.command.WriteCommandAction;import com.intellij.openapi.project.Project;import com.intellij.psi.PsiClass;import com.intellij.psi.PsiField;import com.intellij.psi.search.GlobalSearchScope;import com.intellij.psi.search.PsiShortNamesCache;import com.intellij.psi.xml.XmlFile;import com.intellij.util.xml.DomManager;import com.yishu.plugin.service.Bar;import com.yishu.plugin.service.Foo;import com.yishu.plugin.service.Root;import org.jetbrains.annotations.NotNull;public class GeneratedXMLCode extends AnAction {@Overridepublic void actionPerformed(@NotNull AnActionEvent event) {Project project = event.getProject();XmlFile xmlFile = (XmlFile)event.getData(LangDataKeys.PSI_FILE);DomManager domManager = DomManager.getDomManager(project);Root root = domManager.getFileElement(xmlFile, Root.class).getRootElement();System.out.println("获取value值"+root.getFoo().getBars().get(0).getValue());System.out.println("获取name属性的值"+root.getFoo().getBars().get(0).getName().getValue());WriteCommandAction.runWriteCommandAction(project,()->{//写生成的逻辑//比如根据Foo的name 生成name对应java类的字段//获取fooFoo foo = root.getFoo();//获取classPsiShortNamesCache shortNamesCache = PsiShortNamesCache.getInstance(project);PsiClass cla = shortNamesCache.getClassesByName(foo.getName().getValue(), GlobalSearchScope.projectScope(project))[0];for (PsiField field:cla.getFields()){Bar bar = foo.addBar();bar.getName().setValue(field.getName());bar.setValue("啦啦啦");}});}}

按alt+insert

效果

5.XML跳转Java文件

代码

public class XMLToJavaProvider extends RelatedItemLineMarkerProvider {@Overrideprotected void collectNavigationMarkers(@NotNull PsiElement element, @NotNull Collection<? super RelatedItemLineMarkerInfo> result){//获取当前文件Project project = element.getProject();PsiFile containingFile = element.getContainingFile();DomManager domManager = DomManager.getDomManager(project);XmlFile xmlFile = (XmlFile) containingFile;Root root = domManager.getFileElement(xmlFile, Root.class).getRootElement();if (element instanceof XmlTag){XmlTag tag = (XmlTag)element;System.out.println(tag.getName());switch (tag.getName()){case "foo":String className = root.getFoo().getName().getStringValue();createMark(className,element,result);break;default:break;}}}private void createMark(String className,PsiElement element,Collection<? super RelatedItemLineMarkerInfo> result){Project project = element.getProject();Optional.ofNullable(className).ifPresent(c->{PsiShortNamesCache shortNamesCache = PsiShortNamesCache.getInstance(project);PsiClass psiClass = shortNamesCache.getClassesByName(className, GlobalSearchScope.projectScope(project))[0];Optional.ofNullable(psiClass).ifPresent(p->{NavigationGutterIconBuilder<PsiElement> builder = NavigationGutterIconBuilder.create(AllIcons.Actions.Back).setTarget(psiClass).setTooltipTitle("跳转java文件");result.add(builder.createLineMarkerInfo(element));});});}}

配置:放在extensions标签下

<codeInsight.lineMarkerProvider implementationClass="com.yishu.plugin.provider.XMLToJavaProvider" language="XML"/>

启动看看效果: