mirror of
https://github.com/Cockatrice/Cockatrice.git
synced 2026-04-27 07:48:01 -07:00
* Move logging from QDebug to QCDebug and introduce LoggingCategories. * Lint. * Unlint like one change. * Remove .debug category since this is autofilled by Qt and used to differentiate between QCDebug and QCWarning and QCError. * Uncomment defaults, include main category. * Make PictureLoader logging a bit more useful. * Lint...? * Address comments. * Clean up some unnecessary classes in logging statements. * Add a new message format to the logging handler. * Lint. * Lint. * Support Windows in Regex --------- Co-authored-by: Lukas Brübach <Bruebach.Lukas@bdosecurity.de> Co-authored-by: ZeldaZach <zahalpern+github@gmail.com>
122 lines
4.5 KiB
C++
122 lines
4.5 KiB
C++
#include "tapped_out_interface.h"
|
|
|
|
#include "decklist.h"
|
|
|
|
#include <QDesktopServices>
|
|
#include <QMessageBox>
|
|
#include <QNetworkAccessManager>
|
|
#include <QNetworkReply>
|
|
#include <QNetworkRequest>
|
|
#include <QRegularExpression>
|
|
#include <QUrlQuery>
|
|
|
|
TappedOutInterface::TappedOutInterface(CardDatabase &_cardDatabase, QObject *parent)
|
|
: QObject(parent), cardDatabase(_cardDatabase)
|
|
{
|
|
manager = new QNetworkAccessManager(this);
|
|
connect(manager, &QNetworkAccessManager::finished, this, &TappedOutInterface::queryFinished);
|
|
}
|
|
|
|
void TappedOutInterface::queryFinished(QNetworkReply *reply)
|
|
{
|
|
if (reply->error() != QNetworkReply::NoError) {
|
|
QMessageBox::critical(nullptr, tr("Error"), reply->errorString());
|
|
reply->deleteLater();
|
|
deleteLater();
|
|
return;
|
|
}
|
|
|
|
int httpStatus = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
if (reply->hasRawHeader("Location")) {
|
|
/*
|
|
* If the reply contains a "Location" header, a relative URL to the deck on TappedOut
|
|
* can be extracted from the header. The http status is a 302 "redirect".
|
|
*/
|
|
QString deckUrl = reply->rawHeader("Location");
|
|
qCDebug(TappedOutInterfaceLog) << "Tappedout: good reply, http status" << httpStatus << "location" << deckUrl;
|
|
QDesktopServices::openUrl("https://tappedout.net" + deckUrl);
|
|
} else {
|
|
/*
|
|
* Otherwise, the deck has not been parsed correctly. Error messages can be extracted
|
|
* from the html. Css pseudo selector for errors: $("div.alert-danger > ul > li")
|
|
*/
|
|
QString data(reply->readAll());
|
|
QStringList errorMessageList = {tr("Unable to analyze the deck.")};
|
|
|
|
static const QRegularExpression rx("<div class=\"alert alert-danger.*?<ul>(.*?)</ul>");
|
|
auto match = rx.match(data);
|
|
if (match.hasMatch()) {
|
|
QString errors = match.captured(1);
|
|
static const QRegularExpression rx2("<li>(.*?)</li>");
|
|
static const QRegularExpression rxremove("<[^>]*>");
|
|
auto matchIterator = rx2.globalMatch(errors);
|
|
while (matchIterator.hasNext()) {
|
|
auto match2 = matchIterator.next();
|
|
errorMessageList.append(match2.captured(1).remove(rxremove).simplified());
|
|
}
|
|
}
|
|
|
|
QString errorMessage = errorMessageList.join("\n");
|
|
qCDebug(TappedOutInterfaceLog) << "Tappedout: bad reply, http status" << httpStatus << "size" << data.size()
|
|
<< "message" << errorMessage;
|
|
|
|
QMessageBox::critical(nullptr, tr("Error"), errorMessage);
|
|
}
|
|
|
|
reply->deleteLater();
|
|
deleteLater();
|
|
}
|
|
|
|
void TappedOutInterface::getAnalyzeRequestData(DeckList *deck, QByteArray *data)
|
|
{
|
|
DeckList mainboard, sideboard;
|
|
copyDeckSplitMainAndSide(*deck, mainboard, sideboard);
|
|
|
|
QUrl params;
|
|
QUrlQuery urlQuery;
|
|
urlQuery.addQueryItem("name", deck->getName());
|
|
urlQuery.addQueryItem("mainboard", mainboard.writeToString_Plain(false, true));
|
|
urlQuery.addQueryItem("sideboard", sideboard.writeToString_Plain(false, true));
|
|
params.setQuery(urlQuery);
|
|
data->append(params.query(QUrl::EncodeReserved).toUtf8());
|
|
}
|
|
|
|
void TappedOutInterface::analyzeDeck(DeckList *deck)
|
|
{
|
|
QByteArray data;
|
|
getAnalyzeRequestData(deck, &data);
|
|
|
|
QNetworkRequest request(QUrl("https://tappedout.net/mtg-decks/paste/"));
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
|
|
|
|
manager->post(request, data);
|
|
}
|
|
|
|
struct CopyMainOrSide
|
|
{
|
|
CardDatabase &cardDatabase;
|
|
DeckList &mainboard, &sideboard;
|
|
|
|
CopyMainOrSide(CardDatabase &_cardDatabase, DeckList &_mainboard, DeckList &_sideboard)
|
|
: cardDatabase(_cardDatabase), mainboard(_mainboard), sideboard(_sideboard){};
|
|
|
|
void operator()(const InnerDecklistNode *node, const DecklistCardNode *card) const
|
|
{
|
|
CardInfoPtr dbCard = cardDatabase.getCard(card->getName());
|
|
if (!dbCard || dbCard->getIsToken())
|
|
return;
|
|
|
|
DecklistCardNode *addedCard;
|
|
if (node->getName() == DECK_ZONE_SIDE)
|
|
addedCard = sideboard.addCard(card->getName(), node->getName());
|
|
else
|
|
addedCard = mainboard.addCard(card->getName(), node->getName());
|
|
addedCard->setNumber(card->getNumber());
|
|
}
|
|
};
|
|
|
|
void TappedOutInterface::copyDeckSplitMainAndSide(DeckList &source, DeckList &mainboard, DeckList &sideboard)
|
|
{
|
|
CopyMainOrSide copyMainOrSide(cardDatabase, mainboard, sideboard);
|
|
source.forEachCard(copyMainOrSide);
|
|
}
|