mirror-chatterino2/src/providers/ffz/FfzEmotes.cpp

165 lines
4.9 KiB
C++
Raw Normal View History

2018-06-26 14:09:39 +02:00
#include "providers/ffz/FfzEmotes.hpp"
2018-06-05 18:07:17 +02:00
2018-08-02 14:23:27 +02:00
#include <QJsonArray>
2018-07-15 14:11:46 +02:00
#include "common/NetworkRequest.hpp"
#include "common/Outcome.hpp"
2018-06-26 14:09:39 +02:00
#include "debug/Log.hpp"
#include "messages/Image.hpp"
2018-06-05 18:07:17 +02:00
namespace chatterino {
namespace {
2018-08-02 14:23:27 +02:00
Url getEmoteLink(const QJsonObject &urls, const QString &emoteScale)
2018-06-05 18:07:17 +02:00
{
auto emote = urls.value(emoteScale);
if (emote.isUndefined()) {
2018-08-02 14:23:27 +02:00
return {""};
2018-06-05 18:07:17 +02:00
}
assert(emote.isString());
2018-08-02 14:23:27 +02:00
return {"https:" + emote.toString()};
2018-06-05 18:07:17 +02:00
}
2018-08-06 21:17:03 +02:00
void fillInEmoteData(const QJsonObject &urls, const EmoteName &name,
const QString &tooltip, Emote &emoteData)
2018-06-05 18:07:17 +02:00
{
2018-08-02 14:23:27 +02:00
auto url1x = getEmoteLink(urls, "1");
auto url2x = getEmoteLink(urls, "2");
auto url3x = getEmoteLink(urls, "4");
//, code, tooltip
emoteData.name = name;
emoteData.images =
2018-08-06 21:17:03 +02:00
ImageSet{Image::fromUrl(url1x, 1), Image::fromUrl(url2x, 0.5),
Image::fromUrl(url3x, 0.25)};
2018-08-02 14:23:27 +02:00
emoteData.tooltip = {tooltip};
}
2018-08-11 17:15:17 +02:00
EmotePtr cachedOrMake(Emote &&emote, const EmoteId &id)
{
static std::unordered_map<EmoteId, std::weak_ptr<const Emote>> cache;
static std::mutex mutex;
return cachedOrMakeEmotePtr(std::move(emote), cache, mutex, id);
}
2018-08-11 14:20:53 +02:00
std::pair<Outcome, EmoteMap> parseGlobalEmotes(const QJsonObject &jsonRoot,
const EmoteMap &currentEmotes)
2018-08-02 14:23:27 +02:00
{
auto jsonSets = jsonRoot.value("sets").toObject();
2018-08-11 14:20:53 +02:00
auto emotes = EmoteMap();
2018-06-07 12:25:52 +02:00
2018-08-02 14:23:27 +02:00
for (auto jsonSet : jsonSets) {
auto jsonEmotes = jsonSet.toObject().value("emoticons").toArray();
2018-06-07 12:25:52 +02:00
2018-08-02 14:23:27 +02:00
for (auto jsonEmoteValue : jsonEmotes) {
auto jsonEmote = jsonEmoteValue.toObject();
2018-06-07 12:25:52 +02:00
2018-08-02 14:23:27 +02:00
auto name = EmoteName{jsonEmote.value("name").toString()};
auto id = EmoteId{jsonEmote.value("id").toString()};
auto urls = jsonEmote.value("urls").toObject();
2018-06-07 12:25:52 +02:00
2018-08-02 14:23:27 +02:00
auto emote = Emote();
2018-08-06 21:17:03 +02:00
fillInEmoteData(urls, name, name.string + "<br/>Global FFZ Emote",
emote);
emote.homePage =
Url{QString("https://www.frankerfacez.com/emoticon/%1-%2")
.arg(id.string)
.arg(name.string)};
2018-06-07 12:25:52 +02:00
2018-08-11 14:20:53 +02:00
emotes[name] =
cachedOrMakeEmotePtr(std::move(emote), currentEmotes);
2018-06-07 12:25:52 +02:00
}
2018-08-02 14:23:27 +02:00
}
2018-08-11 14:20:53 +02:00
return {Success, std::move(emotes)};
}
2018-08-11 17:15:17 +02:00
std::pair<Outcome, 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 urls = jsonEmote.value("urls").toObject();
Emote emote;
fillInEmoteData(urls, name, name.string + "<br/>Channel FFZ Emote",
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 {Success, std::move(emotes)};
}
2018-08-11 14:20:53 +02:00
} // namespace
FfzEmotes::FfzEmotes()
: global_(std::make_shared<EmoteMap>())
{
}
std::shared_ptr<const EmoteMap> FfzEmotes::global() const
{
return this->global_.get();
}
boost::optional<EmotePtr> FfzEmotes::global(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::loadGlobal()
{
QString url("https://api.frankerfacez.com/v1/set/global");
NetworkRequest request(url);
request.setCaller(QThread::currentThread());
request.setTimeout(30000);
request.onSuccess([this](auto result) -> Outcome {
auto emotes = this->global();
auto pair = parseGlobalEmotes(result.parseJson(), *emotes);
if (pair.first)
this->global_.set(
std::make_shared<EmoteMap>(std::move(pair.second)));
return pair.first;
});
request.execute();
2018-06-07 12:25:52 +02:00
}
2018-08-11 14:20:53 +02:00
void FfzEmotes::loadChannel(const QString &channelName,
std::function<void(EmoteMap &&)> callback)
2018-06-05 18:07:17 +02:00
{
2018-08-11 17:15:17 +02:00
log("[FFZEmotes] Reload FFZ Channel Emotes for channel %s\n", channelName);
2018-06-05 18:07:17 +02:00
2018-08-11 17:15:17 +02:00
NetworkRequest request("https://api.frankerfacez.com/v1/room/" +
channelName);
request.setCaller(QThread::currentThread());
request.setTimeout(3000);
2018-08-11 17:15:17 +02:00
request.onSuccess([callback = std::move(callback)](auto result) -> Outcome {
auto pair = parseChannelEmotes(result.parseJson());
if (pair.first) callback(std::move(pair.second));
return pair.first;
});
2018-08-02 14:23:27 +02:00
2018-08-11 17:15:17 +02:00
request.execute();
2018-06-05 18:07:17 +02:00
}
} // namespace chatterino