2020-10-18 15:54:48 +02:00
|
|
|
#include "IrcMessageHandler.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
#include "Application.hpp"
|
2020-11-21 16:20:10 +01:00
|
|
|
#include "common/QLogging.hpp"
|
2019-09-18 13:03:16 +02:00
|
|
|
#include "controllers/accounts/AccountController.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
#include "messages/LimitedQueue.hpp"
|
|
|
|
#include "messages/Message.hpp"
|
2019-09-18 13:03:16 +02:00
|
|
|
#include "providers/twitch/TwitchAccountManager.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
#include "providers/twitch/TwitchChannel.hpp"
|
|
|
|
#include "providers/twitch/TwitchHelpers.hpp"
|
2019-09-18 13:03:16 +02:00
|
|
|
#include "providers/twitch/TwitchIrcServer.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
#include "providers/twitch/TwitchMessageBuilder.hpp"
|
|
|
|
#include "singletons/Resources.hpp"
|
|
|
|
#include "singletons/Settings.hpp"
|
|
|
|
#include "singletons/WindowManager.hpp"
|
2021-01-31 13:55:44 +01:00
|
|
|
#include "util/FormatTime.hpp"
|
2021-04-11 14:17:21 +02:00
|
|
|
#include "util/Helpers.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
#include "util/IrcHelpers.hpp"
|
|
|
|
|
|
|
|
#include <IrcMessage>
|
|
|
|
|
|
|
|
#include <unordered_set>
|
|
|
|
|
2020-12-12 14:19:51 +01:00
|
|
|
namespace {
|
|
|
|
using namespace chatterino;
|
2021-05-01 17:42:51 +02:00
|
|
|
|
|
|
|
// Message types below are the ones that might contain special user's message on USERNOTICE
|
|
|
|
static const QSet<QString> specialMessageTypes{
|
|
|
|
"sub", //
|
|
|
|
"subgift", //
|
|
|
|
"resub", // resub messages
|
|
|
|
"bitsbadgetier", // bits badge upgrade
|
|
|
|
"ritual", // new viewer ritual
|
|
|
|
};
|
|
|
|
|
2020-12-12 14:19:51 +01:00
|
|
|
MessagePtr generateBannedMessage(bool confirmedBan)
|
|
|
|
{
|
|
|
|
const auto linkColor = MessageColor(MessageColor::Link);
|
|
|
|
const auto accountsLink = Link(Link::Reconnect, QString());
|
|
|
|
const auto bannedText =
|
|
|
|
confirmedBan
|
|
|
|
? QString("You were banned from this channel!")
|
|
|
|
: QString(
|
|
|
|
"Your connection to this channel was unexpectedly dropped.");
|
|
|
|
|
|
|
|
const auto reconnectPromptText =
|
|
|
|
confirmedBan
|
|
|
|
? QString(
|
|
|
|
"If you believe you have been unbanned, try reconnecting.")
|
|
|
|
: QString("Try reconnecting.");
|
|
|
|
|
|
|
|
MessageBuilder builder;
|
|
|
|
builder.message().flags.set(MessageFlag::System);
|
|
|
|
|
|
|
|
builder.emplace<TimestampElement>();
|
|
|
|
builder.emplace<TextElement>(bannedText, MessageElementFlag::Text,
|
|
|
|
MessageColor::System);
|
|
|
|
builder
|
|
|
|
.emplace<TextElement>(reconnectPromptText, MessageElementFlag::Text,
|
|
|
|
linkColor)
|
|
|
|
->setLink(accountsLink);
|
|
|
|
|
|
|
|
return builder.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
2019-09-08 22:27:57 +02:00
|
|
|
namespace chatterino {
|
|
|
|
|
2020-02-02 14:31:37 +01:00
|
|
|
static float relativeSimilarity(const QString &str1, const QString &str2)
|
|
|
|
{
|
|
|
|
// Longest Common Substring Problem
|
|
|
|
std::vector<std::vector<int>> tree(str1.size(),
|
|
|
|
std::vector<int>(str2.size(), 0));
|
|
|
|
int z = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < str1.size(); ++i)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < str2.size(); ++j)
|
|
|
|
{
|
|
|
|
if (str1[i] == str2[j])
|
|
|
|
{
|
|
|
|
if (i == 0 || j == 0)
|
|
|
|
{
|
|
|
|
tree[i][j] = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tree[i][j] = tree[i - 1][j - 1] + 1;
|
|
|
|
}
|
|
|
|
if (tree[i][j] > z)
|
|
|
|
{
|
|
|
|
z = tree[i][j];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tree[i][j] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 21:05:40 +02:00
|
|
|
// ensure that no div by 0
|
|
|
|
return z == 0 ? 0.f
|
|
|
|
: float(z) /
|
|
|
|
std::max<int>(1, std::max(str1.size(), str2.size()));
|
2020-02-02 14:31:37 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
float IrcMessageHandler::similarity(
|
|
|
|
MessagePtr msg, const LimitedQueueSnapshot<MessagePtr> &messages)
|
|
|
|
{
|
|
|
|
float similarityPercent = 0.0f;
|
|
|
|
int bySameUser = 0;
|
|
|
|
for (int i = 1; bySameUser < getSettings()->hideSimilarMaxMessagesToCheck;
|
|
|
|
++i)
|
|
|
|
{
|
|
|
|
if (messages.size() < i)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
const auto &prevMsg = messages[messages.size() - i];
|
|
|
|
if (prevMsg->parseTime.secsTo(QTime::currentTime()) >=
|
|
|
|
getSettings()->hideSimilarMaxDelay)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (msg->loginName != prevMsg->loginName)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
++bySameUser;
|
|
|
|
similarityPercent = std::max(
|
|
|
|
similarityPercent,
|
|
|
|
relativeSimilarity(msg->messageText, prevMsg->messageText));
|
|
|
|
}
|
|
|
|
return similarityPercent;
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::setSimilarityFlags(MessagePtr msg, ChannelPtr chan)
|
|
|
|
{
|
|
|
|
if (getSettings()->similarityEnabled)
|
|
|
|
{
|
|
|
|
bool isMyself = msg->loginName ==
|
|
|
|
getApp()->accounts->twitch.getCurrent()->getUserName();
|
|
|
|
bool hideMyself = getSettings()->hideSimilarMyself;
|
|
|
|
|
|
|
|
if (isMyself && !hideMyself)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IrcMessageHandler::similarity(msg, chan->getMessageSnapshot()) >
|
|
|
|
getSettings()->similarityPercentage)
|
|
|
|
{
|
|
|
|
msg->flags.set(MessageFlag::Similar, true);
|
|
|
|
if (getSettings()->colorSimilarDisabled)
|
|
|
|
{
|
|
|
|
msg->flags.set(MessageFlag::Disabled, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-08 22:27:57 +02:00
|
|
|
static QMap<QString, QString> parseBadges(QString badgesString)
|
|
|
|
{
|
|
|
|
QMap<QString, QString> badges;
|
|
|
|
|
|
|
|
for (auto badgeData : badgesString.split(','))
|
|
|
|
{
|
|
|
|
auto parts = badgeData.split('/');
|
|
|
|
if (parts.length() != 2)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
badges.insert(parts[0], parts[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return badges;
|
|
|
|
}
|
|
|
|
|
2019-10-07 22:42:34 +02:00
|
|
|
IrcMessageHandler &IrcMessageHandler::instance()
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
|
|
|
static IrcMessageHandler instance;
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<MessagePtr> IrcMessageHandler::parseMessage(
|
|
|
|
Channel *channel, Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
std::vector<MessagePtr> builtMessages;
|
|
|
|
|
|
|
|
auto command = message->command();
|
|
|
|
|
|
|
|
if (command == "PRIVMSG")
|
|
|
|
{
|
|
|
|
return this->parsePrivMessage(
|
|
|
|
channel, static_cast<Communi::IrcPrivateMessage *>(message));
|
|
|
|
}
|
|
|
|
else if (command == "USERNOTICE")
|
|
|
|
{
|
|
|
|
return this->parseUserNoticeMessage(channel, message);
|
|
|
|
}
|
|
|
|
else if (command == "NOTICE")
|
|
|
|
{
|
|
|
|
return this->parseNoticeMessage(
|
|
|
|
static_cast<Communi::IrcNoticeMessage *>(message));
|
|
|
|
}
|
|
|
|
|
|
|
|
return builtMessages;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<MessagePtr> IrcMessageHandler::parsePrivMessage(
|
|
|
|
Channel *channel, Communi::IrcPrivateMessage *message)
|
|
|
|
{
|
|
|
|
std::vector<MessagePtr> builtMessages;
|
|
|
|
MessageParseArgs args;
|
|
|
|
TwitchMessageBuilder builder(channel, message, args, message->content(),
|
|
|
|
message->isAction());
|
|
|
|
if (!builder.isIgnored())
|
|
|
|
{
|
|
|
|
builtMessages.emplace_back(builder.build());
|
|
|
|
builder.triggerHighlights();
|
|
|
|
}
|
|
|
|
return builtMessages;
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handlePrivMessage(Communi::IrcPrivateMessage *message,
|
2019-09-18 13:03:16 +02:00
|
|
|
TwitchIrcServer &server)
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
|
|
|
this->addMessage(message, message->target(), message->content(), server,
|
|
|
|
false, message->isAction());
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::addMessage(Communi::IrcMessage *_message,
|
|
|
|
const QString &target,
|
2019-09-18 13:03:16 +02:00
|
|
|
const QString &content,
|
|
|
|
TwitchIrcServer &server, bool isSub,
|
|
|
|
bool isAction)
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
|
|
|
QString channelName;
|
|
|
|
if (!trimChannelName(target, channelName))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto chan = server.getChannelOrEmpty(channelName);
|
|
|
|
|
|
|
|
if (chan->isEmpty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageParseArgs args;
|
|
|
|
if (isSub)
|
|
|
|
{
|
|
|
|
args.trimSubscriberUsername = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chan->isBroadcaster())
|
|
|
|
{
|
|
|
|
args.isStaffOrBroadcaster = true;
|
|
|
|
}
|
|
|
|
|
2020-08-08 15:37:22 +02:00
|
|
|
auto channel = dynamic_cast<TwitchChannel *>(chan.get());
|
|
|
|
|
|
|
|
const auto &tags = _message->tags();
|
|
|
|
if (const auto &it = tags.find("custom-reward-id"); it != tags.end())
|
|
|
|
{
|
|
|
|
const auto rewardId = it.value().toString();
|
|
|
|
if (!channel->isChannelPointRewardKnown(rewardId))
|
|
|
|
{
|
|
|
|
// Need to wait for pubsub reward notification
|
|
|
|
auto clone = _message->clone();
|
|
|
|
channel->channelPointRewardAdded.connect(
|
|
|
|
[=, &server](ChannelPointReward reward) {
|
|
|
|
if (reward.id == rewardId)
|
|
|
|
{
|
|
|
|
this->addMessage(clone, target, content, server, isSub,
|
|
|
|
isAction);
|
|
|
|
clone->deleteLater();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
args.channelPointRewardId = rewardId;
|
|
|
|
}
|
|
|
|
|
2019-09-08 22:27:57 +02:00
|
|
|
TwitchMessageBuilder builder(chan.get(), _message, args, content, isAction);
|
|
|
|
|
|
|
|
if (isSub || !builder.isIgnored())
|
|
|
|
{
|
|
|
|
if (isSub)
|
|
|
|
{
|
|
|
|
builder->flags.set(MessageFlag::Subscription);
|
|
|
|
builder->flags.unset(MessageFlag::Highlighted);
|
|
|
|
}
|
|
|
|
auto msg = builder.build();
|
2020-02-02 14:31:37 +01:00
|
|
|
|
|
|
|
IrcMessageHandler::setSimilarityFlags(msg, chan);
|
|
|
|
|
|
|
|
if (!msg->flags.has(MessageFlag::Similar) ||
|
|
|
|
(!getSettings()->hideSimilar &&
|
|
|
|
getSettings()->shownSimilarTriggerHighlights))
|
|
|
|
{
|
|
|
|
builder.triggerHighlights();
|
|
|
|
}
|
|
|
|
|
2020-10-24 14:33:15 +02:00
|
|
|
const auto highlighted = msg->flags.has(MessageFlag::Highlighted);
|
|
|
|
const auto showInMentions = msg->flags.has(MessageFlag::ShowInMentions);
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
if (!isSub)
|
|
|
|
{
|
2020-10-24 14:33:15 +02:00
|
|
|
if (highlighted && showInMentions)
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
|
|
|
server.mentionsChannel->addMessage(msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
chan->addMessage(msg);
|
2019-09-18 13:03:16 +02:00
|
|
|
if (auto chatters = dynamic_cast<ChannelChatters *>(chan.get()))
|
|
|
|
{
|
|
|
|
chatters->addRecentChatter(msg->displayName);
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleRoomStateMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
const auto &tags = message->tags();
|
|
|
|
auto app = getApp();
|
|
|
|
|
|
|
|
// get twitch channel
|
|
|
|
QString chanName;
|
|
|
|
if (!trimChannelName(message->parameter(0), chanName))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto chan = app->twitch.server->getChannelOrEmpty(chanName);
|
|
|
|
|
|
|
|
if (auto *twitchChannel = dynamic_cast<TwitchChannel *>(chan.get()))
|
|
|
|
{
|
|
|
|
// room-id
|
|
|
|
decltype(tags.find("xD")) it;
|
|
|
|
|
|
|
|
if ((it = tags.find("room-id")) != tags.end())
|
|
|
|
{
|
|
|
|
auto roomId = it.value().toString();
|
|
|
|
|
|
|
|
twitchChannel->setRoomId(roomId);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Room modes
|
|
|
|
{
|
|
|
|
auto roomModes = *twitchChannel->accessRoomModes();
|
|
|
|
|
|
|
|
if ((it = tags.find("emote-only")) != tags.end())
|
|
|
|
{
|
|
|
|
roomModes.emoteOnly = it.value() == "1";
|
|
|
|
}
|
|
|
|
if ((it = tags.find("subs-only")) != tags.end())
|
|
|
|
{
|
|
|
|
roomModes.submode = it.value() == "1";
|
|
|
|
}
|
|
|
|
if ((it = tags.find("slow")) != tags.end())
|
|
|
|
{
|
|
|
|
roomModes.slowMode = it.value().toInt();
|
|
|
|
}
|
|
|
|
if ((it = tags.find("r9k")) != tags.end())
|
|
|
|
{
|
|
|
|
roomModes.r9k = it.value() == "1";
|
|
|
|
}
|
|
|
|
if ((it = tags.find("broadcaster-lang")) != tags.end())
|
|
|
|
{
|
|
|
|
roomModes.broadcasterLang = it.value().toString();
|
|
|
|
}
|
|
|
|
if ((it = tags.find("followers-only")) != tags.end())
|
|
|
|
{
|
|
|
|
roomModes.followerOnly = it.value().toInt();
|
|
|
|
}
|
|
|
|
twitchChannel->setRoomModes(roomModes);
|
|
|
|
}
|
|
|
|
|
|
|
|
twitchChannel->roomModesChanged.invoke();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleClearChatMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
// check parameter count
|
|
|
|
if (message->parameters().length() < 1)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString chanName;
|
|
|
|
if (!trimChannelName(message->parameter(0), chanName))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto app = getApp();
|
|
|
|
|
|
|
|
// get channel
|
|
|
|
auto chan = app->twitch.server->getChannelOrEmpty(chanName);
|
|
|
|
|
|
|
|
if (chan->isEmpty())
|
|
|
|
{
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "[IrcMessageHandler:handleClearChatMessage] Twitch channel"
|
|
|
|
<< chanName << "not found";
|
2019-09-08 22:27:57 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if the chat has been cleared by a moderator
|
|
|
|
if (message->parameters().length() == 1)
|
|
|
|
{
|
|
|
|
chan->disableAllMessages();
|
|
|
|
chan->addMessage(
|
2020-10-03 13:37:07 +02:00
|
|
|
makeSystemMessage("Chat has been cleared by a moderator.",
|
|
|
|
calculateMessageTimestamp(message)));
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get username, duration and message of the timed out user
|
|
|
|
QString username = message->parameter(1);
|
2021-04-25 15:44:12 +02:00
|
|
|
QString durationInSeconds;
|
2019-09-08 22:27:57 +02:00
|
|
|
QVariant v = message->tag("ban-duration");
|
|
|
|
if (v.isValid())
|
|
|
|
{
|
|
|
|
durationInSeconds = v.toString();
|
|
|
|
}
|
|
|
|
|
2020-10-03 13:37:07 +02:00
|
|
|
auto timeoutMsg =
|
2021-04-25 15:44:12 +02:00
|
|
|
MessageBuilder(timeoutMessage, username, durationInSeconds, false,
|
|
|
|
calculateMessageTimestamp(message))
|
2020-10-03 13:37:07 +02:00
|
|
|
.release();
|
2019-09-08 22:27:57 +02:00
|
|
|
chan->addOrReplaceTimeout(timeoutMsg);
|
|
|
|
|
|
|
|
// refresh all
|
|
|
|
app->windows->repaintVisibleChatWidgets(chan.get());
|
|
|
|
if (getSettings()->hideModerated)
|
|
|
|
{
|
|
|
|
app->windows->forceLayoutChannelViews();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleClearMessageMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
// check parameter count
|
|
|
|
if (message->parameters().length() < 1)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString chanName;
|
|
|
|
if (!trimChannelName(message->parameter(0), chanName))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto app = getApp();
|
|
|
|
|
|
|
|
// get channel
|
|
|
|
auto chan = app->twitch.server->getChannelOrEmpty(chanName);
|
|
|
|
|
|
|
|
if (chan->isEmpty())
|
|
|
|
{
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "[IrcMessageHandler:handleClearMessageMessage] Twitch "
|
|
|
|
"channel"
|
|
|
|
<< chanName << "not found";
|
2019-09-08 22:27:57 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto tags = message->tags();
|
|
|
|
|
|
|
|
QString targetID = tags.value("target-msg-id").toString();
|
|
|
|
|
|
|
|
chan->deleteMessage(targetID);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleUserStateMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
|
|
|
|
QString channelName;
|
|
|
|
if (!trimChannelName(message->parameter(0), channelName))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto c = app->twitch.server->getChannelOrEmpty(channelName);
|
|
|
|
if (c->isEmpty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant _badges = message->tag("badges");
|
|
|
|
if (_badges.isValid())
|
|
|
|
{
|
|
|
|
TwitchChannel *tc = dynamic_cast<TwitchChannel *>(c.get());
|
|
|
|
if (tc != nullptr)
|
|
|
|
{
|
|
|
|
auto parsedBadges = parseBadges(_badges.toString());
|
|
|
|
tc->setVIP(parsedBadges.contains("vip"));
|
|
|
|
tc->setStaff(parsedBadges.contains("staff"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant _mod = message->tag("mod");
|
|
|
|
if (_mod.isValid())
|
|
|
|
{
|
|
|
|
TwitchChannel *tc = dynamic_cast<TwitchChannel *>(c.get());
|
|
|
|
if (tc != nullptr)
|
|
|
|
{
|
|
|
|
tc->setMod(_mod == "1");
|
|
|
|
}
|
|
|
|
}
|
2021-03-21 15:42:45 +01:00
|
|
|
|
|
|
|
// handle emotes
|
|
|
|
app->accounts->twitch.getCurrent()->loadUserstateEmotes(
|
|
|
|
message->tag("emote-sets").toString().split(","));
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleWhisperMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
MessageParseArgs args;
|
|
|
|
|
|
|
|
args.isReceivedWhisper = true;
|
|
|
|
|
|
|
|
auto c = app->twitch.server->whispersChannel.get();
|
|
|
|
|
|
|
|
TwitchMessageBuilder builder(c, message, args, message->parameter(1),
|
|
|
|
false);
|
|
|
|
|
|
|
|
if (!builder.isIgnored())
|
|
|
|
{
|
|
|
|
builder->flags.set(MessageFlag::Whisper);
|
|
|
|
MessagePtr _message = builder.build();
|
|
|
|
builder.triggerHighlights();
|
|
|
|
|
|
|
|
app->twitch.server->lastUserThatWhisperedMe.set(builder.userName);
|
|
|
|
|
|
|
|
if (_message->flags.has(MessageFlag::Highlighted))
|
|
|
|
{
|
|
|
|
app->twitch.server->mentionsChannel->addMessage(_message);
|
|
|
|
}
|
|
|
|
|
|
|
|
c->addMessage(_message);
|
|
|
|
|
|
|
|
auto overrideFlags = boost::optional<MessageFlags>(_message->flags);
|
|
|
|
overrideFlags->set(MessageFlag::DoNotTriggerNotification);
|
|
|
|
overrideFlags->set(MessageFlag::DoNotLog);
|
|
|
|
|
|
|
|
if (getSettings()->inlineWhispers)
|
|
|
|
{
|
|
|
|
app->twitch.server->forEachChannel(
|
|
|
|
[&_message, overrideFlags](ChannelPtr channel) {
|
|
|
|
channel->addMessage(_message, overrideFlags);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<MessagePtr> IrcMessageHandler::parseUserNoticeMessage(
|
|
|
|
Channel *channel, Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
std::vector<MessagePtr> builtMessages;
|
|
|
|
|
|
|
|
auto tags = message->tags();
|
|
|
|
auto parameters = message->parameters();
|
|
|
|
|
|
|
|
QString msgType = tags.value("msg-id", "").toString();
|
|
|
|
QString content;
|
|
|
|
if (parameters.size() >= 2)
|
|
|
|
{
|
|
|
|
content = parameters[1];
|
|
|
|
}
|
|
|
|
|
2021-05-01 17:42:51 +02:00
|
|
|
if (specialMessageTypes.contains(msgType))
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
2021-05-01 17:42:51 +02:00
|
|
|
// Messages are not required, so they might be empty
|
2019-09-08 22:27:57 +02:00
|
|
|
if (!content.isEmpty())
|
|
|
|
{
|
|
|
|
MessageParseArgs args;
|
|
|
|
args.trimSubscriberUsername = true;
|
|
|
|
|
|
|
|
TwitchMessageBuilder builder(channel, message, args, content,
|
|
|
|
false);
|
|
|
|
builder->flags.set(MessageFlag::Subscription);
|
|
|
|
builder->flags.unset(MessageFlag::Highlighted);
|
|
|
|
builtMessages.emplace_back(builder.build());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto it = tags.find("system-msg");
|
|
|
|
|
|
|
|
if (it != tags.end())
|
|
|
|
{
|
2021-05-01 17:42:51 +02:00
|
|
|
// By default, we return value of system-msg tag
|
2021-04-11 14:17:21 +02:00
|
|
|
QString messageText = it.value().toString();
|
|
|
|
|
|
|
|
if (msgType == "bitsbadgetier")
|
|
|
|
{
|
|
|
|
messageText = QString("%1 just earned a new %2 Bits badge!")
|
|
|
|
.arg(tags.value("display-name").toString())
|
|
|
|
.arg(kFormatNumbers(
|
|
|
|
tags.value("msg-param-threshold").toInt()));
|
|
|
|
}
|
|
|
|
|
|
|
|
auto b = MessageBuilder(systemMessage, parseTagString(messageText),
|
|
|
|
calculateMessageTimestamp(message));
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
b->flags.set(MessageFlag::Subscription);
|
|
|
|
auto newMessage = b.release();
|
|
|
|
builtMessages.emplace_back(newMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
return builtMessages;
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleUserNoticeMessage(Communi::IrcMessage *message,
|
2019-09-18 13:03:16 +02:00
|
|
|
TwitchIrcServer &server)
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
|
|
|
auto tags = message->tags();
|
|
|
|
auto parameters = message->parameters();
|
|
|
|
|
|
|
|
auto target = parameters[0];
|
|
|
|
QString msgType = tags.value("msg-id", "").toString();
|
|
|
|
QString content;
|
|
|
|
if (parameters.size() >= 2)
|
|
|
|
{
|
|
|
|
content = parameters[1];
|
|
|
|
}
|
|
|
|
|
2021-05-01 17:42:51 +02:00
|
|
|
if (specialMessageTypes.contains(msgType))
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
2021-05-01 17:42:51 +02:00
|
|
|
// Messages are not required, so they might be empty
|
2019-09-08 22:27:57 +02:00
|
|
|
if (!content.isEmpty())
|
|
|
|
{
|
|
|
|
this->addMessage(message, target, content, server, true, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto it = tags.find("system-msg");
|
|
|
|
|
|
|
|
if (it != tags.end())
|
|
|
|
{
|
2021-05-01 17:42:51 +02:00
|
|
|
// By default, we return value of system-msg tag
|
2021-04-11 14:17:21 +02:00
|
|
|
QString messageText = it.value().toString();
|
|
|
|
|
|
|
|
if (msgType == "bitsbadgetier")
|
|
|
|
{
|
|
|
|
messageText = QString("%1 just earned a new %2 Bits badge!")
|
|
|
|
.arg(tags.value("display-name").toString())
|
|
|
|
.arg(kFormatNumbers(
|
|
|
|
tags.value("msg-param-threshold").toInt()));
|
|
|
|
}
|
|
|
|
|
|
|
|
auto b = MessageBuilder(systemMessage, parseTagString(messageText),
|
|
|
|
calculateMessageTimestamp(message));
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
b->flags.set(MessageFlag::Subscription);
|
|
|
|
auto newMessage = b.release();
|
|
|
|
|
|
|
|
QString channelName;
|
|
|
|
|
|
|
|
if (message->parameters().size() < 1)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!trimChannelName(message->parameter(0), channelName))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto chan = server.getChannelOrEmpty(channelName);
|
|
|
|
|
|
|
|
if (!chan->isEmpty())
|
|
|
|
{
|
|
|
|
chan->addMessage(newMessage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleModeMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
|
|
|
|
auto channel = app->twitch.server->getChannelOrEmpty(
|
|
|
|
message->parameter(0).remove(0, 1));
|
|
|
|
|
|
|
|
if (channel->isEmpty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message->parameter(1) == "+o")
|
|
|
|
{
|
|
|
|
channel->modList.append(message->parameter(2));
|
|
|
|
}
|
|
|
|
else if (message->parameter(1) == "-o")
|
|
|
|
{
|
|
|
|
channel->modList.append(message->parameter(2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<MessagePtr> IrcMessageHandler::parseNoticeMessage(
|
|
|
|
Communi::IrcNoticeMessage *message)
|
|
|
|
{
|
|
|
|
if (message->content().startsWith("Login auth", Qt::CaseInsensitive))
|
|
|
|
{
|
2020-10-04 17:36:38 +02:00
|
|
|
const auto linkColor = MessageColor(MessageColor::Link);
|
|
|
|
const auto accountsLink = Link(Link::OpenAccountsPage, QString());
|
|
|
|
const auto curUser = getApp()->accounts->twitch.getCurrent();
|
|
|
|
const auto expirationText = QString("Login expired for user \"%1\"!")
|
|
|
|
.arg(curUser->getUserName());
|
|
|
|
const auto loginPromptText = QString(" Try adding your account again.");
|
|
|
|
|
|
|
|
auto builder = MessageBuilder();
|
|
|
|
builder.message().flags.set(MessageFlag::System);
|
|
|
|
|
|
|
|
builder.emplace<TimestampElement>();
|
|
|
|
builder.emplace<TextElement>(expirationText, MessageElementFlag::Text,
|
|
|
|
MessageColor::System);
|
|
|
|
builder
|
|
|
|
.emplace<TextElement>(loginPromptText, MessageElementFlag::Text,
|
|
|
|
linkColor)
|
|
|
|
->setLink(accountsLink);
|
|
|
|
|
|
|
|
return {builder.release()};
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
2020-12-12 14:19:51 +01:00
|
|
|
else if (message->content().startsWith("You are permanently banned "))
|
|
|
|
{
|
|
|
|
return {generateBannedMessage(true)};
|
|
|
|
}
|
2021-01-31 13:55:44 +01:00
|
|
|
else if (message->tags()["msg-id"] == "msg_timedout")
|
|
|
|
{
|
2021-03-07 18:03:53 +01:00
|
|
|
std::vector<MessagePtr> builtMessage;
|
2021-01-31 13:55:44 +01:00
|
|
|
|
2021-03-07 18:03:53 +01:00
|
|
|
QString remainingTime =
|
|
|
|
formatTime(message->content().split(" ").value(5));
|
|
|
|
QString formattedMessage =
|
|
|
|
QString("You are timed out for %1.")
|
|
|
|
.arg(remainingTime.isEmpty() ? "0s" : remainingTime);
|
2021-01-31 13:55:44 +01:00
|
|
|
|
2021-03-07 18:03:53 +01:00
|
|
|
builtMessage.emplace_back(makeSystemMessage(
|
|
|
|
formattedMessage, calculateMessageTimestamp(message)));
|
2021-01-31 13:55:44 +01:00
|
|
|
|
2021-03-07 18:03:53 +01:00
|
|
|
return builtMessage;
|
2021-01-31 13:55:44 +01:00
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
std::vector<MessagePtr> builtMessages;
|
|
|
|
|
2020-10-03 13:37:07 +02:00
|
|
|
builtMessages.emplace_back(makeSystemMessage(
|
|
|
|
message->content(), calculateMessageTimestamp(message)));
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
return builtMessages;
|
|
|
|
}
|
|
|
|
} // namespace chatterino
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleNoticeMessage(Communi::IrcNoticeMessage *message)
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
auto builtMessages = this->parseNoticeMessage(message);
|
|
|
|
|
|
|
|
for (auto msg : builtMessages)
|
|
|
|
{
|
|
|
|
QString channelName;
|
|
|
|
if (!trimChannelName(message->target(), channelName) ||
|
|
|
|
channelName == "jtv")
|
|
|
|
{
|
|
|
|
// Notice wasn't targeted at a single channel, send to all twitch
|
|
|
|
// channels
|
|
|
|
app->twitch.server->forEachChannelAndSpecialChannels(
|
|
|
|
[msg](const auto &c) {
|
2020-11-08 12:02:19 +01:00
|
|
|
c->addMessage(msg);
|
2019-09-08 22:27:57 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto channel = app->twitch.server->getChannelOrEmpty(channelName);
|
|
|
|
|
|
|
|
if (channel->isEmpty())
|
|
|
|
{
|
2020-11-21 16:20:10 +01:00
|
|
|
qCDebug(chatterinoTwitch)
|
|
|
|
<< "[IrcManager:handleNoticeMessage] Channel" << channelName
|
|
|
|
<< "not found in channel manager";
|
2019-09-08 22:27:57 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString tags = message->tags().value("msg-id", "").toString();
|
|
|
|
if (tags == "bad_delete_message_error" || tags == "usage_delete")
|
|
|
|
{
|
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
"Usage: \"/delete <msg-id>\" - can't take more "
|
|
|
|
"than one argument"));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
channel->addMessage(msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleJoinMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
auto channel = app->twitch.server->getChannelOrEmpty(
|
|
|
|
message->parameter(0).remove(0, 1));
|
|
|
|
|
|
|
|
if (TwitchChannel *twitchChannel =
|
|
|
|
dynamic_cast<TwitchChannel *>(channel.get()))
|
|
|
|
{
|
2019-09-18 13:03:16 +02:00
|
|
|
if (message->nick() !=
|
|
|
|
getApp()->accounts->twitch.getCurrent()->getUserName() &&
|
|
|
|
getSettings()->showJoins.getValue())
|
|
|
|
{
|
|
|
|
twitchChannel->addJoinedUser(message->nick());
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handlePartMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
auto channel = app->twitch.server->getChannelOrEmpty(
|
|
|
|
message->parameter(0).remove(0, 1));
|
|
|
|
|
|
|
|
if (TwitchChannel *twitchChannel =
|
|
|
|
dynamic_cast<TwitchChannel *>(channel.get()))
|
|
|
|
{
|
2020-12-12 14:19:51 +01:00
|
|
|
const auto selfAccountName =
|
|
|
|
getApp()->accounts->twitch.getCurrent()->getUserName();
|
|
|
|
if (message->nick() != selfAccountName &&
|
2019-10-20 20:11:12 +02:00
|
|
|
getSettings()->showParts.getValue())
|
2019-09-18 13:03:16 +02:00
|
|
|
{
|
|
|
|
twitchChannel->addPartedUser(message->nick());
|
|
|
|
}
|
2020-12-12 14:19:51 +01:00
|
|
|
|
|
|
|
if (message->nick() == selfAccountName)
|
|
|
|
{
|
|
|
|
channel->addMessage(generateBannedMessage(false));
|
|
|
|
}
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} // namespace chatterino
|