Swing實(shí)現(xiàn)Java代碼編輯器 - 語(yǔ)法高亮顯示

當(dāng)JEditorPane被創(chuàng)建時(shí),它會(huì)把createDefaultEditorKit()方法(javax.swing.text.EditorKit的子類(lèi)對(duì)象)的返回值作為默認(rèn)的編輯器工具包,然后將文本的編輯與顯示工作交給這個(gè)工具包。其原型為:

2

3

4

EditorKit基本上什么也沒(méi)有做,只是提供了很多抽象方法給它的子類(lèi)去實(shí)現(xiàn),Swing默認(rèn)已經(jīng)給它添加了一個(gè)子類(lèi)DefaultEditorKit(Swing常用的一招,就是給抽象類(lèi)前面加個(gè)Default進(jìn)行最基本的實(shí)現(xiàn)),既然是Default,那它所提供的功能肯定和一個(gè)記事本沒(méi)有多大區(qū)別,這要是繼承下來(lái),有多少方法需要覆蓋啊,別慌,查看一下Swing的源碼,你會(huì)發(fā)現(xiàn)Swing還提供了一個(gè)繼承自DefaultEditorKit的類(lèi)StyledEditorKit,顧名思義,這個(gè)類(lèi)肯定為我們提供了很多支持格式化顯示的方法,又是一個(gè)巨人,快,趕緊拉過(guò)來(lái)往肩上爬。
接下來(lái)就是覆蓋StyledEditorKit中的相關(guān)方法了,其實(shí)有很多方法都可以覆蓋,但是意義不是很大,比如
EditorKit中有兩個(gè)重要的方法實(shí)現(xiàn)對(duì)文檔的管理與顯示:
創(chuàng)建一個(gè)適合此編輯器類(lèi)型文本存儲(chǔ)模型。EditorKit把對(duì)文本文檔的管理功能交給了這個(gè)方法的返回值。
獲取適合生成此工具包生成的任何模型視圖的工廠(chǎng)。EditorKit把編輯器的顯示功能交給了這個(gè)方法的返回值,比如什么字符顯示成什么樣子,什么顏色等。我們必須覆蓋這兩個(gè)方法以實(shí)現(xiàn)自定義編輯器的功能。
因?yàn)槲覀兊木庉嬈骱蚃EditorPane唯一不同的可能就是代碼怎么來(lái)顯示,所以createDefaultDocument()可以返回一個(gè)默認(rèn)的javax.swing.text.DefaultStyledDocument 就行,對(duì)于getViewFactory,我們需要定制一個(gè)ViewFactory視圖來(lái)實(shí)現(xiàn)編輯器獨(dú)有的各種顯示功能。
ViewFactory在Java中被定義為一個(gè)接口,里面提供了唯一的一個(gè)方法:
這個(gè)方法根據(jù)給定的文檔的結(jié)構(gòu)化元素創(chuàng)建一個(gè)視圖。在這個(gè)方法中,我們只需要返回一個(gè)繼承自View的視圖即可,真正的顯示任務(wù)是交給這個(gè)視圖的。因此,我們的ViewFactory類(lèi)很簡(jiǎn)單:

2

3

4

5

6

7

8

9

10

11

12

接下來(lái)的重點(diǎn)就是這個(gè)JavaEditorView了,所有的語(yǔ)法高亮等顯示功能都是交給它來(lái)完成的
View是一個(gè)抽象類(lèi),Swing默認(rèn)給我們提供了多個(gè)它的子類(lèi),AsyncBoxView, ComponentView, CompositeView, GlyphView, IconView, ImageView, PlainView 以實(shí)現(xiàn)對(duì)不同文檔類(lèi)型的顯示,當(dāng)中只有PlainView是與文本文檔相關(guān)的,它實(shí)現(xiàn)簡(jiǎn)單的多行文本視圖的 View 接口,該文本視圖的文本只有一種字體和顏色,沒(méi)錯(cuò),我們的JavaEditorView需要繼承自PlainView。
PlainView提供了很多方法進(jìn)行文本文檔的視圖顯示,要實(shí)現(xiàn)高亮顯示,我們關(guān)心的有兩個(gè)方法:
一看名字就知道這個(gè)方法是控制選中狀態(tài)下的顯示方式,由于本文只討論非選中狀態(tài)。所以重點(diǎn)看一下另外一個(gè)方法:
這個(gè)方法將模型中給定的范圍呈現(xiàn)為正常的未選定文本。使用前景色或指定的顏色顯示文本。
參數(shù):
g - 圖形上下文(做Swing的人再熟悉不過(guò)了,文本也是畫(huà)出來(lái)的)
x - 起始 X 坐標(biāo),該值 >= 0
y - 起始 Y 坐標(biāo),該值 >= 0
p0 - 模型中的起始位置,該值 >= 0
p1 - 模型中的結(jié)束位置,該值 >= 0
下面是覆蓋后的實(shí)現(xiàn):

2

3

4

5

6

7

8

先是得到從起始位置到結(jié)束位置的長(zhǎng)度,然后再交由scanParagraph去處理指定長(zhǎng)度的文本,其實(shí)也就是怎么把它畫(huà)出來(lái)。
對(duì)于一個(gè)Java代碼編輯器,要考慮類(lèi)名,運(yùn)算符,數(shù)字,關(guān)鍵字等的顯示方式,所以scanParagraph要做的事情很多,本文只以怎么么高亮顯示類(lèi)名為例來(lái)說(shuō)明:

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26



27

28

29

30



31

我只是以類(lèi)名進(jìn)行示范,實(shí)際中可能還要考慮此類(lèi)名是否在注釋當(dāng)中等等...
posted on 2010-02-10 14:27 凱子 閱讀(3146) 評(píng)論(9) 編輯 收藏 所屬分類(lèi): Swing