摘 要: 針對(duì)QT在Windows、Linux、以及嵌入式Linux-arm等不同平臺(tái)下沒(méi)有開(kāi)源、高效的虛擬鍵盤(pán)(包含中文輸入法),以及QT版本升級(jí)造成代碼不兼容等問(wèn)題,設(shè)計(jì)并實(shí)現(xiàn)了一個(gè)跨平臺(tái)的虛擬鍵盤(pán)。該方法首先創(chuàng)建英文虛擬鍵盤(pán)和SQLite3的中文字庫(kù),然后在此基礎(chǔ)上根據(jù)QT底層信號(hào)與槽以及事件過(guò)濾機(jī)制設(shè)計(jì)實(shí)現(xiàn)了中文拼音輸入法接口,并結(jié)合SQLite3數(shù)據(jù)庫(kù)查詢(xún)實(shí)現(xiàn)了中文拼音到漢字的查詢(xún)、匹配、選詞以及顯示等功能,最終實(shí)現(xiàn)中英文輸入功能。最后,在不同平臺(tái)下對(duì)所設(shè)計(jì)虛擬鍵盤(pán)進(jìn)行了測(cè)試,結(jié)果表明效果良好。
關(guān)鍵詞: 虛擬鍵盤(pán);跨平臺(tái);SQLite3;嵌入式系統(tǒng)
0 引言
QT是原Trolltech公司開(kāi)發(fā)的跨平臺(tái)C++GUI應(yīng)用框架,它提供了豐富的部件集合,并具有面向?qū)ο蟆⒔M件編程等優(yōu)勢(shì),近年來(lái)QT由Digia公司運(yùn)作,發(fā)展迅猛,其在嵌入式平臺(tái)以及桌面平臺(tái)中有著不可替代的地位[1]。
QT目前支持幾乎所有的桌面操作系統(tǒng)和嵌入式操作系統(tǒng)[2],但由于各個(gè)版本(尤其是QT4和QT5)之間兼容性較差,造成以前編寫(xiě)的代碼難以在新版本下運(yùn)行或者根本無(wú)法使用等問(wèn)題。當(dāng)前,在桌面系統(tǒng)下,基于QT4的開(kāi)源虛擬鍵盤(pán)(包含中文輸入法)比較少,基于QT5的幾乎沒(méi)有;在嵌入式系統(tǒng)下,基于QT4的虛擬鍵盤(pán)幾乎都是繼承QT4內(nèi)部輸入法類(lèi)QInputContext或者QWSInputMethod,這樣設(shè)計(jì)簡(jiǎn)單、快捷,但是只能在嵌入式系統(tǒng)中使用,而QT5也已經(jīng)刪除了這兩個(gè)內(nèi)部的輸入法類(lèi)。因此,在QT環(huán)境下亟需重新設(shè)計(jì)一個(gè)新的虛擬鍵盤(pán),以此來(lái)兼容QT的不同版本以及滿(mǎn)足不同桌面操作系統(tǒng)、嵌入式設(shè)備中的輸入需求[2]。
本文設(shè)計(jì)一個(gè)在多平臺(tái)(Windows、Linux、嵌入式Linux-arm)下,跨越QT版本限制的虛擬鍵盤(pán)解決方案。該方案完全基于QT底層的信號(hào)與槽機(jī)制以及事件過(guò)濾機(jī)制,沒(méi)有使用特定系統(tǒng)下的接口,因此可以實(shí)現(xiàn)跨平臺(tái)、跨QT版本的目的,同時(shí)兼容嵌入式系統(tǒng)。鍵盤(pán)布局參考實(shí)體鍵盤(pán),支持拖拽移動(dòng),并且聯(lián)合了數(shù)據(jù)庫(kù)技術(shù),實(shí)現(xiàn)自由添加詞組,支持英文和中文拼音兩種輸入模式,并且支持鼠標(biāo)單擊虛擬鍵盤(pán)與實(shí)體鍵盤(pán)同步輸入等功能,豐富了基于QT的虛擬鍵盤(pán)設(shè)計(jì)方案。
1 虛擬鍵盤(pán)的設(shè)計(jì)原理
在GUI的編程中,經(jīng)常需要在各個(gè)界面或者各個(gè)部件之間進(jìn)行對(duì)象間通信。而信號(hào)與槽作為QT的核心機(jī)制[3],主要用來(lái)實(shí)現(xiàn)對(duì)象間通信,以此取代傳統(tǒng)的回調(diào)函數(shù)方式,這也是一種更加安全的機(jī)制。例如,在虛擬鍵盤(pán)上有部件狀態(tài)發(fā)生變化時(shí),相應(yīng)的部件就會(huì)發(fā)射(emit)一個(gè)信號(hào),而與該信號(hào)進(jìn)行關(guān)聯(lián)(connect)的槽函數(shù)則負(fù)責(zé)監(jiān)視接收信號(hào),一旦信號(hào)產(chǎn)生,則執(zhí)行槽函數(shù)里面的相應(yīng)操作。一個(gè)信號(hào)可以與一個(gè)或者多個(gè)槽進(jìn)行關(guān)聯(lián),多個(gè)信號(hào)也可以和一個(gè)槽進(jìn)行關(guān)聯(lián),甚至一個(gè)信號(hào)可以和另一個(gè)信號(hào)進(jìn)行關(guān)聯(lián),這樣就實(shí)現(xiàn)了信息的封裝,確保對(duì)象可以當(dāng)做一個(gè)組件來(lái)編程。
因此,基于QT的GUI編程簡(jiǎn)單、靈活容易擴(kuò)展,其核心也就是如何來(lái)設(shè)計(jì)槽函數(shù)[4]。信號(hào)與槽之間一般通過(guò)QObject::connect函數(shù)進(jìn)行連接,信號(hào)與槽的簡(jiǎn)單連接模型如圖1。
2 虛擬鍵盤(pán)的設(shè)計(jì)與實(shí)現(xiàn)
2.1英文輸入法的設(shè)計(jì)與實(shí)現(xiàn)
該虛擬鍵盤(pán)是在Linux(Ubuntu12.04)環(huán)境下使用QtCreator設(shè)計(jì)的鍵盤(pán)布局,在QWidget下面根據(jù)實(shí)體鍵盤(pán)拖入QToolButton,并對(duì)Button進(jìn)行屬性綁定,以便后面可以根據(jù)屬性批量處理按鍵事件。傳統(tǒng)的虛擬鍵盤(pán)則通過(guò)單個(gè)click信號(hào),連接到相應(yīng)的槽,這樣設(shè)計(jì)思路簡(jiǎn)單,容易理解,但代碼冗余較大[5]。改進(jìn)的虛擬鍵盤(pán)則根據(jù)Button屬性來(lái)捆綁信號(hào)與槽,這樣就降低了代碼冗余度,具體步驟如下。
?。?)根據(jù)實(shí)體鍵盤(pán)設(shè)計(jì)布局,基本實(shí)現(xiàn)全鍵盤(pán),根據(jù)實(shí)際項(xiàng)目需求并考慮兼容嵌入式液晶屏,故刪除了部分不常用的按鍵,具體鍵盤(pán)布局見(jiàn)圖2。
(2)為使鍵盤(pán)支持拖拽操作,將QWidget隱藏了邊框和標(biāo)題欄,然后重新實(shí)現(xiàn)了三個(gè)虛函數(shù)mouseMoveEvent(QMouseEvent*e)、mousePressEvent(QMouseEvent*e)、mouse ReleaseEvent(QMouseEvent*e),這樣虛擬鍵盤(pán)就可以在有鼠標(biāo)的情況下以及支持觸摸的液晶顯示器上,通過(guò)長(zhǎng)按空白處拖拽虛擬鍵盤(pán)。
?。?)設(shè)置QToolButton的objectName并且將所有的字母鍵、數(shù)字鍵以及特殊符號(hào)鍵依次綁定屬性為btn_alphabet、btn_number、btn_special。如按鍵“a”的objectName為btna,屬性設(shè)置為btn_letter,即btna->setProperty(”btn_alphabet”,true),其他按鍵依次設(shè)置。
?。?)創(chuàng)建QToolButton為QList類(lèi)型鏈表,獲取虛擬鍵盤(pán)中所有的QToolButton類(lèi)型的孩子項(xiàng),然后遍歷所有按鍵,依次關(guān)聯(lián)槽函數(shù),代碼如下:
QList<QToolButton*>keyValue=this->findChildren
<QToolButton*>();
foreach(QToolButton*B,keyValue){
connect(B,SIGNAL(clicked()),this,
SLOT(on_keyValue_clicked()));}
?。?)在槽函數(shù)on_keyValue_clicked()中首先通過(guò)sender()函數(shù)獲取激活的Button,然后根據(jù)之前劃分的屬性,批量地處理不同類(lèi)型的按鍵,再根據(jù)objectName實(shí)現(xiàn)特殊鍵的功能,例如Backspace、切換中英文等。
最后將虛擬鍵盤(pán)輸入的字母、數(shù)字以及特殊符號(hào)發(fā)送到QLineEdit中,通過(guò)信號(hào)focusChanged(QWidget*,QWidget*)實(shí)現(xiàn)新舊焦點(diǎn)的切換。
至此實(shí)現(xiàn)了所有虛擬鍵盤(pán)的輸入以及焦點(diǎn)切換等功能,然后在此基礎(chǔ)上聯(lián)合使用了SQLite實(shí)現(xiàn)中文拼音輸入法。
2.2 中文拼音輸入法的設(shè)計(jì)與改進(jìn)
在基于QT的傳統(tǒng)中文輸入法中,尤其是嵌入式系統(tǒng)中使用的中文輸入法,如果是基于QT4,基本都是繼承自QInputContext或者QWSInputMethod,它依賴(lài)于QT/E的C/S架構(gòu),將輸入法安裝在服務(wù)器端,負(fù)責(zé)監(jiān)聽(tīng)來(lái)自鍵盤(pán)、鼠標(biāo)的事件,然后再由服務(wù)端的輸入法進(jìn)行事件處理、分發(fā),這樣設(shè)計(jì)簡(jiǎn)單,但是自由度低、局限性大,而且使用了進(jìn)程間通信的原理[6],只能在嵌入式QT/E中使用,無(wú)法移植到其他桌面系統(tǒng),QT5也刪除了這兩個(gè)基本的內(nèi)部輸入法類(lèi),因此基于此設(shè)計(jì)的虛擬鍵盤(pán)根本無(wú)法在QT5中運(yùn)行。
本文改進(jìn)的中文輸入法則摒棄了這兩個(gè)內(nèi)部輸入法類(lèi),完全基于QT底層的信號(hào)與槽機(jī)制以及事件過(guò)濾機(jī)制,因此可以突破QT版本升級(jí)帶來(lái)的不兼容問(wèn)題,也沒(méi)有使用各個(gè)操作系統(tǒng)的系統(tǒng)函數(shù),因此只需要在虛擬鍵盤(pán)的代碼中預(yù)定義幾個(gè)簡(jiǎn)單的跨平臺(tái)的宏定義即可實(shí)現(xiàn)跨平臺(tái)(Window、Linux、嵌入式Linux-arm)。
在前面實(shí)現(xiàn)的虛擬鍵盤(pán)基礎(chǔ)上,本節(jié)使用SQLite3設(shè)計(jì)制作了中文漢字字庫(kù),設(shè)計(jì)了中文輸入法類(lèi)(Inputdemo)的基本接口,同時(shí)完成了事件過(guò)濾器對(duì)鍵盤(pán)以及鼠標(biāo)事件的分發(fā)調(diào)度。
2.2.1 中文輸入法字庫(kù)結(jié)構(gòu)
SQLite3是一款開(kāi)源免費(fèi)的高性能數(shù)據(jù)庫(kù)系統(tǒng),具有占用資源低、效率高、體積小、使用可靠等優(yōu)點(diǎn),同時(shí)支持Window、Linux以及嵌入式Linux等主流操作系統(tǒng),近幾年其在嵌入式領(lǐng)域得到了長(zhǎng)足的發(fā)展。該中文輸入法使用SQLite3主要考慮了QT本身對(duì)SQLite3提供了很好的驅(qū)動(dòng)支持,同時(shí)考慮了虛擬鍵盤(pán)兼容嵌入式Linux-arm以及在嵌入式中使用效率、查詢(xún)速度等因素[7]。輸入法字庫(kù)字段設(shè)計(jì)如下:
(1)pinyin:漢字以及詞組對(duì)應(yīng)的拼音;
(2)hanzi:漢字以及詞語(yǔ),詞組之間用空格區(qū)分,并一一對(duì)應(yīng)拼音。
?。?)frequency:漢字詞頻。
2.2.2 中文拼音輸入法設(shè)計(jì)以及實(shí)現(xiàn)
QT主要有四種處理事件的方式:重新實(shí)現(xiàn)特定的事件處理器、重寫(xiě)Object:event事件、在對(duì)象中注冊(cè)事件過(guò)濾器以及繼承QApplication重寫(xiě)notify()函數(shù)。本文采用在對(duì)象中注冊(cè)事件過(guò)濾器以及重寫(xiě)event事件的方式[8],來(lái)完成對(duì)鍵盤(pán)、鼠標(biāo)事件的過(guò)濾、分發(fā)、調(diào)度工作。設(shè)置事件過(guò)濾器主要通過(guò)兩個(gè)步驟:
?。?)安裝事件過(guò)濾器,通過(guò)對(duì)目標(biāo)對(duì)象調(diào)用installEventFilter()注冊(cè)事件過(guò)濾器來(lái)注冊(cè)監(jiān)視對(duì)象,安裝事件過(guò)濾器代碼類(lèi)似于:Object.installEventFilter(this)。
(2)重新實(shí)現(xiàn)事件過(guò)濾器,即在監(jiān)視對(duì)象的eventFilter()中處理監(jiān)視的事件,這樣鍵盤(pán)或者鼠標(biāo)事件(嵌入式中是觸摸事件)產(chǎn)生時(shí),首先將事件發(fā)送到過(guò)濾器,然后再交付給程序。事件濾波器原型如下:eventFilter(QObject*obj,QEvent*event)。
鍵盤(pán)以及鼠標(biāo)事件的處理流程如圖3所示。
中文拼音輸入的實(shí)現(xiàn)過(guò)程就是將輸入的拼音字符串轉(zhuǎn)換成相應(yīng)漢字。本輸入法對(duì)漢字的匹配建立在SQL查詢(xún)的基礎(chǔ)上,查詢(xún)簡(jiǎn)單、代碼冗余低,比較容易實(shí)現(xiàn)一些高級(jí)功能,其實(shí)現(xiàn)步驟如下。
?。?)實(shí)現(xiàn)輸入拼音以及顯示漢字功能首先自定義繼承QLabel類(lèi)的MyLabel類(lèi),重載了mouseReleaseEvent(QMouseEvent*)函數(shù),使其可以像Button一樣支持點(diǎn)擊,然后重載void enterEvent(QEvent*)、void leaveEvent(QEvent*),使其可以在選中漢字時(shí)出現(xiàn)顏色變化,最后通過(guò)部件提升的方法將MyLabel添加到QtCreator設(shè)計(jì)器中,這樣添加的MyLabel就可以支持鼠標(biāo)點(diǎn)擊(嵌入式中是觸摸操作)選中并且發(fā)送漢字,然后在ui文件添加一個(gè)QLabel來(lái)顯示輸入的拼音。
?。?)實(shí)現(xiàn)索引功能,根據(jù)之前設(shè)計(jì)的字庫(kù)表,通過(guò)SQL語(yǔ)句查詢(xún)匹配漢字以及詞組,然后將通過(guò)SQL查詢(xún)到的漢字依次添加到漢字鏈表中,并統(tǒng)計(jì)查詢(xún)的漢字或者詞組的個(gè)數(shù)以及使用的頻率,同時(shí)設(shè)計(jì)了接口display_chinese()、choice_chinese(int num)、remove_chinese(),分別用來(lái)顯示漢字或詞組、根據(jù)索引選中漢字或詞組,以及清空漢字或者詞組。最后將查詢(xún)到的漢字顯示到前面建立的MyLabel上,并通過(guò)前翻頁(yè)、后翻頁(yè)查找所需漢字或者詞組。SQL核心代碼如下:
QSqlQuery query;
QString sql="SELECT hanzi FROM pinyin WHERE
pinyin=′"+present_pinyin+"′";
query.exec(sql);
?。?)實(shí)現(xiàn)鼠標(biāo)點(diǎn)擊虛擬鍵盤(pán)與實(shí)體鍵盤(pán)同步輸入功能,即在中文模式下,首先獲取虛擬鍵盤(pán)的所有QToolButton的孩子項(xiàng),然后將實(shí)體鍵盤(pán)的鍵值依次映射到虛擬鍵盤(pán)的QToolButton上面,實(shí)現(xiàn)數(shù)字鍵選中漢字和鼠標(biāo)點(diǎn)擊選中漢字功能,同樣做了類(lèi)似的映射。核心代碼如下:
QList<QToolButton*>keyValue=this->findChildren
<QToolButton*>();
QString key=keyEvent->text();
foreach(QToolButton*Btn,keyValue){
if(Btn->text()==key){
Btn->click();}
?。?)用戶(hù)將選中的漢字或者詞組發(fā)送到QLineEdit中,至此中文輸入法完成。
3 測(cè)試
桌面系統(tǒng)環(huán)境為L(zhǎng)inux(Ubuntu12.04),編譯器g++4.6.4,分別使用QT4.7.2和QT5.3.2進(jìn)行編譯運(yùn)行,編譯順利通過(guò),界面一致;然后在嵌入式QT/Embedded下,使用qvfb進(jìn)行編譯運(yùn)行,可以實(shí)現(xiàn)中英文輸入,運(yùn)行結(jié)果如圖4所示。
4 結(jié)束語(yǔ)
本文完全基于QT的信號(hào)與槽以及事件過(guò)濾機(jī)制,結(jié)合SQLite3數(shù)據(jù)庫(kù)查詢(xún)技術(shù),設(shè)計(jì)實(shí)現(xiàn)了虛擬鍵盤(pán),可以支持中、英文輸入。同時(shí)本文的設(shè)計(jì)方案完全兼容各個(gè)桌面操作系統(tǒng)以及嵌入式Linux-arm操作系統(tǒng),并且不受QT版本升級(jí)的限制,具有廣泛的應(yīng)用價(jià)值。
參考文獻(xiàn)
[1] 屈克文,石奮蘇.基于QT/E和SQLite3的嵌入式中文輸入法的設(shè)計(jì)[J].計(jì)算機(jī)應(yīng)用,2011,31(1):149-151.
[2] BLANCHETTE J, SUMMERFIELD M. C++ GUI Qt4編程(第二版)[M].北京:電子工業(yè)出版社,2008.
[3] 唐新華.QT的信號(hào)與槽機(jī)制介紹[EB/OL].(2005-9-20)[2015-05-15].http://www.ibm.com/developerworks/cn/linux/guitoolkit/qt/signal-slot/index.html.
[4] 謝芬,潘麗,劉守印.基于QT/E的嵌入式Linux系統(tǒng)的軟鍵盤(pán)實(shí)現(xiàn)[J].電子設(shè)計(jì)工程,2012,20(5):177-179.
[5] 韋東山.嵌入式Linux應(yīng)用完全開(kāi)發(fā)手冊(cè)[M].北京:人民郵電出版社,2008.
[6] 章堅(jiān)武,吳寒君.基于嵌入式Linux和Qt4的中文輸入法的實(shí)現(xiàn)與改進(jìn)[J].電子器件,2010,33(3):395-398.
[7] SQLite參考手冊(cè)[EB/OL].(2014-06-04)[2015-05-15].http://www.sqlite.org/lang.html.
[8] QT參考文檔[EB/OL]. (2014-05-06)[2015-05-15].http://doc.qt.io/qt-4.8/qt-embedded-architecture.html.