2018-01-11 20:16:25 +01:00
|
|
|
#include "messages/messageelement.hpp"
|
2018-04-27 22:11:19 +02:00
|
|
|
|
|
|
|
#include "application.hpp"
|
2018-01-11 20:16:25 +01:00
|
|
|
#include "messages/layouts/messagelayoutcontainer.hpp"
|
|
|
|
#include "messages/layouts/messagelayoutelement.hpp"
|
|
|
|
#include "singletons/settingsmanager.hpp"
|
|
|
|
#include "util/benchmark.hpp"
|
|
|
|
#include "util/emotemap.hpp"
|
|
|
|
|
|
|
|
namespace chatterino {
|
|
|
|
namespace messages {
|
|
|
|
|
|
|
|
MessageElement::MessageElement(Flags _flags)
|
|
|
|
: flags(_flags)
|
|
|
|
{
|
2018-04-06 16:37:30 +02:00
|
|
|
util::DebugCount::increase("message elements");
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageElement::~MessageElement()
|
|
|
|
{
|
|
|
|
util::DebugCount::decrease("message elements");
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
MessageElement *MessageElement::setLink(const Link &_link)
|
|
|
|
{
|
|
|
|
this->link = _link;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageElement *MessageElement::setTooltip(const QString &_tooltip)
|
|
|
|
{
|
|
|
|
this->tooltip = _tooltip;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageElement *MessageElement::setTrailingSpace(bool value)
|
|
|
|
{
|
|
|
|
this->trailingSpace = value;
|
2018-01-12 18:42:13 +01:00
|
|
|
return this;
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const QString &MessageElement::getTooltip() const
|
|
|
|
{
|
|
|
|
return this->tooltip;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Link &MessageElement::getLink() const
|
|
|
|
{
|
|
|
|
return this->link;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MessageElement::hasTrailingSpace() const
|
|
|
|
{
|
|
|
|
return this->trailingSpace;
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageElement::Flags MessageElement::getFlags() const
|
|
|
|
{
|
|
|
|
return this->flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
// IMAGE
|
2018-01-17 14:14:31 +01:00
|
|
|
ImageElement::ImageElement(Image *_image, MessageElement::Flags flags)
|
2018-01-11 20:16:25 +01:00
|
|
|
: MessageElement(flags)
|
|
|
|
, image(_image)
|
|
|
|
{
|
2018-01-17 14:14:31 +01:00
|
|
|
this->setTooltip(_image->getTooltip());
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ImageElement::addToContainer(MessageLayoutContainer &container, MessageElement::Flags _flags)
|
|
|
|
{
|
2018-01-22 22:38:44 +01:00
|
|
|
if (_flags & this->getFlags()) {
|
2018-01-28 16:29:47 +01:00
|
|
|
QSize size(this->image->getWidth() * this->image->getScale() * container.getScale(),
|
|
|
|
this->image->getHeight() * this->image->getScale() * container.getScale());
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
container.addElement(
|
|
|
|
(new ImageLayoutElement(*this, this->image, size))->setLink(this->getLink()));
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// EMOTE
|
|
|
|
EmoteElement::EmoteElement(const util::EmoteData &_data, MessageElement::Flags flags)
|
|
|
|
: MessageElement(flags)
|
|
|
|
, data(_data)
|
|
|
|
{
|
|
|
|
if (_data.isValid()) {
|
|
|
|
this->setTooltip(data.image1x->getTooltip());
|
2018-04-03 02:55:32 +02:00
|
|
|
this->textElement.reset(new TextElement(_data.image1x->getName(), MessageElement::Misc));
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
2018-01-22 22:38:44 +01:00
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
void EmoteElement::addToContainer(MessageLayoutContainer &container, MessageElement::Flags _flags)
|
|
|
|
{
|
|
|
|
if (_flags & this->getFlags()) {
|
2018-01-27 21:13:22 +01:00
|
|
|
if (_flags & MessageElement::EmoteImages) {
|
2018-01-22 22:38:44 +01:00
|
|
|
if (!this->data.isValid()) {
|
|
|
|
return;
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-04-27 22:11:19 +02:00
|
|
|
int quality = getApp()->settings->preferredEmoteQuality;
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
Image *_image;
|
|
|
|
if (quality == 3 && this->data.image3x != nullptr) {
|
|
|
|
_image = this->data.image3x;
|
|
|
|
} else if (quality >= 2 && this->data.image2x != nullptr) {
|
|
|
|
_image = this->data.image2x;
|
|
|
|
} else {
|
|
|
|
_image = this->data.image1x;
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-01-28 16:29:47 +01:00
|
|
|
QSize size((int)(container.getScale() * _image->getScaledWidth()),
|
|
|
|
(int)(container.getScale() * _image->getScaledHeight()));
|
2018-01-22 22:38:44 +01:00
|
|
|
|
|
|
|
container.addElement(
|
|
|
|
(new ImageLayoutElement(*this, _image, size))->setLink(this->getLink()));
|
|
|
|
} else {
|
2018-04-03 02:55:32 +02:00
|
|
|
if (this->textElement) {
|
2018-01-22 22:38:44 +01:00
|
|
|
this->textElement->addToContainer(container, MessageElement::Misc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TEXT
|
|
|
|
TextElement::TextElement(const QString &text, MessageElement::Flags flags,
|
|
|
|
const MessageColor &_color, FontStyle _style)
|
|
|
|
: MessageElement(flags)
|
|
|
|
, color(_color)
|
|
|
|
, style(_style)
|
|
|
|
{
|
|
|
|
for (QString word : text.split(' ')) {
|
|
|
|
this->words.push_back({word, -1});
|
2018-04-03 02:55:32 +02:00
|
|
|
// fourtf: add logic to store multiple spaces after message
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextElement::addToContainer(MessageLayoutContainer &container, MessageElement::Flags _flags)
|
|
|
|
{
|
2018-04-27 22:11:19 +02:00
|
|
|
auto app = getApp();
|
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
if (_flags & this->getFlags()) {
|
2018-05-23 04:22:17 +02:00
|
|
|
QFontMetrics metrics = app->fonts->getFontMetrics(this->style, container.getScale());
|
2018-01-22 22:38:44 +01:00
|
|
|
|
|
|
|
for (Word &word : this->words) {
|
|
|
|
auto getTextLayoutElement = [&](QString text, int width, bool trailingSpace) {
|
2018-04-27 22:11:19 +02:00
|
|
|
QColor color = this->color.getColor(*app->themes);
|
|
|
|
app->themes->normalizeColor(color);
|
2018-01-22 22:38:44 +01:00
|
|
|
|
|
|
|
auto e = (new TextLayoutElement(*this, text, QSize(width, metrics.height()), color,
|
2018-01-28 16:29:47 +01:00
|
|
|
this->style, container.getScale()))
|
2018-01-22 22:38:44 +01:00
|
|
|
->setLink(this->getLink());
|
|
|
|
e->setTrailingSpace(trailingSpace);
|
|
|
|
return e;
|
|
|
|
};
|
|
|
|
|
2018-01-23 22:00:58 +01:00
|
|
|
// fourtf: add again
|
|
|
|
// if (word.width == -1) {
|
|
|
|
word.width = metrics.width(word.text);
|
|
|
|
// }
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
// see if the text fits in the current line
|
2018-01-11 20:16:25 +01:00
|
|
|
if (container.fitsInLine(word.width)) {
|
2018-01-16 00:56:17 +01:00
|
|
|
container.addElementNoLineBreak(
|
|
|
|
getTextLayoutElement(word.text, word.width, this->hasTrailingSpace()));
|
2018-01-11 20:16:25 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
// see if the text fits in the next line
|
|
|
|
if (!container.atStartOfLine()) {
|
2018-01-11 20:16:25 +01:00
|
|
|
container.breakLine();
|
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
if (container.fitsInLine(word.width)) {
|
|
|
|
container.addElementNoLineBreak(
|
|
|
|
getTextLayoutElement(word.text, word.width, this->hasTrailingSpace()));
|
|
|
|
continue;
|
2018-01-16 00:56:17 +01:00
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
// we done goofed, we need to wrap the text
|
|
|
|
QString text = word.text;
|
|
|
|
int textLength = text.length();
|
|
|
|
int wordStart = 0;
|
|
|
|
int width = metrics.width(text[0]);
|
|
|
|
int lastWidth = 0;
|
|
|
|
|
|
|
|
for (int i = 1; i < textLength; i++) {
|
|
|
|
int charWidth = metrics.width(text[i]);
|
|
|
|
|
|
|
|
if (!container.fitsInLine(width + charWidth)) {
|
2018-04-10 02:42:41 +02:00
|
|
|
container.addElementNoLineBreak(
|
|
|
|
getTextLayoutElement(text.mid(wordStart, i - wordStart), width, false));
|
2018-01-22 22:38:44 +01:00
|
|
|
container.breakLine();
|
|
|
|
|
|
|
|
wordStart = i;
|
|
|
|
lastWidth = width;
|
|
|
|
width = 0;
|
|
|
|
if (textLength > i + 2) {
|
|
|
|
width += metrics.width(text[i]);
|
|
|
|
width += metrics.width(text[i + 1]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
width += charWidth;
|
|
|
|
}
|
|
|
|
|
2018-04-14 22:32:22 +02:00
|
|
|
UNUSED(lastWidth); // XXX: What should this be used for (if anything)? KKona
|
|
|
|
|
2018-04-10 03:17:44 +02:00
|
|
|
container.addElement(
|
|
|
|
getTextLayoutElement(text.mid(wordStart), width, this->hasTrailingSpace()));
|
2018-01-22 22:38:44 +01:00
|
|
|
container.breakLine();
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TIMESTAMP
|
|
|
|
TimestampElement::TimestampElement(QTime _time)
|
|
|
|
: MessageElement(MessageElement::Timestamp)
|
|
|
|
, time(_time)
|
2018-04-03 02:55:32 +02:00
|
|
|
, element(this->formatTime(_time))
|
2018-01-11 20:16:25 +01:00
|
|
|
{
|
|
|
|
assert(this->element != nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TimestampElement::addToContainer(MessageLayoutContainer &container,
|
|
|
|
MessageElement::Flags _flags)
|
|
|
|
{
|
2018-01-22 22:38:44 +01:00
|
|
|
if (_flags & this->getFlags()) {
|
2018-04-27 22:11:19 +02:00
|
|
|
auto app = getApp();
|
|
|
|
if (app->settings->timestampFormat != this->format) {
|
|
|
|
this->format = app->settings->timestampFormat.getValue();
|
2018-04-03 02:55:32 +02:00
|
|
|
this->element.reset(this->formatTime(this->time));
|
2018-01-22 22:38:44 +01:00
|
|
|
}
|
2018-01-12 23:09:05 +01:00
|
|
|
|
2018-01-22 22:38:44 +01:00
|
|
|
this->element->addToContainer(container, _flags);
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TextElement *TimestampElement::formatTime(const QTime &time)
|
|
|
|
{
|
2018-01-17 18:36:12 +01:00
|
|
|
static QLocale locale("en_US");
|
|
|
|
|
2018-04-27 22:11:19 +02:00
|
|
|
QString format = locale.toString(time, getApp()->settings->timestampFormat);
|
2018-01-11 20:16:25 +01:00
|
|
|
|
2018-05-23 04:22:17 +02:00
|
|
|
return new TextElement(format, Flags::Timestamp, MessageColor::System, FontStyle::ChatMedium);
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TWITCH MODERATION
|
|
|
|
TwitchModerationElement::TwitchModerationElement()
|
|
|
|
: MessageElement(MessageElement::ModeratorTools)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void TwitchModerationElement::addToContainer(MessageLayoutContainer &container,
|
|
|
|
MessageElement::Flags _flags)
|
|
|
|
{
|
2018-01-17 16:52:51 +01:00
|
|
|
if (_flags & MessageElement::ModeratorTools) {
|
2018-01-28 16:29:47 +01:00
|
|
|
QSize size((int)(container.getScale() * 16), (int)(container.getScale() * 16));
|
2018-01-17 16:52:51 +01:00
|
|
|
|
2018-04-27 22:11:19 +02:00
|
|
|
for (const singletons::ModerationAction &m : getApp()->settings->getModerationActions()) {
|
2018-01-17 16:52:51 +01:00
|
|
|
if (m.isImage()) {
|
|
|
|
container.addElement((new ImageLayoutElement(*this, m.getImage(), size))
|
|
|
|
->setLink(Link(Link::UserAction, m.getAction())));
|
|
|
|
} else {
|
|
|
|
container.addElement((new TextIconLayoutElement(*this, m.getLine1(), m.getLine2(),
|
2018-01-28 16:29:47 +01:00
|
|
|
container.getScale(), size))
|
2018-01-17 16:52:51 +01:00
|
|
|
->setLink(Link(Link::UserAction, m.getAction())));
|
|
|
|
}
|
2018-01-17 14:14:31 +01:00
|
|
|
}
|
|
|
|
}
|
2018-01-11 20:16:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace messages
|
|
|
|
} // namespace chatterino
|