qtmobility/examples/qmlcontacts/qmlcontactmodel.cpp
changeset 14 6fbed849b4f4
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/qtmobility/examples/qmlcontacts/qmlcontactmodel.cpp	Wed Jun 23 19:08:38 2010 +0300
@@ -0,0 +1,226 @@
+/****************************************************************************
+**
+** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
+** All rights reserved.
+** Contact: Nokia Corporation (qt-info@nokia.com)
+**
+** This file is part of the Qt Mobility Components.
+**
+** $QT_BEGIN_LICENSE:BSD$
+** You may use this file under the terms of the BSD license as follows:
+**
+** "Redistribution and use in source and binary forms, with or without
+** modification, are permitted provided that the following conditions are
+** met:
+**   * Redistributions of source code must retain the above copyright
+**     notice, this list of conditions and the following disclaimer.
+**   * Redistributions in binary form must reproduce the above copyright
+**     notice, this list of conditions and the following disclaimer in
+**     the documentation and/or other materials provided with the
+**     distribution.
+**   * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor
+**     the names of its contributors may be used to endorse or promote
+**     products derived from this software without specific prior written
+**     permission.
+**
+** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+#include <qcontactdetails.h>
+
+#include "qmlcontactmodel.h"
+#include "qcontactmanager.h"
+#include "qcontactdetailfilter.h"
+#include "qversitreader.h"
+#include "qversitcontactimporter.h"
+
+#include <QColor>
+#include <QHash>
+#include <QDebug>
+#include <QPixmap>
+#include <QFile>
+
+QMLContactModel::QMLContactModel(QObject *parent) :
+    QAbstractListModel(parent),
+    m_manager(0)
+{
+    QHash<int, QByteArray> roleNames;
+    roleNames = QAbstractItemModel::roleNames();
+    roleNames.insert(InterestLabelRole, "interestLabel");
+    roleNames.insert(InterestRole, "interest");
+    roleNames.insert(AvatarRole, "avatar");
+    roleNames.insert(PresenceAvailableRole, "presenceSupported");
+    roleNames.insert(PresenceTextRole, "presenceText");
+    roleNames.insert(PresenceStateRole, "presenceState");
+    roleNames.insert(PresenceMessageRole, "presenceMessage");
+    setRoleNames(roleNames);
+
+    m_fetchHint.setOptimizationHints(QContactFetchHint::NoActionPreferences | QContactFetchHint::NoRelationships);
+    m_fetchHint.setDetailDefinitionsHint(QStringList()
+                                         << QContactPhoneNumber::DefinitionName
+                                         << QContactEmailAddress::DefinitionName
+                                         << QContactThumbnail::DefinitionName
+                                         << QContactAvatar::DefinitionName
+                                         << QContactGlobalPresence::DefinitionName
+                                         << QContactDisplayLabel::DefinitionName
+                                         << QContactOnlineAccount::DefinitionName);
+    m_sortOrder.setDetailDefinitionName(QContactDisplayLabel::DefinitionName, QContactDisplayLabel::FieldLabel);
+    m_sortOrder.setCaseSensitivity(Qt::CaseSensitive);
+
+    connect(&m_contactsRequest, SIGNAL(resultsAvailable()), this, SLOT(resultsReceived()));
+    m_contactsRequest.setFetchHint(m_fetchHint);
+    m_contactsRequest.setSorting(m_sortOrder);
+
+    QContactDetailFilter d;
+    d.setDetailDefinitionName(QContactType::DefinitionName, QContactType::FieldType);
+    d.setValue(QContactType::TypeContact);
+
+    m_contactsRequest.setFilter(d);
+
+    setManager(QString());
+}
+
+QStringList QMLContactModel::availableManagers() const
+{
+    return QContactManager::availableManagers();
+}
+
+QString QMLContactModel::manager()
+{
+    return m_manager->managerName();
+}
+
+void QMLContactModel::fillContactsIntoMemoryEngine(QContactManager* manager)
+{
+    QVersitReader reader;
+    QFile file(":/contents/example.vcf");
+    bool ok = file.open(QIODevice::ReadOnly);
+    if (ok) {
+       reader.setDevice(&file);
+       if (reader.startReading() && reader.waitForFinished()) {
+           QVersitContactImporter importer;
+           importer.importDocuments(reader.results());
+           QList<QContact> contacts = importer.contacts();
+           manager->saveContacts(&contacts, 0);
+       }
+    }
+}
+
+int QMLContactModel::rowCount(const QModelIndex &parent) const
+{
+    Q_UNUSED(parent);
+    return m_contacts.count();
+}
+
+void QMLContactModel::setManager(const QString& managerName)
+{
+    delete m_manager;
+    m_manager = new QContactManager(managerName);
+
+    if (managerName == "memory" && m_manager->contactIds().isEmpty()) {
+        fillContactsIntoMemoryEngine(m_manager);
+    }
+
+    qWarning() << "Changed backend to: " << managerName;
+    m_contactsRequest.setManager(m_manager);
+    connect(m_manager, SIGNAL(dataChanged()), this, SLOT(fetchAgain()));
+    fetchAgain();
+}
+
+void QMLContactModel::resultsReceived()
+{
+    int oldCount = m_contacts.count();
+    int newCount = m_contactsRequest.contacts().count();
+    if (newCount > oldCount) {
+        // Assuming the order is the same
+        beginInsertRows(QModelIndex(), newCount - oldCount, newCount);
+        m_contacts = m_contactsRequest.contacts();
+        endInsertRows();
+    } else {
+        // Hmm, shouldn't happen
+        reset();
+        beginInsertRows(QModelIndex(), 0, m_contactsRequest.contacts().count());
+        m_contacts =  m_contactsRequest.contacts();
+        endInsertRows();
+    }
+}
+
+void QMLContactModel::fetchAgain()
+{
+    m_contacts.clear();
+    reset();
+    m_contactsRequest.start();
+}
+
+QPair<QString, QString> QMLContactModel::interestingDetail(const QContact&c) const
+{
+    // Try a phone number, then email, then online account
+    // This does only check the first detail of each type
+    QContactDetail p = c.details<QContactPhoneNumber>().value(0);
+    if (!p.isEmpty())
+        return qMakePair(tr("Phone"), p.value(QContactPhoneNumber::FieldNumber));
+
+    p = c.details<QContactEmailAddress>().value(0);
+    if (!p.isEmpty())
+        return qMakePair(tr("Email"), p.value(QContactEmailAddress::FieldEmailAddress));
+
+    p = c.details<QContactOnlineAccount>().value(0);
+    if (!p.isEmpty())
+        return qMakePair(p.value(QContactOnlineAccount::FieldServiceProvider), p.value(QContactOnlineAccount::FieldAccountUri));
+
+    // Well, don't know.
+    return qMakePair(QString(), QString());
+}
+
+QVariant QMLContactModel::data(const QModelIndex &index, int role) const
+{
+    QContact c = m_contacts.value(index.row());
+    switch(role) {
+        case Qt::DisplayRole:
+            return c.displayLabel();
+        case InterestLabelRole:
+            return interestingDetail(c).first;
+        case InterestRole:
+            return interestingDetail(c).second;
+        case AvatarRole:
+            if (c.detail<QContactThumbnail>().isEmpty()) {
+                QContactAvatar a = c.detail<QContactAvatar>();
+                if (!a.imageUrl().isEmpty())
+                    return a.imageUrl();
+                else
+                    return QString("qrc:/default.svg");
+            } else {
+                // We have a thumbnail, so return empty
+                return QString("");
+            }
+        case Qt::DecorationRole:
+            {
+                QContactThumbnail t = c.detail<QContactThumbnail>();
+                if (!t.thumbnail().isNull())
+                    return QPixmap::fromImage(t.thumbnail());
+            }
+            return QPixmap();
+        case PresenceAvailableRole:
+            return !c.detail<QContactGlobalPresence>().isEmpty();
+        case PresenceMessageRole:
+            return c.detail<QContactGlobalPresence>().customMessage();
+        case PresenceTextRole:
+            return c.detail<QContactGlobalPresence>().presenceStateText();
+        case PresenceStateRole:
+            return c.detail<QContactGlobalPresence>().presenceState();
+    }
+    return QVariant();
+}
+