Posted on 2008-05-05 13:17
RichardHe 閱讀(548)
評論(0) 編輯 收藏 引用
CEGUI是一個(gè)很流行的GUI開源庫,功能強(qiáng)大,使用方便. 本文針對的是0.2.1版本.此版本目前還沒有內(nèi)嵌對多字節(jié)文字(中文,日語,韓語等)的支持.所以需要修改適當(dāng)?shù)脑创a以達(dá)到目的.
以下以中文舉例,談?wù)勅绾瓮ㄟ^修改適當(dāng)?shù)拇a以增加對中文顯示和輸入的支持,稍后會(huì)放出源代碼.
在開始修改之前,我們需要先了解一下CEGUI是如何處理英文的顯示和輸入的.原理也是很簡單的.
1.準(zhǔn)備一個(gè)貼圖(Texture)來存放字母的位圖,
2.從TTF字體庫中得到每一個(gè)字母的信息,畫到貼圖(Texture)上,并記錄每個(gè)字母對應(yīng)的在貼圖(Texture)上的矩陣
3.顯示的時(shí)候,根據(jù)字母對應(yīng)的矩陣,把貼圖(Texture)中的一塊顯示到屏幕上.
4.輸入: 處理windows的WM_CHAR消息,把鍵值插入CEGUI的系統(tǒng)里.CEGUI會(huì)自動(dòng)處理.
由以上過程不難看出,貼圖(Texture)的大小取決于字母的多少以及字體的大小.而英文字母(包括數(shù)字和符號)只有那么100多個(gè),用一般常用的
10號或者12號字體,貼圖(Texture)并不大.把字母存在貼圖(Texture)上,比直接用GDI的textout等函數(shù)效率高很多.
現(xiàn)在來談?wù)勚形牡娘@示和輸入.本文以簡體中文GB2312為例
GB2312中有7千多個(gè)漢字(包括符號和數(shù)字).如果全部畫到一張貼圖上,以10號字體為例,大概需要一張16M的貼圖來存儲,而這對于目前的硬件水平顯示是不現(xiàn)實(shí)的,那么怎么處理中文呢?也許你也想到了:使用動(dòng)態(tài)存儲.
具體做法:
1. 先準(zhǔn)備一張較大的貼圖, 如512*512,
一個(gè)象素為4個(gè)字節(jié),所以貼圖大小為512*512*4=1M.這種大小的貼圖是可以接受的,但是存儲的漢字?jǐn)?shù)目也是有限的.以本人用的10號字體為例,
可以存放900個(gè)字(包括字母,符號和漢字).而在一般情況下,一次刷新需要顯示的字應(yīng)該不會(huì)超過幾百個(gè)吧.
2.
字母和漢字在貼圖中用相同大小的矩陣(Rect)來表示,這樣方便查找.因?yàn)樽帜甘枪潭ǖ?而且數(shù)目不多,所以字母存在貼圖的前面作為靜態(tài)區(qū),剩下的用來
存儲漢字,這是動(dòng)態(tài)區(qū),當(dāng)要顯示的漢字還沒有畫在貼圖上時(shí),此時(shí)把這個(gè)字畫到貼圖上去.這時(shí)如果貼圖上已經(jīng)畫滿了,則需要先擦除一個(gè)字,騰出一塊地方.
3. 當(dāng)動(dòng)態(tài)更新漢字時(shí),需要一個(gè)算法,保證在一次刷新中,不會(huì)有要顯示的字被擦除掉.這時(shí)需要一個(gè)數(shù)組來保存使用過的漢字的記錄,當(dāng)一次刷新完成后,在把這個(gè)記錄數(shù)組置0.
4.上面解決了顯示的問題,現(xiàn)在來處理輸入的問題.漢字的編碼是雙字節(jié)的,所以只需要在windows的WM_CHAR消息里處理一下就可以了,判斷一
下鍵值是否為漢字,如果是漢字,就把2個(gè)字節(jié)組合成漢字的unicode編碼插入到CEGUI的系統(tǒng)中,否則把一個(gè)字節(jié)直接插入CEGUI的系統(tǒng)中.
OK了,輸入也解決了.
原理就是這樣了,其實(shí)并不復(fù)雜.當(dāng)然還有幾個(gè)細(xì)節(jié)要處理.
比如說,當(dāng)我們輸入"漢"這個(gè)字時(shí),怎么到貼圖
上找到這個(gè)字的位圖的矩陣,所以我們在畫字到貼圖上時(shí),需要建立一個(gè)漢字的unicode碼與其位圖位置的映射表,這樣我們輸入任何一個(gè)漢字,就可以馬上
找到它在貼圖上的位置,或者這個(gè)字還不在貼圖上,則擦除掉一個(gè)還沒有使用過的文字,把這個(gè)字畫到貼圖上去.
原理就講完了,可能有點(diǎn)概括,具體的就看代碼吧.
也許細(xì)心的你又發(fā)現(xiàn)了一個(gè)問題: 如果一次刷新需要顯示很多字,這張貼圖真的夠用嗎?
你的擔(dān)心也不是沒有道理,但是這種情況基本上是不會(huì)出現(xiàn)的(當(dāng)然你要選擇大字體,那就沒辦法了),因?yàn)樵谫N圖上存儲的是不重復(fù)的字,在一次刷新中要顯示
800個(gè)以上的不同的字,應(yīng)該很少碰到吧.如果你確實(shí)需要,可以增大貼圖的大小,比如選擇1024*1024的貼圖,還是以10號字體為例,可以存儲大概
3000個(gè)字,不過貼圖的大小也增長到了4M.
需要修改的幾個(gè)主要文件:
CEGUIFont.cpp
CEGUIFont.h
CEGUITexture.h
CEGUIRenderer.cpp
并增加2個(gè)文件:
CEGUIDbcsSupport.h
CEGUIDbcsSupport.cpp
具體代碼稍后整理好后,放到網(wǎng)上.希望對用到CEHUI的中國用戶能有所幫助.