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

360 lines
11 KiB
C++
Raw Normal View History

2017-06-06 17:18:23 +02:00
#pragma once
2017-01-01 02:30:42 +01:00
#include "common/FlagsEnum.hpp"
#include "messages/LimitedQueue.hpp"
#include "messages/LimitedQueueSnapshot.hpp"
#include "messages/Selection.hpp"
#include "util/ThreadGuard.hpp"
#include "widgets/BaseWidget.hpp"
#include <pajlada/signals/signal.hpp>
#include <QMenu>
2017-01-18 04:33:30 +01:00
#include <QPaintEvent>
2017-01-26 04:26:40 +01:00
#include <QScroller>
#include <QTimer>
#include <QVariantAnimation>
2017-01-26 04:26:40 +01:00
#include <QWheelEvent>
2017-01-18 04:33:30 +01:00
#include <QWidget>
2018-11-03 21:26:57 +01:00
#include <unordered_map>
#include <unordered_set>
2017-01-18 21:30:23 +01:00
namespace chatterino {
enum class HighlightState;
class Channel;
using ChannelPtr = std::shared_ptr<Channel>;
2018-11-03 21:26:57 +01:00
struct Message;
using MessagePtr = std::shared_ptr<const Message>;
enum class MessageFlag : int64_t;
2018-11-03 21:26:57 +01:00
using MessageFlags = FlagsEnum<MessageFlag>;
class MessageLayout;
using MessageLayoutPtr = std::shared_ptr<MessageLayout>;
enum class MessageElementFlag : int64_t;
using MessageElementFlags = FlagsEnum<MessageElementFlag>;
class Scrollbar;
class EffectLabel;
struct Link;
class MessageLayoutElement;
class Split;
class FilterSet;
using FilterSetPtr = std::shared_ptr<FilterSet>;
2018-11-03 21:26:57 +01:00
enum class PauseReason {
Mouse,
Selection,
DoubleClick,
2019-09-16 10:40:02 +02:00
KeyboardModifier,
2018-11-03 21:26:57 +01:00
};
enum class FromTwitchLinkOpenChannelIn {
Split,
Tab,
BrowserPlayer,
Streamlink,
};
2018-11-03 21:26:57 +01:00
using SteadyClock = std::chrono::steady_clock;
class ChannelView final : public BaseWidget
2017-01-01 02:30:42 +01:00
{
Q_OBJECT
2017-09-12 19:06:16 +02:00
2017-01-01 02:30:42 +01:00
public:
enum class Context {
None,
UserCard,
ReplyThread,
Search,
};
explicit ChannelView(BaseWidget *parent = nullptr, Split *split = nullptr,
Context context = Context::None,
size_t messagesLimit = 1000);
void queueUpdate();
2018-01-06 03:48:56 +01:00
Scrollbar &getScrollBar();
QString getSelectedText();
2017-09-21 02:20:02 +02:00
bool hasSelection();
void clearSelection();
/**
* Copies the currently selected text to the users clipboard.
*
* @see ::getSelectedText()
*/
void copySelectedText();
void setEnableScrollingToBottom(bool);
bool getEnableScrollingToBottom() const;
2018-08-07 07:55:31 +02:00
void setOverrideFlags(boost::optional<MessageElementFlags> value);
const boost::optional<MessageElementFlags> &getOverrideFlags() const;
2018-01-23 22:48:33 +01:00
void updateLastReadMessage();
/**
* Attempts to scroll to a message in this channel.
* @return <code>true</code> if the message was found and highlighted.
*/
bool scrollToMessage(const MessagePtr &message);
/**
* Attempts to scroll to a message id in this channel.
* @return <code>true</code> if the message was found and highlighted.
*/
bool scrollToMessageId(const QString &id);
2018-11-14 17:26:08 +01:00
/// Pausing
bool pausable() const;
void setPausable(bool value);
2018-11-03 21:26:57 +01:00
bool paused() const;
void pause(PauseReason reason, boost::optional<uint> msecs = boost::none);
void unpause(PauseReason reason);
MessageElementFlags getFlags() const;
2018-11-14 17:26:08 +01:00
ChannelPtr channel();
2018-06-13 13:27:10 +02:00
void setChannel(ChannelPtr channel_);
2018-11-14 17:26:08 +01:00
void setFilters(const QList<QUuid> &ids);
const QList<QUuid> getFilterIds() const;
FilterSetPtr getFilterSet() const;
ChannelPtr sourceChannel() const;
void setSourceChannel(ChannelPtr sourceChannel);
bool hasSourceChannel() const;
LimitedQueueSnapshot<MessageLayoutPtr> &getMessagesSnapshot();
2018-11-03 21:26:57 +01:00
void queueLayout();
void clearMessages();
Context getContext() const;
/**
* @brief Creates and shows a UserInfoPopup dialog
*
* @param userName The login name of the user
* @param alternativePopoutChannel Optional parameter containing the channel name to use for context
**/
void showUserInfoPopup(const QString &userName,
QString alternativePopoutChannel = QString());
2017-02-07 00:03:15 +01:00
/**
* @brief This method is meant to be used when filtering out channels.
* It <b>must</b> return true if a message belongs in this channel.
* It <b>might</b> return true if a message doesn't belong in this channel.
*/
bool mayContainMessage(const MessagePtr &message);
pajlada::Signals::Signal<QMouseEvent *> mouseDown;
pajlada::Signals::NoArgSignal selectionChanged;
pajlada::Signals::Signal<HighlightState> tabHighlightRequested;
2018-10-13 14:20:06 +02:00
pajlada::Signals::NoArgSignal liveStatusChanged;
pajlada::Signals::Signal<const Link &> linkClicked;
pajlada::Signals::Signal<QString, FromTwitchLinkOpenChannelIn>
openChannelIn;
2017-09-16 16:49:52 +02:00
2017-01-03 21:19:33 +01:00
protected:
2018-07-06 17:11:37 +02:00
void themeChangedEvent() override;
2018-12-02 18:26:21 +01:00
void scaleChangedEvent(float scale) override;
2018-01-24 20:35:26 +01:00
void resizeEvent(QResizeEvent *) override;
2017-01-03 21:19:33 +01:00
void paintEvent(QPaintEvent *) override;
void wheelEvent(QWheelEvent *event) override;
2017-01-26 04:26:40 +01:00
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
void enterEvent(QEnterEvent * /*event*/) override;
#else
void enterEvent(QEvent * /*event*/) override;
#endif
void leaveEvent(QEvent *) override;
void mouseMoveEvent(QMouseEvent *event) override;
void mousePressEvent(QMouseEvent *event) override;
void mouseReleaseEvent(QMouseEvent *event) override;
void mouseDoubleClickEvent(QMouseEvent *event) override;
2018-01-24 21:44:31 +01:00
2018-04-18 09:12:29 +02:00
void hideEvent(QHideEvent *) override;
2018-08-06 21:17:03 +02:00
void handleLinkClick(QMouseEvent *event, const Link &link,
MessageLayout *layout);
2017-02-17 23:51:35 +01:00
2018-08-06 21:17:03 +02:00
bool tryGetMessageAt(QPoint p, std::shared_ptr<MessageLayout> &message,
QPoint &relativePos, int &index);
2017-02-17 23:51:35 +01:00
2017-01-18 04:33:30 +01:00
private:
2018-08-08 20:06:20 +02:00
void initializeLayout();
void initializeScrollbar();
void initializeSignals();
2018-11-03 21:26:57 +01:00
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 messagesUpdated();
2018-08-08 20:06:20 +02:00
void performLayout(bool causedByScrollbar = false);
2018-11-03 21:26:57 +01:00
void layoutVisibleMessages(
const LimitedQueueSnapshot<MessageLayoutPtr> &messages);
void updateScrollbar(const LimitedQueueSnapshot<MessageLayoutPtr> &messages,
2018-11-03 21:26:57 +01:00
bool causedByScrollbar);
2018-07-06 19:23:47 +02:00
void drawMessages(QPainter &painter);
void setSelection(const SelectionItem &start, const SelectionItem &end);
void selectWholeMessage(MessageLayout *layout, int &messageIndex);
2018-10-06 13:43:21 +02:00
void getWordBounds(MessageLayout *layout,
const MessageLayoutElement *element,
const QPoint &relativePos, int &wordStart, int &wordEnd);
2018-07-06 19:23:47 +02:00
2018-08-06 21:17:03 +02:00
void handleMouseClick(QMouseEvent *event,
const MessageLayoutElement *hoveredElement,
MessageLayoutPtr layout);
2018-08-06 21:17:03 +02:00
void addContextMenuItems(const MessageLayoutElement *hoveredElement,
MessageLayoutPtr layout, QMouseEvent *event);
void addImageContextMenuItems(const MessageLayoutElement *hoveredElement,
MessageLayoutPtr layout, QMouseEvent *event,
QMenu &menu);
void addLinkContextMenuItems(const MessageLayoutElement *hoveredElement,
MessageLayoutPtr layout, QMouseEvent *event,
QMenu &menu);
void addMessageContextMenuItems(const MessageLayoutElement *hoveredElement,
MessageLayoutPtr layout, QMouseEvent *event,
QMenu &menu);
void addTwitchLinkContextMenuItems(
const MessageLayoutElement *hoveredElement, MessageLayoutPtr layout,
QMouseEvent *event, QMenu &menu);
void addHiddenContextMenuItems(const MessageLayoutElement *hoveredElement,
MessageLayoutPtr layout, QMouseEvent *event,
QMenu &menu);
void addCommandExecutionContextMenuItems(
const MessageLayoutElement *hoveredElement, MessageLayoutPtr layout,
QMouseEvent *event, QMenu &menu);
2018-07-06 19:23:47 +02:00
int getLayoutWidth() const;
void updatePauses();
void unpaused();
2018-07-06 19:23:47 +02:00
void enableScrolling(const QPointF &scrollStart);
void disableScrolling();
/**
* Scrolls to a message layout that must be from this view.
*
* @param layout Must be from this channel.
* @param messageIdx Must be an index into this channel.
*/
void scrollToMessageLayout(MessageLayout *layout, size_t messageIdx);
void setInputReply(const MessagePtr &message);
void showReplyThreadPopup(const MessagePtr &message);
bool canReplyToMessages() const;
2018-06-13 13:27:10 +02:00
QTimer *layoutCooldown_;
bool layoutQueued_;
2018-06-13 13:27:10 +02:00
QTimer updateTimer_;
bool updateQueued_ = false;
bool messageWasAdded_ = false;
bool lastMessageHasAlternateBackground_ = false;
2018-11-03 21:40:48 +01:00
bool lastMessageHasAlternateBackgroundReverse_ = true;
bool pausable_ = false;
2018-11-03 21:26:57 +01:00
QTimer pauseTimer_;
std::unordered_map<PauseReason, boost::optional<SteadyClock::time_point>>
pauses_;
boost::optional<SteadyClock::time_point> pauseEnd_;
int pauseScrollOffset_ = 0;
// Keeps track how many message indices we need to offset the selection when we resume scrolling
uint32_t pauseSelectionOffset_ = 0;
2018-08-07 07:55:31 +02:00
boost::optional<MessageElementFlags> overrideFlags_;
MessageLayoutPtr lastReadMessage_;
ThreadGuard snapshotGuard_;
LimitedQueueSnapshot<MessageLayoutPtr> snapshot_;
2017-09-17 02:13:57 +02:00
2020-11-01 15:23:58 +01:00
ChannelPtr channel_ = nullptr;
ChannelPtr underlyingChannel_ = nullptr;
ChannelPtr sourceChannel_ = nullptr;
Split *split_ = nullptr;
2017-04-12 17:46:44 +02:00
Scrollbar *scrollBar_;
2018-08-08 15:35:54 +02:00
EffectLabel *goToBottom_;
bool showScrollBar_ = false;
2017-02-07 00:03:15 +01:00
FilterSetPtr channelFilters_;
// Returns true if message should be included
bool shouldIncludeMessage(const MessagePtr &m) const;
// Returns whether the scrollbar should have highlights
bool showScrollbarHighlights() const;
2018-08-06 21:17:03 +02:00
// This variable can be used to decide whether or not we should render the
// "Show latest messages" button
2018-06-13 13:27:10 +02:00
bool showingLatestMessages_ = true;
bool enableScrollingToBottom_ = true;
2018-06-13 13:27:10 +02:00
bool onlyUpdateEmotes_ = false;
2017-01-05 16:07:20 +01:00
2017-04-12 17:46:44 +02:00
// Mouse event variables
bool isLeftMouseDown_ = false;
2018-06-13 13:27:10 +02:00
bool isRightMouseDown_ = false;
2018-10-06 13:43:21 +02:00
bool isDoubleClick_ = false;
2018-11-14 17:26:08 +01:00
DoubleClickSelection doubleClickSelection_;
QPointF lastLeftPressPosition_;
2018-06-13 13:27:10 +02:00
QPointF lastRightPressPosition_;
QPointF lastDClickPosition_;
QTimer *clickTimer_;
2017-01-22 12:46:35 +01:00
bool isScrolling_ = false;
QPointF lastMiddlePressPosition_;
QPointF currentMousePosition_;
QTimer scrollTimer_;
// We're only interested in the pointer, not the contents
MessageLayout *highlightedMessage_ = nullptr;
QVariantAnimation highlightAnimation_;
void setupHighlightAnimationColors();
struct {
QCursor neutral;
QCursor up;
QCursor down;
} cursors_;
Selection selection_;
2018-06-13 13:27:10 +02:00
bool selecting_ = false;
2017-08-18 15:12:07 +02:00
const Context context_;
2018-11-14 17:26:08 +01:00
LimitedQueue<MessageLayoutPtr> messages_;
pajlada::Signals::SignalHolder signalHolder_;
// channelConnections_ will be cleared when the underlying channel of the channelview changes
pajlada::Signals::SignalHolder channelConnections_;
std::unordered_set<std::shared_ptr<MessageLayout>> messagesOnScreen_;
2018-12-02 17:49:15 +01:00
static constexpr int leftPadding = 8;
static constexpr int scrollbarPadding = 8;
2017-01-22 12:46:35 +01:00
private slots:
void wordFlagsChanged()
2017-01-22 12:46:35 +01:00
{
2018-11-03 21:26:57 +01:00
queueLayout();
2017-02-07 00:03:15 +01:00
update();
2017-01-22 12:46:35 +01:00
}
void scrollUpdateRequested();
2017-01-01 02:30:42 +01:00
};
2017-06-06 17:18:23 +02:00
2017-04-14 17:52:22 +02:00
} // namespace chatterino