Sometimes it is a nice thing to extend an editor to have it do some new stuff, like being able to revert an open file to its state on disk. In this post I’m going to quickly demonstrate how to start and finish a plugin to extend theIntelliJ IDEA editor from Jetbrains. These instructions will work if you have the Ultimate or Community edition, and you can make the plugin available for all IntelliJ based editors including WebStorm, PyCharm, RubyMine and PHPStorm.
Whether you can finish these instructions under 30 minutes depends on your internet connection speed, because you have to do a git clone of the IntelliJ Community Edition source code, and that’s a lot of stuff to download.
At my current employer Floobits we are working on an IntelliJ plugin, and through that process I have learned most of this.
System requirements
OS: I have developed plugins for IntelliJ on both Mac OS X and Windows and the steps are the same.
Java: JDK 6 works, JDK 7 might not. I compile with a Java 1.6 JDK.
HD space: Lots of space for multiple copies of IntelliJ IDEA and its source code.
Getting the environment set up.
Resource: Official Jetbrain instructions
Step 1: Download and install Community Edition of IntelliJ IDEA
The first step, is to download and install the Community version of IntelliJ IDEA if you don’t already have it. Even if you have the Ultimate version of IntelliJ IDEA, you will need the Community version. This is because the Ultimate version is closed source and when you’re debugging it is nice to be able to see the source and definitions for IntelliJ resources. Note I still use the Ultimate version of IntelliJ IDEA (version 13) to develop the plugin. I debug and test the plugin using the Community version.
Remember the path to where the Community Edition was installed to, you’ll need this in a moment.
Step 2: Do a shallow git clone of the IntelliJ IDEA Community Edition source code
git clone --depth 1 https://github.com/JetBrains/intellij-community.git
This step will take the longest because it takes forever to do a git clone of the IntelliJ source code. Thus do a shallow git clone using –depth 1 in the command. This is really simple on a Mac or any *nix based system, just use git clone –depth 1. On Windows I used Atlassian’s SourceTree which allows you to specify the depth in the GUI. A depth of 1 reduces the download because the history is truncated to a depth of 1. If you do not do this step, downloading the source code will take you days.
Step 3: Create a project and set up your project structure.
Create a project
Fire up the version of IntelliJ that you want to use to write code. This can be either of the Ultimate or Community versions. Create a new project. Name it whatever you want to call your plugin. I’m going to call mine DiskRead.
Set up the SDK
If you already have a Java JDK skip this paragraph. If you don’t have one already, you’ll have to get a Java JDK. Somewhere or other on the Jetbrains site it recommends Java 1.6, but Oracle will warn you that it is out of date. I have had problems using the JDK 7 when interacting with Intellij jar files compiled with 1.6. Once you have downloaded a Java JDK, you’ll need to create an SDK for this in IntelliJ IDEA even though we will not use it directly. In the project structure dialog that appears after you create a new project, create a new SDK for the version of Java you downloaded similar to the step below, only you’ll be giving it the path to where you installed the Java JDK.
Setup a Plugin SDK by clicking the “new” button next to the Project SDK:
Note that the SDK setup project structure dialog looks a little different on Windows.
It will prompt you to locate the Comunity Edition that you installed. You are not to give it the path to the source you downloaded, this step requires the path to the Community Edition of IntelliJ that you installed in the first step. On a Mac the path for this is:
/Applications/IntelliJ IDEA 13 CE.app
On Windows the path was:
C:\\Program Files (x86)\JetBrains\IntelliJ IDEA Community Edition 13.0.1
Click finish if you are happy with where it will create the project, but you are not yet done setting things up.
You should now see a project window for the project you just created. Go to File->Project Structure from the menu. On a mac you can just hit ⌘+; on Windows it’s Cntrl+Alt+Shift+S. On the left side, click on “SDKs” and check out the classpath tab:
Next click on the sourcepath tab:
Here you’ll add the path to the shallow git clone for the IntelliJ Community edition source you checked out. Click the plus button at the bottom. Find the path in the dialog that appears. Click OK, then you should see a bunch of classes were added. Click apply and ok.
We are still not done settings everything up. Next you’ll want to set up a build and debug configuration. From the top menu go to Run -> Edit configurations:
Give the build configuration a name, it’s not important what it is. I also recommend checking “Show idea.log” because it gives you a nice tab in the debug window to show you info and warning statements.
Congratulations, you are now all done setting up!
Developing the plugin
Resources
IntelliJ IDEA Plugin Development Guide
IntelliJ IDEA Architectural Overview
Making it work.
So now that the environment is all set up, in order to get my file to reflect the state on disk I need to create an action. First thing is to use the GUI to create a new action:
So I right click on src in the project sidebar and click “new” and then click “Action.” This will lead to a dialog to create a new Action. You give it a name, an id, a class name, a helpful description and then assign it to a group which will tell IntelliJ where to put this action. Actions don’t have to be a part of groups, and you can even make buttons that go into the project UI, but in this case I just want it to be in the “toolbar” menu like so:
You can also assign keyboard shortcuts, as I did.
Once you’re all done there click OK and two things should have happened:
Your plugin.xml should be updated to reflect your new action. You can see this XML file by selecting it from within the META-INF directory in your project sidebar. You should see your action in the actions section.
A new class called ReadCurrentFile was created.
For me the new class looks like this:
import com.intellij.openapi.actionSystem.AnAction; import com.intellij.openapi.actionSystem.AnActionEvent; /** * User: bjorn * Date: 1/29/14 * Time: 12:28 PM */ public class ReadCurrentFile extends AnAction { public void actionPerformed(AnActionEvent e) { // TODO: insert action logic here } }
I do some little magic coding and write the code to get my plugin to do what I want:
import com.intellij.openapi.actionSystem.AnAction; import com.intellij.openapi.actionSystem.AnActionEvent; import com.intellij.openapi.application.ApplicationManager; import com.intellij.openapi.command.CommandProcessor; import com.intellij.openapi.editor.Document; import com.intellij.openapi.editor.Editor; import com.intellij.openapi.fileEditor.FileDocumentManager; import com.intellij.openapi.fileEditor.FileEditorManager; import com.intellij.openapi.project.Project; import com.intellij.openapi.vfs.VirtualFile; import com.sun.org.apache.bcel.internal.generic.NEW; import java.io.*; import java.util.regex.Pattern; /** * User: bjorn * Date: 1/29/14 * Time: 12:28 PM */ public class ReadCurrentFile extends AnAction { public void actionPerformed(AnActionEvent e) { final Project project = e.getProject(); if (project == null) { return; } Editor editor = FileEditorManager.getInstance(project).getSelectedTextEditor(); if (editor == null) { return; } final Document document = editor.getDocument(); if (