hide sip address in conversation info if hide_sip_addresses set write account presence note event if empty (otherwise it cannot be cleaned) add chatroom to list when Created state signal received fix navigation to create contact from conversation infos display copy sip address even if participant is in contact list
224 lines
8.4 KiB
C++
224 lines
8.4 KiB
C++
/*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "ChatList.hpp"
|
|
#include "ChatCore.hpp"
|
|
#include "ChatGui.hpp"
|
|
#include "core/App.hpp"
|
|
#include "model/tool/ToolModel.hpp"
|
|
|
|
#include <QSharedPointer>
|
|
#include <linphone++/linphone.hh>
|
|
|
|
// =============================================================================
|
|
|
|
DEFINE_ABSTRACT_OBJECT(ChatList)
|
|
|
|
QSharedPointer<ChatList> ChatList::create() {
|
|
auto model = QSharedPointer<ChatList>(new ChatList(), &QObject::deleteLater);
|
|
model->moveToThread(App::getInstance()->thread());
|
|
model->setSelf(model);
|
|
return model;
|
|
}
|
|
|
|
QSharedPointer<ChatCore> ChatList::createChatCore(const std::shared_ptr<linphone::ChatRoom> &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<ChatCore> chat) {
|
|
connect(
|
|
chat.get(), &ChatCore::deleted, this,
|
|
[this, chat] {
|
|
disconnect(chat.get(), &ChatCore::unreadMessagesCountChanged, this, nullptr);
|
|
disconnect(chat.get(), &ChatCore::lastUpdatedTimeChanged, this, nullptr);
|
|
disconnect(chat.get(), &ChatCore::lastMessageChanged, this, nullptr);
|
|
remove(chat);
|
|
},
|
|
Qt::SingleShotConnection);
|
|
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<ChatList> me) {
|
|
mModelConnection = SafeConnection<ChatList, CoreModel>::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);
|
|
beginResetModel();
|
|
mModelConnection->invokeToModel([this]() {
|
|
mustBeInLinphoneThread(getClassName());
|
|
// Avoid copy to lambdas
|
|
QList<QSharedPointer<ChatCore>> *chats = new QList<QSharedPointer<ChatCore>>();
|
|
auto currentAccount = CoreModel::getInstance()->getCore()->getDefaultAccount();
|
|
if (!currentAccount) {
|
|
mModelConnection->invokeToCore([this, chats]() {
|
|
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<ChatCore>()) {
|
|
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);
|
|
}
|
|
}
|
|
mList.clear();
|
|
for (auto &chat : *chats) {
|
|
connectItem(chat);
|
|
mList.append(chat);
|
|
}
|
|
endResetModel();
|
|
setIsUpdating(false);
|
|
delete chats;
|
|
});
|
|
});
|
|
});
|
|
|
|
mModelConnection->makeConnectToModel(
|
|
&CoreModel::defaultAccountChanged,
|
|
[this](std::shared_ptr<linphone::Core> core, std::shared_ptr<linphone::Account> account) { lUpdate(); });
|
|
|
|
auto addChatToList = [this](const std::shared_ptr<linphone::Core> &core,
|
|
const std::shared_ptr<linphone::ChatRoom> &room,
|
|
const std::shared_ptr<linphone::ChatMessage> &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] { addChatInList(chatCore); });
|
|
};
|
|
mModelConnection->makeConnectToModel(&CoreModel::messageReceived,
|
|
[this, addChatToList](const std::shared_ptr<linphone::Core> &core,
|
|
const std::shared_ptr<linphone::ChatRoom> &room,
|
|
const std::shared_ptr<linphone::ChatMessage> &message) {
|
|
addChatToList(core, room, message);
|
|
});
|
|
mModelConnection->makeConnectToModel(
|
|
&CoreModel::messagesReceived,
|
|
[this, addChatToList](const std::shared_ptr<linphone::Core> &core,
|
|
const std::shared_ptr<linphone::ChatRoom> &room,
|
|
const std::list<std::shared_ptr<linphone::ChatMessage>> &messages) {
|
|
addChatToList(core, room, messages.front());
|
|
});
|
|
mModelConnection->makeConnectToModel(
|
|
&CoreModel::newMessageReaction,
|
|
[this, addChatToList](const std::shared_ptr<linphone::Core> &core,
|
|
const std::shared_ptr<linphone::ChatRoom> &room,
|
|
const std::shared_ptr<linphone::ChatMessage> &message,
|
|
const std::shared_ptr<const linphone::ChatMessageReaction> &reaction) {
|
|
addChatToList(core, room, message);
|
|
});
|
|
mModelConnection->makeConnectToModel(
|
|
&CoreModel::chatRoomStateChanged,
|
|
[this, addChatToList](const std::shared_ptr<linphone::Core> &core,
|
|
const std::shared_ptr<linphone::ChatRoom> &chatRoom, linphone::ChatRoom::State state) {
|
|
if (state == linphone::ChatRoom::State::Created) {
|
|
if (chatRoom->getAccount() != core->getDefaultAccount()) {
|
|
qWarning() << log().arg("Chatroom does not refer to current account, return");
|
|
return;
|
|
}
|
|
auto chatCore = ChatCore::create(chatRoom);
|
|
mModelConnection->invokeToCore([this, chatCore] { addChatInList(chatCore); });
|
|
}
|
|
});
|
|
|
|
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<ChatCore>();
|
|
auto it = std::find_if(chatList.begin(), chatList.end(), [core](const QSharedPointer<ChatCore> item) {
|
|
return item->getIdentifier() == core->getIdentifier();
|
|
});
|
|
return it == chatList.end() ? -1 : std::distance(chatList.begin(), it);
|
|
}
|
|
|
|
bool ChatList::addChatInList(QSharedPointer<ChatCore> chatCore) {
|
|
mustBeInMainThread(log().arg(Q_FUNC_INFO));
|
|
auto chatList = getSharedList<ChatCore>();
|
|
auto it = std::find_if(chatList.begin(), chatList.end(), [chatCore](const QSharedPointer<ChatCore> item) {
|
|
return item && chatCore && item->getIdentifier() == chatCore->getIdentifier();
|
|
});
|
|
if (it == chatList.end()) {
|
|
connectItem(chatCore);
|
|
add(chatCore);
|
|
emit chatAdded();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
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<ChatCore>()));
|
|
return QVariant();
|
|
}
|