mirror-chatterino2/src/providers/twitch/TwitchChannel.cpp

797 lines
23 KiB
C++
Raw Normal View History

2018-06-26 14:09:39 +02:00
#include "providers/twitch/TwitchChannel.hpp"
#include "Application.hpp"
2018-06-26 15:33:51 +02:00
#include "common/Common.hpp"
2018-07-15 14:11:46 +02:00
#include "common/NetworkRequest.hpp"
#include "controllers/accounts/AccountController.hpp"
2018-08-12 15:29:40 +02:00
#include "controllers/notifications/NotificationController.hpp"
2018-06-26 14:09:39 +02:00
#include "debug/Log.hpp"
#include "messages/Message.hpp"
2018-08-11 17:15:17 +02:00
#include "providers/bttv/BttvEmotes.hpp"
2018-08-02 14:23:27 +02:00
#include "providers/bttv/LoadBttvChannelEmote.hpp"
2018-07-06 19:23:47 +02:00
#include "providers/twitch/PubsubClient.hpp"
#include "providers/twitch/TwitchCommon.hpp"
2018-06-26 14:09:39 +02:00
#include "providers/twitch/TwitchMessageBuilder.hpp"
2018-08-02 14:23:27 +02:00
#include "providers/twitch/TwitchParseCheerEmotes.hpp"
2018-06-28 19:46:45 +02:00
#include "singletons/Emotes.hpp"
#include "singletons/Settings.hpp"
2018-08-12 15:29:40 +02:00
#include "singletons/Toasts.hpp"
#include "singletons/WindowManager.hpp"
2018-06-26 14:09:39 +02:00
#include "util/PostToThread.hpp"
2018-08-29 19:25:37 +02:00
#include "widgets/Window.hpp"
2018-02-05 15:11:50 +01:00
#include <IrcConnection>
#include <QJsonArray>
2018-08-02 14:23:27 +02:00
#include <QJsonObject>
#include <QJsonValue>
#include <QThread>
#include <QTimer>
namespace chatterino {
2018-08-10 18:56:17 +02:00
namespace {
auto parseRecentMessages(const QJsonObject &jsonRoot, ChannelPtr channel)
2018-08-15 22:46:20 +02:00
{
QJsonArray jsonMessages = jsonRoot.value("messages").toArray();
std::vector<MessagePtr> messages;
2018-10-21 13:43:02 +02:00
if (jsonMessages.empty())
return messages;
2018-08-15 22:46:20 +02:00
2018-10-21 13:43:02 +02:00
for (const auto jsonMessage : jsonMessages)
{
2018-08-15 22:46:20 +02:00
auto content = jsonMessage.toString().toUtf8();
// passing nullptr as the channel makes the message invalid but we
// don't check for that anyways
auto message = Communi::IrcMessage::fromData(content, nullptr);
auto privMsg = dynamic_cast<Communi::IrcPrivateMessage *>(message);
assert(privMsg);
MessageParseArgs args;
TwitchMessageBuilder builder(channel.get(), privMsg, args);
if (getSettings()->greyOutHistoricMessages)
builder.message().flags.set(MessageFlag::Disabled);
2018-10-21 14:44:59 +02:00
2018-10-21 13:43:02 +02:00
if (!builder.isIgnored())
2018-08-15 22:46:20 +02:00
messages.push_back(builder.build());
2018-08-10 18:56:17 +02:00
}
2018-08-15 22:46:20 +02:00
return messages;
}
std::pair<Outcome, UsernameSet> parseChatters(const QJsonObject &jsonRoot)
{
static QStringList categories = {"moderators", "staff", "admins",
"global_mods", "viewers"};
2018-08-13 13:54:39 +02:00
2018-08-15 22:46:20 +02:00
auto usernames = UsernameSet();
2018-08-13 13:54:39 +02:00
2018-08-15 22:46:20 +02:00
// parse json
QJsonObject jsonCategories = jsonRoot.value("chatters").toObject();
2018-10-21 13:43:02 +02:00
for (const auto &category : categories)
{
for (auto jsonCategory : jsonCategories.value(category).toArray())
{
2018-08-15 22:46:20 +02:00
usernames.insert(jsonCategory.toString());
}
2018-08-13 13:54:39 +02:00
}
2018-08-15 22:46:20 +02:00
return {Success, std::move(usernames)};
}
2018-08-10 18:56:17 +02:00
} // namespace
2018-08-14 17:45:17 +02:00
TwitchChannel::TwitchChannel(const QString &name,
TwitchBadges &globalTwitchBadges, BttvEmotes &bttv,
FfzEmotes &ffz)
2018-08-02 14:23:27 +02:00
: Channel(name, Channel::Type::Twitch)
2018-07-15 20:28:54 +02:00
, subscriptionUrl_("https://www.twitch.tv/subs/" + name)
, channelUrl_("https://twitch.tv/" + name)
, popoutPlayerUrl_("https://player.twitch.tv/?channel=" + name)
2018-08-14 17:45:17 +02:00
, globalTwitchBadges_(globalTwitchBadges)
, globalBttv_(bttv)
, globalFfz_(ffz)
, bttvEmotes_(std::make_shared<EmoteMap>())
, ffzEmotes_(std::make_shared<EmoteMap>())
2018-10-25 21:53:03 +02:00
, ffzCustomModBadge_(name)
2018-07-06 19:23:47 +02:00
, mod_(false)
{
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:{}] Opened", name);
2018-08-19 15:09:00 +02:00
this->liveStatusChanged.connect([this]() {
2018-10-21 13:43:02 +02:00
if (this->isLive() == 1)
{
2018-08-19 15:09:00 +02:00
}
});
2018-07-14 14:24:18 +02:00
this->managedConnect(getApp()->accounts->twitch.currentUserChanged,
[=] { this->setMod(false); });
2018-07-14 14:24:18 +02:00
// pubsub
this->managedConnect(getApp()->accounts->twitch.currentUserChanged,
[=] { this->refreshPubsub(); });
this->refreshPubsub();
2018-08-13 13:54:39 +02:00
this->userStateChanged.connect([this] { this->refreshPubsub(); });
2018-07-14 14:24:18 +02:00
// room id loaded -> refresh live status
2018-08-02 14:23:27 +02:00
this->roomIdChanged.connect([this]() {
this->refreshPubsub();
this->refreshLiveStatus();
2018-08-13 13:54:39 +02:00
this->refreshBadges();
this->refreshCheerEmotes();
2018-08-02 14:23:27 +02:00
});
2018-07-14 14:24:18 +02:00
// timers
QObject::connect(&this->chattersListTimer_, &QTimer::timeout,
2018-08-13 13:54:39 +02:00
[=] { this->refreshChatters(); });
2018-07-14 14:24:18 +02:00
this->chattersListTimer_.start(5 * 60 * 1000);
2018-08-06 21:17:03 +02:00
QObject::connect(&this->liveStatusTimer_, &QTimer::timeout,
[=] { this->refreshLiveStatus(); });
2018-07-14 14:24:18 +02:00
this->liveStatusTimer_.start(60 * 1000);
2018-01-17 18:36:12 +01:00
2018-07-14 14:24:18 +02:00
// --
2018-07-06 19:23:47 +02:00
this->messageSuffix_.append(' ');
this->messageSuffix_.append(QChar(0x206D));
2018-07-14 14:24:18 +02:00
// debugging
2018-06-19 20:34:50 +02:00
#if 0
for (int i = 0; i < 1000; i++) {
2018-08-13 13:54:39 +02:00
this->addMessage(makeSystemMessage("asef"));
2018-06-19 20:34:50 +02:00
}
#endif
}
2018-08-13 13:54:39 +02:00
void TwitchChannel::initialize()
{
this->refreshChatters();
this->refreshChannelEmotes();
2018-10-25 21:53:03 +02:00
this->ffzCustomModBadge_.loadCustomModBadge();
2018-08-13 13:54:39 +02:00
}
bool TwitchChannel::isEmpty() const
{
2018-08-02 14:23:27 +02:00
return this->getName().isEmpty();
}
bool TwitchChannel::canSendMessage() const
{
2017-12-31 00:50:07 +01:00
return !this->isEmpty();
}
2018-07-14 14:24:18 +02:00
void TwitchChannel::refreshChannelEmotes()
{
2018-08-11 17:15:17 +02:00
BttvEmotes::loadChannel(
2018-08-06 21:17:03 +02:00
this->getName(), [this, weak = weakOf<Channel>(this)](auto &&emoteMap) {
2018-08-11 17:15:17 +02:00
if (auto shared = weak.lock())
this->bttvEmotes_.set(
std::make_shared<EmoteMap>(std::move(emoteMap)));
2018-08-06 21:17:03 +02:00
});
2018-08-11 17:15:17 +02:00
FfzEmotes::loadChannel(
2018-08-06 21:17:03 +02:00
this->getName(), [this, weak = weakOf<Channel>(this)](auto &&emoteMap) {
if (auto shared = weak.lock())
2018-08-11 17:15:17 +02:00
this->ffzEmotes_.set(
std::make_shared<EmoteMap>(std::move(emoteMap)));
2018-08-06 21:17:03 +02:00
});
}
void TwitchChannel::sendMessage(const QString &message)
{
auto app = getApp();
2017-12-17 02:18:13 +01:00
2018-10-21 13:43:02 +02:00
if (!app->accounts->twitch.isLoggedIn())
{
2018-08-06 21:17:03 +02:00
// XXX: It would be nice if we could add a link here somehow that opened
// the "account manager" dialog
2018-08-07 01:35:24 +02:00
this->addMessage(
makeSystemMessage("You need to log in to send messages. You can "
"link your Twitch account in the settings."));
return;
}
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:{}] Send message: {}", this->getName(), message);
// Do last message processing
2018-06-05 18:53:49 +02:00
QString parsedMessage = app->emotes->emojis.replaceShortCodes(message);
parsedMessage = parsedMessage.trimmed();
2018-10-21 13:43:02 +02:00
if (parsedMessage.isEmpty())
{
return;
2018-02-05 15:11:50 +01:00
}
2018-10-21 13:43:02 +02:00
if (!this->hasModRights())
{
if (getSettings()->allowDuplicateMessages)
{
if (parsedMessage == this->lastSentMessage_)
{
2018-07-06 19:23:47 +02:00
parsedMessage.append(this->messageSuffix_);
}
}
}
2018-06-06 18:57:22 +02:00
bool messageSent = false;
2018-08-02 14:23:27 +02:00
this->sendMessageSignal.invoke(this->getName(), parsedMessage, messageSent);
2018-02-11 21:13:23 +01:00
2018-10-21 13:43:02 +02:00
if (messageSent)
{
2018-06-06 18:57:22 +02:00
qDebug() << "sent";
2018-07-06 19:23:47 +02:00
this->lastSentMessage_ = parsedMessage;
2018-06-06 18:57:22 +02:00
}
}
bool TwitchChannel::isMod() const
2018-01-17 17:17:26 +01:00
{
2018-07-06 19:23:47 +02:00
return this->mod_;
2018-01-17 17:17:26 +01:00
}
2018-01-17 18:36:12 +01:00
void TwitchChannel::setMod(bool value)
{
2018-10-21 13:43:02 +02:00
if (this->mod_ != value)
{
2018-07-06 19:23:47 +02:00
this->mod_ = value;
2018-01-17 18:36:12 +01:00
this->userStateChanged.invoke();
2018-01-17 18:36:12 +01:00
}
}
bool TwitchChannel::isBroadcaster() const
2018-01-17 17:17:26 +01:00
{
auto app = getApp();
2018-08-02 14:23:27 +02:00
return this->getName() == app->accounts->twitch.getCurrent()->getUserName();
2018-01-17 17:17:26 +01:00
}
2018-08-07 01:35:24 +02:00
void TwitchChannel::addRecentChatter(const MessagePtr &message)
{
2018-08-13 14:38:03 +02:00
this->chatters_.access()->insert(message->displayName);
}
void TwitchChannel::addJoinedUser(const QString &user)
{
2018-07-15 20:28:54 +02:00
auto app = getApp();
2018-05-26 20:26:25 +02:00
if (user == app->accounts->twitch.getCurrent()->getUserName() ||
2018-10-21 13:43:02 +02:00
!getSettings()->showJoins.getValue())
{
return;
}
2018-07-15 20:28:54 +02:00
auto joinedUsers = this->joinedUsers_.access();
joinedUsers->append(user);
2018-10-21 13:43:02 +02:00
if (!this->joinedUsersMergeQueued_)
{
2018-07-06 19:23:47 +02:00
this->joinedUsersMergeQueued_ = true;
2018-07-15 20:28:54 +02:00
QTimer::singleShot(500, &this->lifetimeGuard_, [this] {
auto joinedUsers = this->joinedUsers_.access();
2018-08-07 01:35:24 +02:00
MessageBuilder builder(systemMessage,
"Users joined: " + joinedUsers->join(", "));
2018-08-07 07:55:31 +02:00
builder->flags.set(MessageFlag::Collapsed);
2018-07-15 20:28:54 +02:00
joinedUsers->clear();
2018-08-07 01:35:24 +02:00
this->addMessage(builder.release());
2018-07-06 19:23:47 +02:00
this->joinedUsersMergeQueued_ = false;
});
}
}
void TwitchChannel::addPartedUser(const QString &user)
{
2018-07-14 14:24:18 +02:00
auto app = getApp();
2018-05-26 20:26:25 +02:00
if (user == app->accounts->twitch.getCurrent()->getUserName() ||
2018-10-21 13:43:02 +02:00
!getSettings()->showJoins.getValue())
{
return;
}
2018-07-15 20:28:54 +02:00
auto partedUsers = this->partedUsers_.access();
partedUsers->append(user);
2018-10-21 13:43:02 +02:00
if (!this->partedUsersMergeQueued_)
{
2018-07-06 19:23:47 +02:00
this->partedUsersMergeQueued_ = true;
2018-07-15 20:28:54 +02:00
QTimer::singleShot(500, &this->lifetimeGuard_, [this] {
auto partedUsers = this->partedUsers_.access();
2018-08-07 01:35:24 +02:00
MessageBuilder builder(systemMessage,
"Users parted: " + partedUsers->join(", "));
2018-08-07 07:55:31 +02:00
builder->flags.set(MessageFlag::Collapsed);
2018-08-07 01:35:24 +02:00
this->addMessage(builder.release());
2018-07-15 20:28:54 +02:00
partedUsers->clear();
2018-07-06 19:23:47 +02:00
this->partedUsersMergeQueued_ = false;
});
}
}
2018-08-11 17:15:17 +02:00
QString TwitchChannel::roomId() const
2018-07-14 14:24:18 +02:00
{
2018-08-10 19:00:14 +02:00
return *this->roomID_.access();
2018-07-14 14:24:18 +02:00
}
2018-07-15 20:28:54 +02:00
void TwitchChannel::setRoomId(const QString &id)
2018-07-14 14:24:18 +02:00
{
2018-10-21 13:43:02 +02:00
if (*this->roomID_.accessConst() != id)
{
*this->roomID_.access() = id;
this->roomIdChanged.invoke();
this->loadRecentMessages();
}
2018-07-14 14:24:18 +02:00
}
2018-08-06 21:17:03 +02:00
AccessGuard<const TwitchChannel::RoomModes> TwitchChannel::accessRoomModes()
const
2018-05-24 08:58:34 +02:00
{
2018-08-06 18:25:47 +02:00
return this->roomModes_.accessConst();
2018-05-24 08:58:34 +02:00
}
void TwitchChannel::setRoomModes(const RoomModes &_roomModes)
{
2018-07-15 20:28:54 +02:00
this->roomModes_ = _roomModes;
2018-05-24 08:58:34 +02:00
this->roomModesChanged.invoke();
}
bool TwitchChannel::isLive() const
{
2018-07-15 20:28:54 +02:00
return this->streamStatus_.access()->live;
}
2018-08-06 21:17:03 +02:00
AccessGuard<const TwitchChannel::StreamStatus>
2018-08-15 22:46:20 +02:00
TwitchChannel::accessStreamStatus() const
2018-07-15 20:28:54 +02:00
{
2018-08-06 18:25:47 +02:00
return this->streamStatus_.accessConst();
2018-07-15 20:28:54 +02:00
}
2018-08-13 13:54:39 +02:00
AccessGuard<const UsernameSet> TwitchChannel::accessChatters() const
{
return this->chatters_.accessConst();
}
2018-08-14 17:45:17 +02:00
const TwitchBadges &TwitchChannel::globalTwitchBadges() const
{
return this->globalTwitchBadges_;
}
const BttvEmotes &TwitchChannel::globalBttv() const
{
return this->globalBttv_;
}
const FfzEmotes &TwitchChannel::globalFfz() const
2018-08-02 14:23:27 +02:00
{
return this->globalFfz_;
}
2018-08-11 17:15:17 +02:00
boost::optional<EmotePtr> TwitchChannel::bttvEmote(const EmoteName &name) const
2018-08-02 14:23:27 +02:00
{
2018-08-11 17:15:17 +02:00
auto emotes = this->bttvEmotes_.get();
2018-08-02 14:23:27 +02:00
auto it = emotes->find(name);
2018-10-21 13:43:02 +02:00
if (it == emotes->end())
return boost::none;
2018-08-02 14:23:27 +02:00
return it->second;
}
2018-08-11 17:15:17 +02:00
boost::optional<EmotePtr> TwitchChannel::ffzEmote(const EmoteName &name) const
2018-07-15 20:28:54 +02:00
{
2018-08-13 13:54:39 +02:00
auto emotes = this->ffzEmotes_.get();
2018-08-02 14:23:27 +02:00
auto it = emotes->find(name);
2018-10-21 13:43:02 +02:00
if (it == emotes->end())
return boost::none;
2018-08-02 14:23:27 +02:00
return it->second;
2018-07-15 20:28:54 +02:00
}
2018-08-11 17:15:17 +02:00
std::shared_ptr<const EmoteMap> TwitchChannel::bttvEmotes() const
2018-07-15 20:28:54 +02:00
{
2018-08-11 17:15:17 +02:00
return this->bttvEmotes_.get();
2018-05-24 08:58:34 +02:00
}
2018-08-11 17:15:17 +02:00
std::shared_ptr<const EmoteMap> TwitchChannel::ffzEmotes() const
2018-05-24 08:58:34 +02:00
{
2018-08-11 17:15:17 +02:00
return this->ffzEmotes_.get();
2018-07-15 20:28:54 +02:00
}
2018-08-11 17:15:17 +02:00
const QString &TwitchChannel::subscriptionUrl()
2018-07-15 20:28:54 +02:00
{
return this->subscriptionUrl_;
}
2018-08-11 17:15:17 +02:00
const QString &TwitchChannel::channelUrl()
2018-07-15 20:28:54 +02:00
{
return this->channelUrl_;
}
2018-08-11 17:15:17 +02:00
const QString &TwitchChannel::popoutPlayerUrl()
2018-07-15 20:28:54 +02:00
{
return this->popoutPlayerUrl_;
2018-05-24 08:58:34 +02:00
}
void TwitchChannel::setLive(bool newLiveStatus)
{
bool gotNewLiveStatus = false;
{
2018-07-15 20:28:54 +02:00
auto guard = this->streamStatus_.access();
2018-10-21 13:43:02 +02:00
if (guard->live != newLiveStatus)
{
gotNewLiveStatus = true;
2018-10-21 13:43:02 +02:00
if (newLiveStatus)
{
if (getApp()->notifications->isChannelNotified(
2018-10-21 13:43:02 +02:00
this->getName(), Platform::Twitch))
{
if (Toasts::isEnabled())
{
getApp()->toasts->sendChannelNotification(
this->getName(), Platform::Twitch);
}
2018-10-21 13:43:02 +02:00
if (getSettings()->notificationPlaySound)
{
getApp()->notifications->playSound();
}
2018-10-21 13:43:02 +02:00
if (getSettings()->notificationFlashTaskbar)
{
2018-10-07 12:55:44 +02:00
getApp()->windows->sendAlert();
}
}
auto live = makeSystemMessage(this->getName() + " is live");
this->addMessage(live);
2018-10-21 13:43:02 +02:00
}
else
{
auto offline =
makeSystemMessage(this->getName() + " is offline");
this->addMessage(offline);
2018-08-12 18:54:32 +02:00
}
2018-07-15 20:28:54 +02:00
guard->live = newLiveStatus;
}
}
2018-10-21 13:43:02 +02:00
if (gotNewLiveStatus)
{
2018-07-14 14:24:18 +02:00
this->liveStatusChanged.invoke();
}
}
void TwitchChannel::refreshLiveStatus()
{
2018-08-11 17:15:17 +02:00
auto roomID = this->roomId();
2018-07-14 14:24:18 +02:00
2018-10-21 13:43:02 +02:00
if (roomID.isEmpty())
{
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:{}] Refreshing live status (Missing ID)",
2018-08-06 21:17:03 +02:00
this->getName());
this->setLive(false);
return;
}
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:{}] Refreshing live status", this->getName());
2018-07-14 14:24:18 +02:00
QString url("https://api.twitch.tv/kraken/streams/" + roomID);
//<<<<<<< HEAD
// auto request = makeGetStreamRequest(roomID, QThread::currentThread());
//=======
2018-07-23 15:12:14 +02:00
auto request = NetworkRequest::twitchRequest(url);
request.setCaller(QThread::currentThread());
//>>>>>>> 9bfbdefd2f0972a738230d5b95a009f73b1dd933
2018-08-06 21:17:03 +02:00
request.onSuccess(
[this, weak = weakOf<Channel>(this)](auto result) -> Outcome {
2018-08-06 21:17:03 +02:00
ChannelPtr shared = weak.lock();
2018-10-21 13:43:02 +02:00
if (!shared)
return Failure;
2018-08-06 21:17:03 +02:00
return this->parseLiveStatus(result.parseRapidJson());
});
2018-07-15 20:28:54 +02:00
2018-07-23 15:12:14 +02:00
request.execute();
2018-07-15 20:28:54 +02:00
}
2018-08-02 14:23:27 +02:00
Outcome TwitchChannel::parseLiveStatus(const rapidjson::Document &document)
2018-07-15 20:28:54 +02:00
{
2018-10-21 13:43:02 +02:00
if (!document.IsObject())
{
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:refreshLiveStatus] root is not an object");
2018-08-02 14:23:27 +02:00
return Failure;
2018-07-15 20:28:54 +02:00
}
2018-10-21 13:43:02 +02:00
if (!document.HasMember("stream"))
{
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:refreshLiveStatus] Missing stream in root");
2018-08-02 14:23:27 +02:00
return Failure;
2018-07-15 20:28:54 +02:00
}
2018-07-15 20:28:54 +02:00
const auto &stream = document["stream"];
2018-10-21 13:43:02 +02:00
if (!stream.IsObject())
{
2018-07-15 20:28:54 +02:00
// Stream is offline (stream is most likely null)
this->setLive(false);
2018-08-02 14:23:27 +02:00
return Failure;
2018-07-15 20:28:54 +02:00
}
2018-08-06 21:17:03 +02:00
if (!stream.HasMember("viewers") || !stream.HasMember("game") ||
2018-10-21 13:43:02 +02:00
!stream.HasMember("channel") || !stream.HasMember("created_at"))
{
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:refreshLiveStatus] Missing members in stream");
2018-07-15 20:28:54 +02:00
this->setLive(false);
2018-08-02 14:23:27 +02:00
return Failure;
2018-07-15 20:28:54 +02:00
}
2018-07-15 20:28:54 +02:00
const rapidjson::Value &streamChannel = stream["channel"];
2018-10-21 13:43:02 +02:00
if (!streamChannel.IsObject() || !streamChannel.HasMember("status"))
{
2018-08-11 14:20:53 +02:00
log("[TwitchChannel:refreshLiveStatus] Missing member \"status\" in "
2018-08-06 21:17:03 +02:00
"channel");
2018-08-02 14:23:27 +02:00
return Failure;
2018-07-15 20:28:54 +02:00
}
2018-07-15 20:28:54 +02:00
// Stream is live
2018-07-15 20:28:54 +02:00
{
auto status = this->streamStatus_.access();
status->viewerCount = stream["viewers"].GetUint();
status->game = stream["game"].GetString();
status->title = streamChannel["status"].GetString();
2018-08-06 21:17:03 +02:00
QDateTime since = QDateTime::fromString(
stream["created_at"].GetString(), Qt::ISODate);
2018-07-15 20:28:54 +02:00
auto diff = since.secsTo(QDateTime::currentDateTime());
2018-08-06 21:17:03 +02:00
status->uptime = QString::number(diff / 3600) + "h " +
QString::number(diff % 3600 / 60) + "m";
2018-07-15 20:28:54 +02:00
status->rerun = false;
2018-10-21 13:43:02 +02:00
if (stream.HasMember("stream_type"))
{
2018-07-15 20:28:54 +02:00
status->streamType = stream["stream_type"].GetString();
2018-10-21 13:43:02 +02:00
}
else
{
2018-07-15 20:28:54 +02:00
status->streamType = QString();
}
2018-10-21 13:43:02 +02:00
if (stream.HasMember("broadcast_platform"))
{
2018-07-15 20:28:54 +02:00
const auto &broadcastPlatformValue = stream["broadcast_platform"];
2018-10-21 13:43:02 +02:00
if (broadcastPlatformValue.IsString())
{
2018-08-06 21:17:03 +02:00
const char *broadcastPlatform =
stream["broadcast_platform"].GetString();
2018-10-21 13:43:02 +02:00
if (strcmp(broadcastPlatform, "rerun") == 0)
{
2018-07-15 20:28:54 +02:00
status->rerun = true;
}
}
}
2018-07-15 20:28:54 +02:00
}
2018-08-12 18:54:32 +02:00
setLive(true);
2018-07-15 20:28:54 +02:00
// Signal all listeners that the stream status has been updated
this->liveStatusChanged.invoke();
2018-08-02 14:23:27 +02:00
return Success;
}
2018-07-14 14:24:18 +02:00
void TwitchChannel::loadRecentMessages()
{
static QString genericURL =
2018-08-06 21:17:03 +02:00
"https://tmi.twitch.tv/api/rooms/%1/recent_messages?client_id=" +
getDefaultClientID();
2018-08-11 17:15:17 +02:00
NetworkRequest request(genericURL.arg(this->roomId()));
request.makeAuthorizedV5(getDefaultClientID());
request.setCaller(QThread::currentThread());
2018-08-10 18:56:17 +02:00
// can't be concurrent right now due to SignalVector
// request.setExecuteConcurrently(true);
request.onSuccess([weak = weakOf<Channel>(this)](auto result) -> Outcome {
auto shared = weak.lock();
2018-10-21 13:43:02 +02:00
if (!shared)
return Failure;
2018-07-14 14:24:18 +02:00
auto messages = parseRecentMessages(result.parseJson(), shared);
shared->addMessagesAtStart(messages);
2018-07-14 14:24:18 +02:00
2018-08-10 18:56:17 +02:00
return Success;
});
2018-08-10 18:56:17 +02:00
request.execute();
2018-07-14 14:24:18 +02:00
}
2018-07-14 14:24:18 +02:00
void TwitchChannel::refreshPubsub()
{
// listen to moderation actions
2018-10-21 13:43:02 +02:00
if (!this->hasModRights())
return;
2018-08-11 17:15:17 +02:00
auto roomId = this->roomId();
2018-10-21 13:43:02 +02:00
if (roomId.isEmpty())
return;
2018-07-14 14:24:18 +02:00
auto account = getApp()->accounts->twitch.getCurrent();
2018-08-06 21:17:03 +02:00
getApp()->twitch2->pubsub->listenToChannelModerationActions(roomId,
account);
2018-07-14 14:24:18 +02:00
}
2018-08-13 13:54:39 +02:00
void TwitchChannel::refreshChatters()
2018-07-14 14:24:18 +02:00
{
// setting?
2018-07-15 20:28:54 +02:00
const auto streamStatus = this->accessStreamStatus();
2018-07-14 14:24:18 +02:00
2018-10-21 13:43:02 +02:00
if (getSettings()->onlyFetchChattersForSmallerStreamers)
{
2018-08-06 21:17:03 +02:00
if (streamStatus->live &&
2018-10-21 13:43:02 +02:00
streamStatus->viewerCount > getSettings()->smallStreamerLimit)
{
2018-07-14 14:24:18 +02:00
return;
}
2018-07-14 14:24:18 +02:00
}
2018-07-14 14:24:18 +02:00
// get viewer list
2018-08-06 21:17:03 +02:00
NetworkRequest request("https://tmi.twitch.tv/group/user/" +
this->getName() + "/chatters");
2018-07-14 14:24:18 +02:00
request.setCaller(QThread::currentThread());
2018-08-06 21:17:03 +02:00
request.onSuccess(
[this, weak = weakOf<Channel>(this)](auto result) -> Outcome {
2018-08-06 21:17:03 +02:00
// channel still exists?
auto shared = weak.lock();
2018-10-21 13:43:02 +02:00
if (!shared)
return Failure;
2018-07-14 14:24:18 +02:00
2018-08-13 13:54:39 +02:00
auto pair = parseChatters(result.parseJson());
2018-10-21 13:43:02 +02:00
if (pair.first)
{
2018-08-13 13:54:39 +02:00
*this->chatters_.access() = std::move(pair.second);
}
return pair.first;
2018-08-06 21:17:03 +02:00
});
request.execute();
}
2018-08-13 13:54:39 +02:00
void TwitchChannel::refreshBadges()
2018-08-02 14:23:27 +02:00
{
2018-08-06 21:17:03 +02:00
auto url = Url{"https://badges.twitch.tv/v1/badges/channels/" +
2018-08-11 17:15:17 +02:00
this->roomId() + "/display?language=en"};
2018-08-02 14:23:27 +02:00
NetworkRequest req(url.string);
req.setCaller(QThread::currentThread());
req.onSuccess([this, weak = weakOf<Channel>(this)](auto result) -> Outcome {
auto shared = weak.lock();
2018-10-21 13:43:02 +02:00
if (!shared)
return Failure;
2018-08-02 14:23:27 +02:00
auto badgeSets = this->badgeSets_.access();
auto jsonRoot = result.parseJson();
auto _ = jsonRoot["badge_sets"].toObject();
2018-08-06 21:17:03 +02:00
for (auto jsonBadgeSet = _.begin(); jsonBadgeSet != _.end();
2018-10-21 13:43:02 +02:00
jsonBadgeSet++)
{
2018-08-02 14:23:27 +02:00
auto &versions = (*badgeSets)[jsonBadgeSet.key()];
auto _ = jsonBadgeSet->toObject()["versions"].toObject();
2018-08-06 21:17:03 +02:00
for (auto jsonVersion_ = _.begin(); jsonVersion_ != _.end();
2018-10-21 13:43:02 +02:00
jsonVersion_++)
{
2018-08-02 14:23:27 +02:00
auto jsonVersion = jsonVersion_->toObject();
2018-08-06 21:17:03 +02:00
auto emote = std::make_shared<Emote>(Emote{
EmoteName{},
ImageSet{
Image::fromUrl({jsonVersion["image_url_1x"].toString()},
1),
Image::fromUrl({jsonVersion["image_url_2x"].toString()},
.5),
Image::fromUrl({jsonVersion["image_url_4x"].toString()},
.25)},
Tooltip{jsonVersion["description"].toString()},
2018-08-06 21:17:03 +02:00
Url{jsonVersion["clickURL"].toString()}});
2018-08-02 14:23:27 +02:00
versions.emplace(jsonVersion_.key(), emote);
};
}
return Success;
});
req.execute();
}
2018-08-13 13:54:39 +02:00
void TwitchChannel::refreshCheerEmotes()
2018-08-02 14:23:27 +02:00
{
2018-08-10 18:56:17 +02:00
/*auto url = Url{"https://api.twitch.tv/kraken/bits/actions?channel_id=" +
2018-08-06 21:17:03 +02:00
this->getRoomId()};
2018-08-02 14:23:27 +02:00
auto request = NetworkRequest::twitchRequest(url.string);
request.setCaller(QThread::currentThread());
2018-08-06 21:17:03 +02:00
request.onSuccess(
[this, weak = weakOf<Channel>(this)](auto result) -> Outcome {
auto cheerEmoteSets = ParseCheermoteSets(result.parseRapidJson());
2018-08-10 18:56:17 +02:00
std::vector<CheerEmoteSet> emoteSets;
2018-08-06 21:17:03 +02:00
for (auto &set : cheerEmoteSets) {
auto cheerEmoteSet = CheerEmoteSet();
cheerEmoteSet.regex = QRegularExpression(
"^" + set.prefix.toLower() + "([1-9][0-9]*)$");
for (auto &tier : set.tiers) {
CheerEmote cheerEmote;
cheerEmote.color = QColor(tier.color);
cheerEmote.minBits = tier.minBits;
// TODO(pajlada): We currently hardcode dark here :|
// We will continue to do so for now since we haven't had to
// solve that anywhere else
cheerEmote.animatedEmote = std::make_shared<Emote>(
Emote{EmoteName{"cheer emote"},
ImageSet{
tier.images["dark"]["animated"]["1"],
tier.images["dark"]["animated"]["2"],
tier.images["dark"]["animated"]["4"],
},
Tooltip{}, Url{}});
cheerEmote.staticEmote = std::make_shared<Emote>(
Emote{EmoteName{"cheer emote"},
ImageSet{
tier.images["dark"]["static"]["1"],
tier.images["dark"]["static"]["2"],
tier.images["dark"]["static"]["4"],
},
Tooltip{}, Url{}});
cheerEmoteSet.cheerEmotes.emplace_back(cheerEmote);
}
2018-08-02 14:23:27 +02:00
2018-08-06 21:17:03 +02:00
std::sort(cheerEmoteSet.cheerEmotes.begin(),
cheerEmoteSet.cheerEmotes.end(),
[](const auto &lhs, const auto &rhs) {
return lhs.minBits < rhs.minBits; //
});
2018-08-02 14:23:27 +02:00
2018-08-10 18:56:17 +02:00
emoteSets.emplace_back(cheerEmoteSet);
2018-08-06 21:17:03 +02:00
}
2018-08-10 18:56:17 +02:00
*this->cheerEmoteSets_.access() = std::move(emoteSets);
2018-08-02 14:23:27 +02:00
2018-08-06 21:17:03 +02:00
return Success;
});
2018-08-02 14:23:27 +02:00
request.execute();
2018-08-10 18:56:17 +02:00
*/
2018-08-02 14:23:27 +02:00
}
2018-08-13 13:54:39 +02:00
boost::optional<EmotePtr> TwitchChannel::twitchBadge(
2018-08-06 21:17:03 +02:00
const QString &set, const QString &version) const
2018-08-02 14:23:27 +02:00
{
auto badgeSets = this->badgeSets_.access();
auto it = badgeSets->find(set);
2018-10-21 13:43:02 +02:00
if (it != badgeSets->end())
{
2018-08-02 14:23:27 +02:00
auto it2 = it->second.find(version);
2018-10-21 13:43:02 +02:00
if (it2 != it->second.end())
{
2018-08-02 14:23:27 +02:00
return it2->second;
}
}
return boost::none;
2018-07-14 14:24:18 +02:00
}
2018-10-25 21:53:03 +02:00
boost::optional<EmotePtr> TwitchChannel::ffzCustomModBadge() const
{
if (auto badge = this->ffzCustomModBadge_.badge())
return badge;
return boost::none;
}
} // namespace chatterino