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

914 lines
26 KiB
C++
Raw Normal View History

2017-11-12 17:21:50 +01:00
#include "widgets/helper/channelview.hpp"
2017-06-11 09:31:45 +02:00
#include "channelmanager.hpp"
#include "colorscheme.hpp"
#include "debug/log.hpp"
2017-09-12 19:06:16 +02:00
#include "messages/limitedqueuesnapshot.hpp"
2017-06-11 09:31:45 +02:00
#include "messages/message.hpp"
2017-09-12 19:06:16 +02:00
#include "messages/messageref.hpp"
2017-06-11 09:31:45 +02:00
#include "settingsmanager.hpp"
#include "ui_accountpopupform.h"
#include "util/benchmark.hpp"
2017-06-11 09:31:45 +02:00
#include "util/distancebetweenpoints.hpp"
2017-11-12 17:21:50 +01:00
#include "widgets/split.hpp"
2017-09-17 02:13:57 +02:00
#include "windowmanager.hpp"
2017-01-11 01:08:20 +01:00
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
#include <math.h>
2017-09-12 19:06:16 +02:00
#include <algorithm>
2017-02-07 00:03:15 +01:00
#include <chrono>
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
using namespace chatterino::messages;
2017-01-18 21:30:23 +01:00
namespace chatterino {
namespace widgets {
2017-12-17 02:18:13 +01:00
ChannelView::ChannelView(BaseWidget *parent)
: BaseWidget(parent)
, scrollBar(this)
, userPopupWidget(std::shared_ptr<twitch::TwitchChannel>())
2017-01-01 02:30:42 +01:00
{
#ifndef Q_OS_MAC
2017-09-15 17:23:49 +02:00
// this->setAttribute(Qt::WA_OpaquePaintEvent);
#endif
this->setMouseTracking(true);
2017-01-22 12:46:35 +01:00
2017-04-12 17:46:44 +02:00
QObject::connect(&SettingsManager::getInstance(), &SettingsManager::wordTypeMaskChanged, this,
&ChannelView::wordTypeMaskChanged);
2017-01-26 04:26:40 +01:00
this->scrollBar.getCurrentValueChanged().connect([this] {
2017-06-06 17:18:23 +02:00
// Whenever the scrollbar value has been changed, re-render the ChatWidgetView
2017-08-12 12:09:26 +02:00
this->layoutMessages();
this->goToBottom->setVisible(this->scrollBar.isVisible() && !this->scrollBar.isAtBottom());
this->queueUpdate();
2017-06-06 17:18:23 +02:00
});
2017-09-17 02:13:57 +02:00
2017-12-17 02:18:13 +01:00
WindowManager &windowManager = *WindowManager::instance;
2017-09-17 02:13:57 +02:00
this->repaintGifsConnection =
windowManager.repaintGifs.connect([&] { this->updateGifEmotes(); });
this->layoutConnection = windowManager.layout.connect([&] { this->layoutMessages(); });
this->goToBottom = new RippleEffectLabel(this, 0);
this->goToBottom->setStyleSheet("background-color: rgba(0,0,0,0.5); color: #FFF;");
this->goToBottom->getLabel().setText("Jump to bottom");
this->goToBottom->setVisible(false);
this->managedConnections.emplace_back(FontManager::getInstance().fontChanged.connect([this] {
this->layoutMessages(); //
}));
connect(goToBottom, &RippleEffectLabel::clicked, this,
[this] { QTimer::singleShot(180, [this] { this->scrollBar.scrollToBottom(); }); });
this->updateTimer.setInterval(1000 / 60);
this->updateTimer.setSingleShot(true);
connect(&this->updateTimer, &QTimer::timeout, this, [this] {
if (this->updateQueued) {
this->update();
}
this->updateTimer.start();
});
2017-01-22 12:46:35 +01:00
}
ChannelView::~ChannelView()
2017-01-22 12:46:35 +01:00
{
2017-04-12 17:46:44 +02:00
QObject::disconnect(&SettingsManager::getInstance(), &SettingsManager::wordTypeMaskChanged,
this, &ChannelView::wordTypeMaskChanged);
2017-01-01 02:30:42 +01:00
}
2017-01-03 21:19:33 +01:00
void ChannelView::queueUpdate()
2017-01-03 21:19:33 +01:00
{
if (this->updateTimer.isActive()) {
this->updateQueued = true;
}
update();
this->updateTimer.start();
}
void ChannelView::layoutMessages()
{
this->actuallyLayoutMessages();
}
void ChannelView::actuallyLayoutMessages()
{
2017-10-27 22:05:08 +02:00
// BENCH(timer)
auto messages = this->getMessagesSnapshot();
2017-01-11 01:08:20 +01:00
if (messages.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
this->showingLatestMessages = this->scrollBar.isAtBottom() || !this->scrollBar.isVisible();
2017-06-06 17:18:23 +02:00
2017-09-12 19:06:16 +02:00
size_t start = this->scrollBar.getCurrentValue();
2017-09-15 17:23:49 +02:00
int layoutWidth =
(this->scrollBar.isVisible() ? width() - this->scrollBar.width() : width()) - 4;
// layout the visible messages in the view
if (messages.getLength() > start) {
int y = -(messages[start]->getHeight() * (fmod(this->scrollBar.getCurrentValue(), 1)));
2017-09-12 19:06:16 +02:00
for (size_t i = start; i < messages.getLength(); ++i) {
2017-04-14 17:47:28 +02:00
auto message = messages[i];
2017-12-17 00:06:24 +01:00
redrawRequired |= message->layout(layoutWidth, this->getDpiMultiplier());
2017-02-02 20:35:12 +01:00
y += message->getHeight();
if (y >= height()) {
break;
}
}
}
// layout the messages at the bottom to determine the scrollbar thumb size
2017-04-14 17:47:28 +02:00
int h = height() - 8;
2017-01-26 04:26:40 +01:00
for (std::size_t i = messages.getLength() - 1; i > 0; i--) {
2017-01-26 07:10:46 +01:00
auto *message = messages[i].get();
2017-12-17 00:06:24 +01:00
message->layout(layoutWidth, this->getDpiMultiplier());
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((messages.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
2017-02-07 00:03:15 +01:00
if (!showScrollbar) {
this->scrollBar.setDesiredValue(0);
2017-02-07 00:03:15 +01:00
}
this->scrollBar.setMaximum(messages.getLength());
2017-01-26 07:10:46 +01:00
2017-06-06 21:18:05 +02:00
if (this->showingLatestMessages && showScrollbar) {
2017-06-06 17:18:23 +02:00
// If we were showing the latest messages and the scrollbar now wants to be rendered, scroll
// to bottom
// TODO: Do we want to check if the user is currently moving the scrollbar?
// Perhaps also if the user scrolled with the scrollwheel in this ChatWidget in the last 0.2
// seconds or something
this->scrollBar.scrollToBottom();
2017-06-06 17:18:23 +02:00
}
2017-10-27 22:05:08 +02:00
// MARK(timer);
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();
// Layout chat widget messages, and force an update regardless if there are no messages
this->layoutMessages();
this->queueUpdate();
}
void ChannelView::updateGifEmotes()
{
if (!this->gifEmotes.empty()) {
this->onlyUpdateEmotes = true;
this->queueUpdate();
}
2017-04-12 17:46:44 +02: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
{
LimitedQueueSnapshot<SharedMessageRef> messages = this->getMessagesSnapshot();
2017-09-12 19:06:16 +02:00
QString text;
bool isSingleMessage = this->selection.isSingleMessage();
size_t i = std::max(0, this->selection.min.messageIndex);
int charIndex = 0;
bool first = true;
auto addPart = [&](const WordPart &part, int from = 0, int to = -1) {
if (part.getCopyText().isEmpty()) {
return;
}
if (part.getWord().isText()) {
text += part.getText().mid(from, to);
} else {
text += part.getCopyText();
}
};
// first line
2017-09-12 19:06:16 +02:00
for (const messages::WordPart &part : messages[i]->getWordParts()) {
int charLength = part.getCharacterLength();
if (charIndex + charLength < this->selection.min.charIndex) {
charIndex += charLength;
continue;
}
if (first) {
first = false;
bool isSingleWord =
isSingleMessage &&
this->selection.max.charIndex - charIndex < part.getCharacterLength();
if (isSingleWord) {
// return single word
addPart(part, this->selection.min.charIndex - charIndex,
this->selection.max.charIndex - this->selection.min.charIndex);
return text;
2017-09-12 19:06:16 +02:00
} else {
// add first word of the selection
addPart(part, this->selection.min.charIndex - charIndex);
2017-09-12 19:06:16 +02:00
}
} else if (isSingleMessage && charIndex + charLength >= selection.max.charIndex) {
addPart(part, 0, this->selection.max.charIndex - charIndex);
2017-09-12 19:06:16 +02:00
return text;
} else {
text += part.getCopyText() + (part.hasTrailingSpace() ? " " : "");
2017-09-12 19:06:16 +02:00
}
charIndex += charLength;
}
text += "\n";
// middle lines
2017-09-12 19:06:16 +02:00
for (i++; i < this->selection.max.messageIndex; i++) {
for (const messages::WordPart &part : messages[i]->getWordParts()) {
if (!part.getCopyText().isEmpty()) {
text += part.getCopyText();
2017-09-12 19:06:16 +02:00
if (part.hasTrailingSpace()) {
text += " ";
}
2017-09-12 19:06:16 +02:00
}
}
text += "\n";
}
// last line
2017-09-12 19:06:16 +02:00
charIndex = 0;
for (const messages::WordPart &part :
messages[this->selection.max.messageIndex]->getWordParts()) {
int charLength = part.getCharacterLength();
if (charIndex + charLength >= this->selection.max.charIndex) {
addPart(part, 0, this->selection.max.charIndex - charIndex);
2017-09-12 19:06:16 +02:00
return text;
}
text += part.getCopyText();
if (part.hasTrailingSpace()) {
text += " ";
}
charIndex += charLength;
}
return text;
}
2017-09-21 02:20:02 +02:00
bool ChannelView::hasSelection()
{
return !this->selection.isEmpty();
}
void ChannelView::clearSelection()
{
this->selection = Selection();
layoutMessages();
}
messages::LimitedQueueSnapshot<SharedMessageRef> ChannelView::getMessagesSnapshot()
{
return this->messages.getSnapshot();
}
void ChannelView::setChannel(std::shared_ptr<Channel> channel)
{
if (this->channel) {
this->detachChannel();
}
this->messages.clear();
// on new message
this->messageAppendedConnection =
channel->messageAppended.connect([this](SharedMessage &message) {
SharedMessageRef deleted;
auto messageRef = new MessageRef(message);
if (this->messages.appendItem(SharedMessageRef(messageRef), deleted)) {
2017-12-18 22:13:46 +01:00
if (this->scrollBar.isAtBottom()) {
this->scrollBar.scrollToBottom();
} else {
this->scrollBar.offset(-1);
}
}
layoutMessages();
update();
});
// on message removed
this->messageRemovedConnection =
channel->messageRemovedFromStart.connect([this](SharedMessage &) {
this->selection.min.messageIndex--;
this->selection.max.messageIndex--;
this->selection.start.messageIndex--;
this->selection.end.messageIndex--;
this->layoutMessages();
});
auto snapshot = channel->getMessageSnapshot();
for (size_t i = 0; i < snapshot.getLength(); i++) {
SharedMessageRef deleted;
auto messageRef = new MessageRef(snapshot[i]);
this->messages.appendItem(SharedMessageRef(messageRef), deleted);
}
this->channel = channel;
this->userPopupWidget.setChannel(channel);
}
void ChannelView::detachChannel()
{
// on message added
this->messageAppendedConnection.disconnect();
// on message removed
this->messageRemovedConnection.disconnect();
}
void ChannelView::resizeEvent(QResizeEvent *)
2017-04-12 17:46:44 +02:00
{
this->scrollBar.resize(this->scrollBar.width(), height());
this->scrollBar.move(width() - this->scrollBar.width(), 0);
this->goToBottom->setGeometry(0, this->height() - 32, this->width(), 32);
this->scrollBar.raise();
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
void ChannelView::setSelection(const SelectionItem &start, const SelectionItem &end)
2017-08-18 15:12:07 +02:00
{
// selections
2017-09-12 19:06:16 +02:00
this->selection = Selection(start, end);
2017-08-18 15:12:07 +02:00
2017-09-21 02:20:02 +02:00
this->selectionChanged();
2017-09-12 19:06:16 +02:00
// qDebug() << min.messageIndex << ":" << min.charIndex << " " << max.messageIndex << ":"
// << max.charIndex;
2017-08-18 15:12:07 +02:00
}
void ChannelView::paintEvent(QPaintEvent * /*event*/)
2017-01-05 16:07:20 +01:00
{
// BENCH(timer);
2017-08-18 15:12:07 +02:00
QPainter painter(this);
2017-01-15 16:38:30 +01:00
2017-12-18 22:13:46 +01:00
painter.setRenderHint(QPainter::SmoothPixmapTransform);
2017-01-18 01:04:54 +01:00
2017-08-18 15:12:07 +02:00
// only update gif emotes
#ifndef Q_OS_MAC
2017-09-15 17:23:49 +02:00
// if (this->onlyUpdateEmotes) {
// this->onlyUpdateEmotes = false;
2017-02-07 00:03:15 +01:00
2017-09-15 17:23:49 +02:00
// for (const GifEmoteData &item : this->gifEmotes) {
// painter.fillRect(item.rect, this->colorScheme.ChatBackground);
2017-02-07 00:03:15 +01:00
2017-09-15 17:23:49 +02:00
// painter.drawPixmap(item.rect, *item.image->getPixmap());
// }
2017-02-07 00:03:15 +01:00
2017-09-15 17:23:49 +02:00
// return;
// }
#endif
2017-02-07 00:03:15 +01:00
// update all messages
this->gifEmotes.clear();
2017-02-07 00:03:15 +01:00
2017-08-18 15:12:07 +02:00
painter.fillRect(rect(), this->colorScheme.ChatBackground);
// draw messages
this->drawMessages(painter);
// draw gif emotes
for (GifEmoteData &item : this->gifEmotes) {
// painter.fillRect(item.rect, this->colorScheme.ChatBackground);
2017-02-07 00:03:15 +01:00
2017-08-18 15:12:07 +02:00
painter.drawPixmap(item.rect, *item.image->getPixmap());
}
// MARK(timer);
2017-08-18 15:12:07 +02:00
}
void ChannelView::drawMessages(QPainter &painter)
2017-08-18 15:12:07 +02:00
{
auto messages = this->getMessagesSnapshot();
2017-01-11 01:08:20 +01:00
2017-09-12 19:06:16 +02:00
size_t start = this->scrollBar.getCurrentValue();
2017-01-11 01:08:20 +01:00
if (start >= messages.getLength()) {
2017-01-18 04:33:30 +01:00
return;
}
int y = -(messages[start].get()->getHeight() * (fmod(this->scrollBar.getCurrentValue(), 1)));
2017-01-18 04:33:30 +01:00
2017-08-18 15:12:07 +02:00
for (size_t i = start; i < messages.getLength(); ++i) {
messages::MessageRef *messageRef = messages[i].get();
2017-01-18 04:33:30 +01:00
std::shared_ptr<QPixmap> buffer = messageRef->buffer;
2017-01-13 18:59:11 +01:00
// bool updateBuffer = messageRef->updateBuffer;
bool updateBuffer = false;
2017-01-11 01:08:20 +01:00
if (!buffer) {
buffer = std::shared_ptr<QPixmap>(new QPixmap(width(), messageRef->getHeight()));
updateBuffer = true;
}
2017-01-11 01:08:20 +01:00
2017-09-12 19:06:16 +02:00
updateBuffer |= this->selecting;
2017-02-07 00:03:15 +01:00
// update messages that have been changed
if (updateBuffer) {
this->updateMessageBuffer(messageRef, buffer.get(), i);
2017-10-27 22:05:08 +02:00
// qDebug() << "updating buffer xD";
2017-01-11 01:08:20 +01:00
}
2017-01-07 20:43:55 +01:00
2017-02-07 00:03:15 +01:00
// get gif emotes
for (messages::WordPart const &wordPart : messageRef->getWordParts()) {
if (wordPart.getWord().isImage()) {
messages::LazyLoadedImage &lli = wordPart.getWord().getImage();
if (lli.getAnimated()) {
GifEmoteData gifEmoteData;
gifEmoteData.image = &lli;
2017-04-12 17:46:44 +02:00
QRect rect(wordPart.getX(), wordPart.getY() + y, wordPart.getWidth(),
wordPart.getHeight());
2017-02-07 00:03:15 +01:00
gifEmoteData.rect = rect;
2017-02-07 00:03:15 +01:00
this->gifEmotes.push_back(gifEmoteData);
2017-02-07 00:03:15 +01:00
}
}
}
messageRef->buffer = buffer;
if (buffer) {
painter.drawPixmap(0, y, *buffer.get());
}
y += messageRef->getHeight();
2017-01-20 06:10:28 +01:00
if (y > height()) {
break;
}
2017-01-11 01:08:20 +01:00
}
2017-08-18 15:12:07 +02:00
}
2017-02-07 00:03:15 +01:00
void ChannelView::updateMessageBuffer(messages::MessageRef *messageRef, QPixmap *buffer,
int messageIndex)
2017-08-18 15:12:07 +02:00
{
QPainter painter(buffer);
// draw background
// if (this->selectionMin.messageIndex <= messageIndex &&
// this->selectionMax.messageIndex >= messageIndex) {
// painter.fillRect(buffer->rect(), QColor(24, 55, 25));
//} else {
painter.fillRect(buffer->rect(),
(messageRef->getMessage()->getCanHighlightTab())
? this->colorScheme.ChatBackgroundHighlighted
: this->colorScheme.ChatBackground);
2017-08-18 15:12:07 +02:00
//}
2017-09-12 19:06:16 +02:00
// draw selection
if (!selection.isEmpty()) {
drawMessageSelection(painter, messageRef, messageIndex, buffer->height());
}
// draw message
2017-08-18 15:12:07 +02:00
for (messages::WordPart const &wordPart : messageRef->getWordParts()) {
// image
if (wordPart.getWord().isImage()) {
messages::LazyLoadedImage &lli = wordPart.getWord().getImage();
const QPixmap *image = lli.getPixmap();
if (image != nullptr && !lli.getAnimated()) {
2017-08-18 15:12:07 +02:00
painter.drawPixmap(QRect(wordPart.getX(), wordPart.getY(), wordPart.getWidth(),
wordPart.getHeight()),
*image);
}
}
// text
else {
QColor color = wordPart.getWord().getColor().getColor(this->colorScheme);
2017-02-07 00:03:15 +01:00
2017-08-18 15:12:07 +02:00
this->colorScheme.normalizeColor(color);
painter.setPen(color);
painter.setFont(wordPart.getWord().getFont());
painter.drawText(QRectF(wordPart.getX(), wordPart.getY(), 10000, 10000),
wordPart.getText(), QTextOption(Qt::AlignLeft | Qt::AlignTop));
}
2017-02-07 00:03:15 +01:00
}
2017-08-18 15:12:07 +02:00
messageRef->updateBuffer = false;
2017-01-05 16:07:20 +01:00
}
2017-01-26 04:26:40 +01:00
void ChannelView::drawMessageSelection(QPainter &painter, messages::MessageRef *messageRef,
int messageIndex, int bufferHeight)
2017-09-12 19:06:16 +02:00
{
if (this->selection.min.messageIndex > messageIndex ||
this->selection.max.messageIndex < messageIndex) {
return;
}
2017-12-19 02:37:58 +01:00
QColor selectionColor = this->colorScheme.Selection;
2017-09-12 19:06:16 +02:00
int charIndex = 0;
size_t i = 0;
auto &parts = messageRef->getWordParts();
int currentLineNumber = 0;
QRect rect;
if (parts.size() > 0) {
if (selection.min.messageIndex == messageIndex) {
rect.setTop(parts.at(0).getY());
}
rect.setLeft(parts.at(0).getX());
}
// skip until selection start
if (this->selection.min.messageIndex == messageIndex && this->selection.min.charIndex != 0) {
for (; i < parts.size(); i++) {
const messages::WordPart &part = parts.at(i);
auto characterLength = part.getCharacterLength();
if (characterLength + charIndex > selection.min.charIndex) {
break;
}
charIndex += characterLength;
currentLineNumber = part.getLineNumber();
}
if (i >= parts.size()) {
return;
}
// handle word that has a cut of selection
const messages::WordPart &part = parts.at(i);
// check if selection if single word
int characterLength = part.getCharacterLength();
bool isSingleWord = charIndex + characterLength > this->selection.max.charIndex &&
this->selection.max.messageIndex == messageIndex;
rect = part.getRect();
currentLineNumber = part.getLineNumber();
if (part.getWord().isText()) {
int offset = this->selection.min.charIndex - charIndex;
for (int j = 0; j < offset; j++) {
rect.setLeft(rect.left() + part.getCharWidth(j));
2017-09-12 19:06:16 +02:00
}
if (isSingleWord) {
int length = (this->selection.max.charIndex - charIndex) - offset;
rect.setRight(part.getX());
for (int j = 0; j < offset + length; j++) {
rect.setRight(rect.right() + part.getCharWidth(j));
2017-09-12 19:06:16 +02:00
}
painter.fillRect(rect, selectionColor);
return;
}
} else {
if (isSingleWord) {
if (charIndex + 1 != this->selection.max.charIndex) {
rect.setRight(part.getX() + part.getWord().getImage().getScaledWidth());
}
painter.fillRect(rect, selectionColor);
return;
}
if (charIndex != this->selection.min.charIndex) {
rect.setLeft(part.getX() + part.getWord().getImage().getScaledWidth());
}
}
i++;
charIndex += characterLength;
}
// go through lines and draw selection
for (; i < parts.size(); i++) {
const messages::WordPart &part = parts.at(i);
int charLength = part.getCharacterLength();
bool isLastSelectedWord = this->selection.max.messageIndex == messageIndex &&
charIndex + charLength > this->selection.max.charIndex;
if (part.getLineNumber() == currentLineNumber) {
rect.setLeft(std::min(rect.left(), part.getX()));
rect.setTop(std::min(rect.top(), part.getY()));
rect.setRight(std::max(rect.right(), part.getRight()));
rect.setBottom(std::max(rect.bottom(), part.getBottom() - 1));
} else {
painter.fillRect(rect, selectionColor);
currentLineNumber = part.getLineNumber();
rect = part.getRect();
}
if (isLastSelectedWord) {
if (part.getWord().isText()) {
int offset = this->selection.min.charIndex - charIndex;
int length = (this->selection.max.charIndex - charIndex) - offset;
rect.setRight(part.getX());
for (int j = 0; j < offset + length; j++) {
rect.setRight(rect.right() + part.getCharWidth(j));
2017-09-12 19:06:16 +02:00
}
} else {
if (this->selection.max.charIndex == charIndex) {
rect.setRight(part.getX());
}
}
painter.fillRect(rect, selectionColor);
return;
}
charIndex += charLength;
}
if (this->selection.max.messageIndex != messageIndex) {
rect.setBottom(bufferHeight);
}
painter.fillRect(rect, selectionColor);
}
void ChannelView::wheelEvent(QWheelEvent *event)
2017-01-26 04:26:40 +01:00
{
if (this->scrollBar.isVisible()) {
2017-04-14 17:47:28 +02:00
auto mouseMultiplier = SettingsManager::getInstance().mouseScrollMultiplier.get();
this->scrollBar.setDesiredValue(
this->scrollBar.getDesiredValue() - event->delta() / 10.0 * mouseMultiplier, true);
}
2017-01-26 04:26:40 +01:00
}
2017-02-17 23:51:35 +01:00
void ChannelView::mouseMoveEvent(QMouseEvent *event)
2017-02-17 23:51:35 +01:00
{
std::shared_ptr<messages::MessageRef> message;
QPoint relativePos;
2017-08-18 15:12:07 +02:00
int messageIndex;
2017-02-17 23:51:35 +01:00
// no message under cursor
2017-08-18 15:12:07 +02:00
if (!tryGetMessageAt(event->pos(), message, relativePos, messageIndex)) {
this->setCursor(Qt::ArrowCursor);
return;
}
// message under cursor is collapsed
if (message->isCollapsed()) {
this->setCursor(Qt::PointingHandCursor);
2017-02-17 23:51:35 +01:00
return;
}
// is selecting
2017-08-18 15:12:07 +02:00
if (this->selecting) {
int index = message->getSelectionIndex(relativePos);
2017-09-12 19:06:16 +02:00
this->setSelection(this->selection.start, SelectionItem(messageIndex, index));
2017-08-18 15:12:07 +02:00
this->repaint();
}
2017-04-24 23:00:26 +02:00
// check if word underneath cursor
2017-09-12 19:06:16 +02:00
const messages::Word *hoverWord;
if ((hoverWord = message->tryGetWordPart(relativePos)) == nullptr) {
this->setCursor(Qt::ArrowCursor);
2017-02-17 23:51:35 +01:00
return;
}
// check if word has a link
2017-09-12 19:06:16 +02:00
if (hoverWord->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
{
this->isMouseDown = true;
2017-08-18 15:12:07 +02:00
this->lastPressPosition = event->screenPos();
2017-08-18 15:12:07 +02:00
std::shared_ptr<messages::MessageRef> message;
QPoint relativePos;
int messageIndex;
2017-09-16 16:49:52 +02:00
this->mouseDown(event);
2017-08-18 15:12:07 +02:00
if (!tryGetMessageAt(event->pos(), message, relativePos, messageIndex)) {
setCursor(Qt::ArrowCursor);
auto messages = this->getMessagesSnapshot();
if (messages.getLength() == 0) {
return;
}
// Start selection at the last message at its last index
auto lastMessageIndex = messages.getLength() - 1;
auto lastMessage = messages[lastMessageIndex];
auto lastCharacterIndex = lastMessage->getLastCharacterIndex();
SelectionItem selectionItem(lastMessageIndex, lastCharacterIndex);
this->setSelection(selectionItem, selectionItem);
this->selecting = true;
2017-08-18 15:12:07 +02:00
return;
}
// check if message is collapsed
if (message->isCollapsed()) {
return;
}
2017-08-18 15:12:07 +02:00
int index = message->getSelectionIndex(relativePos);
auto selectionItem = SelectionItem(messageIndex, index);
this->setSelection(selectionItem, selectionItem);
this->selecting = true;
this->repaint();
2017-04-12 17:46:44 +02:00
}
void ChannelView::mouseReleaseEvent(QMouseEvent *event)
2017-04-12 17:46:44 +02:00
{
if (!this->isMouseDown) {
2017-04-12 17:46:44 +02:00
// We didn't grab the mouse press, so we shouldn't be handling the mouse
// release
return;
}
this->isMouseDown = false;
2017-08-18 15:12:07 +02:00
this->selecting = false;
2017-04-12 17:46:44 +02:00
float distance = util::distanceBetweenPoints(this->lastPressPosition, event->screenPos());
2017-04-12 17:46:44 +02:00
2017-12-16 02:09:51 +01:00
// qDebug() << "Distance: " << distance;
2017-04-12 17:46:44 +02:00
if (fabsf(distance) > 15.f) {
// It wasn't a proper click, so we don't care about that here
return;
}
// If you clicked and released less than X pixels away, it counts
// as a click!
// show user thing pajaW
std::shared_ptr<messages::MessageRef> message;
QPoint relativePos;
2017-08-18 15:12:07 +02:00
int messageIndex;
2017-04-12 17:46:44 +02:00
2017-08-18 15:12:07 +02:00
if (!tryGetMessageAt(event->pos(), message, relativePos, messageIndex)) {
2017-04-12 17:46:44 +02:00
// No message at clicked position
this->userPopupWidget.hide();
2017-04-12 17:46:44 +02:00
return;
}
// message under cursor is collapsed
if (message->isCollapsed()) {
message->setCollapsed(false);
this->layoutMessages();
return;
}
2017-09-12 19:06:16 +02:00
const messages::Word *hoverWord;
2017-04-24 23:00:26 +02:00
2017-09-12 19:06:16 +02:00
if ((hoverWord = message->tryGetWordPart(relativePos)) == nullptr) {
2017-04-24 23:00:26 +02:00
return;
}
2017-09-12 19:06:16 +02:00
auto &link = hoverWord->getLink();
2017-04-12 17:46:44 +02:00
2017-04-24 23:00:26 +02:00
switch (link.getType()) {
2017-07-31 00:57:42 +02:00
case messages::Link::UserInfo: {
auto user = link.getValue();
this->userPopupWidget.setName(user);
this->userPopupWidget.move(event->screenPos().toPoint());
this->userPopupWidget.updatePermissions();
this->userPopupWidget.show();
this->userPopupWidget.setFocus();
2017-04-12 17:46:44 +02:00
2017-04-24 23:00:26 +02:00
qDebug() << "Clicked " << user << "s message";
break;
2017-07-26 09:08:19 +02:00
}
2017-07-31 00:57:42 +02:00
case messages::Link::Url: {
2017-07-26 09:08:19 +02:00
QDesktopServices::openUrl(QUrl(link.getValue()));
break;
}
2017-04-24 23:00:26 +02:00
}
2017-04-12 17:46:44 +02:00
}
bool ChannelView::tryGetMessageAt(QPoint p, std::shared_ptr<messages::MessageRef> &_message,
QPoint &relativePos, int &index)
2017-02-17 23:51:35 +01:00
{
auto messages = this->getMessagesSnapshot();
2017-02-17 23:51:35 +01:00
2017-09-12 19:06:16 +02:00
size_t start = this->scrollBar.getCurrentValue();
2017-02-17 23:51:35 +01:00
if (start >= messages.getLength()) {
2017-02-17 23:51:35 +01:00
return false;
}
int y = -(messages[start]->getHeight() * (fmod(this->scrollBar.getCurrentValue(), 1)));
2017-02-17 23:51:35 +01:00
2017-09-12 19:06:16 +02:00
for (size_t i = start; i < messages.getLength(); ++i) {
2017-02-17 23:51:35 +01:00
auto message = messages[i];
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
2017-04-14 17:52:22 +02:00
} // namespace widgets
} // namespace chatterino