#include "MessageBuilder.hpp" #include "Application.hpp" #include "common/LinkParser.hpp" #include "messages/Image.hpp" #include "messages/Message.hpp" #include "messages/MessageElement.hpp" #include "providers/LinkResolver.hpp" #include "providers/twitch/PubsubActions.hpp" #include "singletons/Emotes.hpp" #include "singletons/Resources.hpp" #include "singletons/Theme.hpp" #include "util/FormatTime.hpp" #include "util/IrcHelpers.hpp" #include #include namespace chatterino { MessagePtr makeSystemMessage(const QString &text) { return MessageBuilder(systemMessage, text).release(); } std::pair makeAutomodMessage( const AutomodAction &action) { auto builder = MessageBuilder(); builder.emplace(); builder.message().flags.set(MessageFlag::PubSub); builder .emplace( Image::fromPixmap(getApp()->resources->twitch.automod), MessageElementFlag::BadgeChannelAuthority) ->setTooltip("AutoMod"); builder.emplace("AutoMod:", MessageElementFlag::BoldUsername, MessageColor(QColor("blue")), FontStyle::ChatMediumBold); builder.emplace( "AutoMod:", MessageElementFlag::NonBoldUsername, MessageColor(QColor("blue"))); builder.emplace( ("Held a message for reason: " + action.reason + ". Allow will post it in chat. "), MessageElementFlag::Text, MessageColor::Text); builder .emplace("Allow", MessageElementFlag::Text, MessageColor(QColor("green")), FontStyle::ChatMediumBold) ->setLink({Link::AutoModAllow, action.msgID}); builder .emplace(" Deny", MessageElementFlag::Text, MessageColor(QColor("red")), FontStyle::ChatMediumBold) ->setLink({Link::AutoModDeny, action.msgID}); // builder.emplace(action.msgID, // MessageElementFlag::Text, // MessageColor::Text); builder.message().flags.set(MessageFlag::AutoMod); auto message1 = builder.release(); builder = MessageBuilder(); builder.emplace(); builder.message().flags.set(MessageFlag::PubSub); builder .emplace( action.target.name + ":", MessageElementFlag::BoldUsername, MessageColor(QColor("red")), FontStyle::ChatMediumBold) ->setLink({Link::UserInfo, action.target.name}); builder .emplace(action.target.name + ":", MessageElementFlag::NonBoldUsername, MessageColor(QColor("red"))) ->setLink({Link::UserInfo, action.target.name}); builder.emplace(action.message, MessageElementFlag::Text, MessageColor::Text); builder.message().flags.set(MessageFlag::AutoMod); auto message2 = builder.release(); return std::make_pair(message1, message2); } MessageBuilder::MessageBuilder() : message_(std::make_shared()) { } MessageBuilder::MessageBuilder(const QString &text) : MessageBuilder() { this->emplace(); this->emplace(text, MessageElementFlag::Text, MessageColor::System); this->message().searchText = text; } MessageBuilder::MessageBuilder(SystemMessageTag, const QString &text) : MessageBuilder() { this->emplace(); this->emplace(text, MessageElementFlag::Text, MessageColor::System); this->message().flags.set(MessageFlag::System); this->message().flags.set(MessageFlag::DoNotTriggerNotification); this->message().searchText = text; } MessageBuilder::MessageBuilder(TimeoutMessageTag, const QString &username, const QString &durationInSeconds, const QString &reason, bool multipleTimes) : MessageBuilder() { QString text; text.append(username); if (!durationInSeconds.isEmpty()) { text.append(" has been timed out"); // TODO: Implement who timed the user out text.append(" for "); bool ok = true; int timeoutSeconds = durationInSeconds.toInt(&ok); if (ok) { text.append(formatTime(timeoutSeconds)); } } else { text.append(" has been permanently banned"); } if (reason.length() > 0) { text.append(": \""); text.append(parseTagString(reason)); text.append("\""); } text.append("."); if (multipleTimes) { text.append(" (multiple times)"); } this->message().flags.set(MessageFlag::System); this->message().flags.set(MessageFlag::Timeout); this->message().flags.set(MessageFlag::DoNotTriggerNotification); this->message().timeoutUser = username; this->emplace(); this->emplace(text, MessageElementFlag::Text, MessageColor::System); this->message().searchText = text; } MessageBuilder::MessageBuilder(const BanAction &action, uint32_t count) : MessageBuilder() { this->emplace(); this->message().flags.set(MessageFlag::System); this->message().flags.set(MessageFlag::Timeout); this->message().timeoutUser = action.target.name; this->message().count = count; QString text; if (action.isBan()) { if (action.reason.isEmpty()) { text = QString("%1 banned %2.") // .arg(action.source.name) .arg(action.target.name); } else { text = QString("%1 banned %2: \"%3\".") // .arg(action.source.name) .arg(action.target.name) .arg(action.reason); } } else { if (action.reason.isEmpty()) { text = QString("%1 timed out %2 for %3.") // .arg(action.source.name) .arg(action.target.name) .arg(formatTime(action.duration)); } else { text = QString("%1 timed out %2 for %3: \"%4\".") // .arg(action.source.name) .arg(action.target.name) .arg(formatTime(action.duration)) .arg(action.reason); } if (count > 1) { text.append(QString(" (%1 times)").arg(count)); } } this->emplace(text, MessageElementFlag::Text, MessageColor::System); this->message().searchText = text; } MessageBuilder::MessageBuilder(const UnbanAction &action) : MessageBuilder() { this->emplace(); this->message().flags.set(MessageFlag::System); this->message().flags.set(MessageFlag::Untimeout); this->message().timeoutUser = action.target.name; QString text; if (action.wasBan()) { text = QString("%1 unbanned %2.") // .arg(action.source.name) .arg(action.target.name); } else { text = QString("%1 untimedout %2.") // .arg(action.source.name) .arg(action.target.name); } this->emplace(text, MessageElementFlag::Text, MessageColor::System); this->message().searchText = text; } MessageBuilder::MessageBuilder(const AutomodUserAction &action) : MessageBuilder() { this->emplace(); this->message().flags.set(MessageFlag::System); QString text; switch (action.type) { case AutomodUserAction::AddPermitted: { text = QString("%1 added %2 as a permitted term on AutoMod.") .arg(action.source.name) .arg(action.message); } break; case AutomodUserAction::AddBlocked: { text = QString("%1 added %2 as a blocked term on AutoMod.") .arg(action.source.name) .arg(action.message); } break; case AutomodUserAction::RemovePermitted: { text = QString("%1 removed %2 as a permitted term term on AutoMod.") .arg(action.source.name) .arg(action.message); } break; case AutomodUserAction::RemoveBlocked: { text = QString("%1 removed %2 as a blocked term on AutoMod.") .arg(action.source.name) .arg(action.message); } break; case AutomodUserAction::Properties: { text = QString("%1 modified the AutoMod properties.") .arg(action.source.name); } break; } this->emplace(text, MessageElementFlag::Text, MessageColor::System); } Message *MessageBuilder::operator->() { return this->message_.get(); } Message &MessageBuilder::message() { return *this->message_; } MessagePtr MessageBuilder::release() { std::shared_ptr ptr; this->message_.swap(ptr); return ptr; } std::weak_ptr MessageBuilder::weakOf() { return this->message_; } void MessageBuilder::append(std::unique_ptr element) { this->message().elements.push_back(std::move(element)); } QString MessageBuilder::matchLink(const QString &string) { LinkParser linkParser(string); static QRegularExpression httpRegex( "\\bhttps?://", QRegularExpression::CaseInsensitiveOption); static QRegularExpression ftpRegex( "\\bftps?://", QRegularExpression::CaseInsensitiveOption); static QRegularExpression spotifyRegex( "\\bspotify:", QRegularExpression::CaseInsensitiveOption); if (!linkParser.hasMatch()) { return QString(); } QString captured = linkParser.getCaptured(); if (!captured.contains(httpRegex) && !captured.contains(ftpRegex) && !captured.contains(spotifyRegex)) { captured.insert(0, "http://"); } return captured; } void MessageBuilder::addLink(const QString &origLink, const QString &matchedLink) { static QRegularExpression domainRegex( R"(^(?:(?:ftp|http)s?:\/\/)?([^\/]+)(?:\/.*)?$)", QRegularExpression::CaseInsensitiveOption); QString lowercaseLinkString; auto match = domainRegex.match(origLink); if (match.isValid()) { lowercaseLinkString = origLink.mid(0, match.capturedStart(1)) + match.captured(1).toLower() + origLink.mid(match.capturedEnd(1)); } else { lowercaseLinkString = origLink; } auto linkElement = Link(Link::Url, matchedLink); auto textColor = MessageColor(MessageColor::Link); auto linkMELowercase = this->emplace(lowercaseLinkString, MessageElementFlag::LowercaseLink, textColor) ->setLink(linkElement); auto linkMEOriginal = this->emplace(origLink, MessageElementFlag::OriginalLink, textColor) ->setLink(linkElement); LinkResolver::getLinkInfo(matchedLink, [weakMessage = this->weakOf(), linkMELowercase, linkMEOriginal, matchedLink](QString tooltipText, Link originalLink) { auto shared = weakMessage.lock(); if (!shared) { return; } if (!tooltipText.isEmpty()) { linkMELowercase->setTooltip(tooltipText); linkMEOriginal->setTooltip(tooltipText); } if (originalLink.value != matchedLink && !originalLink.value.isEmpty()) { linkMELowercase->setLink(originalLink)->updateLink(); linkMEOriginal->setLink(originalLink)->updateLink(); } }); } } // namespace chatterino