`
zengxx1989
  • 浏览: 302849 次
  • 性别: Icon_minigender_1
  • 来自: 湖南
社区版块
存档分类
最新评论

Qt Embedded中如何实现自己的keyboard

    博客分类:
  • QT
阅读更多
最近的项目是基于Qt,既然是嵌入式设备,难免就要在根据自己的平台来实现键盘的驱动部分,当然是属于Qt一层,而不是更底层的字符设备驱动。

这里要讲的方法是在不重新编译Qt库的情况,把我们自己的键盘驱动作为一个Qt插件集成到我们的程序中。当然也可以把我们自己的键盘驱动部分直接编译到Qt库里面。

Qt中提供的插件机制(Plugin),可以使得我们很容易根据自己的硬件实现自己的键盘代码。




具体步骤:

1. 首先创建一个静态插件库(当然也可以是动态库),我们这边库名叫做optkeypad。
    a. 创建一个键盘处理类 KeypadHandler, 多继承于QObject 和 QWSKeyboardHandler, 实现一个关键的函数readKpdData(),这个函数很关键,就是用来把我们的硬件值(hardcode)转换为Qt库中定义的键值。
    b. 创建一个插件类 KeypadDriverPlugin, 继承于QKbdDriverPlugin, 这个类主要是创建一个我们自己的键盘处理类KeypadHandler,重新实现QKbdDriverPlugin的create()和keys()函数,这是两个虚函数,我们要在自己的类中重新实现这两个函数。
     c. 我们做成一个静态插件类,所以.pro文件如下。
2. 那么如何使用这个插件类呢,
    a. 在我们的应用程序中(一般在main.cpp中)添加宏 Q_IMPORT_PLUGIN(optkeypad)。optkeypad是库名,注意不包含linux中的前缀lib和扩展名.a。
    b. 在应用程序的.pro文件中添加 QTPLUGIN += optkeypad, 并且要在链接中添加这个库  LIBS += -L(路径)  -loptkeypad
    c. 接下来要为程序添加环境变量 QWS_KEYBOARD,   export QWS_KEYBOARD=OptimusKpdHandler:/dev/input/event1 (这个很重要)
        注意这个OptimusKpdHandler是怎么来的, 我们的插件类KeypadDriverPlugin要根据这个名字来建立一个键盘处理类KeypadHandler,所以如果这个名字不匹配的话,处理类是不会被创建的。
3. 接下来我们的Qt应用程序中的键值,就是KeypadHandler类的readKpdData()处理后的值。

        .pro文件


TEMPLATE = lib
CONFIG += static \
        plugin \
        release
TARGET = optkeypad
HEADERS += keypadhandler.h \
    keypaddriverplugin.h
SOURCES += keypadhandler.cpp \
    keypaddriverplugin.cpp

QMAKE_INCDIR += ../ClockCommon
QMAKE_LIBS += -lClockCommon


        keypadDriverPlugin.h文件


class KeypadDriverPlugin : public QKbdDriverPlugin
{
    Q_OBJECT
public:
    KeypadDriverPlugin(QObject *parent = 0);
    ~KeypadDriverPlugin();

    QWSKeyboardHandler *create(const QString &driver, const QString &device);
    QWSKeyboardHandler *create(const QString &driver);
    QStringList keys() const;
};


        类的实现文件


KeypadDriverPlugin::KeypadDriverPlugin(QObject *parent)
{
   
}

KeypadDriverPlugin::~KeypadDriverPlugin()
{
}

QStringList KeypadDriverPlugin::keys() const
{
    return QStringList() << "OptimusKpdHandler";
}

//
// The create() functions are responsible for returning an instance of
// the keypad driver. We do this only if the driver parameter matches our key.
//
QWSKeyboardHandler *KeypadDriverPlugin::create(const QString &driver, const QString &device)
{
    PR("KeypadDriverPlugin::create###################################: %s\n",       driver.toLocal8Bit().constData());
    if (driver.toLower() == "optimuskpdhandler")
    {
        PR("Before creating KeypadHandler\n");
        return new KeypadHandler(device);
    }

    return 0;
}

QWSKeyboardHandler *KeypadDriverPlugin::create(const QString &driver)
{
    if (driver.toLower() == "optimuskpdhandler")
    {
        PR("Before creating KeypadHandler");
        return new KeypadHandler();
    }

    return 0;
}


        keypadHandler.h文件


class KeypadHandler : public QObject, public QWSKeyboardHandler
{
    Q_OBJECT

public:
    KeypadHandler(const QString &device = QString("/dev/input/event0"));
    ~KeypadHandler();

private:
    QSocketNotifier *m_notifier;
    int kbdFd;

private slots:
    void readKpdData();
};


        类的实现文件
struct InputData
{
    unsigned int dummy1;
    unsigned int dummy2;
    unsigned short type;
    unsigned short code;
    unsigned int value;
    unsigned int dummy3;
    unsigned int dummy4;
    unsigned int dummy5;
    unsigned int dummy6;
};

KeypadHandler::KeypadHandler(const QString &device)
{
    setObjectName("Optimus Keypad Handler");
    this->kbdFd = ::open(device.toLocal8Bit().constData(), O_RDONLY, 0);

    if (kbdFd > 0)
    {
        PR("%s opened as keyboard input.\n", device.toLocal8Bit().constData());
        g_Log.Debugf(LOG_DETAIL_TRACE, 0, L"%s opened as keyboard input.", device.toLocal8Bit().constData());
        this->m_notifier = new QSocketNotifier(kbdFd, QSocketNotifier::Read, this);
        connect(this->m_notifier, SIGNAL(activated(int)), this, SLOT(readKpdData()));
    }
    else
    {
        PR("Cannot open %s for keyboard input. (%s)",
                     device.toLocal8Bit().constData(), strerror(errno));
        g_Log.Errorf(LOG_DETAIL_IMPORTANT, 0, L"Cannot open %s for keyboard input. (%s)",
                     device.toLocal8Bit().constData(), strerror(errno));
        return;
    }
}

KeypadHandler::~KeypadHandler()
{
    if (kbdFd > 0)
        ::close(kbdFd);
}





// Key function
void KeypadHandler::readKpdData()
{
    InputData event;

    int n = read(kbdFd, &event, sizeof(InputData));
    if (n != sizeof(InputData))
    {
        PR("key pressed: n=%d\n", n);
        g_Log.Debugf(LOG_DETAIL_TRACE, 0, L"key pressed: n=%d", n);
        return;
    }

    PR("key pressed: type=%d, code=0x%x, value=%d, %s\n",
                 event.type, event.code, event.value, (event.value != 0)? "(Down)" : "(Up)");
    g_Log.Debugf(LOG_DETAIL_TRACE, 0, L"key pressed: type=%d, code=%d, value=%d, %s",
                 event.type, event.code, event.value, (event.value != 0)? "(Down)" : "(Up)");

    Qt::KeyboardModifiers modifiers = Qt::NoModifier;
    int unicode = 0xffff;
    int key_code = 0;

    // 可以根据自己特定的硬件值来设定。
    switch (event.code)
    {
        case 0x2:
            key_code = Qt::Key_1;
            unicode = '1';
            break;
        case 0x110:
            key_code = Qt::Key_Context1;
            unicode = 0xffff;
            break;
        case 0x100:
            key_code = Qt::Key_Back;
            unicode = 0xffff;
            break;
        default:

            break;
    }

    this->processKeyEvent(unicode, key_code, modifiers, event.value != 0, false);
}
分享到:
评论

相关推荐

    Qt Embedded中如何实现自己的keyboard.doc

    本文将详细介绍如何在Qt Embedded环境中实现自己的键盘驱动,并将其作为Qt插件集成到项目中。 #### 二、实现原理与步骤 ##### 1. 创建键盘驱动插件 为了实现在不重新编译Qt库的情况下集成自定义键盘驱动的目标,...

    Qt_Embedded在嵌入式Linux上的移植.pdf

    文中介绍了Qt_Embedded在构建图形界面、实现事件响应等方面的优点,分析和介绍了Qt_Embedded在嵌入式Linux上的实现方案,并在此基础上给出了详细的设计和实现。 Qt_Embedded在嵌入式Linux上的移植,提供了一种高效...

    QT程序自启动

    最后,在`/etc/init.d/rcS`脚本中添加一行`QT4 &`即可实现在系统启动时自动运行QT程序。 通过以上步骤,不仅能够确保QT程序在开发板上顺利启动,还能实现开机自启动的功能,大大提高了嵌入式系统的用户体验和稳定...

    QT自定义按键原理和使用

    通过以上介绍可以看出,在QT框架中实现自定义按键涉及多个层面的技术要点。从理解QWS架构、编写按键驱动到实现具体的按键逻辑,每一步都需要开发者具备扎实的基础知识和实践经验。无论是直接编译QT源码还是制作插件...

    常用嵌入式GUI比较

    当前,嵌入式Linux系统中有多种GUI解决方案,本文将对MicroWindows、Qt/Embedded、MiniGUI、OpenGUI等几个主流的嵌入式GUI解决方案进行比较。 一、MicroWindows MicroWindows是一个开放源码的嵌入式GUI软件,它...

    基于Linux操作系统的网络应用软件开发.pdf

    例如,配置文件中常常需要设置环境变量,如QTDIR、QWS_DISPLAY、QWS_KEYBOARD、TSLIB等,这些环境变量定义了Qt应用的运行环境以及输入输出设备。编译工具如configure、make、g++等用于编译源代码,生成可执行文件。 ...

    结合telnet与ftp 功能服务器wasd第三版源码

    [note] : /usr/src/qt-embedded-linux-opensource-src-4.4.3.tar.gz will be sent to server's `pwd` as qt-embedded-linux-opensource-src-4.4.3.tar.gz 5 get files [usage] : g remotefilename ...

    Mayavi 参考

    Keyboard interaction From interactive usage to scripting The embedded Python interpreter Recording Mayavi actions to a script Command line arguments mlab: Python scripting for 3D plotting A demo ...

Global site tag (gtag.js) - Google Analytics