homescreenapp/hsdomainmodel/src/hswidgethost.cpp
branchRCL_3
changeset 82 5f0182e07bfb
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/homescreenapp/hsdomainmodel/src/hswidgethost.cpp	Tue Aug 31 15:06:34 2010 +0300
@@ -0,0 +1,776 @@
+/*
+* Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
+* All rights reserved.
+* This component and the accompanying materials are made available
+* under the terms of "Eclipse Public License v1.0"
+* which accompanies this distribution, and is available
+* at the URL "http://www.eclipse.org/legal/epl-v10.html".
+*
+* Initial Contributors:
+* Nokia Corporation - initial contribution.
+*
+* Contributors:
+*
+* Description:
+*
+*/
+
+#include <QApplication>
+#include <QStateMachine>
+#include <QState>
+#include <QFinalState>
+
+#include "hsdatabase.h"
+#include "hsdomainmodeldatastructures.h"
+#include "hsscene.h"
+#include "hspage.h"
+#include "hswidgethost.h"
+#include "hswidgethostvisual.h"
+#include "hshostedwidgetfactory.h"
+
+#include "hswidgetcomponentregistry.h"
+#include "hswidgetcomponent.h"
+#include "hsconfiguration.h"
+#include "hscontentservice.h"
+#include "hsgui.h"
+
+// Helper macros for connecting state entry and exit actions.
+#define ENTRY_ACTION(state, action) \
+    connect(state, SIGNAL(entered()), SLOT(action()));
+#define EXIT_ACTION(state, action) \
+    connect(state, SIGNAL(exited()), SLOT(action()));
+
+
+
+
+/*!
+    \class HsWidgetHost
+    \ingroup group_hsdomainmodel
+    \brief Each widget is controlled by the home screen framework through a widget host.
+*/
+
+/*!
+    Constructs a new widget host with given \a databaseId and
+    \a parent item.
+*/
+HsWidgetHost::HsWidgetHost(int databaseId, QObject *parent)
+  : QObject(parent),
+    mDatabaseId(databaseId),
+    mVisual(new HsWidgetHostVisual),
+    mStateMachine(0),
+    mWidget(0),
+    mPage(0),
+    mComponent(0),
+    mIsFinishing(false)
+{
+    setupStates();
+    mVisual->setVisualModel(this);
+}
+
+/*!
+    Destructor.
+*/
+HsWidgetHost::~HsWidgetHost()
+{
+    delete mVisual;
+}
+
+/*!
+    Creates a new widget host instance based on the give
+    \a widgetData and \a preferences. Returns the created
+    instance. Return 0 in failure cases.
+*/
+HsWidgetHost *HsWidgetHost::createInstance(HsWidgetData &widgetData, 
+                                           const QVariantHash &preferences)
+{
+    HsDatabase *db = HsDatabase::instance();
+
+    if (db->insertWidget(widgetData)) {
+        db->setWidgetPreferences(widgetData.id, preferences);
+        return new HsWidgetHost(widgetData.id);
+    } else {
+        return 0;
+    }
+}
+
+/*!
+    Returns the databaseId.
+*/
+int HsWidgetHost::databaseId() const
+{
+    return mDatabaseId;
+}
+
+/*!
+    Sets the containing \a page for this widget host.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::setPage(HsPage *page)
+{
+    HsDatabase *db = HsDatabase::instance();
+    
+    HsWidgetData data;
+    data.id = mDatabaseId;
+    if (db->widget(data)) {
+        if (!page) {
+            data.pageId = -1;
+        } else {
+            data.pageId = page->databaseId();
+        }
+        if (!db->updateWidget(data)) {
+            return false;
+        }
+    } else {
+        return false;
+    }
+
+    mPage = page;
+    
+    return true;
+}
+ 
+/*!
+    Returns the containing page for this widget. Returns 0 
+    if this widget has no containg page.
+*/
+HsPage *HsWidgetHost::page() const
+{
+    return mPage;
+}
+
+/*!
+    Loads presentation based on the current orientation.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::loadPresentation()
+{
+    return loadPresentation(HsGui::instance()->orientation());
+}
+
+/*!
+    Loads presentation based on the given \a orientation.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::loadPresentation(Qt::Orientation orientation)
+{
+    HsDatabase *db = HsDatabase::instance();
+        
+    HsWidgetPresentationData data;
+    data.orientation = orientation;
+    data.widgetId = mDatabaseId;
+    if (!db->widgetPresentation(data)) {
+        return false;
+    }
+    mVisual->setPos(data.x, data.y);
+    mVisual->setZValue(data.zValue);
+    return true;
+}
+
+/*!
+    Saves the current presentation.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::savePresentation()
+{
+    return savePresentation(HsGui::instance()->orientation());
+}
+
+/*!
+    Saves the current presentation for the given \a orientation.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::savePresentation(Qt::Orientation orientation)
+{
+    HsDatabase *db = HsDatabase::instance();
+        
+    HsWidgetPresentationData data;
+    data.orientation = orientation;
+    data.setPos(mVisual->pos());
+    data.zValue = mVisual->zValue();
+    data.widgetId = mDatabaseId;
+    return db->setWidgetPresentation(data);
+}
+
+/*!
+    Saves the given presentation.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::savePresentation(HsWidgetPresentationData &presentation)
+{
+    HsDatabase *db = HsDatabase::instance();
+    
+    presentation.widgetId = mDatabaseId;
+    return db->setWidgetPresentation(presentation);
+}
+
+/*!
+    Fills the \a presentation based on the orientation field in 
+    the given \a presentation. Returns true on success, otherwise 
+    returns false.
+*/
+bool HsWidgetHost::getPresentation(HsWidgetPresentationData &presentation)
+{
+    HsDatabase *db = HsDatabase::instance();
+        
+    presentation.widgetId = mDatabaseId;
+    return db->widgetPresentation(presentation);
+}
+
+/*!
+    Removes the presentation for the given \a orientation.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::removePresentation(Qt::Orientation orientation)
+{
+    HsDatabase *db = HsDatabase::instance();
+    return db->deleteWidgetPresentation(mDatabaseId, orientation);
+}
+
+HsWidgetHostVisual *HsWidgetHost::visual() const
+{
+    return mVisual;
+}
+
+/*!
+    \fn HsWidgetHost::event_startAndShow()
+    Initiates a transition to show state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_startAndHide()
+    Initiates a transition to hide state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_unload()
+    Initiates a transition to unloaded state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_show()
+    Initiates a transition to show state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_hide()
+    Initiates a transition to hide state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_remove()
+    Initiates a transition to remove state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_close()
+    Initiates a transition to final state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_finished()
+    Initiates a transition to finished state.
+*/
+
+/*!
+    \fn HsWidgetHost::event_faulted()
+    Initiates a transition to faulted state.
+*/
+
+/*!
+    \fn HsWidgetHost::finished()
+    Notifies the home screen framework that this widget
+    host has moved to finished state.
+*/
+
+/*!
+    \fn HsWidgetHost::faulted()
+    Notifies the home screen framework that this widget
+    host has moved to faulted state.
+*/
+
+
+
+/*!
+    \fn HsWidgetHost::available()
+    Notifies the home screen framework that this widget
+    is now available.
+*/
+
+/*!
+    \fn HsWidgetHost::unavailable()
+    Notifies the home screen framework that this widget
+    is temporarily unavailable.
+*/
+
+
+/*!
+    Starts the widget. The \a show parameter defines the
+    visibility of the widget.
+*/
+void HsWidgetHost::startWidget(bool show)
+{
+    if (!mStateMachine->isRunning()) {
+        mStateMachine->start();
+        // This is needed because QStateMachine::start() starts
+        // the state machine asynchronously via the eventloop. 
+        // Here we want the machine to start synchronously.
+        QApplication::processEvents(QEventLoop::ExcludeUserInputEvents);
+    }
+
+    if (show) {
+        emit event_startAndShow();
+    } else {
+        emit event_startAndHide();
+    }
+}
+ 
+/*!
+    Puts the contained widget into show state.
+*/
+void HsWidgetHost::showWidget()
+{
+    emit event_show();
+}
+  
+/*!
+    Puts the contained widget into hidden state.
+*/
+void HsWidgetHost::hideWidget()
+{
+    emit event_hide();
+}
+
+/*!
+    Notifies the conained widget about \a online
+    status changes.
+*/
+void HsWidgetHost::setOnline(bool online)
+{
+    mIsOnlineProperty.write(mWidget, online);
+}
+
+/*!
+    Deletes this widget instance, including the 
+    database entries.
+*/
+void HsWidgetHost::remove()
+{
+    if (mStateMachine->isRunning()) {
+        emit event_remove();
+    } else {
+        action_remove();
+        deleteLater();
+    }
+}
+ 
+/*!
+    Deletes this widget instance, leaving the 
+    database entries untouched.
+*/
+void HsWidgetHost::close()
+{
+    if (mStateMachine->isRunning()) {
+        emit event_close();
+    } else {
+        deleteLater();
+    }
+}
+
+/*!
+    Starts the drag effect.
+*/
+void HsWidgetHost::startDragEffect()
+{
+   mVisual->startDragEffect();
+}
+
+/*!
+    Starts the drop effect.
+*/
+void HsWidgetHost::startDropEffect()
+{
+    mVisual->startDropEffect();
+}
+
+/*!
+    Configures the state machine for this widget host.
+*/
+void HsWidgetHost::setupStates()
+{
+    // State machine
+
+    mStateMachine = new QStateMachine(this);
+    mStateMachine->setAnimated(false);
+    
+    // States
+
+    QState *state_component = new QState;
+    QState *state_unloaded = new QState(state_component);
+    QState *state_running = new QState(state_component);
+    QState *state_show = new QState(state_running);
+    QState *state_hide = new QState(state_running);
+    QState *state_finished = new QState;
+    QState *state_faulted = new QState;
+    QState *state_remove = new QState;
+    QFinalState *state_final = new QFinalState;
+
+    mStateMachine->addState(state_component);
+    mStateMachine->addState(state_finished);
+    mStateMachine->addState(state_faulted);
+    mStateMachine->addState(state_remove);
+    mStateMachine->addState(state_final);
+
+    mStateMachine->setInitialState(state_component);
+    state_component->setInitialState(state_unloaded);
+    state_running->setInitialState(state_hide);
+
+    // Transitions
+
+    state_component->addTransition(
+        this, SIGNAL(event_close()), state_final);
+    state_component->addTransition(
+        this, SIGNAL(event_remove()), state_remove);
+    state_component->addTransition(
+        this, SIGNAL(event_finished()), state_finished);
+    state_component->addTransition(
+        this, SIGNAL(event_faulted()), state_faulted);
+
+    state_unloaded->addTransition(
+        this, SIGNAL(event_startAndShow()), state_show);
+    state_unloaded->addTransition(
+        this, SIGNAL(event_startAndHide()), state_hide);
+
+    state_running->addTransition(
+        this, SIGNAL(event_unload()), state_unloaded);
+    
+    state_show->addTransition(
+        this, SIGNAL(event_hide()), state_hide);
+
+    state_hide->addTransition(
+        this, SIGNAL(event_show()), state_show);
+
+    state_finished->addTransition(
+        this, SIGNAL(event_remove()), state_remove);
+    state_finished->addTransition(
+        this, SIGNAL(event_close()), state_final);
+
+    state_faulted->addTransition(
+        this, SIGNAL(event_remove()), state_remove);
+    state_faulted->addTransition(
+        this, SIGNAL(event_close()), state_final);
+
+    state_remove->addTransition(state_final);
+
+    // Actions
+
+    ENTRY_ACTION(state_component, action_connectComponent)
+    EXIT_ACTION(state_component, action_disconnectComponent)
+
+    ENTRY_ACTION(state_running, action_load)
+    ENTRY_ACTION(state_running, action_initialize)
+    EXIT_ACTION(state_running, action_uninitialize)
+    EXIT_ACTION(state_running, action_unload)
+
+    ENTRY_ACTION(state_show, action_show)
+
+    ENTRY_ACTION(state_hide, action_hide)
+
+    ENTRY_ACTION(state_finished, action_finished)
+
+    ENTRY_ACTION(state_faulted, action_faulted)
+
+    ENTRY_ACTION(state_remove, action_notifyRemove)
+    ENTRY_ACTION(state_remove, action_remove)
+
+    // Delete on finish.
+
+    connect(mStateMachine, SIGNAL(finished()), SLOT(deleteLater()), 
+            Qt::QueuedConnection);
+}
+
+/*!
+    Assigns the meta \a property based on the given property \a name.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::setProperty(const char *name, QMetaProperty &property)
+{
+    const QMetaObject *object = mWidget->metaObject();
+    int index = object->indexOfProperty(name);
+    property = object->property(index);
+    return index >= 0;
+}
+
+/*!
+    Assigns the meta \a method based on the given method \a signature.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::setMethod(const char *signature, QMetaMethod &method)
+{
+    const QMetaObject *object = mWidget->metaObject();
+    int index = object->indexOfMethod(
+        QMetaObject::normalizedSignature(signature));
+    method = object->method(index);
+    return index >= 0;
+}
+
+/*!
+    Checks if the contained widget has implemented the
+    signal with the given \a signature.
+*/
+bool HsWidgetHost::hasSignal(const char *signature)
+{
+    const QMetaObject *object = mWidget->metaObject();
+    int index = object->indexOfSignal(
+        QMetaObject::normalizedSignature(signature));
+    return index >= 0;
+}
+
+
+/*!
+    Assigns preferences for the contained widget.
+    Returns true on success, otherwise returns false.
+*/
+bool HsWidgetHost::setPreferencesToWidget()
+{
+    HsDatabase *db = HsDatabase::instance();
+    
+    QVariantHash preferences;
+    if (!db->widgetPreferences(mDatabaseId, preferences)) {
+        return false;
+    }
+
+    QStringList names = preferences.keys();
+    foreach (QString name, names) {
+        mWidget->setProperty(name.toLatin1(),
+                             preferences.value(name));
+    }
+
+    return true;
+}
+
+/*!
+    Connects this widget host for monitoring changes
+    in its widget component.
+*/
+void HsWidgetHost::action_connectComponent()
+{
+    HsDatabase *db = HsDatabase::instance();
+    
+    HsWidgetData data;
+    data.id = mDatabaseId;
+    if (!db->widget(data)) {
+        emit event_faulted();
+        return;
+    }
+    
+    mComponent = HsWidgetComponentRegistry::instance()->component(data.uri);
+    
+    connect(mComponent, SIGNAL(aboutToUninstall()), SIGNAL(event_unload()));
+    connect(mComponent, SIGNAL(uninstalled()), SIGNAL(event_finished()));
+    connect(mComponent, SIGNAL(unavailable()), SIGNAL(event_unload()));
+    connect(mComponent, SIGNAL(unavailable()), SIGNAL(unavailable()));
+    connect(mComponent, SIGNAL(available()), SIGNAL(available()));
+    connect(mComponent, SIGNAL(updated()), SIGNAL(available()));
+}
+
+/*!
+    Disconnects this widget host from its widget component.
+*/
+void HsWidgetHost::action_disconnectComponent()
+{
+    mComponent->disconnect(this);
+}
+
+/*!
+    Loads the contained widget from a widget plugin and 
+    connects to it's meta interface.
+*/
+void HsWidgetHost::action_load()
+{
+    mWidget = HsHostedWidgetFactory::instance()->createWidget(mComponent->uri());
+    if (!mWidget) {
+        emit event_faulted();
+        return;
+    }
+    // must have fuctions
+    if (!setMethod("onShow()", mOnShowMethod) || 
+        !setMethod("onHide()", mOnHideMethod)) {
+        delete mWidget;
+        mWidget = 0;
+        emit event_faulted();
+        return;
+    }
+    
+    mVisual->setWidget(mWidget);
+
+    setMethod("onInitialize()", mOnInitializeMethod);
+    setMethod("onUninitialize()", mOnUninitializeMethod);
+    
+    setProperty("isOnline", mIsOnlineProperty);
+	setProperty("rootPath", mRootPathProperty);
+    
+    if (hasSignal("setPreferences(const QStringList&)")) {
+        connect(mWidget, SIGNAL(setPreferences(QStringList)),
+            SLOT(onSetPreferences(QStringList)), Qt::QueuedConnection);
+    }
+    if (hasSignal("finished()")) {
+        connect(mWidget, SIGNAL(finished()),
+            SLOT(onFinished()), Qt::QueuedConnection);
+    }
+    if (hasSignal("error()")) {
+        connect(mWidget, SIGNAL(error()),
+            SLOT(onError()), Qt::QueuedConnection);
+    }
+
+    loadPresentation();
+   
+    QString objName(mComponent->uri());
+    objName.append(":");
+    objName.append(QString::number(mDatabaseId));
+    mVisual->setObjectName(objName);
+}
+
+/*!
+    Unloads the contained widget. 
+*/
+void HsWidgetHost::action_unload()
+{
+    mVisual->setWidget(0);
+    delete mWidget;
+    mWidget = 0;
+		// This is needed because QServicePluginCleanup is 
+    // deleted asynchronously via the eventloop (deleteLater). 
+    // Here we want the plugin to unload synchronously.
+    QCoreApplication::sendPostedEvents(0, QEvent::DeferredDelete);
+    	
+    mOnInitializeMethod = QMetaMethod();
+    mOnShowMethod = QMetaMethod();
+    mOnHideMethod = QMetaMethod();
+    mOnUninitializeMethod = QMetaMethod();    
+    mIsOnlineProperty = QMetaProperty();
+	mRootPathProperty = QMetaProperty();    
+}
+
+/*!
+    Initializes the contained widget.
+*/
+void HsWidgetHost::action_initialize()
+{    
+    mRootPathProperty.write(mWidget, mComponent->rootPath());
+    setPreferencesToWidget();
+    setOnline(HsScene::instance()->isOnline());
+    mOnInitializeMethod.invoke(mWidget);
+}
+
+/*!
+    Uninitializes the contained widget.
+*/
+void HsWidgetHost::action_uninitialize()
+{
+    mOnUninitializeMethod.invoke(mWidget);
+}
+
+/*!
+    Puts the contained widget into show state.
+*/
+void HsWidgetHost::action_show()
+{
+    if (!mIsFinishing) {
+        mOnShowMethod.invoke(mWidget);
+    }
+}
+
+/*!
+    Puts the contained widget into hidden state.
+*/
+void HsWidgetHost::action_hide()
+{
+    if (!mIsFinishing) {
+        mOnHideMethod.invoke(mWidget);
+    }
+}
+
+/*!
+    Notifies the home screen framework that this widget
+    host has moved to finished state.
+*/
+void HsWidgetHost::action_finished()
+{
+    emit finished();
+}
+
+/*!
+    Notifies the home screen framework that this widget
+    host has moved to faulted state.
+*/
+void HsWidgetHost::action_faulted()
+{
+    emit faulted();
+}
+
+/*!
+    Removes the contained widget from the home screen
+    database.
+*/
+void HsWidgetHost::action_remove()
+{
+    HsDatabase *db = HsDatabase::instance();
+    
+    db->deleteWidget(mDatabaseId);
+    mDatabaseId = -1;
+}
+
+/*!
+    Notifies the widget removal through the content service.
+*/
+void HsWidgetHost::action_notifyRemove()
+{
+    HsDatabase *db = HsDatabase::instance();
+    QVariantHash preferences;
+    db->widgetPreferences(mDatabaseId, preferences);
+    HsContentService::instance()->emitWidgetRemoved(mComponent->uri(), preferences);
+}
+
+/*!
+    Handles contained widget's finished event.
+    Moves this widget host to finished state.
+*/
+void HsWidgetHost::onFinished()
+{
+    mIsFinishing = true;
+    emit event_finished();
+}
+ 
+/*!
+    Handles contained widget's error event.
+    Moves this widget host to faulted state.
+*/
+void HsWidgetHost::onError()
+{
+    mIsFinishing = true;
+    emit event_faulted();
+}
+
+/*!
+    Handles contained widget's setPreferences event.
+    Stores the preferences for the given \a names to
+    the home screen database.
+*/
+void HsWidgetHost::onSetPreferences(const QStringList &names)
+{
+    if (names.isEmpty()) {
+        return;
+    }
+
+    QVariantHash preferences;
+
+    foreach (QString name, names) {
+        QVariant value = mWidget->property(name.toLatin1());
+        preferences.insert(name, value);
+    }
+
+    HsDatabase *db = HsDatabase::instance();    
+    if (!db->setWidgetPreferences(mDatabaseId, preferences)) {
+        onError();
+    }
+}