/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** All rights reserved. ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** This file is part of the Symbian application wrapper 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 FILES #include #include #ifdef Q_WS_S60 #include #include #include #include #endif #include #include #include #include "qs60mainappui.h" #include #include #include #include #include #include #include "qs60keycapture_p.h" //Animated wallpapers in Qt applications are not supported. const TInt KAknDisableAnimationBackground = 0x02000000; const TInt KAknSingleClickCompatible = 0x01000000; QT_BEGIN_NAMESPACE static QS60KeyCapture *qt_S60KeyCapture = 0; static void installS60KeyCapture(CCoeEnv *env) { if (QApplication::testAttribute(Qt::AA_CaptureMultimediaKeys)) qt_S60KeyCapture = new QS60KeyCapture(env); } static void removeS60KeyCapture() { delete qt_S60KeyCapture; qt_S60KeyCapture = 0; } /*! \class QS60MainAppUi \since 4.6 \brief The QS60MainAppUi class is a helper class for S60 migration. \warning This class is provided only to get access to S60 specific functionality in the application framework classes. It is not portable. We strongly recommend against using it in new applications. The QS60MainAppUi provides a helper class for use in migrating from existing S60 based applications to Qt based applications. It is used in the exact same way as the \c CAknAppUi class from Symbian, but internally provides extensions used by Qt. When modifying old S60 applications that rely on implementing functions in \c CAknAppUi, the class should be modified to inherit from this class instead of \c CAknAppUi. Then the application can choose to override only certain functions. For more information on \c CAknAppUi, please see the S60 documentation. Unlike other Qt classes, QS60MainAppUi behaves like an S60 class, and can throw Symbian leaves. \sa QS60MainDocument, QS60MainApplication */ /*! * \brief Second phase Symbian constructor. * * Constructs all the elements of the class that can cause a leave to happen. * * If you override this function, you should call the base class implementation as well. */ void QS60MainAppUi::ConstructL() { // Cone's heap and handle checks on app destruction are not suitable for Qt apps, as many // objects can still exist in static data at that point. Instead we will print relevant information // so that comparative checks may be made for memory leaks, using ~SPrintExitInfo in corelib. iEikonEnv->DisableExitChecks(ETrue); // Initialise app UI with standard value. // ENoAppResourceFile and ENonStandardResourceFile makes UI to work without // resource files in most SDKs. S60 3rd FP1 public seems to require resource file // even these flags are defined TInt flags = CEikAppUi::ENoScreenFurniture | CEikAppUi::ENonStandardResourceFile; #ifdef Q_WS_S60 flags |= CAknAppUi::EAknEnableSkin; // After 5th Edition S60, native side supports animated wallpapers. // However, there is no support for that feature on Qt side, so indicate to // native UI framework that this application will not support background animations. // Also, add support for single touch for post 5th edition platforms. // This has only impact when launching native dialogs/menus from inside QApplication. if (QSysInfo::s60Version() > QSysInfo::SV_S60_5_0) { flags |= (KAknDisableAnimationBackground | KAknSingleClickCompatible); } #endif BaseConstructL(flags); installS60KeyCapture(iCoeEnv); } /*! * \brief Contructs an instance of QS60MainAppUi. */ QS60MainAppUi::QS60MainAppUi() { // No implementation required } /*! * \brief Destroys the QS60MainAppUi. */ QS60MainAppUi::~QS60MainAppUi() { removeS60KeyCapture(); } /*! * \brief Handles commands produced by the S60 framework. * * \a command holds the ID of the command to handle, and is S60 specific. * * If you override this function, you should call the base class implementation if you do not * handle the command. */ void QS60MainAppUi::HandleCommandL(TInt command) { if (qApp) { QSymbianEvent event(QSymbianEvent::CommandEvent, command); QT_TRYCATCH_LEAVING(qApp->symbianProcessEvent(&event)); } } /*! * \brief Handles a resource change in the S60 framework. * * Resource changes include layout switches. \a type holds the type of resource change that * occurred. * * If you override this function, you should call the base class implementation if you do not * handle the resource change. */ void QS60MainAppUi::HandleResourceChangeL(TInt type) { QS60MainAppUiBase::HandleResourceChangeL(type); if (qApp) { QSymbianEvent event(QSymbianEvent::ResourceChangeEvent, type); QT_TRYCATCH_LEAVING(qApp->symbianProcessEvent(&event)); } } /*! * \brief Handles raw window server events. * * The event type and information is passed in \a wsEvent, while the receiving control is passed in * \a destination. * * If you override this function, you should call the base class implementation if you do not * handle the event. */ void QS60MainAppUi::HandleWsEventL(const TWsEvent &wsEvent, CCoeControl *destination) { int result = 0; if (qApp) { QSymbianEvent event(&wsEvent); QT_TRYCATCH_LEAVING( result = qApp->symbianProcessEvent(&event) ); } if (result <= 0) QS60MainAppUiBase::HandleWsEventL(wsEvent, destination); } /*! * \brief Handles changes to the status pane size. * * Called by the framework when the application status pane size is changed. * * If you override this function, you should call the base class implementation if you do not * handle the size change. */ void QS60MainAppUi::HandleStatusPaneSizeChange() { TRAP_IGNORE(HandleResourceChangeL(KInternalStatusPaneChange)); HandleStackedControlsResourceChange(KInternalStatusPaneChange); } /*! * \brief Dynamically initializes a menu bar. * * The resource associated with the menu is given in \a resourceId, and the actual menu bar is * passed in \a menuBar. * * If you override this function, you should call the base class implementation as well. */ void QS60MainAppUi::DynInitMenuBarL(TInt /* resourceId */, CEikMenuBar * /* menuBar */) { } /*! * \brief Dynamically initializes a menu pane. * * The resource associated with the menu is given in \a resourceId, and the actual menu pane is * passed in \a menuPane. * * If you override this function, you should call the base class implementation as well. */ void QS60MainAppUi::DynInitMenuPaneL(TInt resourceId, CEikMenuPane *menuPane) { #ifdef Q_WS_S60 if (resourceId == R_AVKON_MENUPANE_EMPTY) { // As ccpu is enabled in the editors, default native menu items are now <=2 if (menuPane->NumberOfItemsInPane() <= 2) QT_TRYCATCH_LEAVING(qt_symbian_show_toplevel(menuPane)); } else if (resourceId != R_AVKON_MENUPANE_FEP_DEFAULT && resourceId != R_AVKON_MENUPANE_EDITTEXT_DEFAULT && resourceId != R_AVKON_MENUPANE_LANGUAGE_DEFAULT) { QT_TRYCATCH_LEAVING(qt_symbian_show_submenu(menuPane, resourceId)); } #else QS60MainAppUiBase::DynInitMenuPaneL(resourceId, menuPane); #endif } /*! * \brief Restores a menu window. * * The menu window to restore is given in \a menuWindow. The resource ID and type of menu is given * in \a resourceId and \a menuType, respectively. * * If you override this function, you should call the base class implementation as well. */ void QS60MainAppUi::RestoreMenuL(CCoeControl *menuWindow, TInt resourceId, TMenuType menuType) { #ifdef Q_WS_S60 if (resourceId >= QT_SYMBIAN_FIRST_MENU_ITEM && resourceId <= QT_SYMBIAN_LAST_MENU_ITEM) { if (menuType == EMenuPane) DynInitMenuPaneL(resourceId, (CEikMenuPane*)menuWindow); else DynInitMenuBarL(resourceId, (CEikMenuBar*)menuWindow); } else if(resourceId == R_AVKON_MENUPANE_EMPTY) { CEikMenuBarTitle *title = new(ELeave) CEikMenuBarTitle; CleanupStack::PushL(title); title->iData.iMenuPaneResourceId = R_AVKON_MENUPANE_EMPTY; title->iTitleFlags = 0; S60->menuBar()->TitleArray()->AddTitleL(title); CleanupStack::Pop( title ); } else #endif { QS60MainAppUiBase::RestoreMenuL(menuWindow, resourceId, menuType); } } /*! \internal */ void QS60MainAppUi::Exit() { QS60MainAppUiBase::Exit(); } /*! \internal */ void QS60MainAppUi::SetFadedL(TBool aFaded) { QS60MainAppUiBase::SetFadedL(aFaded); } /*! \internal */ TRect QS60MainAppUi::ApplicationRect() const { return QS60MainAppUiBase::ApplicationRect(); } /*! \internal */ void QS60MainAppUi::HandleScreenDeviceChangedL() { // This function triggers AppUi relayout which also generates // HandleStatusPaneSizeChange(). We don't want to handle // status pane resizes at this point because it causes // Qt window resize and thus EGL surface resize in the middle of // incomplete layout process causing unnecessary overhead. // To prevent status pane resize handling while layout is still // in progress, we guard relayouting with handleStatusPaneResizeNotifications // flag. QSymbianControl checks this flag before doing Qt window // resize due to status pane change. // Eventually when layout is ready, Symbian framework calls // HandleResourceChangeL(KEikDynamicLayoutVariantSwitch) which triggers // resize to Qt window and to its EGL surface. S60->handleStatusPaneResizeNotifications = false; QS60MainAppUiBase::HandleScreenDeviceChangedL(); S60->handleStatusPaneResizeNotifications = true; } /*! \internal */ void QS60MainAppUi::HandleApplicationSpecificEventL(TInt aType, const TWsEvent &aEvent) { QS60MainAppUiBase::HandleApplicationSpecificEventL(aType, aEvent); } /*! \internal */ TTypeUid::Ptr QS60MainAppUi::MopSupplyObject(TTypeUid aId) { return QS60MainAppUiBase::MopSupplyObject(aId); } /*! \internal */ void QS60MainAppUi::ProcessCommandL(TInt aCommand) { QS60MainAppUiBase::ProcessCommandL(aCommand); } /*! \internal */ TErrorHandlerResponse QS60MainAppUi::HandleError (TInt aError, const SExtendedError &aExtErr, TDes &aErrorText, TDes &aContextText) { return QS60MainAppUiBase::HandleError(aError, aExtErr, aErrorText, aContextText); } /*! \internal */ void QS60MainAppUi::HandleViewDeactivation(const TVwsViewId &aViewIdToBeDeactivated, const TVwsViewId &aNewlyActivatedViewId) { QS60MainAppUiBase::HandleViewDeactivation(aViewIdToBeDeactivated, aNewlyActivatedViewId); } /*! \internal */ void QS60MainAppUi::PrepareToExit() { QS60MainAppUiBase::PrepareToExit(); } /*! \internal */ void QS60MainAppUi::HandleTouchPaneSizeChange() { QS60MainAppUiBase::HandleTouchPaneSizeChange(); } /*! \internal */ void QS60MainAppUi::HandleSystemEventL(const TWsEvent &aEvent) { QS60MainAppUiBase::HandleSystemEventL(aEvent); } /*! \internal */ void QS60MainAppUi::Reserved_MtsmPosition() { QS60MainAppUiBase::Reserved_MtsmPosition(); } /*! \internal */ void QS60MainAppUi::Reserved_MtsmObject() { QS60MainAppUiBase::Reserved_MtsmObject(); } /*! \internal */ void QS60MainAppUi::HandleForegroundEventL(TBool aForeground) { QS60MainAppUiBase::HandleForegroundEventL(aForeground); } /*! \internal */ TBool QS60MainAppUi::ProcessCommandParametersL(TApaCommand /*aCommand*/, TFileName &/*aDocumentName*/, const TDesC8 &/*aTail*/) { // bypass CEikAppUi::ProcessCommandParametersL(..) which modifies aDocumentName, preventing apparc document opening from working. // The return value is effectively unused in Qt apps (see QS60MainDocument::OpenFileL) return EFalse; } #ifndef Q_WS_S60 void QS60StubAknAppUi::HandleViewDeactivation(const TVwsViewId &, const TVwsViewId &) {} void QS60StubAknAppUi::HandleTouchPaneSizeChange() {} void QS60StubAknAppUi::HandleStatusPaneSizeChange() {} void QS60StubAknAppUi::Reserved_MtsmPosition() {} void QS60StubAknAppUi::Reserved_MtsmObject() {} #endif QT_END_NAMESPACE