AndroidStuio插件開(kāi)發(fā)適用于jetbrains全家桶
文章目錄 創(chuàng)建項(xiàng)目創(chuàng)建類(lèi)獲取文件解析文件展示解析內(nèi)容寫(xiě)入文件
寫(xiě)了個(gè)類(lèi)似Butter Knife
的開(kāi)發(fā)庫(kù),但是并沒(méi)有與其配套的AndroidStudio插件,抽時(shí)間研究了以下IDEA的api文檔,擼了一個(gè)對(duì)應(yīng)的插件,源碼在這里
之前寫(xiě)的一些小demo可以看看這篇文章
初步編寫(xiě)IDEA\AndroidStudio翻譯插件
以及另外一個(gè)插件項(xiàng)目
https://github.com/huangyuanlove/VariableNameStyleTransfer
本項(xiàng)目代碼參考 android-butterknife-zelezny
所用到的知識(shí)點(diǎn):
- 查找文件
- 解析xml
- 寫(xiě)文件
IDEA插件開(kāi)發(fā)文檔:https://jetbrains.org/intellij/sdk/docs/intro/welcome.html
創(chuàng)建項(xiàng)目
官方推薦創(chuàng)建gradle項(xiàng)目,這里貼個(gè)圖,創(chuàng)建過(guò)程按照官網(wǎng)敘述的創(chuàng)建就好
https://jetbrains.org/intellij/sdk/docs/tutorials/build_system/prerequisites.html
這里說(shuō)明一下,如果想要在AndroidStudio中進(jìn)行debug,閱讀一下這個(gè)
https://jetbrains.org/intellij/sdk/docs/products/android_studio.html
也就是在項(xiàng)目根目錄的的build.gradle
中配置 intellij
和runIde
,具體含義可在網(wǎng)頁(yè)中找到,這里不再贅述
// See https://github.com/JetBrains/gradle-intellij-plugin/ intellij { version '201.8743.12' type 'IC' plugins = ['android', 'java'] } runIde { // Absolute path to installed target 3.5 Android Studio to use as IDE Development Instance // The "Contents" directory is macOS specific. // ideDirectory '/Applications/Android Studio.app/Contents' //for mac // ideDirectory '/home/huangyuan/androidStudio' //for linux ideDirectory 'G:\\AndroidStudio' //for window }
創(chuàng)建類(lèi)
創(chuàng)建一個(gè)繼承AnAction 的類(lèi),這里創(chuàng)建的方式有兩種,一個(gè)是直接創(chuàng)建java類(lèi),然后再去注冊(cè);另外一個(gè)就是通過(guò)想到直接創(chuàng)建(就像我們創(chuàng)建Activity一樣);
具體可以看這里 https://jetbrains.org/intellij/sdk/docs/tutorials/action_system/working_with_custom_actions.html
這里我們需要解析layout文件(xml文件)并且還要寫(xiě)入文件,所以就直接繼承BaseGenerateAction
,重寫(xiě)其中的兩個(gè)方法
@Override public void update(@NotNull AnActionEvent e) { // Using the event, evaluate the context, and enable or disable the action. e.getPresentation().setEnabledAndVisible(e.getProject() != null); } @Override public void actionPerformed(@NotNull AnActionEvent event) { // Using the event, implement an action. For example, create and show a dialog. }
當(dāng)工程處于indexing的時(shí)候,我們不想讓插件生效,可以實(shí)現(xiàn)DumbAware
接口,繼續(xù)向actionPerformed
方法中添加邏輯
@Override public void actionPerformed(@NotNull AnActionEvent event) { //獲取工程對(duì)象,具體信息可以看這里 https://jetbrains.org/intellij/sdk/docs/basics/project_structure.html Project project = event.getData(PlatformDataKeys.PROJECT); if(project ==null){ return ; } Editor editor = event.getData(PlatformDataKeys.EDITOR); if(editor ==null){ return; } DumbService dumbService = DumbService.getInstance(project); if (dumbService.isDumb()) { dumbService.showDumbModeNotification("ViewInject plugin is not available during indexing"); return; } //這里是我們自己的邏輯 analyze(project, editor); }
獲取文件
我們可以獲取到當(dāng)前光標(biāo)所指向的位置,也可以獲取當(dāng)前選中的字符,我們從官方文檔中找到我們自己需要的東西:需要看一下PSI(Program Structure Interface),具體信息在這里https://jetbrains.org/intellij/sdk/docs/basics/architectural_overview/psi.html
關(guān)鍵信息在PSI element 和 PSI Files,項(xiàng)目的中的具體邏輯在GetLayoutFileUtil.java
,這里比較麻煩一些,用到了Module
和GlobalSearchScope
這兩個(gè)類(lèi),具體可以看這里https://jetbrains.org/intellij/sdk/docs/reference_guide/project_model/module.html ,就不再抄一遍+翻譯了
解析文件
這里我們拿到了對(duì)應(yīng)的layout.xml
文件對(duì)象,一個(gè)PsiFile
對(duì)象,調(diào)用文件的遍歷方法layoutFile.accept(PsiElementVisitor visitor)
,這里我們傳入XmlRecursiveElementVisitor
實(shí)例對(duì)象,在解析xml的過(guò)程中,我們可能會(huì)遇到<include>
標(biāo)簽,需要繼續(xù)解析該標(biāo)簽下的xml文件,這里搞個(gè)遞歸。
展示解析內(nèi)容
解析出來(lái)的數(shù)據(jù)存入ArrayList中,在解析過(guò)程中,保存了對(duì)應(yīng)id、判斷是否引用了android name space中id、將id的值轉(zhuǎn)化為對(duì)應(yīng)的字段名字、保存了是是否為自定義的view等信息
public String id; public boolean isAndroidNS = false; public String nameFull; // element name with package public String name; // element name public String fieldName; // name of variable public boolean isValid = false; public boolean used = true; public boolean isClick = true;
展示解析內(nèi)容使用javax.swing
組件,這個(gè)也沒(méi)什么好說(shuō)的。
在展示面板上提供的全選功能;提供了生成代碼的兩種格式
@BindView(idStr = "xxxx") //可在library、application中使用 @BindView(id = R.id.xxx) //僅在application中使用
因?yàn)樵趌ibrary中生成的R文件中的變量不是final類(lèi)型,并且application中的R文件變量,在gradle plugin 5.0之后也不再是final的,所以建議使用idStr的方式,也是默認(rèn)生成的代碼
寫(xiě)入文件
為了方便,寫(xiě)入文件的時(shí)候使用的是PsiClass
對(duì)象進(jìn)行操作的,源碼在這里,至于如何操作PsiFile,可以看這里。寫(xiě)入文件的過(guò)程,看起來(lái)個(gè)使用javapoet
差不多,javapoet可以看這里,github上直接搜索即可
private void generateClick() { for (ElementBean element : mElements) { if (element.isClick) { StringBuilder method = new StringBuilder(); method.append("@ClickResponder(" + element.getGenerateValue(generateId) + ")"); method.append("public void onClick" + Utils.capitalize(element.fieldName) + " (View v) {}"); mClass.add(mFactory.createMethodFromText(method.toString(), mClass)); } } }
在寫(xiě)入類(lèi)字段的時(shí)候,需要判斷是否需要添加前綴,在Constant
中列舉了一些需要特殊處理的對(duì)象
protected void generateFields() { for (ElementBean element : mElements) { if (!element.used) { continue; } StringBuilder injection = new StringBuilder(); injection.append("@BindView"); injection.append('('); injection.append(element.getGenerateValue(generateId)); injection.append(")"); if (element.nameFull != null && element.nameFull.length() > 0) { // custom package+class injection.append(element.nameFull); } else if (Constant.paths.containsKey(element.name)) { // listed class injection.append(Constant.paths.get(element.name)); } else { // android.widget injection.append("android.widget."); injection.append(element.name); } injection.append(" "); injection.append(element.fieldName); injection.append(";"); mClass.add(mFactory.createFieldFromText(injection.toString(), mClass)); } }
寫(xiě)入完成后格式化一下代買(mǎi),要不然寫(xiě)入的字段會(huì)是這樣:android.widget.TextView userNameTextView
JavaCodeStyleManager styleManager = JavaCodeStyleManager.getInstance(mProject); styleManager.optimizeImports(mFile); styleManager.shortenClassReferences(mClass); new ReformatCodeProcessor(mProject, mClass.getContainingFile(), null, false).runWithoutProgress();
到此為止,就已經(jīng)完成了我們的工作。
到此這篇關(guān)于AndroidStuio插件開(kāi)發(fā)適用于jetbrains全家桶的文章就介紹到這了,更多相關(guān)AndroidStuio插件開(kāi)發(fā)jetbrains全家桶內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Kotlin中實(shí)體類(lèi)的創(chuàng)建方式
這篇文章主要介紹了Kotlin中實(shí)體類(lèi)的創(chuàng)建方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Android中使用GridView實(shí)現(xiàn)仿微信圖片上傳功能(附源代碼)
由于工作要求最近在使用GridView完成圖片的批量上傳功能,我的例子當(dāng)中包含仿微信圖片上傳、拍照、本地選擇、相片裁剪等功能,如果有需要的朋友可以看一下2017-08-08Android使用動(dòng)畫(huà)動(dòng)態(tài)添加商品進(jìn)購(gòu)物車(chē)
這篇文章主要為大家詳細(xì)介紹了Android使用動(dòng)畫(huà)動(dòng)態(tài)添加商品進(jìn)購(gòu)物車(chē),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06Input系統(tǒng)之InputReader處理合成事件詳解
這篇文章主要為大家介紹了Input系統(tǒng)之InputReader處理合成事件詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11Android Studio 1.2版安裝設(shè)置圖文教程
這篇文章主要介紹了Android Studio 1.2版安裝設(shè)置圖文教程,本文詳細(xì)講解了下載、安裝Android Studio 1.2教程,以及常用設(shè)置詳細(xì)圖文教程,需要的朋友可以參考下2015-05-05關(guān)于gradle你應(yīng)該知道的一些小事
這篇文章主要給大家介紹了關(guān)于gradle你應(yīng)該知道的一些小事,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用gradle具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-10-10