28#include <QJsonDocument>
30#include <QJsonParseError>
35#include "moc_qgsarcgisvectortileservicedataprovider.cpp"
37using namespace Qt::StringLiterals;
41QString QgsArcGisVectorTileServiceDataProvider::ARCGIS_VT_SERVICE_DATA_PROVIDER_KEY = u
"arcgisvectortileservice"_s;
42QString QgsArcGisVectorTileServiceDataProvider::ARCGIS_VT_SERVICE_DATA_PROVIDER_DESCRIPTION = QObject::tr(
"ArcGIS Vector Tile Service data provider" );
45QgsArcGisVectorTileServiceDataProvider::QgsArcGisVectorTileServiceDataProvider(
const QString &uri,
const ProviderOptions &providerOptions,
Qgis::DataProviderReadFlags flags )
46 : QgsXyzVectorTileDataProviderBase( uri, providerOptions, flags )
48 mIsValid = setupArcgisVectorTileServiceConnection();
54 mLayerMetadata.setIdentifier( mArcgisLayerConfiguration.value( u
"serviceUri"_s ).toString() );
55 const QString parentIdentifier = mArcgisLayerConfiguration.value( u
"serviceItemId"_s ).toString();
56 if ( !parentIdentifier.isEmpty() )
58 mLayerMetadata.setParentIdentifier( parentIdentifier );
60 mLayerMetadata.setType( u
"dataset"_s );
61 mLayerMetadata.setTitle( mArcgisLayerConfiguration.value( u
"name"_s ).toString() );
62 const QString copyright = mArcgisLayerConfiguration.value( u
"copyrightText"_s ).toString();
63 if ( !copyright.isEmpty() )
64 mLayerMetadata.setRights( QStringList() << copyright );
65 mLayerMetadata.addLink(
QgsAbstractMetadataBase::Link( tr(
"Source" ), u
"WWW:LINK"_s, mArcgisLayerConfiguration.value( u
"serviceUri"_s ).toString() ) );
68QgsArcGisVectorTileServiceDataProvider::QgsArcGisVectorTileServiceDataProvider(
const QgsArcGisVectorTileServiceDataProvider &other )
69 : QgsXyzVectorTileDataProviderBase( other )
71 mIsValid = other.mIsValid;
72 mExtent = other.mExtent;
73 mMatrixSet = other.mMatrixSet;
74 mSourcePath = other.mSourcePath;
75 mArcgisLayerConfiguration = other.mArcgisLayerConfiguration;
76 mArcgisStyleConfiguration = other.mArcgisStyleConfiguration;
78 mLayerMetadata = other.mLayerMetadata;
96QString QgsArcGisVectorTileServiceDataProvider::name()
const
100 return ARCGIS_VT_SERVICE_DATA_PROVIDER_KEY;
103QString QgsArcGisVectorTileServiceDataProvider::description()
const
107 return ARCGIS_VT_SERVICE_DATA_PROVIDER_DESCRIPTION;
114 return new QgsArcGisVectorTileServiceDataProvider( *
this );
117QString QgsArcGisVectorTileServiceDataProvider::sourcePath()
const
124bool QgsArcGisVectorTileServiceDataProvider::isValid()
const
131QgsRectangle QgsArcGisVectorTileServiceDataProvider::extent()
const
152QgsLayerMetadata QgsArcGisVectorTileServiceDataProvider::layerMetadata()
const
156 return mLayerMetadata;
159QVariantMap QgsArcGisVectorTileServiceDataProvider::styleDefinition()
const
163 return mArcgisStyleConfiguration;
166QString QgsArcGisVectorTileServiceDataProvider::styleUrl()
const
171 return mArcgisLayerConfiguration.value( u
"serviceUri"_s ).toString()
172 +
'/' + mArcgisLayerConfiguration.value( u
"defaultStyles"_s ).toString();
175QString QgsArcGisVectorTileServiceDataProvider::htmlMetadata()
const
179 if ( !mTileMapUrl.isEmpty() )
180 metadata += u
"<tr><td class=\"highlight\">"_s % tr(
"Tilemap" ) % u
"</td><td><a href=\"%1\">%1</a>"_s.arg( mTileMapUrl ) % u
"</td></tr>\n"_s;
185bool QgsArcGisVectorTileServiceDataProvider::setupArcgisVectorTileServiceConnection()
191 QString tileServiceUri = dsUri.param( u
"url"_s );
193 QUrl url( tileServiceUri );
197 query.addQueryItem( u
"f"_s, u
"pjson"_s );
198 url.setQuery( query );
200 QNetworkRequest request = QNetworkRequest( url );
205 switch ( networkRequest.
get( request ) )
217 const QByteArray raw = content.
content();
221 const QJsonDocument doc = QJsonDocument::fromJson( raw, &err );
227 mArcgisLayerConfiguration = doc.object().toVariantMap();
228 if ( mArcgisLayerConfiguration.contains( u
"error"_s ) )
233 if ( !mArcgisLayerConfiguration.value( u
"tiles"_s ).isValid() )
236 const QString sourceUri = mArcgisLayerConfiguration.value( u
"sources"_s ).toMap().value( u
"esri"_s ).toMap().value( u
"url"_s ).toString();
237 if ( !sourceUri.isEmpty() )
239 QUrl url( sourceUri );
243 query.addQueryItem( u
"f"_s, u
"pjson"_s );
244 url.setQuery( query );
246 QNetworkRequest request = QNetworkRequest( url );
251 switch ( networkRequest.
get( request ) )
263 const QByteArray raw = content.
content();
267 const QJsonDocument doc = QJsonDocument::fromJson( raw, &err );
273 tileServiceUri = sourceUri;
276 mArcgisStyleConfiguration = mArcgisLayerConfiguration;
277 mArcgisLayerConfiguration = doc.object().toVariantMap();
278 if ( mArcgisLayerConfiguration.contains( u
"error"_s ) )
287 const QString tileMapEndpoint = mArcgisLayerConfiguration.value( u
"tileMap"_s ).toString();
288 if ( !tileMapEndpoint.isEmpty() )
290 mTileMapUrl = tileServiceUri +
'/' + tileMapEndpoint;
291 QUrl tilemapUrl( mTileMapUrl );
292 tilemapUrl.setQuery( query );
294 QNetworkRequest tileMapRequest = QNetworkRequest( tilemapUrl );
298 switch ( tileMapNetworkRequest.
get( tileMapRequest ) )
310 const QByteArray tileMapRaw = tileMapContent.
content();
312 const QJsonDocument tileMapDoc = QJsonDocument::fromJson( tileMapRaw, &err );
313 if ( !tileMapDoc.isNull() )
315 tileMap = tileMapDoc.object().toVariantMap();
319 mSourcePath = tileServiceUri +
'/' + mArcgisLayerConfiguration.value( u
"tiles"_s ).toList().value( 0 ).toString();
322 QgsDebugError( u
"Invalid format of URL for XYZ source: "_s + tileServiceUri );
326 mArcgisLayerConfiguration.insert( u
"serviceUri"_s, tileServiceUri );
328 mMatrixSet.fromEsriJson( mArcgisLayerConfiguration, tileMap );
329 mCrs = mMatrixSet.crs();
332 if ( dsUri.hasParam( u
"zmin"_s ) )
333 mMatrixSet.dropMatricesOutsideZoomRange( dsUri.param( u
"zmin"_s ).toInt(), 99 );
335 if ( dsUri.hasParam( u
"zmax"_s ) )
336 mMatrixSet.dropMatricesOutsideZoomRange( 0, dsUri.param( u
"zmax"_s ).toInt() );
338 const QVariantMap fullExtent = mArcgisLayerConfiguration.value( u
"fullExtent"_s ).toMap();
339 if ( !fullExtent.isEmpty() )
342 fullExtent.value( u
"xmin"_s ).toDouble(),
343 fullExtent.value( u
"ymin"_s ).toDouble(),
344 fullExtent.value( u
"xmax"_s ).toDouble(),
345 fullExtent.value( u
"ymax"_s ).toDouble()
352 mExtent = extentTransform.transformBoundingBox( fullExtentRect );
356 QgsDebugError( u
"Could not transform layer fullExtent to layer CRS"_s );
365 mExtent = extentTransform.transformBoundingBox(
QgsRectangle( -20037508.3427892, -20037508.3427892, 20037508.3427892, 20037508.3427892 ) );
369 QgsDebugError( u
"Could not transform layer extent to layer CRS"_s );
381QgsArcGisVectorTileServiceDataProviderMetadata::QgsArcGisVectorTileServiceDataProviderMetadata()
382 :
QgsProviderMetadata( QgsArcGisVectorTileServiceDataProvider::ARCGIS_VT_SERVICE_DATA_PROVIDER_KEY,
383 QgsArcGisVectorTileServiceDataProvider::ARCGIS_VT_SERVICE_DATA_PROVIDER_DESCRIPTION )
387QIcon QgsArcGisVectorTileServiceDataProviderMetadata::icon()
const
399 return new QgsArcGisVectorTileServiceDataProvider( uri, options, flags );
402QVariantMap QgsArcGisVectorTileServiceDataProviderMetadata::decodeUri(
const QString &uri )
const
407 QVariantMap uriComponents;
408 uriComponents.insert( u
"type"_s, u
"xyz"_s );
409 uriComponents.insert( u
"serviceType"_s, u
"arcgis"_s );
410 uriComponents.insert( u
"url"_s, dsUri.
param( u
"url"_s ) );
413 uriComponents.insert( u
"zmin"_s, dsUri.
param( u
"zmin"_s ) );
415 uriComponents.insert( u
"zmax"_s, dsUri.
param( u
"zmax"_s ) );
419 if ( dsUri.
hasParam( u
"styleUrl"_s ) )
420 uriComponents.insert( u
"styleUrl"_s, dsUri.
param( u
"styleUrl"_s ) );
423 if ( !authcfg.isEmpty() )
424 uriComponents.insert( u
"authcfg"_s, authcfg );
426 return uriComponents;
429QString QgsArcGisVectorTileServiceDataProviderMetadata::encodeUri(
const QVariantMap &parts )
const
432 dsUri.
setParam( u
"type"_s, u
"xyz"_s );
433 dsUri.
setParam( u
"serviceType"_s, u
"arcgis"_s );
434 dsUri.
setParam( u
"url"_s, parts.value( u
"url"_s ).toString() );
436 if ( parts.contains( u
"zmin"_s ) )
437 dsUri.
setParam( u
"zmin"_s, parts[ u
"zmin"_s ].toString() );
438 if ( parts.contains( u
"zmax"_s ) )
439 dsUri.
setParam( u
"zmax"_s, parts[ u
"zmax"_s ].toString() );
443 if ( parts.contains( u
"styleUrl"_s ) )
444 dsUri.
setParam( u
"styleUrl"_s, parts[ u
"styleUrl"_s ].toString() );
446 if ( parts.contains( u
"authcfg"_s ) )
452QString QgsArcGisVectorTileServiceDataProviderMetadata::absoluteToRelativeUri(
const QString &uri,
const QgsReadWriteContext & )
const
459QString QgsArcGisVectorTileServiceDataProviderMetadata::relativeToAbsoluteUri(
const QString &uri,
const QgsReadWriteContext & )
const
466QList<Qgis::LayerType> QgsArcGisVectorTileServiceDataProviderMetadata::supportedLayerTypes()
const
@ ReadLayerMetadata
Provider can read layer metadata from data store. See QgsDataProvider::layerMetadata().
@ AlwaysUseTileMatrixSetFromProvider
Vector tile layer must always use the tile matrix set from the data provider, and should never store,...
QFlags< DataProviderFlag > DataProviderFlags
Data provider flags.
@ FastExtent2D
Provider's 2D extent retrieval via QgsDataProvider::extent() is always guaranteed to be trivial/fast ...
QFlags< DataProviderReadFlag > DataProviderReadFlags
Flags which control data provider construction.
QFlags< VectorTileProviderCapability > VectorTileProviderCapabilities
Vector tile data provider capabilities.
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
QFlags< VectorTileProviderFlag > VectorTileProviderFlags
Vector tile data provider flags.
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
static QgsCoordinateReferenceSystem convertSpatialReference(const QVariantMap &spatialReferenceMap)
Converts a spatial reference JSON definition to a QgsCoordinateReferenceSystem value.
A thread safe class for performing blocking (sync) network requests, with full support for QGIS proxy...
ErrorCode get(QNetworkRequest &request, bool forceRefresh=false, QgsFeedback *feedback=nullptr, RequestFlags requestFlags=QgsBlockingNetworkRequest::RequestFlags())
Performs a "get" operation on the specified request.
@ NetworkError
A network error occurred.
@ ServerExceptionError
An exception was raised by the server.
@ NoError
No error was encountered.
@ TimeoutError
Timeout was reached before a reply was received.
QgsNetworkReplyContent reply() const
Returns the content of the network reply, after a get(), post(), head() or put() request has been mad...
Represents a coordinate reference system (CRS).
Custom exception class for Coordinate Reference System related exceptions.
Stores the component parts of a data source URI (e.g.
QByteArray encodedUri() const
Returns the complete encoded URI as a byte array.
bool hasParam(const QString &key) const
Returns true if a parameter with the specified key exists.
void setEncodedUri(const QByteArray &uri)
Sets the complete encoded uri.
void setAuthConfigId(const QString &authcfg)
Sets the authentication configuration ID for the URI.
QgsHttpHeaders httpHeaders() const
Returns http headers.
QString param(const QString &key) const
Returns a generic parameter value corresponding to the specified key.
void setParam(const QString &key, const QString &value)
Sets a generic parameter value on the URI.
QString authConfigId() const
Returns any associated authentication configuration ID stored in the URI.
Encapsulates a network reply within a container which is inexpensive to copy and safe to pass between...
QByteArray content() const
Returns the reply content.
A container for the context for various read/write operations on objects.
A rectangle specified with double values.
Base class for vector tile layer data providers.
Encapsulates properties of a vector tile matrix set, including tile origins and scaling information.
static bool checkXYZUrlTemplate(const QString &url)
Checks whether the URL template string is correct (contains {x}, {y} / {-y}, {z} placeholders).
#define QgsDebugError(str)
#define QgsSetRequestInitiatorClass(request, _class)
#define QGIS_PROTECT_QOBJECT_THREAD_ACCESS
Setting options for creating vector data providers.