Android Studio插件開發(fā)之 - IOC注解生成器

1.概述


上一期我們已經(jīng)分享了Android Studio插件開發(fā)之 - 基礎(chǔ)入門篇。那么現(xiàn)在我們來動手寫一個IOC注解生成器,有點類似于ButterKnife的插件一樣自動給我們生成代碼,在網(wǎng)上找了很多資料國內(nèi)基本就在HelloWorld階段英遭,也有很多哥們向我反應(yīng)插件的代碼還是有點蒙B照卦。代碼方面能理解就理解爬范,不理解也不強(qiáng)求痪宰,如果你能改一改別人已經(jīng)寫好的插件就最好了行疏,實在不行我們干脆也別折騰了大不了不用匆光,本文章旨在給自己想寫一些奇葩插件的哥們一些引導(dǎo)。
  廢話不多說酿联,請看具體效果:

GIF.gif

  自動生成注解代碼铝阐,跟ButterKnife的插件類似,但是我們自己寫的插件生成的注解代碼更加符合google源碼規(guī)范嵌器,而且是基于我們自己動手打造一套IOC注解框架。當(dāng)然我們可以去參考ButterKnife的插件是怎么寫的,但是我看了一下里面的東西太多了螺男,我們干脆自己來吧干签。

所有分享大綱:2017Android進(jìn)階之路與你同行

視頻講解地址:http://pan.baidu.com/s/1gf40cV5

2.實現(xiàn)


2.1 思路整理

我們先來整理一下思路及刻,要實現(xiàn)這么個插件我們需要做一些什么東東:

  • 獲取光標(biāo)所在行的布局文件 --> R.layout.xxxx.xml岂丘;
  • 搜索整個項目獲取到R.layout.xxxx.xml文件;
  • 通過該布局文件去遍歷找出含有id的布局標(biāo)簽蹲姐,當(dāng)然如果考慮完善一點需要考慮include等等磨取;
  • 遍歷完成后生成對話框人柿,讓用戶可以自己選擇需要生成注解的View以及點擊事件,這個是Java GUI里面的內(nèi)容
  • 最后當(dāng)用戶點擊確定生成最終的注解代碼即可

這么說起來還是挺簡單的忙厌,當(dāng)然其中的細(xì)節(jié)還是讓人很蛋疼的凫岖,需要不斷反復(fù)的調(diào)試。

