嵌入式linux上QT標(biāo)準(zhǔn)鍵盤輸入的實(shí)現(xiàn)
掃描二維碼
隨時(shí)隨地手機(jī)看文章
作者:劉洪濤,華清遠(yuǎn)見(jiàn)嵌入式學(xué)院講師。
在嵌入式平臺(tái)上運(yùn)行Qte時(shí),使用的鍵盤通常不是標(biāo)準(zhǔn)鍵盤,而是嵌入式設(shè)備外擴(kuò)的普通按鍵。那么實(shí)現(xiàn)QTE鍵盤輸入的方法大體上可以分為兩類:
(1)編寫一個(gè)普通按鍵驅(qū)動(dòng),然后開(kāi)辟一個(gè)QT線程讀取按鍵值,在通過(guò)信號(hào)把按鍵值發(fā)送出去。需要接收鍵盤輸入的目標(biāo),聲明槽函數(shù),接收鍵盤信號(hào)。
(2)將按鍵驅(qū)動(dòng)編寫成標(biāo)準(zhǔn)鍵盤驅(qū)動(dòng),讓QTE感覺(jué)和標(biāo)準(zhǔn)鍵盤在打交道。
上述兩種方法給有特點(diǎn)。我在一些項(xiàng)目中多數(shù)都是使用第1種方式,感覺(jué)比較直觀容易控制。但也有些情況要選用第2種方法。
第1種方法的實(shí)現(xiàn)比較容易,這里就不多說(shuō)了。下面主要把第2種方法的實(shí)現(xiàn)過(guò)程描述一下。
具體實(shí)現(xiàn)標(biāo)準(zhǔn)鍵盤輸入的過(guò)程可以分為兩步:
(1) 找一個(gè)標(biāo)準(zhǔn)usb鍵盤,測(cè)試QTE能否正確設(shè)別標(biāo)準(zhǔn)鍵盤
(2) 編寫按鍵驅(qū)動(dòng),模擬標(biāo)準(zhǔn)鍵盤輸入
一、第1步的實(shí)現(xiàn)過(guò)程:
● 配置QTE支持標(biāo)準(zhǔn)USB鍵盤
配置qte庫(kù)時(shí),增加鍵盤支持的參數(shù),如下:
./configure …… -qt-kbd-usb ……
● 配置內(nèi)核支持USB鍵盤輸入
● 插入U(xiǎn)SB鍵盤后,產(chǎn)生event設(shè)備節(jié)點(diǎn),如/dev/event2
● 設(shè)置QTE關(guān)聯(lián)的鍵盤設(shè)備的環(huán)境變量
export QWS_KEYBOARD=USB:/dev/event2
● 編寫一個(gè)接收鍵盤事件的QT測(cè)試代碼。
class MyDialog : public QDialog
{
……
protected:
virtual void keyPressEvent(QKeyEvent *k);
};
void MyDialog::keyPressEvent(QKeyEvent *k)
{
qDebug("in press event %x",k->key());
}
● 測(cè)試鍵盤輸入
當(dāng)按下F1~F12時(shí),打印出:
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
說(shuō)明QDialog 的keyPressEvent可以接收到它能獲取的鍵盤信號(hào),即QTE和USB鍵盤連接正確。
二、第2步的實(shí)現(xiàn)過(guò)程:
主要參考/driver/usb/input/usbkbd.c程序,完成鍵盤模擬。程序主要思想是編寫一個(gè)支持EV_KEY的input設(shè)備驅(qū)動(dòng)。下面摘取關(guān)鍵代碼。
● 完成input設(shè)備的注冊(cè)、注銷
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ù)實(shí)際情況注冊(cè)
input_register_device(input_dev);//注冊(cè)input設(shè)備
}
/*注銷*/
static void __exit button_cleanup(void)
{
……
input_unregister_device(input_dev);//注銷input設(shè)備
}
● 中斷處理過(guò)程中完成鍵盤值的獲取及input事件的遞交
static irqreturn_t button_irq(int irq, void *dev_id, struct pt_regs *regs)
{
……
input_report_key(input_dev, 59, 1); //模擬鍵盤碼F1按下過(guò)程
input_report_key(input_dev, 59, 0);
input_sync(input_dev);
}
/*在內(nèi)核include/linux/input.h中
#define KEY_F1 59
*/
上面給出了簡(jiǎn)要的過(guò)程,大家在具體實(shí)現(xiàn)過(guò)程中多參考/driver/usb/input/usbkbd.c文件,及注意按鍵去抖等問(wèn)題。
“本文由華清遠(yuǎn)見(jiàn)http://www.embedu.org/index.htm提供”
華清遠(yuǎn)見(jiàn)