#include "providers/ffz/FfzEmotes.hpp" #include #include "common/NetworkRequest.hpp" #include "common/Outcome.hpp" #include "common/QLogging.hpp" #include "messages/Emote.hpp" #include "messages/Image.hpp" #include "messages/MessageBuilder.hpp" #include "providers/twitch/TwitchChannel.hpp" namespace chatterino { namespace { const QString CHANNEL_HAS_NO_EMOTES( "This channel has no FrankerFaceZ channel emotes."); Url getEmoteLink(const QJsonObject &urls, const QString &emoteScale) { auto emote = urls.value(emoteScale); if (emote.isUndefined() || emote.isNull()) { return {""}; } assert(emote.isString()); return {"https:" + emote.toString()}; } void fillInEmoteData(const QJsonObject &urls, const EmoteName &name, const QString &tooltip, Emote &emoteData) { auto url1x = getEmoteLink(urls, "1"); auto url2x = getEmoteLink(urls, "2"); auto url3x = getEmoteLink(urls, "4"); //, code, tooltip emoteData.name = name; emoteData.images = ImageSet{Image::fromUrl(url1x, 1), url2x.string.isEmpty() ? Image::getEmpty() : Image::fromUrl(url2x, 0.5), url3x.string.isEmpty() ? Image::getEmpty() : Image::fromUrl(url3x, 0.25)}; emoteData.tooltip = {tooltip}; } EmotePtr cachedOrMake(Emote &&emote, const EmoteId &id) { static std::unordered_map> cache; static std::mutex mutex; return cachedOrMakeEmotePtr(std::move(emote), cache, mutex, id); } std::pair parseGlobalEmotes( const QJsonObject &jsonRoot, const EmoteMap ¤tEmotes) { auto jsonSets = jsonRoot.value("sets").toObject(); auto emotes = EmoteMap(); for (auto jsonSet : jsonSets) { auto jsonEmotes = jsonSet.toObject().value("emoticons").toArray(); for (auto jsonEmoteValue : jsonEmotes) { auto jsonEmote = jsonEmoteValue.toObject(); auto name = EmoteName{jsonEmote.value("name").toString()}; auto id = EmoteId{jsonEmote.value("id").toString()}; auto urls = jsonEmote.value("urls").toObject(); auto emote = Emote(); fillInEmoteData(urls, name, name.string + "
Global FFZ Emote", emote); emote.homePage = Url{QString("https://www.frankerfacez.com/emoticon/%1-%2") .arg(id.string) .arg(name.string)}; emotes[name] = cachedOrMakeEmotePtr(std::move(emote), currentEmotes); } } return {Success, std::move(emotes)}; } boost::optional parseAuthorityBadge(const QJsonObject &badgeUrls, const QString tooltip) { boost::optional authorityBadge; if (!badgeUrls.isEmpty()) { auto authorityBadge1x = getEmoteLink(badgeUrls, "1"); auto authorityBadge2x = getEmoteLink(badgeUrls, "2"); auto authorityBadge3x = getEmoteLink(badgeUrls, "4"); auto authorityBadgeImageSet = ImageSet{ Image::fromUrl(authorityBadge1x, 1), authorityBadge2x.string.isEmpty() ? Image::getEmpty() : Image::fromUrl(authorityBadge2x, 0.5), authorityBadge3x.string.isEmpty() ? Image::getEmpty() : Image::fromUrl(authorityBadge3x, 0.25), }; authorityBadge = std::make_shared(Emote{ {""}, authorityBadgeImageSet, Tooltip{tooltip}, authorityBadge1x, }); } return authorityBadge; } EmoteMap parseChannelEmotes(const QJsonObject &jsonRoot) { auto jsonSets = jsonRoot.value("sets").toObject(); auto emotes = EmoteMap(); for (auto jsonSet : jsonSets) { auto jsonEmotes = jsonSet.toObject().value("emoticons").toArray(); for (auto _jsonEmote : jsonEmotes) { auto jsonEmote = _jsonEmote.toObject(); // margins auto id = EmoteId{QString::number(jsonEmote.value("id").toInt())}; auto name = EmoteName{jsonEmote.value("name").toString()}; auto author = EmoteAuthor{jsonEmote.value("owner") .toObject() .value("display_name") .toString()}; auto urls = jsonEmote.value("urls").toObject(); Emote emote; fillInEmoteData(urls, name, QString("%1
Channel FFZ Emote
By: %2") .arg(name.string) .arg(author.string), emote); emote.homePage = Url{QString("https://www.frankerfacez.com/emoticon/%1-%2") .arg(id.string) .arg(name.string)}; emotes[name] = cachedOrMake(std::move(emote), id); } } return emotes; } } // namespace FfzEmotes::FfzEmotes() : global_(std::make_shared()) { } std::shared_ptr FfzEmotes::emotes() const { return this->global_.get(); } boost::optional FfzEmotes::emote(const EmoteName &name) const { auto emotes = this->global_.get(); auto it = emotes->find(name); if (it != emotes->end()) return it->second; return boost::none; } void FfzEmotes::loadEmotes() { QString url("https://api.frankerfacez.com/v1/set/global"); NetworkRequest(url) .timeout(30000) .onSuccess([this](auto result) -> Outcome { auto emotes = this->emotes(); auto pair = parseGlobalEmotes(result.parseJson(), *emotes); if (pair.first) this->global_.set( std::make_shared(std::move(pair.second))); return pair.first; }) .execute(); } void FfzEmotes::loadChannel( std::weak_ptr channel, const QString &channelId, std::function emoteCallback, std::function)> modBadgeCallback, std::function)> vipBadgeCallback, bool manualRefresh) { qCDebug(chatterinoFfzemotes) << "[FFZEmotes] Reload FFZ Channel Emotes for channel" << channelId; NetworkRequest("https://api.frankerfacez.com/v1/room/id/" + channelId) .timeout(20000) .onSuccess([emoteCallback = std::move(emoteCallback), modBadgeCallback = std::move(modBadgeCallback), vipBadgeCallback = std::move(vipBadgeCallback), channel, manualRefresh](auto result) -> Outcome { auto json = result.parseJson(); auto emoteMap = parseChannelEmotes(json); auto modBadge = parseAuthorityBadge( json.value("room").toObject().value("mod_urls").toObject(), "Moderator"); auto vipBadge = parseAuthorityBadge( json.value("room").toObject().value("vip_badge").toObject(), "VIP"); bool hasEmotes = !emoteMap.empty(); emoteCallback(std::move(emoteMap)); modBadgeCallback(std::move(modBadge)); vipBadgeCallback(std::move(vipBadge)); if (auto shared = channel.lock(); manualRefresh) { if (hasEmotes) { shared->addMessage(makeSystemMessage( "FrankerFaceZ channel emotes reloaded.")); } else { shared->addMessage( makeSystemMessage(CHANNEL_HAS_NO_EMOTES)); } } return Success; }) .onError([channelId, channel, manualRefresh](NetworkResult result) { auto shared = channel.lock(); if (!shared) return; if (result.status() == 404) { // User does not have any FFZ emotes if (manualRefresh) shared->addMessage( makeSystemMessage(CHANNEL_HAS_NO_EMOTES)); } else if (result.status() == NetworkResult::timedoutStatus) { // TODO: Auto retry in case of a timeout, with a delay qCWarning(chatterinoFfzemotes) << "Fetching FFZ emotes for channel" << channelId << "failed due to timeout"; shared->addMessage( makeSystemMessage("Failed to fetch FrankerFaceZ channel " "emotes. (timed out)")); } else { qCWarning(chatterinoFfzemotes) << "Error fetching FFZ emotes for channel" << channelId << ", error" << result.status(); shared->addMessage( makeSystemMessage("Failed to fetch FrankerFaceZ channel " "emotes. (unknown error)")); } }) .execute(); } } // namespace chatterino