2.2 具體實現(xiàn)

  • 獲取光標(biāo)所在行的布局文件 --> R.layout.xxxx.xml逢净;
    /**
     * 獲取當(dāng)前光標(biāo)的layout文件
     */
    private String getCurrentLayout(Editor editor) {
        Document document = editor.getDocument();
        CaretModel caretModel = editor.getCaretModel();
        int caretOffset = caretModel.getOffset();
        int lineNum = document.getLineNumber(caretOffset);
        int lineStartOffset = document.getLineStartOffset(lineNum);
        int lineEndOffset = document.getLineEndOffset(lineNum);
        // 獲取當(dāng)前光標(biāo)所在行的所有內(nèi)容
        String lineContent = document.getText(new TextRange(lineStartOffset, lineEndOffset));
        String layoutMatching = "R.layout.";
        if (!TextUtils.isEmpty(lineContent) && lineContent.contains(layoutMatching)) {
            // 獲取layout文件的字符串
            int startPosition = lineContent.indexOf(layoutMatching) + layoutMatching.length();
            int endPosition = lineContent.indexOf(")", startPosition);
            String layoutStr = lineContent.substring(startPosition, endPosition);
            // 可能是另外一種情況 View.inflate
            if (layoutStr.contains(",")) {
                endPosition = lineContent.indexOf(",", startPosition);
                layoutStr = lineContent.substring(startPosition, endPosition);
            }
            return layoutStr;
        }
        return null;
    }
  • 搜索整個項目獲取到R.layout.xxxx.xml文件哥放;
 @Override
    public void actionPerformed(AnActionEvent e) {
        // 獲取project
        Project project = e.getProject();
        // 獲取選中內(nèi)容
        final Editor mEditor = e.getData(PlatformDataKeys.EDITOR);
        if (null == mEditor) {
            return;
        }
        SelectionModel model = mEditor.getSelectionModel();
        mSelectedText = model.getSelectedText();
        // 未選中布局內(nèi)容,顯示dialog
        if (TextUtils.isEmpty(mSelectedText)) {
            // 獲取光標(biāo)所在位置的布局
            mSelectedText = getCurrentLayout(mEditor);
            if (TextUtils.isEmpty(mSelectedText)) {
                mSelectedText = Messages.showInputDialog(project, "布局內(nèi)容:(不需要輸入R.layout.)", "未選中布局內(nèi)容爹土,請輸入layout文件名", Messages.getInformationIcon());
                if (TextUtils.isEmpty(mSelectedText)) {
                    Util.showPopupBalloon(mEditor, "未輸入layout文件名", 5);
                    return;
                }
            }
        }
        // 獲取布局文件甥雕,通過FilenameIndex.getFilesByName獲取
        // GlobalSearchScope.allScope(project)搜索整個項目
        PsiFile[] psiFiles = FilenameIndex.getFilesByName(project, mSelectedText + ".xml", GlobalSearchScope.allScope(project));
        if (psiFiles.length <= 0) {
            Util.showPopupBalloon(mEditor, "未找到選中的布局文件" + mSelectedText, 5);
            return;
        }
        XmlFile xmlFile = (XmlFile) psiFiles[0];
        List<Element> elements = new ArrayList<>();
        Util.getIDsFromLayout(xmlFile, elements);
        // 將代碼寫入文件,不允許在主線程中進(jìn)行實時的文件寫入
        if (elements.size() != 0) {
            PsiFile psiFile = PsiUtilBase.getPsiFileInEditor(mEditor, project);
            PsiClass psiClass = Util.getTargetClass(mEditor, psiFile);
            // 有的話就創(chuàng)建變量和findViewById
            if (mDialog != null && mDialog.isShowing()) {
                mDialog.cancelDialog();
            }
            mDialog = new FindViewByIdDialog(mEditor, project, psiFile, psiClass, elements, mSelectedText);
            mDialog.showDialog();
        } else {
            Util.showPopupBalloon(mEditor, "未找到任何Id", 5);
        }
    }
  • 通過該布局文件去遍歷找出含有id的布局標(biāo)簽胀茵,當(dāng)然如果考慮完善一點需要考慮include等等社露;
 /**
 * 獲取所有id
 *
 * @param file
 * @param elements
 * @return
 */
