/**************************************************************************** ** ** Copyright (C) 2016 Aaron McCarthy ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtLocation module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qgeocodereplyosm.h" #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE QGeoCodeReplyOsm::QGeoCodeReplyOsm(QNetworkReply *reply, bool includeExtraData, QObject *parent) : QGeoCodeReply(*new QGeoCodeReplyOsmPrivate, parent), m_includeExtraData(includeExtraData) { if (!reply) { setError(UnknownError, QStringLiteral("Null reply")); return; } connect(reply, SIGNAL(finished()), this, SLOT(networkReplyFinished())); connect(reply, SIGNAL(error(QNetworkReply::NetworkError)), this, SLOT(networkReplyError(QNetworkReply::NetworkError))); connect(this, &QGeoCodeReply::aborted, reply, &QNetworkReply::abort); connect(this, &QObject::destroyed, reply, &QObject::deleteLater); setLimit(1); setOffset(0); } QGeoCodeReplyOsm::~QGeoCodeReplyOsm() { } static QGeoAddress parseAddressObject(const QJsonObject &object) { QGeoAddress address; address.setText(object.value(QStringLiteral("display_name")).toString()); QJsonObject ao = object.value(QStringLiteral("address")).toObject(); // setCountry address.setCountry(ao.value(QStringLiteral("country")).toString()); // setCountryCode address.setCountryCode(ao.value(QStringLiteral("country_code")).toString()); // setState address.setState(ao.value(QStringLiteral("state")).toString()); // setCity if (ao.contains(QLatin1String("city"))) address.setCity(ao.value(QStringLiteral("city")).toString()); else if (ao.contains(QLatin1String("town"))) address.setCity(ao.value(QLatin1String("town")).toString()); else if (ao.contains(QLatin1String("village"))) address.setCity(ao.value(QLatin1String("village")).toString()); else address.setCity(ao.value(QLatin1String("hamlet")).toString()); // setDistrict address.setDistrict(ao.value(QStringLiteral("suburb")).toString()); // setPostalCode address.setPostalCode(ao.value(QStringLiteral("postcode")).toString()); // setStreet address.setStreet(ao.value(QStringLiteral("road")).toString()); return address; } static void injectExtra(QGeoLocation &location, const QJsonObject &object) { QVariantMap extra; static const QList extraKeys = { QStringLiteral("geojson"), QStringLiteral("icon"), QStringLiteral("importance"), QStringLiteral("type"), QStringLiteral("osm_id"), QStringLiteral("osm_type"), QStringLiteral("licence"), QStringLiteral("place_id"), QStringLiteral("class") }; for (const auto k: extraKeys) { if (object.contains(k)) { extra[k] = object.value(k).toVariant(); if (k == QStringLiteral("geojson")) extra[QStringLiteral("geojson_model")] = QGeoJson::importGeoJson(QJsonDocument::fromVariant(extra[k])); } } location.setExtendedAttributes(extra); } void QGeoCodeReplyOsm::networkReplyFinished() { QNetworkReply *reply = static_cast(sender()); reply->deleteLater(); if (reply->networkError() != QNetworkReply::NoError) return; QList locations; QJsonDocument document = QJsonDocument::fromJson(reply->readAll()); if (document.isObject()) { QJsonObject object = document.object(); QGeoCoordinate coordinate; coordinate.setLatitude(object.value(QStringLiteral("lat")).toString().toDouble()); coordinate.setLongitude(object.value(QStringLiteral("lon")).toString().toDouble()); QGeoLocation location; location.setCoordinate(coordinate); location.setAddress(parseAddressObject(object)); if (m_includeExtraData) injectExtra(location, object); locations.append(location); setLocations(locations); } else if (document.isArray()) { QJsonArray results = document.array(); for (int i = 0; i < results.count(); ++i) { if (!results.at(i).isObject()) continue; QJsonObject object = results.at(i).toObject(); QGeoCoordinate coordinate; coordinate.setLatitude(object.value(QStringLiteral("lat")).toString().toDouble()); coordinate.setLongitude(object.value(QStringLiteral("lon")).toString().toDouble()); QGeoRectangle rectangle; if (object.contains(QStringLiteral("boundingbox"))) { QJsonArray a = object.value(QStringLiteral("boundingbox")).toArray(); if (a.count() == 4) { rectangle.setTopLeft(QGeoCoordinate(a.at(1).toString().toDouble(), a.at(2).toString().toDouble())); rectangle.setBottomRight(QGeoCoordinate(a.at(0).toString().toDouble(), a.at(3).toString().toDouble())); } } QGeoLocation location; location.setCoordinate(coordinate); location.setBoundingBox(rectangle); location.setAddress(parseAddressObject(object)); if (m_includeExtraData) injectExtra(location, object); locations.append(location); } } setLocations(locations); setFinished(true); } void QGeoCodeReplyOsm::networkReplyError(QNetworkReply::NetworkError error) { Q_UNUSED(error); QNetworkReply *reply = static_cast(sender()); reply->deleteLater(); setError(QGeoCodeReply::CommunicationError, reply->errorString()); } QGeoCodeReplyOsmPrivate::QGeoCodeReplyOsmPrivate() { } QGeoCodeReplyOsmPrivate::~QGeoCodeReplyOsmPrivate() { } QVariantMap QGeoCodeReplyOsmPrivate::extraData() const { return m_extraData; } QT_END_NAMESPACE