/* Copyright (C) 2014 by Project Tox This file is part of qTox, a Qt-based graphical interface for Tox. This program is libre 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 COPYING file for more details. */ #include #include #include #include #include #include #include #include #include #include "chatform.h" #include "src/historykeeper.h" #include "src/widget/form/loadhistorydialog.h" #include "src/friend.h" #include "src/widget/friendwidget.h" #include "src/filetransferinstance.h" #include "src/widget/netcamview.h" #include "src/widget/tool/chattextedit.h" #include "src/core.h" #include "src/widget/widget.h" #include "src/widget/maskablepixmapwidget.h" #include "src/widget/croppinglabel.h" #include "src/misc/style.h" #include "src/misc/settings.h" #include "src/misc/cstring.h" #include "src/chatlog/chatmessage.h" #include "src/chatlog/content/filetransferwidget.h" #include "src/chatlog/content/text.h" #include "src/chatlog/chatlog.h" ChatForm::ChatForm(Friend* chatFriend) : f(chatFriend) , callId(0) { nameLabel->setText(f->getDisplayedName()); avatar->setPixmap(QPixmap(":/img/contact_dark.png"), Qt::transparent); statusMessageLabel = new CroppingLabel(); statusMessageLabel->setObjectName("statusLabel"); statusMessageLabel->setFont(Style::getFont(Style::Medium)); statusMessageLabel->setMinimumHeight(Style::getFont(Style::Medium).pixelSize()); netcam = new NetCamView(); timer = nullptr; chatWidget->setTypingNotification(ChatMessage::createChatInfoMessage("", ChatMessage::TYPING, QDateTime())); headTextLayout->addWidget(statusMessageLabel); headTextLayout->addStretch(); callDuration = new QLabel(); headTextLayout->addWidget(callDuration, 1, Qt::AlignCenter); callDuration->hide(); menu.addAction(tr("Load History..."), this, SLOT(onLoadHistory())); connect(Core::getInstance(), &Core::fileSendStarted, this, &ChatForm::startFileSend); connect(sendButton, &QPushButton::clicked, this, &ChatForm::onSendTriggered); connect(fileButton, &QPushButton::clicked, this, &ChatForm::onAttachClicked); connect(callButton, &QPushButton::clicked, this, &ChatForm::onCallTriggered); connect(videoButton, &QPushButton::clicked, this, &ChatForm::onVideoCallTriggered); connect(msgEdit, &ChatTextEdit::enterPressed, this, &ChatForm::onSendTriggered); connect(msgEdit, &ChatTextEdit::textChanged, this, &ChatForm::onTextEditChanged); connect(micButton, SIGNAL(clicked()), this, SLOT(onMicMuteToggle())); connect(volButton, SIGNAL(clicked()), this, SLOT(onVolMuteToggle())); connect(Core::getInstance(), &Core::fileSendFailed, this, &ChatForm::onFileSendFailed); connect(this, SIGNAL(chatAreaCleared()), this, SLOT(clearReciepts())); connect(nameLabel, &CroppingLabel::textChanged, this, [=](QString text, QString orig) { if (text != orig) emit aliasChanged(text); } ); setAcceptDrops(true); } ChatForm::~ChatForm() { delete netcam; } void ChatForm::setStatusMessage(QString newMessage) { statusMessageLabel->setText(newMessage); statusMessageLabel->setToolTip(newMessage); // for overlength messsages } void ChatForm::onSendTriggered() { QString msg = msgEdit->toPlainText(); if (msg.isEmpty()) return; bool isAction = msg.startsWith("/me "); if (isAction) msg = msg = msg.right(msg.length() - 4); QList splittedMsg = Core::splitMessage(msg, TOX_MAX_MESSAGE_LENGTH); QDateTime timestamp = QDateTime::currentDateTime(); for (CString& c_msg : splittedMsg) { QString qt_msg = CString::toString(c_msg.data(), c_msg.size()); QString qt_msg_hist = qt_msg; if (isAction) qt_msg_hist = "/me " + qt_msg; bool status = !Settings::getInstance().getFauxOfflineMessaging(); int id = HistoryKeeper::getInstance()->addChatEntry(f->getToxID().publicKey, qt_msg_hist, Core::getInstance()->getSelfId().publicKey, timestamp, status); ChatMessage::Ptr ma = addSelfMessage(msg, isAction, timestamp, false); int rec; if (isAction) rec = Core::getInstance()->sendAction(f->getFriendID(), qt_msg); else rec = Core::getInstance()->sendMessage(f->getFriendID(), qt_msg); registerReceipt(rec, id, ma); } msgEdit->clear(); } void ChatForm::onTextEditChanged() { bool isNowTyping; if (!Settings::getInstance().isTypingNotificationEnabled()) isNowTyping = false; else isNowTyping = msgEdit->toPlainText().length() > 0; if (isTyping != isNowTyping) { isTyping = isNowTyping; Core::getInstance()->sendTyping(f->getFriendID(), isTyping); } } void ChatForm::onAttachClicked() { QStringList paths = QFileDialog::getOpenFileNames(0,tr("Send a file")); if (paths.isEmpty()) return; for (QString path : paths) { QFile file(path); if (!file.exists() || !file.open(QIODevice::ReadOnly)) { QMessageBox::warning(this, tr("File not read"), tr("qTox wasn't able to open %1").arg(QFileInfo(path).fileName())); continue; } if (file.isSequential()) { QMessageBox::critical(0, tr("Bad Idea"), tr("You're trying to send a special (sequential) file, that's not going to work!")); file.close(); continue; } long long filesize = file.size(); file.close(); QFileInfo fi(path); emit sendFile(f->getFriendID(), fi.fileName(), path, filesize); } } void ChatForm::startFileSend(ToxFile file) { if (file.friendId != f->getFriendID()) return; QString name; if (!previousId.isMine()) { Core* core = Core::getInstance(); name = core->getUsername(); previousId = core->getSelfId(); } insertChatMessage(ChatMessage::createFileTransferMessage(name, file, true, QDateTime::currentDateTime())); } void ChatForm::onFileRecvRequest(ToxFile file) { if (file.friendId != f->getFriendID()) return; Widget* w = Widget::getInstance(); if (!w->isFriendWidgetCurActiveWidget(f)|| w->isMinimized() || !w->isActiveWindow()) { w->newMessageAlert(f->getFriendWidget()); f->setEventFlag(true); f->getFriendWidget()->updateStatusLight(); } QString name; ToxID friendId = f->getToxID(); if (friendId != previousId) { name = f->getDisplayedName(); previousId = friendId; } ChatMessage::Ptr msg = ChatMessage::createFileTransferMessage(name, file, false, QDateTime::currentDateTime()); insertChatMessage(msg); if (!Settings::getInstance().getAutoAcceptDir(f->getToxID()).isEmpty() || Settings::getInstance().getAutoSaveEnabled()) { FileTransferWidget* tfWidget = dynamic_cast(msg->getContent(1)); if(tfWidget) tfWidget->acceptTransfer(Settings::getInstance().getAutoAcceptDir(f->getToxID())); } } void ChatForm::onAvInvite(int FriendId, int CallId, bool video) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvInvite"; callId = CallId; callButton->disconnect(); videoButton->disconnect(); if (video) { callButton->setObjectName("grey"); callButton->style()->polish(callButton); videoButton->setObjectName("yellow"); videoButton->style()->polish(videoButton); connect(videoButton, SIGNAL(clicked()), this, SLOT(onAnswerCallTriggered())); } else { callButton->setObjectName("yellow"); callButton->style()->polish(callButton); videoButton->setObjectName("grey"); videoButton->style()->polish(videoButton); connect(callButton, SIGNAL(clicked()), this, SLOT(onAnswerCallTriggered())); } insertChatMessage(ChatMessage::createChatInfoMessage(tr("%1 calling").arg(f->getDisplayedName()), ChatMessage::INFO, QDateTime::currentDateTime())); Widget* w = Widget::getInstance(); if (!w->isFriendWidgetCurActiveWidget(f)|| w->isMinimized() || !w->isActiveWindow()) { w->newMessageAlert(f->getFriendWidget()); f->setEventFlag(true); f->getFriendWidget()->updateStatusLight(); } } void ChatForm::onAvStart(int FriendId, int CallId, bool video) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvStart"; audioInputFlag = true; audioOutputFlag = true; callId = CallId; callButton->disconnect(); videoButton->disconnect(); if (video) { callButton->setObjectName("grey"); callButton->style()->polish(callButton); videoButton->setObjectName("red"); videoButton->style()->polish(videoButton); connect(videoButton, SIGNAL(clicked()), this, SLOT(onHangupCallTriggered())); netcam->show(Core::getInstance()->getVideoSourceFromCall(CallId), f->getDisplayedName()); } else { callButton->setObjectName("red"); callButton->style()->polish(callButton); videoButton->setObjectName("grey"); videoButton->style()->polish(videoButton); connect(callButton, SIGNAL(clicked()), this, SLOT(onHangupCallTriggered())); } startCounter(); } void ChatForm::onAvCancel(int FriendId, int) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvCancel"; stopCounter(); audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); netcam->hide(); addSystemInfoMessage(tr("%1 stopped calling").arg(f->getDisplayedName()), ChatMessage::INFO, QDateTime::currentDateTime()); } void ChatForm::onAvEnd(int FriendId, int) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvEnd"; audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); netcam->hide(); stopCounter(); } void ChatForm::onAvRinging(int FriendId, int CallId, bool video) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvRinging"; callId = CallId; callButton->disconnect(); videoButton->disconnect(); if (video) { callButton->setObjectName("grey"); callButton->style()->polish(callButton); videoButton->setObjectName("yellow"); videoButton->style()->polish(videoButton); connect(videoButton, SIGNAL(clicked()), this, SLOT(onCancelCallTriggered())); } else { callButton->setObjectName("yellow"); callButton->style()->polish(callButton); videoButton->setObjectName("grey"); videoButton->style()->polish(videoButton); connect(callButton, SIGNAL(clicked()), this, SLOT(onCancelCallTriggered())); } addSystemInfoMessage(tr("Calling to %1").arg(f->getDisplayedName()), ChatMessage::INFO, QDateTime::currentDateTime()); } void ChatForm::onAvStarting(int FriendId, int CallId, bool video) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvStarting"; callButton->disconnect(); videoButton->disconnect(); if (video) { callButton->setObjectName("grey"); callButton->style()->polish(callButton); videoButton->setObjectName("red"); videoButton->style()->polish(videoButton); connect(videoButton, SIGNAL(clicked()), this, SLOT(onHangupCallTriggered())); netcam->show(Core::getInstance()->getVideoSourceFromCall(CallId), f->getDisplayedName()); } else { callButton->setObjectName("red"); callButton->style()->polish(callButton); videoButton->setObjectName("grey"); videoButton->style()->polish(videoButton); connect(callButton, SIGNAL(clicked()), this, SLOT(onHangupCallTriggered())); } startCounter(); } void ChatForm::onAvEnding(int FriendId, int) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvEnding"; audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); callButton->disconnect(); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); videoButton->disconnect(); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); netcam->hide(); stopCounter(); } void ChatForm::onAvRequestTimeout(int FriendId, int) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvRequestTimeout"; audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); callButton->disconnect(); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); videoButton->disconnect(); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); netcam->hide(); } void ChatForm::onAvPeerTimeout(int FriendId, int) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvPeerTimeout"; audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); callButton->disconnect(); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); videoButton->disconnect(); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); netcam->hide(); } void ChatForm::onAvRejected(int FriendId, int) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvRejected"; audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); callButton->disconnect(); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); videoButton->disconnect(); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); insertChatMessage(ChatMessage::createChatInfoMessage(tr("Call rejected"), ChatMessage::INFO, QDateTime::currentDateTime())); netcam->hide(); } void ChatForm::onAvMediaChange(int FriendId, int CallId, bool video) { if (FriendId != f->getFriendID() || CallId != callId) return; qDebug() << "onAvMediaChange"; if (video) { netcam->show(Core::getInstance()->getVideoSourceFromCall(CallId), f->getDisplayedName()); } else { netcam->hide(); } } void ChatForm::onAnswerCallTriggered() { qDebug() << "onAnswerCallTriggered"; audioInputFlag = true; audioOutputFlag = true; emit answerCall(callId); } void ChatForm::onHangupCallTriggered() { qDebug() << "onHangupCallTriggered"; audioInputFlag = false; audioOutputFlag = false; emit hangupCall(callId); micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); } void ChatForm::onCallTriggered() { qDebug() << "onCallTriggered"; audioInputFlag = true; audioOutputFlag = true; callButton->disconnect(); videoButton->disconnect(); emit startCall(f->getFriendID()); } void ChatForm::onVideoCallTriggered() { qDebug() << "onVideoCallTriggered"; audioInputFlag = true; audioOutputFlag = true; callButton->disconnect(); videoButton->disconnect(); emit startVideoCall(f->getFriendID(), true); } void ChatForm::onAvCallFailed(int FriendId) { if (FriendId != f->getFriendID()) return; qDebug() << "onAvCallFailed"; audioInputFlag = false; audioOutputFlag = false; callButton->disconnect(); videoButton->disconnect(); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); } void ChatForm::onCancelCallTriggered() { qDebug() << "onCancelCallTriggered"; audioInputFlag = false; audioOutputFlag = false; micButton->setObjectName("green"); micButton->style()->polish(micButton); volButton->setObjectName("green"); volButton->style()->polish(volButton); callButton->disconnect(); videoButton->disconnect(); callButton->setObjectName("green"); callButton->style()->polish(callButton); callButton->disconnect(); videoButton->setObjectName("green"); videoButton->style()->polish(videoButton); videoButton->disconnect(); connect(callButton, SIGNAL(clicked()), this, SLOT(onCallTriggered())); connect(videoButton, SIGNAL(clicked()), this, SLOT(onVideoCallTriggered())); netcam->hide(); emit cancelCall(callId, f->getFriendID()); } void ChatForm::onMicMuteToggle() { if (audioInputFlag == true) { emit micMuteToggle(callId); if (micButton->objectName() == "red") micButton->setObjectName("green"); else micButton->setObjectName("red"); Style::repolish(micButton); } } void ChatForm::onVolMuteToggle() { if (audioOutputFlag == true) { emit volMuteToggle(callId); if (volButton->objectName() == "red") volButton->setObjectName("green"); else volButton->setObjectName("red"); Style::repolish(volButton); } } void ChatForm::onFileSendFailed(int FriendId, const QString &fname) { if (FriendId != f->getFriendID()) return; addSystemInfoMessage(tr("Failed to send file \"%1\"").arg(fname), ChatMessage::ERROR, QDateTime::currentDateTime()); } void ChatForm::onAvatarChange(int FriendId, const QPixmap &pic) { if (FriendId != f->getFriendID()) return; avatar->setPixmap(pic); } void ChatForm::dragEnterEvent(QDragEnterEvent *ev) { if (ev->mimeData()->hasUrls()) ev->acceptProposedAction(); } void ChatForm::dropEvent(QDropEvent *ev) { if (ev->mimeData()->hasUrls()) { for (QUrl url : ev->mimeData()->urls()) { QFileInfo info(url.path()); if (info.exists()) Core::getInstance()->sendFile(f->getFriendID(), info.fileName(), info.absoluteFilePath(), info.size()); } } } void ChatForm::onAvatarRemoved(int FriendId) { if (FriendId != f->getFriendID()) return; avatar->setPixmap(QPixmap(":/img/contact_dark.png"), Qt::transparent); } void ChatForm::loadHistory(QDateTime since, bool processUndelivered) { QDateTime now = QDateTime::currentDateTime(); if (since > now) return; if (!earliestMessage.isNull()) { if (earliestMessage < since) return; if (earliestMessage < now) { now = earliestMessage; now = now.addMSecs(-1); } } auto msgs = HistoryKeeper::getInstance()->getChatHistory(HistoryKeeper::ctSingle, f->getToxID().publicKey, since, now); ToxID storedPrevId = previousId; ToxID prevId; QList historyMessages; QDate lastDate(1,0,0); for (const auto &it : msgs) { // Show the date every new day QDateTime msgDateTime = it.timestamp.toLocalTime(); QDate msgDate = msgDateTime.date(); if (msgDate > lastDate) { lastDate = msgDate; historyMessages.prepend(ChatMessage::createChatInfoMessage(msgDate.toString(), ChatMessage::INFO, QDateTime::currentDateTime())); } // Show each messages ToxID authorId = ToxID::fromString(it.sender); QString authorStr = authorId.isMine() ? Core::getInstance()->getUsername() : resolveToxID(authorId); bool isAction = it.message.startsWith("/me "); ChatMessage::Ptr msg = ChatMessage::createChatMessage(authorStr, isAction ? it.message.right(it.message.length() - 4) : it.message, isAction, false, authorId.isMine(), QDateTime()); if(!isAction && prevId == authorId) msg->hideSender(); prevId = authorId; if (it.isSent || !authorId.isMine()) { msg->markAsSent(msgDateTime); } else { if (processUndelivered) { int rec; if (!isAction) rec = Core::getInstance()->sendMessage(f->getFriendID(), msg->toString()); else rec = Core::getInstance()->sendAction(f->getFriendID(), msg->toString()); registerReceipt(rec, it.id, msg); } } historyMessages.prepend(msg); } previousId = storedPrevId; int savedSliderPos = chatWidget->verticalScrollBar()->maximum() - chatWidget->verticalScrollBar()->value(); earliestMessage = since; chatWidget->insertChatlineOnTop(historyMessages); savedSliderPos = chatWidget->verticalScrollBar()->maximum() - savedSliderPos; chatWidget->verticalScrollBar()->setValue(savedSliderPos); } void ChatForm::onLoadHistory() { LoadHistoryDialog dlg; if (dlg.exec()) { QDateTime fromTime = dlg.getFromDate(); loadHistory(fromTime); } } void ChatForm::startCounter() { if (!timer) { timer = new QTimer(); connect(timer, SIGNAL(timeout()), this, SLOT(updateTime())); timer->start(1000); timeElapsed.start(); callDuration->show(); } } void ChatForm::stopCounter() { if (timer) { addSystemInfoMessage(tr("Call with %1 ended. %2").arg(f->getDisplayedName(),secondsToDHMS(timeElapsed.elapsed()/1000)), ChatMessage::INFO, QDateTime::currentDateTime()); timer->stop(); callDuration->setText(""); callDuration->hide(); timer = nullptr; delete timer; } } void ChatForm::updateTime() { callDuration->setText(secondsToDHMS(timeElapsed.elapsed()/1000)); } QString ChatForm::secondsToDHMS(quint32 duration) { QString res; QString cD = tr("Call duration: "); int seconds = (int) (duration % 60); duration /= 60; int minutes = (int) (duration % 60); duration /= 60; int hours = (int) (duration % 24); int days = (int) (duration / 24); if (minutes == 0) return cD + res.sprintf("%02ds", seconds); if (hours == 0 && days == 0) return cD + res.sprintf("%02dm %02ds", minutes, seconds); if (days == 0) return cD + res.sprintf("%02dh %02dm %02ds", hours, minutes, seconds); //I assume no one will ever have call longer than ~30days return cD + res.sprintf("%dd%02dh %02dm %02ds", days, hours, minutes, seconds); } void ChatForm::registerReceipt(int receipt, int messageID, ChatMessage::Ptr msg) { receipts[receipt] = messageID; undeliveredMsgs[messageID] = msg; } void ChatForm::dischargeReceipt(int receipt) { auto it = receipts.find(receipt); if (it != receipts.end()) { int mID = it.value(); auto msgIt = undeliveredMsgs.find(mID); if (msgIt != undeliveredMsgs.end()) { HistoryKeeper::getInstance()->markAsSent(mID); msgIt.value()->markAsSent(QDateTime::currentDateTime()); undeliveredMsgs.erase(msgIt); } receipts.erase(it); } } void ChatForm::setFriendTyping(bool isTyping) { chatWidget->setTypingNotificationVisible(isTyping); Text* text = dynamic_cast(chatWidget->getTypingNotification()->getContent(1)); if(text) text->setText("
" + tr("%1 is typing...").arg(f->getDisplayedName()) + "
"); } void ChatForm::clearReciepts() { receipts.clear(); undeliveredMsgs.clear(); } void ChatForm::deliverOfflineMsgs() { if (!Settings::getInstance().getFauxOfflineMessaging()) return; QMap msgs = undeliveredMsgs; clearReciepts(); for (auto iter = msgs.begin(); iter != msgs.end(); iter++) { QString messageText = iter.value()->toString(); int rec; if (iter.value()->isAction()) rec = Core::getInstance()->sendAction(f->getFriendID(), messageText); else rec = Core::getInstance()->sendMessage(f->getFriendID(), messageText); registerReceipt(rec, iter.key(), iter.value()); } }