public static java.util.List<Element> getIDsFromLayout(final PsiFile file, final java.util.List<Element> elements) {
    // To iterate over the elements in a file
    // 遍歷一個文件的所有元素
    file.accept(new XmlRecursiveElementVisitor() {
        @Override
        public void visitElement(PsiElement element) {
            super.visitElement(element);
            // 解析Xml標(biāo)簽
            if (element instanceof XmlTag) {
                XmlTag tag = (XmlTag) element;
                // 獲取Tag的名字(TextView)或者自定義
                String name = tag.getName();
                // 如果有include
                if (name.equalsIgnoreCase("include")) {
                    // 獲取布局
                    XmlAttribute layout = tag.getAttribute("layout", null);
                    // 獲取project
                    Project project = file.getProject();
                    // 布局文件
                    XmlFile include = null;
                    PsiFile[] psiFiles = FilenameIndex.getFilesByName(project, getLayoutName(layout.getValue()) + ".xml", GlobalSearchScope.allScope(project));
                    if (psiFiles.length > 0) {
                        include = (XmlFile) psiFiles[0];
                    }
                    if (include != null) {
                        // 遞歸
                        getIDsFromLayout(include, elements);
                        return;
                    }
                }
                // 獲取id字段屬性
                XmlAttribute id = tag.getAttribute("android:id", null);
                if (id == null) {
                    return;
                }
                // 獲取id的值
                String idValue = id.getValue();
                if (idValue == null) {
                    return;
                }
                XmlAttribute aClass = tag.getAttribute("class", null);
                if (aClass != null) {
                    name = aClass.getValue();
                }
                // 添加到list
                try {
                    Element e = new Element(name, idValue,  tag);
                    elements.add(e);
                } catch (IllegalArgumentException e) {

                }
            }
        }
    });
    return elements;
}
  • 遍歷完成后生成對話框,讓用戶可以自己選擇需要生成注解的View以及點擊事件琼娘,這個是Java GUI里面的內(nèi)容峭弟,我就大致黏貼一部分代碼把,把對話框中間的部分黏貼出來脱拼,具體可以去github下載源碼
    /**
     * 解析mElements瞒瘸,并添加到JPanel
     */
    private void initContentPanel() {
        mContentJPanel.removeAll();
        // 設(shè)置內(nèi)容
        for (int i = 0; i < mElements.size(); i++) {
            Element mElement = mElements.get(i);
            IdBean itemJPanel = new IdBean(new GridLayout(1, 4, 10, 10),
                    new EmptyBorder(5, 10, 5, 10),
                    new JCheckBox(mElement.getName()),
                    new JLabel(mElement.getId()),
                    new JCheckBox(),
                    new JTextField(mElement.getFieldName()),
                    mElement);
            // 監(jiān)聽
            itemJPanel.setEnableActionListener(this);
            itemJPanel.setClickActionListener(clickCheckBox -> mElement.setIsCreateClickMethod(clickCheckBox.isSelected()));
            itemJPanel.setFieldFocusListener(fieldJTextField -> mElement.setFieldName(fieldJTextField.getText()));
            mContentJPanel.add(itemJPanel);
            mContentConstraints.fill = GridBagConstraints.HORIZONTAL;
            mContentConstraints.gridwidth = 0;
            mContentConstraints.gridx = 0;
            mContentConstraints.gridy = i;
            mContentConstraints.weightx = 1;
            mContentLayout.setConstraints(itemJPanel, mContentConstraints);
        }
        mContentJPanel.setLayout(mContentLayout);
        jScrollPane = new JBScrollPane(mContentJPanel);
        jScrollPane.revalidate();
        // 添加到JFrame
        getContentPane().add(jScrollPane, 1);
    }
  • 最后當(dāng)用戶點擊確定生成最終的注解代碼即可,主要生成兩部分代碼@ViewById(R.id.xxx) , @OnClick(R.id.xxx)即可
    /**
     * 創(chuàng)建注解View變量
     */
    private void generateFields() {
        for (Element element : mElements) {
            if (mClass.getText().contains("@ViewById(" + element.getFullID() + ")")) {
                // 不創(chuàng)建新的變量
                continue;
            }
            // 設(shè)置變量名熄浓,獲取text里面的內(nèi)容
            String text = element.getXml().getAttributeValue("android:text");
            if (TextUtils.isEmpty(text)) {
                // 如果是text為空情臭,則獲取hint里面的內(nèi)容
                text = element.getXml().getAttributeValue("android:hint");
            }
            // 如果是@string/app_name類似
            if (!TextUtils.isEmpty(text) && text.contains("@string/")) {
                text = text.replace("@string/", "");
                // 獲取strings.xml
                PsiFile[] psiFiles = FilenameIndex.getFilesByName(mProject, "strings.xml", GlobalSearchScope.allScope(mProject));
                if (psiFiles.length > 0) {
                    for (PsiFile psiFile : psiFiles) {
                        // 獲取src\main\res\values下面的strings.xml文件
                        String dirName = psiFile.getParent().toString();
                        if (dirName.contains("src\\main\\res\\values")) {
                            text = Util.getTextFromStringsXml(psiFile, text);
                        }
                    }
                }
            }

            StringBuilder fromText = new StringBuilder();
            if (!TextUtils.isEmpty(text)) {
                fromText.append("/****" + text + "****/\n");
            }
            fromText.append("@ViewById(" + element.getFullID() + ")\n");
            fromText.append("private ");
            fromText.append(element.getName());
            fromText.append(" ");
            fromText.append(element.getFieldName());
            fromText.append(";");
            // 創(chuàng)建點擊方法
            if (element.isCreateFiled()) {
                // 添加到class
                mClass.add(mFactory.createFieldFromText(fromText.toString(), mClass));
            }
        }
    }

    /**
     * 創(chuàng)建OnClick方法
     */
    private void generateOnClickMethod() {
        for (Element element : mElements) {
            // 可以使用并且可以點擊
            if (element.isCreateClickMethod()) {
                // 需要創(chuàng)建OnClick方法
                String methodName = getClickMethodName(element) + "Click";
                PsiMethod[] onClickMethods = mClass.findMethodsByName(methodName, true);
                boolean clickMethodExist = onClickMethods.length > 0;
                if (!clickMethodExist) {
                    // 創(chuàng)建點擊方法
                    createClickMethod(methodName, element);
                }
            }
        }
    }

    /**
     * 創(chuàng)建一個點擊事件
     */
    private void createClickMethod(String methodName, Element element) {
        // 拼接方法的字符串
        StringBuilder methodBuilder = new StringBuilder();
        methodBuilder.append("@OnClick(" + element.getFullID() + ")\n");
        methodBuilder.append("private void " + methodName + "(" + element.getName() + " " + getClickMethodName(element) + "){");
        methodBuilder.append("\n}");
        // 創(chuàng)建OnClick方法
        mClass.add(mFactory.createMethodFromText(methodBuilder.toString(), mClass));
    }

    /**
     * 獲取點擊方法的名稱
     */
    public String getClickMethodName(Element element) {
        String[] names = element.getId().split("_");
        // aaBbCc
        StringBuilder sb = new StringBuilder();
        for (int i = 0; i < names.length; i++) {
            if (i == 0) {
                sb.append(names[i]);
            } else {
                sb.append(Util.firstToUpperCase(names[i]));
            }
        }
        return sb.toString();
    }

