Skip to content
Snippets Groups Projects
Select Git revision
  • ee100f58996c0b7fa2305c64f5a9b9034458a235
  • master default protected
  • release/202005
  • release/202001
  • release/201912
  • release/201911
  • release/releaseWindowsTestOne
  • release/windowsReleaseTest
  • release/releaseTest
  • release/releaseWindowsTest
  • release/201910
  • release/qt/201910
  • release/windows-test/201910
  • release/201908
  • release/201906
  • release/201905
  • release/201904
  • release/201903
  • release/201902
  • release/201901
  • release/201812
  • 4.0.0
  • 2.2.0
  • 2.1.0
  • 2.0.1
  • 2.0.0
  • 1.4.1
  • 1.4.0
  • 1.3.0
  • 1.2.0
  • 1.1.0
31 results

sipaccountbase.cpp

Blame
  • Code owners
    Assign users and groups as approvers for specific file changes. Learn more.
    smartlistmodel.cpp 14.52 KiB
    /*
     * Copyright (C) 2017-2020 by Savoir-faire Linux
     * Author: Anthony Léonard <anthony.leonard@savoirfairelinux.com>
     * Author: Andreas Traczyk <andreas.traczyk@savoirfairelinux.com>
     * Author: Mingrui Zhang <mingrui.zhang@savoirfairelinux.com>
     *
     * This program is free software; you can redistribute it and/or modify
     * it under the terms of the GNU General Public License as published by
     * the Free Software Foundation; either version 3 of the License, or
     * (at your option) any later version.
     *
     * This program is distributed in the hope that it will be useful,
     * but WITHOUT ANY WARRANTY; without even the implied warranty of
     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     * GNU General Public License for more details.
     *
     * You should have received a copy of the GNU General Public License
     * along with this program.  If not, see <http://www.gnu.org/licenses/>.
     */
    
    #include "smartlistmodel.h"
    
    #include "lrcinstance.h"
    #include "pixbufmanipulator.h"
    #include "utils.h"
    
    #include "api/contactmodel.h"
    #include "globalinstances.h"
    
    #include <QDateTime>
    
    SmartListModel::SmartListModel(const QString &accId,
                                   QObject *parent,
                                   SmartListModel::Type listModelType,
                                   const QString &convUid)
        : QAbstractListModel(parent)
        , accountId_(accId)
        , listModelType_(listModelType)
        , convUid_(convUid)
    {
        if (listModelType_ == Type::CONFERENCE) {
            setConferenceableFilter();
        }
    }
    
    SmartListModel::~SmartListModel() {}
    
    int
    SmartListModel::rowCount(const QModelIndex &parent) const
    {
        if (!parent.isValid()) {
            auto &accInfo = LRCInstance::accountModel().getAccountInfo(accountId_);
            auto &convModel = accInfo.conversationModel;
            if (listModelType_ == Type::TRANSFER) {
                auto filterType = accInfo.profileInfo.type;
                return convModel->getFilteredConversations(filterType).size();
            } else if (listModelType_ == Type::CONFERENCE) {
                auto calls = conferenceables_[ConferenceableItem::CALL];
                auto contacts = conferenceables_[ConferenceableItem::CONTACT];
                auto rowCount = contacts.size();
                if (calls.size()) {
                    rowCount = 2;
                    rowCount += sectionState_[tr("Calls")] ? calls.size() : 0;
                    rowCount += sectionState_[tr("Contacts")] ? contacts.size() : 0;
                }
                return rowCount;
            }
            return conversations_.size();
        }
        return 0;
    }
    
    int
    SmartListModel::columnCount(const QModelIndex &parent) const
    {
        Q_UNUSED(parent);
        return 1;
    }
    
    QVariant
    SmartListModel::data(const QModelIndex &index, int role) const
    {
        if (!index.isValid()) {
            return QVariant();
        }
    
        try {
            auto &accountInfo = LRCInstance::accountModel().getAccountInfo(accountId_);
            auto &convModel = accountInfo.conversationModel;
            lrc::api::conversation::Info item;
            if (listModelType_ == Type::TRANSFER) {
                auto filterType = accountInfo.profileInfo.type;
                item = convModel->getFilteredConversations(filterType).at(index.row());
                return getConversationItemData(item, accountInfo, role);
            } else if (listModelType_ == Type::CONFERENCE) {
                auto calls = conferenceables_[ConferenceableItem::CALL];
                auto contacts = conferenceables_[ConferenceableItem::CONTACT];
                QString itemConvUid{}, itemAccId{};
                if (calls.size() == 0) {
                    itemConvUid = contacts.at(index.row()).at(0).convId;
                    itemAccId = contacts.at(index.row()).at(0).accountId;
                } else {
                    bool callsOpen = sectionState_[tr("Calls")];
                    bool contactsOpen = sectionState_[tr("Contacts")];
                    auto callSectionEnd = callsOpen ? calls.size() + 1 : 1;
                    auto contactSectionEnd = contactsOpen ? callSectionEnd + contacts.size() + 1
                                                          : callSectionEnd + 1;
                    if (index.row() < callSectionEnd) {
                        if (index.row() == 0) {
                            return QVariant(role == Role::SectionName
                                                ? (callsOpen ? "➖ " : "➕ ") + QString(tr("Calls"))
                                                : "");
                        } else {
                            auto idx = index.row() - 1;
                            itemConvUid = calls.at(idx).at(0).convId;
                            itemAccId = calls.at(idx).at(0).accountId;
                        }
                    } else if (index.row() < contactSectionEnd) {
                        if (index.row() == callSectionEnd) {
                            return QVariant(role == Role::SectionName
                                                ? (contactsOpen ? "➖ " : "➕ ") + QString(tr("Contacts"))
                                                : "");
                        } else {
                            auto idx = index.row() - (callSectionEnd + 1);
                            itemConvUid = contacts.at(idx).at(0).convId;
                            itemAccId = contacts.at(idx).at(0).accountId;
                        }
                    }
                }
                if (role == Role::AccountId) {
                    return QVariant(itemAccId);
                }
                auto &itemAccountInfo = LRCInstance::accountModel().getAccountInfo(itemAccId);
                item = itemAccountInfo.conversationModel->getConversationForUID(itemConvUid);
                return getConversationItemData(item, itemAccountInfo, role);
            } else if (listModelType_ == Type::CONVERSATION) {
                item = conversations_.at(index.row());
                return getConversationItemData(item, accountInfo, role);
            }
        } catch (const std::exception &e) {
            qWarning() << e.what();
        }
        return QVariant();
    }
    
    QHash<int, QByteArray>
    SmartListModel::roleNames() const
    {
        QHash<int, QByteArray> roles;
        roles[DisplayName] = "DisplayName";
        roles[DisplayID] = "DisplayID";
        roles[Picture] = "Picture";
        roles[Presence] = "Presence";
        roles[URI] = "URI";
        roles[UnreadMessagesCount] = "UnreadMessagesCount";
        roles[LastInteractionDate] = "LastInteractionDate";
        roles[LastInteraction] = "LastInteraction";
        roles[ContactType] = "ContactType";
        roles[UID] = "UID";
        roles[InCall] = "InCall";
        roles[IsAudioOnly] = "IsAudioOnly";
        roles[CallStackViewShouldShow] = "CallStackViewShouldShow";
        roles[CallStateStr] = "CallStateStr";
        roles[SectionName] = "SectionName";
        roles[AccountId] = "AccountId";
        roles[Draft] = "Draft";
        return roles;
    }
    
    void
    SmartListModel::setConferenceableFilter(const QString &filter)
    {
        beginResetModel();
        auto &accountInfo = LRCInstance::accountModel().getAccountInfo(accountId_);
        auto &convModel = accountInfo.conversationModel;
        conferenceables_ = convModel->getConferenceableConversations(convUid_, filter);
        sectionState_[tr("Calls")] = true;
        sectionState_[tr("Contacts")] = true;
        endResetModel();
    }
    
    void
    SmartListModel::fillConversationsList()
    {
        beginResetModel();
        auto* convModel = LRCInstance::getCurrentConversationModel();
        conversations_.clear();
    
        for (auto convSearch : convModel->getAllSearchResults()) {
            conversations_.emplace_back(convSearch);
        }
    
        for (auto convFilt : convModel->allFilteredConversations()) {
            conversations_.emplace_back(convFilt);
        }
        endResetModel();
    }
    
    void
    SmartListModel::updateConversation(const QString &convUid)
    {
        auto* convModel = LRCInstance::getCurrentConversationModel();
        for (lrc::api::conversation::Info &conversation : conversations_) {
            if (conversation.uid == convUid) {
                conversation = convModel->getConversationForUID(convUid);
                return;
            }
        }
    }
    
    void
    SmartListModel::toggleSection(const QString &section)
    {
        beginResetModel();
        if (section.contains(tr("Calls"))) {
            sectionState_[tr("Calls")] ^= true;
        } else if (section.contains(tr("Contacts"))) {
            sectionState_[tr("Contacts")] ^= true;
        }
        endResetModel();
    }
    
    int
    SmartListModel::currentUidSmartListModelIndex()
    {
        const auto convUid = LRCInstance::getCurrentConvUid();
        for (int i = 0; i < rowCount(); i++) {
            if (convUid == data(index(i, 0), Role::UID))
                return i;
        }
    
        return -1;
    }
    
    QVariant
    SmartListModel::getConversationItemData(const conversation::Info &item,
                                            const account::Info &accountInfo,
                                            int role) const
    {
        if (item.participants.size() <= 0) {
            return QVariant();
        }
        auto &contactModel = accountInfo.contactModel;
        switch (role) {
        case Role::Picture: {
            auto contactImage
                = GlobalInstances::pixmapManipulator().decorationRole(item, accountInfo).value<QImage>();
            return QString::fromLatin1(Utils::QImageToByteArray(contactImage).toBase64().data());
        }
        case Role::DisplayName: {
            if (!item.participants.isEmpty()) {
                auto &contact = contactModel->getContact(item.participants[0]);
                return QVariant(Utils::bestNameForContact(contact));
            }
            return QVariant("");
        }
        case Role::DisplayID: {
            if (!item.participants.isEmpty()) {
                auto &contact = contactModel->getContact(item.participants[0]);
                return QVariant(Utils::bestIdForContact(contact));
            }
            return QVariant("");
        }
        case Role::Presence: {
            if (!item.participants.isEmpty()) {
                auto &contact = contactModel->getContact(item.participants[0]);
                return QVariant(contact.isPresent);
            }
            return QVariant(false);
        }
        case Role::URI: {
            if (!item.participants.isEmpty()) {
                auto &contact = contactModel->getContact(item.participants[0]);
                return QVariant(contact.profileInfo.uri);
            }
            return QVariant("");
        }
        case Role::UnreadMessagesCount:
            return QVariant(item.unreadMessages);
        case Role::LastInteractionDate: {
            if (!item.interactions.empty()) {
                auto &date = item.interactions.at(item.lastMessageUid).timestamp;
                return QVariant(QString::fromStdString(Utils::formatTimeString(date)));
            }
            return QVariant("");
        }
        case Role::LastInteraction: {
            if (!item.interactions.empty()) {
                return QVariant(item.interactions.at(item.lastMessageUid).body);
            }
            return QVariant("");
        }
        case Role::LastInteractionType: {
            if (!item.interactions.empty()) {
                return QVariant(
                    Utils::toUnderlyingValue(item.interactions.at(item.lastMessageUid).type));
            }
            return QVariant(0);
        }
        case Role::ContactType: {
            if (!item.participants.isEmpty()) {
                auto &contact = contactModel->getContact(item.participants[0]);
                return QVariant(Utils::toUnderlyingValue(contact.profileInfo.type));
            }
            return QVariant(0);
        }
        case Role::UID:
            return QVariant(item.uid);
        case Role::InCall: {
            auto* convModel = LRCInstance::getCurrentConversationModel();
            const auto convInfo = convModel->getConversationForUID(item.uid);
            if (!convInfo.uid.isEmpty()) {
                auto* callModel = LRCInstance::getCurrentCallModel();
                return QVariant(callModel->hasCall(convInfo.callId));
            }
            return QVariant(false);
        }
        case Role::IsAudioOnly: {
            auto* convModel = LRCInstance::getCurrentConversationModel();
            const auto convInfo = convModel->getConversationForUID(item.uid);
            if (!convInfo.uid.isEmpty()) {
                auto* call = LRCInstance::getCallInfoForConversation(convInfo);
                if (call) {
                    return QVariant(call->isAudioOnly);
                }
            }
            return QVariant();
        }
        case Role::CallStackViewShouldShow: {
            auto* convModel = LRCInstance::getCurrentConversationModel();
            const auto convInfo = convModel->getConversationForUID(item.uid);
            if (!convInfo.uid.isEmpty()) {
                auto* callModel = LRCInstance::getCurrentCallModel();
                const auto call = callModel->getCall(convInfo.callId);
                return QVariant(callModel->hasCall(convInfo.callId)
                                && ((!call.isOutgoing
                                     && (call.status == lrc::api::call::Status::IN_PROGRESS
                                         || call.status == lrc::api::call::Status::PAUSED))
                                    || call.isOutgoing));
            }
            return QVariant(false);
        }
        case Role::CallStateStr: {
            auto* convModel = LRCInstance::getCurrentConversationModel();
            const auto convInfo = convModel->getConversationForUID(item.uid);
            if (!convInfo.uid.isEmpty()) {
                auto* call = LRCInstance::getCallInfoForConversation(convInfo);
                if (call) {
                    auto statusString = call::to_string(call->status);
                    return QVariant(statusString);
                }
            }
            return QVariant();
        }
        case Role::SectionName:
            return QVariant(QString());
        case Role::Draft: {
            if (!item.uid.isEmpty()) {
                const auto draft = LRCInstance::getContentDraft(item.uid, accountInfo.id);
                if (!draft.isEmpty()) {
                    /*
                     * Pencil Emoji
                     */
                    uint cp = 0x270F;
                    auto emojiString = QString::fromUcs4(&cp, 1);
                    return emojiString + LRCInstance::getContentDraft(item.uid, accountInfo.id);
                }
            }
            return QVariant("");
        }
        }
        return QVariant();
    }
    
    QModelIndex
    SmartListModel::index(int row, int column, const QModelIndex &parent) const
    {
        Q_UNUSED(parent);
        if (column != 0) {
            return QModelIndex();
        }
    
        if (row >= 0 && row < rowCount()) {
            return createIndex(row, column);
        }
        return QModelIndex();
    }
    
    QModelIndex
    SmartListModel::parent(const QModelIndex &child) const
    {
        Q_UNUSED(child);
        return QModelIndex();
    }
    
    Qt::ItemFlags
    SmartListModel::flags(const QModelIndex &index) const
    {
        auto flags = QAbstractItemModel::flags(index) | Qt::ItemNeverHasChildren | Qt::ItemIsSelectable;
        auto type = Utils::toEnum<lrc::api::profile::Type>(data(index, Role::ContactType).value<int>());
        auto uid = data(index, Role::UID).value<QString>();
        if (!index.isValid()) {
            return QAbstractItemModel::flags(index);
        } else if ((type == lrc::api::profile::Type::TEMPORARY && uid.isEmpty())) {
            flags &= ~(Qt::ItemIsSelectable);
        }
        return flags;
    }
    
    void
    SmartListModel::setAccount(const QString &accountId)
    {
        accountId_ = accountId;
    }