/* * ChatModel.cpp * Copyright (C) 2017 Belledonne Communications, Grenoble, France * * 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 2 * 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, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * Created on: February 2, 2017 * Author: Ronan Abhamon */ #include #include #include #include #include #include #include #include #include #include "../../app/App.hpp" #include "../../app/paths/Paths.hpp" #include "../../app/providers/ThumbnailProvider.hpp" #include "../../utils/Utils.hpp" #include "../core/CoreManager.hpp" #include "ChatModel.hpp" #define THUMBNAIL_IMAGE_FILE_HEIGHT 100 #define THUMBNAIL_IMAGE_FILE_WIDTH 100 // In Bytes. #define FILE_SIZE_LIMIT 524288000 using namespace std; // ============================================================================= inline QString getFileId (const shared_ptr &message) { return ::Utils::coreStringToAppString(message->getAppdata()).section(':', 0, 0); } inline QString getDownloadPath (const shared_ptr &message) { return ::Utils::coreStringToAppString(message->getAppdata()).section(':', 1); } inline bool fileWasDownloaded (const shared_ptr &message) { const QString path = ::getDownloadPath(message); return !path.isEmpty() && QFileInfo(path).isFile(); } inline void fillThumbnailProperty (QVariantMap &dest, const shared_ptr &message) { QString fileId = ::getFileId(message); if (!fileId.isEmpty() && !dest.contains("thumbnail")) dest["thumbnail"] = QStringLiteral("image://%1/%2") .arg(ThumbnailProvider::PROVIDER_ID).arg(fileId); } inline void createThumbnail (const shared_ptr &message) { if (!message->getAppdata().empty()) return; QString thumbnailPath = ::Utils::coreStringToAppString(message->getFileTransferFilepath()); QImage image(thumbnailPath); if (image.isNull()) return; QImage thumbnail = image.scaled( THUMBNAIL_IMAGE_FILE_WIDTH, THUMBNAIL_IMAGE_FILE_HEIGHT, Qt::KeepAspectRatio, Qt::SmoothTransformation ); QString uuid = QUuid::createUuid().toString(); QString fileId = QStringLiteral("%1.jpg").arg(uuid.mid(1, uuid.length() - 2)); if (!thumbnail.save(::Utils::coreStringToAppString(Paths::getThumbnailsDirPath()) + fileId, "jpg", 100)) { qWarning() << QStringLiteral("Unable to create thumbnail of: `%1`.").arg(thumbnailPath); return; } message->setAppdata(::Utils::appStringToCoreString(fileId)); } inline void removeFileMessageThumbnail (const shared_ptr &message) { if (message && message->getFileTransferInformation()) { message->cancelFileTransfer(); string fileId = message->getAppdata(); if (!fileId.empty()) { QString thumbnailPath = ::Utils::coreStringToAppString(Paths::getThumbnailsDirPath() + fileId); if (!QFile::remove(thumbnailPath)) qWarning() << QStringLiteral("Unable to remove `%1`.").arg(thumbnailPath); } } } // ----------------------------------------------------------------------------- class ChatModel::MessageHandlers : public linphone::ChatMessageListener { friend class ChatModel; public: MessageHandlers (ChatModel *chatModel) : mChatModel(chatModel) {} ~MessageHandlers () = default; private: QList::iterator findMessageEntry (const shared_ptr &message) { return find_if(mChatModel->mEntries.begin(), mChatModel->mEntries.end(), [&message](const ChatEntryData &pair) { return pair.second == message; }); } void signalDataChanged (const QList::iterator &it) { int row = static_cast(distance(mChatModel->mEntries.begin(), it)); emit mChatModel->dataChanged(mChatModel->index(row, 0), mChatModel->index(row, 0)); } shared_ptr onFileTransferSend ( const shared_ptr &, const shared_ptr &, size_t, size_t ) override { qWarning() << "`onFileTransferSend` called."; return nullptr; } void onFileTransferProgressIndication ( const shared_ptr &message, const shared_ptr &, size_t offset, size_t ) override { if (!mChatModel) return; auto it = findMessageEntry(message); if (it == mChatModel->mEntries.end()) return; (*it).first["fileOffset"] = static_cast(offset); signalDataChanged(it); } void onMsgStateChanged (const shared_ptr &message, linphone::ChatMessageState state) override { if (!mChatModel) return; auto it = findMessageEntry(message); if (it == mChatModel->mEntries.end()) return; // File message downloaded. if (state == linphone::ChatMessageStateFileTransferDone && !message->isOutgoing()) { ::createThumbnail(message); ::fillThumbnailProperty((*it).first, message); message->setAppdata( ::Utils::appStringToCoreString(::getFileId(message)) + ':' + message->getFileTransferFilepath() ); (*it).first["wasDownloaded"] = true; App::getInstance()->getNotifier()->notifyReceivedFileMessage(message); } (*it).first["status"] = state; signalDataChanged(it); } ChatModel *mChatModel; }; // ----------------------------------------------------------------------------- ChatModel::ChatModel (const QString &sipAddress) { CoreManager *core = CoreManager::getInstance(); mCoreHandlers = core->getHandlers(); mMessageHandlers = make_shared(this); setSipAddress(sipAddress); { CoreHandlers *coreHandlers = mCoreHandlers.get(); QObject::connect(coreHandlers, &CoreHandlers::messageReceived, this, &ChatModel::handleMessageReceived); QObject::connect(coreHandlers, &CoreHandlers::callStateChanged, this, &ChatModel::handleCallStateChanged); QObject::connect(coreHandlers, &CoreHandlers::isComposingChanged, this, &ChatModel::handleIsComposingChanged); } } ChatModel::~ChatModel () { mMessageHandlers->mChatModel = nullptr; } QHash ChatModel::roleNames () const { QHash roles; roles[Roles::ChatEntry] = "$chatEntry"; roles[Roles::SectionDate] = "$sectionDate"; return roles; } int ChatModel::rowCount (const QModelIndex &) const { return mEntries.count(); } QVariant ChatModel::data (const QModelIndex &index, int role) const { int row = index.row(); if (!index.isValid() || row < 0 || row >= mEntries.count()) return QVariant(); switch (role) { case Roles::ChatEntry: return QVariant::fromValue(mEntries[row].first); case Roles::SectionDate: return QVariant::fromValue(mEntries[row].first["timestamp"].toDate()); } return QVariant(); } bool ChatModel::removeRow (int row, const QModelIndex &) { return removeRows(row, 1); } bool ChatModel::removeRows (int row, int count, const QModelIndex &parent) { int limit = row + count - 1; if (row < 0 || count < 0 || limit >= mEntries.count()) return false; beginRemoveRows(parent, row, limit); for (int i = 0; i < count; ++i) { removeEntry(mEntries[row]); mEntries.removeAt(row); } endRemoveRows(); if (mEntries.count() == 0) emit allEntriesRemoved(); return true; } QString ChatModel::getSipAddress () const { return ::Utils::coreStringToAppString( mChatRoom->getPeerAddress()->asStringUriOnly() ); } void ChatModel::setSipAddress (const QString &sipAddress) { shared_ptr core = CoreManager::getInstance()->getCore(); mChatRoom = core->getChatRoomFromUri(::Utils::appStringToCoreString(sipAddress)); Q_CHECK_PTR(mChatRoom.get()); if (mChatRoom->getUnreadMessagesCount() > 0) resetMessagesCount(); // Get messages. for (auto &message : mChatRoom->getHistory(0)) { QVariantMap map; fillMessageEntry(map, message); // Old workaround. // It can exist messages with a not delivered status. It's a linphone core bug. if (message->getState() == linphone::ChatMessageStateInProgress) map["status"] = linphone::ChatMessageStateNotDelivered; mEntries << qMakePair(map, static_pointer_cast(message)); } // Get calls. for (auto &callLog : core->getCallHistoryForAddress(mChatRoom->getPeerAddress())) insertCall(callLog); } bool ChatModel::getIsRemoteComposing () const { return mIsRemoteComposing; } // ----------------------------------------------------------------------------- void ChatModel::removeEntry (int id) { qInfo() << QStringLiteral("Removing chat entry: %1 of %2.") .arg(id).arg(getSipAddress()); if (!removeRow(id)) qWarning() << QStringLiteral("Unable to remove chat entry: %1").arg(id); } void ChatModel::removeAllEntries () { qInfo() << QStringLiteral("Removing all chat entries of: %1.").arg(getSipAddress()); beginResetModel(); for (auto &entry : mEntries) removeEntry(entry); mEntries.clear(); endResetModel(); emit allEntriesRemoved(); } // ----------------------------------------------------------------------------- void ChatModel::sendMessage (const QString &message) { shared_ptr _message = mChatRoom->createMessage(::Utils::appStringToCoreString(message)); _message->setListener(mMessageHandlers); insertMessageAtEnd(_message); mChatRoom->sendChatMessage(_message); emit messageSent(_message); } void ChatModel::resendMessage (int id) { if (id < 0 || id > mEntries.count()) { qWarning() << QStringLiteral("Entry %1 not exists.").arg(id); return; } const ChatEntryData entry = mEntries[id]; const QVariantMap map = entry.first; if (map["type"] != EntryType::MessageEntry) { qWarning() << QStringLiteral("Unable to resend entry %1. It's not a message.").arg(id); return; } switch (map["status"].toInt()) { case MessageStatusFileTransferError: case MessageStatusNotDelivered: { shared_ptr message = static_pointer_cast(entry.second); message->setListener(mMessageHandlers); message->resend(); break; } default: qWarning() << QStringLiteral("Unable to resend message: %1. Bad state.").arg(id); } } void ChatModel::sendFileMessage (const QString &path) { QFile file(path); if (!file.exists()) return; qint64 fileSize = file.size(); if (fileSize > FILE_SIZE_LIMIT) { qWarning() << QStringLiteral("Unable to send file. (Size limit=%1)").arg(FILE_SIZE_LIMIT); return; } shared_ptr content = CoreManager::getInstance()->getCore()->createContent(); content->setType("application"); content->setSubtype("octet-stream"); content->setSize(static_cast(fileSize)); content->setName(::Utils::appStringToCoreString(QFileInfo(file).fileName())); shared_ptr message = mChatRoom->createFileTransferMessage(content); message->setFileTransferFilepath(::Utils::appStringToCoreString(path)); message->setListener(mMessageHandlers); ::createThumbnail(message); insertMessageAtEnd(message); mChatRoom->sendChatMessage(message); emit messageSent(message); } // ----------------------------------------------------------------------------- void ChatModel::downloadFile (int id) { const ChatEntryData entry = getFileMessageEntry(id); if (!entry.second) return; shared_ptr message = static_pointer_cast(entry.second); switch (message->getState()) { case MessageStatusDelivered: case MessageStatusDeliveredToUser: case MessageStatusDisplayed: case MessageStatusFileTransferDone: break; default: qWarning() << QStringLiteral("Unable to download file of entry %1. It was not uploaded.").arg(id); return; } bool soFarSoGood; const QString safeFilePath = ::Utils::getSafeFilePath( QStringLiteral("%1%2") .arg(CoreManager::getInstance()->getSettingsModel()->getDownloadFolder()) .arg(entry.first["fileName"].toString()), &soFarSoGood ); if (!soFarSoGood) { qWarning() << QStringLiteral("Unable to create safe file path for: %1.").arg(id); return; } message->setFileTransferFilepath(::Utils::appStringToCoreString(safeFilePath)); message->setListener(mMessageHandlers); if (message->downloadFile() < 0) qWarning() << QStringLiteral("Unable to download file of entry %1.").arg(id); } void ChatModel::openFile (int id, bool showDirectory) { const ChatEntryData entry = getFileMessageEntry(id); if (!entry.second) return; shared_ptr message = static_pointer_cast(entry.second); if (!::fileWasDownloaded(message)) { downloadFile(id); return; } QFileInfo info(::getDownloadPath(message)); QDesktopServices::openUrl( QUrl(QStringLiteral("file:///%1").arg(showDirectory ? info.absolutePath() : info.absoluteFilePath())) ); } bool ChatModel::fileWasDownloaded (int id) { const ChatEntryData entry = getFileMessageEntry(id); return entry.second && ::fileWasDownloaded(static_pointer_cast(entry.second)); } void ChatModel::compose () { mChatRoom->compose(); } // ----------------------------------------------------------------------------- const ChatModel::ChatEntryData ChatModel::getFileMessageEntry (int id) { if (id < 0 || id > mEntries.count()) { qWarning() << QStringLiteral("Entry %1 not exists.").arg(id); return ChatEntryData(); } const ChatEntryData entry = mEntries[id]; if (entry.first["type"] != EntryType::MessageEntry) { qWarning() << QStringLiteral("Unable to download entry %1. It's not a message.").arg(id); return ChatEntryData(); } shared_ptr message = static_pointer_cast(entry.second); if (!message->getFileTransferInformation()) { qWarning() << QStringLiteral("Entry %1 is not a file message.").arg(id); return ChatEntryData(); } return entry; } // ----------------------------------------------------------------------------- void ChatModel::fillMessageEntry (QVariantMap &dest, const shared_ptr &message) { dest["type"] = EntryType::MessageEntry; dest["timestamp"] = QDateTime::fromMSecsSinceEpoch(message->getTime() * 1000); dest["content"] = ::Utils::coreStringToAppString(message->getText()); dest["isOutgoing"] = message->isOutgoing() || message->getState() == linphone::ChatMessageStateIdle; dest["status"] = message->getState(); shared_ptr content = message->getFileTransferInformation(); if (content) { dest["fileSize"] = static_cast(content->getSize()); dest["fileName"] = ::Utils::coreStringToAppString(content->getName()); dest["wasDownloaded"] = ::fileWasDownloaded(message); ::fillThumbnailProperty(dest, message); } } void ChatModel::fillCallStartEntry (QVariantMap &dest, const shared_ptr &callLog) { QDateTime timestamp = QDateTime::fromMSecsSinceEpoch(callLog->getStartDate() * 1000); dest["type"] = EntryType::CallEntry; dest["timestamp"] = timestamp; dest["isOutgoing"] = callLog->getDir() == linphone::CallDirOutgoing; dest["status"] = callLog->getStatus(); dest["isStart"] = true; } void ChatModel::fillCallEndEntry (QVariantMap &dest, const shared_ptr &callLog) { QDateTime timestamp = QDateTime::fromMSecsSinceEpoch((callLog->getStartDate() + callLog->getDuration()) * 1000); dest["type"] = EntryType::CallEntry; dest["timestamp"] = timestamp; dest["isOutgoing"] = callLog->getDir() == linphone::CallDirOutgoing; dest["status"] = callLog->getStatus(); dest["isStart"] = false; } // ----------------------------------------------------------------------------- void ChatModel::removeEntry (ChatEntryData &pair) { int type = pair.first["type"].toInt(); switch (type) { case ChatModel::MessageEntry: { shared_ptr message = static_pointer_cast(pair.second); ::removeFileMessageThumbnail(message); mChatRoom->deleteMessage(message); break; } case ChatModel::CallEntry: { if (pair.first["status"].toInt() == linphone::CallStatusSuccess) { // WARNING: Unable to remove symmetric call here. (start/end) // We are between `beginRemoveRows` and `endRemoveRows`. // A solution is to schedule a `removeEntry` call in the Qt main loop. shared_ptr linphonePtr = pair.second; QTimer::singleShot(0, this, [this, linphonePtr]() { auto it = find_if(mEntries.begin(), mEntries.end(), [linphonePtr](const ChatEntryData &pair) { return pair.second == linphonePtr; }); if (it != mEntries.end()) removeEntry(static_cast(distance(mEntries.begin(), it))); }); } CoreManager::getInstance()->getCore()->removeCallLog(static_pointer_cast(pair.second)); break; } default: qWarning() << QStringLiteral("Unknown chat entry type: %1.").arg(type); } } void ChatModel::insertCall (const shared_ptr &callLog) { linphone::CallStatus status = callLog->getStatus(); switch (status) { case linphone::CallStatusAborted: case linphone::CallStatusEarlyAborted: return; // Ignore aborted calls. case linphone::CallStatusSuccess: case linphone::CallStatusMissed: case linphone::CallStatusDeclined: break; } auto insertEntry = [this]( const ChatEntryData &pair, const QList::iterator *start = NULL ) { auto it = lower_bound(start ? *start : mEntries.begin(), mEntries.end(), pair, [](const ChatEntryData &a, const ChatEntryData &b) { return a.first["timestamp"] < b.first["timestamp"]; }); int row = static_cast(distance(mEntries.begin(), it)); beginInsertRows(QModelIndex(), row, row); it = mEntries.insert(it, pair); endInsertRows(); return it; }; // Add start call. QVariantMap start; fillCallStartEntry(start, callLog); auto it = insertEntry(qMakePair(start, static_pointer_cast(callLog))); // Add end call. (if necessary) if (status == linphone::CallStatusSuccess) { QVariantMap end; fillCallEndEntry(end, callLog); insertEntry(qMakePair(end, static_pointer_cast(callLog)), &it); } } void ChatModel::insertMessageAtEnd (const shared_ptr &message) { int row = mEntries.count(); beginInsertRows(QModelIndex(), row, row); QVariantMap map; fillMessageEntry(map, message); mEntries << qMakePair(map, static_pointer_cast(message)); endInsertRows(); } void ChatModel::resetMessagesCount () { mChatRoom->markAsRead(); emit messagesCountReset(); } // ----------------------------------------------------------------------------- void ChatModel::handleCallStateChanged (const shared_ptr &call, linphone::CallState state) { if ( (state == linphone::CallStateEnd || state == linphone::CallStateError) && mChatRoom == CoreManager::getInstance()->getCore()->getChatRoom(call->getRemoteAddress()) ) insertCall(call->getCallLog()); } void ChatModel::handleIsComposingChanged (const shared_ptr &chatRoom) { if (mChatRoom == chatRoom) { bool isRemoteComposing = mChatRoom->isRemoteComposing(); if (isRemoteComposing != mIsRemoteComposing) { mIsRemoteComposing = isRemoteComposing; emit isRemoteComposingChanged(mIsRemoteComposing); } } } void ChatModel::handleMessageReceived (const shared_ptr &message) { if (mChatRoom == message->getChatRoom()) { insertMessageAtEnd(message); resetMessagesCount(); emit messageReceived(message); } }