mirror-chatterino2/src/widgets/helper/ChannelView.cpp

1236 lines
35 KiB
C++
Raw Normal View History

2018-06-26 14:09:39 +02:00
#include "ChannelView.hpp"
#include "Application.hpp"
#include "common/Common.hpp"
2018-06-26 17:20:03 +02:00
#include "debug/Benchmark.hpp"
2018-06-26 14:09:39 +02:00
#include "debug/Log.hpp"
#include "messages/Emote.hpp"
2018-06-26 14:09:39 +02:00
#include "messages/LimitedQueueSnapshot.hpp"
#include "messages/Message.hpp"
#include "messages/MessageElement.hpp"
2018-06-26 17:20:03 +02:00
#include "messages/layouts/MessageLayout.hpp"
#include "messages/layouts/MessageLayoutElement.hpp"
2018-06-26 14:09:39 +02:00
#include "providers/twitch/TwitchServer.hpp"
2018-06-28 19:46:45 +02:00
#include "singletons/Settings.hpp"
2018-06-28 20:03:04 +02:00
#include "singletons/Theme.hpp"
2018-06-26 14:09:39 +02:00
#include "singletons/WindowManager.hpp"
#include "util/DistanceBetweenPoints.hpp"
#include "widgets/Scrollbar.hpp"
2018-06-26 14:09:39 +02:00
#include "widgets/TooltipWidget.hpp"
2018-06-26 15:11:45 +02:00
#include "widgets/dialogs/UserInfoPopup.hpp"
#include "widgets/helper/EffectLabel.hpp"
2018-06-26 17:20:03 +02:00
#include "widgets/splits/Split.hpp"
2017-01-11 01:08:20 +01:00
2018-01-24 21:16:00 +01:00
#include <QClipboard>
2017-02-01 16:28:28 +01:00
#include <QDebug>
2017-07-26 09:08:19 +02:00
#include <QDesktopServices>
2017-02-07 00:03:15 +01:00
#include <QGraphicsBlurEffect>
2017-01-11 01:08:20 +01:00
#include <QPainter>
2017-06-06 17:18:23 +02:00
2017-09-12 19:06:16 +02:00
#include <algorithm>
2017-02-07 00:03:15 +01:00
#include <chrono>
#include <cmath>
2017-01-18 01:04:54 +01:00
#include <functional>
2017-09-12 19:06:16 +02:00
#include <memory>
2017-01-01 02:30:42 +01:00
#define DRAW_WIDTH (this->width())
#define SELECTION_RESUME_SCROLLING_MSG_THRESHOLD 3
2018-06-13 13:27:10 +02:00
#define CHAT_HOVER_PAUSE_DURATION 1000
2017-01-18 21:30:23 +01:00
namespace chatterino {
2018-08-02 14:23:27 +02:00
namespace {
2018-08-06 21:17:03 +02:00
void addEmoteContextMenuItems(const Emote &emote,
2018-08-07 07:55:31 +02:00
MessageElementFlags creatorFlags, QMenu &menu)
2018-08-02 14:23:27 +02:00
{
auto openAction = menu.addAction("Open");
auto openMenu = new QMenu;
openAction->setMenu(openMenu);
auto copyAction = menu.addAction("Copy");
auto copyMenu = new QMenu;
copyAction->setMenu(copyMenu);
// see if the QMenu actually gets destroyed
QObject::connect(openMenu, &QMenu::destroyed, [] {
2018-08-06 21:17:03 +02:00
QMessageBox(QMessageBox::Information, "xD", "the menu got deleted")
.exec();
2018-08-02 14:23:27 +02:00
});
// Add copy and open links for 1x, 2x, 3x
auto addImageLink = [&](const ImagePtr &image, char scale) {
2018-08-10 18:56:17 +02:00
if (!image->isEmpty()) {
2018-08-06 21:17:03 +02:00
copyMenu->addAction(
QString(scale) + "x link", [url = image->url()] {
QApplication::clipboard()->setText(url.string);
});
openMenu->addAction(QString(scale) + "x link",
[url = image->url()] {
QDesktopServices::openUrl(QUrl(url.string));
});
2018-08-02 14:23:27 +02:00
}
};
addImageLink(emote.images.getImage1(), '1');
addImageLink(emote.images.getImage2(), '2');
addImageLink(emote.images.getImage3(), '3');
// Copy and open emote page link
auto addPageLink = [&](const QString &name) {
copyMenu->addSeparator();
openMenu->addSeparator();
2018-08-06 21:17:03 +02:00
copyMenu->addAction(
"Copy " + name + " emote link", [url = emote.homePage] {
QApplication::clipboard()->setText(url.string); //
});
openMenu->addAction("Open " + name + " emote link",
[url = emote.homePage] {
QDesktopServices::openUrl(QUrl(url.string)); //
});
2018-08-02 14:23:27 +02:00
};
2018-08-07 07:55:31 +02:00
if (creatorFlags.has(MessageElementFlag::BttvEmote)) {
2018-08-02 14:23:27 +02:00
addPageLink("BTTV");
2018-08-07 07:55:31 +02:00
} else if (creatorFlags.has(MessageElementFlag::FfzEmote)) {
2018-08-02 14:23:27 +02:00
addPageLink("FFZ");
}
}
} // namespace
2017-01-18 21:30:23 +01:00
2017-12-17 02:18:13 +01:00
ChannelView::ChannelView(BaseWidget *parent)
: BaseWidget(parent)
, scrollBar_(new Scrollbar(this))
2017-01-01 02:30:42 +01:00
{
this->setMouseTracking(true);
2017-01-22 12:46:35 +01:00
2018-08-08 20:06:20 +02:00
this->initializeLayout();
this->initializeScrollbar();
this->initializeSignals();
2018-08-08 20:06:20 +02:00
this->pauseTimeout_.setSingleShot(true);
QObject::connect(&this->pauseTimeout_, &QTimer::timeout, [this] {
this->pausedTemporarily_ = false;
this->updatePauseStatus();
this->layoutMessages();
2017-06-06 17:18:23 +02:00
});
2017-09-17 02:13:57 +02:00
2018-08-08 20:06:20 +02:00
auto shortcut = new QShortcut(QKeySequence("Ctrl+C"), this);
QObject::connect(shortcut, &QShortcut::activated, [this] {
QGuiApplication::clipboard()->setText(this->getSelectedText());
2018-08-06 21:17:03 +02:00
});
2018-08-08 20:06:20 +02:00
}
2018-06-13 03:58:52 +02:00
2018-08-08 20:06:20 +02:00
void ChannelView::initializeLayout()
{
2018-08-08 15:35:54 +02:00
this->goToBottom_ = new EffectLabel(this, 0);
2018-08-06 21:17:03 +02:00
this->goToBottom_->setStyleSheet(
"background-color: rgba(0,0,0,0.66); color: #FFF;");
2018-06-13 13:27:10 +02:00
this->goToBottom_->getLabel().setText("More messages below");
this->goToBottom_->setVisible(false);
2018-08-08 15:35:54 +02:00
QObject::connect(this->goToBottom_, &EffectLabel::clicked, this, [=] {
QTimer::singleShot(180, [=] {
this->scrollBar_->scrollToBottom(
2018-08-08 20:06:20 +02:00
getApp()
->settings->enableSmoothScrollingNewMessages.getValue());
2018-01-05 03:14:46 +01:00
});
});
2018-08-08 20:06:20 +02:00
}
2018-08-08 20:06:20 +02:00
void ChannelView::initializeScrollbar()
{
this->scrollBar_->getCurrentValueChanged().connect([this] {
2018-08-08 20:06:20 +02:00
this->actuallyLayoutMessages(true);
2018-08-08 20:06:20 +02:00
this->goToBottom_->setVisible(this->enableScrollingToBottom_ &&
this->scrollBar_->isVisible() &&
!this->scrollBar_->isAtBottom());
2018-08-08 20:06:20 +02:00
this->queueUpdate();
});
this->scrollBar_->getDesiredValueChanged().connect([this] {
this->pausedByScrollingUp_ = !this->scrollBar_->isAtBottom();
2018-08-06 21:17:03 +02:00
});
2018-05-23 13:54:42 +02:00
}
2017-01-22 12:46:35 +01:00
2018-08-08 20:06:20 +02:00
void ChannelView::initializeSignals()
2017-01-22 12:46:35 +01:00
{
2018-08-08 20:06:20 +02:00
this->connections_.push_back(
getApp()->windows->wordFlagsChanged.connect([this] {
this->layoutMessages();
this->update();
}));
getApp()->settings->showLastMessageIndicator.connect(
[this](auto, auto) { this->update(); }, this->connections_);
connections_.push_back(
getApp()->windows->repaintGifs.connect([&] { this->queueUpdate(); }));
connections_.push_back(
getApp()->windows->layout.connect([&](Channel *channel) {
if (channel == nullptr || this->channel_.get() == channel)
this->layoutMessages();
}));
connections_.push_back(getApp()->fonts->fontChanged.connect(
[this] { this->layoutMessages(); }));
2017-01-01 02:30:42 +01:00
}
2017-01-03 21:19:33 +01:00
2018-07-06 17:11:37 +02:00
void ChannelView::themeChangedEvent()
2018-01-24 20:35:26 +01:00
{
2018-07-06 17:11:37 +02:00
BaseWidget::themeChangedEvent();
2018-01-24 20:35:26 +01:00
this->layoutMessages();
}
void ChannelView::queueUpdate()
2017-01-03 21:19:33 +01:00
{
// if (this->updateTimer.isActive()) {
// this->updateQueued = true;
// return;
// }
2018-01-19 23:41:02 +01:00
// this->repaint();
this->update();
// this->updateTimer.start();
}
void ChannelView::layoutMessages()
{
2018-04-10 02:07:25 +02:00
// if (!this->layoutCooldown->isActive()) {
this->actuallyLayoutMessages();
2018-04-10 02:07:25 +02:00
// this->layoutCooldown->start();
// } else {
// this->layoutQueued = true;
// }
}
void ChannelView::actuallyLayoutMessages(bool causedByScrollbar)
{
2018-08-09 18:39:46 +02:00
// BenchmarkGuard benchmark("layout");
2018-05-25 12:45:18 +02:00
auto app = getApp();
auto messagesSnapshot = this->getMessagesSnapshot();
2017-01-11 01:08:20 +01:00
if (messagesSnapshot.getLength() == 0) {
this->scrollBar_->setVisible(false);
return;
}
bool redrawRequired = false;
bool showScrollbar = false;
2017-06-06 17:18:23 +02:00
// Bool indicating whether or not we were showing all messages
// True if one of the following statements are true:
// The scrollbar was not visible
// The scrollbar was visible and at the bottom
2018-08-06 21:17:03 +02:00
this->showingLatestMessages_ =
this->scrollBar_->isAtBottom() || !this->scrollBar_->isVisible();
2017-06-06 17:18:23 +02:00
size_t start = size_t(this->scrollBar_->getCurrentValue());
2018-06-13 03:58:52 +02:00
int layoutWidth = this->getLayoutWidth();
2018-08-07 07:55:31 +02:00
MessageElementFlags flags = this->getFlags();
2018-01-17 16:52:51 +01:00
// layout the visible messages in the view
if (messagesSnapshot.getLength() > start) {
2018-06-13 13:27:10 +02:00
int y = int(-(messagesSnapshot[start]->getHeight() *
(fmod(this->scrollBar_->getCurrentValue(), 1))));
for (size_t i = start; i < messagesSnapshot.getLength(); ++i) {
auto message = messagesSnapshot[i];
2018-08-06 21:17:03 +02:00
redrawRequired |=
message->layout(layoutWidth, this->getScale(), flags);
2017-02-02 20:35:12 +01:00
y += message->getHeight();
if (y >= this->height()) {
break;
}
}
}
// layout the messages at the bottom to determine the scrollbar thumb size
int h = this->height() - 8;
2017-01-26 04:26:40 +01:00
2018-06-13 13:27:10 +02:00
for (int i = int(messagesSnapshot.getLength()) - 1; i >= 0; i--) {
auto *message = messagesSnapshot[i].get();
2018-01-25 20:49:49 +01:00
message->layout(layoutWidth, this->getScale(), flags);
2017-01-26 04:26:40 +01:00
2017-01-26 07:10:46 +01:00
h -= message->getHeight();
if (h < 0) {
this->scrollBar_->setLargeChange(
(messagesSnapshot.getLength() - i) +
qreal(h) / message->getHeight());
// this->scrollBar.setDesiredValue(this->scrollBar.getDesiredValue());
2017-01-26 07:10:46 +01:00
showScrollbar = true;
2017-02-01 16:28:28 +01:00
break;
2017-01-26 07:10:46 +01:00
}
2017-01-26 04:26:40 +01:00
}
this->scrollBar_->setVisible(showScrollbar);
2017-01-26 07:10:46 +01:00
2018-06-13 13:27:10 +02:00
if (!showScrollbar && !causedByScrollbar) {
this->scrollBar_->setDesiredValue(0);
2017-02-07 00:03:15 +01:00
}
this->scrollBar_->setMaximum(messagesSnapshot.getLength());
2017-01-26 07:10:46 +01:00
2018-08-06 21:17:03 +02:00
// If we were showing the latest messages and the scrollbar now wants to be
// rendered, scroll to bottom
if (this->enableScrollingToBottom_ && this->showingLatestMessages_ &&
showScrollbar) {
if (!this->isPaused()) {
this->scrollBar_->scrollToBottom(
// this->messageWasAdded &&
app->settings->enableSmoothScrollingNewMessages.getValue());
}
2018-06-13 13:27:10 +02:00
this->messageWasAdded_ = false;
2017-06-06 17:18:23 +02:00
}
if (redrawRequired) {
this->queueUpdate();
}
2017-01-26 04:26:40 +01:00
}
void ChannelView::clearMessages()
{
// Clear all stored messages in this chat widget
this->messages.clear();
2018-08-06 21:17:03 +02:00
// Layout chat widget messages, and force an update regardless if there are
// no messages
this->layoutMessages();
this->queueUpdate();
}
2018-01-06 03:48:56 +01:00
Scrollbar &ChannelView::getScrollBar()
2017-04-12 17:46:44 +02:00
{
return *this->scrollBar_;
2017-04-12 17:46:44 +02:00
}
QString ChannelView::getSelectedText()
2017-09-12 19:06:16 +02:00
{
2018-01-16 02:39:31 +01:00
QString result = "";
2017-09-12 19:06:16 +02:00
2018-08-06 21:17:03 +02:00
LimitedQueueSnapshot<MessageLayoutPtr> messagesSnapshot =
this->getMessagesSnapshot();
2017-09-12 19:06:16 +02:00
2018-06-13 13:27:10 +02:00
Selection _selection = this->selection_;
2017-09-12 19:06:16 +02:00
if (_selection.isEmpty()) {
2018-01-16 02:39:31 +01:00
return result;
}
2017-09-12 19:06:16 +02:00
for (int msg = _selection.selectionMin.messageIndex;
msg <= _selection.selectionMax.messageIndex; msg++) {
2018-01-16 02:39:31 +01:00
MessageLayoutPtr layout = messagesSnapshot[msg];
2018-08-06 21:17:03 +02:00
int from = msg == _selection.selectionMin.messageIndex
? _selection.selectionMin.charIndex
: 0;
int to = msg == _selection.selectionMax.messageIndex
? _selection.selectionMax.charIndex
: layout->getLastCharacterIndex() + 1;
2018-04-10 15:48:56 +02:00
qDebug() << "from:" << from << ", to:" << to;
2017-09-12 19:06:16 +02:00
2018-01-16 02:39:31 +01:00
layout->addSelectionText(result, from, to);
}
qDebug() << "xd <";
2017-09-12 19:06:16 +02:00
2018-01-16 02:39:31 +01:00
return result;
2017-09-12 19:06:16 +02:00
}
2017-09-21 02:20:02 +02:00
bool ChannelView::hasSelection()
{
2018-06-13 13:27:10 +02:00
return !this->selection_.isEmpty();
2017-09-21 02:20:02 +02:00
}
void ChannelView::clearSelection()
{
2018-06-13 13:27:10 +02:00
this->selection_ = Selection();
2017-09-21 02:20:02 +02:00
layoutMessages();
}
void ChannelView::setEnableScrollingToBottom(bool value)
{
2018-06-13 13:27:10 +02:00
this->enableScrollingToBottom_ = value;
}
bool ChannelView::getEnableScrollingToBottom() const
{
2018-06-13 13:27:10 +02:00
return this->enableScrollingToBottom_;
}
2018-08-07 07:55:31 +02:00
void ChannelView::setOverrideFlags(boost::optional<MessageElementFlags> value)
2018-01-27 21:13:22 +01:00
{
2018-06-13 13:27:10 +02:00
this->overrideFlags_ = value;
2018-01-27 21:13:22 +01:00
}
2018-08-07 07:55:31 +02:00
const boost::optional<MessageElementFlags> &ChannelView::getOverrideFlags()
2018-08-06 21:17:03 +02:00
const
2018-01-27 21:13:22 +01:00
{
2018-06-13 13:27:10 +02:00
return this->overrideFlags_;
2018-01-27 21:13:22 +01:00
}
LimitedQueueSnapshot<MessageLayoutPtr> ChannelView::getMessagesSnapshot()
{
if (!this->isPaused() /*|| this->scrollBar_->isVisible()*/) {
2018-06-13 13:27:10 +02:00
this->snapshot_ = this->messages.getSnapshot();
}
2018-06-13 13:27:10 +02:00
return this->snapshot_;
}
void ChannelView::setChannel(ChannelPtr newChannel)
{
2018-06-13 13:27:10 +02:00
if (this->channel_) {
this->detachChannel();
}
2018-04-20 19:54:45 +02:00
this->messages.clear();
// on new message
2018-06-13 03:58:52 +02:00
this->channelConnections_.push_back(
newChannel->messageAppended.connect([this](MessagePtr &message) {
MessageLayoutPtr deleted;
auto messageRef = new MessageLayout(message);
2018-06-13 13:27:10 +02:00
if (this->lastMessageHasAlternateBackground_) {
2018-08-07 07:55:31 +02:00
messageRef->flags.set(MessageLayoutFlag::AlternateBackground);
}
2018-08-06 21:17:03 +02:00
this->lastMessageHasAlternateBackground_ =
!this->lastMessageHasAlternateBackground_;
if (this->isPaused()) {
2018-06-13 13:27:10 +02:00
this->messagesAddedSinceSelectionPause_++;
}
2018-08-06 21:17:03 +02:00
if (this->messages.pushBack(MessageLayoutPtr(messageRef),
deleted)) {
// if (!this->isPaused()) {
if (this->scrollBar_->isAtBottom()) {
this->scrollBar_->scrollToBottom();
} else {
this->scrollBar_->offset(-1);
2017-12-18 22:13:46 +01:00
}
// }
}
2018-08-07 07:55:31 +02:00
if (!message->flags.has(MessageFlag::DoNotTriggerNotification)) {
if (message->flags.has(MessageFlag::Highlighted)) {
2018-08-06 21:17:03 +02:00
this->tabHighlightRequested.invoke(
HighlightState::Highlighted);
} else {
2018-08-06 21:17:03 +02:00
this->tabHighlightRequested.invoke(
HighlightState::NewMessage);
}
}
this->scrollBar_->addHighlight(message->getScrollBarHighlight());
2018-01-06 03:48:56 +01:00
2018-06-13 13:27:10 +02:00
this->messageWasAdded_ = true;
2018-01-05 03:14:46 +01:00
this->layoutMessages();
2018-06-13 03:58:52 +02:00
}));
2018-06-13 03:58:52 +02:00
this->channelConnections_.push_back(
2018-08-06 21:17:03 +02:00
newChannel->messagesAddedAtStart.connect(
[this](std::vector<MessagePtr> &messages) {
std::vector<MessageLayoutPtr> messageRefs;
messageRefs.resize(messages.size());
for (size_t i = 0; i < messages.size(); i++) {
messageRefs.at(i) =
MessageLayoutPtr(new MessageLayout(messages.at(i)));
}
2018-08-06 21:17:03 +02:00
if (!this->isPaused()) {
if (this->messages.pushFront(messageRefs).size() > 0) {
if (this->scrollBar_->isAtBottom()) {
this->scrollBar_->scrollToBottom();
2018-08-06 21:17:03 +02:00
} else {
this->scrollBar_->offset(qreal(messages.size()));
2018-08-06 21:17:03 +02:00
}
2018-01-11 21:03:40 +01:00
}
}
2018-08-06 21:17:03 +02:00
std::vector<ScrollbarHighlight> highlights;
highlights.reserve(messages.size());
for (size_t i = 0; i < messages.size(); i++) {
highlights.push_back(
messages.at(i)->getScrollBarHighlight());
}
2018-01-06 03:48:56 +01:00
this->scrollBar_->addHighlightsAtStart(highlights);
2018-01-06 03:48:56 +01:00
2018-08-06 21:17:03 +02:00
this->messageWasAdded_ = true;
this->layoutMessages();
}));
// on message removed
2018-06-13 03:58:52 +02:00
this->channelConnections_.push_back(
newChannel->messageRemovedFromStart.connect([this](MessagePtr &) {
2018-06-13 13:27:10 +02:00
this->selection_.selectionMin.messageIndex--;
this->selection_.selectionMax.messageIndex--;
this->selection_.start.messageIndex--;
this->selection_.end.messageIndex--;
this->layoutMessages();
2018-06-13 03:58:52 +02:00
}));
2018-01-05 23:14:55 +01:00
// on message replaced
2018-08-06 21:17:03 +02:00
this->channelConnections_.push_back(newChannel->messageReplaced.connect(
[this](size_t index, MessagePtr replacement) {
if (index >= this->messages.getSnapshot().getLength() ||
index < 0) {
2018-06-13 13:27:10 +02:00
return;
}
MessageLayoutPtr newItem(new MessageLayout(replacement));
auto snapshot = this->messages.getSnapshot();
if (index >= snapshot.getLength()) {
2018-08-11 14:20:53 +02:00
log("Tried to replace out of bounds message. Index: {}. "
2018-08-06 21:17:03 +02:00
"Length: {}",
index, snapshot.getLength());
return;
}
const auto &message = snapshot[index];
2018-08-07 07:55:31 +02:00
if (message->flags.has(MessageLayoutFlag::AlternateBackground)) {
newItem->flags.set(MessageLayoutFlag::AlternateBackground);
}
2018-01-05 23:14:55 +01:00
this->scrollBar_->replaceHighlight(
2018-08-06 21:17:03 +02:00
index, replacement->getScrollBarHighlight());
2018-01-06 03:48:56 +01:00
this->messages.replaceItem(message, newItem);
2018-01-05 23:14:55 +01:00
this->layoutMessages();
2018-06-13 03:58:52 +02:00
}));
2018-01-05 23:14:55 +01:00
auto snapshot = newChannel->getMessageSnapshot();
for (size_t i = 0; i < snapshot.getLength(); i++) {
MessageLayoutPtr deleted;
auto messageRef = new MessageLayout(snapshot[i]);
2018-06-13 13:27:10 +02:00
if (this->lastMessageHasAlternateBackground_) {
2018-08-07 07:55:31 +02:00
messageRef->flags.set(MessageLayoutFlag::AlternateBackground);
}
2018-08-06 21:17:03 +02:00
this->lastMessageHasAlternateBackground_ =
!this->lastMessageHasAlternateBackground_;
this->messages.pushBack(MessageLayoutPtr(messageRef), deleted);
}
2018-06-13 13:27:10 +02:00
this->channel_ = newChannel;
2018-01-05 13:42:23 +01:00
this->layoutMessages();
2018-01-07 00:05:32 +01:00
this->queueUpdate();
}
void ChannelView::detachChannel()
{
2018-06-13 03:58:52 +02:00
this->channelConnections_.clear();
}
void ChannelView::pause(int msecTimeout)
{
2018-06-13 13:27:10 +02:00
this->pausedTemporarily_ = true;
this->updatePauseStatus();
2018-06-13 13:27:10 +02:00
if (this->pauseTimeout_.remainingTime() < msecTimeout) {
this->pauseTimeout_.stop();
this->pauseTimeout_.start(msecTimeout);
2018-06-19 20:34:50 +02:00
// qDebug() << "pause" << msecTimeout;
2018-06-13 13:27:10 +02:00
}
}
2018-01-23 22:48:33 +01:00
void ChannelView::updateLastReadMessage()
{
auto _snapshot = this->getMessagesSnapshot();
if (_snapshot.getLength() > 0) {
2018-06-13 13:27:10 +02:00
this->lastReadMessage_ = _snapshot[_snapshot.getLength() - 1];
2018-01-23 22:48:33 +01:00
}
this->update();
}
void ChannelView::resizeEvent(QResizeEvent *)
2017-04-12 17:46:44 +02:00
{
this->scrollBar_->setGeometry(this->width() - this->scrollBar_->width(), 0,
this->scrollBar_->width(), this->height());
2018-06-13 13:27:10 +02:00
this->goToBottom_->setGeometry(0, this->height() - 32, this->width(), 32);
this->scrollBar_->raise();
this->layoutMessages();
2017-02-07 00:03:15 +01:00
2017-06-06 17:18:23 +02:00
this->update();
2017-01-03 21:19:33 +01:00
}
2017-01-05 16:07:20 +01:00
2018-08-06 21:17:03 +02:00
void ChannelView::setSelection(const SelectionItem &start,
const SelectionItem &end)
2017-08-18 15:12:07 +02:00
{
// selections
2018-06-13 13:27:10 +02:00
if (!this->selecting_ && start != end) {
this->messagesAddedSinceSelectionPause_ = 0;
2018-06-13 13:27:10 +02:00
this->selecting_ = true;
this->pausedBySelection_ = true;
}
2018-06-13 13:27:10 +02:00
this->selection_ = Selection(start, end);
2017-08-18 15:12:07 +02:00
this->selectionChanged.invoke();
2017-08-18 15:12:07 +02:00
}
2018-08-07 07:55:31 +02:00
MessageElementFlags ChannelView::getFlags() const
2018-01-17 16:52:51 +01:00
{
auto app = getApp();
2018-06-13 13:27:10 +02:00
if (this->overrideFlags_) {
return this->overrideFlags_.get();
2018-01-27 21:13:22 +01:00
}
2018-08-07 07:55:31 +02:00
MessageElementFlags flags = app->windows->getWordFlags();
2018-01-17 16:52:51 +01:00
Split *split = dynamic_cast<Split *>(this->parentWidget());
if (split != nullptr) {
if (split->getModerationMode()) {
2018-08-07 07:55:31 +02:00
flags.set(MessageElementFlag::ModeratorTools);
2018-01-17 16:52:51 +01:00
}
2018-06-13 13:27:10 +02:00
if (this->channel_ == app->twitch.server->mentionsChannel) {
2018-08-07 07:55:31 +02:00
flags.set(MessageElementFlag::ChannelName);
2018-01-23 23:28:06 +01:00
}
2018-01-17 16:52:51 +01:00
}
return flags;
}
bool ChannelView::isPaused()
{
2018-06-19 18:55:45 +02:00
return false;
2018-08-06 21:17:03 +02:00
// return this->pausedTemporarily_ || this->pausedBySelection_ ||
// this->pausedByScrollingUp_;
}
2018-06-13 13:27:10 +02:00
void ChannelView::updatePauseStatus()
{
if (this->isPaused()) {
this->scrollBar_->pauseHighlights();
2018-06-13 13:27:10 +02:00
} else {
this->scrollBar_->unpauseHighlights();
2018-06-13 13:27:10 +02:00
}
}
void ChannelView::paintEvent(QPaintEvent * /*event*/)
2017-01-05 16:07:20 +01:00
{
2018-08-09 18:39:46 +02:00
// BenchmarkGuard benchmark("paint");
2017-02-07 00:03:15 +01:00
QPainter painter(this);
2017-02-07 00:03:15 +01:00
2018-07-06 17:11:37 +02:00
painter.fillRect(rect(), this->theme->splits.background);
2017-08-18 15:12:07 +02:00
// draw messages
this->drawMessages(painter);
2017-08-18 15:12:07 +02:00
}
2018-08-06 21:17:03 +02:00
// if overlays is false then it draws the message, if true then it draws things
// such as the grey overlay when a message is disabled
void ChannelView::drawMessages(QPainter &painter)
2017-08-18 15:12:07 +02:00
{
auto app = getApp();
auto messagesSnapshot = this->getMessagesSnapshot();
2017-01-11 01:08:20 +01:00
size_t start = size_t(this->scrollBar_->getCurrentValue());
2017-01-11 01:08:20 +01:00
if (start >= messagesSnapshot.getLength()) {
2017-01-18 04:33:30 +01:00
return;
}
2018-05-25 12:45:18 +02:00
int y = int(-(messagesSnapshot[start].get()->getHeight() *
(fmod(this->scrollBar_->getCurrentValue(), 1))));
2017-01-18 04:33:30 +01:00
MessageLayout *end = nullptr;
2018-01-23 22:48:33 +01:00
bool windowFocused = this->window() == QApplication::activeWindow();
for (size_t i = start; i < messagesSnapshot.getLength(); ++i) {
MessageLayout *layout = messagesSnapshot[i].get();
2017-02-07 00:03:15 +01:00
2018-01-23 22:48:33 +01:00
bool isLastMessage = false;
if (app->settings->showLastMessageIndicator) {
2018-06-13 13:27:10 +02:00
isLastMessage = this->lastReadMessage_.get() == layout;
2018-01-23 22:48:33 +01:00
}
2018-08-06 21:17:03 +02:00
layout->paint(painter, DRAW_WIDTH, y, i, this->selection_,
isLastMessage, windowFocused);
y += layout->getHeight();
2017-01-20 06:10:28 +01:00
end = layout;
if (y > this->height()) {
2017-01-20 06:10:28 +01:00
break;
}
2017-01-11 01:08:20 +01:00
}
if (end == nullptr) {
return;
}
// remove messages that are on screen
// the messages that are left at the end get their buffers reset
for (size_t i = start; i < messagesSnapshot.getLength(); ++i) {
2018-06-13 13:27:10 +02:00
auto it = this->messagesOnScreen_.find(messagesSnapshot[i]);
if (it != this->messagesOnScreen_.end()) {
this->messagesOnScreen_.erase(it);
}
}
// delete the message buffers that aren't on screen
for (const std::shared_ptr<MessageLayout> &item : this->messagesOnScreen_) {
item->deleteBuffer();
}
2018-06-13 13:27:10 +02:00
this->messagesOnScreen_.clear();
// add all messages on screen to the map
for (size_t i = start; i < messagesSnapshot.getLength(); ++i) {
std::shared_ptr<MessageLayout> layout = messagesSnapshot[i];
2018-06-13 13:27:10 +02:00
this->messagesOnScreen_.insert(layout);
if (layout.get() == end) {
break;
}
}
2017-08-18 15:12:07 +02:00
}
2017-02-07 00:03:15 +01:00
void ChannelView::wheelEvent(QWheelEvent *event)
2017-01-26 04:26:40 +01:00
{
2018-06-19 18:55:45 +02:00
if (event->orientation() != Qt::Vertical) {
return;
}
2018-06-13 13:27:10 +02:00
this->pausedBySelection_ = false;
this->pausedTemporarily_ = false;
this->updatePauseStatus();
2018-06-11 15:04:54 +02:00
if (event->modifiers() & Qt::ControlModifier) {
event->ignore();
return;
}
if (this->scrollBar_->isVisible()) {
auto app = getApp();
float mouseMultiplier = app->settings->mouseScrollMultiplier;
2017-04-14 17:47:28 +02:00
qreal desired = this->scrollBar_->getDesiredValue();
2018-06-13 13:27:10 +02:00
qreal delta = event->delta() * qreal(1.5) * mouseMultiplier;
2018-01-05 02:55:24 +01:00
auto snapshot = this->getMessagesSnapshot();
2018-06-13 13:27:10 +02:00
int snapshotLength = int(snapshot.getLength());
2018-07-07 11:41:01 +02:00
int i = std::min<int>(int(desired), snapshotLength);
2018-01-05 02:55:24 +01:00
if (delta > 0) {
2018-06-13 13:27:10 +02:00
qreal scrollFactor = fmod(desired, 1);
2018-08-06 21:17:03 +02:00
qreal currentScrollLeft =
int(scrollFactor * snapshot[i]->getHeight());
2018-01-05 02:55:24 +01:00
for (; i >= 0; i--) {
if (delta < currentScrollLeft) {
desired -= scrollFactor * (delta / currentScrollLeft);
break;
} else {
delta -= currentScrollLeft;
desired -= scrollFactor;
}
if (i == 0) {
desired = 0;
} else {
2018-08-06 21:17:03 +02:00
snapshot[i - 1]->layout(this->getLayoutWidth(),
this->getScale(), this->getFlags());
2018-01-05 02:55:24 +01:00
scrollFactor = 1;
currentScrollLeft = snapshot[i - 1]->getHeight();
}
}
} else {
delta = -delta;
2018-06-13 13:27:10 +02:00
qreal scrollFactor = 1 - fmod(desired, 1);
2018-08-06 21:17:03 +02:00
qreal currentScrollLeft =
int(scrollFactor * snapshot[i]->getHeight());
2018-01-05 02:55:24 +01:00
for (; i < snapshotLength; i++) {
2018-01-05 02:55:24 +01:00
if (delta < currentScrollLeft) {
2018-08-06 21:17:03 +02:00
desired +=
scrollFactor * (qreal(delta) / currentScrollLeft);
2018-01-05 02:55:24 +01:00
break;
} else {
delta -= currentScrollLeft;
desired += scrollFactor;
}
if (i == snapshotLength - 1) {
2018-01-05 02:55:24 +01:00
desired = snapshot.getLength();
} else {
2018-08-06 21:17:03 +02:00
snapshot[i + 1]->layout(this->getLayoutWidth(),
this->getScale(), this->getFlags());
2018-01-05 02:55:24 +01:00
scrollFactor = 1;
currentScrollLeft = snapshot[i + 1]->getHeight();
}
}
}
this->scrollBar_->setDesiredValue(desired, true);
}
2017-01-26 04:26:40 +01:00
}
2017-02-17 23:51:35 +01:00
void ChannelView::enterEvent(QEvent *)
{
// this->pause(PAUSE_TIME);
}
void ChannelView::leaveEvent(QEvent *)
{
2018-06-13 13:27:10 +02:00
this->pausedTemporarily_ = false;
this->updatePauseStatus();
this->layoutMessages();
}
void ChannelView::mouseMoveEvent(QMouseEvent *event)
2017-02-17 23:51:35 +01:00
{
if (event->modifiers() & (Qt::AltModifier | Qt::ControlModifier)) {
this->unsetCursor();
event->ignore();
return;
}
auto app = getApp();
if (app->settings->pauseChatHover.getValue()) {
this->pause(CHAT_HOVER_PAUSE_DURATION);
}
auto tooltipWidget = TooltipWidget::getInstance();
std::shared_ptr<MessageLayout> layout;
2017-02-17 23:51:35 +01:00
QPoint relativePos;
2017-08-18 15:12:07 +02:00
int messageIndex;
2017-02-17 23:51:35 +01:00
// no message under cursor
if (!tryGetMessageAt(event->pos(), layout, relativePos, messageIndex)) {
this->setCursor(Qt::ArrowCursor);
tooltipWidget->hide();
return;
}
// is selecting
2018-06-13 13:27:10 +02:00
if (this->isMouseDown_) {
this->pause(300);
int index = layout->getSelectionIndex(relativePos);
2017-08-18 15:12:07 +02:00
2018-08-06 21:17:03 +02:00
this->setSelection(this->selection_.start,
SelectionItem(messageIndex, index));
2017-08-18 15:12:07 +02:00
this->queueUpdate();
}
// message under cursor is collapsed
2018-08-07 07:55:31 +02:00
if (layout->flags.has(MessageLayoutFlag::Collapsed)) {
this->setCursor(Qt::PointingHandCursor);
tooltipWidget->hide();
return;
2017-08-18 15:12:07 +02:00
}
2017-04-24 23:00:26 +02:00
// check if word underneath cursor
2018-08-06 21:17:03 +02:00
const MessageLayoutElement *hoverLayoutElement =
layout->getElementAt(relativePos);
if (hoverLayoutElement == nullptr) {
this->setCursor(Qt::ArrowCursor);
tooltipWidget->hide();
2017-02-17 23:51:35 +01:00
return;
}
const auto &tooltip = hoverLayoutElement->getCreator().getTooltip();
2017-12-19 03:36:05 +01:00
2018-01-17 03:26:32 +01:00
if (tooltip.isEmpty()) {
tooltipWidget->hide();
} else {
tooltipWidget->moveTo(this, event->globalPos());
2018-01-17 03:26:32 +01:00
tooltipWidget->setText(tooltip);
2018-06-22 17:45:11 +02:00
tooltipWidget->adjustSize();
2018-01-17 03:26:32 +01:00
tooltipWidget->show();
tooltipWidget->raise();
2018-01-17 03:26:32 +01:00
}
2017-12-19 03:36:05 +01:00
// check if word has a link
2018-01-17 14:14:31 +01:00
if (hoverLayoutElement->getLink().isValid()) {
this->setCursor(Qt::PointingHandCursor);
2017-02-17 23:51:35 +01:00
} else {
this->setCursor(Qt::ArrowCursor);
2017-02-17 23:51:35 +01:00
}
}
void ChannelView::mousePressEvent(QMouseEvent *event)
2017-04-12 17:46:44 +02:00
{
auto app = getApp();
2018-06-04 21:44:03 +02:00
this->mouseDown.invoke(event);
std::shared_ptr<MessageLayout> layout;
2017-08-18 15:12:07 +02:00
QPoint relativePos;
int messageIndex;
if (!tryGetMessageAt(event->pos(), layout, relativePos, messageIndex)) {
2017-08-18 15:12:07 +02:00
setCursor(Qt::ArrowCursor);
auto messagesSnapshot = this->getMessagesSnapshot();
if (messagesSnapshot.getLength() == 0) {
return;
}
// Start selection at the last message at its last index
if (event->button() == Qt::LeftButton) {
auto lastMessageIndex = messagesSnapshot.getLength() - 1;
auto lastMessage = messagesSnapshot[lastMessageIndex];
auto lastCharacterIndex = lastMessage->getLastCharacterIndex();
SelectionItem selectionItem(lastMessageIndex, lastCharacterIndex);
this->setSelection(selectionItem, selectionItem);
}
return;
2017-08-18 15:12:07 +02:00
}
// check if message is collapsed
switch (event->button()) {
case Qt::LeftButton: {
2018-06-13 13:27:10 +02:00
this->lastPressPosition_ = event->screenPos();
this->isMouseDown_ = true;
2018-06-05 14:24:01 +02:00
2018-08-07 07:55:31 +02:00
if (layout->flags.has(MessageLayoutFlag::Collapsed)) {
2018-06-05 14:24:01 +02:00
return;
}
if (app->settings->linksDoubleClickOnly.getValue()) {
this->pause(200);
}
int index = layout->getSelectionIndex(relativePos);
2017-04-12 17:46:44 +02:00
auto selectionItem = SelectionItem(messageIndex, index);
this->setSelection(selectionItem, selectionItem);
} break;
case Qt::RightButton: {
2018-06-13 13:27:10 +02:00
this->lastRightPressPosition_ = event->screenPos();
this->isRightMouseDown_ = true;
} break;
default:;
}
this->update();
}
2017-04-12 17:46:44 +02:00
void ChannelView::mouseReleaseEvent(QMouseEvent *event)
{
// check if mouse was pressed
if (event->button() == Qt::LeftButton) {
2018-06-13 13:27:10 +02:00
if (this->isMouseDown_) {
this->isMouseDown_ = false;
2018-08-06 21:17:03 +02:00
if (fabsf(distanceBetweenPoints(this->lastPressPosition_,
event->screenPos())) > 15.f) {
return;
}
} else {
return;
}
} else if (event->button() == Qt::RightButton) {
2018-06-13 13:27:10 +02:00
if (this->isRightMouseDown_) {
this->isRightMouseDown_ = false;
2017-04-12 17:46:44 +02:00
2018-08-06 21:17:03 +02:00
if (fabsf(distanceBetweenPoints(this->lastRightPressPosition_,
event->screenPos())) > 15.f) {
return;
}
} else {
return;
}
} else {
// not left or right button
2017-04-12 17:46:44 +02:00
return;
}
// find message
2018-02-06 00:10:30 +01:00
this->layoutMessages();
2017-04-12 17:46:44 +02:00
std::shared_ptr<MessageLayout> layout;
2017-04-12 17:46:44 +02:00
QPoint relativePos;
2017-08-18 15:12:07 +02:00
int messageIndex;
2017-04-12 17:46:44 +02:00
// no message found
if (!tryGetMessageAt(event->pos(), layout, relativePos, messageIndex)) {
2017-04-12 17:46:44 +02:00
// No message at clicked position
return;
}
// message under cursor is collapsed
2018-08-07 07:55:31 +02:00
if (layout->flags.has(MessageLayoutFlag::Collapsed)) {
layout->flags.set(MessageLayoutFlag::Expanded);
layout->flags.set(MessageLayoutFlag::RequiresLayout);
2018-04-18 09:12:29 +02:00
this->layoutMessages();
return;
}
2018-08-06 21:17:03 +02:00
const MessageLayoutElement *hoverLayoutElement =
layout->getElementAt(relativePos);
2017-04-24 23:00:26 +02:00
if (hoverLayoutElement == nullptr) {
2017-04-24 23:00:26 +02:00
return;
}
// handle the click
this->handleMouseClick(event, hoverLayoutElement, layout.get());
}
2018-08-06 21:17:03 +02:00
void ChannelView::handleMouseClick(QMouseEvent *event,
const MessageLayoutElement *hoveredElement,
MessageLayout *layout)
{
switch (event->button()) {
case Qt::LeftButton: {
2018-06-13 13:27:10 +02:00
if (this->selecting_) {
if (this->messagesAddedSinceSelectionPause_ >
SELECTION_RESUME_SCROLLING_MSG_THRESHOLD) {
2018-06-13 13:27:10 +02:00
this->showingLatestMessages_ = false;
}
2018-06-13 13:27:10 +02:00
// this->pausedBySelection = false;
this->selecting_ = false;
// this->pauseTimeout.stop();
// this->pausedTemporarily = false;
this->layoutMessages();
}
auto &link = hoveredElement->getLink();
if (!getApp()->settings->linksDoubleClickOnly) {
this->handleLinkClick(event, link, layout);
this->linkClicked.invoke(link);
}
} break;
case Qt::RightButton: {
auto &link = hoveredElement->getLink();
if (link.type == Link::UserInfo) {
Split *split = dynamic_cast<Split *>(this->parentWidget());
if (split != nullptr) {
split->insertTextToInput("@" + link.value + ", ");
}
} else {
this->addContextMenuItems(hoveredElement, layout);
}
} break;
default:;
}
}
2018-08-06 21:17:03 +02:00
void ChannelView::addContextMenuItems(
const MessageLayoutElement *hoveredElement, MessageLayout *layout)
{
const auto &creator = hoveredElement->getCreator();
auto creatorFlags = creator.getFlags();
static QMenu *menu = new QMenu;
menu->clear();
2018-05-23 21:16:34 +02:00
// Emote actions
2018-08-07 07:55:31 +02:00
if (creatorFlags.hasAny({MessageElementFlag::EmoteImages,
MessageElementFlag::EmojiImage})) {
2018-08-02 14:23:27 +02:00
const auto emoteElement = dynamic_cast<const EmoteElement *>(&creator);
2018-08-06 21:17:03 +02:00
if (emoteElement)
addEmoteContextMenuItems(*emoteElement->getEmote(), creatorFlags,
*menu);
}
// add seperator
if (!menu->actions().empty()) {
menu->addSeparator();
}
// Link copy
if (hoveredElement->getLink().type == Link::Url) {
QString url = hoveredElement->getLink().value;
2018-05-23 21:16:34 +02:00
2018-08-06 21:17:03 +02:00
menu->addAction("Open link",
[url] { QDesktopServices::openUrl(QUrl(url)); });
menu->addAction("Copy link",
[url] { QApplication::clipboard()->setText(url); });
menu->addSeparator();
}
// Copy actions
2018-06-13 13:27:10 +02:00
if (!this->selection_.isEmpty()) {
2018-08-06 21:17:03 +02:00
menu->addAction("Copy selection", [this] {
QGuiApplication::clipboard()->setText(this->getSelectedText());
});
}
2018-06-05 14:24:54 +02:00
menu->addAction("Copy message", [layout] {
QString copyString;
layout->addSelectionText(copyString);
QGuiApplication::clipboard()->setText(copyString);
});
// menu->addAction("Quote message", [layout] {
// QString copyString;
// layout->addSelectionText(copyString);
// // insert into input
// });
menu->popup(QCursor::pos());
menu->raise();
2018-01-24 21:44:31 +01:00
return;
2018-01-24 21:44:31 +01:00
}
void ChannelView::mouseDoubleClickEvent(QMouseEvent *event)
{
auto app = getApp();
if (app->settings->linksDoubleClickOnly) {
std::shared_ptr<MessageLayout> layout;
2018-01-24 21:44:31 +01:00
QPoint relativePos;
int messageIndex;
if (!tryGetMessageAt(event->pos(), layout, relativePos, messageIndex)) {
return;
}
// message under cursor is collapsed
2018-08-07 07:55:31 +02:00
if (layout->flags.has(MessageLayoutFlag::Collapsed)) {
2018-01-24 21:44:31 +01:00
return;
}
2017-04-12 17:46:44 +02:00
2018-08-06 21:17:03 +02:00
const MessageLayoutElement *hoverLayoutElement =
layout->getElementAt(relativePos);
2018-01-24 21:44:31 +01:00
if (hoverLayoutElement == nullptr) {
return;
}
2017-04-12 17:46:44 +02:00
2018-01-24 21:44:31 +01:00
auto &link = hoverLayoutElement->getLink();
this->handleLinkClick(event, link, layout.get());
}
}
2018-04-18 09:12:29 +02:00
void ChannelView::hideEvent(QHideEvent *)
{
2018-06-13 13:27:10 +02:00
for (auto &layout : this->messagesOnScreen_) {
2018-04-18 09:12:29 +02:00
layout->deleteBuffer();
}
2018-06-13 13:27:10 +02:00
this->messagesOnScreen_.clear();
2018-04-18 09:12:29 +02:00
}
2018-08-06 21:17:03 +02:00
void ChannelView::handleLinkClick(QMouseEvent *event, const Link &link,
MessageLayout *layout)
2018-01-24 21:44:31 +01:00
{
2018-05-23 21:16:34 +02:00
if (event->button() != Qt::LeftButton) {
return;
}
2018-01-28 03:52:52 +01:00
switch (link.type) {
case Link::UserInfo: {
2018-01-28 03:52:52 +01:00
auto user = link.value;
2018-06-06 13:35:06 +02:00
auto *userPopup = new UserInfoPopup;
2018-06-13 13:27:10 +02:00
userPopup->setData(user, this->channel_);
userPopup->setActionOnFocusLoss(BaseWindow::Delete);
2018-08-06 21:17:03 +02:00
QPoint offset(int(150 * this->getScale()),
int(70 * this->getScale()));
userPopup->move(QCursor::pos() - offset);
2018-06-06 10:46:23 +02:00
userPopup->show();
2017-04-24 23:00:26 +02:00
qDebug() << "Clicked " << user << "s message";
2018-08-11 14:20:53 +02:00
} break;
case Link::Url: {
2018-05-23 21:16:34 +02:00
QDesktopServices::openUrl(QUrl(link.value));
2018-08-11 14:20:53 +02:00
} break;
case Link::UserAction: {
2018-01-28 03:52:52 +01:00
QString value = link.value;
2018-01-17 14:14:31 +01:00
value.replace("{user}", layout->getMessage()->loginName);
2018-06-13 13:27:10 +02:00
this->channel_->sendMessage(value);
2018-08-11 14:20:53 +02:00
} break;
default:;
2017-04-24 23:00:26 +02:00
}
2017-04-12 17:46:44 +02:00
}
2018-08-06 21:17:03 +02:00
bool ChannelView::tryGetMessageAt(QPoint p,
std::shared_ptr<MessageLayout> &_message,
QPoint &relativePos, int &index)
2017-02-17 23:51:35 +01:00
{
auto messagesSnapshot = this->getMessagesSnapshot();
2017-02-17 23:51:35 +01:00
size_t start = this->scrollBar_->getCurrentValue();
2017-02-17 23:51:35 +01:00
if (start >= messagesSnapshot.getLength()) {
2017-02-17 23:51:35 +01:00
return false;
}
2018-08-06 21:17:03 +02:00
int y = -(messagesSnapshot[start]->getHeight() *
(fmod(this->scrollBar_->getCurrentValue(), 1)));
2017-02-17 23:51:35 +01:00
for (size_t i = start; i < messagesSnapshot.getLength(); ++i) {
auto message = messagesSnapshot[i];
2017-02-17 23:51:35 +01:00
2017-04-24 23:00:26 +02:00
if (p.y() < y + message->getHeight()) {
relativePos = QPoint(p.x(), p.y() - y);
2017-02-17 23:51:35 +01:00
_message = message;
2017-08-18 15:12:07 +02:00
index = i;
2017-02-17 23:51:35 +01:00
return true;
}
2017-04-24 23:00:26 +02:00
y += message->getHeight();
2017-02-17 23:51:35 +01:00
}
return false;
}
2017-06-06 17:18:23 +02:00
2018-06-13 03:58:52 +02:00
int ChannelView::getLayoutWidth() const
{
if (this->scrollBar_->isVisible())
2018-08-06 21:17:03 +02:00
return int(this->width() - 8 * this->getScale());
2018-06-13 03:58:52 +02:00
return this->width();
}
2017-04-14 17:52:22 +02:00
} // namespace chatterino