• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            逛奔的蝸牛

            我不聰明,但我會很努力

               ::  :: 新隨筆 ::  ::  :: 管理 ::

            寫自己的文本編輯器(): 高亮關鍵字

            . 高亮的內容:

            需要高亮的內容有:

            1. 關鍵字, public, int, true

            2. 運算符, +, -, *, /

            3. 數字

            4. 高亮字符串, "example of string"

            5. 高亮單行注釋

            6. 高亮多行注釋


            . 實現高亮的核心方法:

            StyledDocument.setCharacterAttributes(int offset, int length, AttributeSet s, boolean replace) 


            . 文本編輯器選擇.

            Java中提供的多行文本編輯器有: JTextComponent, JTextArea, JTextPane, JEditorPane, 都可以使用. 但是因為語法著色中文本要使用多種風格的樣式, 所以這些文本編輯器的document要使用StyledDocument. 

            JTextArea使用的是PlainDocument, document不能進行多種格式的著色.

            JTextPane, JEditorPane使用的是StyledDocument, 默認就可以使用

            為了實現語法著色, 可以繼承自DefaultStyledDocument, 設置其為這些文本編輯器的documet, 或者也可以直接使用JTextPane, JEditorPane來做. 為了方便, 這里就直接使用JTextPane.


            . 何時進行著色.

            當文本編輯器中有字符被插入或者刪除時, 文本的內容就發生了變化, 這時檢查, 進行著色.

            為了監視到文本的內容發生了變化, 要給document添加一個DocumentListener監聽器, 在他的removeUpdateinsertUpdate中進行著色處理.

            changedUpdate方法在文本的屬性例如前景色, 背景色, 字體等風格改變時才會被調用.

            @Override

            public void changedUpdate(DocumentEvent e) {


            }


            @Override

            public void insertUpdate(DocumentEvent e) {

            try {

            colouring((StyledDocument) e.getDocument(), e.getOffset(), e.getLength());

            } catch (BadLocationException e1) {

            e1.printStackTrace();

            }

            }


            @Override

            public void removeUpdate(DocumentEvent e) {

            try {

            // 因為刪除后光標緊接著影響的單詞兩邊, 所以長度就不需要了

            colouring((StyledDocument) e.getDocument(), e.getOffset(), 0);

            } catch (BadLocationException e1) {

            e1.printStackTrace();

            }

            }


            . 著色范圍

            pos: 指變化前光標的位置.

            len: 指變化的字符數.

            例如有關鍵字public, int

            單詞"publicint", "public""int"中插入一個空格后變成"public int", 一個單詞變成了兩個, 這時對"public" "int"進行著色.

            著色范圍是publicp的位置和intt的位置加1, 即是pos前面單詞開始的下標和pos+len開始單詞結束的下標. 所以上例中要著色的范圍是"public int". 

            提供了方法indexOfWordStart來取得pos前單詞開始的下標, 方法indexOfWordEnd來取得pos后單詞結束的下標.

            public int indexOfWordStart(Document doc, int pos) throws BadLocationException {

            // pos開始向前找到第一個非單詞字符.

            for (; pos > 0 && isWordCharacter(doc, pos - 1); --pos);

            return pos;

            }


            public int indexOfWordEnd(Document doc, int pos) throws BadLocationException {

            // pos開始向前找到第一個非單詞字符.

            for (; isWordCharacter(doc, pos); ++pos);

            return pos;

            }



            一個字符是單詞的有效字符: 是字母, 數字, 下劃線.

            public boolean isWordCharacter(Document doc, int pos) throws BadLocationException {

            char ch = getCharAt(doc, pos); // 取得在文檔中pos位置處的字符

            if (Character.isLetter(ch) || Character.isDigit(ch) || ch == '_') { return true; }

            return false;

            }


            所以著色的范圍是[start, end] :

            int start = indexOfWordStart(doc, pos);

            int end = indexOfWordEnd(doc, pos + len);


            . 關鍵字著色.

            從著色范圍的開始下標起進行判斷, 如果是以字母開或者下劃線開頭, 則說明是單詞, 那么先取得這個單詞, 如果這個單詞是關鍵字, 就進行關鍵字著色, 如果不是, 就進行普通的著色. 著色完這個單詞后, 繼續后面的著色處理. 已經著色過的字符, 就不再進行著色了.

            public void colouring(StyledDocument doc, int pos, int len) throws BadLocationException {

            // 取得插入或者刪除后影響到的單詞.

            // 例如"public"b后插入一個空格, 就變成了:"pub lic", 這時就有兩個單詞要處理:"pub""lic"

            // 這時要取得的范圍是pubp前面的位置和licc后面的位置

            int start = indexOfWordStart(doc, pos);

            int end = indexOfWordEnd(doc, pos + len);


            char ch;

            while (start < end) {

            ch = getCharAt(doc, start);

            if (Character.isLetter(ch) || ch == '_') {

            // 如果是以字母或者下劃線開頭, 說明是單詞

            // pos為處理后的最后一個下標

            start = colouringWord(doc, start);

            } else {

            //SwingUtilities.invokeLater(new ColouringTask(doc, pos, wordEnd - pos, normalStyle));

            ++start;

            }

            }

            }



            public int colouringWord(StyledDocument doc, int pos) throws BadLocationException {

            int wordEnd = indexOfWordEnd(doc, pos);

            String word = doc.getText(pos, wordEnd - pos); // 要進行著色的單詞


            if (keywords.contains(word)) {

            // 如果是關鍵字, 就進行關鍵字的著色, 否則使用普通的著色.

            // 這里有一點要注意, insertUpdateremoveUpdate的方法調用的過程中, 不能修改doc的屬性.

            // 但我們又要達到能夠修改doc的屬性, 所以把此任務放到這個方法的外面去執行.

            // 實現這一目的, 可以使用新線程, 但放到swing的事件隊列里去處理更輕便一點.

            SwingUtilities.invokeLater(new ColouringTask(doc, pos, wordEnd - pos, keywordStyle));

            } else {

            SwingUtilities.invokeLater(new ColouringTask(doc, pos, wordEnd - pos, normalStyle));

            }


            return wordEnd;

            }


            因為在insertUpdateremoveUpdate方法中不能修改document的屬性, 所以著色的任務放到這兩個方法外面, 所以使用了SwingUtilities.invokeLater來實現.

            private class ColouringTask implements Runnable {

            private StyledDocument doc;

            private Style style;

            private int pos;

            private int len;


            public ColouringTask(StyledDocument doc, int pos, int len, Style style) {

            this.doc = doc;

            this.pos = pos;

            this.len = len;

            this.style = style;

            }


            public void run() {

            try {

            // 這里就是對字符進行著色

            doc.setCharacterAttributes(pos, len, style, true);

            } catch (Exception e) {}

            }

            }



            : 源碼

            關鍵字著色的完成代碼如下, 可以直接編譯運行. 對于數字, 運算符, 字符串等的著色處理在以后的教程中會繼續進行詳解.

            import java.awt.Color;

            import java.util.HashSet;

            import java.util.Set;


            import javax.swing.JFrame;

            import javax.swing.JTextPane;

            import javax.swing.SwingUtilities;

            import javax.swing.event.DocumentEvent;

            import javax.swing.event.DocumentListener;

            import javax.swing.text.BadLocationException;

            import javax.swing.text.Document;

            import javax.swing.text.Style;

            import javax.swing.text.StyleConstants;

            import javax.swing.text.StyledDocument;


            public class HighlightKeywordsDemo {

            public static void main(String[] args) {

            JFrame frame = new JFrame();


            JTextPane editor = new JTextPane();

            editor.getDocument().addDocumentListener(new SyntaxHighlighter(editor));

            frame.getContentPane().add(editor);


            frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

            frame.setSize(500, 500);

            frame.setVisible(true);

            }

            }


            /**

             * 當文本輸入區的有字符插入或者刪除時, 進行高亮.

             * 

             * 要進行語法高亮, 文本輸入組件的document要是styled document才行. 所以不要用JTextArea. 可以使用JTextPane.

             * 

             * @author Biao

             * 

             */

            class SyntaxHighlighter implements DocumentListener {

            private Set<String> keywords;

            private Style keywordStyle;

            private Style normalStyle;


            public SyntaxHighlighter(JTextPane editor) {

            // 準備著色使用的樣式

            keywordStyle = ((StyledDocument) editor.getDocument()).addStyle("Keyword_Style", null);

            normalStyle = ((StyledDocument) editor.getDocument()).addStyle("Keyword_Style", null);

            StyleConstants.setForeground(keywordStyle, Color.RED);

            StyleConstants.setForeground(normalStyle, Color.BLACK);


            // 準備關鍵字

            keywords = new HashSet<String>();

            keywords.add("public");

            keywords.add("protected");

            keywords.add("private");

            keywords.add("_int9");

            keywords.add("float");

            keywords.add("double");

            }


            public void colouring(StyledDocument doc, int pos, int len) throws BadLocationException {

            // 取得插入或者刪除后影響到的單詞.

            // 例如"public"b后插入一個空格, 就變成了:"pub lic", 這時就有兩個單詞要處理:"pub""lic"

            // 這時要取得的范圍是pubp前面的位置和licc后面的位置

            int start = indexOfWordStart(doc, pos);

            int end = indexOfWordEnd(doc, pos + len);


            char ch;

            while (start < end) {

            ch = getCharAt(doc, start);

            if (Character.isLetter(ch) || ch == '_') {

            // 如果是以字母或者下劃線開頭, 說明是單詞

            // pos為處理后的最后一個下標

            start = colouringWord(doc, start);

            } else {

            SwingUtilities.invokeLater(new ColouringTask(doc, start, 1, normalStyle));

            ++start;

            }

            }

            }


            /**

            * 對單詞進行著色, 并返回單詞結束的下標.

            * @param doc

            * @param pos

            * @return

            * @throws BadLocationException

            */

            public int colouringWord(StyledDocument doc, int pos) throws BadLocationException {

            int wordEnd = indexOfWordEnd(doc, pos);

            String word = doc.getText(pos, wordEnd - pos);


            if (keywords.contains(word)) {

            // 如果是關鍵字, 就進行關鍵字的著色, 否則使用普通的著色.

            // 這里有一點要注意, insertUpdateremoveUpdate的方法調用的過程中, 不能修改doc的屬性.

            // 但我們又要達到能夠修改doc的屬性, 所以把此任務放到這個方法的外面去執行.

            // 實現這一目的, 可以使用新線程, 但放到swing的事件隊列里去處理更輕便一點.

            SwingUtilities.invokeLater(new ColouringTask(doc, pos, wordEnd - pos, keywordStyle));

            } else {

            SwingUtilities.invokeLater(new ColouringTask(doc, pos, wordEnd - pos, normalStyle));

            }


            return wordEnd;

            }


            /**

            * 取得在文檔中下標在pos處的字符.

            * 如果posdoc.getLength(), 返回的是一個文檔的結束符, 不會拋出異常. 如果pos<0, 則會拋出異常.

            * 所以pos的有效值是[0, doc.getLength()]

            * @param doc

            * @param pos

            * @return

            * @throws BadLocationException

            */

            public char getCharAt(Document doc, int pos) throws BadLocationException {

            return doc.getText(pos, 1).charAt(0);

            }


            /**

            * 取得下標為pos, 它所在的單詞開始的下標. ±wor^d± (^表示pos, ±表示開始或結束的下標)

            * @param doc

            * @param pos

            * @return

            * @throws BadLocationException

            */

            public int indexOfWordStart(Document doc, int pos) throws BadLocationException {

            // pos開始向前找到第一個非單詞字符.

            for (; pos > 0 && isWordCharacter(doc, pos - 1); --pos);


            return pos;

            }


            /**

            * 取得下標為pos, 它所在的單詞結束的下標. ±wor^d± (^表示pos, ±表示開始或結束的下標)

            * @param doc

            * @param pos

            * @return

            * @throws BadLocationException

            */

            public int indexOfWordEnd(Document doc, int pos) throws BadLocationException {

            // pos開始向前找到第一個非單詞字符.

            for (; isWordCharacter(doc, pos); ++pos);


            return pos;

            }


            /**

            * 如果一個字符是字母, 數字, 下劃線, 則返回true.

            * @param doc

            * @param pos

            * @return

            * @throws BadLocationException

            */

            public boolean isWordCharacter(Document doc, int pos) throws BadLocationException {

            char ch = getCharAt(doc, pos);

            if (Character.isLetter(ch) || Character.isDigit(ch) || ch == '_') { return true; }

            return false;

            }


            @Override

            public void changedUpdate(DocumentEvent e) {


            }


            @Override

            public void insertUpdate(DocumentEvent e) {

            try {

            colouring((StyledDocument) e.getDocument(), e.getOffset(), e.getLength());

            } catch (BadLocationException e1) {

            e1.printStackTrace();

            }

            }


            @Override

            public void removeUpdate(DocumentEvent e) {

            try {

            // 因為刪除后光標緊接著影響的單詞兩邊, 所以長度就不需要了

            colouring((StyledDocument) e.getDocument(), e.getOffset(), 0);

            } catch (BadLocationException e1) {

            e1.printStackTrace();

            }

            }


            /**

            * 完成著色任務

            * @author Biao

            */

            private class ColouringTask implements Runnable {

            private StyledDocument doc;

            private Style style;

            private int pos;

            private int len;


            public ColouringTask(StyledDocument doc, int pos, int len, Style style) {

            this.doc = doc;

            this.pos = pos;

            this.len = len;

            this.style = style;

            }


            public void run() {

            try {

            // 這里就是對字符進行著色

            doc.setCharacterAttributes(pos, len, style, true);

            } catch (Exception e) {}

            }

            }

            }

            posted on 2010-02-05 19:00 逛奔的蝸牛 閱讀(9188) 評論(0)  編輯 收藏 引用 所屬分類: Java
            久久精品国产亚洲av瑜伽| 麻豆国内精品久久久久久| 狠狠色丁香久久婷婷综合_中| 久久午夜无码鲁丝片午夜精品| 亚洲国产日韩欧美综合久久| 久久99精品久久久大学生| 久久精品无码午夜福利理论片| 亚洲国产二区三区久久| 偷偷做久久久久网站| 26uuu久久五月天| 亚洲精品午夜国产VA久久成人| 青青草国产精品久久| 欧美亚洲国产精品久久高清| 久久精品一区二区国产| 偷窥少妇久久久久久久久| 91久久成人免费| 久久精品水蜜桃av综合天堂| 亚洲国产精品成人久久蜜臀| 久久亚洲国产午夜精品理论片| 久久亚洲熟女cc98cm| 久久人妻少妇嫩草AV蜜桃| 国产亚洲欧美成人久久片| 无码国内精品久久人妻| 久久99九九国产免费看小说| 精品久久久久久无码人妻热| 精品久久久久久无码专区| 亚洲欧美日韩中文久久| 香港aa三级久久三级老师2021国产三级精品三级在 | 久久嫩草影院免费看夜色| 久久青青草原综合伊人| 国产精品久久永久免费| 精品久久久久久无码中文字幕一区| 久久久黄色大片| 色综合久久夜色精品国产| 国产激情久久久久影院老熟女| 99国产精品久久久久久久成人热| 伊人久久久AV老熟妇色| 中文字幕久久精品无码| 久久久久久午夜成人影院| 久久青青草原精品国产| 国产V综合V亚洲欧美久久|