如果在公司的時候比較閑比如說我,那么還是可以去了解一下插件開發(fā)的赌蔑,我們可以利用它去生成代碼或者修改代碼找沒有用到的資源等等等等俯在,還是蠻不錯的,如果是天天加班還是應(yīng)該考慮一下學(xué)習(xí)成本娃惯,因為有些地方剛接觸還是容易蒙B朝巫。

附上源碼地址:https://github.com/Shenmowen/DarrenIOC

所有分享大綱:2017Android進(jìn)階之路與你同行

視頻講解地址:http://pan.baidu.com/s/1gf40cV5

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市石景,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌拙吉,老刑警劉巖潮孽,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異筷黔,居然都是意外死亡往史,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進(jìn)店門佛舱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來椎例,“玉大人挨决,你說我怎么就攤上這事《┩幔” “怎么了脖祈?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長刷晋。 經(jīng)常有香客問我盖高,道長,這世上最難降的妖魔是什么眼虱? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任喻奥,我火速辦了婚禮,結(jié)果婚禮上捏悬,老公的妹妹穿的比我還像新娘撞蚕。我一直安慰自己,他們只是感情好过牙,可當(dāng)我...
    茶點故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布甥厦。 她就那樣靜靜地躺著,像睡著了一般抒和。 火紅的嫁衣襯著肌膚如雪矫渔。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天摧莽,我揣著相機(jī)與錄音庙洼,去河邊找鬼。 笑死镊辕,一個胖子當(dāng)著我的面吹牛油够,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播征懈,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼石咬,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了卖哎?” 一聲冷哼從身側(cè)響起鬼悠,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎亏娜,沒想到半個月后焕窝,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡维贺,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年它掂,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片溯泣。...
    茶點故事閱讀 39,690評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡虐秋,死狀恐怖榕茧,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情客给,我是刑警寧澤用押,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站起愈,受9級特大地震影響只恨,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜抬虽,卻給世界環(huán)境...
    茶點故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一官觅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧阐污,春花似錦休涤、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至手幢,卻和暖如春捷凄,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背围来。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工跺涤, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人监透。 一個月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓桶错,卻偏偏與公主長得像,于是被迫代替她去往敵國和親胀蛮。 傳聞我的和親對象是個殘疾皇子院刁,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,577評論 2 353

推薦閱讀更多精彩內(nèi)容