嵌入式linux上QT標(biāo)準(zhǔn)鍵盤(pán)輸入的實(shí)現(xiàn)
掃描二維碼
隨時(shí)隨地手機(jī)看文章
作者:劉洪濤,華清遠(yuǎn)見(jiàn)嵌入式學(xué)院講師。
在嵌入式平臺(tái)上運(yùn)行Qte時(shí),使用的鍵盤(pán)通常不是標(biāo)準(zhǔn)鍵盤(pán),而是嵌入式設(shè)備外擴(kuò)的普通按鍵。那么實(shí)現(xiàn)QTE鍵盤(pán)輸入的方法大體上可以分為兩類:
(1)編寫(xiě)一個(gè)普通按鍵驅(qū)動(dòng),然后開(kāi)辟一個(gè)QT線程讀取按鍵值,在通過(guò)信號(hào)把按鍵值發(fā)送出去。需要接收鍵盤(pán)輸入的目標(biāo),聲明槽函數(shù),接收鍵盤(pán)信號(hào)。
(2)將按鍵驅(qū)動(dòng)編寫(xiě)成標(biāo)準(zhǔn)鍵盤(pán)驅(qū)動(dòng),讓QTE感覺(jué)和標(biāo)準(zhǔn)鍵盤(pá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)鍵盤(pán)輸入的過(guò)程可以分為兩步:
(1) 找一個(gè)標(biāo)準(zhǔn)usb鍵盤(pán),測(cè)試QTE能否正確設(shè)別標(biāo)準(zhǔn)鍵盤(pán)
(2) 編寫(xiě)按鍵驅(qū)動(dòng),模擬標(biāo)準(zhǔn)鍵盤(pán)輸入
一、第1步的實(shí)現(xiàn)過(guò)程:
● 配置QTE支持標(biāo)準(zhǔn)USB鍵盤(pán)
配置qte庫(kù)時(shí),增加鍵盤(pán)支持的參數(shù),如下:
./configure …… -qt-kbd-usb ……
● 配置內(nèi)核支持USB鍵盤(pán)輸入
● 插入U(xiǎn)SB鍵盤(pán)后,產(chǎn)生event設(shè)備節(jié)點(diǎn),如/dev/event2
● 設(shè)置QTE關(guān)聯(lián)的鍵盤(pán)設(shè)備的環(huán)境變量
export QWS_KEYBOARD=USB:/dev/event2
● 編寫(xiě)一個(gè)接收鍵盤(pán)事件的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è)試鍵盤(pán)輸入
當(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可以接收到它能獲取的鍵盤(pán)信號(hào),即QTE和USB鍵盤(pán)連接正確。
二、第2步的實(shí)現(xiàn)過(guò)程:
主要參考/driver/usb/input/usbkbd.c程序,完成鍵盤(pán)模擬。程序主要思想是編寫(xiě)一個(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為要支持的鍵盤(pán)事件
for (i = 0; i < 255; i++)
set_bit(usb_kbd_keycode[i], input_dev->keybit);
//設(shè)置支持的鍵盤(pán)碼,可根據(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ò)程中完成鍵盤(pán)值的獲取及input事件的遞交
static irqreturn_t button_irq(int irq, void *dev_id, struct pt_regs *regs)
{
……
input_report_key(input_dev, 59, 1); //模擬鍵盤(pán)碼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)