/* * 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 "ChatList.hpp" #include "ChatCore.hpp" #include "ChatGui.hpp" #include "core/App.hpp" #include "model/tool/ToolModel.hpp" #include #include // ============================================================================= DEFINE_ABSTRACT_OBJECT(ChatList) QSharedPointer ChatList::create() { auto model = QSharedPointer(new ChatList(), &QObject::deleteLater); model->moveToThread(App::getInstance()->thread()); model->setSelf(model); return model; } QSharedPointer ChatList::createChatCore(const std::shared_ptr &chatroom) { auto chatCore = ChatCore::create(chatroom); return chatCore; } ChatList::ChatList(QObject *parent) : ListProxy(parent) { mustBeInMainThread(getClassName()); App::getInstance()->mEngine->setObjectOwnership(this, QQmlEngine::CppOwnership); } ChatList::~ChatList() { mustBeInMainThread("~" + getClassName()); mModelConnection = nullptr; } void ChatList::connectItem(QSharedPointer chat) { connect(chat.get(), &ChatCore::deleted, this, [this, chat] { disconnect(chat.get()); remove(chat); // We cannot use countChanged here because it is called before mList // really has removed the item, then emit specific signal emit chatRemoved(chat ? new ChatGui(chat) : nullptr); }); auto dataChange = [this, chat] { int i = -1; get(chat.get(), &i); if (i != -1) { auto modelIndex = index(i); emit dataChanged(modelIndex, modelIndex); } }; connect(chat.get(), &ChatCore::unreadMessagesCountChanged, this, [this, dataChange] { dataChange(); auto defaultAccount = App::getInstance()->getAccountList()->getDefaultAccountCore(); if (defaultAccount) emit defaultAccount->lRefreshNotifications(); }); connect(chat.get(), &ChatCore::lastUpdatedTimeChanged, this, dataChange); connect(chat.get(), &ChatCore::lastMessageChanged, this, dataChange); } void ChatList::setSelf(QSharedPointer me) { mModelConnection = SafeConnection::create(me, CoreModel::getInstance()); mModelConnection->makeConnectToCore(&ChatList::lUpdate, [this]() { if (mIsUpdating) { connect(this, &ChatList::isUpdatingChanged, this, [this] { if (!mIsUpdating) { disconnect(this, &ChatList::isUpdatingChanged, this, nullptr); lUpdate(); } }); return; } setIsUpdating(true); mModelConnection->invokeToModel([this]() { mustBeInLinphoneThread(getClassName()); beginResetModel(); mList.clear(); // Avoid copy to lambdas QList> *chats = new QList>(); auto currentAccount = CoreModel::getInstance()->getCore()->getDefaultAccount(); if (!currentAccount) { setIsUpdating(false); endResetModel(); return; } auto linphoneChatRooms = currentAccount->filterChatRooms(Utils::appStringToCoreString(mFilter)); for (auto it : linphoneChatRooms) { auto model = createChatCore(it); chats->push_back(model); } mModelConnection->invokeToCore([this, chats]() { mustBeInMainThread(getClassName()); for (auto &chat : getSharedList()) { if (chat) { disconnect(chat.get(), &ChatCore::deleted, this, nullptr); disconnect(chat.get(), &ChatCore::unreadMessagesCountChanged, this, nullptr); disconnect(chat.get(), &ChatCore::lastUpdatedTimeChanged, this, nullptr); disconnect(chat.get(), &ChatCore::lastMessageChanged, this, nullptr); } } for (auto &chat : *chats) { connectItem(chat); } add(*chats); endResetModel(); setIsUpdating(false); delete chats; }); }); }); mModelConnection->makeConnectToModel( &CoreModel::defaultAccountChanged, [this](std::shared_ptr core, std::shared_ptr account) { lUpdate(); }); auto addChatToList = [this](const std::shared_ptr &core, const std::shared_ptr &room, const std::shared_ptr &message) { mustBeInLinphoneThread(log().arg(Q_FUNC_INFO)); if (!message) return; if (room->getAccount() != core->getDefaultAccount()) { qWarning() << log().arg("Chat room does not refer to current account, return"); return; } auto chatCore = ChatCore::create(room); mModelConnection->invokeToCore([this, chatCore] { auto chatList = getSharedList(); auto it = std::find_if(chatList.begin(), chatList.end(), [chatCore](const QSharedPointer item) { return item && chatCore && item->getModel() && chatCore->getModel() && item->getModel()->getMonitor() == chatCore->getModel()->getMonitor(); }); if (it == chatList.end()) { connectItem(chatCore); add(chatCore); emit chatAdded(); } }); }; mModelConnection->makeConnectToModel(&CoreModel::messageReceived, [this, addChatToList](const std::shared_ptr &core, const std::shared_ptr &room, const std::shared_ptr &message) { addChatToList(core, room, message); }); mModelConnection->makeConnectToModel( &CoreModel::messagesReceived, [this, addChatToList](const std::shared_ptr &core, const std::shared_ptr &room, const std::list> &messages) { addChatToList(core, room, messages.front()); }); mModelConnection->makeConnectToModel( &CoreModel::newMessageReaction, [this, addChatToList](const std::shared_ptr &core, const std::shared_ptr &room, const std::shared_ptr &message, const std::shared_ptr &reaction) { addChatToList(core, room, message); }); connect(this, &ChatList::filterChanged, [this](QString filter) { mFilter = filter; lUpdate(); }); lUpdate(); } int ChatList::findChatIndex(ChatGui *chatGui) { if (!chatGui) return -1; auto core = chatGui->mCore; auto chatList = getSharedList(); auto it = std::find_if(chatList.begin(), chatList.end(), [core](const QSharedPointer item) { return item->getIdentifier() == core->getIdentifier(); }); return it == chatList.end() ? -1 : std::distance(chatList.begin(), it); } void ChatList::addChatInList(QSharedPointer chatCore) { auto chatList = getSharedList(); auto it = std::find_if(chatList.begin(), chatList.end(), [chatCore](const QSharedPointer item) { return item && chatCore && item->getModel() && chatCore->getModel() && item->getModel()->getMonitor() == chatCore->getModel()->getMonitor(); }); if (it == chatList.end()) { connectItem(chatCore); add(chatCore); emit chatAdded(); } } QVariant ChatList::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 ChatGui(mList[row].objectCast())); return QVariant(); }