src/hbcore/inputfw/hbinputcontextproxy.cpp
author hgs
Tue, 13 Jul 2010 22:03:02 +0300
changeset 13 5168dbe2168a
parent 7 923ff622b8b9
child 21 4633027730f5
child 34 ed14f46c0e55
permissions -rw-r--r--
201027_1

/****************************************************************************
**
** Copyright (C) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (developer.feedback@nokia.com)
**
** This file is part of the HbCore module of the UI Extensions for Mobile.
**
** 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.
**
** If you have questions regarding the use of this file, please contact
** Nokia at developer.feedback@nokia.com.
**
****************************************************************************/
#include "hbinputcontextproxy_p.h"
#include <hbwidget.h>
#include <hbgraphicsscene.h>
#include <hbinputfocusobject.h>
#include <QGraphicsProxyWidget>
#include <QGraphicsView>
#include <hbinputmethod.h>
#include <hbinputsettingproxy.h>

/*!
@alpha
@hbcore
\class HbInputContextProxy
\brief A proxy class forwarding calls from QInputContext to HbInputMethod

This class is needed because Qt's input context system assumes the ownership
of the installed context and deletes the old one when a new context is installed.
HbInput framework wants to cache active input methods and keep the ownership of those
objects. The proxy sits between QInputContext and HbInputMethod and is the one that
is deleted when Qt level input context is switched.
real implementation.

Also a set of common events (such as Qt's input panel events) are handled on this level.
This class is not needed outside of framework code.

\sa QInputContext
\sa HbInputMethod
*/

HbInputContextProxy::HbInputContextProxy(HbInputMethod *target) : mTarget(target)
{
}

HbInputContextProxy::~HbInputContextProxy()
{
}

/*!
\internal
\reimp
*/
QList<QAction *> HbInputContextProxy::actions()
{
    if (mTarget) {
        return mTarget->actions();
    }

    return QList<QAction *>();
}

/*!
\internal
Returns true if given editor widget already has input framework focus.
*/
bool HbInputContextProxy::hasAlreadyInputFrameworkFocus(HbInputMethod *activeMethod, QObject *editorWidget) const
{
    if (activeMethod) {
        HbInputFocusObject *focusObject = activeMethod->focusObject();
        if (focusObject) {
            return focusObject->object() == editorWidget;
        }
    }

    return false;
}

/*!
\internal
Sets input framework focus to given widget if it is valid.
*/
void HbInputContextProxy::setInputFrameworkFocus(QObject *widget)
{
    if (mTarget) {
        if (!widget) {
            mTarget->setFocusObject(0);
        } else if (HbInputFocusObject::isEditor(widget) && !HbInputFocusObject::isReadOnlyWidget(widget)) {
            if (mTarget->focusObject() && mTarget->focusObject()->object() == widget) {
                // Already focused to given widget.                 
                return;
            } 
            mTarget->setFocusObject(new HbInputFocusObject(widget));
        }
    }
}

/*!
\internal
\reimp
*/
bool HbInputContextProxy::filterEvent(const QEvent *event)
{
    if (mTarget) {
        bool orientationCompleted = HbInputSettingProxy::instance()->orientationChangeCompleted();
        if (event->type() == QEvent::CloseSoftwareInputPanel && orientationCompleted) {
            setInputFrameworkFocus(0);
            return true;
        } else if (event->type() == QEvent::RequestSoftwareInputPanel && orientationCompleted) {
            if (QWidget *focusedWidget =  qApp->focusWidget()) {
                // see if the focused widget is graphics view, if so get the focused graphics item in the view
                // and acivate inputmethod for the focused graphics item
                if (QGraphicsView *graphicsView = qobject_cast<QGraphicsView *>(focusedWidget)) {
                    if (QGraphicsScene *scene = graphicsView->scene()) {
                        if (QGraphicsItem *fItem = scene->focusItem()) {
                            QGraphicsProxyWidget *proxy  = qgraphicsitem_cast<QGraphicsProxyWidget *>(fItem);
                            if (proxy) {
                                setInputFrameworkFocus(proxy->widget()->focusWidget());
                            } else {
                                setInputFrameworkFocus(static_cast<QGraphicsWidget *>(fItem));
                            }
                        }
                    }
                } else {
                    // focused wiget is not graphics view, let see if it is native qt editor
                    // and activate inputmethod for the focused widget
                    setInputFrameworkFocus(focusedWidget);
                }
            }
            return true;
        }

#ifdef Q_OS_SYMBIAN
        const quint32 HbInputContextProxyExternalKeyboardModifier = 0x00200000;

        if (event->type() == QEvent::QEvent::KeyPress || event->type() == QEvent::KeyRelease) {
            const QKeyEvent *keyEvent = static_cast<const QKeyEvent *>(event);
            if (keyEvent->nativeModifiers() & HbInputContextProxyExternalKeyboardModifier) {
                // Operating system indicates that the event originated from an external keyboard.
                // We let it pass here untouched.
                if (mTarget) {
                    mTarget->reset();
                }
                return false;
            }
        }
#endif
        if (event->type() == QEvent::KeyPress || event->type() == QEvent::KeyRelease) {
            const QKeyEvent *keyEvent = static_cast<const QKeyEvent *>(event);
            if (Qt::Key_unknown == keyEvent->key()) {
                return false;
            }
        }
        return mTarget->filterEvent(event);
    }

    return false;
}

/*!
\internal
\reimp
*/
QFont HbInputContextProxy::font() const
{
    if (mTarget) {
        return mTarget->font();
    }

    return QFont();
}

/*!
\internal
\reimp
*/
QString HbInputContextProxy::identifierName()
{
    if (mTarget) {
        return mTarget->identifierName();
    }

    return QString();
}

/*!
\internal
\reimp
*/
bool HbInputContextProxy::isComposing() const
{
    if (mTarget) {
        return mTarget->isComposing();
    }

    return false;
}

/*!
\internal
\reimp
*/
QString HbInputContextProxy::language()
{
    if (mTarget) {
        return mTarget->language();
    }

    return QString();
}

/*!
\internal
\reimp
*/
void HbInputContextProxy::mouseHandler(int x, QMouseEvent *event)
{
    if (mTarget) {
        mTarget->mouseHandler(x, event);
    }
}

/*!
\internal
\reimp
*/
void HbInputContextProxy::reset()
{
    if (mTarget) {
        mTarget->reset();
    }
}

/*!
\internal
\reimp
*/
void HbInputContextProxy::sendEvent(const QInputMethodEvent &event)
{
    if (mTarget) {
        mTarget->sendEvent(event);
    }
}

/*!
\internal
\reimp
*/
void HbInputContextProxy::update()
{
    if (mTarget) {
        mTarget->update();
    }
}

/*!
\internal
\reimp
*/
void HbInputContextProxy::widgetDestroyed(QWidget *widget)
{
    if (mTarget) {
        mTarget->widgetDestroyed(widget);
    }
}

/*!
\internal
\reimp
*/
void HbInputContextProxy::setFocusWidget(QWidget *widget)
{
    if (mTarget) {
        mTarget->setFocusWidget(widget);
    }
}

// End of file