天天看點

嵌入式linux上QT标準鍵盤輸入的實作

在嵌入式平台上運作QTE時,使用的鍵盤通常不是标準鍵盤,而是嵌入式裝置外擴的普通按鍵。那麼實作QTE鍵盤輸入的方 法大體上可以分為兩類:

(1)編寫一個普通按鍵驅動,然後開辟一個QT線程讀取按鍵值,在通過信号把按鍵值發送出 去。需要接收鍵盤輸入的目标,聲明槽函數,接收鍵盤信号。

        (2)将按鍵驅動編寫成标準鍵盤驅動,讓QTE感覺和标準鍵盤在打交道。

上述兩種方法給有特點。我在一些項目中多數都是使用第1種方式,感覺比較直覺容易控制。但 也有些情況要選用第2種方法。

第1種方法的實作比較容易,這裡就不多說了。下面主要把第2種方法的實作過程描述一下。

具體實作标準鍵盤輸入的過程可以分為兩步:

(1) 找一個标準usb鍵盤,測試QTE能否正确設别标準鍵盤

        (2) 編寫按鍵驅動,模拟标準鍵盤輸入

一、第1步的實作過程:

●    配置QTE支援标準USB鍵盤

        配置qte庫時,增加鍵盤支援的參數,如下:

        ./configure …… -qt-kbd-usb ……

        ●    配置核心支援USB鍵盤輸入

        ●    插入USB鍵盤後,産生event裝置節點,如/dev/event2

        ●    設定QTE關聯的鍵盤裝置的環境變量

                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步的實作過程:

主要參考/driver/usb/input/usbkbd.c程式,完成鍵盤模拟。程式 主要思想是編寫一個支援EV_KEY的input裝置驅動。下面摘取關鍵代碼。

●    完成input裝置的注冊、登出

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裝置

                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);

                 //設定支援的鍵盤碼,可根據實際情況注冊

                 input_register_device(input_dev);//注冊input裝置

}

static void __exit button_cleanup(void)

        {

        ……

                input_unregister_device(input_dev);//登出input裝置

        }

●    中斷處理過程中完成鍵盤值的擷取及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);

        }

上面給出了簡要的過程,大家在具體實作過程中多參考/driver/usb/input /usbkbd.c檔案,及注意按鍵去抖等問題。