Added result code to ignore/unignore calls

Add ignore/unignore calls that take the user ID to avoid double api
calls
Fully implement the account popup ignore/unignore feature
Fix #247
This commit is contained in:
Rasmus Karlsson 2018-05-13 17:53:24 +02:00
parent d4c31309f0
commit 7bc63ba38f
6 changed files with 260 additions and 122 deletions

View file

@ -130,7 +130,7 @@ QString CommandController::execCommand(const QString &text, ChannelPtr channel,
return ""; return "";
} }
user->ignore(target, [channel](const QString &message) { user->ignore(target, [channel](auto resultCode, const QString &message) {
channel->addMessage(messages::Message::createSystemMessage(message)); channel->addMessage(messages::Message::createSystemMessage(message));
}); });
@ -147,7 +147,7 @@ QString CommandController::execCommand(const QString &text, ChannelPtr channel,
return ""; return "";
} }
user->unignore(target, [channel](const QString &message) { user->unignore(target, [channel](auto resultCode, const QString &message) {
channel->addMessage(messages::Message::createSystemMessage(message)); channel->addMessage(messages::Message::createSystemMessage(message));
}); });

View file

@ -3,6 +3,7 @@
#include "const.hpp" #include "const.hpp"
#include "debug/log.hpp" #include "debug/log.hpp"
#include "util/networkrequest.hpp" #include "util/networkrequest.hpp"
#include "util/rapidjson-helpers.hpp"
#include "util/urlfetch.hpp" #include "util/urlfetch.hpp"
namespace chatterino { namespace chatterino {
@ -118,9 +119,16 @@ void TwitchAccount::loadIgnores()
} }
void TwitchAccount::ignore(const QString &targetName, void TwitchAccount::ignore(const QString &targetName,
std::function<void(const QString &message)> onFinished) std::function<void(IgnoreResult, const QString &)> onFinished)
{ {
util::twitch::getUserID(targetName, QThread::currentThread(), [=](QString targetUserID) { util::twitch::getUserID(targetName, QThread::currentThread(), [=](QString targetUserID) {
this->ignoreByID(targetUserID, targetName, onFinished); //
});
}
void TwitchAccount::ignoreByID(const QString &targetUserID, const QString &targetName,
std::function<void(IgnoreResult, const QString &)> onFinished)
{
QString url("https://api.twitch.tv/kraken/users/" + this->getUserId() + "/blocks/" + QString url("https://api.twitch.tv/kraken/users/" + this->getUserId() + "/blocks/" +
targetUserID); targetUserID);
@ -130,21 +138,22 @@ void TwitchAccount::ignore(const QString &targetName,
req.makeAuthorizedV5(this->getOAuthClient(), this->getOAuthToken()); req.makeAuthorizedV5(this->getOAuthClient(), this->getOAuthToken());
req.onError([=](int errorCode) { req.onError([=](int errorCode) {
onFinished("An unknown error occured while trying to ignore user " + targetName + " (" + onFinished(IgnoreResult_Failed, "An unknown error occured while trying to ignore user " +
QString::number(errorCode) + ")"); targetName + " (" + QString::number(errorCode) + ")");
return true; return true;
}); });
req.onSuccess([=](const rapidjson::Document &document) { req.onSuccess([=](const rapidjson::Document &document) {
if (!document.IsObject()) { if (!document.IsObject()) {
onFinished("Bad JSON data while ignoring user " + targetName); onFinished(IgnoreResult_Failed, "Bad JSON data while ignoring user " + targetName);
return false; return false;
} }
auto userIt = document.FindMember("user"); auto userIt = document.FindMember("user");
if (userIt == document.MemberEnd()) { if (userIt == document.MemberEnd()) {
onFinished("Bad JSON data while ignoring user (missing user) " + targetName); onFinished(IgnoreResult_Failed,
"Bad JSON data while ignoring user (missing user) " + targetName);
return false; return false;
} }
@ -156,23 +165,31 @@ void TwitchAccount::ignore(const QString &targetName,
if (!res.second) { if (!res.second) {
const TwitchUser &existingUser = *(res.first); const TwitchUser &existingUser = *(res.first);
existingUser.update(ignoredUser); existingUser.update(ignoredUser);
onFinished("User " + targetName + " is already ignored"); onFinished(IgnoreResult_AlreadyIgnored,
"User " + targetName + " is already ignored");
return false; return false;
} }
} }
onFinished("Successfully ignored user " + targetName); onFinished(IgnoreResult_Success, "Successfully ignored user " + targetName);
return true; return true;
}); });
req.execute(); req.execute();
});
} }
void TwitchAccount::unignore(const QString &targetName, void TwitchAccount::unignore(const QString &targetName,
std::function<void(const QString &message)> onFinished) std::function<void(UnignoreResult, const QString &message)> onFinished)
{ {
util::twitch::getUserID(targetName, QThread::currentThread(), [=](QString targetUserID) { util::twitch::getUserID(targetName, QThread::currentThread(), [=](QString targetUserID) {
this->unignoreByID(targetUserID, targetName, onFinished); //
});
}
void TwitchAccount::unignoreByID(
const QString &targetUserID, const QString &targetName,
std::function<void(UnignoreResult, const QString &message)> onFinished)
{
QString url("https://api.twitch.tv/kraken/users/" + this->getUserId() + "/blocks/" + QString url("https://api.twitch.tv/kraken/users/" + this->getUserId() + "/blocks/" +
targetUserID); targetUserID);
@ -182,8 +199,9 @@ void TwitchAccount::unignore(const QString &targetName,
req.makeAuthorizedV5(this->getOAuthClient(), this->getOAuthToken()); req.makeAuthorizedV5(this->getOAuthClient(), this->getOAuthToken());
req.onError([=](int errorCode) { req.onError([=](int errorCode) {
onFinished("An unknown error occured while trying to unignore user " + targetName + onFinished(UnignoreResult_Failed,
" (" + QString::number(errorCode) + ")"); "An unknown error occured while trying to unignore user " + targetName + " (" +
QString::number(errorCode) + ")");
return true; return true;
}); });
@ -196,13 +214,41 @@ void TwitchAccount::unignore(const QString &targetName,
this->ignores.erase(ignoredUser); this->ignores.erase(ignoredUser);
} }
onFinished("Successfully unignored user " + targetName); onFinished(UnignoreResult_Success, "Successfully unignored user " + targetName);
return true; return true;
}); });
req.execute(); req.execute();
}
void TwitchAccount::checkFollow(const QString targetUserID,
std::function<void(FollowResult)> onFinished)
{
QString url("https://api.twitch.tv/kraken/users/" + this->getUserId() + "/follows/channels/" +
targetUserID);
util::NetworkRequest req(url);
req.setRequestType(util::NetworkRequest::GetRequest);
req.setCaller(QThread::currentThread());
req.makeAuthorizedV5(this->getOAuthClient(), this->getOAuthToken());
req.onError([=](int errorCode) {
if (errorCode == 203) {
onFinished(FollowResult_NotFollowing);
} else {
onFinished(FollowResult_Failed);
}
return true;
}); });
req.onSuccess([=](const rapidjson::Document &document) {
onFinished(FollowResult_Following);
return true;
});
req.execute();
} }
std::set<TwitchUser> TwitchAccount::getIgnores() const std::set<TwitchUser> TwitchAccount::getIgnores() const

