refactor: make a single MessageBuilder (#5548)

This commit is contained in:
pajlada 2024-08-24 12:18:27 +02:00 committed by GitHub
parent 5170085d7c
commit 175afa8b16
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
33 changed files with 2819 additions and 3014 deletions

View file

@ -72,6 +72,7 @@
- Dev: Refactored a few `#define`s into `const(expr)` and cleaned includes. (#5527) - Dev: Refactored a few `#define`s into `const(expr)` and cleaned includes. (#5527)
- Dev: Added `FlagsEnum::isEmpty`. (#5550) - Dev: Added `FlagsEnum::isEmpty`. (#5550)
- Dev: Prepared for Qt 6.8 by addressing some deprecations. (#5529) - Dev: Prepared for Qt 6.8 by addressing some deprecations. (#5529)
- Dev: Refactored `MessageBuilder` to be a single class. (#5548)
- Dev: Recent changes are now shown in the nightly release description. (#5553, #5554) - Dev: Recent changes are now shown in the nightly release description. (#5553, #5554)
## 2.5.1 ## 2.5.1

View file

@ -4,7 +4,7 @@
#include "controllers/highlights/HighlightController.hpp" #include "controllers/highlights/HighlightController.hpp"
#include "controllers/highlights/HighlightPhrase.hpp" #include "controllers/highlights/HighlightPhrase.hpp"
#include "messages/Message.hpp" #include "messages/Message.hpp"
#include "messages/SharedMessageBuilder.hpp" #include "messages/MessageBuilder.hpp"
#include "mocks/BaseApplication.hpp" #include "mocks/BaseApplication.hpp"
#include "singletons/Settings.hpp" #include "singletons/Settings.hpp"
#include "util/Helpers.hpp" #include "util/Helpers.hpp"
@ -16,15 +16,16 @@
using namespace chatterino; using namespace chatterino;
class BenchmarkMessageBuilder : public SharedMessageBuilder class BenchmarkMessageBuilder : public MessageBuilder
{ {
public: public:
explicit BenchmarkMessageBuilder( explicit BenchmarkMessageBuilder(
Channel *_channel, const Communi::IrcPrivateMessage *_ircMessage, Channel *_channel, const Communi::IrcPrivateMessage *_ircMessage,
const MessageParseArgs &_args) const MessageParseArgs &_args)
: SharedMessageBuilder(_channel, _ircMessage, _args) : MessageBuilder(_channel, _ircMessage, _args)
{ {
} }
virtual MessagePtr build() virtual MessagePtr build()
{ {
// PARSE // PARSE

View file

@ -42,7 +42,6 @@
#include "providers/twitch/PubSubMessages.hpp" #include "providers/twitch/PubSubMessages.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchIrcServer.hpp" #include "providers/twitch/TwitchIrcServer.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/CrashHandler.hpp" #include "singletons/CrashHandler.hpp"
#include "singletons/Emotes.hpp" #include "singletons/Emotes.hpp"
#include "singletons/Fonts.hpp" #include "singletons/Fonts.hpp"
@ -738,11 +737,9 @@ void Application::initPubSub()
return; return;
} }
MessageBuilder msg; auto msg = MessageBuilder::makeDeletionMessageFromPubSub(action);
TwitchMessageBuilder::deletionMessage(action, &msg);
msg->flags.set(MessageFlag::PubSub);
postToThread([chan, msg = msg.release()] { postToThread([chan, msg] {
auto replaced = false; auto replaced = false;
LimitedQueueSnapshot<MessagePtr> snapshot = LimitedQueueSnapshot<MessagePtr> snapshot =
chan->getMessageSnapshot(); chan->getMessageSnapshot();
@ -827,10 +824,8 @@ void Application::initPubSub()
} }
postToThread([twitchChannel, action] { postToThread([twitchChannel, action] {
const auto p = const auto p = MessageBuilder::makeLowTrustUserMessage(
TwitchMessageBuilder::makeLowTrustUserMessage( action, twitchChannel->getName(), twitchChannel.get());
action, twitchChannel->getName(),
twitchChannel.get());
twitchChannel->addMessage(p.first, twitchChannel->addMessage(p.first,
MessageContext::Original); MessageContext::Original);
twitchChannel->addMessage(p.second, twitchChannel->addMessage(p.second,
@ -871,7 +866,7 @@ void Application::initPubSub()
postToThread([chan, action] { postToThread([chan, action] {
auto msg = auto msg =
TwitchMessageBuilder::makeLowTrustUpdateMessage(action); MessageBuilder::makeLowTrustUpdateMessage(action);
chan->addMessage(msg, MessageContext::Original); chan->addMessage(msg, MessageContext::Original);
}); });
}); });
@ -951,8 +946,7 @@ void Application::initPubSub()
ActionUser{msg.senderUserID, msg.senderUserLogin, ActionUser{msg.senderUserID, msg.senderUserLogin,
senderDisplayName, senderColor}; senderDisplayName, senderColor};
postToThread([chan, action] { postToThread([chan, action] {
const auto p = const auto p = MessageBuilder::makeAutomodMessage(
TwitchMessageBuilder::makeAutomodMessage(
action, chan->getName()); action, chan->getName());
chan->addMessage(p.first, MessageContext::Original); chan->addMessage(p.first, MessageContext::Original);
chan->addMessage(p.second, chan->addMessage(p.second,
@ -1004,8 +998,8 @@ void Application::initPubSub()
} }
postToThread([chan, action] { postToThread([chan, action] {
const auto p = TwitchMessageBuilder::makeAutomodMessage( const auto p =
action, chan->getName()); MessageBuilder::makeAutomodMessage(action, chan->getName());
chan->addMessage(p.first, MessageContext::Original); chan->addMessage(p.first, MessageContext::Original);
chan->addMessage(p.second, MessageContext::Original); chan->addMessage(p.second, MessageContext::Original);
}); });
@ -1043,8 +1037,7 @@ void Application::initPubSub()
} }
postToThread([chan, action] { postToThread([chan, action] {
const auto p = const auto p = MessageBuilder::makeAutomodInfoMessage(action);
TwitchMessageBuilder::makeAutomodInfoMessage(action);
chan->addMessage(p, MessageContext::Original); chan->addMessage(p, MessageContext::Original);
}); });
}); });

View file

@ -280,9 +280,6 @@ set(SOURCE_FILES
messages/MessageThread.cpp messages/MessageThread.cpp
messages/MessageThread.hpp messages/MessageThread.hpp
messages/SharedMessageBuilder.cpp
messages/SharedMessageBuilder.hpp
messages/layouts/MessageLayout.cpp messages/layouts/MessageLayout.cpp
messages/layouts/MessageLayout.hpp messages/layouts/MessageLayout.hpp
messages/layouts/MessageLayoutContainer.cpp messages/layouts/MessageLayoutContainer.cpp
@ -405,8 +402,6 @@ set(SOURCE_FILES
providers/twitch/TwitchHelpers.hpp providers/twitch/TwitchHelpers.hpp
providers/twitch/TwitchIrcServer.cpp providers/twitch/TwitchIrcServer.cpp
providers/twitch/TwitchIrcServer.hpp providers/twitch/TwitchIrcServer.hpp
providers/twitch/TwitchMessageBuilder.cpp
providers/twitch/TwitchMessageBuilder.hpp
providers/twitch/TwitchUser.cpp providers/twitch/TwitchUser.cpp
providers/twitch/TwitchUser.hpp providers/twitch/TwitchUser.hpp

View file

@ -3,7 +3,6 @@
#include "common/Channel.hpp" #include "common/Channel.hpp"
#include "messages/Message.hpp" #include "messages/Message.hpp"
#include "messages/MessageBuilder.hpp" #include "messages/MessageBuilder.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include <QColor> #include <QColor>
@ -39,11 +38,10 @@ void ChannelChatters::addJoinedUser(const QString &user)
auto joinedUsers = this->joinedUsers_.access(); auto joinedUsers = this->joinedUsers_.access();
joinedUsers->sort(); joinedUsers->sort();
MessageBuilder builder; this->channel_.addMessage(
TwitchMessageBuilder::listOfUsersSystemMessage( MessageBuilder::makeListOfUsersMessage(
"Users joined:", *joinedUsers, &this->channel_, &builder); "Users joined:", *joinedUsers, &this->channel_,
builder->flags.set(MessageFlag::Collapsed); {MessageFlag::Collapsed}),
this->channel_.addMessage(builder.release(),
MessageContext::Original); MessageContext::Original);
joinedUsers->clear(); joinedUsers->clear();
@ -65,11 +63,10 @@ void ChannelChatters::addPartedUser(const QString &user)
auto partedUsers = this->partedUsers_.access(); auto partedUsers = this->partedUsers_.access();
partedUsers->sort(); partedUsers->sort();
MessageBuilder builder; this->channel_.addMessage(
TwitchMessageBuilder::listOfUsersSystemMessage( MessageBuilder::makeListOfUsersMessage(
"Users parted:", *partedUsers, &this->channel_, &builder); "Users parted:", *partedUsers, &this->channel_,
builder->flags.set(MessageFlag::Collapsed); {MessageFlag::Collapsed}),
this->channel_.addMessage(builder.release(),
MessageContext::Original); MessageContext::Original);
partedUsers->clear(); partedUsers->clear();

View file

@ -6,7 +6,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "util/Twitch.hpp" #include "util/Twitch.hpp"
namespace chatterino::commands { namespace chatterino::commands {

View file

@ -6,7 +6,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "util/Twitch.hpp" #include "util/Twitch.hpp"
namespace chatterino::commands { namespace chatterino::commands {

View file

@ -11,7 +11,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Theme.hpp" #include "singletons/Theme.hpp"
#include <QApplication> #include <QApplication>
@ -125,11 +124,9 @@ QString testChatters(const CommandContext &ctx)
prefix += QString("(%1):").arg(result.total); prefix += QString("(%1):").arg(result.total);
} }
MessageBuilder builder; channel->addMessage(MessageBuilder::makeListOfUsersMessage(
TwitchMessageBuilder::listOfUsersSystemMessage( prefix, entries, twitchChannel),
prefix, entries, twitchChannel, &builder); MessageContext::Original);
channel->addMessage(builder.release(), MessageContext::Original);
}, },
[channel{ctx.channel}](auto error, auto message) { [channel{ctx.channel}](auto error, auto message) {
auto errorMessage = formatChattersError(error, message); auto errorMessage = formatChattersError(error, message);

View file

@ -5,7 +5,6 @@
#include "messages/MessageBuilder.hpp" #include "messages/MessageBuilder.hpp"
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
namespace { namespace {
@ -77,11 +76,10 @@ QString getModerators(const CommandContext &ctx)
// TODO: sort results? // TODO: sort results?
MessageBuilder builder; channel->addMessage(MessageBuilder::makeListOfUsersMessage(
TwitchMessageBuilder::listOfUsersSystemMessage( "The moderators of this channel are",
"The moderators of this channel are", result, twitchChannel, result, twitchChannel),
&builder); MessageContext::Original);
channel->addMessage(builder.release(), MessageContext::Original);
}, },
[channel{ctx.channel}](auto error, auto message) { [channel{ctx.channel}](auto error, auto message) {
auto errorMessage = formatModsError(error, message); auto errorMessage = formatModsError(error, message);

View file

@ -7,7 +7,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
namespace { namespace {
@ -106,11 +105,10 @@ QString getVIPs(const CommandContext &ctx)
auto messagePrefix = QString("The VIPs of this channel are"); auto messagePrefix = QString("The VIPs of this channel are");
// TODO: sort results? // TODO: sort results?
MessageBuilder builder;
TwitchMessageBuilder::listOfUsersSystemMessage(
messagePrefix, vipList, twitchChannel, &builder);
channel->addMessage(builder.release(), MessageContext::Original); channel->addMessage(MessageBuilder::makeListOfUsersMessage(
messagePrefix, vipList, twitchChannel),
MessageContext::Original);
}, },
[channel{ctx.channel}](auto error, auto message) { [channel{ctx.channel}](auto error, auto message) {
auto errorMessage = formatGetVIPsError(error, message); auto errorMessage = formatGetVIPsError(error, message);

View file

@ -6,7 +6,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "util/Twitch.hpp" #include "util/Twitch.hpp"
namespace chatterino::commands { namespace chatterino::commands {

View file

@ -8,7 +8,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "util/Twitch.hpp" #include "util/Twitch.hpp"
namespace chatterino::commands { namespace chatterino::commands {

View file

@ -6,7 +6,6 @@
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
namespace { namespace {

View file

@ -1,8 +1,7 @@
#include "controllers/highlights/HighlightBadge.hpp" #include "controllers/highlights/HighlightBadge.hpp"
#include "messages/SharedMessageBuilder.hpp"
#include "providers/twitch/TwitchBadge.hpp" #include "providers/twitch/TwitchBadge.hpp"
#include "singletons/Resources.hpp" #include "util/IrcHelpers.hpp"
namespace chatterino { namespace chatterino {
@ -97,7 +96,7 @@ bool HighlightBadge::compare(const QString &id, const Badge &badge) const
{ {
if (this->hasVersions_) if (this->hasVersions_)
{ {
auto parts = SharedMessageBuilder::slashKeyValue(id); auto parts = slashKeyValue(id);
return parts.first.compare(badge.key_, Qt::CaseInsensitive) == 0 && return parts.first.compare(badge.key_, Qt::CaseInsensitive) == 0 &&
parts.second.compare(badge.value_, Qt::CaseInsensitive) == 0; parts.second.compare(badge.value_, Qt::CaseInsensitive) == 0;
} }

View file

@ -5,9 +5,9 @@
#include "controllers/notifications/NotificationModel.hpp" #include "controllers/notifications/NotificationModel.hpp"
#include "controllers/sound/ISoundController.hpp" #include "controllers/sound/ISoundController.hpp"
#include "messages/Message.hpp" #include "messages/Message.hpp"
#include "messages/MessageBuilder.hpp"
#include "providers/twitch/api/Helix.hpp" #include "providers/twitch/api/Helix.hpp"
#include "providers/twitch/TwitchIrcServer.hpp" #include "providers/twitch/TwitchIrcServer.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Settings.hpp" #include "singletons/Settings.hpp"
#include "singletons/StreamerMode.hpp" #include "singletons/StreamerMode.hpp"
#include "singletons/Toasts.hpp" #include "singletons/Toasts.hpp"
@ -137,11 +137,9 @@ void NotificationController::notifyTwitchChannelLive(
} }
// Message in /live channel // Message in /live channel
MessageBuilder builder;
TwitchMessageBuilder::liveMessage(payload.displayName, &builder);
builder.message().id = payload.channelId;
getApp()->getTwitch()->getLiveChannel()->addMessage( getApp()->getTwitch()->getLiveChannel()->addMessage(
builder.release(), MessageContext::Original); MessageBuilder::makeLiveMessage(payload.displayName, payload.channelId),
MessageContext::Original);
// Notify on all channels with a ping sound // Notify on all channels with a ping sound
if (showNotification && !playedSound && if (showNotification && !playedSound &&

File diff suppressed because it is too large Load diff

View file

@ -1,15 +1,25 @@
#pragma once #pragma once
#include "common/Aliases.hpp"
#include "common/Outcome.hpp"
#include "messages/MessageColor.hpp" #include "messages/MessageColor.hpp"
#include "messages/MessageFlag.hpp"
#include "providers/twitch/pubsubmessages/LowTrustUsers.hpp"
#include <IrcMessage>
#include <QRegularExpression> #include <QRegularExpression>
#include <QString>
#include <QTime> #include <QTime>
#include <QVariant>
#include <ctime> #include <ctime>
#include <memory> #include <memory>
#include <optional>
#include <unordered_map>
#include <utility> #include <utility>
namespace chatterino { namespace chatterino {
struct BanAction; struct BanAction;
struct UnbanAction; struct UnbanAction;
struct WarnAction; struct WarnAction;
@ -24,6 +34,15 @@ class TextElement;
struct Emote; struct Emote;
using EmotePtr = std::shared_ptr<const Emote>; using EmotePtr = std::shared_ptr<const Emote>;
class Channel;
class TwitchChannel;
class MessageThread;
class IgnorePhrase;
struct HelixVip;
using HelixModerator = HelixVip;
struct ChannelPointReward;
struct DeleteAction;
namespace linkparser { namespace linkparser {
struct Parsed; struct Parsed;
} // namespace linkparser } // namespace linkparser
@ -67,10 +86,36 @@ struct MessageParseArgs {
QString channelPointRewardId = ""; QString channelPointRewardId = "";
}; };
struct TwitchEmoteOccurrence {
int start;
int end;
EmotePtr ptr;
EmoteName name;
bool operator==(const TwitchEmoteOccurrence &other) const
{
return std::tie(this->start, this->end, this->ptr, this->name) ==
std::tie(other.start, other.end, other.ptr, other.name);
}
};
class MessageBuilder class MessageBuilder
{ {
public: public:
/// Build a message without a base IRC message.
MessageBuilder(); MessageBuilder();
/// Build a message based on an incoming IRC PRIVMSG
explicit MessageBuilder(Channel *_channel,
const Communi::IrcPrivateMessage *_ircMessage,
const MessageParseArgs &_args);
/// Build a message based on an incoming IRC message (e.g. notice)
explicit MessageBuilder(Channel *_channel,
const Communi::IrcMessage *_ircMessage,
const MessageParseArgs &_args, QString content,
bool isAction);
MessageBuilder(SystemMessageTag, const QString &text, MessageBuilder(SystemMessageTag, const QString &text,
const QTime &time = QTime::currentTime()); const QTime &time = QTime::currentTime());
MessageBuilder(TimeoutMessageTag, const QString &timeoutUser, MessageBuilder(TimeoutMessageTag, const QString &timeoutUser,
@ -106,7 +151,16 @@ public:
const QString &deletionLink, size_t imagesStillQueued = 0, const QString &deletionLink, size_t imagesStillQueued = 0,
size_t secondsLeft = 0); size_t secondsLeft = 0);
virtual ~MessageBuilder() = default; MessageBuilder(const MessageBuilder &) = delete;
MessageBuilder(MessageBuilder &&) = delete;
MessageBuilder &operator=(const MessageBuilder &) = delete;
MessageBuilder &operator=(MessageBuilder &&) = delete;
~MessageBuilder() = default;
QString userName;
TwitchChannel *twitchChannel = nullptr;
Message *operator->(); Message *operator->();
Message &message(); Message &message();
@ -117,10 +171,7 @@ public:
void addLink(const linkparser::Parsed &parsedLink, const QString &source); void addLink(const linkparser::Parsed &parsedLink, const QString &source);
template <typename T, typename... Args> template <typename T, typename... Args>
// clang-format off
// clang-format can be enabled once clang-format v11+ has been installed in CI
T *emplace(Args &&...args) T *emplace(Args &&...args)
// clang-format on
{ {
static_assert(std::is_base_of<MessageElement, T>::value, static_assert(std::is_base_of<MessageElement, T>::value,
"T must extend MessageElement"); "T must extend MessageElement");
@ -131,9 +182,70 @@ public:
return pointer; return pointer;
} }
[[nodiscard]] bool isIgnored() const;
bool isIgnoredReply() const;
void triggerHighlights();
MessagePtr build();
void setThread(std::shared_ptr<MessageThread> thread);
void setParent(MessagePtr parent);
void setMessageOffset(int offset);
void appendChannelPointRewardMessage(const ChannelPointReward &reward,
bool isMod, bool isBroadcaster);
static MessagePtr makeChannelPointRewardMessage(
const ChannelPointReward &reward, bool isMod, bool isBroadcaster);
/// Make a "CHANNEL_NAME has gone live!" message
static MessagePtr makeLiveMessage(const QString &channelName,
const QString &channelID,
MessageFlags extraFlags = {});
// Messages in normal chat for channel stuff
static MessagePtr makeOfflineSystemMessage(const QString &channelName,
const QString &channelID);
static MessagePtr makeHostingSystemMessage(const QString &channelName,
bool hostOn);
static MessagePtr makeDeletionMessageFromIRC(
const MessagePtr &originalMessage);
static MessagePtr makeDeletionMessageFromPubSub(const DeleteAction &action);
static MessagePtr makeListOfUsersMessage(QString prefix, QStringList users,
Channel *channel,
MessageFlags extraFlags = {});
static MessagePtr makeListOfUsersMessage(
QString prefix, const std::vector<HelixModerator> &users,
Channel *channel, MessageFlags extraFlags = {});
static MessagePtr buildHypeChatMessage(Communi::IrcPrivateMessage *message);
static std::pair<MessagePtr, MessagePtr> makeAutomodMessage(
const AutomodAction &action, const QString &channelName);
static MessagePtr makeAutomodInfoMessage(const AutomodInfoAction &action);
static std::pair<MessagePtr, MessagePtr> makeLowTrustUserMessage(
const PubSubLowTrustUsersMessage &action, const QString &channelName,
const TwitchChannel *twitchChannel);
static MessagePtr makeLowTrustUpdateMessage(
const PubSubLowTrustUsersMessage &action);
static std::unordered_map<QString, QString> parseBadgeInfoTag(
const QVariantMap &tags);
// Parses "badges" tag which contains a comma separated list of key-value elements
static std::vector<Badge> parseBadgeTag(const QVariantMap &tags);
static std::vector<TwitchEmoteOccurrence> parseTwitchEmotes(
const QVariantMap &tags, const QString &originalMessage,
int messageOffset);
static void processIgnorePhrases(
const std::vector<IgnorePhrase> &phrases, QString &originalMessage,
std::vector<TwitchEmoteOccurrence> &twitchEmotes);
protected: protected:
virtual void addTextOrEmoji(EmotePtr emote); void addTextOrEmoji(EmotePtr emote);
virtual void addTextOrEmoji(const QString &value); void addTextOrEmoji(const QString &string_);
bool isEmpty() const; bool isEmpty() const;
MessageElement &back(); MessageElement &back();
@ -141,7 +253,6 @@ protected:
MessageColor textColor_ = MessageColor::Text; MessageColor textColor_ = MessageColor::Text;
private:
// Helper method that emplaces some text stylized as system text // Helper method that emplaces some text stylized as system text
// and then appends that text to the QString parameter "toUpdate". // and then appends that text to the QString parameter "toUpdate".
// Returns the TextElement that was emplaced. // Returns the TextElement that was emplaced.
@ -149,6 +260,70 @@ private:
QString &toUpdate); QString &toUpdate);
std::shared_ptr<Message> message_; std::shared_ptr<Message> message_;
void parse();
void parseUsernameColor();
void parseUsername();
void parseMessageID();
void parseRoomID();
// Parse & build thread information into the message
// Will read information from thread_ or from IRC tags
void parseThread();
// parseHighlights only updates the visual state of the message, but leaves the playing of alerts and sounds to the triggerHighlights function
void parseHighlights();
void appendChannelName();
void appendUsername();
Outcome tryAppendEmote(const EmoteName &name);
void addWords(const QStringList &words,
const std::vector<TwitchEmoteOccurrence> &twitchEmotes);
void appendTwitchBadges();
void appendChatterinoBadges();
void appendFfzBadges();
void appendSeventvBadges();
Outcome tryParseCheermote(const QString &string);
bool shouldAddModerationElements() const;
QString roomID_;
bool hasBits_ = false;
QString bits;
int bitsLeft{};
bool bitsStacked = false;
bool historicalMessage_ = false;
std::shared_ptr<MessageThread> thread_;
MessagePtr parent_;
/**
* Starting offset to be used on index-based operations on `originalMessage_`.
*
* For example:
* originalMessage_ = "there"
* messageOffset_ = 4
* (the irc message is "hey there")
*
* then the index 6 would resolve to 6 - 4 = 2 => 'e'
*/
int messageOffset_ = 0;
QString userId_;
bool senderIsBroadcaster{};
Channel *channel = nullptr;
const Communi::IrcMessage *ircMessage;
MessageParseArgs args;
const QVariantMap tags;
QString originalMessage_;
const bool action_{};
QColor usernameColor_ = {153, 153, 153};
bool highlightAlert_ = false;
bool highlightSound_ = false;
std::optional<QUrl> highlightSoundCustomUrl_{};
}; };
} // namespace chatterino } // namespace chatterino

View file

@ -1,294 +0,0 @@
#include "messages/SharedMessageBuilder.hpp"
#include "Application.hpp"
#include "common/QLogging.hpp"
#include "controllers/highlights/HighlightController.hpp"
#include "controllers/ignores/IgnoreController.hpp"
#include "controllers/ignores/IgnorePhrase.hpp"
#include "controllers/nicknames/Nickname.hpp"
#include "controllers/sound/ISoundController.hpp"
#include "messages/Message.hpp"
#include "messages/MessageElement.hpp"
#include "providers/twitch/TwitchBadge.hpp"
#include "singletons/Settings.hpp"
#include "singletons/StreamerMode.hpp"
#include "singletons/WindowManager.hpp"
#include "util/Helpers.hpp"
#include <QFileInfo>
#include <optional>
namespace {
using namespace chatterino;
/**
* Gets the default sound url if the user set one,
* or the chatterino default ping sound if no url is set.
*/
QUrl getFallbackHighlightSound()
{
QString path = getSettings()->pathHighlightSound;
bool fileExists =
!path.isEmpty() && QFileInfo::exists(path) && QFileInfo(path).isFile();
if (fileExists)
{
return QUrl::fromLocalFile(path);
}
return QUrl("qrc:/sounds/ping2.wav");
}
} // namespace
namespace chatterino {
SharedMessageBuilder::SharedMessageBuilder(
Channel *_channel, const Communi::IrcPrivateMessage *_ircMessage,
const MessageParseArgs &_args)
: channel(_channel)
, ircMessage(_ircMessage)
, args(_args)
, tags(this->ircMessage->tags())
, originalMessage_(_ircMessage->content())
, action_(_ircMessage->isAction())
{
}
SharedMessageBuilder::SharedMessageBuilder(
Channel *_channel, const Communi::IrcMessage *_ircMessage,
const MessageParseArgs &_args, QString content, bool isAction)
: channel(_channel)
, ircMessage(_ircMessage)
, args(_args)
, tags(this->ircMessage->tags())
, originalMessage_(content)
, action_(isAction)
{
}
void SharedMessageBuilder::parse()
{
this->parseUsernameColor();
if (this->action_)
{
this->textColor_ = this->usernameColor_;
this->message().flags.set(MessageFlag::Action);
}
this->parseUsername();
this->message().flags.set(MessageFlag::Collapsed);
}
// "foo/bar/baz,tri/hard" can be a valid badge-info tag
// In that case, valid map content should be 'split by slash' only once:
// {"foo": "bar/baz", "tri": "hard"}
std::pair<QString, QString> SharedMessageBuilder::slashKeyValue(
const QString &kvStr)
{
return {
// part before first slash (index 0 of section)
kvStr.section('/', 0, 0),
// part after first slash (index 1 of section)
kvStr.section('/', 1, -1),
};
}
std::vector<Badge> SharedMessageBuilder::parseBadgeTag(const QVariantMap &tags)
{
std::vector<Badge> b;
auto badgesIt = tags.constFind("badges");
if (badgesIt == tags.end())
{
return b;
}
auto badges = badgesIt.value().toString().split(',', Qt::SkipEmptyParts);
for (const QString &badge : badges)
{
if (!badge.contains('/'))
{
continue;
}
auto pair = SharedMessageBuilder::slashKeyValue(badge);
b.emplace_back(Badge{pair.first, pair.second});
}
return b;
}
bool SharedMessageBuilder::isIgnored() const
{
return isIgnoredMessage({
/*.message = */ this->originalMessage_,
});
}
void SharedMessageBuilder::parseUsernameColor()
{
if (getSettings()->colorizeNicknames)
{
this->usernameColor_ = getRandomColor(this->ircMessage->nick());
}
}
void SharedMessageBuilder::parseUsername()
{
// username
this->userName = this->ircMessage->nick();
this->message().loginName = this->userName;
}
void SharedMessageBuilder::parseHighlights()
{
if (getSettings()->isBlacklistedUser(this->message().loginName))
{
// Do nothing. We ignore highlights from this user.
return;
}
auto badges = SharedMessageBuilder::parseBadgeTag(this->tags);
auto [highlighted, highlightResult] = getApp()->getHighlights()->check(
this->args, badges, this->message().loginName, this->originalMessage_,
this->message().flags);
if (!highlighted)
{
return;
}
// This message triggered one or more highlights, act upon the highlight result
this->message().flags.set(MessageFlag::Highlighted);
this->highlightAlert_ = highlightResult.alert;
this->highlightSound_ = highlightResult.playSound;
this->highlightSoundCustomUrl_ = highlightResult.customSoundUrl;
this->message().highlightColor = highlightResult.color;
if (highlightResult.showInMentions)
{
this->message().flags.set(MessageFlag::ShowInMentions);
}
}
void SharedMessageBuilder::appendChannelName()
{
QString channelName("#" + this->channel->getName());
Link link(Link::JumpToChannel, this->channel->getName());
this->emplace<TextElement>(channelName, MessageElementFlag::ChannelName,
MessageColor::System)
->setLink(link);
}
void SharedMessageBuilder::triggerHighlights()
{
SharedMessageBuilder::triggerHighlights(
this->channel->getName(), this->highlightSound_,
this->highlightSoundCustomUrl_, this->highlightAlert_);
}
void SharedMessageBuilder::triggerHighlights(
const QString &channelName, bool playSound,
const std::optional<QUrl> &customSoundUrl, bool windowAlert)
{
if (getApp()->getStreamerMode()->isEnabled() &&
getSettings()->streamerModeMuteMentions)
{
// We are in streamer mode with muting mention sounds enabled. Do nothing.
return;
}
if (getSettings()->isMutedChannel(channelName))
{
// Do nothing. Pings are muted in this channel.
return;
}
const bool hasFocus = (QApplication::focusWidget() != nullptr);
const bool resolveFocus =
!hasFocus || getSettings()->highlightAlwaysPlaySound;
if (playSound && resolveFocus)
{
// TODO(C++23): optional or_else
QUrl soundUrl;
if (customSoundUrl)
{
soundUrl = *customSoundUrl;
}
else
{
soundUrl = getFallbackHighlightSound();
}
getApp()->getSound()->play(soundUrl);
}
if (windowAlert)
{
getApp()->getWindows()->sendAlert();
}
}
QString SharedMessageBuilder::stylizeUsername(const QString &username,
const Message &message)
{
const QString &localizedName = message.localizedName;
bool hasLocalizedName = !localizedName.isEmpty();
// The full string that will be rendered in the chat widget
QString usernameText;
switch (getSettings()->usernameDisplayMode.getValue())
{
case UsernameDisplayMode::Username: {
usernameText = username;
}
break;
case UsernameDisplayMode::LocalizedName: {
if (hasLocalizedName)
{
usernameText = localizedName;
}
else
{
usernameText = username;
}
}
break;
default:
case UsernameDisplayMode::UsernameAndLocalizedName: {
if (hasLocalizedName)
{
usernameText = username + "(" + localizedName + ")";
}
else
{
usernameText = username;
}
}
break;
}
if (auto nicknameText = getSettings()->matchNickname(usernameText))
{
usernameText = *nicknameText;
}
return usernameText;
}
} // namespace chatterino

View file

@ -1,84 +0,0 @@
#pragma once
#include "common/Aliases.hpp"
#include "common/Outcome.hpp"
#include "messages/MessageBuilder.hpp"
#include <IrcMessage>
#include <QColor>
#include <QUrl>
#include <optional>
namespace chatterino {
class Badge;
class Channel;
class SharedMessageBuilder : public MessageBuilder
{
public:
SharedMessageBuilder() = delete;
explicit SharedMessageBuilder(Channel *_channel,
const Communi::IrcPrivateMessage *_ircMessage,
const MessageParseArgs &_args);
explicit SharedMessageBuilder(Channel *_channel,
const Communi::IrcMessage *_ircMessage,
const MessageParseArgs &_args,
QString content, bool isAction);
QString userName;
[[nodiscard]] virtual bool isIgnored() const;
// triggerHighlights triggers any alerts or sounds parsed by parseHighlights
virtual void triggerHighlights();
virtual MessagePtr build() = 0;
static std::pair<QString, QString> slashKeyValue(const QString &kvStr);
// Parses "badges" tag which contains a comma separated list of key-value elements
static std::vector<Badge> parseBadgeTag(const QVariantMap &tags);
static QString stylizeUsername(const QString &username,
const Message &message);
protected:
virtual void parse();
virtual void parseUsernameColor();
virtual void parseUsername();
virtual Outcome tryAppendEmote(const EmoteName &name)
{
(void)name;
return Failure;
}
// parseHighlights only updates the visual state of the message, but leaves the playing of alerts and sounds to the triggerHighlights function
virtual void parseHighlights();
static void triggerHighlights(const QString &channelName, bool playSound,
const std::optional<QUrl> &customSoundUrl,
bool windowAlert);
void appendChannelName();
Channel *channel;
const Communi::IrcMessage *ircMessage;
MessageParseArgs args;
const QVariantMap tags;
QString originalMessage_;
const bool action_{};
QColor usernameColor_ = {153, 153, 153};
bool highlightAlert_ = false;
bool highlightSound_ = false;
std::optional<QUrl> highlightSoundCustomUrl_{};
};
} // namespace chatterino

View file

@ -4,7 +4,6 @@
#include "common/network/NetworkResult.hpp" #include "common/network/NetworkResult.hpp"
#include "common/QLogging.hpp" #include "common/QLogging.hpp"
#include "providers/recentmessages/Impl.hpp" #include "providers/recentmessages/Impl.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "util/PostToThread.hpp" #include "util/PostToThread.hpp"
namespace { namespace {

View file

@ -2,9 +2,9 @@
#include "common/Env.hpp" #include "common/Env.hpp"
#include "common/QLogging.hpp" #include "common/QLogging.hpp"
#include "messages/MessageBuilder.hpp"
#include "providers/twitch/IrcMessageHandler.hpp" #include "providers/twitch/IrcMessageHandler.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "util/FormatTime.hpp" #include "util/FormatTime.hpp"
#include <QJsonArray> #include <QJsonArray>

View file

@ -20,7 +20,6 @@
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchHelpers.hpp" #include "providers/twitch/TwitchHelpers.hpp"
#include "providers/twitch/TwitchIrcServer.hpp" #include "providers/twitch/TwitchIrcServer.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Resources.hpp" #include "singletons/Resources.hpp"
#include "singletons/Settings.hpp" #include "singletons/Settings.hpp"
#include "singletons/StreamerMode.hpp" #include "singletons/StreamerMode.hpp"
@ -126,7 +125,7 @@ int stripLeadingReplyMention(const QVariantMap &tags, QString &content)
void updateReplyParticipatedStatus(const QVariantMap &tags, void updateReplyParticipatedStatus(const QVariantMap &tags,
const QString &senderLogin, const QString &senderLogin,
TwitchMessageBuilder &builder, MessageBuilder &builder,
std::shared_ptr<MessageThread> &thread, std::shared_ptr<MessageThread> &thread,
bool isNew) bool isNew)
{ {
@ -245,7 +244,7 @@ QMap<QString, QString> parseBadges(const QString &badgesString)
void populateReply(TwitchChannel *channel, Communi::IrcMessage *message, void populateReply(TwitchChannel *channel, Communi::IrcMessage *message,
const std::vector<MessagePtr> &otherLoaded, const std::vector<MessagePtr> &otherLoaded,
TwitchMessageBuilder &builder) MessageBuilder &builder)
{ {
const auto &tags = message->tags(); const auto &tags = message->tags();
if (const auto it = tags.find("reply-thread-parent-msg-id"); if (const auto it = tags.find("reply-thread-parent-msg-id");
@ -481,8 +480,7 @@ std::vector<MessagePtr> parseUserNoticeMessage(Channel *channel,
MessageParseArgs args; MessageParseArgs args;
args.trimSubscriberUsername = true; args.trimSubscriberUsername = true;
TwitchMessageBuilder builder(channel, message, args, content, MessageBuilder builder(channel, message, args, content, false);
false);
builder->flags.set(MessageFlag::Subscription); builder->flags.set(MessageFlag::Subscription);
builder->flags.unset(MessageFlag::Highlighted); builder->flags.unset(MessageFlag::Highlighted);
builtMessages.emplace_back(builder.build()); builtMessages.emplace_back(builder.build());
@ -566,7 +564,7 @@ std::vector<MessagePtr> parsePrivMessage(Channel *channel,
std::vector<MessagePtr> builtMessages; std::vector<MessagePtr> builtMessages;
MessageParseArgs args; MessageParseArgs args;
TwitchMessageBuilder builder(channel, message, args, message->content(), MessageBuilder builder(channel, message, args, message->content(),
message->isAction()); message->isAction());
if (!builder.isIgnored()) if (!builder.isIgnored())
{ {
@ -576,7 +574,7 @@ std::vector<MessagePtr> parsePrivMessage(Channel *channel,
if (message->tags().contains(u"pinned-chat-paid-amount"_s)) if (message->tags().contains(u"pinned-chat-paid-amount"_s))
{ {
auto ptr = TwitchMessageBuilder::buildHypeChatMessage(message); auto ptr = MessageBuilder::buildHypeChatMessage(message);
if (ptr) if (ptr)
{ {
builtMessages.emplace_back(std::move(ptr)); builtMessages.emplace_back(std::move(ptr));
@ -618,7 +616,7 @@ std::vector<MessagePtr> IrcMessageHandler::parseMessageWithReply(
QString content = privMsg->content(); QString content = privMsg->content();
int messageOffset = stripLeadingReplyMention(privMsg->tags(), content); int messageOffset = stripLeadingReplyMention(privMsg->tags(), content);
MessageParseArgs args; MessageParseArgs args;
TwitchMessageBuilder builder(channel, message, args, content, MessageBuilder builder(channel, message, args, content,
privMsg->isAction()); privMsg->isAction());
builder.setMessageOffset(messageOffset); builder.setMessageOffset(messageOffset);
@ -716,7 +714,7 @@ void IrcMessageHandler::handlePrivMessage(Communi::IrcPrivateMessage *message,
if (message->tags().contains(u"pinned-chat-paid-amount"_s)) if (message->tags().contains(u"pinned-chat-paid-amount"_s))
{ {
auto ptr = TwitchMessageBuilder::buildHypeChatMessage(message); auto ptr = MessageBuilder::buildHypeChatMessage(message);
if (ptr) if (ptr)
{ {
chan->addMessage(ptr, MessageContext::Original); chan->addMessage(ptr, MessageContext::Original);
@ -865,9 +863,8 @@ void IrcMessageHandler::handleClearMessageMessage(Communi::IrcMessage *message)
msg->flags.set(MessageFlag::Disabled); msg->flags.set(MessageFlag::Disabled);
if (!getSettings()->hideDeletionActions) if (!getSettings()->hideDeletionActions)
{ {
MessageBuilder builder; chan->addMessage(MessageBuilder::makeDeletionMessageFromIRC(msg),
TwitchMessageBuilder::deletionMessage(msg, &builder); MessageContext::Original);
chan->addMessage(builder.release(), MessageContext::Original);
} }
} }
@ -947,7 +944,7 @@ void IrcMessageHandler::handleWhisperMessage(Communi::IrcMessage *ircMessage)
auto *c = getApp()->getTwitch()->getWhispersChannel().get(); auto *c = getApp()->getTwitch()->getWhispersChannel().get();
TwitchMessageBuilder builder( MessageBuilder builder(
c, ircMessage, args, c, ircMessage, args,
ircMessage->parameter(1).replace(COMBINED_FIXER, ZERO_WIDTH_JOINER), ircMessage->parameter(1).replace(COMBINED_FIXER, ZERO_WIDTH_JOINER),
false); false);
@ -1163,10 +1160,9 @@ void IrcMessageHandler::handleNoticeMessage(Communi::IrcNoticeMessage *message)
{ {
hostedChannelName.chop(1); hostedChannelName.chop(1);
} }
MessageBuilder builder; channel->addMessage(MessageBuilder::makeHostingSystemMessage(
TwitchMessageBuilder::hostingSystemMessage(hostedChannelName, hostedChannelName, hostOn),
&builder, hostOn); MessageContext::Original);
channel->addMessage(builder.release(), MessageContext::Original);
} }
else if (tags == "room_mods" || tags == "vips_success") else if (tags == "room_mods" || tags == "vips_success")
{ {
@ -1193,9 +1189,9 @@ void IrcMessageHandler::handleNoticeMessage(Communi::IrcNoticeMessage *message)
.mid(1) // there is a space before the first user .mid(1) // there is a space before the first user
.split(", "); .split(", ");
users.sort(Qt::CaseInsensitive); users.sort(Qt::CaseInsensitive);
TwitchMessageBuilder::listOfUsersSystemMessage(msgParts.at(0), channel->addMessage(MessageBuilder::makeListOfUsersMessage(
users, tc, &builder); msgParts.at(0), users, tc),
channel->addMessage(builder.release(), MessageContext::Original); MessageContext::Original);
} }
else else
{ {
@ -1367,7 +1363,7 @@ void IrcMessageHandler::addMessage(Communi::IrcMessage *message,
QString content = originalContent; QString content = originalContent;
int messageOffset = stripLeadingReplyMention(tags, content); int messageOffset = stripLeadingReplyMention(tags, content);
TwitchMessageBuilder builder(channel, message, args, content, isAction); MessageBuilder builder(channel, message, args, content, isAction);
builder.setMessageOffset(messageOffset); builder.setMessageOffset(messageOffset);
if (const auto it = tags.find("reply-thread-parent-msg-id"); if (const auto it = tags.find("reply-thread-parent-msg-id");

View file

@ -12,6 +12,7 @@
#include "messages/Image.hpp" #include "messages/Image.hpp"
#include "messages/Link.hpp" #include "messages/Link.hpp"
#include "messages/Message.hpp" #include "messages/Message.hpp"
#include "messages/MessageBuilder.hpp"
#include "messages/MessageElement.hpp" #include "messages/MessageElement.hpp"
#include "messages/MessageThread.hpp" #include "messages/MessageThread.hpp"
#include "providers/bttv/BttvEmotes.hpp" #include "providers/bttv/BttvEmotes.hpp"
@ -31,7 +32,6 @@
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchCommon.hpp" #include "providers/twitch/TwitchCommon.hpp"
#include "providers/twitch/TwitchIrcServer.hpp" #include "providers/twitch/TwitchIrcServer.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Emotes.hpp" #include "singletons/Emotes.hpp"
#include "singletons/Settings.hpp" #include "singletons/Settings.hpp"
#include "singletons/StreamerMode.hpp" #include "singletons/StreamerMode.hpp"
@ -311,10 +311,9 @@ void TwitchChannel::addChannelPointReward(const ChannelPointReward &reward)
if (!reward.isUserInputRequired) if (!reward.isUserInputRequired)
{ {
MessageBuilder builder; this->addMessage(MessageBuilder::makeChannelPointRewardMessage(
TwitchMessageBuilder::appendChannelPointRewardMessage( reward, this->isMod(), this->isBroadcaster()),
reward, &builder, this->isMod(), this->isBroadcaster()); MessageContext::Original);
this->addMessage(builder.release(), MessageContext::Original);
return; return;
} }
@ -434,11 +433,11 @@ void TwitchChannel::onLiveStatusChanged(bool isLive, bool isInitialUpdate)
}); });
// Channel live message // Channel live message
MessageBuilder builder; this->addMessage(
TwitchMessageBuilder::liveSystemMessage(this->getDisplayName(), MessageBuilder::makeLiveMessage(
&builder); this->getDisplayName(), this->roomId(),
builder.message().id = this->roomId(); {MessageFlag::System, MessageFlag::DoNotTriggerNotification}),
this->addMessage(builder.release(), MessageContext::Original); MessageContext::Original);
} }
else else
{ {
@ -446,10 +445,9 @@ void TwitchChannel::onLiveStatusChanged(bool isLive, bool isInitialUpdate)
<< "[TwitchChannel " << this->getName() << "] Offline"; << "[TwitchChannel " << this->getName() << "] Offline";
// Channel offline message // Channel offline message
MessageBuilder builder; this->addMessage(MessageBuilder::makeOfflineSystemMessage(
TwitchMessageBuilder::offlineSystemMessage(this->getDisplayName(), this->getDisplayName(), this->roomId()),
&builder); MessageContext::Original);
this->addMessage(builder.release(), MessageContext::Original);
getApp()->getNotifications()->notifyTwitchChannelOffline( getApp()->getNotifications()->notifyTwitchChannelOffline(
this->roomId()); this->roomId());
@ -1077,20 +1075,28 @@ bool TwitchChannel::tryReplaceLastLiveUpdateAddOrRemove(
// Update the message // Update the message
this->lastLiveUpdateEmoteNames_.push_back(emoteName); this->lastLiveUpdateEmoteNames_.push_back(emoteName);
MessageBuilder replacement; auto makeReplacement = [&](MessageFlag op) -> MessageBuilder {
if (op == MessageFlag::LiveUpdatesAdd) if (op == MessageFlag::LiveUpdatesAdd)
{ {
replacement = return {
MessageBuilder(liveUpdatesAddEmoteMessage, platform, liveUpdatesAddEmoteMessage,
last->loginName, this->lastLiveUpdateEmoteNames_); platform,
} last->loginName,
else // op == RemoveEmoteMessage this->lastLiveUpdateEmoteNames_,
{ };
replacement =
MessageBuilder(liveUpdatesRemoveEmoteMessage, platform,
last->loginName, this->lastLiveUpdateEmoteNames_);
} }
// op == RemoveEmoteMessage
return {
liveUpdatesRemoveEmoteMessage,
platform,
last->loginName,
this->lastLiveUpdateEmoteNames_,
};
};
auto replacement = makeReplacement(op);
replacement->flags = last->flags; replacement->flags = last->flags;
auto msg = replacement.release(); auto msg = replacement.release();

View file

@ -458,7 +458,7 @@ private:
std::vector<boost::signals2::scoped_connection> bSignals_; std::vector<boost::signals2::scoped_connection> bSignals_;
friend class TwitchIrcServer; friend class TwitchIrcServer;
friend class TwitchMessageBuilder; friend class MessageBuilder;
friend class IrcMessageHandler; friend class IrcMessageHandler;
friend class Commands_E2E_Test; friend class Commands_E2E_Test;
}; };

File diff suppressed because it is too large Load diff

View file

@ -1,166 +0,0 @@
#pragma once
#include "common/Aliases.hpp"
#include "common/Outcome.hpp"
#include "messages/SharedMessageBuilder.hpp"
#include "pubsubmessages/LowTrustUsers.hpp"
#include <IrcMessage>
#include <QString>
#include <QVariant>
#include <optional>
#include <unordered_map>
namespace chatterino {
struct Emote;
using EmotePtr = std::shared_ptr<const Emote>;
class Channel;
class TwitchChannel;
class MessageThread;
class IgnorePhrase;
struct HelixVip;
using HelixModerator = HelixVip;
struct ChannelPointReward;
struct DeleteAction;
struct TwitchEmoteOccurrence {
int start;
int end;
EmotePtr ptr;
EmoteName name;
bool operator==(const TwitchEmoteOccurrence &other) const
{
return std::tie(this->start, this->end, this->ptr, this->name) ==
std::tie(other.start, other.end, other.ptr, other.name);
}
};
class TwitchMessageBuilder : public SharedMessageBuilder
{
public:
TwitchMessageBuilder() = delete;
explicit TwitchMessageBuilder(Channel *_channel,
const Communi::IrcPrivateMessage *_ircMessage,
const MessageParseArgs &_args);
explicit TwitchMessageBuilder(Channel *_channel,
const Communi::IrcMessage *_ircMessage,
const MessageParseArgs &_args,
QString content, bool isAction);
TwitchChannel *twitchChannel;
[[nodiscard]] bool isIgnored() const override;
bool isIgnoredReply() const;
void triggerHighlights() override;
MessagePtr build() override;
void setThread(std::shared_ptr<MessageThread> thread);
void setParent(MessagePtr parent);
void setMessageOffset(int offset);
static void appendChannelPointRewardMessage(
const ChannelPointReward &reward, MessageBuilder *builder, bool isMod,
bool isBroadcaster);
// Message in the /live chat for channel going live
static void liveMessage(const QString &channelName,
MessageBuilder *builder);
// Messages in normal chat for channel stuff
static void liveSystemMessage(const QString &channelName,
MessageBuilder *builder);
static void offlineSystemMessage(const QString &channelName,
MessageBuilder *builder);
static void hostingSystemMessage(const QString &channelName,
MessageBuilder *builder, bool hostOn);
static void deletionMessage(const MessagePtr originalMessage,
MessageBuilder *builder);
static void deletionMessage(const DeleteAction &action,
MessageBuilder *builder);
static void listOfUsersSystemMessage(QString prefix, QStringList users,
Channel *channel,
MessageBuilder *builder);
static void listOfUsersSystemMessage(
QString prefix, const std::vector<HelixModerator> &users,
Channel *channel, MessageBuilder *builder);
static MessagePtr buildHypeChatMessage(Communi::IrcPrivateMessage *message);
static std::pair<MessagePtr, MessagePtr> makeAutomodMessage(
const AutomodAction &action, const QString &channelName);
static MessagePtr makeAutomodInfoMessage(const AutomodInfoAction &action);
static std::pair<MessagePtr, MessagePtr> makeLowTrustUserMessage(
const PubSubLowTrustUsersMessage &action, const QString &channelName,
const TwitchChannel *twitchChannel);
static MessagePtr makeLowTrustUpdateMessage(
const PubSubLowTrustUsersMessage &action);
// Shares some common logic from SharedMessageBuilder::parseBadgeTag
static std::unordered_map<QString, QString> parseBadgeInfoTag(
const QVariantMap &tags);
static std::vector<TwitchEmoteOccurrence> parseTwitchEmotes(
const QVariantMap &tags, const QString &originalMessage,
int messageOffset);
static void processIgnorePhrases(
const std::vector<IgnorePhrase> &phrases, QString &originalMessage,
std::vector<TwitchEmoteOccurrence> &twitchEmotes);
private:
void parseUsernameColor() override;
void parseUsername() override;
void parseMessageID();
void parseRoomID();
// Parse & build thread information into the message
// Will read information from thread_ or from IRC tags
void parseThread();
void appendUsername();
Outcome tryAppendEmote(const EmoteName &name) override;
void addWords(const QStringList &words,
const std::vector<TwitchEmoteOccurrence> &twitchEmotes);
void addTextOrEmoji(EmotePtr emote) override;
void addTextOrEmoji(const QString &value) override;
void appendTwitchBadges();
void appendChatterinoBadges();
void appendFfzBadges();
void appendSeventvBadges();
Outcome tryParseCheermote(const QString &string);
bool shouldAddModerationElements() const;
QString roomID_;
bool hasBits_ = false;
QString bits;
int bitsLeft{};
bool bitsStacked = false;
bool historicalMessage_ = false;
std::shared_ptr<MessageThread> thread_;
MessagePtr parent_;
/**
* Starting offset to be used on index-based operations on `originalMessage_`.
*
* For example:
* originalMessage_ = "there"
* messageOffset_ = 4
* (the irc message is "hey there")
*
* then the index 6 would resolve to 6 - 4 = 2 => 'e'
*/
int messageOffset_ = 0;
QString userId_;
bool senderIsBroadcaster{};
};
} // namespace chatterino

View file

@ -7,7 +7,6 @@
#include "common/QLogging.hpp" #include "common/QLogging.hpp"
#include "debug/Benchmark.hpp" #include "debug/Benchmark.hpp"
#include "messages/MessageBuilder.hpp" #include "messages/MessageBuilder.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Paths.hpp" #include "singletons/Paths.hpp"
#include "singletons/Settings.hpp" #include "singletons/Settings.hpp"
#include "util/CombinePath.hpp" #include "util/CombinePath.hpp"

View file

@ -6,6 +6,7 @@
#include <cmath> #include <cmath>
#include <optional> #include <optional>
#include <utility>
#include <vector> #include <vector>
namespace chatterino { namespace chatterino {

View file

@ -97,4 +97,17 @@ inline QDateTime calculateMessageTime(const Communi::IrcMessage *message)
return QDateTime::currentDateTime(); return QDateTime::currentDateTime();
} }
// "foo/bar/baz,tri/hard" can be a valid badge-info tag
// In that case, valid map content should be 'split by slash' only once:
// {"foo": "bar/baz", "tri": "hard"}
inline std::pair<QString, QString> slashKeyValue(const QString &kvStr)
{
return {
// part before first slash (index 0 of section)
kvStr.section('/', 0, 0),
// part after first slash (index 1 of section)
kvStr.section('/', 1, -1),
};
}
} // namespace chatterino } // namespace chatterino

View file

@ -15,7 +15,6 @@
#include "providers/twitch/TwitchAccount.hpp" #include "providers/twitch/TwitchAccount.hpp"
#include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchIrcServer.hpp" #include "providers/twitch/TwitchIrcServer.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Fonts.hpp" #include "singletons/Fonts.hpp"
#include "singletons/ImageUploader.hpp" #include "singletons/ImageUploader.hpp"
#include "singletons/Settings.hpp" #include "singletons/Settings.hpp"

View file

@ -22,7 +22,7 @@ set(test_SOURCES
${CMAKE_CURRENT_LIST_DIR}/src/UtilTwitch.cpp ${CMAKE_CURRENT_LIST_DIR}/src/UtilTwitch.cpp
${CMAKE_CURRENT_LIST_DIR}/src/IrcHelpers.cpp ${CMAKE_CURRENT_LIST_DIR}/src/IrcHelpers.cpp
${CMAKE_CURRENT_LIST_DIR}/src/TwitchPubSubClient.cpp ${CMAKE_CURRENT_LIST_DIR}/src/TwitchPubSubClient.cpp
${CMAKE_CURRENT_LIST_DIR}/src/TwitchMessageBuilder.cpp ${CMAKE_CURRENT_LIST_DIR}/src/MessageBuilder.cpp
${CMAKE_CURRENT_LIST_DIR}/src/HighlightController.cpp ${CMAKE_CURRENT_LIST_DIR}/src/HighlightController.cpp
${CMAKE_CURRENT_LIST_DIR}/src/FormatTime.cpp ${CMAKE_CURRENT_LIST_DIR}/src/FormatTime.cpp
${CMAKE_CURRENT_LIST_DIR}/src/LimitedQueue.cpp ${CMAKE_CURRENT_LIST_DIR}/src/LimitedQueue.cpp

View file

@ -3,6 +3,7 @@
#include "controllers/filters/lang/Filter.hpp" #include "controllers/filters/lang/Filter.hpp"
#include "controllers/filters/lang/Types.hpp" #include "controllers/filters/lang/Types.hpp"
#include "controllers/highlights/HighlightController.hpp" #include "controllers/highlights/HighlightController.hpp"
#include "messages/MessageBuilder.hpp"
#include "mocks/Channel.hpp" #include "mocks/Channel.hpp"
#include "mocks/ChatterinoBadges.hpp" #include "mocks/ChatterinoBadges.hpp"
#include "mocks/EmptyApplication.hpp" #include "mocks/EmptyApplication.hpp"
@ -11,7 +12,6 @@
#include "providers/ffz/FfzBadges.hpp" #include "providers/ffz/FfzBadges.hpp"
#include "providers/seventv/SeventvBadges.hpp" #include "providers/seventv/SeventvBadges.hpp"
#include "providers/twitch/TwitchBadge.hpp" #include "providers/twitch/TwitchBadge.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Emotes.hpp" #include "singletons/Emotes.hpp"
#include "Test.hpp" #include "Test.hpp"
@ -280,7 +280,7 @@ TEST_F(FiltersF, TypingContextChecks)
QString originalMessage = privmsg->content(); QString originalMessage = privmsg->content();
TwitchMessageBuilder builder(&channel, privmsg, MessageParseArgs{}); MessageBuilder builder(&channel, privmsg, MessageParseArgs{});
auto msg = builder.build(); auto msg = builder.build();
EXPECT_NE(msg.get(), nullptr); EXPECT_NE(msg.get(), nullptr);

View file

@ -1,10 +1,8 @@
#include "providers/twitch/TwitchMessageBuilder.hpp" #include "messages/MessageBuilder.hpp"
#include "common/Channel.hpp"
#include "controllers/accounts/AccountController.hpp" #include "controllers/accounts/AccountController.hpp"
#include "controllers/highlights/HighlightController.hpp" #include "controllers/highlights/HighlightController.hpp"
#include "controllers/ignores/IgnorePhrase.hpp" #include "controllers/ignores/IgnorePhrase.hpp"
#include "messages/MessageBuilder.hpp"
#include "mocks/Channel.hpp" #include "mocks/Channel.hpp"
#include "mocks/ChatterinoBadges.hpp" #include "mocks/ChatterinoBadges.hpp"
#include "mocks/DisabledStreamerMode.hpp" #include "mocks/DisabledStreamerMode.hpp"
@ -16,6 +14,7 @@
#include "providers/twitch/TwitchBadge.hpp" #include "providers/twitch/TwitchBadge.hpp"
#include "singletons/Emotes.hpp" #include "singletons/Emotes.hpp"
#include "Test.hpp" #include "Test.hpp"
#include "util/IrcHelpers.hpp"
#include <IrcConnection> #include <IrcConnection>
#include <QDebug> #include <QDebug>
@ -115,7 +114,7 @@ public:
} // namespace } // namespace
TEST(TwitchMessageBuilder, CommaSeparatedListTagParsing) TEST(MessageBuilder, CommaSeparatedListTagParsing)
{ {
struct TestCase { struct TestCase {
QString input; QString input;
@ -151,14 +150,14 @@ TEST(TwitchMessageBuilder, CommaSeparatedListTagParsing)
for (const auto &test : testCases) for (const auto &test : testCases)
{ {
auto output = TwitchMessageBuilder::slashKeyValue(test.input); auto output = slashKeyValue(test.input);
EXPECT_EQ(output, test.expectedOutput) EXPECT_EQ(output, test.expectedOutput)
<< "Input " << test.input << " failed"; << "Input " << test.input << " failed";
} }
} }
class TestTwitchMessageBuilder : public ::testing::Test class TestMessageBuilder : public ::testing::Test
{ {
protected: protected:
void SetUp() override void SetUp() override
@ -174,7 +173,7 @@ protected:
std::unique_ptr<MockApplication> mockApplication; std::unique_ptr<MockApplication> mockApplication;
}; };
TEST(TwitchMessageBuilder, BadgeInfoParsing) TEST(MessageBuilder, BadgeInfoParsing)
{ {
struct TestCase { struct TestCase {
QByteArray input; QByteArray input;
@ -235,12 +234,11 @@ TEST(TwitchMessageBuilder, BadgeInfoParsing)
Communi::IrcPrivateMessage::fromData(test.input, nullptr); Communi::IrcPrivateMessage::fromData(test.input, nullptr);
auto outputBadgeInfo = auto outputBadgeInfo =
TwitchMessageBuilder::parseBadgeInfoTag(privmsg->tags()); MessageBuilder::parseBadgeInfoTag(privmsg->tags());
EXPECT_EQ(outputBadgeInfo, test.expectedBadgeInfo) EXPECT_EQ(outputBadgeInfo, test.expectedBadgeInfo)
<< "Input for badgeInfo " << test.input << " failed"; << "Input for badgeInfo " << test.input << " failed";
auto outputBadges = auto outputBadges = MessageBuilder::parseBadgeTag(privmsg->tags());
SharedMessageBuilder::parseBadgeTag(privmsg->tags());
EXPECT_EQ(outputBadges, test.expectedBadges) EXPECT_EQ(outputBadges, test.expectedBadges)
<< "Input for badges " << test.input << " failed"; << "Input for badges " << test.input << " failed";
@ -248,7 +246,7 @@ TEST(TwitchMessageBuilder, BadgeInfoParsing)
} }
} }
TEST_F(TestTwitchMessageBuilder, ParseTwitchEmotes) TEST_F(TestMessageBuilder, ParseTwitchEmotes)
{ {
struct TestCase { struct TestCase {
QByteArray input; QByteArray input;
@ -416,7 +414,7 @@ TEST_F(TestTwitchMessageBuilder, ParseTwitchEmotes)
QString originalMessage = privmsg->content(); QString originalMessage = privmsg->content();
// TODO: Add tests with replies // TODO: Add tests with replies
auto actualTwitchEmotes = TwitchMessageBuilder::parseTwitchEmotes( auto actualTwitchEmotes = MessageBuilder::parseTwitchEmotes(
privmsg->tags(), originalMessage, 0); privmsg->tags(), originalMessage, 0);
EXPECT_EQ(actualTwitchEmotes, test.expectedTwitchEmotes) EXPECT_EQ(actualTwitchEmotes, test.expectedTwitchEmotes)
@ -426,7 +424,7 @@ TEST_F(TestTwitchMessageBuilder, ParseTwitchEmotes)
} }
} }
TEST_F(TestTwitchMessageBuilder, ParseMessage) TEST_F(TestMessageBuilder, ParseMessage)
{ {
MockChannel channel("pajlada"); MockChannel channel("pajlada");
@ -484,7 +482,7 @@ TEST_F(TestTwitchMessageBuilder, ParseMessage)
QString originalMessage = privmsg->content(); QString originalMessage = privmsg->content();
TwitchMessageBuilder builder(&channel, privmsg, MessageParseArgs{}); MessageBuilder builder(&channel, privmsg, MessageParseArgs{});
auto msg = builder.build(); auto msg = builder.build();
EXPECT_NE(msg.get(), nullptr); EXPECT_NE(msg.get(), nullptr);
@ -493,7 +491,7 @@ TEST_F(TestTwitchMessageBuilder, ParseMessage)
} }
} }
TEST_F(TestTwitchMessageBuilder, IgnoresReplace) TEST_F(TestMessageBuilder, IgnoresReplace)
{ {
struct TestCase { struct TestCase {
std::vector<IgnorePhrase> phrases; std::vector<IgnorePhrase> phrases;
@ -619,8 +617,7 @@ TEST_F(TestTwitchMessageBuilder, IgnoresReplace)
{ {
auto message = test.input; auto message = test.input;
auto emotes = test.twitchEmotes; auto emotes = test.twitchEmotes;
TwitchMessageBuilder::processIgnorePhrases(test.phrases, message, MessageBuilder::processIgnorePhrases(test.phrases, message, emotes);
emotes);
EXPECT_EQ(message, test.expectedMessage) EXPECT_EQ(message, test.expectedMessage)
<< "Message not equal for input '" << test.input << "Message not equal for input '" << test.input