#include "common/Channel.hpp" #include "Application.hpp" #include "debug/Log.hpp" #include "messages/Message.hpp" #include "singletons/Emotes.hpp" #include "singletons/Logging.hpp" #include "singletons/WindowManager.hpp" #include #include #include #include #include #include #include namespace chatterino { Channel::Channel(const QString &_name, Type type) : name(_name) , completionModel(this->name) , type_(type) { QObject::connect(&this->clearCompletionModelTimer_, &QTimer::timeout, [this]() { this->completionModel.clearExpiredStrings(); // }); this->clearCompletionModelTimer_.start(60 * 1000); } Channel::~Channel() { this->destroyed.invoke(); } Channel::Type Channel::getType() const { return this->type_; } bool Channel::isTwitchChannel() const { return this->type_ >= Type::Twitch && this->type_ < Type::TwitchEnd; } bool Channel::isEmpty() const { return this->name.isEmpty(); } LimitedQueueSnapshot Channel::getMessageSnapshot() { return this->messages_.getSnapshot(); } void Channel::addMessage(MessagePtr message) { auto app = getApp(); MessagePtr deleted; const QString &username = message->loginName; if (!username.isEmpty()) { // TODO: Add recent chatters display name. This should maybe be a setting this->addRecentChatter(message); } // FOURTF: change this when adding more providers if (this->isTwitchChannel()) { app->logging->addMessage(this->name, message); } if (this->messages_.pushBack(message, deleted)) { this->messageRemovedFromStart.invoke(deleted); } this->messageAppended.invoke(message); } void Channel::addOrReplaceTimeout(MessagePtr message) { LimitedQueueSnapshot snapshot = this->getMessageSnapshot(); int snapshotLength = snapshot.getLength(); int end = std::max(0, snapshotLength - 20); bool addMessage = true; QTime minimumTime = QTime::currentTime().addSecs(-5); for (int i = snapshotLength - 1; i >= end; --i) { auto &s = snapshot[i]; qDebug() << s->parseTime << minimumTime; if (s->parseTime < minimumTime) { break; } if (s->flags.HasFlag(Message::Untimeout) && s->timeoutUser == message->timeoutUser) { break; } if (s->flags.HasFlag(Message::Timeout) && s->timeoutUser == message->timeoutUser) { if (message->flags.HasFlag(Message::PubSub) && !s->flags.HasFlag(Message::PubSub)) { this->replaceMessage(s, message); addMessage = false; break; } if (!message->flags.HasFlag(Message::PubSub) && s->flags.HasFlag(Message::PubSub)) { addMessage = false; break; } int count = s->count + 1; MessagePtr replacement(Message::createSystemMessage( message->searchText + QString(" (") + QString::number(count) + " times)")); replacement->timeoutUser = message->timeoutUser; replacement->count = count; replacement->flags = message->flags; this->replaceMessage(s, replacement); return; } } // disable the messages from the user for (int i = 0; i < snapshotLength; i++) { auto &s = snapshot[i]; if ((s->flags & (Message::Timeout | Message::Untimeout)) == 0 && s->loginName == message->timeoutUser) { s->flags.EnableFlag(Message::Disabled); } } if (addMessage) { this->addMessage(message); } // XXX: Might need the following line // WindowManager::getInstance().repaintVisibleChatWidgets(this); } void Channel::disableAllMessages() { LimitedQueueSnapshot snapshot = this->getMessageSnapshot(); int snapshotLength = snapshot.getLength(); for (int i = 0; i < snapshotLength; i++) { auto &s = snapshot[i]; if (s->flags & Message::System || s->flags & Message::Timeout) { continue; } s->flags.EnableFlag(Message::Disabled); } } void Channel::addMessagesAtStart(std::vector &_messages) { std::vector addedMessages = this->messages_.pushFront(_messages); if (addedMessages.size() != 0) { this->messagesAddedAtStart.invoke(addedMessages); } } void Channel::replaceMessage(MessagePtr message, MessagePtr replacement) { int index = this->messages_.replaceItem(message, replacement); if (index >= 0) { this->messageReplaced.invoke((size_t)index, replacement); } } void Channel::addRecentChatter(const std::shared_ptr &message) { // Do nothing by default } bool Channel::canSendMessage() const { return false; } void Channel::sendMessage(const QString &message) { } bool Channel::isMod() const { return false; } bool Channel::isBroadcaster() const { return false; } bool Channel::hasModRights() const { // fourtf: check if staff return this->isMod() || this->isBroadcaster(); } std::shared_ptr Channel::getEmpty() { static std::shared_ptr channel(new Channel("", Type::None)); return channel; } void Channel::onConnected() { } std::weak_ptr Channel::weak_from_this() { return std::weak_ptr(this->shared_from_this()); } } // namespace chatterino