2019-09-24 16:08:12 +02:00
|
|
|
#include "NuulsUploader.hpp"
|
|
|
|
|
|
|
|
#include "common/Env.hpp"
|
|
|
|
#include "common/NetworkRequest.hpp"
|
|
|
|
#include "providers/twitch/TwitchMessageBuilder.hpp"
|
|
|
|
|
2019-09-24 18:28:28 +02:00
|
|
|
#include <QHttpMultiPart>
|
2019-09-24 16:08:12 +02:00
|
|
|
|
|
|
|
namespace chatterino {
|
|
|
|
bool isUploading = false;
|
|
|
|
|
|
|
|
std::queue<TypedBytes> uploadQueue;
|
2019-09-24 18:28:28 +02:00
|
|
|
|
2019-09-24 16:08:12 +02:00
|
|
|
void uploadImageToNuuls(TypedBytes imageData, ChannelPtr channel,
|
|
|
|
ResizingTextEdit &textEdit)
|
|
|
|
{
|
|
|
|
const char *boundary = "thisistheboudaryasd";
|
2019-09-25 12:51:17 +02:00
|
|
|
static QUrl url(Env::get().imageUploaderUrl);
|
2019-09-24 16:08:12 +02:00
|
|
|
|
2019-09-24 18:28:28 +02:00
|
|
|
QHttpMultiPart *payload = new QHttpMultiPart(QHttpMultiPart::FormDataType);
|
|
|
|
QHttpPart part = QHttpPart();
|
|
|
|
part.setBody(imageData.data);
|
|
|
|
part.setHeader(QNetworkRequest::ContentTypeHeader,
|
|
|
|
QString("image/%1").arg(imageData.type));
|
|
|
|
part.setHeader(QNetworkRequest::ContentLengthHeader,
|
|
|
|
QVariant(imageData.data.length()));
|
|
|
|
part.setHeader(
|
|
|
|
QNetworkRequest::ContentDispositionHeader,
|
|
|
|
QString("form-data; name=\"attachment\"; filename=\"control_v.%1\""));
|
|
|
|
payload->setBoundary(boundary);
|
|
|
|
payload->append(part);
|
2019-09-24 16:08:12 +02:00
|
|
|
NetworkRequest(url, NetworkRequestType::Post)
|
|
|
|
.header("Content-Type", (std::string("multipart/form-data; boundary=") +
|
|
|
|
std::string(boundary))
|
|
|
|
.c_str())
|
|
|
|
|
2019-09-24 18:28:28 +02:00
|
|
|
.multiPart(payload)
|
2019-09-24 16:08:12 +02:00
|
|
|
.onSuccess([&textEdit, channel](NetworkResult result) -> Outcome {
|
|
|
|
textEdit.insertPlainText(result.getData() + QString(" "));
|
2019-09-25 22:21:26 +02:00
|
|
|
if (uploadQueue.empty())
|
|
|
|
{
|
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
QString("Your image has been uploaded.")));
|
|
|
|
}
|
|
|
|
else
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
QString("Your image has been uploaded. %1 left. Please "
|
|
|
|
"wait until all of them are uploaded. About %2 "
|
|
|
|
"seconds left.")
|
|
|
|
.arg(uploadQueue.size())
|
|
|
|
.arg(uploadQueue.size() * 3)));
|
|
|
|
// Argument number 2 is the ETA.
|
|
|
|
// 2 seconds for the timer that's there not to spam Nuuls' server
|
|
|
|
// and 1 second of actual uploading.
|
|
|
|
}
|
|
|
|
isUploading = false;
|
2019-09-25 22:21:26 +02:00
|
|
|
if (uploadQueue.size())
|
|
|
|
{
|
|
|
|
QTimer::singleShot(2000, [channel, &textEdit]() {
|
2019-09-24 16:08:12 +02:00
|
|
|
uploadImageToNuuls(uploadQueue.front(), channel, textEdit);
|
|
|
|
uploadQueue.pop();
|
2019-09-25 22:21:26 +02:00
|
|
|
});
|
|
|
|
}
|
2019-09-24 16:08:12 +02:00
|
|
|
return Success;
|
|
|
|
})
|
|
|
|
.onError([channel](NetworkResult result) -> bool {
|
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
QString("An error happened while uploading your image: %1")
|
|
|
|
.arg(result.status())));
|
|
|
|
isUploading = false;
|
|
|
|
return true;
|
|
|
|
})
|
|
|
|
.execute();
|
|
|
|
}
|
|
|
|
|
2019-09-25 22:21:26 +02:00
|
|
|
void upload(const QMimeData *source, ChannelPtr channel,
|
|
|
|
ResizingTextEdit &outputTextEdit)
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
|
|
|
if (isUploading)
|
|
|
|
{
|
|
|
|
channel->addMessage(makeSystemMessage(
|
2019-09-25 22:21:26 +02:00
|
|
|
QString("Please wait until the upload finishes.")));
|
2019-09-24 16:08:12 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
isUploading = true;
|
|
|
|
channel->addMessage(makeSystemMessage(QString("Started upload...")));
|
|
|
|
|
|
|
|
if (source->hasFormat("image/png"))
|
|
|
|
{
|
2019-09-24 18:28:28 +02:00
|
|
|
uploadImageToNuuls({source->data("image/png"), "png"}, channel,
|
|
|
|
outputTextEdit);
|
2019-09-24 16:08:12 +02:00
|
|
|
}
|
2019-09-25 22:21:26 +02:00
|
|
|
else if (source->hasFormat("image/jpeg"))
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
2019-09-25 22:21:26 +02:00
|
|
|
uploadImageToNuuls({source->data("image/jpeg"), "jpeg"}, channel,
|
|
|
|
outputTextEdit);
|
|
|
|
}
|
|
|
|
else if (source->hasFormat("image/gif"))
|
|
|
|
{
|
|
|
|
uploadImageToNuuls({source->data("image/gif"), "gif"}, channel,
|
|
|
|
outputTextEdit);
|
|
|
|
}
|
|
|
|
else if (source->hasUrls())
|
|
|
|
{
|
|
|
|
for (QUrl path : source->urls())
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
2019-09-25 22:21:26 +02:00
|
|
|
if (getImageFileFormat(path.toLocalFile()) != QString())
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
2019-09-25 22:21:26 +02:00
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
QString("Uploading image: %1").arg(path.toLocalFile())));
|
|
|
|
QImage img = QImage(path.toLocalFile());
|
|
|
|
if (img.isNull())
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
2019-09-25 22:21:26 +02:00
|
|
|
channel->addMessage(
|
|
|
|
makeSystemMessage(QString("Couldn't load image :(")));
|
|
|
|
return;
|
2019-09-24 16:08:12 +02:00
|
|
|
}
|
2019-09-25 22:21:26 +02:00
|
|
|
QByteArray imageData;
|
|
|
|
QBuffer buf(&imageData);
|
|
|
|
buf.open(QIODevice::WriteOnly);
|
|
|
|
img.save(&buf, "png");
|
|
|
|
|
|
|
|
TypedBytes data = {imageData, "png"};
|
|
|
|
uploadQueue.push(data);
|
|
|
|
}
|
|
|
|
else if (path.toLocalFile().endsWith(".gif"))
|
|
|
|
{
|
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
QString("Uploading GIF: %1").arg(path.toLocalFile())));
|
|
|
|
QFile file(path.toLocalFile());
|
|
|
|
bool isOkay = file.open(QIODevice::ReadOnly);
|
|
|
|
if (!isOkay)
|
2019-09-24 16:08:12 +02:00
|
|
|
{
|
2019-09-25 22:21:26 +02:00
|
|
|
channel->addMessage(
|
|
|
|
makeSystemMessage(QString("Failed to open file. :(")));
|
|
|
|
return;
|
2019-09-24 16:08:12 +02:00
|
|
|
}
|
2019-09-25 22:21:26 +02:00
|
|
|
TypedBytes data = {file.readAll(), "gif"};
|
|
|
|
uploadQueue.push(data);
|
|
|
|
file.close();
|
|
|
|
// file.readAll() => might be a bit big but it /should/ work
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
channel->addMessage(makeSystemMessage(
|
|
|
|
QString("Cannot upload file: %1, not an image")
|
|
|
|
.arg(path.toLocalFile())));
|
2019-09-24 16:08:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (uploadQueue.size())
|
|
|
|
{
|
|
|
|
uploadImageToNuuls(uploadQueue.front(), channel, outputTextEdit);
|
|
|
|
uploadQueue.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // not PNG, try loading it into QImage and save it to a PNG.
|
|
|
|
QImage image = qvariant_cast<QImage>(source->imageData());
|
|
|
|
QByteArray imageData;
|
|
|
|
QBuffer buf(&imageData);
|
|
|
|
buf.open(QIODevice::WriteOnly);
|
|
|
|
image.save(&buf, "png");
|
|
|
|
|
2019-09-24 18:28:28 +02:00
|
|
|
uploadImageToNuuls({imageData, "png"}, channel, outputTextEdit);
|
2019-09-24 16:08:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} // namespace chatterino
|
2019-09-25 22:21:26 +02:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
QString getImageFileFormat(QString path)
|
|
|
|
{
|
|
|
|
static QStringList listOfImageFormats = {".png", ".jpg", ".jpeg"};
|
|
|
|
for (const QString &format : listOfImageFormats)
|
|
|
|
{
|
|
|
|
if (path.endsWith(format))
|
|
|
|
{
|
|
|
|
return format.mid(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
} // namespace
|