blob: fc9d0cb8d843cecf887798342273a97c56887c0d [file] [log] [blame]
/****************************************************************************
**
** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (qt-info@nokia.com)
**
** This file is part of the plugins of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser General Public
** License version 2.1 as published by the Free Software Foundation and
** appearing in the file LICENSE.LGPL included in the packaging of this
** file. Please review the following information to ensure the GNU Lesser
** General Public License version 2.1 requirements will be met:
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights. These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU General
** Public License version 3.0 as published by the Free Software Foundation
** and appearing in the file LICENSE.GPL included in the packaging of this
** file. Please review the following information to ensure the GNU General
** Public License version 3.0 requirements will be met:
** http://www.gnu.org/copyleft/gpl.html.
**
** Other Usage
** Alternatively, this file may be used in accordance with the terms and
** conditions contained in a signed written agreement between you and Nokia.
**
**
**
**
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "qdirectfbkeyboard.h"
#ifndef QT_NO_QWS_DIRECTFB
#include "qdirectfbscreen.h"
#include <qobject.h>
#include <qsocketnotifier.h>
#include <qhash.h>
#include <directfb.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
QT_BEGIN_NAMESPACE
class KeyMap : public QHash<DFBInputDeviceKeySymbol, Qt::Key>
{
public:
KeyMap();
};
Q_GLOBAL_STATIC(KeyMap, keymap);
class QDirectFBKeyboardHandlerPrivate : public QObject
{
Q_OBJECT
public:
QDirectFBKeyboardHandlerPrivate(QDirectFBKeyboardHandler *handler);
~QDirectFBKeyboardHandlerPrivate();
void suspend();
void resume();
private:
QDirectFBKeyboardHandler *handler;
IDirectFBEventBuffer *eventBuffer;
QSocketNotifier *keyboardNotifier;
DFBEvent event;
int bytesRead;
int lastUnicode, lastKeycode;
Qt::KeyboardModifiers lastModifiers;
private Q_SLOTS:
void readKeyboardData();
};
QDirectFBKeyboardHandlerPrivate::QDirectFBKeyboardHandlerPrivate(QDirectFBKeyboardHandler *h)
: handler(h), eventBuffer(0), keyboardNotifier(0), bytesRead(0),
lastUnicode(0), lastKeycode(0), lastModifiers(0)
{
Q_ASSERT(qt_screen);
IDirectFB *fb = QDirectFBScreen::instance()->dfb();
if (!fb) {
qCritical("QDirectFBKeyboardHandler: DirectFB not initialized");
return;
}
DFBResult result;
result = fb->CreateInputEventBuffer(fb, DICAPS_KEYS, DFB_TRUE,
&eventBuffer);
if (result != DFB_OK) {
DirectFBError("QDirectFBKeyboardHandler: "
"Unable to create input event buffer", result);
return;
}
int fd;
result = eventBuffer->CreateFileDescriptor(eventBuffer, &fd);
if (result != DFB_OK) {
DirectFBError("QDirectFBKeyboardHandler: "
"Unable to create file descriptor", result);
return;
}
int flags = ::fcntl(fd, F_GETFL, 0);
::fcntl(fd, F_SETFL, flags | O_NONBLOCK);
memset(&event, 0, sizeof(event));
keyboardNotifier = new QSocketNotifier(fd, QSocketNotifier::Read, this);
connect(keyboardNotifier, SIGNAL(activated(int)),
this, SLOT(readKeyboardData()));
resume();
}
void QDirectFBKeyboardHandlerPrivate::suspend()
{
keyboardNotifier->setEnabled(false);
}
void QDirectFBKeyboardHandlerPrivate::resume()
{
eventBuffer->Reset(eventBuffer);
keyboardNotifier->setEnabled(true);
}
QDirectFBKeyboardHandlerPrivate::~QDirectFBKeyboardHandlerPrivate()
{
if (eventBuffer)
eventBuffer->Release(eventBuffer);
}
void QDirectFBKeyboardHandlerPrivate::readKeyboardData()
{
if(!qt_screen)
return;
for (;;) {
// GetEvent returns DFB_UNSUPPORTED after CreateFileDescriptor().
// This seems stupid and I really hope it's a bug which will be fixed.
// DFBResult ret = eventBuffer->GetEvent(eventBuffer, &event);
char *buf = reinterpret_cast<char*>(&event);
int ret = ::read(keyboardNotifier->socket(),
buf + bytesRead, sizeof(DFBEvent) - bytesRead);
if (ret == -1) {
if (errno != EAGAIN)
qWarning("QDirectFBKeyboardHandlerPrivate::readKeyboardData(): %s",
strerror(errno));
return;
}
Q_ASSERT(ret >= 0);
bytesRead += ret;
if (bytesRead < int(sizeof(DFBEvent)))
break;
bytesRead = 0;
Q_ASSERT(event.clazz == DFEC_INPUT);
const DFBInputEvent input = event.input;
Qt::KeyboardModifiers modifiers = Qt::NoModifier;
// Not implemented:
// if (input.modifiers & DIMM_SUPER)
// if (input.modifiers & DIMM_HYPER)
if (!(input.flags & DIEF_KEYSYMBOL) ||
!(input.flags & DIEF_KEYID) ||
!(input.type & (DIET_KEYPRESS|DIET_KEYRELEASE)))
{
static bool first = true;
if (first) {
qWarning("QDirectFBKeyboardHandler - Getting unexpected non-keyboard related events");
first = false;
}
break;
}
if (input.flags & DIEF_MODIFIERS) {
if (input.modifiers & DIMM_SHIFT)
modifiers |= Qt::ShiftModifier;
if (input.modifiers & DIMM_CONTROL)
modifiers |= Qt::ControlModifier;
if (input.modifiers & DIMM_ALT)
modifiers |= Qt::AltModifier;
if (input.modifiers & DIMM_ALTGR)
modifiers |= Qt::AltModifier;
if (input.modifiers & DIMM_META)
modifiers |= Qt::MetaModifier;
}
const bool press = input.type & DIET_KEYPRESS;
DFBInputDeviceKeySymbol symbol = input.key_symbol;
int unicode = -1;
int keycode = 0;
keycode = keymap()->value(symbol);
if (DFB_KEY_TYPE(symbol) == DIKT_UNICODE)
unicode = symbol;
if (unicode != -1 || keycode != 0) {
bool autoRepeat = false;
if (press) {
if (unicode == lastUnicode && keycode == lastKeycode && modifiers == lastModifiers) {
autoRepeat = true;
} else {
lastUnicode = unicode;
lastKeycode = keycode;
lastModifiers = modifiers;
}
} else {
lastUnicode = lastKeycode = -1;
lastModifiers = 0;
}
if (autoRepeat) {
handler->processKeyEvent(unicode, keycode,
modifiers, false, autoRepeat);
}
handler->processKeyEvent(unicode, keycode,
modifiers, press, autoRepeat);
}
}
}
QDirectFBKeyboardHandler::QDirectFBKeyboardHandler(const QString &device)
: QWSKeyboardHandler()
{
Q_UNUSED(device);
d = new QDirectFBKeyboardHandlerPrivate(this);
}
QDirectFBKeyboardHandler::~QDirectFBKeyboardHandler()
{
delete d;
}
KeyMap::KeyMap()
{
insert(DIKS_BACKSPACE , Qt::Key_Backspace);
insert(DIKS_TAB , Qt::Key_Tab);
insert(DIKS_RETURN , Qt::Key_Return);
insert(DIKS_ESCAPE , Qt::Key_Escape);
insert(DIKS_DELETE , Qt::Key_Delete);
insert(DIKS_CURSOR_LEFT , Qt::Key_Left);
insert(DIKS_CURSOR_RIGHT , Qt::Key_Right);
insert(DIKS_CURSOR_UP , Qt::Key_Up);
insert(DIKS_CURSOR_DOWN , Qt::Key_Down);
insert(DIKS_INSERT , Qt::Key_Insert);
insert(DIKS_HOME , Qt::Key_Home);
insert(DIKS_END , Qt::Key_End);
insert(DIKS_PAGE_UP , Qt::Key_PageUp);
insert(DIKS_PAGE_DOWN , Qt::Key_PageDown);
insert(DIKS_PRINT , Qt::Key_Print);
insert(DIKS_PAUSE , Qt::Key_Pause);
insert(DIKS_SELECT , Qt::Key_Select);
insert(DIKS_GOTO , Qt::Key_OpenUrl);
insert(DIKS_CLEAR , Qt::Key_Clear);
insert(DIKS_MENU , Qt::Key_Menu);
insert(DIKS_HELP , Qt::Key_Help);
insert(DIKS_INTERNET , Qt::Key_HomePage);
insert(DIKS_MAIL , Qt::Key_LaunchMail);
insert(DIKS_FAVORITES , Qt::Key_Favorites);
insert(DIKS_BACK , Qt::Key_Back);
insert(DIKS_FORWARD , Qt::Key_Forward);
insert(DIKS_VOLUME_UP , Qt::Key_VolumeUp);
insert(DIKS_VOLUME_DOWN , Qt::Key_VolumeDown);
insert(DIKS_MUTE , Qt::Key_VolumeMute);
insert(DIKS_PLAYPAUSE , Qt::Key_Pause);
insert(DIKS_PLAY , Qt::Key_MediaPlay);
insert(DIKS_STOP , Qt::Key_MediaStop);
insert(DIKS_RECORD , Qt::Key_MediaRecord);
insert(DIKS_PREVIOUS , Qt::Key_MediaPrevious);
insert(DIKS_NEXT , Qt::Key_MediaNext);
insert(DIKS_F1 , Qt::Key_F1);
insert(DIKS_F2 , Qt::Key_F2);
insert(DIKS_F3 , Qt::Key_F3);
insert(DIKS_F4 , Qt::Key_F4);
insert(DIKS_F5 , Qt::Key_F5);
insert(DIKS_F6 , Qt::Key_F6);
insert(DIKS_F7 , Qt::Key_F7);
insert(DIKS_F8 , Qt::Key_F8);
insert(DIKS_F9 , Qt::Key_F9);
insert(DIKS_F10 , Qt::Key_F10);
insert(DIKS_F11 , Qt::Key_F11);
insert(DIKS_F12 , Qt::Key_F12);
insert(DIKS_SHIFT , Qt::Key_Shift);
insert(DIKS_CONTROL , Qt::Key_Control);
insert(DIKS_ALT , Qt::Key_Alt);
insert(DIKS_ALTGR , Qt::Key_AltGr);
insert(DIKS_META , Qt::Key_Meta);
insert(DIKS_SUPER , Qt::Key_Super_L); // ???
insert(DIKS_HYPER , Qt::Key_Hyper_L); // ???
insert(DIKS_CAPS_LOCK , Qt::Key_CapsLock);
insert(DIKS_NUM_LOCK , Qt::Key_NumLock);
insert(DIKS_SCROLL_LOCK , Qt::Key_ScrollLock);
insert(DIKS_DEAD_ABOVEDOT , Qt::Key_Dead_Abovedot);
insert(DIKS_DEAD_ABOVERING , Qt::Key_Dead_Abovering);
insert(DIKS_DEAD_ACUTE , Qt::Key_Dead_Acute);
insert(DIKS_DEAD_BREVE , Qt::Key_Dead_Breve);
insert(DIKS_DEAD_CARON , Qt::Key_Dead_Caron);
insert(DIKS_DEAD_CEDILLA , Qt::Key_Dead_Cedilla);
insert(DIKS_DEAD_CIRCUMFLEX , Qt::Key_Dead_Circumflex);
insert(DIKS_DEAD_DIAERESIS , Qt::Key_Dead_Diaeresis);
insert(DIKS_DEAD_DOUBLEACUTE , Qt::Key_Dead_Doubleacute);
insert(DIKS_DEAD_GRAVE , Qt::Key_Dead_Grave);
insert(DIKS_DEAD_IOTA , Qt::Key_Dead_Iota);
insert(DIKS_DEAD_MACRON , Qt::Key_Dead_Macron);
insert(DIKS_DEAD_OGONEK , Qt::Key_Dead_Ogonek);
insert(DIKS_DEAD_SEMIVOICED_SOUND , Qt::Key_Dead_Semivoiced_Sound);
insert(DIKS_DEAD_TILDE , Qt::Key_Dead_Tilde);
insert(DIKS_DEAD_VOICED_SOUND , Qt::Key_Dead_Voiced_Sound);
insert(DIKS_SPACE , Qt::Key_Space);
insert(DIKS_EXCLAMATION_MARK , Qt::Key_Exclam);
insert(DIKS_QUOTATION , Qt::Key_QuoteDbl);
insert(DIKS_NUMBER_SIGN , Qt::Key_NumberSign);
insert(DIKS_DOLLAR_SIGN , Qt::Key_Dollar);
insert(DIKS_PERCENT_SIGN , Qt::Key_Percent);
insert(DIKS_AMPERSAND , Qt::Key_Ampersand);
insert(DIKS_APOSTROPHE , Qt::Key_Apostrophe);
insert(DIKS_PARENTHESIS_LEFT , Qt::Key_ParenLeft);
insert(DIKS_PARENTHESIS_RIGHT , Qt::Key_ParenRight);
insert(DIKS_ASTERISK , Qt::Key_Asterisk);
insert(DIKS_PLUS_SIGN , Qt::Key_Plus);
insert(DIKS_COMMA , Qt::Key_Comma);
insert(DIKS_MINUS_SIGN , Qt::Key_Minus);
insert(DIKS_PERIOD , Qt::Key_Period);
insert(DIKS_SLASH , Qt::Key_Slash);
insert(DIKS_0 , Qt::Key_0);
insert(DIKS_1 , Qt::Key_1);
insert(DIKS_2 , Qt::Key_2);
insert(DIKS_3 , Qt::Key_3);
insert(DIKS_4 , Qt::Key_4);
insert(DIKS_5 , Qt::Key_5);
insert(DIKS_6 , Qt::Key_6);
insert(DIKS_7 , Qt::Key_7);
insert(DIKS_8 , Qt::Key_8);
insert(DIKS_9 , Qt::Key_9);
insert(DIKS_COLON , Qt::Key_Colon);
insert(DIKS_SEMICOLON , Qt::Key_Semicolon);
insert(DIKS_LESS_THAN_SIGN , Qt::Key_Less);
insert(DIKS_EQUALS_SIGN , Qt::Key_Equal);
insert(DIKS_GREATER_THAN_SIGN , Qt::Key_Greater);
insert(DIKS_QUESTION_MARK , Qt::Key_Question);
insert(DIKS_AT , Qt::Key_At);
insert(DIKS_CAPITAL_A , Qt::Key_A);
insert(DIKS_CAPITAL_B , Qt::Key_B);
insert(DIKS_CAPITAL_C , Qt::Key_C);
insert(DIKS_CAPITAL_D , Qt::Key_D);
insert(DIKS_CAPITAL_E , Qt::Key_E);
insert(DIKS_CAPITAL_F , Qt::Key_F);
insert(DIKS_CAPITAL_G , Qt::Key_G);
insert(DIKS_CAPITAL_H , Qt::Key_H);
insert(DIKS_CAPITAL_I , Qt::Key_I);
insert(DIKS_CAPITAL_J , Qt::Key_J);
insert(DIKS_CAPITAL_K , Qt::Key_K);
insert(DIKS_CAPITAL_L , Qt::Key_L);
insert(DIKS_CAPITAL_M , Qt::Key_M);
insert(DIKS_CAPITAL_N , Qt::Key_N);
insert(DIKS_CAPITAL_O , Qt::Key_O);
insert(DIKS_CAPITAL_P , Qt::Key_P);
insert(DIKS_CAPITAL_Q , Qt::Key_Q);
insert(DIKS_CAPITAL_R , Qt::Key_R);
insert(DIKS_CAPITAL_S , Qt::Key_S);
insert(DIKS_CAPITAL_T , Qt::Key_T);
insert(DIKS_CAPITAL_U , Qt::Key_U);
insert(DIKS_CAPITAL_V , Qt::Key_V);
insert(DIKS_CAPITAL_W , Qt::Key_W);
insert(DIKS_CAPITAL_X , Qt::Key_X);
insert(DIKS_CAPITAL_Y , Qt::Key_Y);
insert(DIKS_CAPITAL_Z , Qt::Key_Z);
insert(DIKS_SQUARE_BRACKET_LEFT , Qt::Key_BracketLeft);
insert(DIKS_BACKSLASH , Qt::Key_Backslash);
insert(DIKS_SQUARE_BRACKET_RIGHT , Qt::Key_BracketRight);
insert(DIKS_CIRCUMFLEX_ACCENT , Qt::Key_AsciiCircum);
insert(DIKS_UNDERSCORE , Qt::Key_Underscore);
insert(DIKS_SMALL_A , Qt::Key_A);
insert(DIKS_SMALL_B , Qt::Key_B);
insert(DIKS_SMALL_C , Qt::Key_C);
insert(DIKS_SMALL_D , Qt::Key_D);
insert(DIKS_SMALL_E , Qt::Key_E);
insert(DIKS_SMALL_F , Qt::Key_F);
insert(DIKS_SMALL_G , Qt::Key_G);
insert(DIKS_SMALL_H , Qt::Key_H);
insert(DIKS_SMALL_I , Qt::Key_I);
insert(DIKS_SMALL_J , Qt::Key_J);
insert(DIKS_SMALL_K , Qt::Key_K);
insert(DIKS_SMALL_L , Qt::Key_L);
insert(DIKS_SMALL_M , Qt::Key_M);
insert(DIKS_SMALL_N , Qt::Key_N);
insert(DIKS_SMALL_O , Qt::Key_O);
insert(DIKS_SMALL_P , Qt::Key_P);
insert(DIKS_SMALL_Q , Qt::Key_Q);
insert(DIKS_SMALL_R , Qt::Key_R);
insert(DIKS_SMALL_S , Qt::Key_S);
insert(DIKS_SMALL_T , Qt::Key_T);
insert(DIKS_SMALL_U , Qt::Key_U);
insert(DIKS_SMALL_V , Qt::Key_V);
insert(DIKS_SMALL_W , Qt::Key_W);
insert(DIKS_SMALL_X , Qt::Key_X);
insert(DIKS_SMALL_Y , Qt::Key_Y);
insert(DIKS_SMALL_Z , Qt::Key_Z);
insert(DIKS_CURLY_BRACKET_LEFT , Qt::Key_BraceLeft);
insert(DIKS_VERTICAL_BAR , Qt::Key_Bar);
insert(DIKS_CURLY_BRACKET_RIGHT , Qt::Key_BraceRight);
insert(DIKS_TILDE , Qt::Key_AsciiTilde);
}
QT_END_NAMESPACE
#include "qdirectfbkeyboard.moc"
#endif // QT_NO_QWS_DIRECTFB