青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

逛奔的蝸牛

我不聰明,但我會很努力

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

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

. 高亮的內容:

需要高亮的內容有:

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 逛奔的蝸牛 閱讀(9216) 評論(0)  編輯 收藏 引用 所屬分類: Java
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久亚洲电影| 在线亚洲成人| 久久视频在线看| 欧美在线免费一级片| 国产日韩欧美一区二区三区在线观看| 欧美一区二区三区日韩| 亚洲欧美国产日韩天堂区| 国产欧美一区二区精品性| 久久九九热免费视频| 久久精品国产亚洲精品| 亚洲第一在线综合网站| 亚洲国产精品成人综合| 欧美精品性视频| 亚洲欧美中文日韩v在线观看| 午夜亚洲视频| 亚洲国产精品久久久久婷婷老年| 亚洲国产一区二区三区高清| 国产精品国产三级国产a| 久久九九99视频| 欧美国产综合一区二区| 午夜久久福利| 免费欧美日韩| 亚洲欧美日韩系列| 久久这里只有精品视频首页| 日韩午夜高潮| 欧美一区二区免费观在线| 亚洲国产精品免费| 亚洲男人的天堂在线| 在线观看成人小视频| 一区二区三区日韩在线观看| 激情综合亚洲| 亚洲网站在线| 亚洲精品美女久久7777777| 亚洲影院高清在线| 亚洲精品乱码久久久久| 亚洲欧美日韩久久精品| 日韩手机在线导航| 久久免费视频在线观看| 午夜精品亚洲一区二区三区嫩草| 女人天堂亚洲aⅴ在线观看| 亚洲欧美日韩网| 欧美日韩蜜桃| 欧美激情精品久久久久久黑人| 国产精品私人影院| 亚洲精品少妇| 亚洲激情av| 久久久久久电影| 久久狠狠婷婷| 国产精品一区久久久| 99re在线精品| 日韩一级精品| 欧美精品一线| 亚洲高清二区| 91久久夜色精品国产九色| 欧美一级在线视频| 久久爱www久久做| 国产精品成人久久久久| 亚洲美女视频| 国产精品99久久久久久久vr| 欧美精品成人一区二区在线观看 | 久久久国际精品| 久久精品视频在线看| 国产一区二区三区av电影 | 久久成人精品视频| 国产精品影视天天线| 亚洲视频碰碰| 欧美一区二区三区视频免费| 欧美午夜不卡| 亚洲午夜性刺激影院| 午夜激情一区| 国产曰批免费观看久久久| 亚洲欧美日韩在线播放| 欧美综合77777色婷婷| 狠狠入ady亚洲精品| 欧美日韩国产三级| 亚洲韩国日本中文字幕| 亚洲国产婷婷| 欧美激情视频一区二区三区免费| 亚洲高清不卡av| 亚洲最新色图| 国产精品视频精品视频| 亚洲欧美在线aaa| 久久在线播放| 亚洲精品在线一区二区| 欧美日韩免费一区二区三区| 在线亚洲一区二区| 欧美一区午夜视频在线观看| 国模 一区 二区 三区| 久久亚洲综合网| 日韩视频在线观看免费| 欧美亚洲一区二区三区| 亚洲第一色中文字幕| 香蕉成人啪国产精品视频综合网| 亚洲午夜久久久| 国产欧美日韩精品a在线观看| 欧美在线中文字幕| 亚洲电影中文字幕| 亚洲一区二区动漫| 国内一区二区在线视频观看| 免费av成人在线| 亚洲在线成人精品| 欧美激情在线观看| 午夜精品福利视频| 亚洲电影视频在线| 国产精品久久久久久久7电影| 久久精品女人天堂| 一区二区欧美视频| 欧美成人第一页| 欧美呦呦网站| 一二三四社区欧美黄| 国产一区二区三区久久| 欧美日韩视频在线观看一区二区三区| 欧美一区二区三区四区夜夜大片| 亚洲国产日韩一区| 久久久综合网| 亚洲欧美激情视频| 亚洲精品视频在线观看免费| 国产精品影视天天线| 欧美精品在欧美一区二区少妇| 亚洲欧美在线磁力| av成人天堂| 亚洲高清在线精品| 久久婷婷国产麻豆91天堂| 在线一区二区三区四区五区| 伊人成人在线视频| 国产精品一区二区三区四区| 欧美另类在线播放| 嫩草国产精品入口| 久久九九国产精品| 西西人体一区二区| 亚洲欧美国产视频| 亚洲天堂偷拍| 99re热这里只有精品免费视频| 亚洲福利视频二区| 男同欧美伦乱| 久久综合色播五月| 久久久久久久久蜜桃| 小辣椒精品导航| 亚洲伊人网站| 一本到高清视频免费精品| 亚洲三级网站| 亚洲欧洲日产国码二区| 亚洲高清视频在线观看| 伊人男人综合视频网| 国产综合久久久久久| 国产欧美日韩在线| 国产欧美一区在线| 国产一区二区剧情av在线| 国产精品视频精品视频| 国产伦精品一区二区三区视频黑人 | 欧美特黄一级| 国产精品家庭影院| 国产精品视频一区二区高潮| 欧美性猛片xxxx免费看久爱 | 这里只有精品电影| 一区二区三区日韩欧美| 亚洲一区视频| 欧美一区二区三区的| 欧美一区二区三区在线| 久久精品人人做人人爽电影蜜月| 久久三级视频| 欧美成人自拍| 欧美午夜视频| 国产一本一道久久香蕉| 黄色成人小视频| 亚洲黄色一区| 亚洲一区视频在线观看视频| 久久国产精品电影| 久久午夜精品一区二区| 欧美激情自拍| 亚洲社区在线观看| 久久精品亚洲乱码伦伦中文| 欧美大片免费观看| 国产精品久久久久久av下载红粉| 国产欧美一级| 亚洲毛片在线看| 久久精品视频网| 亚洲二区视频| 午夜精彩视频在线观看不卡| 久久精品在线观看| 欧美日韩国产一区二区三区| 国产欧美精品在线| 亚洲激情婷婷| 欧美中文字幕第一页| 欧美韩日精品| 亚洲欧美日韩精品久久久久| 噜噜噜噜噜久久久久久91| 国产精品成人观看视频免费| 亚洲大片精品永久免费| 午夜精品亚洲| 亚洲精品免费一二三区| 香蕉av777xxx色综合一区| 欧美高清在线观看| 国产在线精品一区二区中文 | 国内精品嫩模av私拍在线观看 | 亚洲在线播放| 亚洲国产女人aaa毛片在线| 欧美有码视频| 国产精品久久久久国产a级| 亚洲乱码国产乱码精品精可以看 |