implemented pausing on hover

This commit is contained in:
fourtf 2018-11-03 21:26:57 +01:00
parent 22cf4368bd
commit 5453c65f0f
10 changed files with 395 additions and 304 deletions

View file

@ -88,7 +88,7 @@ void Channel::addMessage(MessagePtr message,
void Channel::addOrReplaceTimeout(MessagePtr message)
{
LimitedQueueSnapshot<MessagePtr> snapshot = this->getMessageSnapshot();
int snapshotLength = snapshot.getLength();
int snapshotLength = snapshot.size();
int end = std::max(0, snapshotLength - 20);
@ -169,7 +169,7 @@ void Channel::addOrReplaceTimeout(MessagePtr message)
void Channel::disableAllMessages()
{
LimitedQueueSnapshot<MessagePtr> snapshot = this->getMessageSnapshot();
int snapshotLength = snapshot.getLength();
int snapshotLength = snapshot.size();
for (int i = 0; i < snapshotLength; i++)
{
auto &message = snapshot[i];

View file

@ -22,7 +22,7 @@ public:
{
}
std::size_t getLength()
std::size_t size()
{
return this->length_;
}

View file

@ -244,13 +244,13 @@ void AbstractIrcServer::onConnected()
LimitedQueueSnapshot<MessagePtr> snapshot = chan->getMessageSnapshot();
bool replaceMessage = snapshot.getLength() > 0 &&
snapshot[snapshot.getLength() - 1]->flags.has(
bool replaceMessage = snapshot.size() > 0 &&
snapshot[snapshot.size() - 1]->flags.has(
MessageFlag::DisconnectedMessage);
if (replaceMessage)
{
chan->replaceMessage(snapshot[snapshot.getLength() - 1],
chan->replaceMessage(snapshot[snapshot.size() - 1],
reconnected);
continue;
}

View file

@ -4,6 +4,8 @@
#include "common/Common.hpp"
#include "controllers/accounts/AccountController.hpp"
#include "controllers/highlights/HighlightController.hpp"
#include "messages/Message.hpp"
#include "messages/MessageBuilder.hpp"
#include "providers/twitch/IrcMessageHandler.hpp"
#include "providers/twitch/PubsubClient.hpp"
#include "providers/twitch/TwitchAccount.hpp"
@ -182,6 +184,23 @@ std::shared_ptr<Channel> TwitchServer::getCustomChannel(
return this->mentionsChannel;
}
if (channelName == "$$$")
{
static auto channel =
std::make_shared<Channel>("$$$", chatterino::Channel::Type::Misc);
static auto timer = [&] {
auto timer = new QTimer;
QObject::connect(timer, &QTimer::timeout, [] {
channel->addMessage(
makeSystemMessage(QTime::currentTime().toString()));
});
timer->start(500);
return timer;
}();
return channel;
}
return nullptr;
}

View file

@ -264,7 +264,7 @@ void Scrollbar::paintEvent(QPaintEvent *)
// draw highlights
auto snapshot = this->getHighlightSnapshot();
size_t snapshotLength = snapshot.getLength();
size_t snapshotLength = snapshot.size();
if (snapshotLength == 0)
{

View file

@ -172,7 +172,7 @@ void EmotePopup::loadChannel(ChannelPtr _channel)
this->subEmotesView_->setChannel(subChannel);
this->channelEmotesView_->setChannel(channelChannel);
if (subChannel->getMessageSnapshot().getLength() == 0)
if (subChannel->getMessageSnapshot().size() == 0)
{
MessageBuilder builder;
builder->flags.set(MessageFlag::Centered);

View file

@ -114,11 +114,13 @@ ChannelView::ChannelView(BaseWidget *parent)
this->initializeScrollbar();
this->initializeSignals();
this->pauseTimeout_.setSingleShot(true);
QObject::connect(&this->pauseTimeout_, &QTimer::timeout, [this] {
this->pausedTemporarily_ = false;
this->updatePauseStatus();
this->layoutMessages();
this->pauseTimer_.setSingleShot(true);
QObject::connect(&this->pauseTimer_, &QTimer::timeout, this, [this] {
/// remove elements that are finite
for (auto it = this->pauses_.begin(); it != this->pauses_.end();)
it = it->second ? this->pauses_.erase(it) : ++it;
this->updatePauseTimer();
});
auto shortcut = new QShortcut(QKeySequence("Ctrl+C"), this);
@ -150,7 +152,7 @@ void ChannelView::initializeLayout()
void ChannelView::initializeScrollbar()
{
this->scrollBar_->getCurrentValueChanged().connect([this] {
this->actuallyLayoutMessages(true);
this->performLayout(true);
this->goToBottom_->setVisible(this->enableScrollingToBottom_ &&
this->scrollBar_->isVisible() &&
@ -158,17 +160,13 @@ void ChannelView::initializeScrollbar()
this->queueUpdate();
});
this->scrollBar_->getDesiredValueChanged().connect([this] {
this->pausedByScrollingUp_ = !this->scrollBar_->isAtBottom();
});
}
void ChannelView::initializeSignals()
{
this->connections_.push_back(
getApp()->windows->wordFlagsChanged.connect([this] {
this->layoutMessages();
this->queueLayout();
this->update();
}));
@ -181,18 +179,100 @@ void ChannelView::initializeSignals()
connections_.push_back(
getApp()->windows->layout.connect([&](Channel *channel) {
if (channel == nullptr || this->channel_.get() == channel)
this->layoutMessages();
this->queueLayout();
}));
connections_.push_back(getApp()->fonts->fontChanged.connect(
[this] { this->layoutMessages(); }));
connections_.push_back(
getApp()->fonts->fontChanged.connect([this] { this->queueLayout(); }));
}
bool ChannelView::paused() const
{
/// No elements in the map -> not paused
return !this->pauses_.empty();
}
void ChannelView::pause(PauseReason reason, boost::optional<uint> msecs)
{
if (msecs)
{
/// Msecs has a value
auto timePoint =
SteadyClock::now() + std::chrono::milliseconds(msecs.get());
auto it = this->pauses_.find(reason);
if (it == this->pauses_.end())
{
/// No value found so we insert a new one.
this->pauses_[reason] = timePoint;
}
else
{
/// If the new time point is newer then we override.
if (it->second && it->second.get() < timePoint)
it->second = timePoint;
}
}
else
{
/// Msecs is none -> pause is infinite.
/// We just override the value.
this->pauses_[reason] = boost::none;
}
this->updatePauseTimer();
}
void ChannelView::unpause(PauseReason reason)
{
/// Remove the value from the map
this->pauses_.erase(reason);
this->updatePauseTimer();
}
void ChannelView::updatePauseTimer()
{
using namespace std::chrono;
if (this->pauses_.empty())
{
/// No pauses so we can stop the timer
this->pauseEnd = boost::none;
this->pauseTimer_.stop();
this->queueLayout();
}
else if (std::any_of(this->pauses_.begin(), this->pauses_.end(),
[](auto &&value) { return !value.second; }))
{
/// Some of the pauses are infinite
this->pauseEnd = boost::none;
this->pauseTimer_.stop();
}
else
{
/// Get the maximum pause
auto max = std::max_element(
this->pauses_.begin(), this->pauses_.end(),
[](auto &&a, auto &&b) { return a.second > b.second; })
->second.get();
if (max != this->pauseEnd)
{
/// Start the timer
this->pauseEnd = max;
this->pauseTimer_.start(
duration_cast<milliseconds>(max - SteadyClock::now()));
}
}
}
void ChannelView::themeChangedEvent()
{
BaseWidget::themeChangedEvent();
this->layoutMessages();
this->queueLayout();
}
void ChannelView::queueUpdate()
@ -208,10 +288,10 @@ void ChannelView::queueUpdate()
// this->updateTimer.start();
}
void ChannelView::layoutMessages()
void ChannelView::queueLayout()
{
// if (!this->layoutCooldown->isActive()) {
this->actuallyLayoutMessages();
this->performLayout();
// this->layoutCooldown->start();
// } else {
@ -219,79 +299,86 @@ void ChannelView::layoutMessages()
// }
}
void ChannelView::actuallyLayoutMessages(bool causedByScrollbar)
void ChannelView::performLayout(bool causedByScrollbar)
{
// BenchmarkGuard benchmark("layout");
auto messagesSnapshot = this->getMessagesSnapshot();
/// Get messages and check if there are at least 1
auto messages = this->getMessagesSnapshot();
if (messagesSnapshot.getLength() == 0)
{
this->scrollBar_->setVisible(false);
return;
}
bool redrawRequired = false;
bool showScrollbar = false;
// 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();
size_t start = size_t(this->scrollBar_->getCurrentValue());
int layoutWidth = this->getLayoutWidth();
/// Layout visible messages
this->layoutVisibleMessages(messages);
MessageElementFlags flags = this->getFlags();
/// Update scrollbar
this->updateScrollbar(messages, causedByScrollbar);
}
// layout the visible messages in the view
if (messagesSnapshot.getLength() > start)
void ChannelView::layoutVisibleMessages(
LimitedQueueSnapshot<MessageLayoutPtr> &messages)
{
const auto start = size_t(this->scrollBar_->getCurrentValue());
const auto layoutWidth = this->getLayoutWidth();
const auto flags = this->getFlags();
auto redrawRequired = false;
if (messages.size() > start)
{
int y = int(-(messagesSnapshot[start]->getHeight() *
auto y = int(-(messages[start]->getHeight() *
(fmod(this->scrollBar_->getCurrentValue(), 1))));
for (size_t i = start; i < messagesSnapshot.getLength(); ++i)
for (auto i = start; i < messages.size() && y >= this->height(); i++)
{
auto message = messagesSnapshot[i];
auto message = messages[i];
redrawRequired |=
message->layout(layoutWidth, this->getScale(), flags);
y += message->getHeight();
}
}
if (y >= this->height())
if (redrawRequired)
this->queueUpdate();
}
void ChannelView::updateScrollbar(
LimitedQueueSnapshot<MessageLayoutPtr> &messages, bool causedByScrollbar)
{
if (messages.size() == 0)
{
break;
}
}
this->scrollBar_->setVisible(false);
return;
}
// layout the messages at the bottom to determine the scrollbar thumb size
int h = this->height() - 8;
/// Layout the messages at the bottom
auto h = this->height() - 8;
auto flags = this->getFlags();
auto layoutWidth = this->getLayoutWidth();
auto showScrollbar = false;
for (int i = int(messagesSnapshot.getLength()) - 1; i >= 0; i--)
// convert i to int since it checks >= 0
for (auto i = int(messages.size()) - 1; i >= 0; i--)
{
auto *message = messagesSnapshot[i].get();
auto *message = messages[i].get();
message->layout(layoutWidth, this->getScale(), flags);
h -= message->getHeight();
if (h < 0)
if (h < 0) // break condition
{
this->scrollBar_->setLargeChange(
(messagesSnapshot.getLength() - i) +
this->scrollBar_->setLargeChange((messages.size() - i) +
qreal(h) / message->getHeight());
// this->scrollBar.setDesiredValue(this->scrollBar.getDesiredValue());
showScrollbar = true;
break;
}
}
/// Update scrollbar values
this->scrollBar_->setVisible(showScrollbar);
if (!showScrollbar && !causedByScrollbar)
@ -299,26 +386,18 @@ void ChannelView::actuallyLayoutMessages(bool causedByScrollbar)
this->scrollBar_->setDesiredValue(0);
}
this->scrollBar_->setMaximum(messagesSnapshot.getLength());
this->scrollBar_->setMaximum(messages.size());
// 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 &&
getSettings()->enableSmoothScrollingNewMessages.getValue());
}
this->messageWasAdded_ = false;
}
if (redrawRequired)
{
this->queueUpdate();
}
}
void ChannelView::clearMessages()
@ -329,7 +408,7 @@ void ChannelView::clearMessages()
// Layout chat widget messages, and force an update regardless if there are
// no messages
this->layoutMessages();
this->queueLayout();
this->queueUpdate();
}
@ -363,11 +442,8 @@ QString ChannelView::getSelectedText()
? _selection.selectionMax.charIndex
: layout->getLastCharacterIndex() + 1;
qDebug() << "from:" << from << ", to:" << to;
layout->addSelectionText(result, from, to);
}
qDebug() << "xd <";
return result;
}
@ -380,7 +456,7 @@ bool ChannelView::hasSelection()
void ChannelView::clearSelection()
{
this->selection_ = Selection();
layoutMessages();
queueLayout();
}
void ChannelView::setEnableScrollingToBottom(bool value)
@ -406,7 +482,7 @@ const boost::optional<MessageElementFlags> &ChannelView::getOverrideFlags()
LimitedQueueSnapshot<MessageLayoutPtr> ChannelView::getMessagesSnapshot()
{
if (!this->isPaused() /*|| this->scrollBar_->isVisible()*/)
if (!this->paused() /*|| this->scrollBar_->isVisible()*/)
{
this->snapshot_ = this->messages.getSnapshot();
}
@ -427,6 +503,63 @@ void ChannelView::setChannel(ChannelPtr newChannel)
this->channelConnections_.push_back(newChannel->messageAppended.connect(
[this](MessagePtr &message,
boost::optional<MessageFlags> overridingFlags) {
this->messageAppended(message, overridingFlags);
}));
this->channelConnections_.push_back(
newChannel->messagesAddedAtStart.connect(
[this](std::vector<MessagePtr> &messages) {
this->messageAddedAtStart(messages);
}));
// on message removed
this->channelConnections_.push_back(
newChannel->messageRemovedFromStart.connect(
[this](MessagePtr &message) {
this->messageRemoveFromStart(message);
}));
// on message replaced
this->channelConnections_.push_back(newChannel->messageReplaced.connect(
[this](size_t index, MessagePtr replacement) {
this->messageReplaced(index, replacement);
}));
auto snapshot = newChannel->getMessageSnapshot();
for (size_t i = 0; i < snapshot.size(); i++)
{
MessageLayoutPtr deleted;
auto messageRef = new MessageLayout(snapshot[i]);
if (this->lastMessageHasAlternateBackground_)
{
messageRef->flags.set(MessageLayoutFlag::AlternateBackground);
}
this->lastMessageHasAlternateBackground_ =
!this->lastMessageHasAlternateBackground_;
this->messages.pushBack(MessageLayoutPtr(messageRef), deleted);
}
this->channel_ = newChannel;
this->queueLayout();
this->queueUpdate();
// Notifications
if (auto tc = dynamic_cast<TwitchChannel *>(newChannel.get()))
{
tc->liveStatusChanged.connect([this]() {
this->liveStatusChanged.invoke(); //
});
}
}
void ChannelView::messageAppended(MessagePtr &message,
boost::optional<MessageFlags> overridingFlags)
{
MessageLayoutPtr deleted;
auto *messageFlags = &message->flags;
@ -448,11 +581,6 @@ void ChannelView::setChannel(ChannelPtr newChannel)
this->lastMessageHasAlternateBackground_ =
!this->lastMessageHasAlternateBackground_;
if (this->isPaused())
{
this->messagesAddedSinceSelectionPause_++;
}
if (this->messages.pushBack(MessageLayoutPtr(messageRef), deleted))
{
// if (!this->isPaused()) {
@ -471,38 +599,33 @@ void ChannelView::setChannel(ChannelPtr newChannel)
{
if (messageFlags->has(MessageFlag::Highlighted))
{
this->tabHighlightRequested.invoke(
HighlightState::Highlighted);
this->tabHighlightRequested.invoke(HighlightState::Highlighted);
}
else
{
this->tabHighlightRequested.invoke(
HighlightState::NewMessage);
this->tabHighlightRequested.invoke(HighlightState::NewMessage);
}
}
if (this->channel_->getType() != Channel::Type::TwitchMentions)
{
this->scrollBar_->addHighlight(
message->getScrollBarHighlight());
this->scrollBar_->addHighlight(message->getScrollBarHighlight());
}
this->messageWasAdded_ = true;
this->layoutMessages();
}));
this->queueLayout();
}
this->channelConnections_.push_back(
newChannel->messagesAddedAtStart.connect(
[this](std::vector<MessagePtr> &messages) {
void ChannelView::messageAddedAtStart(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)));
messageRefs.at(i) = MessageLayoutPtr(new MessageLayout(messages.at(i)));
}
if (!this->isPaused())
// if (!this->isPaused())
{
if (this->messages.pushFront(messageRefs).size() > 0)
{
@ -521,42 +644,39 @@ void ChannelView::setChannel(ChannelPtr newChannel)
highlights.reserve(messages.size());
for (size_t i = 0; i < messages.size(); i++)
{
highlights.push_back(
messages.at(i)->getScrollBarHighlight());
highlights.push_back(messages.at(i)->getScrollBarHighlight());
}
this->scrollBar_->addHighlightsAtStart(highlights);
this->messageWasAdded_ = true;
this->layoutMessages();
}));
this->queueLayout();
}
// on message removed
this->channelConnections_.push_back(
newChannel->messageRemovedFromStart.connect([this](MessagePtr &) {
void ChannelView::messageRemoveFromStart(MessagePtr &message)
{
this->selection_.selectionMin.messageIndex--;
this->selection_.selectionMax.messageIndex--;
this->selection_.start.messageIndex--;
this->selection_.end.messageIndex--;
this->layoutMessages();
}));
this->queueLayout();
}
// on message replaced
this->channelConnections_.push_back(newChannel->messageReplaced.connect(
[this](size_t index, MessagePtr replacement) {
if (index >= this->messages.getSnapshot().getLength() || index < 0)
void ChannelView::messageReplaced(size_t index, MessagePtr &replacement)
{
if (index >= this->messages.getSnapshot().size() || index < 0)
{
return;
}
MessageLayoutPtr newItem(new MessageLayout(replacement));
auto snapshot = this->messages.getSnapshot();
if (index >= snapshot.getLength())
if (index >= snapshot.size())
{
log("Tried to replace out of bounds message. Index: {}. "
"Length: {}",
index, snapshot.getLength());
index, snapshot.size());
return;
}
@ -566,43 +686,11 @@ void ChannelView::setChannel(ChannelPtr newChannel)
newItem->flags.set(MessageLayoutFlag::AlternateBackground);
}
this->scrollBar_->replaceHighlight(
index, replacement->getScrollBarHighlight());
this->scrollBar_->replaceHighlight(index,
replacement->getScrollBarHighlight());
this->messages.replaceItem(message, newItem);
this->layoutMessages();
}));
auto snapshot = newChannel->getMessageSnapshot();
for (size_t i = 0; i < snapshot.getLength(); i++)
{
MessageLayoutPtr deleted;
auto messageRef = new MessageLayout(snapshot[i]);
if (this->lastMessageHasAlternateBackground_)
{
messageRef->flags.set(MessageLayoutFlag::AlternateBackground);
}
this->lastMessageHasAlternateBackground_ =
!this->lastMessageHasAlternateBackground_;
this->messages.pushBack(MessageLayoutPtr(messageRef), deleted);
}
this->channel_ = newChannel;
this->layoutMessages();
this->queueUpdate();
// Notifications
if (auto tc = dynamic_cast<TwitchChannel *>(newChannel.get()))
{
tc->liveStatusChanged.connect([this]() {
this->liveStatusChanged.invoke(); //
});
}
this->queueLayout();
}
void ChannelView::detachChannel()
@ -610,27 +698,13 @@ void ChannelView::detachChannel()
this->channelConnections_.clear();
}
void ChannelView::pause(int msecTimeout)
{
this->pausedTemporarily_ = true;
this->updatePauseStatus();
if (this->pauseTimeout_.remainingTime() < msecTimeout)
{
this->pauseTimeout_.stop();
this->pauseTimeout_.start(msecTimeout);
// qDebug() << "pause" << msecTimeout;
}
}
void ChannelView::updateLastReadMessage()
{
auto _snapshot = this->getMessagesSnapshot();
if (_snapshot.getLength() > 0)
if (_snapshot.size() > 0)
{
this->lastReadMessage_ = _snapshot[_snapshot.getLength() - 1];
this->lastReadMessage_ = _snapshot[_snapshot.size() - 1];
}
this->update();
@ -645,7 +719,7 @@ void ChannelView::resizeEvent(QResizeEvent *)
this->scrollBar_->raise();
this->layoutMessages();
this->queueLayout();
this->update();
}
@ -656,10 +730,10 @@ void ChannelView::setSelection(const SelectionItem &start,
// selections
if (!this->selecting_ && start != end)
{
this->messagesAddedSinceSelectionPause_ = 0;
// this->messagesAddedSinceSelectionPause_ = 0;
this->selecting_ = true;
this->pausedBySelection_ = true;
// this->pausedBySelection_ = true;
}
this->selection_ = Selection(start, end);
@ -695,25 +769,6 @@ MessageElementFlags ChannelView::getFlags() const
return flags;
}
bool ChannelView::isPaused()
{
// return false;
return this->pausedTemporarily_;
// || this->pausedBySelection_ || this->pausedByScrollingUp_;
}
void ChannelView::updatePauseStatus()
{
if (this->isPaused())
{
this->scrollBar_->pauseHighlights();
}
else
{
this->scrollBar_->unpauseHighlights();
}
}
void ChannelView::paintEvent(QPaintEvent * /*event*/)
{
// BenchmarkGuard benchmark("paint");
@ -724,6 +779,15 @@ void ChannelView::paintEvent(QPaintEvent * /*event*/)
// draw messages
this->drawMessages(painter);
// draw paused sign
if (this->paused())
{
auto a = this->getScale() * 16;
auto brush = QBrush(QColor(127, 127, 127, 63));
painter.fillRect(QRectF(this->width() - a, a / 4, a / 4, a), brush);
painter.fillRect(QRectF(this->width() - a / 2, a / 4, a / 4, a), brush);
}
}
// if overlays is false then it draws the message, if true then it draws things
@ -734,7 +798,7 @@ void ChannelView::drawMessages(QPainter &painter)
size_t start = size_t(this->scrollBar_->getCurrentValue());
if (start >= messagesSnapshot.getLength())
if (start >= messagesSnapshot.size())
{
return;
}
@ -745,7 +809,7 @@ void ChannelView::drawMessages(QPainter &painter)
MessageLayout *end = nullptr;
bool windowFocused = this->window() == QApplication::activeWindow();
for (size_t i = start; i < messagesSnapshot.getLength(); ++i)
for (size_t i = start; i < messagesSnapshot.size(); ++i)
{
MessageLayout *layout = messagesSnapshot[i].get();
@ -774,7 +838,7 @@ void ChannelView::drawMessages(QPainter &painter)
// 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)
for (size_t i = start; i < messagesSnapshot.size(); ++i)
{
auto it = this->messagesOnScreen_.find(messagesSnapshot[i]);
if (it != this->messagesOnScreen_.end())
@ -792,7 +856,7 @@ void ChannelView::drawMessages(QPainter &painter)
this->messagesOnScreen_.clear();
// add all messages on screen to the map
for (size_t i = start; i < messagesSnapshot.getLength(); ++i)
for (size_t i = start; i < messagesSnapshot.size(); ++i)
{
std::shared_ptr<MessageLayout> layout = messagesSnapshot[i];
@ -808,13 +872,7 @@ void ChannelView::drawMessages(QPainter &painter)
void ChannelView::wheelEvent(QWheelEvent *event)
{
if (event->orientation() != Qt::Vertical)
{
return;
}
this->pausedBySelection_ = false;
this->pausedTemporarily_ = false;
this->updatePauseStatus();
if (event->modifiers() & Qt::ControlModifier)
{
@ -830,7 +888,7 @@ void ChannelView::wheelEvent(QWheelEvent *event)
qreal delta = event->delta() * qreal(1.5) * mouseMultiplier;
auto snapshot = this->getMessagesSnapshot();
int snapshotLength = int(snapshot.getLength());
int snapshotLength = int(snapshot.size());
int i = std::min<int>(int(desired), snapshotLength);
if (delta > 0)
@ -888,7 +946,7 @@ void ChannelView::wheelEvent(QWheelEvent *event)
if (i == snapshotLength - 1)
{
desired = snapshot.getLength();
desired = snapshot.size();
}
else
{
@ -912,9 +970,7 @@ void ChannelView::enterEvent(QEvent *)
void ChannelView::leaveEvent(QEvent *)
{
this->pausedTemporarily_ = false;
this->updatePauseStatus();
this->layoutMessages();
this->queueLayout();
}
void ChannelView::mouseMoveEvent(QMouseEvent *event)
@ -927,9 +983,10 @@ void ChannelView::mouseMoveEvent(QMouseEvent *event)
return;
}
/// Pause on hover
if (getSettings()->pauseChatOnHover.getValue())
{
this->pause(CHAT_HOVER_PAUSE_DURATION);
this->pause(PauseReason::Mouse, 500);
}
auto tooltipWidget = TooltipWidget::getInstance();
@ -948,7 +1005,7 @@ void ChannelView::mouseMoveEvent(QMouseEvent *event)
// is selecting
if (this->isMouseDown_)
{
this->pause(300);
// this->pause(PauseReason::Selecting, 300);
int index = layout->getSelectionIndex(relativePos);
this->setSelection(this->selection_.start,
@ -1154,7 +1211,7 @@ void ChannelView::mousePressEvent(QMouseEvent *event)
{
setCursor(Qt::ArrowCursor);
auto messagesSnapshot = this->getMessagesSnapshot();
if (messagesSnapshot.getLength() == 0)
if (messagesSnapshot.size() == 0)
{
return;
}
@ -1162,7 +1219,7 @@ void ChannelView::mousePressEvent(QMouseEvent *event)
// Start selection at the last message at its last index
if (event->button() == Qt::LeftButton)
{
auto lastMessageIndex = messagesSnapshot.getLength() - 1;
auto lastMessageIndex = messagesSnapshot.size() - 1;
auto lastMessage = messagesSnapshot[lastMessageIndex];
auto lastCharacterIndex = lastMessage->getLastCharacterIndex();
@ -1181,13 +1238,11 @@ void ChannelView::mousePressEvent(QMouseEvent *event)
this->isMouseDown_ = true;
if (layout->flags.has(MessageLayoutFlag::Collapsed))
{
return;
}
if (getSettings()->linksDoubleClickOnly.getValue())
{
this->pause(200);
this->pause(PauseReason::DoubleClick, 200);
}
int index = layout->getSelectionIndex(relativePos);
@ -1265,7 +1320,7 @@ void ChannelView::mouseReleaseEvent(QMouseEvent *event)
return;
}
// find message
this->layoutMessages();
this->queueLayout();
std::shared_ptr<MessageLayout> layout;
QPoint relativePos;
@ -1284,7 +1339,7 @@ void ChannelView::mouseReleaseEvent(QMouseEvent *event)
layout->flags.set(MessageLayoutFlag::Expanded);
layout->flags.set(MessageLayoutFlag::RequiresLayout);
this->layoutMessages();
this->queueLayout();
return;
}
@ -1319,18 +1374,12 @@ void ChannelView::handleMouseClick(QMouseEvent *event,
{
if (this->selecting_)
{
if (this->messagesAddedSinceSelectionPause_ >
SELECTION_RESUME_SCROLLING_MSG_THRESHOLD)
{
this->showingLatestMessages_ = false;
}
// this->pausedBySelection = false;
this->selecting_ = false;
// this->pauseTimeout.stop();
// this->pausedTemporarily = false;
this->layoutMessages();
this->queueLayout();
}
auto &link = hoveredElement->getLink();
@ -1601,7 +1650,7 @@ bool ChannelView::tryGetMessageAt(QPoint p,
size_t start = this->scrollBar_->getCurrentValue();
if (start >= messagesSnapshot.getLength())
if (start >= messagesSnapshot.size())
{
return false;
}
@ -1609,7 +1658,7 @@ bool ChannelView::tryGetMessageAt(QPoint p,
int y = -(messagesSnapshot[start]->getHeight() *
(fmod(this->scrollBar_->getCurrentValue(), 1)));
for (size_t i = start; i < messagesSnapshot.getLength(); ++i)
for (size_t i = start; i < messagesSnapshot.size(); ++i)
{
auto message = messagesSnapshot[i];

View file

@ -13,6 +13,7 @@
#include <QWidget>
#include <pajlada/signals/signal.hpp>
#include <unordered_map>
#include <unordered_set>
namespace chatterino {
@ -21,6 +22,12 @@ enum class HighlightState;
class Channel;
using ChannelPtr = std::shared_ptr<Channel>;
struct Message;
using MessagePtr = std::shared_ptr<const Message>;
enum class MessageFlag : uint16_t;
using MessageFlags = FlagsEnum<MessageFlag>;
class MessageLayout;
using MessageLayoutPtr = std::shared_ptr<MessageLayout>;
@ -32,6 +39,14 @@ class EffectLabel;
struct Link;
class MessageLayoutElement;
enum class PauseReason {
Mouse,
Selection,
DoubleClick,
};
using SteadyClock = std::chrono::steady_clock;
class ChannelView final : public BaseWidget
{
Q_OBJECT
@ -48,12 +63,15 @@ public:
bool getEnableScrollingToBottom() const;
void setOverrideFlags(boost::optional<MessageElementFlags> value);
const boost::optional<MessageElementFlags> &getOverrideFlags() const;
void pause(int msecTimeout);
void updateLastReadMessage();
bool paused() const;
void pause(PauseReason reason, boost::optional<uint> msecs = boost::none);
void unpause(PauseReason reason);
void setChannel(ChannelPtr channel_);
LimitedQueueSnapshot<MessageLayoutPtr> getMessagesSnapshot();
void layoutMessages();
void queueLayout();
void clearMessages();
void showUserInfoPopup(const QString &userName);
@ -94,18 +112,23 @@ private:
void initializeScrollbar();
void initializeSignals();
// void messageAppended(MessagePtr &message);
// void messageAddedAtStart(std::vector<MessagePtr> &messages);
// void messageRemoveFromStart(MessagePtr &message);
void messageAppended(MessagePtr &message,
boost::optional<MessageFlags> overridingFlags);
void messageAddedAtStart(std::vector<MessagePtr> &messages);
void messageRemoveFromStart(MessagePtr &message);
void messageReplaced(size_t index, MessagePtr &replacement);
void updatePauseStatus();
void detachChannel();
void actuallyLayoutMessages(bool causedByScollbar = false);
void performLayout(bool causedByScollbar = false);
void layoutVisibleMessages(
LimitedQueueSnapshot<MessageLayoutPtr> &messages);
void updateScrollbar(LimitedQueueSnapshot<MessageLayoutPtr> &messages,
bool causedByScrollbar);
void drawMessages(QPainter &painter);
void setSelection(const SelectionItem &start, const SelectionItem &end);
MessageElementFlags getFlags() const;
bool isPaused();
void selectWholeMessage(MessageLayout *layout, int &messageIndex);
void getWordBounds(MessageLayout *layout,
const MessageLayoutElement *element,
@ -117,6 +140,7 @@ private:
void addContextMenuItems(const MessageLayoutElement *hoveredElement,
MessageLayout *layout);
int getLayoutWidth() const;
void updatePauseTimer();
QTimer *layoutCooldown_;
bool layoutQueued_;
@ -126,12 +150,11 @@ private:
bool messageWasAdded_ = false;
bool lastMessageHasAlternateBackground_ = false;
bool pausedTemporarily_ = false;
bool pausedBySelection_ = false;
bool pausedByScrollingUp_ = false;
int messagesAddedSinceSelectionPause_ = 0;
QTimer pauseTimer_;
std::unordered_map<PauseReason, boost::optional<SteadyClock::time_point>>
pauses_;
boost::optional<SteadyClock::time_point> pauseEnd;
QTimer pauseTimeout_;
boost::optional<MessageElementFlags> overrideFlags_;
MessageLayoutPtr lastReadMessage_;
@ -179,7 +202,7 @@ private:
private slots:
void wordFlagsChanged()
{
layoutMessages();
queueLayout();
update();
}
};

View file

@ -74,7 +74,7 @@ void SearchPopup::performSearch()
ChannelPtr channel(new Channel("search", Channel::Type::None));
for (size_t i = 0; i < this->snapshot_.getLength(); i++)
for (size_t i = 0; i < this->snapshot_.size(); i++)
{
MessagePtr message = this->snapshot_[i];

View file

@ -274,7 +274,7 @@ void Split::setModerationMode(bool value)
{
this->moderationMode_ = value;
this->header_->updateModerationModeIcon();
this->view_->layoutMessages();
this->view_->queueLayout();
}
bool Split::getModerationMode() const
@ -322,7 +322,7 @@ void Split::showChangeChannelPopup(const char *dialogTitle, bool empty,
void Split::layoutMessages()
{
this->view_->layoutMessages();
this->view_->queueLayout();
}
void Split::updateGifEmotes()