mirror-chatterino2/src/messages/layouts/MessageLayout.cpp

302 lines
8.7 KiB
C++
Raw Normal View History

2018-06-26 14:09:39 +02:00
#include "messages/layouts/MessageLayout.hpp"
2018-06-26 14:09:39 +02:00
#include "Application.hpp"
2018-06-26 17:20:03 +02:00
#include "debug/Benchmark.hpp"
#include "messages/Message.hpp"
#include "messages/MessageElement.hpp"
#include "messages/layouts/MessageLayoutContainer.hpp"
2018-06-28 19:46:45 +02:00
#include "singletons/Emotes.hpp"
#include "singletons/Settings.hpp"
#include "singletons/Theme.hpp"
2018-06-26 14:09:39 +02:00
#include "singletons/WindowManager.hpp"
2018-08-07 01:35:24 +02:00
#include "util/DebugCount.hpp"
#include <QApplication>
#include <QDebug>
#include <QPainter>
#include <QThread>
#include <QtGlobal>
#define MARGIN_LEFT (int)(8 * this->scale)
#define MARGIN_RIGHT (int)(8 * this->scale)
#define MARGIN_TOP (int)(4 * this->scale)
#define MARGIN_BOTTOM (int)(4 * this->scale)
#define COMPACT_EMOTES_OFFSET 6
namespace chatterino {
2018-05-25 13:02:14 +02:00
MessageLayout::MessageLayout(MessagePtr message)
: message_(message)
, container_(std::make_shared<MessageLayoutContainer>())
{
2018-06-26 17:06:17 +02:00
DebugCount::increase("message layout");
2018-04-06 16:37:30 +02:00
}
MessageLayout::~MessageLayout()
{
2018-06-26 17:06:17 +02:00
DebugCount::decrease("message layout");
}
2018-08-07 01:35:24 +02:00
const Message *MessageLayout::getMessage()
{
return this->message_.get();
}
// Height
int MessageLayout::getHeight() const
{
return container_->getHeight();
}
// Layout
// return true if redraw is required
2018-08-07 07:55:31 +02:00
bool MessageLayout::layout(int width, float scale, MessageElementFlags flags)
{
// BenchmarkGuard benchmark("MessageLayout::layout()");
auto app = getApp();
bool layoutRequired = false;
// check if width changed
bool widthChanged = width != this->currentLayoutWidth_;
layoutRequired |= widthChanged;
this->currentLayoutWidth_ = width;
// check if layout state changed
if (this->layoutState_ != app->windows->getGeneration()) {
layoutRequired = true;
2018-08-07 07:55:31 +02:00
this->flags.set(MessageLayoutFlag::RequiresBufferUpdate);
this->layoutState_ = app->windows->getGeneration();
}
// check if work mask changed
layoutRequired |= this->currentWordFlags_ != flags;
this->currentWordFlags_ = flags; // getSettings()->getWordTypeMask();
2018-05-25 12:45:18 +02:00
// check if layout was requested manually
2018-08-07 07:55:31 +02:00
layoutRequired |= this->flags.has(MessageLayoutFlag::RequiresLayout);
this->flags.unset(MessageLayoutFlag::RequiresLayout);
2018-05-25 12:45:18 +02:00
// check if dpi changed
layoutRequired |= this->scale_ != scale;
this->scale_ = scale;
2018-05-24 23:12:50 +02:00
if (!layoutRequired) {
return false;
}
int oldHeight = this->container_->getHeight();
2018-01-17 16:52:51 +01:00
this->actuallyLayout(width, flags);
if (widthChanged || this->container_->getHeight() != oldHeight) {
this->deleteBuffer();
}
this->invalidateBuffer();
return true;
}
2018-08-07 07:55:31 +02:00
void MessageLayout::actuallyLayout(int width, MessageElementFlags _flags)
{
2018-08-07 07:55:31 +02:00
auto messageFlags = this->message_->flags;
2018-08-07 07:55:31 +02:00
if (this->flags.has(MessageLayoutFlag::Expanded) ||
(_flags.has(MessageElementFlag::ModeratorTools) &&
!this->message_->flags.has(MessageFlag::Disabled))) //
{
messageFlags.unset(MessageFlag::Collapsed);
}
this->container_->begin(width, this->scale_, messageFlags);
2018-08-07 01:35:24 +02:00
for (const auto &element : this->message_->elements) {
element->addToContainer(*this->container_, _flags);
}
if (this->height_ != this->container_->getHeight()) {
this->deleteBuffer();
}
this->container_->end();
this->height_ = this->container_->getHeight();
// collapsed state
2018-08-07 07:55:31 +02:00
this->flags.unset(MessageLayoutFlag::Collapsed);
if (this->container_->isCollapsed()) {
2018-08-07 07:55:31 +02:00
this->flags.set(MessageLayoutFlag::Collapsed);
}
}
// Painting
void MessageLayout::paint(QPainter &painter, int width, int y, int messageIndex,
2018-08-06 21:17:03 +02:00
Selection &selection, bool isLastReadMessage,
bool isWindowFocused)
{
auto app = getApp();
QPixmap *pixmap = this->buffer_.get();
// create new buffer if required
if (!pixmap) {
#ifdef Q_OS_MACOS
pixmap = new QPixmap(int(width * painter.device()->devicePixelRatioF()),
int(container_->getHeight() *
2018-08-06 21:17:03 +02:00
painter.device()->devicePixelRatioF()));
pixmap->setDevicePixelRatio(painter.device()->devicePixelRatioF());
#else
pixmap =
new QPixmap(width, std::max(16, this->container_->getHeight()));
#endif
this->buffer_ = std::shared_ptr<QPixmap>(pixmap);
this->bufferValid_ = false;
2018-06-26 17:06:17 +02:00
DebugCount::increase("message drawing buffers");
}
if (!this->bufferValid_ || !selection.isEmpty()) {
this->updateBuffer(pixmap, messageIndex, selection);
}
// draw on buffer
2018-01-23 22:51:15 +01:00
painter.drawPixmap(0, y, *pixmap);
2018-08-06 21:17:03 +02:00
// painter.drawPixmap(0, y, this->container.width,
// this->container.getHeight(), *pixmap);
// draw gif emotes
this->container_->paintAnimatedElements(painter, y);
// draw disabled
2018-08-07 07:55:31 +02:00
if (this->message_->flags.has(MessageFlag::Disabled)) {
2018-08-06 21:17:03 +02:00
painter.fillRect(0, y, pixmap->width(), pixmap->height(),
app->themes->messages.disabled);
}
2018-04-10 03:29:00 +02:00
// draw selection
if (!selection.isEmpty()) {
this->container_->paintSelection(painter, messageIndex, selection, y);
2018-04-10 03:29:00 +02:00
}
// draw message seperation line
if (getSettings()->separateMessages.getValue()) {
painter.fillRect(0, y, this->container_->getWidth(), 1,
app->themes->splits.messageSeperator);
}
2018-01-23 22:48:33 +01:00
// draw last read message line
if (isLastReadMessage) {
QColor color;
if (getSettings()->lastMessageColor != "") {
color = QColor(getSettings()->lastMessageColor.getValue());
} else {
color =
isWindowFocused
? app->themes->tabs.selected.backgrounds.regular.color()
: app->themes->tabs.selected.backgrounds.unfocused.color();
}
2018-01-23 22:48:33 +01:00
2018-08-06 21:17:03 +02:00
QBrush brush(color, static_cast<Qt::BrushStyle>(
getSettings()->lastMessagePattern.getValue()));
2018-01-23 22:48:33 +01:00
painter.fillRect(0, y + this->container_->getHeight() - 1,
2018-08-06 21:17:03 +02:00
pixmap->width(), 1, brush);
2018-01-23 22:48:33 +01:00
}
this->bufferValid_ = true;
}
2018-08-06 21:17:03 +02:00
void MessageLayout::updateBuffer(QPixmap *buffer, int /*messageIndex*/,
Selection & /*selection*/)
{
auto app = getApp();
QPainter painter(buffer);
painter.setRenderHint(QPainter::SmoothPixmapTransform);
// draw background
2018-10-21 13:29:52 +02:00
QColor backgroundColor = app->themes->messages.backgrounds.regular;
if (this->message_->flags.has(MessageFlag::Highlighted) &&
!this->flags.has(MessageLayoutFlag::IgnoreHighlights)) {
backgroundColor = app->themes->messages.backgrounds.highlighted;
2018-08-07 07:55:31 +02:00
} else if (this->message_->flags.has(MessageFlag::Subscription)) {
2018-06-04 12:23:23 +02:00
backgroundColor = app->themes->messages.backgrounds.subscription;
} else if (getSettings()->alternateMessageBackground.getValue() &&
2018-08-07 07:55:31 +02:00
this->flags.has(MessageLayoutFlag::AlternateBackground)) {
backgroundColor = app->themes->messages.backgrounds.alternate;
}
2018-10-21 13:29:52 +02:00
painter.fillRect(buffer->rect(), backgroundColor);
// draw message
this->container_->paintElements(painter);
#ifdef FOURTF
// debug
painter.setPen(QColor(255, 0, 0));
2018-08-06 21:17:03 +02:00
painter.drawRect(buffer->rect().x(), buffer->rect().y(),
buffer->rect().width() - 1, buffer->rect().height() - 1);
QTextOption option;
option.setAlignment(Qt::AlignRight | Qt::AlignTop);
2018-04-10 02:42:41 +02:00
painter.drawText(QRectF(1, 1, this->container.getWidth() - 3, 1000),
QString::number(++this->bufferUpdatedCount), option);
2018-01-11 20:26:32 +01:00
#endif
}
void MessageLayout::invalidateBuffer()
{
this->bufferValid_ = false;
}
void MessageLayout::deleteBuffer()
{
if (this->buffer_ != nullptr) {
2018-06-26 17:06:17 +02:00
DebugCount::decrease("message drawing buffers");
this->buffer_ = nullptr;
}
}
2018-04-18 09:12:29 +02:00
void MessageLayout::deleteCache()
{
this->deleteBuffer();
#ifdef XD
this->container_->clear();
2018-04-18 09:12:29 +02:00
#endif
}
// Elements
// assert(QThread::currentThread() == QApplication::instance()->thread());
// returns nullptr if none was found
// fourtf: this should return a MessageLayoutItem
const MessageLayoutElement *MessageLayout::getElementAt(QPoint point)
{
// go through all words and return the first one that contains the point.
return this->container_->getElementAt(point);
}
int MessageLayout::getLastCharacterIndex() const
{
return this->container_->getLastCharacterIndex();
}
int MessageLayout::getFirstMessageCharacterIndex() const
{
return this->container_->getFirstMessageCharacterIndex();
}
int MessageLayout::getSelectionIndex(QPoint position)
{
return this->container_->getSelectionIndex(position);
}
2018-01-16 02:39:31 +01:00
2018-08-16 00:16:33 +02:00
void MessageLayout::addSelectionText(QString &str, int from, int to,
CopyMode copymode)
2018-01-16 02:39:31 +01:00
{
2018-08-16 00:16:33 +02:00
this->container_->addSelectionText(str, from, to, copymode);
2018-01-16 02:39:31 +01:00
}
2018-04-01 16:43:30 +02:00
} // namespace chatterino