/* * Copyright (c) 2010-2024 Belledonne Communications SARL. * * This file is part of linphone-desktop * (see https://www.linphone.org). * * 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 . */ #include "MagicSearchList.hpp" #include "core/App.hpp" #include "core/friend/FriendCore.hpp" #include "model/tool/ToolModel.hpp" #include "tool/Utils.hpp" #include #include // ============================================================================= DEFINE_ABSTRACT_OBJECT(MagicSearchList) QSharedPointer MagicSearchList::create() { auto model = QSharedPointer(new MagicSearchList(), &QObject::deleteLater); model->moveToThread(App::getInstance()->thread()); model->setSelf(model); return model; } MagicSearchList::MagicSearchList(QObject *parent) : ListProxy(parent) { mustBeInMainThread(getClassName()); App::getInstance()->mEngine->setObjectOwnership(this, QQmlEngine::CppOwnership); mSourceFlags = (int)linphone::MagicSearch::Source::Friends | (int)linphone::MagicSearch::Source::LdapServers; mAggregationFlag = LinphoneEnums::MagicSearchAggregation::Friend; mSearchFilter = "*"; } MagicSearchList::~MagicSearchList() { mustBeInMainThread("~" + getClassName()); } void MagicSearchList::setSelf(QSharedPointer me) { mCoreModelConnection = QSharedPointer>( new SafeConnection(me, CoreModel::getInstance()), &QObject::deleteLater); mCoreModelConnection->makeConnectToModel( &CoreModel::friendCreated, [this](const std::shared_ptr &f) { auto friendCore = FriendCore::create(f); auto haveContact = std::find_if(mList.begin(), mList.end(), [friendCore](const QSharedPointer &item) { return item.objectCast()->getDefaultAddress() == friendCore->getDefaultAddress(); }); if (haveContact == mList.end()) { connect(friendCore.get(), &FriendCore::removed, this, qOverload(&MagicSearchList::remove)); add(friendCore); emit friendCreated(getCount() - 1, new FriendGui(friendCore)); } }); mCoreModelConnection->invokeToModel([this] { auto linphoneSearch = CoreModel::getInstance()->getCore()->createMagicSearch(); linphoneSearch->setLimitedSearch(false); auto magicSearch = Utils::makeQObject_ptr(linphoneSearch); mCoreModelConnection->invokeToCore([this, magicSearch] { mMagicSearch = magicSearch; mMagicSearch->setSelf(mMagicSearch); mModelConnection = QSharedPointer>( new SafeConnection(mCoreModelConnection->mCore.mQData, mMagicSearch), &QObject::deleteLater); mModelConnection->makeConnectToCore( &MagicSearchList::lSearch, [this](QString filter, int sourceFlags, LinphoneEnums::MagicSearchAggregation aggregationFlag, int maxResults) { resetData(); mModelConnection->invokeToModel([this, filter, sourceFlags, aggregationFlag, maxResults]() { mMagicSearch->search(filter, sourceFlags, aggregationFlag, maxResults); }); }); mModelConnection->makeConnectToModel( &MagicSearchModel::searchResultsReceived, [this](const std::list> &results) { auto *contacts = new QList>(); auto ldapContacts = ToolModel::getLdapFriendList(); for (auto it : results) { QSharedPointer contact; auto linphoneFriend = it->getFriend(); bool isStored = false; if (linphoneFriend) { isStored = (ldapContacts->findFriendByAddress(linphoneFriend->getAddress()) != linphoneFriend); contact = FriendCore::create(linphoneFriend, isStored); contacts->append(contact); } else if (auto address = it->getAddress()) { auto linphoneFriend = CoreModel::getInstance()->getCore()->createFriend(); linphoneFriend->setAddress(address); contact = FriendCore::create(linphoneFriend, isStored); auto displayName = Utils::coreStringToAppString(address->getDisplayName()); auto splitted = displayName.split(" "); if (!displayName.isEmpty() && splitted.size() > 0) { contact->setGivenName(splitted[0]); splitted.removeFirst(); contact->setFamilyName(splitted.join(" ")); } else { contact->setGivenName(Utils::coreStringToAppString(address->getUsername())); } contact->setDefaultFullAddress(Utils::coreStringToAppString( address->asString())); // linphone Friend object remove specific address. contacts->append(contact); } else if (!it->getPhoneNumber().empty()) { linphoneFriend = CoreModel::getInstance()->getCore()->createFriend(); linphoneFriend->setAddress(address); contact = FriendCore::create(linphoneFriend, isStored); contact->setGivenName(Utils::coreStringToAppString(it->getPhoneNumber())); contact->appendPhoneNumber(tr("Phone"), Utils::coreStringToAppString(it->getPhoneNumber())); contacts->append(contact); } } mModelConnection->invokeToCore([this, contacts]() { setResults(*contacts); delete contacts; }); }); qDebug() << log().arg("Initialized"); emit initialized(); }); }); } void MagicSearchList::connectContact(FriendCore *data) { connect(data, &FriendCore::removed, this, qOverload(&MagicSearchList::remove)); connect(data, &FriendCore::starredChanged, this, &MagicSearchList::friendStarredChanged); } void MagicSearchList::setResults(const QList> &contacts) { for (auto item : mList) { auto isFriendCore = item.objectCast(); if (!isFriendCore) continue; disconnect(isFriendCore.get()); } qDebug() << log().arg("SetResults: %1").arg(contacts.size()); resetData(contacts); for (auto it : contacts) { connectContact(it.get()); } } void MagicSearchList::add(QSharedPointer contact) { connectContact(contact.get()); ListProxy::add(contact); } void MagicSearchList::setSearch(const QString &search) { mSearchFilter = search; if (!search.isEmpty()) { emit lSearch(search, mSourceFlags, mAggregationFlag, mMaxResults); } else { beginResetModel(); mList.clear(); endResetModel(); } } int MagicSearchList::getSourceFlags() const { return mSourceFlags; } void MagicSearchList::setSourceFlags(int flags) { if (mSourceFlags != flags) { mSourceFlags = flags; emit sourceFlagsChanged(mSourceFlags); } } int MagicSearchList::getMaxResults() const { return mMaxResults; } void MagicSearchList::setMaxResults(int maxResults) { if (mMaxResults != maxResults) { mMaxResults = maxResults; emit maxResultsChanged(mMaxResults); } } LinphoneEnums::MagicSearchAggregation MagicSearchList::getAggregationFlag() const { return mAggregationFlag; } void MagicSearchList::setAggregationFlag(LinphoneEnums::MagicSearchAggregation flags) { if (mAggregationFlag != flags) { mAggregationFlag = flags; emit aggregationFlagChanged(mAggregationFlag); } } QHash MagicSearchList::roleNames() const { QHash roles; roles[Qt::DisplayRole] = "$modelData"; roles[Qt::DisplayRole + 1] = "isStored"; return roles; } QVariant MagicSearchList::data(const QModelIndex &index, int role) const { int row = index.row(); if (!index.isValid() || row < 0 || row >= mList.count()) return QVariant(); if (role == Qt::DisplayRole) { return QVariant::fromValue(new FriendGui(mList[row].objectCast())); } else if (role == Qt::DisplayRole + 1) { return mList[row].objectCast()->getIsStored() || mList[row].objectCast()->isLdap(); } return QVariant(); } int MagicSearchList::findFriendIndexByAddress(const QString &address) { for (int i = 0; i < getCount(); ++i) { auto friendCore = getAt(i); if (!friendCore) continue; for (auto &friendAddress : friendCore->getAllAddresses()) { auto map = friendAddress.toMap(); if (map["address"].toString() == address) { return i; } } } return -1; } QSharedPointer MagicSearchList::findFriendByAddress(const QString &address) { for (int i = 0; i < getCount(); ++i) { auto friendCore = getAt(i); if (!friendCore) continue; for (auto &friendAddress : friendCore->getAllAddresses()) { auto map = friendAddress.toMap(); if (map["address"].toString() == address) { return friendCore; } } } return nullptr; }