2018-01-01 23:54:54 +01:00
|
|
|
#include "ircmessagehandler.hpp"
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
#include "debug/log.hpp"
|
|
|
|
#include "messages/limitedqueue.hpp"
|
|
|
|
#include "messages/message.hpp"
|
|
|
|
#include "singletons/channelmanager.hpp"
|
|
|
|
#include "singletons/resourcemanager.hpp"
|
|
|
|
#include "singletons/windowmanager.hpp"
|
|
|
|
#include "twitch/twitchchannel.hpp"
|
|
|
|
|
|
|
|
using namespace chatterino::messages;
|
|
|
|
|
|
|
|
namespace chatterino {
|
|
|
|
namespace singletons {
|
|
|
|
namespace helper {
|
|
|
|
|
|
|
|
IrcMessageHandler::IrcMessageHandler(ChannelManager &_channelManager,
|
|
|
|
ResourceManager &_resourceManager)
|
|
|
|
: channelManager(_channelManager)
|
|
|
|
, resourceManager(_resourceManager)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
IrcMessageHandler &IrcMessageHandler::getInstance()
|
|
|
|
{
|
|
|
|
static IrcMessageHandler instance(ChannelManager::getInstance(),
|
|
|
|
ResourceManager::getInstance());
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleRoomStateMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
const auto &tags = message->tags();
|
|
|
|
|
|
|
|
auto iterator = tags.find("room-id");
|
|
|
|
|
|
|
|
if (iterator != tags.end()) {
|
|
|
|
auto roomID = iterator.value().toString();
|
|
|
|
|
|
|
|
auto channel =
|
|
|
|
this->channelManager.getTwitchChannel(QString(message->toData()).split("#").at(1));
|
|
|
|
auto twitchChannel = dynamic_cast<twitch::TwitchChannel *>(channel.get());
|
|
|
|
if (twitchChannel != nullptr) {
|
|
|
|
twitchChannel->setRoomID(roomID);
|
|
|
|
}
|
|
|
|
|
|
|
|
this->resourceManager.loadChannelData(roomID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleClearChatMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
assert(message->parameters().length() >= 1);
|
|
|
|
|
|
|
|
auto rawChannelName = message->parameter(0);
|
|
|
|
|
|
|
|
assert(rawChannelName.length() >= 2);
|
|
|
|
|
|
|
|
auto trimmedChannelName = rawChannelName.mid(1);
|
|
|
|
|
|
|
|
auto c = this->channelManager.getTwitchChannel(trimmedChannelName);
|
|
|
|
|
|
|
|
if (!c) {
|
|
|
|
debug::Log(
|
|
|
|
"[IrcMessageHandler:handleClearChatMessage] Channel {} not found in channel manager",
|
|
|
|
trimmedChannelName);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if the chat has been cleared by a moderator
|
|
|
|
if (message->parameters().length() == 1) {
|
2018-01-11 20:16:25 +01:00
|
|
|
c->addMessage(Message::createSystemMessage("Chat has been cleared by a moderator."));
|
2018-01-01 23:54:54 +01:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(message->parameters().length() >= 2);
|
|
|
|
|
|
|
|
// get username, duration and message of the timed out user
|
|
|
|
QString username = message->parameter(1);
|
|
|
|
QString durationInSeconds, reason;
|
|
|
|
QVariant v = message->tag("ban-duration");
|
|
|
|
if (v.isValid()) {
|
|
|
|
durationInSeconds = v.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
v = message->tag("ban-reason");
|
|
|
|
if (v.isValid()) {
|
|
|
|
reason = v.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
// add the notice that the user has been timed out
|
2018-01-11 20:16:25 +01:00
|
|
|
LimitedQueueSnapshot<MessagePtr> snapshot = c->getMessageSnapshot();
|
2018-01-05 23:14:55 +01:00
|
|
|
bool addMessage = true;
|
2018-01-11 20:16:25 +01:00
|
|
|
int snapshotLength = snapshot.getLength();
|
2018-01-05 23:14:55 +01:00
|
|
|
|
2018-01-11 20:16:25 +01:00
|
|
|
for (int i = std::max(0, snapshotLength - 20); i < snapshotLength; i++) {
|
|
|
|
if (snapshot[i]->hasFlags(Message::Timeout) && snapshot[i]->loginName == username) {
|
|
|
|
MessagePtr replacement(
|
2018-01-05 23:14:55 +01:00
|
|
|
Message::createTimeoutMessage(username, durationInSeconds, reason, true));
|
|
|
|
c->replaceMessage(snapshot[i], replacement);
|
|
|
|
addMessage = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-01-01 23:54:54 +01:00
|
|
|
|
2018-01-05 23:14:55 +01:00
|
|
|
if (addMessage) {
|
2018-01-11 20:16:25 +01:00
|
|
|
c->addMessage(Message::createTimeoutMessage(username, durationInSeconds, reason, false));
|
2018-01-05 23:14:55 +01:00
|
|
|
}
|
2018-01-01 23:54:54 +01:00
|
|
|
|
|
|
|
// disable the messages from the user
|
2018-01-11 20:16:25 +01:00
|
|
|
for (int i = 0; i < snapshotLength; i++) {
|
|
|
|
if (!snapshot[i]->hasFlags(Message::Timeout) && snapshot[i]->loginName == username) {
|
2018-01-17 02:22:57 +01:00
|
|
|
snapshot[i]->addFlags(Message::Disabled);
|
2018-01-01 23:54:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// refresh all
|
2018-01-05 23:14:55 +01:00
|
|
|
WindowManager::getInstance().repaintVisibleChatWidgets(c.get());
|
2018-01-01 23:54:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleUserStateMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
2018-01-17 18:36:12 +01:00
|
|
|
QVariant _mod = message->tag("mod");
|
|
|
|
|
|
|
|
if (_mod.isValid()) {
|
|
|
|
auto rawChannelName = message->parameters().at(0);
|
|
|
|
auto trimmedChannelName = rawChannelName.mid(1);
|
|
|
|
|
|
|
|
auto c = this->channelManager.getTwitchChannel(trimmedChannelName);
|
|
|
|
twitch::TwitchChannel *tc = dynamic_cast<twitch::TwitchChannel *>(c.get());
|
|
|
|
if (tc != nullptr) {
|
|
|
|
tc->setMod(_mod == "1");
|
|
|
|
}
|
|
|
|
}
|
2018-01-01 23:54:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleWhisperMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
// TODO: Implement
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleUserNoticeMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleModeMessage(Communi::IrcMessage *message)
|
|
|
|
{
|
|
|
|
auto channel = channelManager.getTwitchChannel(message->parameter(0).remove(0, 1));
|
|
|
|
if (message->parameter(1) == "+o") {
|
|
|
|
channel->modList.append(message->parameter(2));
|
|
|
|
} else if (message->parameter(1) == "-o") {
|
|
|
|
channel->modList.append(message->parameter(2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleNoticeMessage(Communi::IrcNoticeMessage *message)
|
|
|
|
{
|
|
|
|
auto rawChannelName = message->target();
|
|
|
|
|
|
|
|
bool broadcast = rawChannelName.length() < 2;
|
2018-01-11 20:16:25 +01:00
|
|
|
MessagePtr msg = Message::createSystemMessage(message->content());
|
2018-01-01 23:54:54 +01:00
|
|
|
|
|
|
|
if (broadcast) {
|
|
|
|
this->channelManager.doOnAll([msg](const auto &c) {
|
|
|
|
c->addMessage(msg); //
|
|
|
|
});
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto trimmedChannelName = rawChannelName.mid(1);
|
|
|
|
|
|
|
|
auto c = this->channelManager.getTwitchChannel(trimmedChannelName);
|
|
|
|
|
|
|
|
if (!c) {
|
|
|
|
debug::Log("[IrcManager:handleNoticeMessage] Channel {} not found in channel manager",
|
|
|
|
trimmedChannelName);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
c->addMessage(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IrcMessageHandler::handleWriteConnectionNoticeMessage(Communi::IrcNoticeMessage *message)
|
|
|
|
{
|
|
|
|
QVariant v = message->tag("msg-id");
|
|
|
|
if (!v.isValid()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QString msg_id = v.toString();
|
|
|
|
|
|
|
|
static QList<QString> idsToSkip = {"timeout_success", "ban_success"};
|
|
|
|
|
|
|
|
if (idsToSkip.contains(msg_id)) {
|
|
|
|
// Already handled in the read-connection
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->handleNoticeMessage(message);
|
|
|
|
}
|
2018-01-17 02:22:57 +01:00
|
|
|
} // namespace helper
|
|
|
|
} // namespace singletons
|
|
|
|
} // namespace chatterino
|