qtdoc/examples/demos/osmbuildings/request.cpp

343 lines
12 KiB
C++

// Copyright (C) 2023 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#include "request.h"
#include <QFileInfo>
#include <QGeoPolygon>
#include <QJsonDocument>
#include <QJsonObject>
#include <QNetworkReply>
#include <QPoint>
#include <QtMath>
using namespace Qt::StringLiterals;
qreal OSMTileData::distanceTo(QPoint p) const
{
const qreal deltaX = p.x() - TileX;
const qreal deltaY = p.y() - TileY;
return qSqrt(deltaX * deltaX + deltaY * deltaY);
}
//%1 = zoom level(15 the default and only one here that seems working), %2 = x tile number, %3 = y tile number
static constexpr auto URL_OSMB_JSON = "https://983wdxn2c2.execute-api.eu-north-1.amazonaws.com/production/osmbuildingstile?z=%1&x=%2&y=%3&token=%4"_L1;
//%1 = zoom level(is dynamic), %2 = x tile number, %3 = y tile number
static constexpr auto URL_OSMB_MAP = "https://tile-a.openstreetmap.fr/hot/%1/%2/%3.png"_L1;
size_t qHash(OSMTileData data, size_t seed) noexcept
{
return qHashMulti(seed, data.TileX, data.TileY, data.ZoomLevel);
}
static QString tileKey(const OSMTileData &tile)
{
return QString::number(tile.ZoomLevel) + u',' + QString::number(tile.TileX)
+ u',' + QString::number(tile.TileY);
}
QGeoCoordinate importPosition(const QVariant &position)
{
QGeoCoordinate returnedCoordinates;
const auto positionList = position.value<QVariantList>();
if (!positionList.isEmpty()) {
returnedCoordinates.setLongitude(positionList.constFirst().toDouble());
if (positionList.size() > 1) {
returnedCoordinates.setLatitude(positionList.at(1).toDouble());
if (positionList.size() > 2)
returnedCoordinates.setAltitude(positionList.at(2).toDouble());
}
}
return returnedCoordinates;
}
QList<QGeoCoordinate> importArrayOfPositions(const QVariant &arrayOfPositions)
{
QList<QGeoCoordinate> returnedCoordinates;
const auto positionsList = arrayOfPositions.value<QVariantList>();
for (const auto &position : positionsList) { // Iterating the LineString coordinates nested arrays
QGeoCoordinate coordinate = importPosition(position);
if ( coordinate.isValid() )
returnedCoordinates.append(coordinate); // Populating the QList of coordinates
}
return returnedCoordinates;
}
QList<QList<QGeoCoordinate>> importArrayOfArrayOfPositions(const QVariant &arrayOfArrayofPositions)
{
QList<QList<QGeoCoordinate>> returnedCoordinates;
const auto positionsList = arrayOfArrayofPositions.value<QVariantList>();
for (const QVariant &position : positionsList) // Iterating the Polygon coordinates nested arrays
returnedCoordinates << importArrayOfPositions(position);
return returnedCoordinates;
}
QGeoPolygon importPolygon(const QVariantMap &inputMap)
{
QGeoPolygon returnedObject;
const QVariant valueCoordinates = inputMap.value("coordinates"_L1);
const QList<QList<QGeoCoordinate>> perimeters = importArrayOfArrayOfPositions(valueCoordinates);
for (qsizetype i = 0; i < perimeters.size(); ++i) { // Import an array of QList<QGeocoordinates>
if (i == 0)
returnedObject.setPerimeter(perimeters.at(i)); // External perimeter
else
returnedObject.addHole(perimeters.at(i)); // Inner perimeters
}
return returnedObject;
}
QVariantMap importGeometry(const QVariantMap &inputMap)
{
QVariantMap returnedObject;
const int geometryTypesLen = 1;
QString geometryTypes[] = {
"Polygon"_L1,
};
enum geoTypeSwitch {
Polygon,
};
for (int i = 0; i < geometryTypesLen; ++i) {
if (inputMap.value("type"_L1).value<QString>() == geometryTypes[i]) {
switch (i) {
case Polygon: {
returnedObject.insert("type"_L1, "Polygon"_L1);
returnedObject.insert("data"_L1, QVariant::fromValue(importPolygon(inputMap)));
break;
}
default:
break;
}
}
}
return returnedObject;
}
static QVariantList importFeatureCollection(const QVariantMap &inputMap)
{
QVariantList returnedObject;
const auto featuresList = inputMap.value("features"_L1).value<QVariantList>();
for (const QVariant &inputfeature : featuresList) {
auto inputFeatureMap = inputfeature.value<QVariantMap>();
auto singleFeatureMap = importGeometry(inputFeatureMap.value("geometry"_L1).value<QVariantMap>());
const auto importedProperties = inputFeatureMap.value("properties"_L1).value<QVariantMap>();
singleFeatureMap.insert("properties"_L1, importedProperties);
const auto it = inputFeatureMap.constFind("id"_L1);
if (it != inputFeatureMap.cend()) {
auto importedId = it.value().value<QVariant>();
singleFeatureMap.insert("id"_L1, importedId);
}
returnedObject.append(singleFeatureMap);
}
return returnedObject;
}
static QVariantList importGeoJson(const QJsonDocument &geoJson)
{
QVariantList returnedList;
QJsonObject object = geoJson.object(); // Read json object from imported doc
QVariantMap rootGeoJsonObject = object.toVariantMap(); // Extraced map using Qt's API
const int geometryTypesLen = 3;
QString geoType[] = {
"Polygon"_L1,
"Feature"_L1,
"FeatureCollection"_L1
};
enum geoTypeSwitch {
Polygon,
Feature,
FeatureCollection
};
QVariantMap parsedGeoJsonMap;
// Checking whether the JSON object has a "type" member
const QVariant keyVariant = rootGeoJsonObject.value("type"_L1);
auto valueType = keyVariant.value<QString>();
// Checking whether the "type" member has a GeoJSON admitted value
for (int i = 0; i < geometryTypesLen; ++i) {
if (valueType == geoType[i]) {
switch (i) {
case Polygon: {
QGeoPolygon poly = importPolygon(rootGeoJsonObject);
QVariant dataNodeValue = QVariant::fromValue(poly);
parsedGeoJsonMap.insert("type"_L1, "Polygon"_L1);
parsedGeoJsonMap.insert("data"_L1, dataNodeValue);
break;
}
// Single GeoJson geometry object with properties
case Feature: {
parsedGeoJsonMap = importGeometry(rootGeoJsonObject.value("geometry"_L1).value<QVariantMap>());
auto importedProperties = rootGeoJsonObject.value("properties"_L1).value<QVariantMap>();
parsedGeoJsonMap.insert("properties"_L1, importedProperties);
const auto it = rootGeoJsonObject.constFind("id"_L1);
if (it != rootGeoJsonObject.cend()){
auto importedId = it.value().value<QVariant>();
parsedGeoJsonMap.insert("id"_L1, importedId);
}
break;
}
// Heterogeneous list of GeoJSON geometries with properties
case FeatureCollection: {
QVariantList featCollection = importFeatureCollection(rootGeoJsonObject);
QVariant dataNodeValue = QVariant::fromValue(featCollection);
parsedGeoJsonMap.insert("type"_L1, "FeatureCollection"_L1);
parsedGeoJsonMap.insert("data"_L1, dataNodeValue);
break;
}
default:
break;
}
QVariant bboxNodeValue = rootGeoJsonObject.value("bbox"_L1);
if (bboxNodeValue.isValid()) {
parsedGeoJsonMap.insert("bbox"_L1, bboxNodeValue);
}
returnedList.append(parsedGeoJsonMap);
} else if (i >= 9) {
// Error
break;
}
}
return returnedList;
}
OSMRequest::OSMRequest(QObject *parent)
: QObject{parent}
{
connect( &m_queuesTimer, &QTimer::timeout, this, [this](){
if ( m_buildingsQueue.isEmpty() && m_mapsQueue.isEmpty() ) {
m_queuesTimer.stop();
}
else {
#ifdef QT_DEBUG
const int numConcurrentRequests = 1;
#else
const int numConcurrentRequests = 6;
#endif
if ( !m_buildingsQueue.isEmpty() && m_buildingsNumberOfRequestsInFlight < numConcurrentRequests ) {
getBuildingsDataRequest(m_buildingsQueue.dequeue());
++m_buildingsNumberOfRequestsInFlight;
}
if ( !m_mapsQueue.isEmpty() && m_mapsNumberOfRequestsInFlight < numConcurrentRequests ) {
getMapsDataRequest(m_mapsQueue.dequeue());
++m_mapsNumberOfRequestsInFlight;
}
}
});
m_queuesTimer.setInterval(0);
}
bool OSMRequest::isDemoToken() const
{
return m_token.isEmpty();
}
QString OSMRequest::token() const
{
return m_token;
}
void OSMRequest::setToken(const QString &token)
{
m_token = token;
}
void OSMRequest::getBuildingsData(const QQueue<OSMTileData> &buildingsQueue) {
if ( buildingsQueue.isEmpty() )
return;
m_buildingsQueue = buildingsQueue;
if ( !m_queuesTimer.isActive() )
m_queuesTimer.start();
}
void OSMRequest::getBuildingsDataRequest(const OSMTileData &tile)
{
const QString fileName = "data/"_L1 + tileKey(tile) + ".json"_L1;
QFileInfo file(fileName);
if ( file.size() > 0 ) {
QFile file(fileName);
if (file.open(QFile::ReadOnly)){
QByteArray data = file.readAll();
file.close();
emit buildingsDataReady( importGeoJson(QJsonDocument::fromJson( data )), tile.TileX, tile.TileY, tile.ZoomLevel );
--m_buildingsNumberOfRequestsInFlight;
return;
}
}
QUrl url = QUrl(QString(URL_OSMB_JSON).arg(QString::number(tile.ZoomLevel),
QString::number(tile.TileX),
QString::number(tile.TileY),
m_token));
QNetworkReply * reply = m_networkAccessManager.get( QNetworkRequest(url));
connect( reply, &QNetworkReply::finished, this, [this, reply, tile](){
reply->deleteLater();
if ( reply->error() == QNetworkReply::NoError ) {
QByteArray data = reply->readAll();
emit buildingsDataReady( importGeoJson(QJsonDocument::fromJson( data )), tile.TileX, tile.TileY, tile.ZoomLevel );
} else {
const QByteArray message = reply->readAll();
static QByteArray lastMessage;
if (message != lastMessage) {
lastMessage = message;
qWarning().noquote() << "OSMRequest::getBuildingsData " << reply->error()
<< reply->url() << message;
}
}
--m_buildingsNumberOfRequestsInFlight;
} );
}
void OSMRequest::getMapsData(const QQueue<OSMTileData> &mapsQueue) {
if ( mapsQueue.isEmpty() )
return;
m_mapsQueue = mapsQueue;
if ( !m_queuesTimer.isActive() )
m_queuesTimer.start();
}
void OSMRequest::getMapsDataRequest(const OSMTileData &tile)
{
const QString fileName = "data/"_L1 + tileKey(tile) + ".png"_L1;
QFileInfo file(fileName);
if ( file.size() > 0) {
QFile file(fileName);
if (file.open(QFile::ReadOnly)){
QByteArray data = file.readAll();
file.close();
emit mapsDataReady( data, tile.TileX, tile.TileY, tile.ZoomLevel );
--m_mapsNumberOfRequestsInFlight;
return;
}
}
QUrl url = QUrl(QString(URL_OSMB_MAP).arg(QString::number(tile.ZoomLevel),
QString::number(tile.TileX),
QString::number(tile.TileY)));
QNetworkReply * reply = m_networkAccessManager.get( QNetworkRequest(url));
connect( reply, &QNetworkReply::finished, this, [this, reply, tile](){
reply->deleteLater();
if ( reply->error() == QNetworkReply::NoError ) {
QByteArray data = reply->readAll();
emit mapsDataReady( data, tile.TileX, tile.TileY, tile.ZoomLevel );
} else {
const QByteArray message = reply->readAll();
static QByteArray lastMessage;
if (message != lastMessage) {
lastMessage = message;
qWarning().noquote() << "OSMRequest::getMapsDataRequest" << reply->error()
<< reply->url() << message;
}
}
--m_mapsNumberOfRequestsInFlight;
} );
}