1
0
Fork 0
flightgear/utils/fgqcanvas/fgqcanvasimageloader.cpp
James Turner 61bf6b0f58 QtQuick-based rendering for remote canvas
Currently displayed alongside the QPainter version, while establishing
the accuracy of each mode, but this is temporary.
2017-10-29 16:44:32 +00:00

125 lines
3.6 KiB
C++

//
// Copyright (C) 2017 James Turner zakalawe@mac.com
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License as
// published by the Free Software Foundation; either version 2 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#include "fgqcanvasimageloader.h"
#include <QDebug>
#include <QNetworkAccessManager>
static FGQCanvasImageLoader* static_instance = nullptr;
class TransferSignalHolder : public QObject
{
Q_OBJECT
public:
TransferSignalHolder(QObject* pr) : QObject(pr) { }
signals:
void trigger();
};
FGQCanvasImageLoader::FGQCanvasImageLoader(QNetworkAccessManager* dl)
: m_downloader(dl)
{
}
void FGQCanvasImageLoader::onDownloadFinished()
{
QNetworkReply* reply = qobject_cast<QNetworkReply*>(sender());
QPixmap pm;
if (!pm.loadFromData(reply->readAll())) {
qWarning() << "image loading failed";
} else {
qDebug() << "did download:" << reply->property("image").toByteArray();
m_cache.insert(reply->property("image").toByteArray(), pm);
TransferSignalHolder* signalHolder = reply->findChild<TransferSignalHolder*>("holder");
if (signalHolder) {
qDebug() << "triggering image updates";
signalHolder->trigger();
}
}
m_transfers.removeOne(reply);
reply->deleteLater();
}
FGQCanvasImageLoader *FGQCanvasImageLoader::instance()
{
return static_instance;
}
void FGQCanvasImageLoader::initialise(QNetworkAccessManager *dl)
{
Q_ASSERT(static_instance == nullptr);
static_instance = new FGQCanvasImageLoader(dl);
}
void FGQCanvasImageLoader::setHost(QString hostName, int portNumber)
{
m_hostName = hostName;
m_port = portNumber;
}
QPixmap FGQCanvasImageLoader::getImage(const QByteArray &imagePath)
{
if (m_cache.contains(imagePath)) {
// cached, easy
return m_cache.value(imagePath);
}
QUrl url;
url.setScheme("http");
url.setHost(m_hostName);
url.setPort(m_port);
url.setPath("/aircraft-dir/" + imagePath);
Q_FOREACH (QNetworkReply* transfer, m_transfers) {
if (transfer->url() == url) {
return QPixmap(); // transfer already active
}
}
qDebug() << "requesting image" << url;
QNetworkReply* reply = m_downloader->get(QNetworkRequest(url));
reply->setProperty("image", imagePath);
connect(reply, &QNetworkReply::finished, this, &FGQCanvasImageLoader::onDownloadFinished);
m_transfers.append(reply);
return QPixmap();
}
void FGQCanvasImageLoader::connectToImageLoaded(const QByteArray &imagePath, QObject *receiver, const char *slot)
{
Q_FOREACH (QNetworkReply* transfer, m_transfers) {
if (transfer->property("image").toByteArray() == imagePath) {
QObject* signalHolder = transfer->findChild<QObject*>("holder");
if (!signalHolder) {
signalHolder = new TransferSignalHolder(transfer);
signalHolder->setObjectName("holder");
}
connect(signalHolder, SIGNAL(trigger()), receiver, slot);
return;
}
}
qWarning() << "no transfer active for" << imagePath;
}
#include "fgqcanvasimageloader.moc"