QGIS API Documentation 4.1.0-Master (659fe69c07c)
Loading...
Searching...
No Matches
qgsarcgisrestutils.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsarcgisrestutils.cpp
3 ----------------------
4 begin : Nov 25, 2015
5 copyright : (C) 2015 by Sandro Mani
7***************************************************************************
8* *
9* This program is free software; you can redistribute it and/or modify *
10* it under the terms of the GNU General Public License as published by *
11* the Free Software Foundation; either version 2 of the License, or *
12* (at your option) any later version. *
13* *
14***************************************************************************/
15
16#include "qgsarcgisrestutils.h"
17
19#include "qgscircularstring.h"
26#include "qgscolorrampimpl.h"
27#include "qgscurve.h"
28#include "qgsfields.h"
29#include "qgsgeometryengine.h"
31#include "qgslinestring.h"
32#include "qgslinesymbollayer.h"
33#include "qgslogger.h"
34#include "qgsmulticurve.h"
35#include "qgsmultilinestring.h"
36#include "qgsmultipoint.h"
37#include "qgsmultipolygon.h"
38#include "qgsmultisurface.h"
39#include "qgspallabeling.h"
40#include "qgspolygon.h"
42#include "qgsrectangle.h"
43#include "qgsrenderer.h"
46#include "qgssymbol.h"
48#include "qgssymbollayer.h"
49#include "qgsvariantutils.h"
51
52#include <QRegularExpression>
53#include <QString>
54#include <QUrl>
55
56#include "moc_qgsarcgisrestutils.cpp"
57
58using namespace Qt::StringLiterals;
59
60QMetaType::Type QgsArcGisRestUtils::convertFieldType( const QString &esriFieldType )
61{
62 if ( esriFieldType == "esriFieldTypeInteger"_L1 )
63 return QMetaType::Type::LongLong;
64 if ( esriFieldType == "esriFieldTypeSmallInteger"_L1 )
65 return QMetaType::Type::Int;
66 if ( esriFieldType == "esriFieldTypeDouble"_L1 )
67 return QMetaType::Type::Double;
68 if ( esriFieldType == "esriFieldTypeSingle"_L1 )
69 return QMetaType::Type::Double;
70 if ( esriFieldType == "esriFieldTypeString"_L1 )
71 return QMetaType::Type::QString;
72 if ( esriFieldType == "esriFieldTypeDate"_L1 )
73 return QMetaType::Type::QDateTime;
74 if ( esriFieldType == "esriFieldTypeGeometry"_L1 )
75 return QMetaType::Type::UnknownType; // Geometry column should not appear as field
76 if ( esriFieldType == "esriFieldTypeOID"_L1 )
77 return QMetaType::Type::LongLong;
78 if ( esriFieldType == "esriFieldTypeBlob"_L1 )
79 return QMetaType::Type::QByteArray;
80 if ( esriFieldType == "esriFieldTypeGlobalID"_L1 )
81 return QMetaType::Type::QString;
82 if ( esriFieldType == "esriFieldTypeRaster"_L1 )
83 return QMetaType::Type::QByteArray;
84 if ( esriFieldType == "esriFieldTypeGUID"_L1 )
85 return QMetaType::Type::QString;
86 if ( esriFieldType == "esriFieldTypeXML"_L1 )
87 return QMetaType::Type::QString;
88 return QMetaType::Type::UnknownType;
89}
90
92{
93 // http://resources.arcgis.com/en/help/arcobjects-cpp/componenthelp/index.html#//000w0000001p000000
94 if ( esriGeometryType == "esriGeometryNull"_L1 )
96 else if ( esriGeometryType == "esriGeometryPoint"_L1 )
98 else if ( esriGeometryType == "esriGeometryMultipoint"_L1 )
100 else if ( esriGeometryType == "esriGeometryPolyline"_L1 )
102 else if ( esriGeometryType == "esriGeometryPolygon"_L1 )
104 else if ( esriGeometryType == "esriGeometryEnvelope"_L1 )
106 // Unsupported (either by qgis, or format unspecified by the specification)
107 // esriGeometryCircularArc
108 // esriGeometryEllipticArc
109 // esriGeometryBezier3Curve
110 // esriGeometryPath
111 // esriGeometryRing
112 // esriGeometryLine
113 // esriGeometryAny
114 // esriGeometryMultiPatch
115 // esriGeometryTriangleStrip
116 // esriGeometryTriangleFan
117 // esriGeometryRay
118 // esriGeometrySphere
119 // esriGeometryTriangles
120 // esriGeometryBag
122}
123
124std::unique_ptr< QgsPoint > QgsArcGisRestUtils::convertPoint( const QVariantList &coordList, Qgis::WkbType pointType )
125{
126 const int nCoords = static_cast< int >( coordList.size() );
127 if ( nCoords < 2 )
128 return nullptr;
129 bool xok = false, yok = false;
130 const double x = coordList[0].toDouble( &xok );
131 const double y = coordList[1].toDouble( &yok );
132 if ( !xok || !yok )
133 return nullptr;
134 const bool hasZ = QgsWkbTypes::hasZ( pointType );
135 const double z = hasZ && nCoords >= 3 ? coordList[2].toDouble() : std::numeric_limits< double >::quiet_NaN();
136
137 // if point has just M but not Z, then the point dimension list will only have X, Y, M, otherwise it will have X, Y, Z, M
138 const double m = QgsWkbTypes::hasM( pointType ) && ( ( hasZ && nCoords >= 4 ) || ( !hasZ && nCoords >= 3 ) ) ? coordList[hasZ ? 3 : 2].toDouble() : std::numeric_limits< double >::quiet_NaN();
139 return std::make_unique< QgsPoint >( pointType, x, y, z, m );
140}
141
142std::unique_ptr< QgsCircularString > QgsArcGisRestUtils::convertCircularString( const QVariantMap &curveData, Qgis::WkbType pointType, const QgsPoint &startPoint )
143{
144 const QVariantList coordsList = curveData[u"c"_s].toList();
145 if ( coordsList.isEmpty() )
146 return nullptr;
147 const int coordsListSize = static_cast< int >( coordsList.size() );
148
149 QVector<QgsPoint> points;
150 points.reserve( coordsListSize + 1 );
151 points.append( startPoint );
152
153 for ( int i = 0; i < coordsListSize - 1; )
154 {
155 // first point is end point, second is point on curve
156 // i.e. the opposite to what QGIS requires!
157 std::unique_ptr< QgsPoint > endPoint( convertPoint( coordsList.at( i ).toList(), pointType ) );
158 if ( !endPoint )
159 return nullptr;
160 i++;
161 std::unique_ptr< QgsPoint > interiorPoint( convertPoint( coordsList.at( i ).toList(), pointType ) );
162 if ( !interiorPoint )
163 return nullptr;
164 i++;
165 points << *interiorPoint;
166 points << *endPoint;
167 }
168 auto curve = std::make_unique< QgsCircularString>();
169 curve->setPoints( points );
170 return curve;
171}
172
173std::unique_ptr< QgsCurve > QgsArcGisRestUtils::convertCompoundCurve( const QVariantList &curvesList, Qgis::WkbType pointType )
174{
175 // [[6,3],[5,3],{"b":[[3,2],[6,1],[2,4]]},[1,2],{"c": [[3,3],[1,4]]}]
176 auto compoundCurve = std::make_unique< QgsCompoundCurve >();
177
178 QVector< double > lineX;
179 QVector< double > lineY;
180 QVector< double > lineZ;
181 QVector< double > lineM;
182 const int maxCurveListSize = static_cast< int >( curvesList.size() );
183 lineX.resize( maxCurveListSize );
184 lineY.resize( maxCurveListSize );
185
186 const bool hasZ = QgsWkbTypes::hasZ( pointType );
187 if ( hasZ )
188 lineZ.resize( maxCurveListSize );
189 const bool hasM = QgsWkbTypes::hasM( pointType );
190 if ( hasM )
191 lineM.resize( maxCurveListSize );
192
193 double *outLineX = lineX.data();
194 double *outLineY = lineY.data();
195 double *outLineZ = lineZ.data();
196 double *outLineM = lineM.data();
197 int actualLineSize = 0;
198
199 bool xok = false;
200 bool yok = false;
201
202 int curveListIndex = 0;
203 for ( const QVariant &curveData : curvesList )
204 {
205 if ( curveData.userType() == QMetaType::Type::QVariantList )
206 {
207 const QVariantList coordList = curveData.toList();
208 const int nCoords = static_cast< int >( coordList.size() );
209 if ( nCoords < 2 )
210 return nullptr;
211
212 const double x = coordList[0].toDouble( &xok );
213 const double y = coordList[1].toDouble( &yok );
214 if ( !xok || !yok )
215 return nullptr;
216
217 actualLineSize++;
218 *outLineX++ = x;
219 *outLineY++ = y;
220 if ( hasZ )
221 {
222 *outLineZ++ = nCoords >= 3 ? coordList[2].toDouble() : std::numeric_limits< double >::quiet_NaN();
223 }
224
225 if ( hasM )
226 {
227 // if point has just M but not Z, then the point dimension list will only have X, Y, M, otherwise it will have X, Y, Z, M
228 *outLineM++ = ( ( hasZ && nCoords >= 4 ) || ( !hasZ && nCoords >= 3 ) ) ? coordList[hasZ ? 3 : 2].toDouble() : std::numeric_limits< double >::quiet_NaN();
229 }
230 }
231 else if ( curveData.userType() == QMetaType::Type::QVariantMap )
232 {
233 // The last point of the linestring is the start point of this circular string
234 QgsPoint lastLineStringPoint;
235 if ( actualLineSize > 0 )
236 {
237 lastLineStringPoint
238 = QgsPoint( lineX.at( actualLineSize - 1 ), lineY.at( actualLineSize - 1 ), hasZ ? lineZ.at( actualLineSize - 1 ) : std::numeric_limits< double >::quiet_NaN(), hasM ? lineM.at( actualLineSize - 1 ) : std::numeric_limits< double >::quiet_NaN() );
239 }
240 std::unique_ptr< QgsCircularString > circularString( convertCircularString( curveData.toMap(), pointType, lastLineStringPoint ) );
241 if ( !circularString )
242 {
243 return nullptr;
244 }
245
246 if ( actualLineSize > 0 )
247 {
248 lineX.resize( actualLineSize );
249 lineY.resize( actualLineSize );
250 if ( hasZ )
251 lineZ.resize( actualLineSize );
252 if ( hasM )
253 lineM.resize( actualLineSize );
254
255 compoundCurve->addCurve( new QgsLineString( lineX, lineY, lineZ, lineM ) );
256 lineX.resize( maxCurveListSize - curveListIndex );
257 lineY.resize( maxCurveListSize - curveListIndex );
258 if ( hasZ )
259 lineZ.resize( maxCurveListSize - curveListIndex );
260 if ( hasM )
261 lineM.resize( maxCurveListSize - curveListIndex );
262 outLineX = lineX.data();
263 outLineY = lineY.data();
264 outLineZ = lineZ.data();
265 outLineM = lineM.data();
266 }
267
268 // If the previous curve had less than two points, remove it
269 if ( compoundCurve->curveAt( compoundCurve->nCurves() - 1 )->nCoordinates() < 2 )
270 compoundCurve->removeCurve( compoundCurve->nCurves() - 1 );
271
272 const QgsPoint endPointCircularString = circularString->endPoint();
273 compoundCurve->addCurve( circularString.release() );
274
275 // Prepare a new line string
276 actualLineSize = 1;
277 *outLineX++ = endPointCircularString.x();
278 *outLineY++ = endPointCircularString.y();
279 if ( hasZ )
280 *outLineZ++ = endPointCircularString.z();
281 if ( hasM )
282 *outLineM++ = endPointCircularString.m();
283 }
284 curveListIndex++;
285 }
286
287 if ( actualLineSize == 1 && compoundCurve->nCurves() > 0 )
288 {
289 const QgsCurve *finalCurve = compoundCurve->curveAt( compoundCurve->nCurves() - 1 );
290 const QgsPoint finalCurveEndPoint = finalCurve->endPoint();
291 if ( qgsDoubleNear( finalCurveEndPoint.x(), lineX.at( 0 ) )
292 && qgsDoubleNear( finalCurveEndPoint.y(), lineY.at( 0 ) )
293 && ( !hasZ || qgsDoubleNear( finalCurveEndPoint.z(), lineZ.at( 0 ) ) )
294 && ( !hasM || qgsDoubleNear( finalCurveEndPoint.m(), lineM.at( 0 ) ) ) )
295 {
296 actualLineSize = 0; // redundant final curve containing a duplicate vertex
297 }
298 }
299
300 if ( actualLineSize > 0 )
301 {
302 lineX.resize( actualLineSize );
303 lineY.resize( actualLineSize );
304 if ( hasZ )
305 lineZ.resize( actualLineSize );
306 if ( hasM )
307 lineM.resize( actualLineSize );
308 compoundCurve->addCurve( new QgsLineString( lineX, lineY, lineZ, lineM ) );
309 }
310
311 return compoundCurve;
312}
313
314std::unique_ptr<QgsLineString> QgsArcGisRestUtils::convertLineString( const QVariantList &curvesList, Qgis::WkbType pointType )
315{
316 auto linestring = std::make_unique< QgsLineString >();
317
318 QVector< double > lineX;
319 QVector< double > lineY;
320 QVector< double > lineZ;
321 QVector< double > lineM;
322 const int maxCurveListSize = static_cast< int >( curvesList.size() );
323 lineX.resize( maxCurveListSize );
324 lineY.resize( maxCurveListSize );
325
326 const bool hasZ = QgsWkbTypes::hasZ( pointType );
327 if ( hasZ )
328 lineZ.resize( maxCurveListSize );
329 const bool hasM = QgsWkbTypes::hasM( pointType );
330 if ( hasM )
331 lineM.resize( maxCurveListSize );
332
333 double *outLineX = lineX.data();
334 double *outLineY = lineY.data();
335 double *outLineZ = lineZ.data();
336 double *outLineM = lineM.data();
337
338 bool xok = false;
339 bool yok = false;
340
341 int actualLineSize = 0;
342 for ( const QVariant &curveData : curvesList )
343 {
344 if ( curveData.userType() == QMetaType::Type::QVariantList )
345 {
346 const QVariantList coordList = curveData.toList();
347 const int nCoords = static_cast< int >( coordList.size() );
348 if ( nCoords < 2 )
349 return nullptr;
350
351 const double x = coordList[0].toDouble( &xok );
352 const double y = coordList[1].toDouble( &yok );
353 if ( !xok || !yok )
354 return nullptr;
355
356 actualLineSize++;
357 *outLineX++ = x;
358 *outLineY++ = y;
359 if ( hasZ )
360 {
361 *outLineZ++ = nCoords >= 3 ? coordList[2].toDouble() : std::numeric_limits< double >::quiet_NaN();
362 }
363
364 if ( hasM )
365 {
366 // if point has just M but not Z, then the point dimension list will only have X, Y, M, otherwise it will have X, Y, Z, M
367 *outLineM++ = ( ( hasZ && nCoords >= 4 ) || ( !hasZ && nCoords >= 3 ) ) ? coordList[hasZ ? 3 : 2].toDouble() : std::numeric_limits< double >::quiet_NaN();
368 }
369 }
370 else
371 {
372 QgsDebugError( u"Found unexpected value when parsing ESRI json line string. Expected list, got %1"_s.arg( curveData.metaType().name() ) );
373 return nullptr;
374 }
375 }
376
377 if ( actualLineSize == 0 )
378 return nullptr;
379
380 lineX.resize( actualLineSize );
381 lineY.resize( actualLineSize );
382 if ( hasZ )
383 lineZ.resize( actualLineSize );
384 if ( hasM )
385 lineM.resize( actualLineSize );
386 return std::make_unique< QgsLineString>( lineX, lineY, lineZ, lineM );
387}
388
389std::unique_ptr< QgsPoint > QgsArcGisRestUtils::convertGeometryPoint( const QVariantMap &geometryData, Qgis::WkbType pointType )
390{
391 // {"x" : <x>, "y" : <y>, "z" : <z>, "m" : <m>}
392 bool xok = false, yok = false;
393 double x = geometryData[u"x"_s].toDouble( &xok );
394 double y = geometryData[u"y"_s].toDouble( &yok );
395 if ( !xok || !yok )
396 return nullptr;
397 double z = geometryData[u"z"_s].toDouble();
398 double m = geometryData[u"m"_s].toDouble();
399 return std::make_unique< QgsPoint >( pointType, x, y, z, m );
400}
401
402std::unique_ptr< QgsMultiPoint > QgsArcGisRestUtils::convertMultiPoint( const QVariantMap &geometryData, Qgis::WkbType pointType )
403{
404 // {"points" : [[ <x1>, <y1>, <z1>, <m1> ] , [ <x2>, <y2>, <z2>, <m2> ], ... ]}
405 const QVariantList coordsList = geometryData[u"points"_s].toList();
406
407 auto multiPoint = std::make_unique< QgsMultiPoint >();
408 multiPoint->reserve( static_cast< int >( coordsList.size() ) );
409 for ( const QVariant &coordData : coordsList )
410 {
411 const QVariantList coordList = coordData.toList();
412 std::unique_ptr< QgsPoint > p = convertPoint( coordList, pointType );
413 if ( !p )
414 {
415 continue;
416 }
417 multiPoint->addGeometry( p.release() );
418 }
419
420 // second chance -- sometimes layers are reported as multipoint but features have single
421 // point geometries. Silently handle this and upgrade to multipoint.
422 std::unique_ptr< QgsPoint > p = convertGeometryPoint( geometryData, pointType );
423 if ( p )
424 multiPoint->addGeometry( p.release() );
425
426 if ( multiPoint->numGeometries() == 0 )
427 {
428 // didn't find any points, so reset geometry to null
429 multiPoint.reset();
430 }
431 return multiPoint;
432}
433
434std::unique_ptr< QgsMultiCurve > QgsArcGisRestUtils::convertGeometryPolyline( const QVariantMap &geometryData, Qgis::WkbType pointType, bool allowCurves )
435{
436 // {"curvePaths": [[[0,0], {"c": [[3,3],[1,4]]} ]]}
437 QVariantList pathsList;
438 if ( geometryData[u"paths"_s].isValid() )
439 pathsList = geometryData[u"paths"_s].toList();
440 else if ( geometryData[u"curvePaths"_s].isValid() )
441 pathsList = geometryData[u"curvePaths"_s].toList();
442 if ( pathsList.isEmpty() )
443 return nullptr;
444 std::unique_ptr< QgsMultiCurve > multiCurve = allowCurves ? std::make_unique< QgsMultiCurve >() : std::make_unique< QgsMultiLineString >();
445 multiCurve->reserve( static_cast< int >( pathsList.size() ) );
446 for ( const QVariant &pathData : std::as_const( pathsList ) )
447 {
448 std::unique_ptr< QgsCurve > curve = allowCurves ? convertCompoundCurve( pathData.toList(), pointType ) : convertLineString( pathData.toList(), pointType );
449 if ( !curve )
450 {
451 return nullptr;
452 }
453 multiCurve->addGeometry( curve.release() );
454 }
455 return multiCurve;
456}
457
458std::unique_ptr< QgsMultiSurface > QgsArcGisRestUtils::convertGeometryPolygon( const QVariantMap &geometryData, Qgis::WkbType pointType, bool allowCurves )
459{
460 // {"curveRings": [[[0,0], {"c": [[3,3],[1,4]]} ]]}
461 QVariantList ringsList;
462 if ( geometryData[u"rings"_s].isValid() )
463 ringsList = geometryData[u"rings"_s].toList();
464 else if ( geometryData[u"ringPaths"_s].isValid() )
465 ringsList = geometryData[u"ringPaths"_s].toList();
466 if ( ringsList.isEmpty() )
467 return nullptr;
468
469 QList< QgsCurve * > curves;
470 for ( int i = 0, n = static_cast< int >( ringsList.size() ); i < n; ++i )
471 {
472 std::unique_ptr< QgsCurve > curve = allowCurves ? convertCompoundCurve( ringsList[i].toList(), pointType ) : convertLineString( ringsList[i].toList(), pointType );
473 if ( !curve )
474 {
475 continue;
476 }
477 curves.append( curve.release() );
478 }
479 if ( curves.count() == 0 )
480 return nullptr;
481
482 std::unique_ptr< QgsMultiSurface > result = allowCurves ? std::make_unique< QgsMultiSurface >() : std::make_unique< QgsMultiPolygon >();
483 if ( curves.count() == 1 )
484 {
485 // shortcut for exterior ring only
486 std::unique_ptr< QgsCurvePolygon > newPolygon = allowCurves ? std::make_unique< QgsCurvePolygon >() : std::make_unique< QgsPolygon >();
487 newPolygon->setExteriorRing( curves.takeAt( 0 ) );
488 result->addGeometry( newPolygon.release() );
489 return result;
490 }
491
492 std::sort( curves.begin(), curves.end(), []( const QgsCurve *a, const QgsCurve *b ) -> bool {
493 double a_area = 0.0;
494 double b_area = 0.0;
495 a->sumUpArea( a_area );
496 b->sumUpArea( b_area );
497 return std::abs( a_area ) > std::abs( b_area );
498 } );
499 result->reserve( static_cast< int >( curves.size() ) );
500 while ( !curves.isEmpty() )
501 {
502 QgsCurve *exterior = curves.takeFirst();
503 QgsCurvePolygon *newPolygon = allowCurves ? new QgsCurvePolygon() : new QgsPolygon();
504 newPolygon->setExteriorRing( exterior );
505 std::unique_ptr<QgsGeometryEngine> engine( QgsGeometry::createGeometryEngine( newPolygon ) );
506 engine->prepareGeometry();
507
508 QMutableListIterator< QgsCurve * > it( curves );
509 while ( it.hasNext() )
510 {
511 QgsCurve *curve = it.next();
512 QgsRectangle boundingBox = newPolygon->boundingBox();
513 if ( boundingBox.intersects( curve->boundingBox() ) )
514 {
515 QgsPoint point = curve->startPoint();
516 if ( engine->contains( &point ) )
517 {
518 newPolygon->addInteriorRing( curve );
519 it.remove();
520 engine.reset( QgsGeometry::createGeometryEngine( newPolygon ) );
521 engine->prepareGeometry();
522 }
523 }
524 }
525 result->addGeometry( newPolygon );
526 }
527 if ( result->numGeometries() == 0 )
528 return nullptr;
529
530 return result;
531}
532
533std::unique_ptr< QgsPolygon > QgsArcGisRestUtils::convertEnvelope( const QVariantMap &geometryData )
534{
535 // {"xmin" : -109.55, "ymin" : 25.76, "xmax" : -86.39, "ymax" : 49.94}
536 bool xminOk = false, yminOk = false, xmaxOk = false, ymaxOk = false;
537 double xmin = geometryData[u"xmin"_s].toDouble( &xminOk );
538 double ymin = geometryData[u"ymin"_s].toDouble( &yminOk );
539 double xmax = geometryData[u"xmax"_s].toDouble( &xmaxOk );
540 double ymax = geometryData[u"ymax"_s].toDouble( &ymaxOk );
541 if ( !xminOk || !yminOk || !xmaxOk || !ymaxOk )
542 return nullptr;
543 auto ext = std::make_unique< QgsLineString>();
544 ext->addVertex( QgsPoint( xmin, ymin ) );
545 ext->addVertex( QgsPoint( xmax, ymin ) );
546 ext->addVertex( QgsPoint( xmax, ymax ) );
547 ext->addVertex( QgsPoint( xmin, ymax ) );
548 ext->addVertex( QgsPoint( xmin, ymin ) );
549 auto poly = std::make_unique< QgsPolygon >();
550 poly->setExteriorRing( ext.release() );
551 return poly;
552}
553
554std::unique_ptr< QgsAbstractGeometry > QgsArcGisRestUtils::convertGeometry(
555 const QVariantMap &geometryData, const QString &esriGeometryType, bool readM, bool readZ, bool allowCurves, QgsCoordinateReferenceSystem *crs
556)
557{
558 Qgis::WkbType pointType = QgsWkbTypes::zmType( Qgis::WkbType::Point, readZ, readM );
559 if ( crs )
560 {
561 *crs = convertSpatialReference( geometryData[u"spatialReference"_s].toMap() );
562 }
563
564 // http://resources.arcgis.com/en/help/arcgis-rest-api/index.html#/Geometry_Objects/02r3000000n1000000/
565 if ( esriGeometryType == "esriGeometryNull"_L1 )
566 return nullptr;
567 else if ( esriGeometryType == "esriGeometryPoint"_L1 )
568 return convertGeometryPoint( geometryData, pointType );
569 else if ( esriGeometryType == "esriGeometryMultipoint"_L1 )
570 return convertMultiPoint( geometryData, pointType );
571 else if ( esriGeometryType == "esriGeometryPolyline"_L1 )
572 return convertGeometryPolyline( geometryData, pointType, allowCurves );
573 else if ( esriGeometryType == "esriGeometryPolygon"_L1 )
574 return convertGeometryPolygon( geometryData, pointType, allowCurves );
575 else if ( esriGeometryType == "esriGeometryEnvelope"_L1 )
576 return convertEnvelope( geometryData );
577 // Unsupported (either by qgis, or format unspecified by the specification)
578 // esriGeometryCircularArc
579 // esriGeometryEllipticArc
580 // esriGeometryBezier3Curve
581 // esriGeometryPath
582 // esriGeometryRing
583 // esriGeometryLine
584 // esriGeometryAny
585 // esriGeometryMultiPatch
586 // esriGeometryTriangleStrip
587 // esriGeometryTriangleFan
588 // esriGeometryRay
589 // esriGeometrySphere
590 // esriGeometryTriangles
591 // esriGeometryBag
592 return nullptr;
593}
594
596{
598
599 QString spatialReference = spatialReferenceMap[u"latestWkid"_s].toString();
600 if ( spatialReference.isEmpty() )
601 spatialReference = spatialReferenceMap[u"wkid"_s].toString();
602
603 // prefer using authority/id wherever we can
604 if ( !spatialReference.isEmpty() )
605 {
606 crs.createFromString( u"EPSG:%1"_s.arg( spatialReference ) );
607 if ( !crs.isValid() )
608 {
609 // Try as an ESRI auth
610 crs.createFromString( u"ESRI:%1"_s.arg( spatialReference ) );
611 }
612 }
613 else if ( !spatialReferenceMap[u"wkt"_s].toString().isEmpty() )
614 {
615 // otherwise fallback to WKT
616 crs.createFromWkt( spatialReferenceMap[u"wkt"_s].toString() );
617 }
618
619 if ( !crs.isValid() )
620 {
621 // If no spatial reference, just use WGS84
622 // TODO -- this needs further investigation! Most ESRI server services default to 3857, so that would likely be
623 // a safer fallback to use...
624 crs.createFromString( u"EPSG:4326"_s );
625 }
626 return crs;
627}
628
629std::unique_ptr< QgsSymbol > QgsArcGisRestUtils::convertSymbol( const QVariantMap &symbolData, QgsSymbolConverterContext &context )
630{
631 return QgsSymbolConverterEsriRest().createSymbol( symbolData, context );
632}
633
634std::unique_ptr< QgsSymbol > QgsArcGisRestUtils::convertSymbol( const QVariantMap &symbolData )
635{
636 QgsReadWriteContext rwContext;
637 QgsSymbolConverterContext context( rwContext );
638 return convertSymbol( symbolData, context );
639}
640
641std::unique_ptr<QgsAbstractVectorLayerLabeling > QgsArcGisRestUtils::convertLabeling( const QVariantList &labelingData )
642{
643 if ( labelingData.empty() )
644 return nullptr;
645
646 QgsRuleBasedLabeling::Rule *root = new QgsRuleBasedLabeling::Rule( new QgsPalLayerSettings(), 0, 0, QString(), QString(), false );
647 root->setActive( true );
648
649 int i = 1;
650 for ( const QVariant &lbl : labelingData )
651 {
652 const QVariantMap labeling = lbl.toMap();
653
655 QgsTextFormat format;
656
657 const QString placement = labeling.value( u"labelPlacement"_s ).toString();
658 if ( placement == "esriServerPointLabelPlacementAboveCenter"_L1 )
659 {
662 }
663 else if ( placement == "esriServerPointLabelPlacementBelowCenter"_L1 )
664 {
667 }
668 else if ( placement == "esriServerPointLabelPlacementCenterCenter"_L1 )
669 {
672 }
673 else if ( placement == "esriServerPointLabelPlacementAboveLeft"_L1 )
674 {
677 }
678 else if ( placement == "esriServerPointLabelPlacementBelowLeft"_L1 )
679 {
682 }
683 else if ( placement == "esriServerPointLabelPlacementCenterLeft"_L1 )
684 {
687 }
688 else if ( placement == "esriServerPointLabelPlacementAboveRight"_L1 )
689 {
692 }
693 else if ( placement == "esriServerPointLabelPlacementBelowRight"_L1 )
694 {
697 }
698 else if ( placement == "esriServerPointLabelPlacementCenterRight"_L1 )
699 {
702 }
703 else if ( placement == "esriServerLinePlacementAboveAfter"_L1 || placement == "esriServerLinePlacementAboveStart"_L1 || placement == "esriServerLinePlacementAboveAlong"_L1 )
704 {
707 }
708 else if ( placement == "esriServerLinePlacementBelowAfter"_L1 || placement == "esriServerLinePlacementBelowStart"_L1 || placement == "esriServerLinePlacementBelowAlong"_L1 )
709 {
712 }
713 else if ( placement == "esriServerLinePlacementCenterAfter"_L1 || placement == "esriServerLinePlacementCenterStart"_L1 || placement == "esriServerLinePlacementCenterAlong"_L1 )
714 {
717 }
718 else if ( placement == "esriServerPolygonPlacementAlwaysHorizontal"_L1 )
719 {
721 }
722
723 const double minScale = labeling.value( u"minScale"_s ).toDouble();
724 const double maxScale = labeling.value( u"maxScale"_s ).toDouble();
725
726 QVariantMap symbol = labeling.value( u"symbol"_s ).toMap();
727 format.setColor( convertColor( symbol.value( u"color"_s ) ) );
728 const double haloSize = symbol.value( u"haloSize"_s ).toDouble();
729 if ( !qgsDoubleNear( haloSize, 0.0 ) )
730 {
732 buffer.setEnabled( true );
733 buffer.setSize( haloSize );
735 buffer.setColor( convertColor( symbol.value( u"haloColor"_s ) ) );
736 format.setBuffer( buffer );
737 }
738
739 const QString fontFamily = symbol.value( u"font"_s ).toMap().value( u"family"_s ).toString();
740 const QString fontStyle = symbol.value( u"font"_s ).toMap().value( u"style"_s ).toString();
741 const QString fontWeight = symbol.value( u"font"_s ).toMap().value( u"weight"_s ).toString();
742 const int fontSize = symbol.value( u"font"_s ).toMap().value( u"size"_s ).toInt();
743 QFont font( fontFamily, fontSize );
744 font.setStyleName( fontStyle );
745 font.setWeight( fontWeight == "bold"_L1 ? QFont::Bold : QFont::Normal );
746
747 format.setFont( font );
748 format.setSize( fontSize );
750
751 settings->setFormat( format );
752
753 QString where = labeling.value( u"where"_s ).toString();
754 QgsExpression exp( where );
755 // If the where clause isn't parsed as valid, don't use its
756 if ( !exp.isValid() )
757 where.clear();
758
759 settings->fieldName = convertLabelingExpression( labeling.value( u"labelExpression"_s ).toString() );
760 settings->isExpression = true;
761
762 QgsRuleBasedLabeling::Rule *child = new QgsRuleBasedLabeling::Rule( settings, maxScale, minScale, where, QObject::tr( "ASF label %1" ).arg( i++ ), false );
763 child->setActive( true );
764 root->appendChild( child );
765 }
766
767 return std::make_unique< QgsRuleBasedLabeling >( root );
768}
769
770void QgsArcGisRestUtils::applyVisualVariables( const QVariantMap &rendererData, QgsSymbol *symbol, QgsSymbolConverterContext &context )
771{
772 if ( !symbol )
773 return;
774
775 const QVariantList visualVariablesData = rendererData.value( u"visualVariables"_s ).toList();
776 for ( const QVariant &visualVariable : visualVariablesData )
777 {
778 const QVariantMap visualVariableData = visualVariable.toMap();
779 const QString variableType = visualVariableData.value( u"type"_s ).toString();
780
781 if ( variableType == "rotationInfo"_L1 )
782 {
783 const QString field = visualVariableData.value( u"field"_s ).toString();
784 if ( field.isEmpty() )
785 {
786 // Check if it was a valueExpression that we don't support yet
787 if ( !visualVariableData.value( u"valueExpression"_s ).toString().isEmpty() )
788 context.pushWarning( QObject::tr( "ESRI rotationInfo valueExpression is not yet supported" ) );
789 continue;
790 }
791
792 const QString rotationType = visualVariableData.value( u"rotationType"_s ).toString();
793
794 QgsProperty angleProperty;
795 if ( rotationType == "arithmetic"_L1 )
796 {
797 // ArcGIS arithmetic: 0° = East, counter-clockwise
798 // QGIS: 0° = North, clockwise
799 // Conversion: QGIS_angle = 90 - ArcGIS_angle
800 angleProperty = QgsProperty::fromExpression( u"90 - %1"_s.arg( QgsExpression::quotedColumnRef( field ) ) );
801 }
802 else if ( rotationType == "geographic"_L1 )
803 {
804 // ArcGIS geographic: 0° = North, clockwise (same as QGIS)
805 angleProperty = QgsProperty::fromField( field );
806 }
807 else
808 {
809 context.pushWarning( QObject::tr( "ESRI rotationInfo rotationType '%1' is not supported" ).arg( rotationType ) );
810 continue;
811 }
812
813 // Apply rotation to all symbol layers
814 for ( int layer = 0; layer < symbol->symbolLayerCount(); ++layer )
815 {
816 symbol->symbolLayer( layer )->setDataDefinedProperty( QgsSymbolLayer::Property::Angle, angleProperty );
817 }
818 }
819 else
820 {
821 context.pushWarning( QObject::tr( "ESRI visualVariable type '%1' is not currently supported" ).arg( variableType ) );
822 }
823 }
824}
825
826std::unique_ptr< QgsFeatureRenderer > QgsArcGisRestUtils::convertRenderer( const QVariantMap &rendererData, QgsSymbolConverterContext &context )
827{
828 const QString type = rendererData.value( u"type"_s ).toString();
829 if ( type == "simple"_L1 )
830 {
831 const QVariantMap symbolProps = rendererData.value( u"symbol"_s ).toMap();
832 std::unique_ptr< QgsSymbol > symbol( convertSymbol( symbolProps, context ) );
833 if ( symbol )
834 {
835 // Apply visual variables (e.g., rotation) to the symbol
836 applyVisualVariables( rendererData, symbol.get(), context );
837 return std::make_unique< QgsSingleSymbolRenderer >( symbol.release() );
838 }
839 else
840 return nullptr;
841 }
842 else if ( type == "uniqueValue"_L1 )
843 {
844 const QString field1 = rendererData.value( u"field1"_s ).toString();
845 const QString field2 = rendererData.value( u"field2"_s ).toString();
846 const QString field3 = rendererData.value( u"field3"_s ).toString();
847 QString attribute;
848 if ( !field2.isEmpty() || !field3.isEmpty() )
849 {
850 const QString delimiter = rendererData.value( u"fieldDelimiter"_s ).toString();
851 if ( !field3.isEmpty() )
852 {
853 attribute = u"concat(\"%1\",'%2',\"%3\",'%4',\"%5\")"_s.arg( field1, delimiter, field2, delimiter, field3 );
854 }
855 else
856 {
857 attribute = u"concat(\"%1\",'%2',\"%3\")"_s.arg( field1, delimiter, field2 );
858 }
859 }
860 else
861 {
862 attribute = field1;
863 }
864
865 const QVariantList categories = rendererData.value( u"uniqueValueInfos"_s ).toList();
866 QgsCategoryList categoryList;
867 for ( const QVariant &category : categories )
868 {
869 const QVariantMap categoryData = category.toMap();
870 const QString value = categoryData.value( u"value"_s ).toString();
871 const QString label = categoryData.value( u"label"_s ).toString();
872 std::unique_ptr< QgsSymbol > symbol( QgsArcGisRestUtils::convertSymbol( categoryData.value( u"symbol"_s ).toMap(), context ) );
873 if ( symbol )
874 {
875 // Apply visual variables (e.g., rotation) to the symbol
876 applyVisualVariables( rendererData, symbol.get(), context );
877
878 categoryList.append( QgsRendererCategory( value, symbol.release(), label ) );
879 }
880 }
881
882 std::unique_ptr< QgsSymbol > defaultSymbol( convertSymbol( rendererData.value( u"defaultSymbol"_s ).toMap(), context ) );
883 if ( defaultSymbol )
884 {
885 // Apply visual variables (e.g., rotation) to the symbol
886 applyVisualVariables( rendererData, defaultSymbol.get(), context );
887
888 categoryList.append( QgsRendererCategory( QVariant(), defaultSymbol.release(), rendererData.value( u"defaultLabel"_s ).toString() ) );
889 }
890
891 if ( categoryList.empty() )
892 return nullptr;
893
894 auto renderer = std::make_unique< QgsCategorizedSymbolRenderer >( attribute, categoryList );
895 return renderer;
896 }
897 else if ( type == "classBreaks"_L1 )
898 {
899 const QString attrName = rendererData.value( u"field"_s ).toString();
900
901 const QVariantList classBreakInfos = rendererData.value( u"classBreakInfos"_s ).toList();
902 const QVariantMap authoringInfo = rendererData.value( u"authoringInfo"_s ).toMap();
903 QVariantMap symbolData;
904
905 QString esriMode = authoringInfo.value( u"classificationMethod"_s ).toString();
906 if ( esriMode.isEmpty() )
907 {
908 esriMode = rendererData.value( u"classificationMethod"_s ).toString();
909 }
910
911 if ( !classBreakInfos.isEmpty() )
912 {
913 symbolData = classBreakInfos.at( 0 ).toMap().value( u"symbol"_s ).toMap();
914 }
915 std::unique_ptr< QgsSymbol > symbol( QgsArcGisRestUtils::convertSymbol( symbolData, context ) );
916 if ( !symbol )
917 return nullptr;
918
919 const double transparency = rendererData.value( u"transparency"_s ).toDouble();
920 const double opacity = ( 100.0 - transparency ) / 100.0;
921 symbol->setOpacity( opacity );
922
923 const QVariantList visualVariablesData = rendererData.value( u"visualVariables"_s ).toList();
924
925 for ( const QVariant &visualVariable : visualVariablesData )
926 {
927 const QVariantMap visualVariableData = visualVariable.toMap();
928 const QString variableType = visualVariableData.value( u"type"_s ).toString();
929 if ( variableType == "sizeInfo"_L1 )
930 {
931 continue;
932 }
933 else if ( variableType == "colorInfo"_L1 )
934 {
935 const QVariantList stops = visualVariableData.value( u"stops"_s ).toList();
936 if ( stops.size() < 2 )
937 continue;
938
939 // layer has continuous coloring, so convert to a symbol using color ramp assistant
940 bool ok = false;
941 const double minValue = stops.front().toMap().value( u"value"_s ).toDouble( &ok );
942 if ( !ok )
943 continue;
944 const QColor minColor = convertColor( stops.front().toMap().value( u"color"_s ) );
945
946 const double maxValue = stops.back().toMap().value( u"value"_s ).toDouble( &ok );
947 if ( !ok )
948 continue;
949 const QColor maxColor = convertColor( stops.back().toMap().value( u"color"_s ) );
950
951 QgsGradientStopsList gradientStops;
952 for ( int i = 1; i < stops.size() - 1; ++i )
953 {
954 const QVariantMap stopData = stops.at( i ).toMap();
955 const double breakpoint = stopData.value( u"value"_s ).toDouble();
956 const double scaledBreakpoint = ( breakpoint - minValue ) / ( maxValue - minValue );
957 const QColor fillColor = convertColor( stopData.value( u"color"_s ) );
958
959 gradientStops.append( QgsGradientStop( scaledBreakpoint, fillColor ) );
960 }
961
962 auto colorRamp = std::make_unique< QgsGradientColorRamp >( minColor, maxColor, false, gradientStops );
963
964 QgsProperty colorProperty = QgsProperty::fromField( attrName );
965 colorProperty.setTransformer( new QgsColorRampTransformer( minValue, maxValue, colorRamp.release() ) );
966 for ( int layer = 0; layer < symbol->symbolLayerCount(); ++layer )
967 {
968 symbol->symbolLayer( layer )->setDataDefinedProperty( QgsSymbolLayer::Property::FillColor, colorProperty );
969 }
970
971 return std::make_unique< QgsSingleSymbolRenderer >( symbol.release() );
972 }
973 else if ( variableType == "rotationInfo"_L1 )
974 {
975 // Rotation will be handled below after the renderer is created
976 continue;
977 }
978 else
979 {
980 context.pushWarning( QObject::tr( "ESRI visualVariable type '%1' is not currently supported" ).arg( variableType ) );
981 }
982 }
983
984 double lastValue = rendererData.value( u"minValue"_s ).toDouble();
985
986 auto graduatedRenderer = std::make_unique< QgsGraduatedSymbolRenderer >( attrName );
987
988 graduatedRenderer->setSourceSymbol( symbol.release() );
989
990 if ( esriMode == "esriClassifyDefinedInterval"_L1 )
991 {
993 graduatedRenderer->setClassificationMethod( method );
994 }
995 else if ( esriMode == "esriClassifyEqualInterval"_L1 )
996 {
998 graduatedRenderer->setClassificationMethod( method );
999 }
1000 else if ( esriMode == "esriClassifyGeometricalInterval"_L1 )
1001 {
1003 graduatedRenderer->setClassificationMethod( method );
1004 }
1005 else if ( esriMode == "esriClassifyManual"_L1 )
1006 {
1008 graduatedRenderer->setClassificationMethod( method );
1009 }
1010 else if ( esriMode == "esriClassifyNaturalBreaks"_L1 )
1011 {
1013 graduatedRenderer->setClassificationMethod( method );
1014 }
1015 else if ( esriMode == "esriClassifyQuantile"_L1 )
1016 {
1018 graduatedRenderer->setClassificationMethod( method );
1019 }
1020 else if ( esriMode == "esriClassifyStandardDeviation"_L1 )
1021 {
1023 graduatedRenderer->setClassificationMethod( method );
1024 }
1025 else if ( !esriMode.isEmpty() )
1026 {
1027 context.pushWarning( QObject::tr( "ESRI classification mode '%1' is not currently supported" ).arg( esriMode ) );
1028 }
1029
1030 for ( const QVariant &classBreakInfo : classBreakInfos )
1031 {
1032 const QVariantMap symbolData = classBreakInfo.toMap().value( u"symbol"_s ).toMap();
1033 std::unique_ptr< QgsSymbol > symbol( QgsArcGisRestUtils::convertSymbol( symbolData, context ) );
1034 double classMaxValue = classBreakInfo.toMap().value( u"classMaxValue"_s ).toDouble();
1035 const QString label = classBreakInfo.toMap().value( u"label"_s ).toString();
1036
1037 // Apply visual variables (e.g., rotation) to the symbol
1038 applyVisualVariables( rendererData, symbol.get(), context );
1039
1040 QgsRendererRange range;
1041
1042 range.setLowerValue( lastValue );
1043 range.setUpperValue( classMaxValue );
1044 range.setLabel( label );
1045 range.setSymbol( symbol.release() );
1046
1047 lastValue = classMaxValue;
1048 graduatedRenderer->addClass( range );
1049 }
1050
1051 return graduatedRenderer;
1052 }
1053 else if ( type == "heatmap"_L1 )
1054 {
1055 // currently unsupported
1056 return nullptr;
1057 }
1058 else if ( type == "vectorField"_L1 )
1059 {
1060 // currently unsupported
1061 return nullptr;
1062 }
1063 return nullptr;
1064}
1065
1066std::unique_ptr< QgsFeatureRenderer > QgsArcGisRestUtils::convertRenderer( const QVariantMap &rendererData )
1067{
1068 QgsReadWriteContext rwContext;
1069 QgsSymbolConverterContext context( rwContext );
1070 return convertRenderer( rendererData, context );
1071}
1072
1074{
1075 QString expression = string;
1076
1077 // Replace a few ArcGIS token to QGIS equivalents
1078 const thread_local QRegularExpression rx1 = QRegularExpression( u"(?=([^\"\\\\]*(\\\\.|\"([^\"\\\\]*\\\\.)*[^\"\\\\]*\"))*[^\"]*$)(\\s|^)CONCAT(\\s|$)"_s );
1079 expression = expression.replace( rx1, u"\\4||\\5"_s );
1080
1081 const thread_local QRegularExpression rx2 = QRegularExpression( u"(?=([^\"\\\\]*(\\\\.|\"([^\"\\\\]*\\\\.)*[^\"\\\\]*\"))*[^\"]*$)(\\s|^)NEWLINE(\\s|$)"_s );
1082 expression = expression.replace( rx2, u"\\4'\\n'\\5"_s );
1083
1084 // ArcGIS's double quotes are single quotes in QGIS
1085 const thread_local QRegularExpression rx3 = QRegularExpression( u"\"(.*?(?<!\\\\))\""_s );
1086 expression = expression.replace( rx3, u"'\\1'"_s );
1087 const thread_local QRegularExpression rx4 = QRegularExpression( u"\\\\\""_s );
1088 expression = expression.replace( rx4, u"\""_s );
1089
1090 // ArcGIS's square brakets are double quotes in QGIS
1091 const thread_local QRegularExpression rx5 = QRegularExpression( u"\\[([^]]*)\\]"_s );
1092 expression = expression.replace( rx5, u"\"\\1\""_s );
1093
1094 return expression;
1095}
1096
1097QColor QgsArcGisRestUtils::convertColor( const QVariant &colorData )
1098{
1099 return QgsSymbolConverterEsriRest::convertColor( colorData );
1100}
1101
1102Qt::PenStyle QgsArcGisRestUtils::convertLineStyle( const QString &style )
1103{
1105}
1106
1107Qt::BrushStyle QgsArcGisRestUtils::convertFillStyle( const QString &style )
1108{
1110}
1111
1112QDateTime QgsArcGisRestUtils::convertDateTime( const QVariant &value )
1113{
1114 if ( QgsVariantUtils::isNull( value ) )
1115 return QDateTime();
1116 bool ok = false;
1117 QDateTime dt = QDateTime::fromMSecsSinceEpoch( value.toLongLong( &ok ) );
1118 if ( !ok )
1119 {
1120 QgsDebugError( u"Invalid value %1 for datetime"_s.arg( value.toString() ) );
1121 return QDateTime();
1122 }
1123 else
1124 return dt;
1125}
1126
1128{
1129 if ( QgsVariantUtils::isNull( value ) )
1130 return QgsRectangle();
1131
1132 const QVariantMap coords = value.toMap();
1133 if ( coords.isEmpty() )
1134 return QgsRectangle();
1135
1136 bool ok;
1137
1138 const double xmin = coords.value( u"xmin"_s ).toDouble( &ok );
1139 if ( !ok )
1140 return QgsRectangle();
1141
1142 const double ymin = coords.value( u"ymin"_s ).toDouble( &ok );
1143 if ( !ok )
1144 return QgsRectangle();
1145
1146 const double xmax = coords.value( u"xmax"_s ).toDouble( &ok );
1147 if ( !ok )
1148 return QgsRectangle();
1149
1150 const double ymax = coords.value( u"ymax"_s ).toDouble( &ok );
1151 if ( !ok )
1152 return QgsRectangle();
1153
1154 return QgsRectangle( xmin, ymin, xmax, ymax );
1155}
1156
1157
1159{
1160 QVariantMap res;
1161 if ( geometry.isNull() )
1162 return QVariantMap();
1163
1164 const QgsAbstractGeometry *geom = geometry.constGet()->simplifiedTypeRef();
1165 switch ( QgsWkbTypes::flatType( geom->wkbType() ) )
1166 {
1169 return QVariantMap();
1170
1172 res = pointToJson( qgsgeometry_cast< const QgsPoint * >( geom ) );
1173 break;
1174
1176 res = lineStringToJson( qgsgeometry_cast< const QgsLineString * >( geom ) );
1177 break;
1178
1181 res = curveToJson( qgsgeometry_cast< const QgsCurve * >( geom ) );
1182 break;
1183
1185 res = polygonToJson( qgsgeometry_cast< const QgsPolygon * >( geom ) );
1186 break;
1187
1189 res = multiPointToJson( qgsgeometry_cast< const QgsMultiPoint * >( geom ) );
1190 break;
1191
1193 res = multiLineStringToJson( qgsgeometry_cast< const QgsMultiLineString * >( geom ) );
1194 break;
1195
1197 res = multiCurveToJson( qgsgeometry_cast< const QgsMultiCurve * >( geom ) );
1198 break;
1199
1201 res = multiPolygonToJson( qgsgeometry_cast< const QgsMultiPolygon * >( geom ) );
1202 break;
1203
1205 res = curvePolygonToJson( qgsgeometry_cast< const QgsCurvePolygon * >( geom ) );
1206 break;
1207
1209 res = multiSurfaceToJson( qgsgeometry_cast< const QgsMultiSurface * >( geom ) );
1210 break;
1211
1213 return QVariantMap(); // not supported by REST API
1214
1216 return QVariantMap(); //not yet supported, but could be
1217
1218 default:
1219 return QVariantMap(); //unreachable
1220 }
1221
1222 if ( crs.isValid() )
1223 {
1224 // add spatialReference information
1225 res.insert( u"spatialReference"_s, crsToJson( crs ) );
1226 }
1227
1228 return res;
1229}
1230
1231QVariantMap QgsArcGisRestUtils::pointToJson( const QgsPoint *point )
1232{
1233 QVariantMap data;
1234 if ( point->isEmpty() )
1235 data[u"x"_s] = u"NaN"_s;
1236 else
1237 {
1238 data[u"x"_s] = point->x();
1239 data[u"y"_s] = point->y();
1240
1241 if ( point->is3D() )
1242 data[u"z"_s] = !std::isnan( point->z() ) ? QVariant( point->z() ) : QVariant( u"NaN"_s );
1243
1244 if ( point->isMeasure() )
1245 data[u"m"_s] = !std::isnan( point->m() ) ? QVariant( point->m() ) : QVariant( u"NaN"_s );
1246 }
1247 return data;
1248}
1249
1250QVariantMap QgsArcGisRestUtils::multiPointToJson( const QgsMultiPoint *multiPoint )
1251{
1252 QVariantMap data;
1253 const bool hasZ = multiPoint->is3D();
1254 const bool hasM = multiPoint->isMeasure();
1255 data[u"hasM"_s] = hasM;
1256 data[u"hasZ"_s] = hasZ;
1257
1258 QVariantList pointsList;
1259 const int size = multiPoint->numGeometries();
1260 pointsList.reserve( size );
1261
1262 QVariantList pointList;
1263 for ( int i = 0; i < size; ++i )
1264 {
1265 const QgsPoint *point = multiPoint->pointN( i );
1266
1267 pointList.clear();
1268 pointList.append( point->x() );
1269 pointList.append( point->y() );
1270 if ( hasZ )
1271 pointList.append( point->z() );
1272 if ( hasM && !std::isnan( point->m() ) )
1273 pointList.append( point->m() );
1274
1275 pointsList.push_back( pointList );
1276 }
1277
1278 data[u"points"_s] = pointsList;
1279 return data;
1280}
1281
1282QVariantList QgsArcGisRestUtils::lineStringToJsonPath( const QgsLineString *line )
1283{
1284 const bool hasZ = line->is3D();
1285 const bool hasM = line->isMeasure();
1286
1287 QVariantList pointsList;
1288 const int size = line->numPoints();
1289 pointsList.reserve( size );
1290
1291 QVariantList pointList;
1292 const double *xData = line->xData();
1293 const double *yData = line->yData();
1294 const double *zData = hasZ ? line->zData() : nullptr;
1295 const double *mData = hasM ? line->mData() : nullptr;
1296
1297 for ( int i = 0; i < size; ++i )
1298 {
1299 pointList.clear();
1300 pointList.append( *xData++ );
1301 pointList.append( *yData++ );
1302
1303 if ( hasZ )
1304 pointList.append( *zData++ );
1305
1306 if ( hasM && !std::isnan( *mData ) )
1307 pointList.append( *mData );
1308 if ( hasM )
1309 mData++;
1310
1311 pointsList.push_back( pointList );
1312 }
1313 return pointsList;
1314}
1315
1316QVariantList QgsArcGisRestUtils::curveToJsonCurve( const QgsCurve *curve, bool includeStart )
1317{
1318 const bool hasZ = curve->is3D();
1319 const bool hasM = curve->isMeasure();
1320
1321 auto pointToList = [hasZ, hasM]( const QgsPoint &point ) -> QVariantList {
1322 QVariantList pointList;
1323
1324 pointList.append( point.x() );
1325 pointList.append( point.y() );
1326
1327 if ( hasZ )
1328 pointList.append( point.z() );
1329
1330 if ( hasM && !std::isnan( point.m() ) )
1331 pointList.append( point.m() );
1332
1333 return pointList;
1334 };
1335
1336 QVariantList res;
1337 switch ( QgsWkbTypes::flatType( curve->wkbType() ) )
1338 {
1340 {
1341 QVariantList part = lineStringToJsonPath( qgsgeometry_cast< const QgsLineString *>( curve ) );
1342 if ( !part.isEmpty() && !includeStart )
1343 part.removeAt( 0 );
1344 res = part;
1345 break;
1346 }
1347
1349 {
1350 const QgsCircularString *circularString = qgsgeometry_cast<const QgsCircularString * >( curve );
1351 if ( includeStart && !circularString->isEmpty() )
1352 {
1353 res.push_back( pointToList( circularString->startPoint() ) );
1354 }
1355
1356 const int size = circularString->numPoints();
1357 for ( int i = 1; i + 1 < size; i += 2 )
1358 {
1359 // end point comes BEFORE interior point!
1360 QVariantMap curvePart;
1361 QVariantList curveList;
1362 curveList.push_back( pointToList( circularString->pointN( i + 1 ) ) );
1363
1364 curveList.push_back( pointToList( circularString->pointN( i ) ) );
1365
1366 curvePart.insert( u"c"_s, curveList );
1367 res.push_back( curvePart );
1368 }
1369 break;
1370 }
1371
1373 {
1374 const QgsCompoundCurve *compoundCurve = qgsgeometry_cast<const QgsCompoundCurve * >( curve );
1375
1376 const int size = compoundCurve->nCurves();
1377 for ( int i = 0; i < size; ++i )
1378 {
1379 const QgsCurve *subCurve = compoundCurve->curveAt( i );
1380 res.append( curveToJsonCurve( subCurve, i == 0 ) );
1381 }
1382 break;
1383 }
1384
1385 default:
1386 break;
1387 }
1388 return res;
1389}
1390
1391QVariantMap QgsArcGisRestUtils::lineStringToJson( const QgsLineString *line )
1392{
1393 QVariantMap data;
1394 const bool hasZ = line->is3D();
1395 const bool hasM = line->isMeasure();
1396 data[u"hasM"_s] = hasM;
1397 data[u"hasZ"_s] = hasZ;
1398
1399 const QVariantList pointsList = lineStringToJsonPath( line );
1400
1401 QVariantList pointsData = QVariantList();
1402 pointsData.push_back( pointsList );
1403 data[u"paths"_s] = pointsData;
1404
1405 return data;
1406}
1407
1408QVariantMap QgsArcGisRestUtils::curveToJson( const QgsCurve *curve )
1409{
1410 QVariantMap data;
1411 const bool hasZ = curve->is3D();
1412 const bool hasM = curve->isMeasure();
1413 data[u"hasM"_s] = hasM;
1414 data[u"hasZ"_s] = hasZ;
1415
1416 const QVariantList curveList = curveToJsonCurve( curve, true );
1417
1418 QVariantList curveData = QVariantList();
1419 curveData.push_back( curveList );
1420 data[u"curvePaths"_s] = curveData;
1421
1422 return data;
1423}
1424
1425QVariantMap QgsArcGisRestUtils::multiLineStringToJson( const QgsMultiLineString *multiLine )
1426{
1427 QVariantMap data;
1428 const bool hasZ = multiLine->is3D();
1429 const bool hasM = multiLine->isMeasure();
1430 data[u"hasM"_s] = hasM;
1431 data[u"hasZ"_s] = hasZ;
1432
1433 const int size = multiLine->numGeometries();
1434 QVariantList paths;
1435 paths.reserve( size );
1436 for ( int i = 0; i < size; ++i )
1437 {
1438 const QgsLineString *line = multiLine->lineStringN( i );
1439 paths.push_back( lineStringToJsonPath( line ) );
1440 }
1441
1442 data[u"paths"_s] = paths;
1443 return data;
1444}
1445
1446QVariantMap QgsArcGisRestUtils::multiCurveToJson( const QgsMultiCurve *multiCurve )
1447{
1448 QVariantMap data;
1449 const bool hasZ = multiCurve->is3D();
1450 const bool hasM = multiCurve->isMeasure();
1451 data[u"hasM"_s] = hasM;
1452 data[u"hasZ"_s] = hasZ;
1453
1454 const int size = multiCurve->numGeometries();
1455 QVariantList paths;
1456 paths.reserve( size );
1457 for ( int i = 0; i < size; ++i )
1458 {
1459 const QgsCurve *curve = multiCurve->curveN( i );
1460 paths.push_back( curveToJsonCurve( curve, true ) );
1461 }
1462
1463 data[u"curvePaths"_s] = paths;
1464 return data;
1465}
1466
1467QVariantList QgsArcGisRestUtils::polygonToJsonRings( const QgsPolygon *polygon )
1468{
1469 QVariantList rings;
1470 const int numInteriorRings = polygon->numInteriorRings();
1471 rings.reserve( numInteriorRings + 1 );
1472
1473 if ( const QgsLineString *exterior = qgsgeometry_cast< const QgsLineString * >( polygon->exteriorRing() ) )
1474 {
1475 // exterior ring MUST be clockwise
1476 switch ( exterior->orientation() )
1477 {
1479 rings.push_back( lineStringToJsonPath( exterior ) );
1480 break;
1481
1483 {
1484 std::unique_ptr< QgsLineString > reversed( exterior->reversed() );
1485 rings.push_back( lineStringToJsonPath( reversed.get() ) );
1486 break;
1487 }
1489 break;
1490 }
1491 }
1492
1493 for ( int i = 0; i < numInteriorRings; ++i )
1494 {
1495 const QgsLineString *ring = qgsgeometry_cast< const QgsLineString * >( polygon->interiorRing( i ) );
1496 // holes MUST be counter-clockwise
1497 switch ( ring->orientation() )
1498 {
1500 rings.push_back( lineStringToJsonPath( ring ) );
1501 break;
1502
1504 {
1505 std::unique_ptr< QgsLineString > reversed( ring->reversed() );
1506 rings.push_back( lineStringToJsonPath( reversed.get() ) );
1507 break;
1508 }
1510 break;
1511 }
1512 }
1513 return rings;
1514}
1515
1516QVariantList QgsArcGisRestUtils::curvePolygonToJsonRings( const QgsCurvePolygon *polygon )
1517{
1518 QVariantList rings;
1519 const int numInteriorRings = polygon->numInteriorRings();
1520 rings.reserve( numInteriorRings + 1 );
1521
1522 if ( const QgsCurve *exterior = qgsgeometry_cast< const QgsCurve * >( polygon->exteriorRing() ) )
1523 {
1524 // exterior ring MUST be clockwise
1525 switch ( exterior->orientation() )
1526 {
1528 rings.push_back( curveToJsonCurve( exterior, true ) );
1529 break;
1530
1532 {
1533 std::unique_ptr< QgsCurve > reversed( exterior->reversed() );
1534 rings.push_back( curveToJsonCurve( reversed.get(), true ) );
1535 break;
1536 }
1538 break;
1539 }
1540 }
1541
1542 for ( int i = 0; i < numInteriorRings; ++i )
1543 {
1544 const QgsCurve *ring = qgsgeometry_cast< const QgsCurve * >( polygon->interiorRing( i ) );
1545 // holes MUST be counter-clockwise
1546 switch ( ring->orientation() )
1547 {
1549 rings.push_back( curveToJsonCurve( ring, true ) );
1550 break;
1551
1553 {
1554 std::unique_ptr< QgsCurve > reversed( ring->reversed() );
1555 rings.push_back( curveToJsonCurve( reversed.get(), true ) );
1556 break;
1557 }
1559 break;
1560 }
1561 }
1562 return rings;
1563}
1564
1565QVariantMap QgsArcGisRestUtils::polygonToJson( const QgsPolygon *polygon )
1566{
1567 QVariantMap data;
1568 const bool hasZ = polygon->is3D();
1569 const bool hasM = polygon->isMeasure();
1570 data[u"hasM"_s] = hasM;
1571 data[u"hasZ"_s] = hasZ;
1572 data[u"rings"_s] = polygonToJsonRings( polygon );
1573 return data;
1574}
1575
1576QVariantMap QgsArcGisRestUtils::curvePolygonToJson( const QgsCurvePolygon *polygon )
1577{
1578 QVariantMap data;
1579 const bool hasZ = polygon->is3D();
1580 const bool hasM = polygon->isMeasure();
1581 data[u"hasM"_s] = hasM;
1582 data[u"hasZ"_s] = hasZ;
1583 data[u"curveRings"_s] = curvePolygonToJsonRings( polygon );
1584 return data;
1585}
1586
1587QVariantMap QgsArcGisRestUtils::multiPolygonToJson( const QgsMultiPolygon *multiPolygon )
1588{
1589 QVariantMap data;
1590 const bool hasZ = multiPolygon->is3D();
1591 const bool hasM = multiPolygon->isMeasure();
1592 data[u"hasM"_s] = hasM;
1593 data[u"hasZ"_s] = hasZ;
1594
1595 const int size = multiPolygon->numGeometries();
1596 QVariantList rings;
1597 for ( int i = 0; i < size; ++i )
1598 {
1599 const QgsPolygon *polygon = multiPolygon->polygonN( i );
1600 rings.append( polygonToJsonRings( polygon ) );
1601 }
1602
1603 data[u"rings"_s] = rings;
1604 return data;
1605}
1606
1607QVariantMap QgsArcGisRestUtils::multiSurfaceToJson( const QgsMultiSurface *multiSurface )
1608{
1609 QVariantMap data;
1610 const bool hasZ = multiSurface->is3D();
1611 const bool hasM = multiSurface->isMeasure();
1612 data[u"hasM"_s] = hasM;
1613 data[u"hasZ"_s] = hasZ;
1614
1615 const int size = multiSurface->numGeometries();
1616 QVariantList rings;
1617 for ( int i = 0; i < size; ++i )
1618 {
1619 const QgsCurvePolygon *polygon = qgsgeometry_cast< const QgsCurvePolygon * >( multiSurface->geometryN( i ) );
1620 if ( !polygon )
1621 continue;
1622
1623 rings.append( curvePolygonToJsonRings( polygon ) );
1624 }
1625
1626 data[u"curveRings"_s] = rings;
1627 return data;
1628}
1629
1631{
1632 QVariantMap res;
1633 if ( !crs.isValid() )
1634 return res;
1635
1636 const QString authid = crs.authid();
1637 if ( !authid.isEmpty() )
1638 {
1639 const thread_local QRegularExpression rxAuthid( u"(\\w+):(\\d+)"_s );
1640 const QRegularExpressionMatch match = rxAuthid.match( authid );
1641 if ( match.hasMatch() && ( ( match.captured( 1 ).compare( "EPSG"_L1, Qt::CaseInsensitive ) == 0 ) || ( match.captured( 1 ).compare( "ESRI"_L1, Qt::CaseInsensitive ) == 0 ) ) )
1642 {
1643 const QString wkid = match.captured( 2 );
1644 res.insert( u"wkid"_s, wkid );
1645 return res;
1646 }
1647 }
1648
1649 // docs don't mention the WKT version support, so let's hope for 2.0...
1650 res.insert( u"wkt"_s, crs.toWkt( Qgis::CrsWktVariant::Wkt2_2019Simplified ) );
1651
1652 return res;
1653}
1654
1656{
1657 QVariantMap res;
1658 if ( ( flags & FeatureToJsonFlag::IncludeGeometry ) && feature.hasGeometry() )
1659 {
1660 res.insert( u"geometry"_s, geometryToJson( feature.geometry(), context, crs ) );
1661 }
1662
1663 QVariantMap attributes;
1664 const QgsFields fields = feature.fields();
1665 for ( const QgsField &field : fields )
1666 {
1667 QVariant value = feature.attribute( field.name() );
1668 if ( value.userType() == qMetaTypeId< QgsUnsetAttributeValue >() )
1669 {
1670 if ( flags.testFlag( FeatureToJsonFlag::SkipUnsetAttributes ) )
1671 continue;
1672 else
1673 value = QVariant(); // reset to null, we can't store 'QgsUnsetAttributeValue' as json
1674 }
1675
1676 if ( ( flags & FeatureToJsonFlag::IncludeNonObjectIdAttributes ) || field.name() == context.objectIdFieldName() )
1677 attributes.insert( field.name(), variantToAttributeValue( value, field.type(), context ) );
1678 }
1679 if ( !attributes.isEmpty() )
1680 {
1681 res.insert( u"attributes"_s, attributes );
1682 }
1683 return res;
1684}
1685
1686QVariant QgsArcGisRestUtils::variantToAttributeValue( const QVariant &variant, QMetaType::Type expectedType, const QgsArcGisRestContext &context )
1687{
1688 if ( QgsVariantUtils::isNull( variant ) )
1689 return QVariant();
1690
1691 switch ( expectedType )
1692 {
1693 case QMetaType::Type::QString:
1694 {
1695 const QString escaped = variant.toString().replace( '\\', "\\\\"_L1 ).replace( '"', "\\\""_L1 );
1696 return QString( QUrl::toPercentEncoding( escaped, "'" ) );
1697 }
1698
1699 case QMetaType::Type::QDateTime:
1700 case QMetaType::Type::QDate:
1701 {
1702 switch ( variant.userType() )
1703 {
1704 case QMetaType::Type::QDateTime:
1705 return variant.toDateTime().toMSecsSinceEpoch();
1706
1707 case QMetaType::Type::QDate:
1708 // for date values, assume start of day -- the REST api requires datetime values only, not plain dates
1709 if ( context.timeZone().isValid() )
1710 return QDateTime( variant.toDate(), QTime( 0, 0, 0 ), context.timeZone() ).toMSecsSinceEpoch();
1711 else
1712 return QDateTime( variant.toDate(), QTime( 0, 0, 0 ) ).toMSecsSinceEpoch();
1713
1714 default:
1715 return QVariant();
1716 }
1717 }
1718
1719 default:
1720 return variant;
1721 }
1722}
1723
1725{
1726 QVariantMap res;
1727 res.insert( u"name"_s, field.name() );
1728
1729 QString fieldType;
1730 switch ( field.type() )
1731 {
1732 case QMetaType::Type::LongLong:
1733 fieldType = u"esriFieldTypeInteger"_s;
1734 break;
1735
1736 case QMetaType::Type::Int:
1737 fieldType = u"esriFieldTypeSmallInteger"_s;
1738 break;
1739
1740 case QMetaType::Type::Double:
1741 fieldType = u"esriFieldTypeDouble"_s;
1742 break;
1743
1744 case QMetaType::Type::QString:
1745 fieldType = u"esriFieldTypeString"_s;
1746 break;
1747
1748 case QMetaType::Type::QDateTime:
1749 case QMetaType::Type::QDate:
1750 fieldType = u"esriFieldTypeDate"_s;
1751 break;
1752
1753 case QMetaType::Type::QByteArray:
1754 fieldType = u"esriFieldTypeBlob"_s;
1755 break;
1756
1757 default:
1758 // fallback to string
1759 fieldType = u"esriFieldTypeString"_s;
1760 break;
1761 }
1762 res.insert( u"type"_s, fieldType );
1763
1764 if ( !field.alias().isEmpty() )
1765 res.insert( u"alias"_s, field.alias() );
1766
1767 // nullable
1769 res.insert( u"nullable"_s, !notNullable );
1770
1771 // editable
1772 res.insert( u"editable"_s, true );
1773
1774 return res;
1775}
1776
1778{
1779 if ( type.compare( "FeatureServer"_L1, Qt::CaseInsensitive ) == 0 )
1781 else if ( type.compare( "MapServer"_L1, Qt::CaseInsensitive ) == 0 )
1783 else if ( type.compare( "ImageServer"_L1, Qt::CaseInsensitive ) == 0 )
1785 else if ( type.compare( "GlobeServer"_L1, Qt::CaseInsensitive ) == 0 )
1787 else if ( type.compare( "GPServer"_L1, Qt::CaseInsensitive ) == 0 )
1789 else if ( type.compare( "GeocodeServer"_L1, Qt::CaseInsensitive ) == 0 )
1791 else if ( type.compare( "SceneServer"_L1, Qt::CaseInsensitive ) == 0 )
1793
1795}
1796
1798{
1799 const QStringList parts = capabilities.split( ',' );
1800
1802 res.setFlag( Qgis::ArcGisRestServiceCapability::Query, parts.contains( "query"_L1, Qt::CaseInsensitive ) );
1803 res.setFlag( Qgis::ArcGisRestServiceCapability::Map, parts.contains( "map"_L1, Qt::CaseInsensitive ) );
1804 res.setFlag( Qgis::ArcGisRestServiceCapability::Update, parts.contains( "update"_L1, Qt::CaseInsensitive ) );
1805 res.setFlag( Qgis::ArcGisRestServiceCapability::Delete, parts.contains( "delete"_L1, Qt::CaseInsensitive ) );
1806 res.setFlag( Qgis::ArcGisRestServiceCapability::Create, parts.contains( "create"_L1, Qt::CaseInsensitive ) );
1807 res.setFlag( Qgis::ArcGisRestServiceCapability::Image, parts.contains( "image"_L1, Qt::CaseInsensitive ) );
1808 res.setFlag( Qgis::ArcGisRestServiceCapability::TilesOnly, parts.contains( "tilesonly"_L1, Qt::CaseInsensitive ) );
1809
1810 return res;
1811}
1812
1814{
1815 if ( pixelType.compare( "U8"_L1, Qt::CaseInsensitive ) == 0
1816 || pixelType.compare( "U4"_L1, Qt::CaseInsensitive ) == 0
1817 || pixelType.compare( "U2"_L1, Qt::CaseInsensitive ) == 0
1818 || pixelType.compare( "U1"_L1, Qt::CaseInsensitive ) == 0 )
1819 {
1820 return Qgis::DataType::Byte;
1821 }
1822 else if ( pixelType.compare( "S8"_L1, Qt::CaseInsensitive ) == 0 )
1823 {
1824 return Qgis::DataType::Int8;
1825 }
1826 else if ( pixelType.compare( "U16"_L1, Qt::CaseInsensitive ) == 0 )
1827 {
1829 }
1830 else if ( pixelType.compare( "S16"_L1, Qt::CaseInsensitive ) == 0 )
1831 {
1832 return Qgis::DataType::Int16;
1833 }
1834 else if ( pixelType.compare( "U32"_L1, Qt::CaseInsensitive ) == 0 )
1835 {
1837 }
1838 else if ( pixelType.compare( "S32"_L1, Qt::CaseInsensitive ) == 0 )
1839 {
1840 return Qgis::DataType::Int32;
1841 }
1842 else if ( pixelType.compare( "F32"_L1, Qt::CaseInsensitive ) == 0 )
1843 {
1845 }
1846 else if ( pixelType.compare( "F64"_L1, Qt::CaseInsensitive ) == 0 )
1847 {
1849 }
1850 else if ( pixelType.compare( "C64"_L1, Qt::CaseInsensitive ) == 0 )
1851 {
1852 // C64 = 32-bit real + 32-bit imaginary
1854 }
1855 else if ( pixelType.compare( "C128"_L1, Qt::CaseInsensitive ) == 0 )
1856 {
1857 // C128 = 64-bit real + 64-bit imaginary
1859 }
1860 else
1861 {
1862 QgsDebugError( u"Unknown pixelType: %1"_s.arg( pixelType ) );
1863 }
1864
1866}
1867
1869{
1870 if ( pixelType.compare( "U8"_L1, Qt::CaseInsensitive ) == 0
1871 || pixelType.compare( "U4"_L1, Qt::CaseInsensitive ) == 0
1872 || pixelType.compare( "U2"_L1, Qt::CaseInsensitive ) == 0
1873 || pixelType.compare( "U1"_L1, Qt::CaseInsensitive ) == 0
1874 || pixelType.compare( "S8"_L1, Qt::CaseInsensitive ) == 0
1875 || pixelType.compare( "U16"_L1, Qt::CaseInsensitive ) == 0
1876 || pixelType.compare( "S16"_L1, Qt::CaseInsensitive ) == 0 )
1877 {
1878 return PixelTypeLimitUsefulness { true, true };
1879 }
1880 else if ( pixelType.compare( "U32"_L1, Qt::CaseInsensitive ) == 0 )
1881 {
1882 return PixelTypeLimitUsefulness { true, false };
1883 }
1884 else if ( pixelType.compare( "S32"_L1, Qt::CaseInsensitive ) == 0
1885 || pixelType.compare( "F32"_L1, Qt::CaseInsensitive ) == 0
1886 || pixelType.compare( "F64"_L1, Qt::CaseInsensitive ) == 0
1887 || pixelType.compare( "C64"_L1, Qt::CaseInsensitive ) == 0
1888 || pixelType.compare( "C128"_L1, Qt::CaseInsensitive ) == 0 )
1889 {
1890 return PixelTypeLimitUsefulness { false, false };
1891 }
1892 else
1893 {
1894 QgsDebugError( u"Unknown pixelType: %1"_s.arg( pixelType ) );
1895 }
1896
1897 return PixelTypeLimitUsefulness { false, false };
1898}
1899
1900std::optional<std::pair<double, double> > QgsArcGisRestUtils::rangeForPixelType( const QString &pixelType )
1901{
1902 if ( pixelType.compare( "U8"_L1, Qt::CaseInsensitive ) == 0 )
1903 {
1904 return std::make_pair( 0.0, 255.0 );
1905 }
1906 else if ( pixelType.compare( "U4"_L1, Qt::CaseInsensitive ) == 0 )
1907 {
1908 return std::make_pair( 0.0, 15.0 );
1909 }
1910 else if ( pixelType.compare( "U2"_L1, Qt::CaseInsensitive ) == 0 )
1911 {
1912 return std::make_pair( 0.0, 3.0 );
1913 }
1914 else if ( pixelType.compare( "U1"_L1, Qt::CaseInsensitive ) == 0 )
1915 {
1916 return std::make_pair( 0.0, 1.0 );
1917 }
1918 else if ( pixelType.compare( "S8"_L1, Qt::CaseInsensitive ) == 0 )
1919 {
1920 return std::make_pair( -128.0, 127.0 );
1921 }
1922 else if ( pixelType.compare( "U16"_L1, Qt::CaseInsensitive ) == 0 )
1923 {
1924 return std::make_pair( 0.0, 65535.0 );
1925 }
1926 else if ( pixelType.compare( "S16"_L1, Qt::CaseInsensitive ) == 0 )
1927 {
1928 return std::make_pair( -32768.0, 32767.0 );
1929 }
1930 else if ( pixelType.compare( "U32"_L1, Qt::CaseInsensitive ) == 0 )
1931 {
1932 return std::make_pair( 0.0, static_cast<double>( std::numeric_limits<uint32_t>::max() ) );
1933 }
1934 else if ( pixelType.compare( "S32"_L1, Qt::CaseInsensitive ) == 0 )
1935 {
1936 return std::make_pair( static_cast<double>( std::numeric_limits<int32_t>::lowest() ), static_cast<double>( std::numeric_limits<int32_t>::max() ) );
1937 }
1938 else if ( pixelType.compare( "F32"_L1, Qt::CaseInsensitive ) == 0 )
1939 {
1940 return std::make_pair( static_cast<double>( std::numeric_limits<float>::lowest() ), static_cast<double>( std::numeric_limits<float>::max() ) );
1941 }
1942 else if ( pixelType.compare( "F64"_L1, Qt::CaseInsensitive ) == 0 )
1943 {
1944 return std::make_pair( std::numeric_limits<double>::lowest(), std::numeric_limits<double>::max() );
1945 }
1946 else if ( pixelType.compare( "C64"_L1, Qt::CaseInsensitive ) == 0 )
1947 {
1948 // C64 = 32-bit real + 32-bit imaginary
1949 return std::make_pair( static_cast<double>( std::numeric_limits<float>::lowest() ), static_cast<double>( std::numeric_limits<float>::max() ) );
1950 }
1951 else if ( pixelType.compare( "C128"_L1, Qt::CaseInsensitive ) == 0 )
1952 {
1953 // C128 = 64-bit real + 64-bit imaginary
1954 return std::make_pair( std::numeric_limits<double>::lowest(), std::numeric_limits<double>::max() );
1955 }
1956 else
1957 {
1958 QgsDebugError( u"Unknown pixelType: %1"_s.arg( pixelType ) );
1959 }
1960
1961 return std::nullopt;
1962}
1963
1965{
1966 if ( bandName.isEmpty() )
1967 {
1969 }
1970
1971 if ( bandName.compare( "Red"_L1, Qt::CaseInsensitive ) == 0 )
1973 else if ( bandName.compare( "Green"_L1, Qt::CaseInsensitive ) == 0 )
1975 else if ( bandName.compare( "Blue"_L1, Qt::CaseInsensitive ) == 0 )
1977 else if ( bandName.compare( "Alpha"_L1, Qt::CaseInsensitive ) == 0 )
1979 else if ( bandName.compare( "NIR"_L1, Qt::CaseInsensitive ) == 0
1980 || bandName.compare( "NearInfrared"_L1, Qt::CaseInsensitive ) == 0
1981 || bandName.compare( "NearIR"_L1, Qt::CaseInsensitive ) == 0
1982 || bandName.compare( "NarrowNIR"_L1, Qt::CaseInsensitive ) == 0 )
1984 else if ( bandName.startsWith( "SWIR"_L1, Qt::CaseInsensitive ) )
1986 else if ( bandName.startsWith( "VRE"_L1, Qt::CaseInsensitive ) || bandName.compare( "RedEdge"_L1, Qt::CaseInsensitive ) == 0 )
1988 else if ( bandName.startsWith( "Coastal"_L1, Qt::CaseInsensitive ) )
1990 else if ( bandName.startsWith( "Pan"_L1, Qt::CaseInsensitive ) )
1992 else if ( bandName.startsWith( "Thermal"_L1, Qt::CaseInsensitive ) || bandName.startsWith( "TIR"_L1, Qt::CaseInsensitive ) )
1994 else if ( bandName.compare( "Gray"_L1, Qt::CaseInsensitive ) == 0 || bandName.compare( "Grey"_L1, Qt::CaseInsensitive ) == 0 )
1996 else if ( bandName.compare( "Cyan"_L1, Qt::CaseInsensitive ) == 0 )
1998 else if ( bandName.compare( "Magenta"_L1, Qt::CaseInsensitive ) == 0 )
2000 else if ( bandName.compare( "Yellow"_L1, Qt::CaseInsensitive ) == 0 )
2002 else if ( bandName.compare( "Black"_L1, Qt::CaseInsensitive ) == 0 )
2004 else if ( bandName.compare( "Hue"_L1, Qt::CaseInsensitive ) == 0 )
2006 else if ( bandName.compare( "Saturation"_L1, Qt::CaseInsensitive ) == 0 )
2008 else if ( bandName.compare( "Lightness"_L1, Qt::CaseInsensitive ) == 0 )
2010
2011 // we don't log failures here -- a lot of bands will have non-interpretable names,
2012 // eg "Band 1"
2014}
2015
2017{
2018 ok = false;
2019 switch ( type )
2020 {
2022 return 0;
2023
2025 ok = true;
2026 return std::numeric_limits<quint8>::max();
2027
2029 ok = true;
2030 return std::numeric_limits<qint8>::lowest();
2031
2033 ok = true;
2034 return std::numeric_limits<quint16>::max();
2035
2037 ok = true;
2038 return std::numeric_limits<qint16>::lowest();
2039
2041 ok = true;
2042 return std::numeric_limits<quint32>::max();
2043
2045 ok = true;
2046 return std::numeric_limits<qint32>::lowest();
2047
2049 ok = true;
2050 return std::numeric_limits<float>::quiet_NaN();
2051
2053 ok = true;
2054 return std::numeric_limits<double>::quiet_NaN();
2055
2062 return 0;
2063 }
2065}
@ BelowLine
Labels can be placed below a line feature. Unless MapOrientation is also specified this mode respects...
Definition qgis.h:1385
@ MapOrientation
Signifies that the AboveLine and BelowLine flags should respect the map's orientation rather than the...
Definition qgis.h:1386
@ OnLine
Labels can be placed directly over a line feature.
Definition qgis.h:1383
@ AboveLine
Labels can be placed above a line feature. Unless MapOrientation is also specified this mode respects...
Definition qgis.h:1384
@ NoOrientation
Unknown orientation or sentinel value.
Definition qgis.h:3617
@ CounterClockwise
Counter-clockwise direction.
Definition qgis.h:3616
@ Clockwise
Clockwise direction.
Definition qgis.h:3615
@ OverPoint
Arranges candidates over a point (or centroid of a polygon), or at a preset offset from the point....
Definition qgis.h:1276
@ Line
Arranges candidates parallel to a generalised line representing the feature or parallel to a polygon'...
Definition qgis.h:1277
@ Horizontal
Arranges horizontal candidates scattered throughout a polygon feature. Applies to polygon layers only...
Definition qgis.h:1279
ArcGisRestServiceType
Available ArcGIS REST service types.
Definition qgis.h:4741
@ GeocodeServer
GeocodeServer.
Definition qgis.h:4747
@ SceneServer
SceneServer.
Definition qgis.h:4749
@ Unknown
Other unknown/unsupported type.
Definition qgis.h:4748
@ GlobeServer
GlobeServer.
Definition qgis.h:4745
@ ImageServer
ImageServer.
Definition qgis.h:4744
@ FeatureServer
FeatureServer.
Definition qgis.h:4742
@ AboveRight
Above right.
Definition qgis.h:1365
@ BelowLeft
Below left.
Definition qgis.h:1369
@ Above
Above center.
Definition qgis.h:1364
@ BelowRight
Below right.
Definition qgis.h:1371
@ Right
Right middle.
Definition qgis.h:1368
@ AboveLeft
Above left.
Definition qgis.h:1363
@ Below
Below center.
Definition qgis.h:1370
@ Over
Center middle.
Definition qgis.h:1367
DataType
Raster data types.
Definition qgis.h:393
@ CInt32
Complex Int32.
Definition qgis.h:404
@ Float32
Thirty two bit floating point (float).
Definition qgis.h:401
@ CFloat64
Complex Float64.
Definition qgis.h:406
@ Int16
Sixteen bit signed integer (qint16).
Definition qgis.h:398
@ ARGB32_Premultiplied
Color, alpha, red, green, blue, 4 bytes the same as QImage::Format_ARGB32_Premultiplied.
Definition qgis.h:408
@ Int8
Eight bit signed integer (qint8) (added in QGIS 3.30).
Definition qgis.h:396
@ UInt16
Sixteen bit unsigned integer (quint16).
Definition qgis.h:397
@ Byte
Eight bit unsigned integer (quint8).
Definition qgis.h:395
@ UnknownDataType
Unknown or unspecified type.
Definition qgis.h:394
@ ARGB32
Color, alpha, red, green, blue, 4 bytes the same as QImage::Format_ARGB32.
Definition qgis.h:407
@ Int32
Thirty two bit signed integer (qint32).
Definition qgis.h:400
@ Float64
Sixty four bit floating point (double).
Definition qgis.h:402
@ CFloat32
Complex Float32.
Definition qgis.h:405
@ CInt16
Complex Int16.
Definition qgis.h:403
@ UInt32
Thirty two bit unsigned integer (quint32).
Definition qgis.h:399
@ Points
Points (e.g., for font sizes).
Definition qgis.h:5612
QFlags< ArcGisRestServiceCapability > ArcGisRestServiceCapabilities
Available ArcGIS REST service capabilities.
Definition qgis.h:4778
@ Update
Update features.
Definition qgis.h:4765
@ Create
Create features.
Definition qgis.h:4767
@ TilesOnly
Service supports tiled image requests only.
Definition qgis.h:4769
@ Image
Image capabilities.
Definition qgis.h:4768
@ Delete
Delete features.
Definition qgis.h:4766
RasterColorInterpretation
Raster color interpretation.
Definition qgis.h:5136
@ NIRBand
Near-InfraRed (NIR) band [0.75 - 1.40 um].
Definition qgis.h:5162
@ GreenBand
Green band of RGBA image, or green spectral band [0.51 - 0.60 um].
Definition qgis.h:5141
@ SaturationBand
Saturation band of HLS image.
Definition qgis.h:5145
@ MagentaBand
Magenta band of CMYK image.
Definition qgis.h:5148
@ BlackBand
Black band of CMLY image.
Definition qgis.h:5150
@ AlphaBand
Alpha (0=transparent, 255=opaque).
Definition qgis.h:5143
@ SWIRBand
Short-Wavelength InfraRed (SWIR) band [1.40 - 3.00 um].
Definition qgis.h:5163
@ BlueBand
Blue band of RGBA image, or blue spectral band [0.45 - 0.53 um].
Definition qgis.h:5142
@ RedEdgeBand
Red-edge band [0.69 - 0.79 um].
Definition qgis.h:5161
@ YellowBand
Yellow band of CMYK image, or yellow spectral band [0.58 - 0.62 um].
Definition qgis.h:5149
@ CyanBand
Cyan band of CMYK image.
Definition qgis.h:5147
@ LightnessBand
Lightness band of HLS image.
Definition qgis.h:5146
@ CoastalBand
Coastal band [0.40 - 0.45 um].
Definition qgis.h:5160
@ HueBand
Hue band of HLS image.
Definition qgis.h:5144
@ TIRBand
Thermal InfraRed (TIR) band (MWIR or LWIR) [3 - 15 um].
Definition qgis.h:5166
@ RedBand
Red band of RGBA image, or red spectral band [0.62 - 0.69 um].
Definition qgis.h:5140
@ PanBand
Panchromatic band [0.40 - 1.00 um].
Definition qgis.h:5159
WkbType
The WKB type describes the number of dimensions a geometry has.
Definition qgis.h:294
@ CompoundCurve
CompoundCurve.
Definition qgis.h:305
@ Point
Point.
Definition qgis.h:296
@ LineString
LineString.
Definition qgis.h:297
@ MultiPoint
MultiPoint.
Definition qgis.h:300
@ Polygon
Polygon.
Definition qgis.h:298
@ MultiPolygon
MultiPolygon.
Definition qgis.h:302
@ Triangle
Triangle.
Definition qgis.h:299
@ NoGeometry
No geometry.
Definition qgis.h:312
@ MultiLineString
MultiLineString.
Definition qgis.h:301
@ Unknown
Unknown.
Definition qgis.h:295
@ CircularString
CircularString.
Definition qgis.h:304
@ GeometryCollection
GeometryCollection.
Definition qgis.h:303
@ MultiCurve
MultiCurve.
Definition qgis.h:307
@ CurvePolygon
CurvePolygon.
Definition qgis.h:306
@ MultiSurface
MultiSurface.
Definition qgis.h:308
@ Wkt2_2019Simplified
WKT2_2019 with the simplification rule of WKT2_SIMPLIFIED.
Definition qgis.h:2580
Abstract base class for all geometries.
virtual const QgsAbstractGeometry * simplifiedTypeRef() const
Returns a reference to the simplest lossless representation of this geometry, e.g.
bool isMeasure() const
Returns true if the geometry contains m values.
bool is3D() const
Returns true if the geometry is 3D and contains a z-value.
Qgis::WkbType wkbType() const
Returns the WKB type of the geometry.
Contains the context of a ArcGIS REST service operation.
QTimeZone timeZone() const
Returns the time zone for datetime values.
QString objectIdFieldName() const
Returns the name of the objectId field.
static std::unique_ptr< QgsAbstractGeometry > convertGeometry(const QVariantMap &geometry, const QString &esriGeometryType, bool hasM, bool hasZ, bool allowCurves=true, QgsCoordinateReferenceSystem *crs=nullptr)
Converts an ESRI REST geometry JSON definition to a QgsAbstractGeometry.
static QVariantMap fieldDefinitionToJson(const QgsField &field)
Converts a field's definition to an ArcGIS REST JSON representation.
static QgsCoordinateReferenceSystem convertSpatialReference(const QVariantMap &spatialReferenceMap)
Converts a spatial reference JSON definition to a QgsCoordinateReferenceSystem value.
static QDateTime convertDateTime(const QVariant &value)
Converts a date time value to a QDateTime.
static Qgis::WkbType convertGeometryType(const QString &type)
Converts an ESRI REST geometry type to a WKB type.
static QString convertLabelingExpression(const QString &string)
Converts an ESRI labeling expression to a QGIS expression string.
static QVariantMap geometryToJson(const QgsGeometry &geometry, const QgsArcGisRestContext &context, const QgsCoordinateReferenceSystem &crs=QgsCoordinateReferenceSystem())
Converts a geometry to an ArcGIS REST JSON representation.
static double defaultNoDataForDataType(Qgis::DataType type, bool &ok)
Returns a sensible no-data value to use for the specified data type.
static PixelTypeLimitUsefulness pixelTypeLimitUsefulness(const QString &pixelType)
Returns whether the theoretical minimum and maximum values for a given ESRI pixelType are practically...
static Qgis::ArcGisRestServiceType serviceTypeFromString(const QString &type)
Converts a string value to a REST service type.
static QVariant variantToAttributeValue(const QVariant &variant, QMetaType::Type expectedType, const QgsArcGisRestContext &context)
Converts a variant to a REST attribute value.
static QVariantMap featureToJson(const QgsFeature &feature, const QgsArcGisRestContext &context, const QgsCoordinateReferenceSystem &crs=QgsCoordinateReferenceSystem(), QgsArcGisRestUtils::FeatureToJsonFlags flags=QgsArcGisRestUtils::FeatureToJsonFlags(static_cast< int >(QgsArcGisRestUtils::FeatureToJsonFlag::IncludeGeometry)|static_cast< int >(QgsArcGisRestUtils::FeatureToJsonFlag::IncludeNonObjectIdAttributes)))
Converts a feature to an ArcGIS REST JSON representation.
static Qt::PenStyle convertLineStyle(const QString &style)
Converts an ESRI line style to a Qt pen style.
@ IncludeGeometry
Whether to include the geometry definition.
@ IncludeNonObjectIdAttributes
Whether to include any non-objectId attributes.
static Qgis::ArcGisRestServiceCapabilities serviceCapabilitiesFromString(const QString &capabilities)
Parses a capabilities string to known values.
static std::optional< std::pair< double, double > > rangeForPixelType(const QString &pixelType)
Returns the valid data range given an ESRI pixelType string.
static QMetaType::Type convertFieldType(const QString &type)
Converts an ESRI REST field type to a QVariant type.
static Qt::BrushStyle convertFillStyle(const QString &style)
Converts an ESRI fill style to a Qt brush style.
static std::unique_ptr< QgsFeatureRenderer > convertRenderer(const QVariantMap &rendererData, QgsSymbolConverterContext &context)
Converts renderer JSON data to an equivalent QgsFeatureRenderer.
static Qgis::RasterColorInterpretation colorInterpretationFromBandName(const QString &bandName)
Attempts to match arbitrary band name strings to a QGIS raster color interpretation.
static QVariantMap crsToJson(const QgsCoordinateReferenceSystem &crs)
Converts a crs to an ArcGIS REST JSON representation.
static std::unique_ptr< QgsAbstractVectorLayerLabeling > convertLabeling(const QVariantList &data)
Converts labeling JSON data to an equivalent QGIS vector labeling.
QFlags< FeatureToJsonFlag > FeatureToJsonFlags
Flags which control the behavior of converting features to JSON.
static Qgis::DataType dataTypeFromString(const QString &pixelType)
Returns the raster data type corresponding to an ESRI pixelType string.
static std::unique_ptr< QgsSymbol > convertSymbol(const QVariantMap &definition, QgsSymbolConverterContext &context)
Converts a symbol JSON definition to a QgsSymbol.
static QColor convertColor(const QVariant &data)
Converts ESRI JSON color data to a QColor object.
static QgsRectangle convertRectangle(const QVariant &value)
Converts a rectangle value to a QgsRectangle.
A dummy implementation class method which does not compute any breaks.
A classification method which uses equal width intervals.
Implementation of a fixed interval classification.
A classification method for natural breaks, based on Jenks method.
A classification method which creates classes based on quantiles.
A classification method which classifies based on standard deviation of values.
QgsPropertyTransformer subclass for transforming a numeric value into a color from a color ramp.
int nCurves() const
Returns the number of curves in the geometry.
const QgsCurve * curveAt(int i) const
Returns the curve at the specified index.
Represents a coordinate reference system (CRS).
bool isValid() const
Returns whether this CRS is correctly initialized and usable.
bool createFromWkt(const QString &wkt)
Sets this CRS using a WKT definition.
bool createFromString(const QString &definition)
Set up this CRS from a string definition.
QString toWkt(Qgis::CrsWktVariant variant=Qgis::CrsWktVariant::Wkt1Gdal, bool multiline=false, int indentationWidth=4) const
Returns a WKT representation of this CRS.
Curve polygon geometry type.
int numInteriorRings() const
Returns the number of interior rings contained with the curve polygon.
const QgsCurve * exteriorRing() const
Returns the curve polygon's exterior ring.
const QgsCurve * interiorRing(int i) const
Retrieves an interior ring from the curve polygon.
Abstract base class for curved geometry type.
Definition qgscurve.h:36
Qgis::AngularDirection orientation() const
Returns the curve's orientation, e.g.
Definition qgscurve.cpp:291
virtual QgsCurve * reversed() const =0
Returns a reversed copy of the curve, where the direction of the curve has been flipped.
virtual QgsPoint endPoint() const =0
Returns the end point of the curve.
Handles parsing and evaluation of expressions (formerly called "search strings").
static QString quotedColumnRef(QString name)
Returns a quoted column reference (in double quotes).
bool isValid() const
Checks if this expression is valid.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
QgsFields fields
Definition qgsfeature.h:65
QgsGeometry geometry
Definition qgsfeature.h:66
bool hasGeometry() const
Returns true if the feature has an associated geometry.
Q_INVOKABLE QVariant attribute(const QString &name) const
Lookup attribute value by attribute name.
@ ConstraintNotNull
Field may not be null.
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:56
QMetaType::Type type
Definition qgsfield.h:63
QString name
Definition qgsfield.h:65
QString alias
Definition qgsfield.h:66
QgsFieldConstraints constraints
Definition qgsfield.h:68
Container of fields for a vector layer.
Definition qgsfields.h:46
int numGeometries() const
Returns the number of geometries within the collection.
const QgsAbstractGeometry * geometryN(int n) const
Returns a const reference to a geometry from within the collection.
A geometry is the spatial representation of a feature.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
Represents a color stop within a QgsGradientColorRamp color ramp.
void setPlacementFlags(Qgis::LabelLinePlacementFlags flags)
Returns the line placement flags, which dictate how line labels can be placed above or below the line...
void setQuadrant(Qgis::LabelQuadrantPosition quadrant)
Sets the quadrant in which to offset labels from the point.
Line string geometry type, with support for z-dimension and m-values.
QgsLineString * reversed() const override
Returns a reversed copy of the curve, where the direction of the curve has been flipped.
Multi curve geometry collection.
QgsCurve * curveN(int index)
Returns the curve with the specified index.
Multi line string geometry collection.
QgsLineString * lineStringN(int index)
Returns the line string with the specified index.
Multi point geometry collection.
QgsPoint * pointN(int index)
Returns the point with the specified index.
Multi polygon geometry collection.
QgsPolygon * polygonN(int index)
Returns the polygon with the specified index.
Multi surface geometry collection.
Contains settings for how a map layer will be labeled.
void setFormat(const QgsTextFormat &format)
Sets the label text formatting settings, e.g., font settings, buffer settings, etc.
Qgis::LabelPlacement placement
Label placement mode.
bool isExpression
true if this label is made from a expression string, e.g., FieldName || 'mm'
const QgsLabelLineSettings & lineSettings() const
Returns the label line settings, which contain settings related to how the label engine places and fo...
QString fieldName
Name of field (or an expression) to use for label text.
const QgsLabelPointSettings & pointSettings() const
Returns the label point settings, which contain settings related to how the label engine places and f...
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:53
void clear() override
Clears the geometry, ie reset it to a null geometry.
Definition qgspoint.cpp:391
double z
Definition qgspoint.h:58
double x
Definition qgspoint.h:56
bool isEmpty() const override
Returns true if the geometry is empty.
Definition qgspoint.cpp:777
double m
Definition qgspoint.h:59
double y
Definition qgspoint.h:57
Polygon geometry type.
Definition qgspolygon.h:37
A store for object properties.
void setTransformer(QgsPropertyTransformer *transformer)
Sets an optional transformer to use for manipulating the calculated values for the property.
static QgsProperty fromExpression(const QString &expression, bool isActive=true)
Returns a new ExpressionBasedProperty created from the specified expression.
static QgsProperty fromField(const QString &fieldName, bool isActive=true)
Returns a new FieldBasedProperty created from the specified field name.
A container for the context for various read/write operations on objects.
A rectangle specified with double values.
Represents an individual category (class) from a QgsCategorizedSymbolRenderer.
Represents a value range for a QgsGraduatedSymbolRenderer.
void setUpperValue(double upperValue)
Sets the upper bound of the range.
void setSymbol(QgsSymbol *s)
Sets the symbol used for the range.
void setLabel(const QString &label)
Sets the label used for the range.
void setLowerValue(double lowerValue)
Sets the lower bound of the range.
A child rule for QgsRuleBasedLabeling.
void setActive(bool state)
Sets if this rule is active.
void appendChild(QgsRuleBasedLabeling::Rule *rule)
add child rule, take ownership, sets this as parent
int numPoints() const override
Returns the number of points in the curve.
bool isEmpty() const override
Returns true if the geometry is empty.
const double * mData() const
Returns a const pointer to the m vertex data, or nullptr if the simple curve does not have m values.
const double * yData() const
Returns a const pointer to the y vertex data.
QgsPoint startPoint() const override
Returns the starting point of the curve.
QgsPoint pointN(int i) const
Returns the specified point from inside the simple curve.
const double * xData() const
Returns a const pointer to the x vertex data.
const double * zData() const
Returns a const pointer to the z vertex data, or nullptr if the simple curve does not have z values.
Represents the context in which a QgsSymbolConverter conversion occurs.
void pushWarning(const QString &warning)
Pushes a warning message generated during the conversion.
A symbol converter for converting ESRI REST JSON symbols.
static Qt::PenStyle convertLineStyle(const QString &style)
Converts an ESRI line style to a Qt pen style.
std::unique_ptr< QgsSymbol > createSymbol(const QVariant &variant, QgsSymbolConverterContext &context) const override
Creates a new QgsSymbol from a QVariant representation.
static Qt::BrushStyle convertFillStyle(const QString &style)
Converts an ESRI fill style to a Qt brush style.
static QColor convertColor(const QVariant &data)
Converts ESRI JSON color data to a QColor object.
virtual void setDataDefinedProperty(Property key, const QgsProperty &property)
Sets a data defined property for the layer.
Abstract base class for all rendered symbols.
Definition qgssymbol.h:227
QgsSymbolLayer * symbolLayer(int layer)
Returns the symbol layer at the specified index.
int symbolLayerCount() const
Returns the total number of symbol layers contained in the symbol.
Definition qgssymbol.h:357
Container for settings relating to a text buffer.
void setColor(const QColor &color)
Sets the color for the buffer.
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units used for the buffer size.
void setEnabled(bool enabled)
Sets whether the text buffer will be drawn.
void setSize(double size)
Sets the size of the buffer.
Container for all settings relating to text rendering.
void setColor(const QColor &color)
Sets the color that text will be rendered in.
void setSize(double size)
Sets the size for rendered text.
void setFont(const QFont &font)
Sets the font used for rendering text.
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units for the size of rendered text.
void setBuffer(const QgsTextBufferSettings &bufferSettings)
Sets the text's buffer settings.
static bool isNull(const QVariant &variant, bool silenceNullWarnings=false)
Returns true if the specified variant should be considered a NULL value.
static Qgis::WkbType zmType(Qgis::WkbType type, bool hasZ, bool hasM)
Returns the modified input geometry type according to hasZ / hasM.
static Q_INVOKABLE bool hasZ(Qgis::WkbType type)
Tests whether a WKB type contains the z-dimension.
static Q_INVOKABLE bool hasM(Qgis::WkbType type)
Tests whether a WKB type contains m values.
static Qgis::WkbType flatType(Qgis::WkbType type)
Returns the flat type for a WKB type.
#define BUILTIN_UNREACHABLE
Definition qgis.h:7974
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference).
Definition qgis.h:7340
T qgsgeometry_cast(QgsAbstractGeometry *geom)
QList< QgsRendererCategory > QgsCategoryList
QList< QgsGradientStop > QgsGradientStopsList
List of gradient stops.
#define QgsDebugError(str)
Definition qgslogger.h:71
Struct representing whether the theoretical limits of a pixel type are useful for describing actual d...