1
0
mirror of https://github.com/qTox/qTox.git synced 2024-03-22 14:00:36 +08:00
qTox/src/chatlog/chatmessage.cpp

313 lines
12 KiB
C++
Raw Normal View History

2014-11-16 19:58:43 +08:00
/*
Copyright © 2014-2015 by The qTox Project
2014-11-16 19:58:43 +08:00
This file is part of qTox, a Qt-based graphical interface for Tox.
qTox is libre software: you can redistribute it and/or modify
2014-11-16 19:58:43 +08:00
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.
qTox is distributed in the hope that it will be useful,
2014-11-16 19:58:43 +08:00
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.
2014-11-16 19:58:43 +08:00
You should have received a copy of the GNU General Public License
along with qTox. If not, see <http://www.gnu.org/licenses/>.
2014-11-16 19:58:43 +08:00
*/
2014-11-12 21:11:25 +08:00
#include "chatmessage.h"
2015-01-03 22:03:33 +08:00
#include "chatlinecontentproxy.h"
2014-11-12 23:45:24 +08:00
#include "content/text.h"
#include "content/timestamp.h"
2014-11-12 23:45:24 +08:00
#include "content/spinner.h"
2015-01-03 22:03:33 +08:00
#include "content/filetransferwidget.h"
#include "content/image.h"
2015-01-20 02:04:19 +08:00
#include "content/notificationicon.h"
2014-11-12 21:11:25 +08:00
#include <QDebug>
2015-06-06 07:44:47 +08:00
#include "src/persistence/settings.h"
#include "src/persistence/smileypack.h"
#include "src/widget/style.h"
2015-01-03 03:07:45 +08:00
2015-01-11 05:21:33 +08:00
#define NAME_COL_WIDTH 90.0
2015-01-05 21:06:14 +08:00
#define TIME_COL_WIDTH 90.0
2014-11-12 23:45:24 +08:00
2015-01-05 01:21:35 +08:00
ChatMessage::ChatMessage()
2014-11-12 23:45:24 +08:00
{
2014-11-16 19:58:43 +08:00
2014-11-12 23:45:24 +08:00
}
2015-02-15 17:51:54 +08:00
ChatMessage::Ptr ChatMessage::createChatMessage(const QString &sender, const QString &rawMessage, MessageType type, bool isMe, const QDateTime &date)
2015-01-03 22:03:33 +08:00
{
2015-01-05 01:21:35 +08:00
ChatMessage::Ptr msg = ChatMessage::Ptr(new ChatMessage);
2015-01-03 22:03:33 +08:00
2015-02-16 22:22:29 +08:00
QString text = rawMessage.toHtmlEscaped();
2015-02-15 17:51:54 +08:00
QString senderText = sender;
const QColor actionColor = QColor("#1818FF"); // has to match the color in innerStyle.css (div.action)
2015-01-04 02:05:38 +08:00
2015-01-11 05:21:33 +08:00
//smileys
if (Settings::getInstance().getUseEmoticons())
2015-01-04 02:05:38 +08:00
text = SmileyPack::getInstance().smileyfied(text);
//quotes (green text)
text = detectQuotes(detectAnchors(text), type);
2015-01-03 22:03:33 +08:00
//markdown
if (Settings::getInstance().getMarkdownPreference() != NONE)
text = detectMarkdown(text);
2015-02-15 17:51:54 +08:00
switch(type)
2015-01-03 22:03:33 +08:00
{
2015-02-15 17:51:54 +08:00
case ACTION:
senderText = "*";
text = wrapDiv(QString("%1 %2").arg(sender.toHtmlEscaped(), text), "action");
2015-01-03 22:03:33 +08:00
msg->setAsAction();
2015-02-15 17:51:54 +08:00
break;
case ALERT:
text = wrapDiv(text, "alert");
break;
default:
text = wrapDiv(text, "msg");
2015-01-04 02:06:10 +08:00
}
2015-01-03 22:03:33 +08:00
2015-02-15 17:51:54 +08:00
// Note: Eliding cannot be enabled for RichText items. (QTBUG-17207)
msg->addColumn(new Text(senderText, isMe ? Style::getFont(Style::BigBold) : Style::getFont(Style::Big), true, sender, type == ACTION ? actionColor : Qt::black), ColumnFormat(NAME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
2015-05-27 06:28:19 +08:00
msg->addColumn(new Text(text, Style::getFont(Style::Big), false, ((type == ACTION) && isMe) ? QString("%1 %2").arg(sender, rawMessage) : rawMessage), ColumnFormat(1.0, ColumnFormat::VariableSize));
2015-02-06 19:21:13 +08:00
msg->addColumn(new Spinner(":/ui/chatArea/spinner.svg", QSize(16, 16), 360.0/1.6), ColumnFormat(TIME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
2015-01-03 22:03:33 +08:00
if (!date.isNull())
2015-01-03 22:03:33 +08:00
msg->markAsSent(date);
return msg;
}
2015-01-05 01:21:35 +08:00
ChatMessage::Ptr ChatMessage::createChatInfoMessage(const QString &rawMessage, SystemMessageType type, const QDateTime &date)
2015-01-03 22:03:33 +08:00
{
2015-01-05 01:21:35 +08:00
ChatMessage::Ptr msg = ChatMessage::Ptr(new ChatMessage);
2015-02-16 22:22:29 +08:00
QString text = rawMessage.toHtmlEscaped();
2015-01-03 22:03:33 +08:00
2015-01-06 21:30:24 +08:00
QString img;
switch(type)
{
case INFO: img = ":/ui/chatArea/info.svg"; break;
case ERROR: img = ":/ui/chatArea/error.svg"; break;
case TYPING: img = ":/ui/chatArea/typing.svg"; break;
2015-01-06 21:30:24 +08:00
}
2015-01-27 18:20:35 +08:00
msg->addColumn(new Image(QSize(18, 18), img), ColumnFormat(NAME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
2015-01-27 17:52:23 +08:00
msg->addColumn(new Text("<b>" + text + "</b>", Style::getFont(Style::Big), false, ""), ColumnFormat(1.0, ColumnFormat::VariableSize, ColumnFormat::Left));
msg->addColumn(new Timestamp(date, Settings::getInstance().getTimestampFormat(), Style::getFont(Style::Big)), ColumnFormat(TIME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
2015-01-03 22:03:33 +08:00
return msg;
}
2015-01-05 01:21:35 +08:00
ChatMessage::Ptr ChatMessage::createFileTransferMessage(const QString& sender, ToxFile file, bool isMe, const QDateTime& date)
2015-01-03 22:03:33 +08:00
{
2015-01-05 01:21:35 +08:00
ChatMessage::Ptr msg = ChatMessage::Ptr(new ChatMessage);
2015-01-03 22:03:33 +08:00
msg->addColumn(new Text(sender, isMe ? Style::getFont(Style::BigBold) : Style::getFont(Style::Big), true), ColumnFormat(NAME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
2015-02-16 00:21:00 +08:00
msg->addColumn(new ChatLineContentProxy(new FileTransferWidget(0, file), 320, 0.6f), ColumnFormat(1.0, ColumnFormat::VariableSize));
msg->addColumn(new Timestamp(date, Settings::getInstance().getTimestampFormat(), Style::getFont(Style::Big)), ColumnFormat(TIME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
2015-01-03 22:03:33 +08:00
return msg;
}
ChatMessage::Ptr ChatMessage::createTypingNotification()
{
ChatMessage::Ptr msg = ChatMessage::Ptr(new ChatMessage);
// Note: "[user]..." is just a placeholder. The actual text is set in ChatForm::setFriendTyping()
//
// FIXME: Due to circumstances, placeholder is being used in a case where
// user received typing notifications constantly since contact came online.
// This causes "[user]..." to be displayed in place of user nick, as long
// as user will keep typing. Issue #1280
2015-02-02 18:01:01 +08:00
msg->addColumn(new NotificationIcon(QSize(18, 18)), ColumnFormat(NAME_COL_WIDTH, ColumnFormat::FixedSize, ColumnFormat::Right));
msg->addColumn(new Text("[user]...", Style::getFont(Style::Big), false, ""), ColumnFormat(1.0, ColumnFormat::VariableSize, ColumnFormat::Left));
return msg;
}
ChatMessage::Ptr ChatMessage::createBusyNotification()
{
ChatMessage::Ptr msg = ChatMessage::Ptr(new ChatMessage);
// TODO: Bigger font
msg->addColumn(new Text(QObject::tr("Resizing"), Style::getFont(Style::ExtraBig), false, ""), ColumnFormat(1.0, ColumnFormat::VariableSize, ColumnFormat::Center));
return msg;
}
2014-11-12 23:45:24 +08:00
void ChatMessage::markAsSent(const QDateTime &time)
2014-11-12 21:11:25 +08:00
{
2014-11-12 23:45:24 +08:00
// remove the spinner and replace it by $time
replaceContent(2, new Timestamp(time, Settings::getInstance().getTimestampFormat(), Style::getFont(Style::Big)));
2014-11-12 21:11:25 +08:00
}
2014-11-14 01:27:32 +08:00
QString ChatMessage::toString() const
{
2015-01-05 01:21:35 +08:00
ChatLineContent* c = getContent(1);
if (c)
2015-01-05 01:21:35 +08:00
return c->getText();
return QString();
2014-11-14 01:27:32 +08:00
}
2014-12-14 04:11:03 +08:00
bool ChatMessage::isAction() const
{
return action;
}
void ChatMessage::setAsAction()
{
action = true;
}
2015-01-03 22:03:33 +08:00
2015-01-05 01:21:35 +08:00
void ChatMessage::hideSender()
{
ChatLineContent* c = getContent(0);
if (c)
2015-01-05 01:21:35 +08:00
c->hide();
}
void ChatMessage::hideDate()
{
ChatLineContent* c = getContent(2);
if (c)
2015-01-05 01:21:35 +08:00
c->hide();
}
QString ChatMessage::detectMarkdown(const QString &str)
{
QString out = str;
// Create regex for certain markdown syntax
QRegExp exp("(\\*\\*)([^\\*\\*]{2,})(\\*\\*)" // Bold **text**
"|(\\*)([^\\*]{2,})(\\*)" // Italics *text*
"|(\\_)([^\\_]{2,})(\\_)" // Italics _text_
"|(\\_\\_)([^\\_\\_]{2,})(\\_\\_)" // Bold __text__
"|(\\-)([^\\-]{2,})(\\-)" // Underline -text-
"|(\\~)([^\\~]{2,})(\\~)" // Strike ~text~
"|(\\~~)([^\\~\\~]{2,})(\\~~)" // Strike ~~text~~
"|(\\`)([^\\`]{2,})(\\`)" // Codeblock `text`
);
int offset = 0;
while ((offset = exp.indexIn(out, offset)) != -1)
{
QString snipCheck = out.mid(offset-1,exp.cap(0).length()+2);
QString snippet = exp.cap(0).trimmed();
QString htmledSnippet;
// Only parse if surrounded by spaces, newline(s) and/or beginning/end of line
if ((snipCheck.startsWith(' ') || snipCheck.startsWith('>') || offset == 0) && ((snipCheck.endsWith(' ') || snipCheck.endsWith('<')) || offset + snippet.toHtmlEscaped().length() == out.toHtmlEscaped().length()))
{
int mul = 0; // Determines how many characters to strip from markdown text
// Set mul depending on markdownPreference
if (Settings::getInstance().getMarkdownPreference() == WITHOUT_CHARS)
mul = 2;
// Match captured string to corresponding md format
if (exp.cap(1) == "**") // Bold **text**
htmledSnippet = QString(" <b>%1</b> ").arg(snippet.mid(mul,snippet.length()-2*mul));
else if (exp.cap(4) == "*" && snippet.length() > 2) // Italics *text*
htmledSnippet = QString(" <i>%1</i> ").arg(snippet.mid(mul/2,snippet.length()-mul));
else if (exp.cap(7) == "_" && snippet.length() > 2) // Italics _text_
htmledSnippet = QString(" <i>%1</i> ").arg(snippet.mid(mul/2,snippet.length()-mul));
else if (exp.cap(10) == "__"&& snippet.length() > 4) // Bold __text__
htmledSnippet = QString(" <b>%1</b> ").arg(snippet.mid(mul,snippet.length()-2*mul));
else if (exp.cap(13) == "-" && snippet.length() > 2) // Underline -text-
htmledSnippet = QString(" <u>%1</u> ").arg(snippet.mid(mul/2,snippet.length()-mul));
else if (exp.cap(16) == "~" && snippet.length() > 2) // Strikethrough ~text~
htmledSnippet = QString(" <s>%1</s> ").arg(snippet.mid(mul/2,snippet.length()-mul));
else if (exp.cap(19) == "~~" && snippet.length() > 4) // Strikethrough ~~text~~
htmledSnippet = QString(" <s>%1</s> ").arg(snippet.mid(mul,snippet.length()-2*mul));
else if (exp.cap(22) == "`" && snippet.length() > 2) // Codeblock `text`
htmledSnippet = QString("<font color=#595959><code>%1</code></font>").arg(snippet.mid(mul/2,snippet.length()-mul));
else
htmledSnippet = snippet;
out.replace(offset, exp.cap().length(), htmledSnippet);
offset += htmledSnippet.length();
} else
offset += snippet.length();
}
return out;
}
2015-01-03 22:03:33 +08:00
QString ChatMessage::detectAnchors(const QString &str)
{
QString out = str;
2015-01-03 22:03:33 +08:00
// detect URIs
QRegExp exp("("
"(?:\\b)((www\\.)|(http[s]?|ftp)://)" // (protocol)://(printable - non-special character)
// http://ONEORMOREALHPA-DIGIT
"\\w+\\S+)" // any other character, lets domains and other
// ↓ link to a file, or samba share
// https://en.wikipedia.org/wiki/File_URI_scheme
"|(?:\\b)((file|smb)://)([\\S| ]*)"
"|(?:\\b)(tox:[a-zA-Z\\d]{76})" //link with full user address
2015-09-29 20:33:26 +08:00
"|(?:\\b)(mailto:\\S+@\\S+\\.\\S+)" //@mail link
"|(?:\\b)(tox:\\S+@\\S+)"); // starts with `tox` then : and only alpha-digits till the end
// also accepts tox:agilob@net as simplified TOX ID
int offset = 0;
while ((offset = exp.indexIn(out, offset)) != -1)
2015-01-03 22:03:33 +08:00
{
QString url = exp.cap();
// If there's a trailing " it's a HTML attribute, e.g. a smiley img's title=":tox:"
if (url == "tox:\"")
2015-01-03 22:03:33 +08:00
{
offset += url.length();
continue;
2015-01-03 22:03:33 +08:00
}
QString htmledUrl;
// add scheme if not specified
if (exp.cap(2) == "www.")
htmledUrl = QString("<a href=\"http://%1\">%1</a>").arg(url);
else
htmledUrl = QString("<a href=\"%1\">%1</a>").arg(url);
out.replace(offset, exp.cap().length(), htmledUrl);
offset += htmledUrl.length();
2015-01-03 22:03:33 +08:00
}
return out;
2015-01-03 22:03:33 +08:00
}
QString ChatMessage::detectQuotes(const QString& str, MessageType type)
2015-01-03 22:03:33 +08:00
{
// detect text quotes
QStringList messageLines = str.split("\n");
QString quotedText;
for (int i = 0; i < messageLines.size(); ++i)
2015-01-03 22:03:33 +08:00
{
// don't quote first line in action message. This makes co-existence of
// quotes and action messages possible, since only first line can cause
// problems in case where there is quote in it used.
if (QRegExp("^(&gt;|).*").exactMatch(messageLines[i])) {
if (i > 0 || type != ACTION)
quotedText += "<span class=quote>" + messageLines[i] + "</span>";
else
quotedText += messageLines[i];
} else {
2015-01-03 22:03:33 +08:00
quotedText += messageLines[i];
}
2015-01-03 22:03:33 +08:00
if (i < messageLines.size() - 1)
quotedText += "<br/>";
}
return quotedText;
}
2015-02-15 17:51:54 +08:00
QString ChatMessage::wrapDiv(const QString &str, const QString &div)
{
return QString("<p class=%1>%2</p>").arg(div, /*QChar(0x200E) + */QString(str));
2015-02-15 17:51:54 +08:00
}