/**************************************************************************** ** ** Copyright (C) 2017 Mapbox, Inc. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtFoo 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 "qplacesearchreplymapbox.h" #include "qplacemanagerenginemapbox.h" #include "qmapboxcommon.h" #include #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE namespace { // https://www.mapbox.com/api-documentation/#response-object QPlaceResult parsePlaceResult(const QJsonObject &response, const QString &attribution) { QPlace place; place.setAttribution(attribution); place.setPlaceId(response.value(QStringLiteral("id")).toString()); place.setVisibility(QLocation::PublicVisibility); QString placeName = response.value(QStringLiteral("text")).toString(); if (placeName.isEmpty()) placeName = response.value(QStringLiteral("place_name")).toString(); place.setName(placeName); place.setDetailsFetched(true); // Unused data: type, place_type, relevance, properties.short_code, // properties.landmark, properties.wikidata // The property object is unstable and only Carmen GeoJSON properties are // guaranteed. This implementation should check for the presence of these // values in a response before it attempts to use them. if (response.value(QStringLiteral("properties")).isObject()) { const QJsonObject properties = response.value(QStringLiteral("properties")).toObject(); const QString makiString = properties.value(QStringLiteral("maki")).toString(); if (!makiString.isEmpty()) { QVariantMap iconParameters; iconParameters.insert(QPlaceIcon::SingleUrl, QUrl::fromLocalFile(QStringLiteral(":/mapbox/") + makiString + QStringLiteral(".svg"))); QPlaceIcon icon; icon.setParameters(iconParameters); place.setIcon(icon); } const QString phoneString = properties.value(QStringLiteral("tel")).toString(); if (!phoneString.isEmpty()) { QPlaceContactDetail phoneDetail; phoneDetail.setLabel(QPlaceContactDetail::Phone); phoneDetail.setValue(phoneString); place.setContactDetails(QPlaceContactDetail::Phone, QList() << phoneDetail); } const QString categoryString = properties.value(QStringLiteral("category")).toString(); if (!categoryString.isEmpty()) { QList categories; for (const QString &categoryId : categoryString.split(QStringLiteral(", "), QString::SkipEmptyParts)) { QPlaceCategory category; category.setName(QMapboxCommon::mapboxNameForCategory(categoryId)); category.setCategoryId(categoryId); categories.append(category); } place.setCategories(categories); } } // XXX: matching_text, matching_place_name // XXX: text_{language}, place_name_{language} // XXX: language, language_{language} place.setLocation(QMapboxCommon::parseGeoLocation(response)); // XXX: geometry, geometry.type, geometry.coordinates, geometry.interpolated QPlaceResult result; result.setPlace(place); result.setTitle(place.name()); return result; } } // namespace QPlaceSearchReplyMapbox::QPlaceSearchReplyMapbox(const QPlaceSearchRequest &request, QNetworkReply *reply, QPlaceManagerEngineMapbox *parent) : QPlaceSearchReply(parent) { Q_ASSERT(parent); if (!reply) { setError(UnknownError, QStringLiteral("Null reply")); return; } setRequest(request); connect(reply, &QNetworkReply::finished, this, &QPlaceSearchReplyMapbox::onReplyFinished); connect(reply, QOverload::of(&QNetworkReply::error), this, &QPlaceSearchReplyMapbox::onNetworkError); connect(this, &QPlaceReply::aborted, reply, &QNetworkReply::abort); connect(this, &QObject::destroyed, reply, &QObject::deleteLater); } QPlaceSearchReplyMapbox::~QPlaceSearchReplyMapbox() { } void QPlaceSearchReplyMapbox::setError(QPlaceReply::Error errorCode, const QString &errorString) { QPlaceReply::setError(errorCode, errorString); emit error(errorCode, errorString); setFinished(true); emit finished(); } void QPlaceSearchReplyMapbox::onReplyFinished() { QNetworkReply *reply = static_cast(sender()); reply->deleteLater(); if (reply->error() != QNetworkReply::NoError) return; const QJsonDocument document = QJsonDocument::fromJson(reply->readAll()); if (!document.isObject()) { setError(ParseError, tr("Response parse error")); return; } const QJsonArray features = document.object().value(QStringLiteral("features")).toArray(); const QString attribution = document.object().value(QStringLiteral("attribution")).toString(); const QGeoCoordinate searchCenter = request().searchArea().center(); const QList categories = request().categories(); QList results; for (const QJsonValue &feature : features) { QPlaceResult placeResult = parsePlaceResult(feature.toObject(), attribution); if (!categories.isEmpty()) { const QList placeCategories = placeResult.place().categories(); bool categoryMatch = false; if (!placeCategories.isEmpty()) { for (const QPlaceCategory &placeCategory : placeCategories) { if (categories.contains(placeCategory)) { categoryMatch = true; break; } } } if (!categoryMatch) continue; } placeResult.setDistance(searchCenter.distanceTo(placeResult.place().location().coordinate())); results.append(placeResult); } if (request().relevanceHint() == QPlaceSearchRequest::DistanceHint) { std::sort(results.begin(), results.end(), [](const QPlaceResult &a, const QPlaceResult &b) -> bool { return a.distance() < b.distance(); }); } else if (request().relevanceHint() == QPlaceSearchRequest::LexicalPlaceNameHint) { std::sort(results.begin(), results.end(), [](const QPlaceResult &a, const QPlaceResult &b) -> bool { return a.place().name() < b.place().name(); }); } setResults(results); setFinished(true); emit finished(); } void QPlaceSearchReplyMapbox::onNetworkError(QNetworkReply::NetworkError error) { Q_UNUSED(error); QNetworkReply *reply = static_cast(sender()); reply->deleteLater(); setError(CommunicationError, reply->errorString()); } QT_END_NAMESPACE