View file

@ -9,6 +9,24 @@
#include <set> #include <set>
namespace chatterino { namespace chatterino {
enum IgnoreResult {
IgnoreResult_Success,
IgnoreResult_AlreadyIgnored,
IgnoreResult_Failed,
};
enum UnignoreResult {
UnignoreResult_Success,
UnignoreResult_Failed,
};
enum FollowResult {
FollowResult_Following,
FollowResult_NotFollowing,
FollowResult_Failed,
};
namespace providers { namespace providers {
namespace twitch { namespace twitch {
@ -37,8 +55,17 @@ public:
bool isAnon() const; bool isAnon() const;
void loadIgnores(); void loadIgnores();
void ignore(const QString &targetName, std::function<void(const QString &)> onFinished);
void unignore(const QString &targetName, std::function<void(const QString &)> onFinished); void ignore(const QString &targetName,
std::function<void(IgnoreResult, const QString &)> onFinished);
void ignoreByID(const QString &targetUserID, const QString &targetName,
std::function<void(IgnoreResult, const QString &)> onFinished);
void unignore(const QString &targetName,
std::function<void(UnignoreResult, const QString &)> onFinished);
void unignoreByID(const QString &targetUserID, const QString &targetName,
std::function<void(UnignoreResult, const QString &message)> onFinished);
void checkFollow(const QString targetUserID, std::function<void(FollowResult)> onFinished);
std::set<TwitchUser> getIgnores() const; std::set<TwitchUser> getIgnores() const;

View file

@ -372,7 +372,7 @@ private:
QObject::connect(worker, &NetworkWorker::doneUrl, this->data.caller, QObject::connect(worker, &NetworkWorker::doneUrl, this->data.caller,
[data = this->data](auto reply) mutable { [data = this->data](auto reply) mutable {
if (reply->error() != QNetworkReply::NetworkError::NoError) { if (reply->error() != QNetworkReply::NetworkError::NoError) {
// TODO: We might want to call an onError callback here data.onError(reply->error());
return; return;
} }

View file

@ -98,31 +98,50 @@ AccountPopupWidget::AccountPopupWidget(ChannelPtr _channel)
"/follows/channels/" + this->popupWidgetUser.userID); "/follows/channels/" + this->popupWidgetUser.userID);
this->ui->follow->setEnabled(false); this->ui->follow->setEnabled(false);
if (!this->relationship.following) { if (!this->relationship.isFollowing()) {
util::twitch::put(requestUrl, [this](QJsonObject obj) { util::twitch::put(requestUrl, [this](QJsonObject obj) {
qDebug() << "follows channel: " << obj; qDebug() << "follows channel: " << obj;
this->relationship.following = true; this->relationship.setFollowing(true);
emit refreshButtons(); emit refreshButtons();
}); });
} else { } else {
util::twitch::sendDelete(requestUrl, [this] { util::twitch::sendDelete(requestUrl, [this] {
this->relationship.following = false; this->relationship.setFollowing(false);
emit refreshButtons(); emit refreshButtons();
}); });
} }
}); });
QObject::connect(this->ui->ignore, &QPushButton::clicked, this, [=]() { QObject::connect(this->ui->ignore, &QPushButton::clicked, this, [=]() {
QUrl requestUrl("https://api.twitch.tv/kraken/users/" + this->loggedInUser.userID + auto currentUser = getApp()->accounts->Twitch.getCurrent();
"/blocks/" + this->popupWidgetUser.userID);
if (!this->relationship.ignoring) { if (!this->relationship.isIgnoring()) {
util::twitch::put(requestUrl, [this](auto) { currentUser->ignoreByID(this->popupWidgetUser.userID, this->popupWidgetUser.username,
this->relationship.ignoring = true; // [=](auto result, const auto &message) {
switch (result) {
case IgnoreResult_Success: {
this->relationship.setIgnoring(true);
emit refreshButtons();
} break;
case IgnoreResult_AlreadyIgnored: {
this->relationship.setIgnoring(true);
emit refreshButtons();
} break;
case IgnoreResult_Failed: {
} break;
}
}); });
} else { } else {
util::twitch::sendDelete(requestUrl, [this] { currentUser->unignoreByID(this->popupWidgetUser.userID, this->popupWidgetUser.username,
this->relationship.ignoring = false; // [=](auto result, const auto &message) {
switch (result) {
case UnignoreResult_Success: {
this->relationship.setIgnoring(false);
emit refreshButtons();
} break;
case UnignoreResult_Failed: {
} break;
}
}); });
} }
}); });
@ -158,8 +177,7 @@ AccountPopupWidget::AccountPopupWidget(ChannelPtr _channel)
void AccountPopupWidget::setName(const QString &name) void AccountPopupWidget::setName(const QString &name)
{ {
this->relationship.following = false; this->relationship.reset();
this->relationship.ignoring = false;
this->popupWidgetUser.username = name; this->popupWidgetUser.username = name;
this->ui->lblUsername->setText(name); this->ui->lblUsername->setText(name);
@ -206,17 +224,24 @@ void AccountPopupWidget::getUserData()
this->loadAvatar(QUrl(obj.value("logo").toString())); this->loadAvatar(QUrl(obj.value("logo").toString()));
}); });
util::twitch::get("https://api.twitch.tv/kraken/users/" + this->loggedInUser.userID + auto app = getApp();
"/follows/channels/" + this->popupWidgetUser.userID, auto currentUser = app->accounts->Twitch.getCurrent();
this, [=](const QJsonObject &obj) {
this->ui->follow->setEnabled(true); currentUser->checkFollow(this->popupWidgetUser.userID, [=](auto result) {
this->relationship.following = obj.contains("channel"); this->relationship.setFollowing(result == FollowResult_Following);
emit refreshButtons(); emit refreshButtons();
}); });
// TODO: Get ignore relationship between logged in user and popup widget user and update bool isIgnoring = false;
// relationship.ignoring for (const auto &ignoredUser : currentUser->getIgnores()) {
if (this->popupWidgetUser.userID == ignoredUser.id) {
isIgnoring = true;
break;
}
}
this->relationship.setIgnoring(isIgnoring);
emit refreshButtons();
} }
void AccountPopupWidget::loadAvatar(const QUrl &avatarUrl) void AccountPopupWidget::loadAvatar(const QUrl &avatarUrl)
@ -327,28 +352,24 @@ void AccountPopupWidget::refreshLayouts()
void AccountPopupWidget::actuallyRefreshButtons() void AccountPopupWidget::actuallyRefreshButtons()
{ {
if (this->relationship.following) { if (this->relationship.isFollowingSet()) {
if (this->ui->follow->text() != "Unfollow") { if (this->relationship.isFollowing()) {
this->ui->follow->setText("Unfollow"); this->ui->follow->setText("Unfollow");
this->ui->follow->setEnabled(true);
}
} else { } else {
if (this->ui->follow->text() != "Follow") {
this->ui->follow->setText("Follow"); this->ui->follow->setText("Follow");
this->ui->follow->setEnabled(true);
}
} }
if (this->relationship.ignoring) { this->ui->follow->setEnabled(true);
if (this->ui->ignore->text() != "Unignore") { }
if (this->relationship.isIgnoringSet()) {
if (this->relationship.isIgnoring()) {
this->ui->ignore->setText("Unignore"); this->ui->ignore->setText("Unignore");
this->ui->ignore->setEnabled(true);
}
} else { } else {
if (this->ui->ignore->text() != "Ignore") {
this->ui->ignore->setText("Ignore"); this->ui->ignore->setText("Ignore");
this->ui->ignore->setEnabled(true);
} }
this->ui->ignore->setEnabled(true);
} }
} }
@ -368,8 +389,7 @@ void AccountPopupWidget::showEvent(QShowEvent *)
this->popupWidgetUser.refreshUserType(this->channel, false); this->popupWidgetUser.refreshUserType(this->channel, false);
this->ui->follow->setEnabled(false); this->ui->follow->setEnabled(false);
// XXX: Uncomment when ignore/unignore is fully implemented this->ui->ignore->setEnabled(false);
// this->ui->ignore->setEnabled(false);
this->refreshButtons(); this->refreshButtons();

View file

@ -71,8 +71,53 @@ private:
User popupWidgetUser; User popupWidgetUser;
struct { struct {
bool following = false; void reset()
bool ignoring = false; {
this->following = -1;
this->ignoring = -1;
}
bool isFollowing() const
{
return this->following == 1;
}
bool isFollowingSet() const
{
return this->following != -1;
}
void setFollowing(bool newVal)
{
if (newVal) {
this->following = 1;
} else {
this->following = 0;
}
}
bool isIgnoring() const
{
return this->ignoring == 1;
}
bool isIgnoringSet() const
{
return this->ignoring != -1;
}
void setIgnoring(bool newVal)
{
if (newVal) {
this->ignoring = 1;
} else {
this->ignoring = 0;
}
}
private:
int following = -1;
int ignoring = -1;
} relationship; } relationship;
protected: protected: