• <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 逛奔的蝸牛 閱讀(9189) 評論(0)  編輯 收藏 引用 所屬分類: Java
            国产高潮久久免费观看| 久久精品国产亚洲av麻豆色欲| 久久久久久无码Av成人影院| 伊人久久综合无码成人网| 亚洲精品白浆高清久久久久久| 久久精品99久久香蕉国产色戒| 久久国产成人精品麻豆| 久久亚洲电影| 国产亚洲精品自在久久| 很黄很污的网站久久mimi色| 精品久久久久久久久免费影院| 久久青青草原亚洲av无码app| 日本免费一区二区久久人人澡| 伊人伊成久久人综合网777| 久久国产精品成人影院| 亚洲欧美一级久久精品| 久久精品国产亚洲精品2020| 久久久久人妻一区精品| av国内精品久久久久影院| 思思久久好好热精品国产| 99久久久精品| 色婷婷综合久久久久中文| 国产精品免费久久| 国产精品久久国产精麻豆99网站 | 久久夜色精品国产噜噜麻豆| 久久久久国产视频电影| 久久99精品国产麻豆| 亚洲AV无一区二区三区久久| 亚洲另类欧美综合久久图片区| 久久亚洲国产欧洲精品一| 久久久久久国产精品免费无码 | 国产精品一区二区久久国产| 一本色综合久久| 久久综合伊人77777| 久久青青草原精品国产不卡| 国产成人精品久久亚洲高清不卡 | 99精品国产综合久久久久五月天| 久久精品国产欧美日韩| 很黄很污的网站久久mimi色| 久久国产高清一区二区三区| 久久精品无码一区二区三区日韩 |