嵌入式linux上QT標準鍵盤輸入的實現(xiàn)
作者:劉洪濤,華清遠見嵌入式學院講師。
在嵌入式平臺上運行Qte時,使用的鍵盤通常不是標準鍵盤,而是嵌入式設(shè)備外擴的普通按鍵。那么實現(xiàn)QTE鍵盤輸入的方法大體上可以分為兩類:
(1)編寫一個普通按鍵驅(qū)動,然后開辟一個QT線程讀取按鍵值,在通過信號把按鍵值發(fā)送出去。需要接收鍵盤輸入的目標,聲明槽函數(shù),接收鍵盤信號。
(2)將按鍵驅(qū)動編寫成標準鍵盤驅(qū)動,讓QTE感覺和標準鍵盤在打交道。
上述兩種方法給有特點。我在一些項目中多數(shù)都是使用第1種方式,感覺比較直觀容易控制。但也有些情況要選用第2種方法。
第1種方法的實現(xiàn)比較容易,這里就不多說了。下面主要把第2種方法的實現(xiàn)過程描述一下。
具體實現(xiàn)標準鍵盤輸入的過程可以分為兩步:
(1) 找一個標準usb鍵盤,測試QTE能否正確設(shè)別標準鍵盤
(2) 編寫按鍵驅(qū)動,模擬標準鍵盤輸入
一、第1步的實現(xiàn)過程:
● 配置QTE支持標準USB鍵盤
配置qte庫時,增加鍵盤支持的參數(shù),如下:
./configure …… -qt-kbd-usb ……
● 配置內(nèi)核支持USB鍵盤輸入
● 插入USB鍵盤后,產(chǎn)生event設(shè)備節(jié)點,如/dev/event2
● 設(shè)置QTE關(guān)聯(lián)的鍵盤設(shè)備的環(huán)境變量
export QWS_KEYBOARD=USB:/dev/event2
● 編寫一個接收鍵盤事件的QT測試代碼。
class MyDialog : public QDialog
{
……
protected:
virtual void keyPressEvent(QKeyEvent *k);
};
void MyDialog::keyPressEvent(QKeyEvent *k)
{
qDebug("in press event %x",k->key());
}
● 測試鍵盤輸入
當按下F1~F12時,打印出:
in press event 1000030
in press event 1000031
in press event 1000032
in press event 1000033
in press event 1000034
in press event 1000035
in press event 1000036
in press event 1000037
in press event 1000038
in press event 1000039
in press event 100003a
查QT幫助 Key_F1=0x1000030
說明QDialog 的keyPressEvent可以接收到它能獲取的鍵盤信號,即QTE和USB鍵盤連接正確。
二、第2步的實現(xiàn)過程:
主要參考/driver/usb/input/usbkbd.c程序,完成鍵盤模擬。程序主要思想是編寫一個支持EV_KEY的input設(shè)備驅(qū)動。下面摘取關(guān)鍵代碼。
● 完成input設(shè)備的注冊、注銷
struct input_dev *input_dev;
static unsigned char usb_kbd_keycode[256] = {
0, 0, 0, 0, 30, 48, 46, 32, 18, 33, 34, 35, 23, 36, 37, 38,
50, 49, 24, 25, 16, 19, 31, 20, 22, 47, 17, 45, 21, 44, 2, 3,
4, 5, 6, 7, 8, 9, 10, 11, 28, 1, 14, 15, 57, 12, 13, 26,
27, 43, 43, 39, 40, 41, 51, 52, 53, 58, 59, 60, 61, 62, 63, 64,
65, 66, 67, 68, 87, 88, 99, 70,119,110,102,104,111,107,109,106,
105,108,103, 69, 98, 55, 74, 78, 96, 79, 80, 81, 75, 76, 77, 71,
72, 73, 82, 83, 86,127,116,117,183,184,185,186,187,188,189,190,
191,192,193,194,134,138,130,132,128,129,131,137,133,135,136,113,
115,114, 0, 0, 0,121, 0, 89, 93,124, 92, 94, 95, 0, 0, 0,
122,123, 90, 91, 85, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
29, 42, 56,125, 97, 54,100,126,164,166,165,163,161,115,114,113,
150,158,159,128,136,177,178,176,142,152,173,140
};
/*初始化*/
static int button_init(void)
{
……
input_dev = input_allocate_device();//分配input設(shè)備
input_dev->evbit[0] = BIT(EV_KEY) | BIT(EV_LED) | BIT(EV_REP);
// EV_KEY為要支持的鍵盤事件
for (i = 0; i < 255; i++)
set_bit(usb_kbd_keycode[i], input_dev->keybit);
//設(shè)置支持的鍵盤碼,可根據(jù)實際情況注冊
input_register_device(input_dev);//注冊input設(shè)備
}
/*注銷*/
static void __exit button_cleanup(void)
{
……
input_unregister_device(input_dev);//注銷input設(shè)備
}
● 中斷處理過程中完成鍵盤值的獲取及input事件的遞交
static irqreturn_t button_irq(int irq, void *dev_id, struct pt_regs *regs)
{
……
input_report_key(input_dev, 59, 1); //模擬鍵盤碼F1按下過程
input_report_key(input_dev, 59, 0);
input_sync(input_dev);
}
/*在內(nèi)核include/linux/input.h中
#define KEY_F1 59
*/
上面給出了簡要的過程,大家在具體實現(xiàn)過程中多參考/driver/usb/input/usbkbd.c文件,及注意按鍵去抖等問題。
“本文由華清遠見http://www.embedu.org/index.htm提供”
華清遠見