/****************************************************************************
**
** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (qt-info@nokia.com)
**
** This file is part of the QtGui module of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
** No Commercial Usage
** This file contains pre-release code and may not be distributed.
** You may use this file in accordance with the terms and conditions
** contained in the Technology Preview License Agreement accompanying
** this package.
**
** GNU Lesser General Public License Usage
** Alternatively, 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.
**
** If you have questions regarding the use of this file, please contact
** Nokia at qt-info@nokia.com.
**
**
**
**
**
**
**
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "qapplication.h"
#include "qevent.h"
#ifdef Q_WS_S60
#include "qstyle.h"
#include "private/qt_s60_p.h"
#endif
#include "private/qsoftkeymanager_p.h"
#include "private/qobject_p.h"
#ifndef QT_NO_SOFTKEYMANAGER
QT_BEGIN_NAMESPACE
#ifdef Q_WS_S60
static const int s60CommandStart = 6000;
#endif
class QSoftKeyManagerPrivate : public QObjectPrivate
{
Q_DECLARE_PUBLIC(QSoftKeyManager)
public:
static void updateSoftKeys_sys(const QList<QAction*> &softKeys);
private:
QHash<QAction*, Qt::Key> keyedActions;
static QSoftKeyManager *self;
static QWidget *softKeySource;
};
QWidget *QSoftKeyManagerPrivate::softKeySource = 0;
QSoftKeyManager *QSoftKeyManagerPrivate::self = 0;
const char *QSoftKeyManager::standardSoftKeyText(StandardSoftKey standardKey)
{
const char *softKeyText = 0;
switch (standardKey) {
case OkSoftKey:
softKeyText = QT_TRANSLATE_NOOP("QSoftKeyManager", "Ok");
break;
case SelectSoftKey:
softKeyText = QT_TRANSLATE_NOOP("QSoftKeyManager", "Select");
break;
case DoneSoftKey:
softKeyText = QT_TRANSLATE_NOOP("QSoftKeyManager", "Done");
break;
case MenuSoftKey:
softKeyText = QT_TRANSLATE_NOOP("QSoftKeyManager", "Options");
break;
case CancelSoftKey:
softKeyText = QT_TRANSLATE_NOOP("QSoftKeyManager", "Cancel");
break;
default:
break;
};
return softKeyText;
}
QSoftKeyManager *QSoftKeyManager::instance()
{
if (!QSoftKeyManagerPrivate::self)
QSoftKeyManagerPrivate::self = new QSoftKeyManager;
return QSoftKeyManagerPrivate::self;
}
QSoftKeyManager::QSoftKeyManager() : QObject(*(new QSoftKeyManagerPrivate), 0)
{
}
QAction *QSoftKeyManager::createAction(StandardSoftKey standardKey, QWidget *actionWidget)
{
const char* text = standardSoftKeyText(standardKey);
QAction *action = new QAction(QSoftKeyManager::tr(text), actionWidget);
QAction::SoftKeyRole softKeyRole = QAction::NoSoftKey;
switch (standardKey) {
case OkSoftKey:
case SelectSoftKey:
case DoneSoftKey:
case MenuSoftKey:
softKeyRole = QAction::PositiveSoftKey;
break;
case CancelSoftKey:
softKeyRole = QAction::NegativeSoftKey;
break;
}
action->setSoftKeyRole(softKeyRole);
return action;
}
/*! \internal
Creates a QAction and registers the 'triggered' signal to send the given key event to
\a actionWidget as a convenience.
*/
QAction *QSoftKeyManager::createKeyedAction(StandardSoftKey standardKey, Qt::Key key, QWidget *actionWidget)
{
#ifndef QT_NO_ACTION
QScopedPointer<QAction> action(createAction(standardKey, actionWidget));
connect(action.data(), SIGNAL(triggered()), QSoftKeyManager::instance(), SLOT(sendKeyEvent()));
connect(action.data(), SIGNAL(destroyed(QObject*)), QSoftKeyManager::instance(), SLOT(cleanupHash(QObject*)));
QSoftKeyManager::instance()->d_func()->keyedActions.insert(action.data(), key);
return action.take();
#endif //QT_NO_ACTION
}
void QSoftKeyManager::cleanupHash(QObject* obj)
{
Q_D(QSoftKeyManager);
QAction *action = qobject_cast<QAction*>(obj);
d->keyedActions.remove(action);
}
void QSoftKeyManager::sendKeyEvent()
{
Q_D(QSoftKeyManager);
QAction *action = qobject_cast<QAction*>(sender());
if (!action)
return;
Qt::Key keyToSend = d->keyedActions.value(action, Qt::Key_unknown);
if (keyToSend != Qt::Key_unknown)
QApplication::postEvent(action->parentWidget(),
new QKeyEvent(QEvent::KeyPress, keyToSend, Qt::NoModifier));
}
void QSoftKeyManager::updateSoftKeys()
{
QEvent *event = new QEvent(QEvent::UpdateSoftKeys);
QApplication::postEvent(QSoftKeyManager::instance(), event);
}
bool QSoftKeyManager::event(QEvent *e)
{
#ifndef QT_NO_ACTION
if (e->type() == QEvent::UpdateSoftKeys) {
QList<QAction*> softKeys;
QWidget *source = QApplication::focusWidget();
do {
if (source) {
QList<QAction*> actions = source->actions();
for (int i = 0; i < actions.count(); ++i) {
if (actions.at(i)->softKeyRole() != QAction::NoSoftKey)
softKeys.append(actions.at(i));
}
QWidget *parent = source->parentWidget();
if (parent && softKeys.isEmpty() && !source->isWindow())
source = parent;
else
break;
} else {
source = QApplication::activeWindow();
}
} while (source);
QSoftKeyManagerPrivate::softKeySource = source;
QSoftKeyManagerPrivate::updateSoftKeys_sys(softKeys);
return true;
}
#endif //QT_NO_ACTION
return false;
}
#ifdef Q_WS_S60
void QSoftKeyManagerPrivate::updateSoftKeys_sys(const QList<QAction*> &softkeys)
{
// lets not update softkeys if s60 native dialog or menu is shown
if (QApplication::testAttribute(Qt::AA_S60DontConstructApplicationPanes)
|| CCoeEnv::Static()->AppUi()->IsDisplayingMenuOrDialog())
return;
CEikButtonGroupContainer* nativeContainer = S60->buttonGroupContainer();
nativeContainer->DrawableWindow()->SetOrdinalPosition(0);
nativeContainer->DrawableWindow()->SetPointerCapturePriority(1); //keep softkeys available in modal dialog
nativeContainer->DrawableWindow()->SetFaded(EFalse, RWindowTreeNode::EFadeIncludeChildren);
int position = -1;
bool needsExitButton = true;
QT_TRAP_THROWING(
//Using -1 instead of EAknSoftkeyEmpty to avoid flickering.
nativeContainer->SetCommandL(0, -1, KNullDesC);
nativeContainer->SetCommandL(2, -1, KNullDesC);
);
for (int index = 0; index < softkeys.count(); index++) {
const QAction* softKeyAction = softkeys.at(index);
switch (softKeyAction->softKeyRole()) {
// Positive Actions on the LSK
case QAction::PositiveSoftKey:
position = 0;
break;
case QAction::SelectSoftKey:
position = 0;
break;
// Negative Actions on the RSK
case QAction::NegativeSoftKey:
needsExitButton = false;
position = 2;
break;
default:
break;
}
int command = (softKeyAction->objectName().contains(QLatin1String("_q_menuSoftKeyAction")))
? EAknSoftkeyOptions
: s60CommandStart + index;
// _q_menuSoftKeyAction action is set to "invisible" and all invisible actions are by default
// disabled. However we never want to dim options softkey, even it is set to "invisible"
bool dimmed = (command == EAknSoftkeyOptions) ? false : !softKeyAction->isEnabled();
if (position != -1) {
const int underlineShortCut = QApplication::style()->styleHint(QStyle::SH_UnderlineShortcut);
QString iconText = softKeyAction->iconText();
TPtrC text = qt_QString2TPtrC( underlineShortCut ? softKeyAction->text() : iconText);
QT_TRAP_THROWING(
nativeContainer->SetCommandL(position, command, text);
nativeContainer->DimCommand(command, dimmed);
);
}
}
const Qt::WindowType sourceWindowType = QSoftKeyManagerPrivate::softKeySource
? QSoftKeyManagerPrivate::softKeySource->window()->windowType()
: Qt::Widget;
if (needsExitButton && sourceWindowType != Qt::Dialog && sourceWindowType != Qt::Popup)
QT_TRAP_THROWING(
nativeContainer->SetCommandL(2, EAknSoftkeyExit, qt_QString2TPtrC(QSoftKeyManager::tr("Exit"))));
nativeContainer->DrawDeferred(); // 3.1 needs an extra invitation
}
bool QSoftKeyManager::handleCommand(int command)
{
if (command >= s60CommandStart && QSoftKeyManagerPrivate::softKeySource) {
int index = command - s60CommandStart;
const QList<QAction*>& softKeys = QSoftKeyManagerPrivate::softKeySource->actions();
for (int i = 0, j = 0; i < softKeys.count(); ++i) {
QAction *action = softKeys.at(i);
if (action->softKeyRole() != QAction::NoSoftKey) {
if (j == index) {
QWidget *parent = action->parentWidget();
if (parent && parent->isEnabled()) {
action->activate(QAction::Trigger);
return true;
}
}
j++;
}
}
}
return false;
}
#else
void QSoftKeyManagerPrivate::updateSoftKeys_sys(const QList<QAction*> &)
{
}
#endif
QT_END_NAMESPACE
#endif //QT_NO_SOFTKEYMANAGER