QGIS API Documentation 4.1.0-Master (0cdd3ae6384)
Loading...
Searching...
No Matches
qgsmapboxglstyleconverter.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsmapboxglstyleconverter.cpp
3 --------------------------------------
4 Date : September 2020
5 Copyright : (C) 2020 by Nyall Dawson
6 Email : nyall dot dawson at gmail dot com
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
17/*
18 * Ported from original work by Martin Dobias, and extended by the MapTiler team!
19 */
20
22
23#include "qgis.h"
24#include "qgsapplication.h"
25#include "qgsblureffect.h"
26#include "qgseffectstack.h"
27#include "qgsfillsymbol.h"
28#include "qgsfillsymbollayer.h"
29#include "qgsfontmanager.h"
30#include "qgsfontutils.h"
31#include "qgsjsonutils.h"
32#include "qgslinesymbol.h"
33#include "qgslinesymbollayer.h"
34#include "qgslogger.h"
35#include "qgsmarkersymbol.h"
37#include "qgspainteffect.h"
38#include "qgsproviderregistry.h"
39#include "qgsrasterlayer.h"
40#include "qgsrasterpipe.h"
41#include "qgssymbollayer.h"
42#include "qgssymbollayerutils.h"
46
47#include <QBuffer>
48#include <QRegularExpression>
49#include <QString>
50
51#include "moc_qgsmapboxglstyleconverter.cpp"
52
53using namespace Qt::StringLiterals;
54
57
59{
60 mError.clear();
61 mWarnings.clear();
62
63 if ( style.contains( u"sources"_s ) )
64 {
65 parseSources( style.value( u"sources"_s ).toMap(), context );
66 }
67
68 if ( style.contains( u"layers"_s ) )
69 {
70 parseLayers( style.value( u"layers"_s ).toList(), context );
71 }
72 else
73 {
74 mError = QObject::tr( "Could not find layers list in JSON" );
75 return NoLayerList;
76 }
77 return Success;
78}
79
84
86{
87 qDeleteAll( mSources );
88}
89
91{
92 std::unique_ptr< QgsMapBoxGlStyleConversionContext > tmpContext;
93 if ( !context )
94 {
95 tmpContext = std::make_unique< QgsMapBoxGlStyleConversionContext >();
96 context = tmpContext.get();
97 }
98
99 QList<QgsVectorTileBasicRendererStyle> rendererStyles;
100 QList<QgsVectorTileBasicLabelingStyle> labelingStyles;
101
102 QgsVectorTileBasicRendererStyle rendererBackgroundStyle;
103 bool hasRendererBackgroundStyle = false;
104
105 for ( const QVariant &layer : layers )
106 {
107 const QVariantMap jsonLayer = layer.toMap();
108
109 const QString layerType = jsonLayer.value( u"type"_s ).toString();
110 if ( layerType == "background"_L1 )
111 {
112 hasRendererBackgroundStyle = parseFillLayer( jsonLayer, rendererBackgroundStyle, *context, true );
113 if ( hasRendererBackgroundStyle )
114 {
115 rendererBackgroundStyle.setStyleName( layerType );
116 rendererBackgroundStyle.setLayerName( layerType );
117 rendererBackgroundStyle.setFilterExpression( QString() );
118 rendererBackgroundStyle.setEnabled( true );
119 }
120 continue;
121 }
122
123 const QString styleId = jsonLayer.value( u"id"_s ).toString();
124 context->setLayerId( styleId );
125
126 if ( layerType.compare( "raster"_L1, Qt::CaseInsensitive ) == 0 )
127 {
128 QgsMapBoxGlStyleRasterSubLayer raster( styleId, jsonLayer.value( u"source"_s ).toString() );
129 const QVariantMap jsonPaint = jsonLayer.value( u"paint"_s ).toMap();
130 if ( jsonPaint.contains( u"raster-opacity"_s ) )
131 {
132 const QVariant jsonRasterOpacity = jsonPaint.value( u"raster-opacity"_s );
133 double defaultOpacity = 1;
134 raster.dataDefinedProperties().setProperty( QgsRasterPipe::Property::RendererOpacity, parseInterpolateByZoom( jsonRasterOpacity.toMap(), *context, 100, &defaultOpacity ) );
135 }
136
137 mRasterSubLayers.append( raster );
138 continue;
139 }
140
141 const QString layerName = jsonLayer.value( u"source-layer"_s ).toString();
142
143 const int minZoom = jsonLayer.value( u"minzoom"_s, u"-1"_s ).toInt();
144
145 // WARNING -- the QGIS renderers for vector tiles treat maxzoom different to the MapBox Style Specifications.
146 // from the MapBox Specifications:
147 //
148 // "The maximum zoom level for the layer. At zoom levels equal to or greater than the maxzoom, the layer will be hidden."
149 //
150 // However the QGIS styles will be hidden if the zoom level is GREATER THAN (not equal to) maxzoom.
151 // Accordingly we need to subtract 1 from the maxzoom value in the JSON:
152 int maxZoom = jsonLayer.value( u"maxzoom"_s, u"-1"_s ).toInt();
153 if ( maxZoom != -1 )
154 maxZoom--;
155
156 QString visibilyStr;
157 if ( jsonLayer.contains( u"visibility"_s ) )
158 {
159 visibilyStr = jsonLayer.value( u"visibility"_s ).toString();
160 }
161 else if ( jsonLayer.contains( u"layout"_s ) && jsonLayer.value( u"layout"_s ).userType() == QMetaType::Type::QVariantMap )
162 {
163 const QVariantMap jsonLayout = jsonLayer.value( u"layout"_s ).toMap();
164 visibilyStr = jsonLayout.value( u"visibility"_s ).toString();
165 }
166
167 const bool enabled = visibilyStr != "none"_L1;
168
169 QString filterExpression;
170 if ( jsonLayer.contains( u"filter"_s ) )
171 {
172 filterExpression = parseExpression( jsonLayer.value( u"filter"_s ).toList(), *context );
173 }
174
177
178 bool hasRendererStyle = false;
179 bool hasLabelingStyle = false;
180 if ( layerType == "fill"_L1 )
181 {
182 hasRendererStyle = parseFillLayer( jsonLayer, rendererStyle, *context );
183 }
184 else if ( layerType == "line"_L1 )
185 {
186 hasRendererStyle = parseLineLayer( jsonLayer, rendererStyle, *context );
187 }
188 else if ( layerType == "circle"_L1 )
189 {
190 hasRendererStyle = parseCircleLayer( jsonLayer, rendererStyle, *context );
191 }
192 else if ( layerType == "symbol"_L1 )
193 {
194 parseSymbolLayer( jsonLayer, rendererStyle, hasRendererStyle, labelingStyle, hasLabelingStyle, *context );
195 }
196 else
197 {
198 mWarnings << QObject::tr( "%1: Skipping unknown layer type %2" ).arg( context->layerId(), layerType );
199 QgsDebugError( mWarnings.constLast() );
200 continue;
201 }
202
203 if ( hasRendererStyle )
204 {
205 rendererStyle.setStyleName( styleId );
206 rendererStyle.setLayerName( layerName );
207 rendererStyle.setFilterExpression( filterExpression );
208 rendererStyle.setMinZoomLevel( minZoom );
209 rendererStyle.setMaxZoomLevel( maxZoom );
210 rendererStyle.setEnabled( enabled );
211 rendererStyles.append( rendererStyle );
212 }
213
214 if ( hasLabelingStyle )
215 {
216 labelingStyle.setStyleName( styleId );
217 labelingStyle.setLayerName( layerName );
218 labelingStyle.setFilterExpression( filterExpression );
219 labelingStyle.setMinZoomLevel( minZoom );
220 labelingStyle.setMaxZoomLevel( maxZoom );
221 labelingStyle.setEnabled( enabled );
222 labelingStyles.append( labelingStyle );
223 }
224
225 mWarnings.append( context->warnings() );
226 context->clearWarnings();
227 }
228
229 if ( hasRendererBackgroundStyle )
230 rendererStyles.prepend( rendererBackgroundStyle );
231
232 auto renderer = std::make_unique< QgsVectorTileBasicRenderer >();
233 renderer->setStyles( rendererStyles );
234 mRenderer = std::move( renderer );
235
236 auto labeling = std::make_unique< QgsVectorTileBasicLabeling >();
237 labeling->setStyles( labelingStyles );
238 mLabeling = std::move( labeling );
239}
240
241bool QgsMapBoxGlStyleConverter::parseFillLayer( const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &style, QgsMapBoxGlStyleConversionContext &context, bool isBackgroundStyle )
242{
243 const QVariantMap jsonPaint = jsonLayer.value( u"paint"_s ).toMap();
244
245 QgsPropertyCollection ddProperties;
246 QgsPropertyCollection ddRasterProperties;
247
248 bool colorIsDataDefined = false;
249
250 std::unique_ptr< QgsSymbol > symbol( std::make_unique< QgsFillSymbol >() );
251
252 // fill color
253 QColor fillColor;
254 if ( jsonPaint.contains( isBackgroundStyle ? u"background-color"_s : u"fill-color"_s ) )
255 {
256 const QVariant jsonFillColor = jsonPaint.value( isBackgroundStyle ? u"background-color"_s : u"fill-color"_s );
257 switch ( jsonFillColor.userType() )
258 {
259 case QMetaType::Type::QVariantMap:
260 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseInterpolateColorByZoom( jsonFillColor.toMap(), context, &fillColor ) );
261 break;
262
263 case QMetaType::Type::QVariantList:
264 case QMetaType::Type::QStringList:
265 colorIsDataDefined = true;
266 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseValueList( jsonFillColor.toList(), PropertyType::Color, context, 1, 255, &fillColor ) );
267 break;
268
269 case QMetaType::Type::QString:
270 fillColor = parseColor( jsonFillColor.toString(), context );
271 break;
272
273 default:
274 {
275 context.pushWarning( QObject::tr( "%1: Skipping unsupported fill-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonFillColor.userType() ) ) ) );
276 break;
277 }
278 }
279 }
280 else
281 {
282 // defaults to #000000
283 fillColor = QColor( 0, 0, 0 );
284 }
285
286 QColor fillOutlineColor;
287 if ( !isBackgroundStyle )
288 {
289 if ( !jsonPaint.contains( u"fill-outline-color"_s ) )
290 {
291 if ( fillColor.isValid() )
292 fillOutlineColor = fillColor;
293
294 // match fill color data defined property when active
295 if ( ddProperties.isActive( QgsSymbolLayer::Property::FillColor ) )
297 }
298 else
299 {
300 const QVariant jsonFillOutlineColor = jsonPaint.value( u"fill-outline-color"_s );
301 switch ( jsonFillOutlineColor.userType() )
302 {
303 case QMetaType::Type::QVariantMap:
304 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseInterpolateColorByZoom( jsonFillOutlineColor.toMap(), context, &fillOutlineColor ) );
305 break;
306
307 case QMetaType::Type::QVariantList:
308 case QMetaType::Type::QStringList:
309 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseValueList( jsonFillOutlineColor.toList(), PropertyType::Color, context, 1, 255, &fillOutlineColor ) );
310 break;
311
312 case QMetaType::Type::QString:
313 fillOutlineColor = parseColor( jsonFillOutlineColor.toString(), context );
314 break;
315
316 default:
317 context.pushWarning(
318 QObject::tr( "%1: Skipping unsupported fill-outline-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonFillOutlineColor.userType() ) ) )
319 );
320 break;
321 }
322 }
323 }
324
325 double fillOpacity = -1.0;
326 double rasterOpacity = -1.0;
327 if ( jsonPaint.contains( isBackgroundStyle ? u"background-opacity"_s : u"fill-opacity"_s ) )
328 {
329 const QVariant jsonFillOpacity = jsonPaint.value( isBackgroundStyle ? u"background-opacity"_s : u"fill-opacity"_s );
330 switch ( jsonFillOpacity.userType() )
331 {
332 case QMetaType::Type::Int:
333 case QMetaType::Type::LongLong:
334 case QMetaType::Type::Double:
335 fillOpacity = jsonFillOpacity.toDouble();
336 rasterOpacity = fillOpacity;
337 break;
338
339 case QMetaType::Type::QVariantMap:
340 if ( ddProperties.isActive( QgsSymbolLayer::Property::FillColor ) )
341 {
342 symbol->setDataDefinedProperty( QgsSymbol::Property::Opacity, parseInterpolateByZoom( jsonFillOpacity.toMap(), context, 100 ) );
343 }
344 else
345 {
346 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseInterpolateOpacityByZoom( jsonFillOpacity.toMap(), fillColor.isValid() ? fillColor.alpha() : 255, &context ) );
347 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseInterpolateOpacityByZoom( jsonFillOpacity.toMap(), fillOutlineColor.isValid() ? fillOutlineColor.alpha() : 255, &context ) );
348 ddRasterProperties.setProperty( QgsSymbolLayer::Property::Opacity, parseInterpolateByZoom( jsonFillOpacity.toMap(), context, 100, &rasterOpacity ) );
349 }
350 break;
351
352 case QMetaType::Type::QVariantList:
353 case QMetaType::Type::QStringList:
354 if ( ddProperties.isActive( QgsSymbolLayer::Property::FillColor ) )
355 {
356 symbol->setDataDefinedProperty( QgsSymbol::Property::Opacity, parseValueList( jsonFillOpacity.toList(), PropertyType::Numeric, context, 100, 100 ) );
357 }
358 else
359 {
360 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseValueList( jsonFillOpacity.toList(), PropertyType::Opacity, context, 1, fillColor.isValid() ? fillColor.alpha() : 255 ) );
361 ddProperties
362 .setProperty( QgsSymbolLayer::Property::StrokeColor, parseValueList( jsonFillOpacity.toList(), PropertyType::Opacity, context, 1, fillOutlineColor.isValid() ? fillOutlineColor.alpha() : 255 ) );
363 ddRasterProperties.setProperty( QgsSymbolLayer::Property::Opacity, parseValueList( jsonFillOpacity.toList(), PropertyType::Numeric, context, 100, 255, nullptr, &rasterOpacity ) );
364 }
365 break;
366
367 default:
368 context.pushWarning( QObject::tr( "%1: Skipping unsupported fill-opacity type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonFillOpacity.userType() ) ) ) );
369 break;
370 }
371 }
372
373 // fill-translate
374 QPointF fillTranslate;
375 if ( jsonPaint.contains( u"fill-translate"_s ) )
376 {
377 const QVariant jsonFillTranslate = jsonPaint.value( u"fill-translate"_s );
378 switch ( jsonFillTranslate.userType() )
379 {
380 case QMetaType::Type::QVariantMap:
381 ddProperties.setProperty( QgsSymbolLayer::Property::Offset, parseInterpolatePointByZoom( jsonFillTranslate.toMap(), context, context.pixelSizeConversionFactor(), &fillTranslate ) );
382 break;
383
384 case QMetaType::Type::QVariantList:
385 case QMetaType::Type::QStringList:
386 fillTranslate
387 = QPointF( jsonFillTranslate.toList().value( 0 ).toDouble() * context.pixelSizeConversionFactor(), jsonFillTranslate.toList().value( 1 ).toDouble() * context.pixelSizeConversionFactor() );
388 break;
389
390 default:
391 context.pushWarning(
392 QObject::tr( "%1: Skipping unsupported fill-translate type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonFillTranslate.userType() ) ) )
393 );
394 break;
395 }
396 }
397
398 QgsSimpleFillSymbolLayer *fillSymbol = dynamic_cast< QgsSimpleFillSymbolLayer * >( symbol->symbolLayer( 0 ) );
399 Q_ASSERT( fillSymbol ); // should not fail since QgsFillSymbol() constructor instantiates a QgsSimpleFillSymbolLayer
400
401 // set render units
402 symbol->setOutputUnit( context.targetUnit() );
403 fillSymbol->setOutputUnit( context.targetUnit() );
404
405 if ( !fillTranslate.isNull() )
406 {
407 fillSymbol->setOffset( fillTranslate );
408 }
409 fillSymbol->setOffsetUnit( context.targetUnit() );
410
411 if ( jsonPaint.contains( isBackgroundStyle ? u"background-pattern"_s : u"fill-pattern"_s ) )
412 {
413 // get fill-pattern to set sprite
414
415 const QVariant fillPatternJson = jsonPaint.value( isBackgroundStyle ? u"background-pattern"_s : u"fill-pattern"_s );
416
417 // fill-pattern disabled dillcolor
418 fillColor = QColor();
419 fillOutlineColor = QColor();
420
421 // fill-pattern can be String or Object
422 // String: {"fill-pattern": "dash-t"}
423 // Object: {"fill-pattern":{"stops":[[11,"wetland8"],[12,"wetland16"]]}}
424
425 QSize spriteSize;
426 QString spriteProperty, spriteSizeProperty;
427 const QString sprite = retrieveSpriteAsBase64WithProperties( fillPatternJson, context, spriteSize, spriteProperty, spriteSizeProperty );
428 if ( !sprite.isEmpty() )
429 {
430 // when fill-pattern exists, set and insert QgsRasterFillSymbolLayer
432 rasterFill->setImageFilePath( sprite );
433 rasterFill->setWidth( spriteSize.width() );
434 rasterFill->setSizeUnit( context.targetUnit() );
436
437 if ( rasterOpacity >= 0 )
438 {
439 rasterFill->setOpacity( rasterOpacity );
440 }
441
442 if ( !spriteProperty.isEmpty() )
443 {
444 ddRasterProperties.setProperty( QgsSymbolLayer::Property::File, QgsProperty::fromExpression( spriteProperty ) );
445 ddRasterProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( spriteSizeProperty ) );
446 }
447
448 rasterFill->setDataDefinedProperties( ddRasterProperties );
449 symbol->appendSymbolLayer( rasterFill );
450 }
451 }
452
453 fillSymbol->setDataDefinedProperties( ddProperties );
454
455 if ( fillOpacity != -1 )
456 {
457 symbol->setOpacity( fillOpacity );
458 }
459
460 // some complex logic here!
461 // by default a MapBox fill style will share the same stroke color as the fill color.
462 // This is generally desirable and the 1px stroke can help to hide boundaries between features which
463 // would otherwise be visible due to antialiasing effects.
464 // BUT if the outline color is semi-transparent, then drawing the stroke will result in a double rendering
465 // of strokes for adjacent polygons, resulting in visible seams between tiles. Accordingly, we only
466 // set the stroke color if it's a completely different color to the fill (ie the style designer explicitly
467 // wants a visible stroke) OR the stroke color is opaque and the double-rendering artifacts aren't an issue
468 if ( fillOutlineColor.isValid() && ( fillOutlineColor.alpha() == 255 || fillOutlineColor != fillColor ) )
469 {
470 // mapbox fill strokes are always 1 px wide
471 fillSymbol->setStrokeWidth( 0 );
472 fillSymbol->setStrokeColor( fillOutlineColor );
473 }
474 else
475 {
476 fillSymbol->setStrokeStyle( Qt::NoPen );
477 }
478
479 if ( fillColor.isValid() )
480 {
481 fillSymbol->setFillColor( fillColor );
482 }
483 else if ( colorIsDataDefined )
484 {
485 fillSymbol->setFillColor( QColor( Qt::transparent ) );
486 }
487 else
488 {
489 fillSymbol->setBrushStyle( Qt::NoBrush );
490 }
491
493 style.setSymbol( symbol.release() );
494 return true;
495}
496
498{
499 if ( !jsonLayer.contains( u"paint"_s ) )
500 {
501 context.pushWarning( QObject::tr( "%1: Style has no paint property, skipping" ).arg( context.layerId() ) );
502 return false;
503 }
504
505 QgsPropertyCollection ddProperties;
506 QString rasterLineSprite;
507
508 const QVariantMap jsonPaint = jsonLayer.value( u"paint"_s ).toMap();
509 if ( jsonPaint.contains( u"line-pattern"_s ) )
510 {
511 const QVariant jsonLinePattern = jsonPaint.value( u"line-pattern"_s );
512 switch ( jsonLinePattern.userType() )
513 {
514 case QMetaType::Type::QVariantMap:
515 case QMetaType::Type::QString:
516 {
517 QSize spriteSize;
518 QString spriteProperty, spriteSizeProperty;
519 rasterLineSprite = retrieveSpriteAsBase64WithProperties( jsonLinePattern, context, spriteSize, spriteProperty, spriteSizeProperty );
521 break;
522 }
523
524 case QMetaType::Type::QVariantList:
525 case QMetaType::Type::QStringList:
526 default:
527 break;
528 }
529
530 if ( rasterLineSprite.isEmpty() )
531 {
532 // unsupported line-pattern definition, moving on
533 context.pushWarning( QObject::tr( "%1: Skipping unsupported line-pattern property" ).arg( context.layerId() ) );
534 return false;
535 }
536 }
537
538 // line color
539 QColor lineColor;
540 if ( jsonPaint.contains( u"line-color"_s ) )
541 {
542 const QVariant jsonLineColor = jsonPaint.value( u"line-color"_s );
543 switch ( jsonLineColor.userType() )
544 {
545 case QMetaType::Type::QVariantMap:
546 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseInterpolateColorByZoom( jsonLineColor.toMap(), context, &lineColor ) );
548 break;
549
550 case QMetaType::Type::QVariantList:
551 case QMetaType::Type::QStringList:
552 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseValueList( jsonLineColor.toList(), PropertyType::Color, context, 1, 255, &lineColor ) );
554 break;
555
556 case QMetaType::Type::QString:
557 lineColor = parseColor( jsonLineColor.toString(), context );
558 break;
559
560 default:
561 context.pushWarning( QObject::tr( "%1: Skipping unsupported line-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonLineColor.userType() ) ) ) );
562 break;
563 }
564 }
565 else
566 {
567 // defaults to #000000
568 lineColor = QColor( 0, 0, 0 );
569 }
570
571
572 double lineWidth = 1.0 * context.pixelSizeConversionFactor();
573 QgsProperty lineWidthProperty;
574 if ( jsonPaint.contains( u"line-width"_s ) )
575 {
576 const QVariant jsonLineWidth = jsonPaint.value( u"line-width"_s );
577 switch ( jsonLineWidth.userType() )
578 {
579 case QMetaType::Type::Int:
580 case QMetaType::Type::LongLong:
581 case QMetaType::Type::Double:
582 lineWidth = jsonLineWidth.toDouble() * context.pixelSizeConversionFactor();
583 break;
584
585 case QMetaType::Type::QVariantMap:
586 {
587 lineWidth = -1;
588 lineWidthProperty = parseInterpolateByZoom( jsonLineWidth.toMap(), context, context.pixelSizeConversionFactor(), &lineWidth );
589 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeWidth, lineWidthProperty );
590 // set symbol layer visibility depending on line width since QGIS displays line with 0 width as hairlines
591 QgsProperty layerEnabledProperty = QgsProperty( lineWidthProperty );
592 layerEnabledProperty.setExpressionString( u"(%1) > 0"_s.arg( lineWidthProperty.expressionString() ) );
593 ddProperties.setProperty( QgsSymbolLayer::Property::LayerEnabled, layerEnabledProperty );
594 break;
595 }
596
597 case QMetaType::Type::QVariantList:
598 case QMetaType::Type::QStringList:
599 {
600 lineWidthProperty = parseValueList( jsonLineWidth.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &lineWidth );
601 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeWidth, lineWidthProperty );
602 // set symbol layer visibility depending on line width since QGIS displays line with 0 width as hairlines
603 QgsProperty layerEnabledProperty = QgsProperty( lineWidthProperty );
604 layerEnabledProperty.setExpressionString( u"(%1) > 0"_s.arg( lineWidthProperty.expressionString() ) );
605 ddProperties.setProperty( QgsSymbolLayer::Property::LayerEnabled, layerEnabledProperty );
606 break;
607 }
608
609 default:
610 context.pushWarning( QObject::tr( "%1: Skipping unsupported fill-width type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonLineWidth.userType() ) ) ) );
611 break;
612 }
613 }
614
615 double lineOffset = 0.0;
616 if ( jsonPaint.contains( u"line-offset"_s ) )
617 {
618 const QVariant jsonLineOffset = jsonPaint.value( u"line-offset"_s );
619 switch ( jsonLineOffset.userType() )
620 {
621 case QMetaType::Type::Int:
622 case QMetaType::Type::LongLong:
623 case QMetaType::Type::Double:
624 lineOffset = -jsonLineOffset.toDouble() * context.pixelSizeConversionFactor();
625 break;
626
627 case QMetaType::Type::QVariantMap:
628 lineWidth = -1;
629 ddProperties.setProperty( QgsSymbolLayer::Property::Offset, parseInterpolateByZoom( jsonLineOffset.toMap(), context, context.pixelSizeConversionFactor() * -1, &lineOffset ) );
630 break;
631
632 case QMetaType::Type::QVariantList:
633 case QMetaType::Type::QStringList:
634 ddProperties.setProperty( QgsSymbolLayer::Property::Offset, parseValueList( jsonLineOffset.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor() * -1, 255, nullptr, &lineOffset ) );
635 break;
636
637 default:
638 context.pushWarning( QObject::tr( "%1: Skipping unsupported line-offset type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonLineOffset.userType() ) ) ) );
639 break;
640 }
641 }
642
643 double lineOpacity = -1.0;
644 QgsProperty lineOpacityProperty;
645 if ( jsonPaint.contains( u"line-opacity"_s ) )
646 {
647 const QVariant jsonLineOpacity = jsonPaint.value( u"line-opacity"_s );
648 switch ( jsonLineOpacity.userType() )
649 {
650 case QMetaType::Type::Int:
651 case QMetaType::Type::LongLong:
652 case QMetaType::Type::Double:
653 lineOpacity = jsonLineOpacity.toDouble();
654 break;
655
656 case QMetaType::Type::QVariantMap:
658 {
659 double defaultValue = 1.0;
660 lineOpacityProperty = parseInterpolateByZoom( jsonLineOpacity.toMap(), context, 100, &defaultValue );
661 }
662 else
663 {
664 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseInterpolateOpacityByZoom( jsonLineOpacity.toMap(), lineColor.isValid() ? lineColor.alpha() : 255, &context ) );
665 }
666 break;
667
668 case QMetaType::Type::QVariantList:
669 case QMetaType::Type::QStringList:
671 {
672 double defaultValue = 1.0;
673 QColor invalidColor;
674 lineOpacityProperty = parseValueList( jsonLineOpacity.toList(), PropertyType::Numeric, context, 100, 255, &invalidColor, &defaultValue );
675 }
676 else
677 {
678 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseValueList( jsonLineOpacity.toList(), PropertyType::Opacity, context, 1, lineColor.isValid() ? lineColor.alpha() : 255 ) );
679 }
680 break;
681
682 default:
683 context.pushWarning( QObject::tr( "%1: Skipping unsupported line-opacity type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonLineOpacity.userType() ) ) ) );
684 break;
685 }
686 }
687
688 QVector< double > dashVector;
689 if ( jsonPaint.contains( u"line-dasharray"_s ) )
690 {
691 const QVariant jsonLineDashArray = jsonPaint.value( u"line-dasharray"_s );
692 switch ( jsonLineDashArray.userType() )
693 {
694 case QMetaType::Type::QVariantMap:
695 {
696 QString arrayExpression;
697 if ( !lineWidthProperty.asExpression().isEmpty() )
698 {
699 arrayExpression = u"array_to_string(array_foreach(%1,@element * (%2)), ';')"_s // skip-keyword-check
700 .arg( parseArrayStops( jsonLineDashArray.toMap().value( u"stops"_s ).toList(), context, 1 ), lineWidthProperty.asExpression() );
701 }
702 else
703 {
704 arrayExpression = u"array_to_string(%1, ';')"_s.arg( parseArrayStops( jsonLineDashArray.toMap().value( u"stops"_s ).toList(), context, lineWidth ) );
705 }
707
708 const QVariantList dashSource = jsonLineDashArray.toMap().value( u"stops"_s ).toList().first().toList().value( 1 ).toList();
709 for ( const QVariant &v : dashSource )
710 {
711 dashVector << v.toDouble() * lineWidth;
712 }
713 break;
714 }
715
716 case QMetaType::Type::QVariantList:
717 case QMetaType::Type::QStringList:
718 {
719 const QVariantList dashSource = jsonLineDashArray.toList();
720
721 if ( !dashSource.empty() )
722 {
723 if ( dashSource.at( 0 ).userType() == QMetaType::Type::QString )
724 {
725 QgsProperty property = parseValueList( dashSource, PropertyType::DashArray, context, 1, 255, nullptr, nullptr );
726 if ( !lineWidthProperty.asExpression().isEmpty() )
727 {
729 u"array_to_string(array_foreach(%1,@element * (%2)), ';')"_s // skip-keyword-check
730 .arg( property.asExpression(), lineWidthProperty.asExpression() )
731 );
732 }
733 else
734 {
735 property = QgsProperty::fromExpression( u"array_to_string(%1, ';')"_s.arg( property.asExpression() ) );
736 }
737 ddProperties.setProperty( QgsSymbolLayer::Property::CustomDash, property );
738 }
739 else
740 {
741 QVector< double > rawDashVectorSizes;
742 rawDashVectorSizes.reserve( dashSource.size() );
743 for ( const QVariant &v : dashSource )
744 {
745 rawDashVectorSizes << v.toDouble();
746 }
747
748 // handle non-compliant dash vector patterns
749 if ( rawDashVectorSizes.size() == 1 )
750 {
751 // match behavior of MapBox style rendering -- if a user makes a line dash array with one element, it's ignored
752 rawDashVectorSizes.clear();
753 }
754 else if ( rawDashVectorSizes.size() % 2 == 1 )
755 {
756 // odd number of dash pattern sizes -- this isn't permitted by Qt/QGIS, but isn't explicitly blocked by the MapBox specs
757 // MapBox seems to implicitly add a 0 length gap to the array if odd length.
758 rawDashVectorSizes.append( 0 );
759 }
760
761 if ( !rawDashVectorSizes.isEmpty() && ( !lineWidthProperty.asExpression().isEmpty() ) )
762 {
763 QStringList dashArrayStringParts;
764 dashArrayStringParts.reserve( rawDashVectorSizes.size() );
765 for ( double v : std::as_const( rawDashVectorSizes ) )
766 {
767 dashArrayStringParts << qgsDoubleToString( v );
768 }
769
770 QString arrayExpression = u"array_to_string(array_foreach(array(%1),@element * (%2)), ';')"_s // skip-keyword-check
771 .arg( dashArrayStringParts.join( ',' ), lineWidthProperty.asExpression() );
773 }
774
775 // dash vector sizes for QGIS symbols must be multiplied by the target line width
776 for ( double v : std::as_const( rawDashVectorSizes ) )
777 {
778 dashVector << v * lineWidth;
779 }
780 }
781 }
782 break;
783 }
784
785 default:
786 context.pushWarning(
787 QObject::tr( "%1: Skipping unsupported line-dasharray type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonLineDashArray.userType() ) ) )
788 );
789 break;
790 }
791 }
792
793 Qt::PenCapStyle penCapStyle = Qt::FlatCap;
794 Qt::PenJoinStyle penJoinStyle = Qt::MiterJoin;
795 if ( jsonLayer.contains( u"layout"_s ) )
796 {
797 const QVariantMap jsonLayout = jsonLayer.value( u"layout"_s ).toMap();
798 if ( jsonLayout.contains( u"line-cap"_s ) )
799 {
800 penCapStyle = parseCapStyle( jsonLayout.value( u"line-cap"_s ).toString() );
801 }
802 if ( jsonLayout.contains( u"line-join"_s ) )
803 {
804 penJoinStyle = parseJoinStyle( jsonLayout.value( u"line-join"_s ).toString() );
805 }
806 }
807
808 std::unique_ptr< QgsSymbol > symbol( std::make_unique< QgsLineSymbol >() );
809 symbol->setOutputUnit( context.targetUnit() );
810
811 if ( !rasterLineSprite.isEmpty() )
812 {
813 QgsRasterLineSymbolLayer *lineSymbol = new QgsRasterLineSymbolLayer( rasterLineSprite );
814 lineSymbol->setOutputUnit( context.targetUnit() );
815 lineSymbol->setPenCapStyle( penCapStyle );
816 lineSymbol->setPenJoinStyle( penJoinStyle );
817 lineSymbol->setDataDefinedProperties( ddProperties );
818 lineSymbol->setOffset( lineOffset );
819 lineSymbol->setOffsetUnit( context.targetUnit() );
820
821 if ( lineOpacity != -1 )
822 {
823 symbol->setOpacity( lineOpacity );
824 }
825 if ( !lineOpacityProperty.asExpression().isEmpty() )
826 {
827 QgsPropertyCollection ddProperties;
828 ddProperties.setProperty( QgsSymbol::Property::Opacity, lineOpacityProperty );
829 symbol->setDataDefinedProperties( ddProperties );
830 }
831 if ( lineWidth != -1 )
832 {
833 lineSymbol->setWidth( lineWidth );
834 }
835 symbol->changeSymbolLayer( 0, lineSymbol );
836 }
837 else
838 {
839 QgsSimpleLineSymbolLayer *lineSymbol = dynamic_cast< QgsSimpleLineSymbolLayer * >( symbol->symbolLayer( 0 ) );
840 Q_ASSERT( lineSymbol ); // should not fail since QgsLineSymbol() constructor instantiates a QgsSimpleLineSymbolLayer
841
842 // set render units
843 lineSymbol->setOutputUnit( context.targetUnit() );
844 lineSymbol->setPenCapStyle( penCapStyle );
845 lineSymbol->setPenJoinStyle( penJoinStyle );
846 lineSymbol->setDataDefinedProperties( ddProperties );
847 lineSymbol->setOffset( lineOffset );
848 lineSymbol->setOffsetUnit( context.targetUnit() );
849
850 if ( lineOpacity != -1 )
851 {
852 symbol->setOpacity( lineOpacity );
853 }
854 if ( !lineOpacityProperty.asExpression().isEmpty() )
855 {
856 QgsPropertyCollection ddProperties;
857 ddProperties.setProperty( QgsSymbol::Property::Opacity, lineOpacityProperty );
858 symbol->setDataDefinedProperties( ddProperties );
859 }
860 if ( lineColor.isValid() )
861 {
862 lineSymbol->setColor( lineColor );
863 }
864 if ( lineWidth != -1 )
865 {
866 lineSymbol->setWidth( lineWidth );
867 }
868 if ( !dashVector.empty() )
869 {
870 lineSymbol->setUseCustomDashPattern( true );
871 lineSymbol->setCustomDashVector( dashVector );
872 }
873 }
874
876 style.setSymbol( symbol.release() );
877 return true;
878}
879
881{
882 if ( !jsonLayer.contains( u"paint"_s ) )
883 {
884 context.pushWarning( QObject::tr( "%1: Style has no paint property, skipping" ).arg( context.layerId() ) );
885 return false;
886 }
887
888 const QVariantMap jsonPaint = jsonLayer.value( u"paint"_s ).toMap();
889
890 QgsPropertyCollection ddProperties;
891
892 // circle color
893 QColor circleFillColor;
894 if ( jsonPaint.contains( u"circle-color"_s ) )
895 {
896 const QVariant jsonCircleColor = jsonPaint.value( u"circle-color"_s );
897 switch ( jsonCircleColor.userType() )
898 {
899 case QMetaType::Type::QVariantMap:
900 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseInterpolateColorByZoom( jsonCircleColor.toMap(), context, &circleFillColor ) );
901 break;
902
903 case QMetaType::Type::QVariantList:
904 case QMetaType::Type::QStringList:
905 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseValueList( jsonCircleColor.toList(), PropertyType::Color, context, 1, 255, &circleFillColor ) );
906 break;
907
908 case QMetaType::Type::QString:
909 circleFillColor = parseColor( jsonCircleColor.toString(), context );
910 break;
911
912 default:
913 context.pushWarning( QObject::tr( "%1: Skipping unsupported circle-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonCircleColor.userType() ) ) ) );
914 break;
915 }
916 }
917 else
918 {
919 // defaults to #000000
920 circleFillColor = QColor( 0, 0, 0 );
921 }
922
923 // circle radius
924 double circleDiameter = 10.0;
925 if ( jsonPaint.contains( u"circle-radius"_s ) )
926 {
927 const QVariant jsonCircleRadius = jsonPaint.value( u"circle-radius"_s );
928 switch ( jsonCircleRadius.userType() )
929 {
930 case QMetaType::Type::Int:
931 case QMetaType::Type::LongLong:
932 case QMetaType::Type::Double:
933 circleDiameter = jsonCircleRadius.toDouble() * context.pixelSizeConversionFactor() * 2;
934 break;
935
936 case QMetaType::Type::QVariantMap:
937 circleDiameter = -1;
938 ddProperties.setProperty( QgsSymbolLayer::Property::Size, parseInterpolateByZoom( jsonCircleRadius.toMap(), context, context.pixelSizeConversionFactor() * 2, &circleDiameter ) );
939 break;
940
941 case QMetaType::Type::QVariantList:
942 case QMetaType::Type::QStringList:
943 ddProperties.setProperty( QgsSymbolLayer::Property::Size, parseValueList( jsonCircleRadius.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor() * 2, 255, nullptr, &circleDiameter ) );
944 break;
945
946 default:
947 context.pushWarning(
948 QObject::tr( "%1: Skipping unsupported circle-radius type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonCircleRadius.userType() ) ) )
949 );
950 break;
951 }
952 }
953
954 double circleOpacity = -1.0;
955 if ( jsonPaint.contains( u"circle-opacity"_s ) )
956 {
957 const QVariant jsonCircleOpacity = jsonPaint.value( u"circle-opacity"_s );
958 switch ( jsonCircleOpacity.userType() )
959 {
960 case QMetaType::Type::Int:
961 case QMetaType::Type::LongLong:
962 case QMetaType::Type::Double:
963 circleOpacity = jsonCircleOpacity.toDouble();
964 break;
965
966 case QMetaType::Type::QVariantMap:
967 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseInterpolateOpacityByZoom( jsonCircleOpacity.toMap(), circleFillColor.isValid() ? circleFillColor.alpha() : 255, &context ) );
968 break;
969
970 case QMetaType::Type::QVariantList:
971 case QMetaType::Type::QStringList:
972 ddProperties.setProperty( QgsSymbolLayer::Property::FillColor, parseValueList( jsonCircleOpacity.toList(), PropertyType::Opacity, context, 1, circleFillColor.isValid() ? circleFillColor.alpha() : 255 ) );
973 break;
974
975 default:
976 context.pushWarning(
977 QObject::tr( "%1: Skipping unsupported circle-opacity type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonCircleOpacity.userType() ) ) )
978 );
979 break;
980 }
981 }
982 if ( ( circleOpacity != -1 ) && circleFillColor.isValid() )
983 {
984 circleFillColor.setAlphaF( circleOpacity );
985 }
986
987 // circle stroke color
988 QColor circleStrokeColor;
989 if ( jsonPaint.contains( u"circle-stroke-color"_s ) )
990 {
991 const QVariant jsonCircleStrokeColor = jsonPaint.value( u"circle-stroke-color"_s );
992 switch ( jsonCircleStrokeColor.userType() )
993 {
994 case QMetaType::Type::QVariantMap:
995 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseInterpolateColorByZoom( jsonCircleStrokeColor.toMap(), context, &circleStrokeColor ) );
996 break;
997
998 case QMetaType::Type::QVariantList:
999 case QMetaType::Type::QStringList:
1000 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseValueList( jsonCircleStrokeColor.toList(), PropertyType::Color, context, 1, 255, &circleStrokeColor ) );
1001 break;
1002
1003 case QMetaType::Type::QString:
1004 circleStrokeColor = parseColor( jsonCircleStrokeColor.toString(), context );
1005 break;
1006
1007 default:
1008 context.pushWarning(
1009 QObject::tr( "%1: Skipping unsupported circle-stroke-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonCircleStrokeColor.userType() ) ) )
1010 );
1011 break;
1012 }
1013 }
1014
1015 // circle stroke width
1016 double circleStrokeWidth = -1.0;
1017 if ( jsonPaint.contains( u"circle-stroke-width"_s ) )
1018 {
1019 const QVariant circleStrokeWidthJson = jsonPaint.value( u"circle-stroke-width"_s );
1020 switch ( circleStrokeWidthJson.userType() )
1021 {
1022 case QMetaType::Type::Int:
1023 case QMetaType::Type::LongLong:
1024 case QMetaType::Type::Double:
1025 circleStrokeWidth = circleStrokeWidthJson.toDouble() * context.pixelSizeConversionFactor();
1026 break;
1027
1028 case QMetaType::Type::QVariantMap:
1029 circleStrokeWidth = -1.0;
1030 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeWidth, parseInterpolateByZoom( circleStrokeWidthJson.toMap(), context, context.pixelSizeConversionFactor(), &circleStrokeWidth ) );
1031 break;
1032
1033 case QMetaType::Type::QVariantList:
1034 case QMetaType::Type::QStringList:
1035 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeWidth, parseValueList( circleStrokeWidthJson.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &circleStrokeWidth ) );
1036 break;
1037
1038 default:
1039 context.pushWarning(
1040 QObject::tr( "%1: Skipping unsupported circle-stroke-width type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( circleStrokeWidthJson.userType() ) ) )
1041 );
1042 break;
1043 }
1044 }
1045
1046 double circleStrokeOpacity = -1.0;
1047 if ( jsonPaint.contains( u"circle-stroke-opacity"_s ) )
1048 {
1049 const QVariant jsonCircleStrokeOpacity = jsonPaint.value( u"circle-stroke-opacity"_s );
1050 switch ( jsonCircleStrokeOpacity.userType() )
1051 {
1052 case QMetaType::Type::Int:
1053 case QMetaType::Type::LongLong:
1054 case QMetaType::Type::Double:
1055 circleStrokeOpacity = jsonCircleStrokeOpacity.toDouble();
1056 break;
1057
1058 case QMetaType::Type::QVariantMap:
1059 ddProperties.setProperty( QgsSymbolLayer::Property::StrokeColor, parseInterpolateOpacityByZoom( jsonCircleStrokeOpacity.toMap(), circleStrokeColor.isValid() ? circleStrokeColor.alpha() : 255, &context ) );
1060 break;
1061
1062 case QMetaType::Type::QVariantList:
1063 case QMetaType::Type::QStringList:
1064 ddProperties
1065 .setProperty( QgsSymbolLayer::Property::StrokeColor, parseValueList( jsonCircleStrokeOpacity.toList(), PropertyType::Opacity, context, 1, circleStrokeColor.isValid() ? circleStrokeColor.alpha() : 255 ) );
1066 break;
1067
1068 default:
1069 context.pushWarning(
1070 QObject::tr( "%1: Skipping unsupported circle-stroke-opacity type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonCircleStrokeOpacity.userType() ) ) )
1071 );
1072 break;
1073 }
1074 }
1075 if ( ( circleStrokeOpacity != -1 ) && circleStrokeColor.isValid() )
1076 {
1077 circleStrokeColor.setAlphaF( circleStrokeOpacity );
1078 }
1079
1080 // translate
1081 QPointF circleTranslate;
1082 if ( jsonPaint.contains( u"circle-translate"_s ) )
1083 {
1084 const QVariant jsonCircleTranslate = jsonPaint.value( u"circle-translate"_s );
1085 switch ( jsonCircleTranslate.userType() )
1086 {
1087 case QMetaType::Type::QVariantMap:
1088 ddProperties.setProperty( QgsSymbolLayer::Property::Offset, parseInterpolatePointByZoom( jsonCircleTranslate.toMap(), context, context.pixelSizeConversionFactor(), &circleTranslate ) );
1089 break;
1090
1091 case QMetaType::Type::QVariantList:
1092 case QMetaType::Type::QStringList:
1093 circleTranslate
1094 = QPointF( jsonCircleTranslate.toList().value( 0 ).toDouble() * context.pixelSizeConversionFactor(), jsonCircleTranslate.toList().value( 1 ).toDouble() * context.pixelSizeConversionFactor() );
1095 break;
1096
1097 default:
1098 context.pushWarning(
1099 QObject::tr( "%1: Skipping unsupported circle-translate type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonCircleTranslate.userType() ) ) )
1100 );
1101 break;
1102 }
1103 }
1104
1105 std::unique_ptr< QgsSymbol > symbol( std::make_unique< QgsMarkerSymbol >() );
1106 QgsSimpleMarkerSymbolLayer *markerSymbolLayer = dynamic_cast< QgsSimpleMarkerSymbolLayer * >( symbol->symbolLayer( 0 ) );
1107 Q_ASSERT( markerSymbolLayer );
1108
1109 // set render units
1110 symbol->setOutputUnit( context.targetUnit() );
1111 markerSymbolLayer->setDataDefinedProperties( ddProperties );
1112
1113 if ( !circleTranslate.isNull() )
1114 {
1115 markerSymbolLayer->setOffset( circleTranslate );
1116 markerSymbolLayer->setOffsetUnit( context.targetUnit() );
1117 }
1118
1119 if ( circleFillColor.isValid() )
1120 {
1121 markerSymbolLayer->setFillColor( circleFillColor );
1122 }
1123 if ( circleDiameter != -1 )
1124 {
1125 markerSymbolLayer->setSize( circleDiameter );
1126 markerSymbolLayer->setSizeUnit( context.targetUnit() );
1127 }
1128 if ( circleStrokeColor.isValid() )
1129 {
1130 markerSymbolLayer->setStrokeColor( circleStrokeColor );
1131 }
1132 if ( circleStrokeWidth != -1 )
1133 {
1134 markerSymbolLayer->setStrokeWidth( circleStrokeWidth );
1135 markerSymbolLayer->setStrokeWidthUnit( context.targetUnit() );
1136 }
1137
1139 style.setSymbol( symbol.release() );
1140 return true;
1141}
1142
1144 const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &renderer, bool &hasRenderer, QgsVectorTileBasicLabelingStyle &labelingStyle, bool &hasLabeling, QgsMapBoxGlStyleConversionContext &context
1145)
1146{
1147 hasLabeling = false;
1148 hasRenderer = false;
1149
1150 if ( !jsonLayer.contains( u"layout"_s ) )
1151 {
1152 context.pushWarning( QObject::tr( "%1: Style layer has no layout property, skipping" ).arg( context.layerId() ) );
1153 return;
1154 }
1155 const QVariantMap jsonLayout = jsonLayer.value( u"layout"_s ).toMap();
1156 if ( !jsonLayout.contains( u"text-field"_s ) )
1157 {
1158 hasRenderer = parseSymbolLayerAsRenderer( jsonLayer, renderer, context );
1159 return;
1160 }
1161
1162 const QVariantMap jsonPaint = jsonLayer.value( u"paint"_s ).toMap();
1163
1164 QgsPropertyCollection ddLabelProperties;
1165
1166 double textSize = 16.0 * context.pixelSizeConversionFactor();
1167 QgsProperty textSizeProperty;
1168 if ( jsonLayout.contains( u"text-size"_s ) )
1169 {
1170 const QVariant jsonTextSize = jsonLayout.value( u"text-size"_s );
1171 switch ( jsonTextSize.userType() )
1172 {
1173 case QMetaType::Type::Int:
1174 case QMetaType::Type::LongLong:
1175 case QMetaType::Type::Double:
1176 textSize = jsonTextSize.toDouble() * context.pixelSizeConversionFactor();
1177 break;
1178
1179 case QMetaType::Type::QVariantMap:
1180 textSize = -1;
1181 textSizeProperty = parseInterpolateByZoom( jsonTextSize.toMap(), context, context.pixelSizeConversionFactor(), &textSize );
1182
1183 break;
1184
1185 case QMetaType::Type::QVariantList:
1186 case QMetaType::Type::QStringList:
1187 textSize = -1;
1188 textSizeProperty = parseValueList( jsonTextSize.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &textSize );
1189 break;
1190
1191 default:
1192 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-size type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextSize.userType() ) ) ) );
1193 break;
1194 }
1195
1196 if ( textSizeProperty )
1197 {
1198 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::Size, textSizeProperty );
1199 }
1200 }
1201
1202 // a rough average of ems to character count conversion for a variety of fonts
1203 constexpr double EM_TO_CHARS = 2.0;
1204
1205 double textMaxWidth = -1;
1206 if ( jsonLayout.contains( u"text-max-width"_s ) )
1207 {
1208 const QVariant jsonTextMaxWidth = jsonLayout.value( u"text-max-width"_s );
1209 switch ( jsonTextMaxWidth.userType() )
1210 {
1211 case QMetaType::Type::Int:
1212 case QMetaType::Type::LongLong:
1213 case QMetaType::Type::Double:
1214 textMaxWidth = jsonTextMaxWidth.toDouble() * EM_TO_CHARS;
1215 break;
1216
1217 case QMetaType::Type::QVariantMap:
1218 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::AutoWrapLength, parseInterpolateByZoom( jsonTextMaxWidth.toMap(), context, EM_TO_CHARS, &textMaxWidth ) );
1219 break;
1220
1221 case QMetaType::Type::QVariantList:
1222 case QMetaType::Type::QStringList:
1223 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::AutoWrapLength, parseValueList( jsonTextMaxWidth.toList(), PropertyType::Numeric, context, EM_TO_CHARS, 255, nullptr, &textMaxWidth ) );
1224 break;
1225
1226 default:
1227 context.pushWarning(
1228 QObject::tr( "%1: Skipping unsupported text-max-width type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextMaxWidth.userType() ) ) )
1229 );
1230 break;
1231 }
1232 }
1233 else
1234 {
1235 // defaults to 10
1236 textMaxWidth = 10 * EM_TO_CHARS;
1237 }
1238
1239 double textLetterSpacing = -1;
1240 if ( jsonLayout.contains( u"text-letter-spacing"_s ) )
1241 {
1242 const QVariant jsonTextLetterSpacing = jsonLayout.value( u"text-letter-spacing"_s );
1243 switch ( jsonTextLetterSpacing.userType() )
1244 {
1245 case QMetaType::Type::Int:
1246 case QMetaType::Type::LongLong:
1247 case QMetaType::Type::Double:
1248 textLetterSpacing = jsonTextLetterSpacing.toDouble();
1249 break;
1250
1251 case QMetaType::Type::QVariantMap:
1252 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::FontLetterSpacing, parseInterpolateByZoom( jsonTextLetterSpacing.toMap(), context, 1, &textLetterSpacing ) );
1253 break;
1254
1255 case QMetaType::Type::QVariantList:
1256 case QMetaType::Type::QStringList:
1257 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::FontLetterSpacing, parseValueList( jsonTextLetterSpacing.toList(), PropertyType::Numeric, context, 1, 255, nullptr, &textLetterSpacing ) );
1258 break;
1259
1260 default:
1261 context.pushWarning(
1262 QObject::tr( "%1: Skipping unsupported text-letter-spacing type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextLetterSpacing.userType() ) ) )
1263 );
1264 break;
1265 }
1266 }
1267
1268 QFont textFont;
1269 bool foundFont = false;
1270 QString fontName;
1271 QString fontStyleName;
1272
1273 bool allowOverlap = jsonLayout.contains( u"text-allow-overlap"_s ) && jsonLayout.value( u"text-allow-overlap"_s ).toBool();
1274
1275 if ( jsonLayout.contains( u"text-font"_s ) )
1276 {
1277 auto splitFontFamily = []( const QString &fontName, QString &family, QString &style ) -> bool {
1278 QString matchedFamily;
1279 const QStringList textFontParts = fontName.split( ' ' );
1280 for ( int i = textFontParts.size() - 1; i >= 1; --i )
1281 {
1282 const QString candidateFontFamily = textFontParts.mid( 0, i ).join( ' ' );
1283 const QString candidateFontStyle = textFontParts.mid( i ).join( ' ' );
1284
1285 const QString processedFontFamily = QgsApplication::fontManager()->processFontFamilyName( candidateFontFamily );
1286 if ( QgsFontUtils::fontFamilyHasStyle( processedFontFamily, candidateFontStyle ) )
1287 {
1288 family = processedFontFamily;
1289 style = candidateFontStyle;
1290 return true;
1291 }
1292 else if ( QgsApplication::fontManager()->tryToDownloadFontFamily( processedFontFamily, matchedFamily ) )
1293 {
1294 if ( processedFontFamily == matchedFamily )
1295 {
1296 family = processedFontFamily;
1297 style = candidateFontStyle;
1298 }
1299 else
1300 {
1301 family = matchedFamily;
1302 style = processedFontFamily;
1303 style.replace( matchedFamily, QString() );
1304 style = style.trimmed();
1305 if ( !style.isEmpty() && !candidateFontStyle.isEmpty() )
1306 {
1307 style += u" %1"_s.arg( candidateFontStyle );
1308 }
1309 }
1310 return true;
1311 }
1312 }
1313
1314 const QString processedFontFamily = QgsApplication::fontManager()->processFontFamilyName( fontName );
1315 if ( QFontDatabase().hasFamily( processedFontFamily ) )
1316 {
1317 // the json isn't following the spec correctly!!
1318 family = processedFontFamily;
1319 style.clear();
1320 return true;
1321 }
1322 else if ( QgsApplication::fontManager()->tryToDownloadFontFamily( processedFontFamily, matchedFamily ) )
1323 {
1324 family = matchedFamily;
1325 style.clear();
1326 return true;
1327 }
1328 return false;
1329 };
1330
1331 const QVariant jsonTextFont = jsonLayout.value( u"text-font"_s );
1332 if ( jsonTextFont.userType() != QMetaType::Type::QVariantList
1333 && jsonTextFont.userType() != QMetaType::Type::QStringList
1334 && jsonTextFont.userType() != QMetaType::Type::QString
1335 && jsonTextFont.userType() != QMetaType::Type::QVariantMap )
1336 {
1337 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-font type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextFont.userType() ) ) ) );
1338 }
1339 else
1340 {
1341 switch ( jsonTextFont.userType() )
1342 {
1343 case QMetaType::Type::QVariantList:
1344 case QMetaType::Type::QStringList:
1345 fontName = jsonTextFont.toList().value( 0 ).toString();
1346 break;
1347
1348 case QMetaType::Type::QString:
1349 fontName = jsonTextFont.toString();
1350 break;
1351
1352 case QMetaType::Type::QVariantMap:
1353 {
1354 QString familyCaseString = u"CASE "_s;
1355 QString styleCaseString = u"CASE "_s;
1356 QString fontFamily;
1357 const QVariantList stops = jsonTextFont.toMap().value( u"stops"_s ).toList();
1358
1359 bool error = false;
1360 for ( int i = 0; i < stops.length() - 1; ++i )
1361 {
1362 // bottom zoom and value
1363 const QVariant bz = stops.value( i ).toList().value( 0 );
1364 const QString bv = stops.value( i ).toList().value( 1 ).userType() == QMetaType::Type::QString ? stops.value( i ).toList().value( 1 ).toString()
1365 : stops.value( i ).toList().value( 1 ).toList().value( 0 ).toString();
1366 if ( bz.userType() == QMetaType::Type::QVariantList || bz.userType() == QMetaType::Type::QStringList )
1367 {
1368 context.pushWarning( QObject::tr( "%1: Expressions in interpolation function are not supported, skipping." ).arg( context.layerId() ) );
1369 error = true;
1370 break;
1371 }
1372
1373 // top zoom
1374 const QVariant tz = stops.value( i + 1 ).toList().value( 0 );
1375 if ( tz.userType() == QMetaType::Type::QVariantList || tz.userType() == QMetaType::Type::QStringList )
1376 {
1377 context.pushWarning( QObject::tr( "%1: Expressions in interpolation function are not supported, skipping." ).arg( context.layerId() ) );
1378 error = true;
1379 break;
1380 }
1381
1382 if ( splitFontFamily( bv, fontFamily, fontStyleName ) )
1383 {
1384 familyCaseString += QStringLiteral(
1385 "WHEN @vector_tile_zoom > %1 AND @vector_tile_zoom <= %2 "
1386 "THEN %3 "
1387 )
1388 .arg( bz.toString(), tz.toString(), QgsExpression::quotedValue( fontFamily ) );
1389 styleCaseString += QStringLiteral(
1390 "WHEN @vector_tile_zoom > %1 AND @vector_tile_zoom <= %2 "
1391 "THEN %3 "
1392 )
1393 .arg( bz.toString(), tz.toString(), QgsExpression::quotedValue( fontStyleName ) );
1394 }
1395 else
1396 {
1397 context.pushWarning( QObject::tr( "%1: Referenced font %2 is not available on system" ).arg( context.layerId(), bv ) );
1398 }
1399 }
1400 if ( error )
1401 break;
1402
1403 const QString bv = stops.constLast().toList().value( 1 ).userType() == QMetaType::Type::QString ? stops.constLast().toList().value( 1 ).toString()
1404 : stops.constLast().toList().value( 1 ).toList().value( 0 ).toString();
1405 if ( splitFontFamily( bv, fontFamily, fontStyleName ) )
1406 {
1407 familyCaseString += u"ELSE %1 END"_s.arg( QgsExpression::quotedValue( fontFamily ) );
1408 styleCaseString += u"ELSE %1 END"_s.arg( QgsExpression::quotedValue( fontStyleName ) );
1409 }
1410 else
1411 {
1412 context.pushWarning( QObject::tr( "%1: Referenced font %2 is not available on system" ).arg( context.layerId(), bv ) );
1413 }
1414
1415 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::Family, QgsProperty::fromExpression( familyCaseString ) );
1417
1418 foundFont = true;
1419 fontName = fontFamily;
1420
1421 break;
1422 }
1423
1424 default:
1425 break;
1426 }
1427
1428 QString fontFamily;
1429 if ( splitFontFamily( fontName, fontFamily, fontStyleName ) )
1430 {
1431 textFont = QgsFontUtils::createFont( fontFamily );
1432 if ( !fontStyleName.isEmpty() )
1433 textFont.setStyleName( fontStyleName );
1434 foundFont = true;
1435 }
1436 }
1437 }
1438 else
1439 {
1440 // Defaults to ["Open Sans Regular","Arial Unicode MS Regular"].
1441 if ( QgsFontUtils::fontFamilyHasStyle( u"Open Sans"_s, u"Regular"_s ) )
1442 {
1443 fontName = u"Open Sans"_s;
1444 textFont = QgsFontUtils::createFont( fontName );
1445 textFont.setStyleName( u"Regular"_s );
1446 fontStyleName = u"Regular"_s;
1447 foundFont = true;
1448 }
1449 else if ( QgsFontUtils::fontFamilyHasStyle( u"Arial Unicode MS"_s, u"Regular"_s ) )
1450 {
1451 fontName = u"Arial Unicode MS"_s;
1452 textFont = QgsFontUtils::createFont( fontName );
1453 textFont.setStyleName( u"Regular"_s );
1454 fontStyleName = u"Regular"_s;
1455 foundFont = true;
1456 }
1457 else
1458 {
1459 fontName = u"Open Sans, Arial Unicode MS"_s;
1460 }
1461 }
1462 if ( !foundFont && !fontName.isEmpty() )
1463 {
1464 context.pushWarning( QObject::tr( "%1: Referenced font %2 is not available on system" ).arg( context.layerId(), fontName ) );
1465 }
1466
1467 // text color
1468 QColor textColor;
1469 if ( jsonPaint.contains( u"text-color"_s ) )
1470 {
1471 const QVariant jsonTextColor = jsonPaint.value( u"text-color"_s );
1472 switch ( jsonTextColor.userType() )
1473 {
1474 case QMetaType::Type::QVariantMap:
1475 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::Color, parseInterpolateColorByZoom( jsonTextColor.toMap(), context, &textColor ) );
1476 break;
1477
1478 case QMetaType::Type::QVariantList:
1479 case QMetaType::Type::QStringList:
1480 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::Color, parseValueList( jsonTextColor.toList(), PropertyType::Color, context, 1, 255, &textColor ) );
1481 break;
1482
1483 case QMetaType::Type::QString:
1484 textColor = parseColor( jsonTextColor.toString(), context );
1485 break;
1486
1487 default:
1488 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextColor.userType() ) ) ) );
1489 break;
1490 }
1491 }
1492 else
1493 {
1494 // defaults to #000000
1495 textColor = QColor( 0, 0, 0 );
1496 }
1497
1498 // buffer color
1499 QColor bufferColor( 0, 0, 0, 0 );
1500 if ( jsonPaint.contains( u"text-halo-color"_s ) )
1501 {
1502 const QVariant jsonBufferColor = jsonPaint.value( u"text-halo-color"_s );
1503 switch ( jsonBufferColor.userType() )
1504 {
1505 case QMetaType::Type::QVariantMap:
1506 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::BufferColor, parseInterpolateColorByZoom( jsonBufferColor.toMap(), context, &bufferColor ) );
1507 break;
1508
1509 case QMetaType::Type::QVariantList:
1510 case QMetaType::Type::QStringList:
1511 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::BufferColor, parseValueList( jsonBufferColor.toList(), PropertyType::Color, context, 1, 255, &bufferColor ) );
1512 break;
1513
1514 case QMetaType::Type::QString:
1515 bufferColor = parseColor( jsonBufferColor.toString(), context );
1516 break;
1517
1518 default:
1519 context.pushWarning(
1520 QObject::tr( "%1: Skipping unsupported text-halo-color type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonBufferColor.userType() ) ) )
1521 );
1522 break;
1523 }
1524 }
1525
1526 double bufferSize = 0.0;
1527 // the pixel based text buffers appear larger when rendered on the web - so automatically scale
1528 // them up when converting to a QGIS style
1529 // (this number is based on trial-and-error comparisons only!)
1530 constexpr double BUFFER_SIZE_SCALE = 2.0;
1531 if ( jsonPaint.contains( u"text-halo-width"_s ) )
1532 {
1533 const QVariant jsonHaloWidth = jsonPaint.value( u"text-halo-width"_s );
1534 QString bufferSizeDataDefined;
1535 switch ( jsonHaloWidth.userType() )
1536 {
1537 case QMetaType::Type::Int:
1538 case QMetaType::Type::LongLong:
1539 case QMetaType::Type::Double:
1540 bufferSize = jsonHaloWidth.toDouble() * context.pixelSizeConversionFactor() * BUFFER_SIZE_SCALE;
1541 break;
1542
1543 case QMetaType::Type::QVariantMap:
1544 bufferSize = 1;
1545 bufferSizeDataDefined = parseInterpolateByZoom( jsonHaloWidth.toMap(), context, context.pixelSizeConversionFactor() * BUFFER_SIZE_SCALE, &bufferSize ).asExpression();
1546 break;
1547
1548 case QMetaType::Type::QVariantList:
1549 case QMetaType::Type::QStringList:
1550 bufferSize = 1;
1551 bufferSizeDataDefined = parseValueList( jsonHaloWidth.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor() * BUFFER_SIZE_SCALE, 255, nullptr, &bufferSize ).asExpression();
1552 break;
1553
1554 default:
1555 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-halo-width type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonHaloWidth.userType() ) ) ) );
1556 break;
1557 }
1558
1559 // from the specs halo should not be larger than 1/4 of the text-size
1560 // https://docs.mapbox.com/style-spec/reference/layers/#paint-symbol-text-halo-width
1561 if ( bufferSize > 0 )
1562 {
1563 if ( textSize > 0 && bufferSizeDataDefined.isEmpty() )
1564 {
1565 bufferSize = std::min( bufferSize, textSize * BUFFER_SIZE_SCALE / 4 );
1566 }
1567 else if ( textSize > 0 && !bufferSizeDataDefined.isEmpty() )
1568 {
1569 bufferSizeDataDefined = u"min(%1/4, %2)"_s.arg( textSize * BUFFER_SIZE_SCALE ).arg( bufferSizeDataDefined );
1570 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::BufferSize, QgsProperty::fromExpression( bufferSizeDataDefined ) );
1571 }
1572 else if ( !bufferSizeDataDefined.isEmpty() )
1573 {
1574 bufferSizeDataDefined = u"min(%1*%2/4, %3)"_s.arg( textSizeProperty.asExpression() ).arg( BUFFER_SIZE_SCALE ).arg( bufferSizeDataDefined );
1575 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::BufferSize, QgsProperty::fromExpression( bufferSizeDataDefined ) );
1576 }
1577 else if ( bufferSizeDataDefined.isEmpty() )
1578 {
1579 bufferSizeDataDefined = u"min(%1*%2/4, %3)"_s.arg( textSizeProperty.asExpression() ).arg( BUFFER_SIZE_SCALE ).arg( bufferSize );
1580 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::BufferSize, QgsProperty::fromExpression( bufferSizeDataDefined ) );
1581 }
1582 }
1583 }
1584
1585 double haloBlurSize = 0;
1586 if ( jsonPaint.contains( u"text-halo-blur"_s ) )
1587 {
1588 const QVariant jsonTextHaloBlur = jsonPaint.value( u"text-halo-blur"_s );
1589 switch ( jsonTextHaloBlur.userType() )
1590 {
1591 case QMetaType::Type::Int:
1592 case QMetaType::Type::LongLong:
1593 case QMetaType::Type::Double:
1594 {
1595 haloBlurSize = jsonTextHaloBlur.toDouble() * context.pixelSizeConversionFactor();
1596 break;
1597 }
1598
1599 default:
1600 context.pushWarning(
1601 QObject::tr( "%1: Skipping unsupported text-halo-blur type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextHaloBlur.userType() ) ) )
1602 );
1603 break;
1604 }
1605 }
1606
1607 QgsTextFormat format;
1608 format.setSizeUnit( context.targetUnit() );
1609 if ( textColor.isValid() )
1610 format.setColor( textColor );
1611 if ( textSize >= 0 )
1612 format.setSize( textSize );
1613 if ( foundFont )
1614 {
1615 format.setFont( textFont );
1616 if ( !fontStyleName.isEmpty() )
1617 format.setNamedStyle( fontStyleName );
1618 }
1619 if ( textLetterSpacing > 0 )
1620 {
1621 QFont f = format.font();
1622 f.setLetterSpacing( QFont::AbsoluteSpacing, textLetterSpacing );
1623 format.setFont( f );
1624 }
1625
1626 if ( bufferSize > 0 )
1627 {
1628 // Color and opacity are separate components in QGIS
1629 const double opacity = bufferColor.alphaF();
1630 bufferColor.setAlphaF( 1.0 );
1631
1632 format.buffer().setEnabled( true );
1633 format.buffer().setSize( bufferSize );
1634 format.buffer().setSizeUnit( context.targetUnit() );
1635 format.buffer().setColor( bufferColor );
1636 format.buffer().setOpacity( opacity );
1637
1638 if ( haloBlurSize > 0 )
1639 {
1640 QgsEffectStack *stack = new QgsEffectStack();
1641 QgsBlurEffect *blur = new QgsBlurEffect();
1642 blur->setEnabled( true );
1643 blur->setBlurUnit( context.targetUnit() );
1644 blur->setBlurLevel( haloBlurSize );
1646 stack->appendEffect( blur );
1647 stack->setEnabled( true );
1648 format.buffer().setPaintEffect( stack );
1649 }
1650 }
1651
1652 QgsPalLayerSettings labelSettings;
1653 if ( allowOverlap )
1654 {
1655 QgsLabelPlacementSettings placementSettings = labelSettings.placementSettings();
1657 placementSettings.setAllowDegradedPlacement( true );
1658 labelSettings.setPlacementSettings( placementSettings );
1659 }
1660
1661 if ( textMaxWidth > 0 )
1662 {
1663 labelSettings.autoWrapLength = textMaxWidth;
1664 }
1665
1666 // convert field name
1667 if ( jsonLayout.contains( u"text-field"_s ) )
1668 {
1669 const QVariant jsonTextField = jsonLayout.value( u"text-field"_s );
1670 switch ( jsonTextField.userType() )
1671 {
1672 case QMetaType::Type::QString:
1673 {
1674 labelSettings.fieldName = processLabelField( jsonTextField.toString(), labelSettings.isExpression );
1675 break;
1676 }
1677
1678 case QMetaType::Type::QVariantList:
1679 case QMetaType::Type::QStringList:
1680 {
1681 const QVariantList textFieldList = jsonTextField.toList();
1682 /*
1683 * e.g.
1684 * "text-field": ["format",
1685 * "foo", { "font-scale": 1.2 },
1686 * "bar", { "font-scale": 0.8 }
1687 * ]
1688 */
1689 if ( textFieldList.size() > 2 && textFieldList.at( 0 ).toString() == "format"_L1 )
1690 {
1691 QStringList parts;
1692 for ( int i = 1; i < textFieldList.size(); ++i )
1693 {
1694 bool isExpression = false;
1695 const QString part = processLabelField( textFieldList.at( i ).toString(), isExpression );
1696 if ( !isExpression )
1697 parts << QgsExpression::quotedColumnRef( part );
1698 else
1699 parts << part;
1700 // TODO -- we could also translate font color, underline, overline, strikethrough to HTML tags!
1701 i += 1;
1702 }
1703 labelSettings.fieldName = u"concat(%1)"_s.arg( parts.join( ',' ) );
1704 labelSettings.isExpression = true;
1705 }
1706 else
1707 {
1708 /*
1709 * e.g.
1710 * "text-field": ["to-string", ["get", "name"]]
1711 */
1712 labelSettings.fieldName = parseExpression( textFieldList, context );
1713 labelSettings.isExpression = true;
1714 }
1715 break;
1716 }
1717
1718 case QMetaType::Type::QVariantMap:
1719 {
1720 const QVariantList stops = jsonTextField.toMap().value( u"stops"_s ).toList();
1721 if ( !stops.empty() )
1722 {
1723 labelSettings.fieldName = parseLabelStops( stops, context );
1724 labelSettings.isExpression = true;
1725 }
1726 else
1727 {
1728 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-field dictionary" ).arg( context.layerId() ) );
1729 }
1730 break;
1731 }
1732
1733 default:
1734 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-field type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextField.userType() ) ) ) );
1735 break;
1736 }
1737 }
1738
1739 if ( jsonLayout.contains( u"text-rotate"_s ) )
1740 {
1741 const QVariant jsonTextRotate = jsonLayout.value( u"text-rotate"_s );
1742 switch ( jsonTextRotate.userType() )
1743 {
1744 case QMetaType::Type::Double:
1745 case QMetaType::Type::Int:
1746 {
1747 labelSettings.angleOffset = jsonTextRotate.toDouble();
1748 break;
1749 }
1750
1751 case QMetaType::Type::QVariantList:
1752 case QMetaType::Type::QStringList:
1753 {
1754 const QgsProperty property = parseValueList( jsonTextRotate.toList(), PropertyType::Numeric, context );
1755 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::LabelRotation, property );
1756 break;
1757 }
1758
1759 case QMetaType::Type::QVariantMap:
1760 {
1761 QVariantMap rotateMap = jsonTextRotate.toMap();
1762 if ( rotateMap.contains( u"property"_s ) && rotateMap[u"type"_s].toString() == "identity"_L1 )
1763 {
1764 const QgsProperty property = QgsProperty::fromExpression( rotateMap[u"property"_s].toString() );
1765 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::LabelRotation, property );
1766 }
1767 else
1768 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-rotate map content (%2)" ).arg( context.layerId(), QString( QJsonDocument::fromVariant( rotateMap ).toJson() ) ) );
1769 break;
1770 }
1771
1772 default:
1773 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-rotate type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextRotate.userType() ) ) ) );
1774 break;
1775 }
1776 }
1777
1778 if ( jsonLayout.contains( u"text-transform"_s ) )
1779 {
1780 const QString textTransform = jsonLayout.value( u"text-transform"_s ).toString();
1781 if ( textTransform == "uppercase"_L1 )
1782 {
1783 labelSettings.fieldName = u"upper(%1)"_s.arg( labelSettings.isExpression ? labelSettings.fieldName : QgsExpression::quotedColumnRef( labelSettings.fieldName ) );
1784 }
1785 else if ( textTransform == "lowercase"_L1 )
1786 {
1787 labelSettings.fieldName = u"lower(%1)"_s.arg( labelSettings.isExpression ? labelSettings.fieldName : QgsExpression::quotedColumnRef( labelSettings.fieldName ) );
1788 }
1789 labelSettings.isExpression = true;
1790 }
1791
1794 if ( jsonLayout.contains( u"symbol-placement"_s ) )
1795 {
1796 const QString symbolPlacement = jsonLayout.value( u"symbol-placement"_s ).toString();
1797 if ( symbolPlacement == "line"_L1 )
1798 {
1801 geometryType = Qgis::GeometryType::Line;
1802
1803 if ( jsonLayout.contains( u"text-rotation-alignment"_s ) )
1804 {
1805 const QString textRotationAlignment = jsonLayout.value( u"text-rotation-alignment"_s ).toString();
1806 if ( textRotationAlignment == "viewport"_L1 )
1807 {
1809 }
1810 }
1811
1812 if ( labelSettings.placement == Qgis::LabelPlacement::Curved )
1813 {
1814 QPointF textOffset;
1815 QgsProperty textOffsetProperty;
1816 if ( jsonLayout.contains( u"text-offset"_s ) )
1817 {
1818 const QVariant jsonTextOffset = jsonLayout.value( u"text-offset"_s );
1819
1820 // units are ems!
1821 switch ( jsonTextOffset.userType() )
1822 {
1823 case QMetaType::Type::QVariantMap:
1824 textOffsetProperty = parseInterpolatePointByZoom( jsonTextOffset.toMap(), context, !textSizeProperty ? textSize : 1.0, &textOffset );
1825 if ( !textSizeProperty )
1826 {
1827 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::LabelDistance, u"abs(array_get(%1,1))-%2"_s.arg( textOffsetProperty.asExpression() ).arg( textSize ) );
1828 }
1829 else
1830 {
1831 ddLabelProperties.setProperty(
1833 u"with_variable('text_size',%2,abs(array_get(%1,1))*@text_size-@text_size)"_s.arg( textOffsetProperty.asExpression(), textSizeProperty.asExpression() )
1834 );
1835 }
1836 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::LinePlacementOptions, u"if(array_get(%1,1)>0,'BL','AL')"_s.arg( textOffsetProperty.asExpression() ) );
1837 break;
1838
1839 case QMetaType::Type::QVariantList:
1840 case QMetaType::Type::QStringList:
1841 textOffset = QPointF( jsonTextOffset.toList().value( 0 ).toDouble() * textSize, jsonTextOffset.toList().value( 1 ).toDouble() * textSize );
1842 break;
1843
1844 default:
1845 context.pushWarning(
1846 QObject::tr( "%1: Skipping unsupported text-offset type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextOffset.userType() ) ) )
1847 );
1848 break;
1849 }
1850
1851 if ( !textOffset.isNull() )
1852 {
1853 labelSettings.distUnits = context.targetUnit();
1854 labelSettings.dist = std::abs( textOffset.y() ) - textSize;
1856 if ( textSizeProperty && !textOffsetProperty )
1857 {
1858 ddLabelProperties.setProperty(
1860 u"with_variable('text_size',%2,%1*@text_size-@text_size)"_s.arg( std::abs( textOffset.y() / textSize ) ).arg( textSizeProperty.asExpression() )
1861 );
1862 }
1863 }
1864 }
1865
1866 if ( textOffset.isNull() )
1867 {
1869 }
1870 }
1871 }
1872 }
1873
1874 if ( jsonLayout.contains( u"text-justify"_s ) )
1875 {
1876 const QVariant jsonTextJustify = jsonLayout.value( u"text-justify"_s );
1877
1878 // default is center
1879 QString textAlign = u"center"_s;
1880
1881 const QVariantMap conversionMap { { u"left"_s, u"left"_s }, { u"center"_s, u"center"_s }, { u"right"_s, u"right"_s }, { u"auto"_s, u"follow"_s } };
1882
1883 switch ( jsonTextJustify.userType() )
1884 {
1885 case QMetaType::Type::QString:
1886 textAlign = jsonTextJustify.toString();
1887 break;
1888
1889 case QMetaType::Type::QVariantList:
1890 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::OffsetQuad, QgsProperty::fromExpression( parseStringStops( jsonTextJustify.toList(), context, conversionMap, &textAlign ) ) );
1891 break;
1892
1893 case QMetaType::Type::QVariantMap:
1894 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::OffsetQuad, parseInterpolateStringByZoom( jsonTextJustify.toMap(), context, conversionMap, &textAlign ) );
1895 break;
1896
1897 default:
1898 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-justify type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextJustify.userType() ) ) ) );
1899 break;
1900 }
1901
1902 if ( textAlign == "left"_L1 )
1904 else if ( textAlign == "right"_L1 )
1906 else if ( textAlign == "center"_L1 )
1908 else if ( textAlign == "follow"_L1 )
1910 }
1911 else
1912 {
1914 }
1915
1916 if ( labelSettings.placement == Qgis::LabelPlacement::OverPoint )
1917 {
1918 if ( jsonLayout.contains( u"text-anchor"_s ) )
1919 {
1920 const QVariant jsonTextAnchor = jsonLayout.value( u"text-anchor"_s );
1921 QString textAnchor;
1922
1923 const QVariantMap conversionMap {
1924 { u"center"_s, 4 },
1925 { u"left"_s, 5 },
1926 { u"right"_s, 3 },
1927 { u"top"_s, 7 },
1928 { u"bottom"_s, 1 },
1929 { u"top-left"_s, 8 },
1930 { u"top-right"_s, 6 },
1931 { u"bottom-left"_s, 2 },
1932 { u"bottom-right"_s, 0 },
1933 };
1934
1935 switch ( jsonTextAnchor.userType() )
1936 {
1937 case QMetaType::Type::QString:
1938 textAnchor = jsonTextAnchor.toString();
1939 break;
1940
1941 case QMetaType::Type::QVariantList:
1942 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::OffsetQuad, QgsProperty::fromExpression( parseStringStops( jsonTextAnchor.toList(), context, conversionMap, &textAnchor ) ) );
1943 break;
1944
1945 case QMetaType::Type::QVariantMap:
1946 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::OffsetQuad, parseInterpolateStringByZoom( jsonTextAnchor.toMap(), context, conversionMap, &textAnchor ) );
1947 break;
1948
1949 default:
1950 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-anchor type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextAnchor.userType() ) ) ) );
1951 break;
1952 }
1953
1954 if ( textAnchor == "center"_L1 )
1956 else if ( textAnchor == "left"_L1 )
1958 else if ( textAnchor == "right"_L1 )
1960 else if ( textAnchor == "top"_L1 )
1962 else if ( textAnchor == "bottom"_L1 )
1964 else if ( textAnchor == "top-left"_L1 )
1966 else if ( textAnchor == "top-right"_L1 )
1968 else if ( textAnchor == "bottom-left"_L1 )
1970 else if ( textAnchor == "bottom-right"_L1 )
1972 }
1973
1974 QPointF textOffset;
1975 if ( jsonLayout.contains( u"text-offset"_s ) )
1976 {
1977 const QVariant jsonTextOffset = jsonLayout.value( u"text-offset"_s );
1978
1979 // units are ems!
1980 switch ( jsonTextOffset.userType() )
1981 {
1982 case QMetaType::Type::QVariantMap:
1983 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::OffsetXY, parseInterpolatePointByZoom( jsonTextOffset.toMap(), context, textSize, &textOffset ) );
1984 break;
1985
1986 case QMetaType::Type::QVariantList:
1987 case QMetaType::Type::QStringList:
1988 textOffset = QPointF( jsonTextOffset.toList().value( 0 ).toDouble() * textSize, jsonTextOffset.toList().value( 1 ).toDouble() * textSize );
1989 break;
1990
1991 default:
1992 context.pushWarning( QObject::tr( "%1: Skipping unsupported text-offset type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonTextOffset.userType() ) ) ) );
1993 break;
1994 }
1995
1996 if ( !textOffset.isNull() )
1997 {
1998 labelSettings.offsetUnits = context.targetUnit();
1999 labelSettings.xOffset = textOffset.x();
2000 labelSettings.yOffset = textOffset.y();
2001 }
2002 }
2003 }
2004
2005 if ( jsonLayout.contains( u"icon-image"_s ) && ( labelSettings.placement == Qgis::LabelPlacement::Horizontal || labelSettings.placement == Qgis::LabelPlacement::Curved ) )
2006 {
2007 QSize spriteSize;
2008 QString spriteProperty, spriteSizeProperty;
2009 const QString sprite = retrieveSpriteAsBase64WithProperties( jsonLayout.value( u"icon-image"_s ), context, spriteSize, spriteProperty, spriteSizeProperty );
2010 if ( !sprite.isEmpty() )
2011 {
2012 double size = 1.0;
2013 if ( jsonLayout.contains( u"icon-size"_s ) )
2014 {
2015 QgsProperty property;
2016 const QVariant jsonIconSize = jsonLayout.value( u"icon-size"_s );
2017 switch ( jsonIconSize.userType() )
2018 {
2019 case QMetaType::Type::Int:
2020 case QMetaType::Type::LongLong:
2021 case QMetaType::Type::Double:
2022 {
2023 size = jsonIconSize.toDouble();
2024 if ( !spriteSizeProperty.isEmpty() )
2025 {
2026 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::ShapeSizeX, QgsProperty::fromExpression( u"with_variable('marker_size',%1,%2*@marker_size)"_s.arg( spriteSizeProperty ).arg( size ) ) );
2027 }
2028 break;
2029 }
2030
2031 case QMetaType::Type::QVariantMap:
2032 property = parseInterpolateByZoom( jsonIconSize.toMap(), context, 1, &size );
2033 break;
2034
2035 case QMetaType::Type::QVariantList:
2036 case QMetaType::Type::QStringList:
2037 property = parseValueList( jsonIconSize.toList(), PropertyType::Numeric, context );
2038 break;
2039 default:
2040 context.pushWarning(
2041 QObject::tr( "%1: Skipping non-implemented icon-size type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonIconSize.userType() ) ) )
2042 );
2043 break;
2044 }
2045
2046 if ( !property.expressionString().isEmpty() )
2047 {
2048 if ( !spriteSizeProperty.isEmpty() )
2049 {
2050 ddLabelProperties
2051 .setProperty( QgsPalLayerSettings::Property::ShapeSizeX, QgsProperty::fromExpression( u"with_variable('marker_size',%1,(%2)*@marker_size)"_s.arg( spriteSizeProperty ).arg( property.expressionString() ) ) );
2052 }
2053 else
2054 {
2055 ddLabelProperties.setProperty( QgsPalLayerSettings::Property::ShapeSizeX, QgsProperty::fromExpression( u"(%2)*%1"_s.arg( spriteSize.width() ).arg( property.expressionString() ) ) );
2056 }
2057 }
2058 }
2059
2061 markerLayer->setPath( sprite );
2062 markerLayer->setSize( spriteSize.width() );
2063 markerLayer->setSizeUnit( context.targetUnit() );
2064
2065 if ( !spriteProperty.isEmpty() )
2066 {
2067 QgsPropertyCollection markerDdProperties;
2068 markerDdProperties.setProperty( QgsSymbolLayer::Property::Name, QgsProperty::fromExpression( spriteProperty ) );
2069 markerLayer->setDataDefinedProperties( markerDdProperties );
2070 }
2071
2072 QgsTextBackgroundSettings backgroundSettings;
2073 backgroundSettings.setEnabled( true );
2075 backgroundSettings.setSize( spriteSize * size );
2076 backgroundSettings.setSizeUnit( context.targetUnit() );
2078 backgroundSettings.setMarkerSymbol( new QgsMarkerSymbol( QgsSymbolLayerList() << markerLayer ) );
2079 format.setBackground( backgroundSettings );
2080 }
2081 }
2082
2083 if ( jsonLayout.contains( u"symbol-spacing"_s ) )
2084 {
2085 double spacing;
2086 const QVariant jsonSpacing = jsonLayout.value( u"symbol-spacing"_s );
2087
2088 // main checkbox in labeling GUI
2089 QgsLabelThinningSettings thinningSettings = labelSettings.thinningSettings();
2090 thinningSettings.setAllowDuplicateRemoval( true );
2091 thinningSettings.setMinimumDistanceToDuplicateUnit( context.targetUnit() );
2092 labelSettings.setThinningSettings( thinningSettings );
2093
2094 QgsProperty spacingProp;
2095
2096 switch ( jsonSpacing.userType() )
2097 {
2098 case QMetaType::Type::Int:
2099 case QMetaType::Type::LongLong:
2100 case QMetaType::Type::Double:
2101 {
2102 spacing = jsonSpacing.toDouble() * context.pixelSizeConversionFactor();
2103 spacingProp = QgsProperty::fromValue( spacing );
2104 break;
2105 }
2106
2107 case QMetaType::Type::QVariantMap:
2108 {
2109 spacingProp = parseInterpolateByZoom( jsonSpacing.toMap(), context, context.pixelSizeConversionFactor(), &spacing );
2110 break;
2111 }
2112
2113 case QMetaType::Type::QVariantList:
2114 case QMetaType::Type::QStringList:
2115 {
2116 spacingProp = parseValueList( jsonSpacing.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &spacing );
2117 break;
2118 }
2119
2120 default:
2121 context.pushWarning( QObject::tr( "%1: Skipping unsupported symbol-spacing type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonSpacing.userType() ) ) ) );
2122 break;
2123 }
2124
2125 spacingProp.setActive( true );
2127 }
2128
2129 if ( textSize >= 0 )
2130 {
2131 // TODO -- this probably needs revisiting -- it was copied from the MapTiler code, but may be wrong...
2132 labelSettings.priority = std::min( textSize / ( context.pixelSizeConversionFactor() * 3 ), 10.0 );
2133 }
2134
2135 labelSettings.setFormat( format );
2136
2137 // use a low obstacle weight for layers by default -- we'd rather have more labels for these layers, even if placement isn't ideal
2138 labelSettings.obstacleSettings().setFactor( 0.1 );
2139
2140 labelSettings.setDataDefinedProperties( ddLabelProperties );
2141
2142 labelingStyle.setGeometryType( geometryType );
2143 labelingStyle.setLabelSettings( labelSettings );
2144
2145 hasLabeling = true;
2146
2147 hasRenderer = parseSymbolLayerAsRenderer( jsonLayer, renderer, context );
2148}
2149
2151{
2152 if ( !jsonLayer.contains( u"layout"_s ) )
2153 {
2154 context.pushWarning( QObject::tr( "%1: Style layer has no layout property, skipping" ).arg( context.layerId() ) );
2155 return false;
2156 }
2157 const QVariantMap jsonLayout = jsonLayer.value( u"layout"_s ).toMap();
2158
2159 if ( jsonLayout.value( u"symbol-placement"_s ).toString() == "line"_L1 && !jsonLayout.contains( u"text-field"_s ) )
2160 {
2161 QgsPropertyCollection ddProperties;
2162
2163 double spacing = -1.0;
2164 if ( jsonLayout.contains( u"symbol-spacing"_s ) )
2165 {
2166 const QVariant jsonSpacing = jsonLayout.value( u"symbol-spacing"_s );
2167 switch ( jsonSpacing.userType() )
2168 {
2169 case QMetaType::Type::Int:
2170 case QMetaType::Type::LongLong:
2171 case QMetaType::Type::Double:
2172 spacing = jsonSpacing.toDouble() * context.pixelSizeConversionFactor();
2173 break;
2174
2175 case QMetaType::Type::QVariantMap:
2176 ddProperties.setProperty( QgsSymbolLayer::Property::Interval, parseInterpolateByZoom( jsonSpacing.toMap(), context, context.pixelSizeConversionFactor(), &spacing ) );
2177 break;
2178
2179 case QMetaType::Type::QVariantList:
2180 case QMetaType::Type::QStringList:
2181 ddProperties.setProperty( QgsSymbolLayer::Property::Interval, parseValueList( jsonSpacing.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &spacing ) );
2182 break;
2183
2184 default:
2185 context.pushWarning( QObject::tr( "%1: Skipping unsupported symbol-spacing type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonSpacing.userType() ) ) ) );
2186 break;
2187 }
2188 }
2189 else
2190 {
2191 // defaults to 250
2192 spacing = 250 * context.pixelSizeConversionFactor();
2193 }
2194
2195 bool rotateMarkers = true;
2196 if ( jsonLayout.contains( u"icon-rotation-alignment"_s ) )
2197 {
2198 const QString alignment = jsonLayout.value( u"icon-rotation-alignment"_s ).toString();
2199 if ( alignment == "map"_L1 || alignment == "auto"_L1 )
2200 {
2201 rotateMarkers = true;
2202 }
2203 else if ( alignment == "viewport"_L1 )
2204 {
2205 rotateMarkers = false;
2206 }
2207 }
2208
2209 QgsPropertyCollection markerDdProperties;
2210 double rotation = 0.0;
2211 if ( jsonLayout.contains( u"icon-rotate"_s ) )
2212 {
2213 const QVariant jsonIconRotate = jsonLayout.value( u"icon-rotate"_s );
2214 switch ( jsonIconRotate.userType() )
2215 {
2216 case QMetaType::Type::Int:
2217 case QMetaType::Type::LongLong:
2218 case QMetaType::Type::Double:
2219 rotation = jsonIconRotate.toDouble();
2220 break;
2221
2222 case QMetaType::Type::QVariantMap:
2223 markerDdProperties.setProperty( QgsSymbolLayer::Property::Angle, parseInterpolateByZoom( jsonIconRotate.toMap(), context, context.pixelSizeConversionFactor(), &rotation ) );
2224 break;
2225
2226 case QMetaType::Type::QVariantList:
2227 case QMetaType::Type::QStringList:
2228 markerDdProperties.setProperty( QgsSymbolLayer::Property::Angle, parseValueList( jsonIconRotate.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &rotation ) );
2229 break;
2230
2231 default:
2232 context.pushWarning( QObject::tr( "%1: Skipping unsupported icon-rotate type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonIconRotate.userType() ) ) ) );
2233 break;
2234 }
2235 }
2236
2237 QgsMarkerLineSymbolLayer *lineSymbol = new QgsMarkerLineSymbolLayer( rotateMarkers, spacing > 0 ? spacing : 1 );
2238 lineSymbol->setOutputUnit( context.targetUnit() );
2239 lineSymbol->setDataDefinedProperties( ddProperties );
2240 if ( spacing < 1 )
2241 {
2242 // if spacing isn't specified, it's a central point marker only
2244 }
2245
2247 QSize spriteSize;
2248 QString spriteProperty, spriteSizeProperty;
2249 const QString sprite = retrieveSpriteAsBase64WithProperties( jsonLayout.value( u"icon-image"_s ), context, spriteSize, spriteProperty, spriteSizeProperty );
2250 if ( !sprite.isNull() )
2251 {
2252 markerLayer->setPath( sprite );
2253 markerLayer->setSize( spriteSize.width() );
2254 markerLayer->setSizeUnit( context.targetUnit() );
2255
2256 if ( !spriteProperty.isEmpty() )
2257 {
2258 markerDdProperties.setProperty( QgsSymbolLayer::Property::Name, QgsProperty::fromExpression( spriteProperty ) );
2259 markerDdProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( spriteSizeProperty ) );
2260 }
2261 }
2262
2263 if ( jsonLayout.contains( u"icon-size"_s ) )
2264 {
2265 const QVariant jsonIconSize = jsonLayout.value( u"icon-size"_s );
2266 double size = 1.0;
2267 QgsProperty property;
2268 switch ( jsonIconSize.userType() )
2269 {
2270 case QMetaType::Type::Int:
2271 case QMetaType::Type::LongLong:
2272 case QMetaType::Type::Double:
2273 {
2274 size = jsonIconSize.toDouble();
2275 if ( !spriteSizeProperty.isEmpty() )
2276 {
2277 markerDdProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( u"with_variable('marker_size',%1,%2*@marker_size)"_s.arg( spriteSizeProperty ).arg( size ) ) );
2278 }
2279 break;
2280 }
2281
2282 case QMetaType::Type::QVariantMap:
2283 property = parseInterpolateByZoom( jsonIconSize.toMap(), context, 1, &size );
2284 break;
2285
2286 case QMetaType::Type::QVariantList:
2287 case QMetaType::Type::QStringList:
2288 property = parseValueList( jsonIconSize.toList(), PropertyType::Numeric, context );
2289 break;
2290 default:
2291 context.pushWarning( QObject::tr( "%1: Skipping non-implemented icon-size type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonIconSize.userType() ) ) ) );
2292 break;
2293 }
2294 markerLayer->setSize( size * spriteSize.width() );
2295 if ( !property.expressionString().isEmpty() )
2296 {
2297 if ( !spriteSizeProperty.isEmpty() )
2298 {
2299 markerDdProperties
2300 .setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( u"with_variable('marker_size',%1,(%2)*@marker_size)"_s.arg( spriteSizeProperty ).arg( property.expressionString() ) ) );
2301 }
2302 else
2303 {
2304 markerDdProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( u"(%2)*%1"_s.arg( spriteSize.width() ).arg( property.expressionString() ) ) );
2305 }
2306 }
2307 }
2308
2309 markerLayer->setDataDefinedProperties( markerDdProperties );
2310 markerLayer->setAngle( rotation );
2311 lineSymbol->setSubSymbol( new QgsMarkerSymbol( QgsSymbolLayerList() << markerLayer ) );
2312
2313 std::unique_ptr< QgsSymbol > symbol = std::make_unique< QgsLineSymbol >( QgsSymbolLayerList() << lineSymbol );
2314
2315 // set render units
2316 symbol->setOutputUnit( context.targetUnit() );
2317 lineSymbol->setOutputUnit( context.targetUnit() );
2318
2320 rendererStyle.setSymbol( symbol.release() );
2321 return true;
2322 }
2323 else if ( jsonLayout.contains( u"icon-image"_s ) )
2324 {
2325 const QVariantMap jsonPaint = jsonLayer.value( u"paint"_s ).toMap();
2326
2327 QSize spriteSize;
2328 QString spriteProperty, spriteSizeProperty;
2329 const QString sprite = retrieveSpriteAsBase64WithProperties( jsonLayout.value( u"icon-image"_s ), context, spriteSize, spriteProperty, spriteSizeProperty );
2330 if ( !sprite.isEmpty() || !spriteProperty.isEmpty() )
2331 {
2333 rasterMarker->setPath( sprite );
2334 rasterMarker->setSize( spriteSize.width() );
2335 rasterMarker->setSizeUnit( context.targetUnit() );
2336
2337 QgsPropertyCollection markerDdProperties;
2338 if ( !spriteProperty.isEmpty() )
2339 {
2340 markerDdProperties.setProperty( QgsSymbolLayer::Property::Name, QgsProperty::fromExpression( spriteProperty ) );
2341 markerDdProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( spriteSizeProperty ) );
2342 }
2343
2344 if ( jsonLayout.contains( u"icon-size"_s ) )
2345 {
2346 const QVariant jsonIconSize = jsonLayout.value( u"icon-size"_s );
2347 double size = 1.0;
2348 QgsProperty property;
2349 switch ( jsonIconSize.userType() )
2350 {
2351 case QMetaType::Type::Int:
2352 case QMetaType::Type::LongLong:
2353 case QMetaType::Type::Double:
2354 {
2355 size = jsonIconSize.toDouble();
2356 if ( !spriteSizeProperty.isEmpty() )
2357 {
2358 markerDdProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( u"with_variable('marker_size',%1,%2*@marker_size)"_s.arg( spriteSizeProperty ).arg( size ) ) );
2359 }
2360 break;
2361 }
2362
2363 case QMetaType::Type::QVariantMap:
2364 property = parseInterpolateByZoom( jsonIconSize.toMap(), context, 1, &size );
2365 break;
2366
2367 case QMetaType::Type::QVariantList:
2368 case QMetaType::Type::QStringList:
2369 property = parseValueList( jsonIconSize.toList(), PropertyType::Numeric, context );
2370 break;
2371 default:
2372 context.pushWarning(
2373 QObject::tr( "%1: Skipping non-implemented icon-size type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonIconSize.userType() ) ) )
2374 );
2375 break;
2376 }
2377 rasterMarker->setSize( size * spriteSize.width() );
2378 if ( !property.expressionString().isEmpty() )
2379 {
2380 if ( !spriteSizeProperty.isEmpty() )
2381 {
2382 markerDdProperties
2383 .setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( u"with_variable('marker_size',%1,(%2)*@marker_size)"_s.arg( spriteSizeProperty ).arg( property.expressionString() ) ) );
2384 }
2385 else
2386 {
2387 markerDdProperties.setProperty( QgsSymbolLayer::Property::Width, QgsProperty::fromExpression( u"(%2)*%1"_s.arg( spriteSize.width() ).arg( property.expressionString() ) ) );
2388 }
2389 }
2390 }
2391
2392 double rotation = 0.0;
2393 if ( jsonLayout.contains( u"icon-rotate"_s ) )
2394 {
2395 const QVariant jsonIconRotate = jsonLayout.value( u"icon-rotate"_s );
2396 switch ( jsonIconRotate.userType() )
2397 {
2398 case QMetaType::Type::Int:
2399 case QMetaType::Type::LongLong:
2400 case QMetaType::Type::Double:
2401 rotation = jsonIconRotate.toDouble();
2402 break;
2403
2404 case QMetaType::Type::QVariantMap:
2405 markerDdProperties.setProperty( QgsSymbolLayer::Property::Angle, parseInterpolateByZoom( jsonIconRotate.toMap(), context, context.pixelSizeConversionFactor(), &rotation ) );
2406 break;
2407
2408 case QMetaType::Type::QVariantList:
2409 case QMetaType::Type::QStringList:
2410 markerDdProperties.setProperty( QgsSymbolLayer::Property::Angle, parseValueList( jsonIconRotate.toList(), PropertyType::Numeric, context, context.pixelSizeConversionFactor(), 255, nullptr, &rotation ) );
2411 break;
2412
2413 default:
2414 context.pushWarning(
2415 QObject::tr( "%1: Skipping unsupported icon-rotate type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonIconRotate.userType() ) ) )
2416 );
2417 break;
2418 }
2419 }
2420
2421 double iconOpacity = -1.0;
2422 if ( jsonPaint.contains( u"icon-opacity"_s ) )
2423 {
2424 const QVariant jsonIconOpacity = jsonPaint.value( u"icon-opacity"_s );
2425 switch ( jsonIconOpacity.userType() )
2426 {
2427 case QMetaType::Type::Int:
2428 case QMetaType::Type::LongLong:
2429 case QMetaType::Type::Double:
2430 iconOpacity = jsonIconOpacity.toDouble();
2431 break;
2432
2433 case QMetaType::Type::QVariantMap:
2434 markerDdProperties.setProperty( QgsSymbolLayer::Property::Opacity, parseInterpolateByZoom( jsonIconOpacity.toMap(), context, 100, &iconOpacity ) );
2435 break;
2436
2437 case QMetaType::Type::QVariantList:
2438 case QMetaType::Type::QStringList:
2439 markerDdProperties.setProperty( QgsSymbolLayer::Property::Opacity, parseValueList( jsonIconOpacity.toList(), PropertyType::Numeric, context, 100, 255, nullptr, &iconOpacity ) );
2440 break;
2441
2442 default:
2443 context.pushWarning(
2444 QObject::tr( "%1: Skipping unsupported icon-opacity type (%2)" ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( jsonIconOpacity.userType() ) ) )
2445 );
2446 break;
2447 }
2448 }
2449
2450 rasterMarker->setDataDefinedProperties( markerDdProperties );
2451 rasterMarker->setAngle( rotation );
2452 if ( iconOpacity >= 0 )
2453 rasterMarker->setOpacity( iconOpacity );
2454
2455 QgsMarkerSymbol *markerSymbol = new QgsMarkerSymbol( QgsSymbolLayerList() << rasterMarker );
2456 rendererStyle.setSymbol( markerSymbol );
2458 return true;
2459 }
2460 }
2461
2462 return false;
2463}
2464
2466{
2467 const double base = json.value( u"base"_s, u"1"_s ).toDouble();
2468 const QVariantList stops = json.value( u"stops"_s ).toList();
2469 if ( stops.empty() )
2470 return QgsProperty();
2471
2472 QString caseString = u"CASE "_s;
2473 const QString colorComponent( "color_part(%1,'%2')" );
2474
2475 for ( int i = 0; i < stops.length() - 1; ++i )
2476 {
2477 // step bottom zoom
2478 const QString bz = stops.at( i ).toList().value( 0 ).toString();
2479 // step top zoom
2480 const QString tz = stops.at( i + 1 ).toList().value( 0 ).toString();
2481
2482 const QVariant bcVariant = stops.at( i ).toList().value( 1 );
2483 const QVariant tcVariant = stops.at( i + 1 ).toList().value( 1 );
2484
2485 const QColor bottomColor = parseColor( bcVariant.toString(), context );
2486 const QColor topColor = parseColor( tcVariant.toString(), context );
2487
2488 if ( i == 0 && bottomColor.isValid() )
2489 {
2490 int bcHue;
2491 int bcSat;
2492 int bcLight;
2493 int bcAlpha;
2494 colorAsHslaComponents( bottomColor, bcHue, bcSat, bcLight, bcAlpha );
2495 caseString += u"WHEN @vector_tile_zoom < %1 THEN color_hsla(%2, %3, %4, %5) "_s.arg( bz ).arg( bcHue ).arg( bcSat ).arg( bcLight ).arg( bcAlpha );
2496 }
2497
2498 if ( bottomColor.isValid() && topColor.isValid() )
2499 {
2500 int bcHue;
2501 int bcSat;
2502 int bcLight;
2503 int bcAlpha;
2504 colorAsHslaComponents( bottomColor, bcHue, bcSat, bcLight, bcAlpha );
2505 int tcHue;
2506 int tcSat;
2507 int tcLight;
2508 int tcAlpha;
2509 colorAsHslaComponents( topColor, tcHue, tcSat, tcLight, tcAlpha );
2510 caseString += QStringLiteral(
2511 "WHEN @vector_tile_zoom >= %1 AND @vector_tile_zoom < %2 THEN color_hsla("
2512 "%3, %4, %5, %6) "
2513 )
2514 .arg(
2515 bz,
2516 tz,
2518 bz.toDouble(),
2519 tz.toDouble(),
2520 bcHue,
2521 tcHue,
2522 base,
2523 1,
2524 json.value( u"x1"_s ).toDouble(),
2525 json.value( u"y1"_s ).toDouble(),
2526 json.value( u"x2"_s ).toDouble(),
2527 json.value( u"y2"_s ).toDouble(),
2528 type,
2529 &context
2530 ),
2532 bz.toDouble(),
2533 tz.toDouble(),
2534 bcSat,
2535 tcSat,
2536 base,
2537 1,
2538 json.value( u"x1"_s ).toDouble(),
2539 json.value( u"y1"_s ).toDouble(),
2540 json.value( u"x2"_s ).toDouble(),
2541 json.value( u"y2"_s ).toDouble(),
2542 type,
2543 &context
2544 ),
2546 bz.toDouble(),
2547 tz.toDouble(),
2548 bcLight,
2549 tcLight,
2550 base,
2551 1,
2552 json.value( u"x1"_s ).toDouble(),
2553 json.value( u"y1"_s ).toDouble(),
2554 json.value( u"x2"_s ).toDouble(),
2555 json.value( u"y2"_s ).toDouble(),
2556 type,
2557 &context
2558 ),
2560 bz.toDouble(),
2561 tz.toDouble(),
2562 bcAlpha,
2563 tcAlpha,
2564 base,
2565 1,
2566 json.value( u"x1"_s ).toDouble(),
2567 json.value( u"y1"_s ).toDouble(),
2568 json.value( u"x2"_s ).toDouble(),
2569 json.value( u"y2"_s ).toDouble(),
2570 type,
2571 &context
2572 )
2573 );
2574 }
2575 else
2576 {
2577 const QString bottomColorExpr = parseColorExpression( bcVariant, context );
2578 const QString topColorExpr = parseColorExpression( tcVariant, context );
2579
2580 caseString += QStringLiteral(
2581 "WHEN @vector_tile_zoom >= %1 AND @vector_tile_zoom < %2 THEN color_hsla("
2582 "%3, %4, %5, %6) "
2583 )
2584 .arg(
2585 bz,
2586 tz,
2588 bz.toDouble(),
2589 tz.toDouble(),
2590 colorComponent.arg( bottomColorExpr ).arg( "hsl_hue" ),
2591 colorComponent.arg( topColorExpr ).arg( "hsl_hue" ),
2592 base,
2593 1,
2594 json.value( u"x1"_s ).toDouble(),
2595 json.value( u"y1"_s ).toDouble(),
2596 json.value( u"x2"_s ).toDouble(),
2597 json.value( u"y2"_s ).toDouble(),
2598 type,
2599 &context
2600 ),
2602 bz.toDouble(),
2603 tz.toDouble(),
2604 colorComponent.arg( bottomColorExpr ).arg( "hsl_saturation" ),
2605 colorComponent.arg( topColorExpr ).arg( "hsl_saturation" ),
2606 base,
2607 1,
2608 json.value( u"x1"_s ).toDouble(),
2609 json.value( u"y1"_s ).toDouble(),
2610 json.value( u"x2"_s ).toDouble(),
2611 json.value( u"y2"_s ).toDouble(),
2612 type,
2613 &context
2614 ),
2616 bz.toDouble(),
2617 tz.toDouble(),
2618 colorComponent.arg( bottomColorExpr ).arg( "lightness" ),
2619 colorComponent.arg( topColorExpr ).arg( "lightness" ),
2620 base,
2621 1,
2622 json.value( u"x1"_s ).toDouble(),
2623 json.value( u"y1"_s ).toDouble(),
2624 json.value( u"x2"_s ).toDouble(),
2625 json.value( u"y2"_s ).toDouble(),
2626 type,
2627 &context
2628 ),
2630 bz.toDouble(),
2631 tz.toDouble(),
2632 colorComponent.arg( bottomColorExpr ).arg( "alpha" ),
2633 colorComponent.arg( topColorExpr ).arg( "alpha" ),
2634 base,
2635 1,
2636 json.value( u"x1"_s ).toDouble(),
2637 json.value( u"y1"_s ).toDouble(),
2638 json.value( u"x2"_s ).toDouble(),
2639 json.value( u"y2"_s ).toDouble(),
2640 type,
2641 &context
2642 )
2643 );
2644 }
2645 }
2646
2647 // top color
2648 const QString tz = stops.last().toList().value( 0 ).toString();
2649 const QVariant tcVariant = stops.last().toList().value( 1 );
2650 QColor topColor;
2651 if ( tcVariant.userType() == QMetaType::Type::QString )
2652 {
2653 topColor = parseColor( tcVariant, context );
2654 if ( topColor.isValid() )
2655 {
2656 int tcHue;
2657 int tcSat;
2658 int tcLight;
2659 int tcAlpha;
2660 colorAsHslaComponents( topColor, tcHue, tcSat, tcLight, tcAlpha );
2661 caseString += QStringLiteral(
2662 "WHEN @vector_tile_zoom >= %1 THEN color_hsla(%2, %3, %4, %5) "
2663 "ELSE color_hsla(%2, %3, %4, %5) END"
2664 )
2665 .arg( tz )
2666 .arg( tcHue )
2667 .arg( tcSat )
2668 .arg( tcLight )
2669 .arg( tcAlpha );
2670 }
2671 }
2672 else if ( tcVariant.userType() == QMetaType::QVariantList )
2673 {
2674 const QString topColorExpr = parseColorExpression( tcVariant, context );
2675
2676 caseString += QStringLiteral(
2677 "WHEN @vector_tile_zoom >= %1 THEN color_hsla(%2, %3, %4, %5) "
2678 "ELSE color_hsla(%2, %3, %4, %5) END"
2679 )
2680 .arg( tz )
2681 .arg( colorComponent.arg( topColorExpr ).arg( "hsl_hue" ) )
2682 .arg( colorComponent.arg( topColorExpr ).arg( "hsl_saturation" ) )
2683 .arg( colorComponent.arg( topColorExpr ).arg( "lightness" ) )
2684 .arg( colorComponent.arg( topColorExpr ).arg( "alpha" ) );
2685 }
2686
2687 if ( !stops.empty() && defaultColor )
2688 *defaultColor = parseColor( stops.value( 0 ).toList().value( 1 ).toString(), context );
2689
2690 return QgsProperty::fromExpression( caseString );
2691}
2692
2693QgsProperty QgsMapBoxGlStyleConverter::parseInterpolateByZoom( const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, double multiplier, double *defaultNumber, InterpolationType type )
2694{
2695 const double base = json.value( u"base"_s, u"1"_s ).toDouble();
2696 const QVariantList stops = json.value( u"stops"_s ).toList();
2697 if ( stops.empty() )
2698 return QgsProperty();
2699
2700 QString scaleExpression;
2701 if ( stops.size() <= 2 )
2702 {
2703 scaleExpression = interpolateExpression(
2704 stops.value( 0 ).toList().value( 0 ).toDouble(), // zoomMin
2705 stops.last().toList().value( 0 ).toDouble(), // zoomMax
2706 stops.value( 0 ).toList().value( 1 ), // valueMin
2707 stops.last().toList().value( 1 ), // valueMax
2708 base,
2709 multiplier,
2710 json.value( u"x1"_s ).toDouble(),
2711 json.value( u"y1"_s ).toDouble(),
2712 json.value( u"x2"_s ).toDouble(),
2713 json.value( u"y2"_s ).toDouble(),
2714 type,
2715 &context
2716 );
2717 }
2718 else
2719 {
2720 scaleExpression
2721 = parseStops( base, stops, multiplier, context, type, json.value( u"x1"_s ).toDouble(), json.value( u"y1"_s ).toDouble(), json.value( u"x2"_s ).toDouble(), json.value( u"y2"_s ).toDouble() );
2722 }
2723
2724 if ( !stops.empty() && defaultNumber )
2725 *defaultNumber = stops.value( 0 ).toList().value( 1 ).toDouble() * multiplier;
2726
2727 return QgsProperty::fromExpression( scaleExpression );
2728}
2729
2731{
2733 if ( contextPtr )
2734 {
2735 context = *contextPtr;
2736 }
2737 const double base = json.value( u"base"_s, u"1"_s ).toDouble();
2738 const QVariantList stops = json.value( u"stops"_s ).toList();
2739 if ( stops.empty() )
2740 return QgsProperty();
2741
2742 QString scaleExpression;
2743 if ( stops.length() <= 2 )
2744 {
2745 const QVariant bv = stops.value( 0 ).toList().value( 1 );
2746 const QVariant tv = stops.last().toList().value( 1 );
2747 double bottom = 0.0;
2748 double top = 0.0;
2749 const bool numeric = numericArgumentsOnly( bv, tv, bottom, top );
2750 scaleExpression = u"set_color_part(@symbol_color, 'alpha', %1)"_s.arg( interpolateExpression(
2751 stops.value( 0 ).toList().value( 0 ).toDouble(),
2752 stops.last().toList().value( 0 ).toDouble(),
2753 numeric ? QString::number( bottom * maxOpacity ) : QString( "(%1) * %2" ).arg( parseValue( bv, context ) ).arg( maxOpacity ),
2754 numeric ? QString::number( top * maxOpacity ) : QString( "(%1) * %2" ).arg( parseValue( tv, context ) ).arg( maxOpacity ),
2755 base,
2756 1,
2757 json.value( u"x1"_s ).toDouble(),
2758 json.value( u"y1"_s ).toDouble(),
2759 json.value( u"x2"_s ).toDouble(),
2760 json.value( u"y2"_s ).toDouble(),
2761 type,
2762 &context
2763 ) );
2764 }
2765 else
2766 {
2767 scaleExpression
2768 = parseOpacityStops( base, stops, maxOpacity, context, type, json.value( u"x1"_s ).toDouble(), json.value( u"y1"_s ).toDouble(), json.value( u"x2"_s ).toDouble(), json.value( u"y2"_s ).toDouble() );
2769 }
2770 return QgsProperty::fromExpression( scaleExpression );
2771}
2772
2774 double base, const QVariantList &stops, int maxOpacity, QgsMapBoxGlStyleConversionContext &context, InterpolationType type, double x1, double y1, double x2, double y2
2775)
2776{
2777 QString caseString = u"CASE WHEN @vector_tile_zoom < %1 THEN set_color_part(@symbol_color, 'alpha', %2)"_s.arg( stops.value( 0 ).toList().value( 0 ).toString() )
2778 .arg( stops.value( 0 ).toList().value( 1 ).toDouble() * maxOpacity );
2779
2780 for ( int i = 0; i < stops.size() - 1; ++i )
2781 {
2782 const QVariant bv = stops.value( i ).toList().value( 1 );
2783 const QVariant tv = stops.value( i + 1 ).toList().value( 1 );
2784 double bottom = 0.0;
2785 double top = 0.0;
2786 const bool numeric = numericArgumentsOnly( bv, tv, bottom, top );
2787
2788 caseString += QStringLiteral(
2789 " WHEN @vector_tile_zoom >= %1 AND @vector_tile_zoom < %2 "
2790 "THEN set_color_part(@symbol_color, 'alpha', %3)"
2791 )
2792 .arg(
2793 stops.value( i ).toList().value( 0 ).toString(),
2794 stops.value( i + 1 ).toList().value( 0 ).toString(),
2796 stops.value( i ).toList().value( 0 ).toDouble(),
2797 stops.value( i + 1 ).toList().value( 0 ).toDouble(),
2798 numeric ? QString::number( bottom * maxOpacity ) : QString( "(%1) * %2" ).arg( parseValue( bv, context ) ).arg( maxOpacity ),
2799 numeric ? QString::number( top * maxOpacity ) : QString( "(%1) * %2" ).arg( parseValue( tv, context ) ).arg( maxOpacity ),
2800 base,
2801 1,
2802 x1,
2803 y1,
2804 x2,
2805 y2,
2806 type,
2807 &context
2808 )
2809 );
2810 }
2811
2812
2813 bool numeric = false;
2814 const QVariant vv = stops.last().toList().value( 1 );
2815 double dv = vv.toDouble( &numeric );
2816
2817 caseString += QStringLiteral(
2818 " WHEN @vector_tile_zoom >= %1 "
2819 "THEN set_color_part(@symbol_color, 'alpha', %2) END"
2820 )
2821 .arg( stops.last().toList().value( 0 ).toString(), numeric ? QString::number( dv * maxOpacity ) : QString( "(%1) * %2" ).arg( parseValue( vv, context ) ).arg( maxOpacity ) );
2822 return caseString;
2823}
2824
2825QgsProperty QgsMapBoxGlStyleConverter::parseInterpolatePointByZoom( const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, double multiplier, QPointF *defaultPoint, InterpolationType type )
2826{
2827 const double base = json.value( u"base"_s, u"1"_s ).toDouble();
2828 const QVariantList stops = json.value( u"stops"_s ).toList();
2829 if ( stops.empty() )
2830 return QgsProperty();
2831
2832 QString scaleExpression;
2833 if ( stops.size() <= 2 )
2834 {
2835 scaleExpression = u"array(%1,%2)"_s.arg(
2837 stops.value( 0 ).toList().value( 0 ).toDouble(),
2838 stops.last().toList().value( 0 ).toDouble(),
2839 stops.value( 0 ).toList().value( 1 ).toList().value( 0 ),
2840 stops.last().toList().value( 1 ).toList().value( 0 ),
2841 base,
2842 multiplier,
2843 json.value( u"x1"_s ).toDouble(),
2844 json.value( u"y1"_s ).toDouble(),
2845 json.value( u"x2"_s ).toDouble(),
2846 json.value( u"y2"_s ).toDouble(),
2847 type,
2848 &context
2849 ),
2851 stops.value( 0 ).toList().value( 0 ).toDouble(),
2852 stops.last().toList().value( 0 ).toDouble(),
2853 stops.value( 0 ).toList().value( 1 ).toList().value( 1 ),
2854 stops.last().toList().value( 1 ).toList().value( 1 ),
2855 base,
2856 multiplier,
2857 json.value( u"x1"_s ).toDouble(),
2858 json.value( u"y1"_s ).toDouble(),
2859 json.value( u"x2"_s ).toDouble(),
2860 json.value( u"y2"_s ).toDouble(),
2861 type,
2862 &context
2863 )
2864 );
2865 }
2866 else
2867 {
2868 scaleExpression
2869 = parsePointStops( base, stops, context, multiplier, type, json.value( u"x1"_s ).toDouble(), json.value( u"y1"_s ).toDouble(), json.value( u"x2"_s ).toDouble(), json.value( u"y2"_s ).toDouble() );
2870 }
2871
2872 if ( !stops.empty() && defaultPoint )
2873 *defaultPoint = QPointF( stops.value( 0 ).toList().value( 1 ).toList().value( 0 ).toDouble() * multiplier, stops.value( 0 ).toList().value( 1 ).toList().value( 1 ).toDouble() * multiplier );
2874
2875 return QgsProperty::fromExpression( scaleExpression );
2876}
2877
2878QgsProperty QgsMapBoxGlStyleConverter::parseInterpolateStringByZoom( const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, const QVariantMap &conversionMap, QString *defaultString )
2879{
2880 const QVariantList stops = json.value( u"stops"_s ).toList();
2881 if ( stops.empty() )
2882 return QgsProperty();
2883
2884 const QString scaleExpression = parseStringStops( stops, context, conversionMap, defaultString );
2885
2886 return QgsProperty::fromExpression( scaleExpression );
2887}
2888
2890 double base, const QVariantList &stops, QgsMapBoxGlStyleConversionContext &context, double multiplier, InterpolationType type, double x1, double y1, double x2, double y2
2891)
2892{
2893 QString caseString = u"CASE "_s;
2894
2895 for ( int i = 0; i < stops.length() - 1; ++i )
2896 {
2897 // bottom zoom and value
2898 const QVariant bz = stops.value( i ).toList().value( 0 );
2899 const QVariant bv = stops.value( i ).toList().value( 1 );
2900 if ( bv.userType() != QMetaType::Type::QVariantList && bv.userType() != QMetaType::Type::QStringList )
2901 {
2902 context.pushWarning( QObject::tr( "%1: Skipping unsupported offset interpolation type (%2)." ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( bz.userType() ) ) ) );
2903 return QString();
2904 }
2905
2906 // top zoom and value
2907 const QVariant tz = stops.value( i + 1 ).toList().value( 0 );
2908 const QVariant tv = stops.value( i + 1 ).toList().value( 1 );
2909 if ( tv.userType() != QMetaType::Type::QVariantList && tv.userType() != QMetaType::Type::QStringList )
2910 {
2911 context.pushWarning( QObject::tr( "%1: Skipping unsupported offset interpolation type (%2)." ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( tz.userType() ) ) ) );
2912 return QString();
2913 }
2914
2915 caseString += QStringLiteral(
2916 "WHEN @vector_tile_zoom > %1 AND @vector_tile_zoom <= %2 "
2917 "THEN array(%3,%4)"
2918 )
2919 .arg(
2920 bz.toString(),
2921 tz.toString(),
2922 interpolateExpression( bz.toDouble(), tz.toDouble(), bv.toList().value( 0 ), tv.toList().value( 0 ), base, multiplier, x1, y1, x2, y2, type, &context ),
2923 interpolateExpression( bz.toDouble(), tz.toDouble(), bv.toList().value( 1 ), tv.toList().value( 1 ), base, multiplier, x1, y1, x2, y2, type, &context )
2924 );
2925 }
2926 caseString += "END"_L1;
2927 return caseString;
2928}
2929
2930QString QgsMapBoxGlStyleConverter::parseArrayStops( const QVariantList &stops, QgsMapBoxGlStyleConversionContext &, double multiplier )
2931{
2932 if ( stops.length() < 2 )
2933 return QString();
2934
2935 QString caseString = u"CASE"_s;
2936
2937 for ( int i = 0; i < stops.length(); ++i )
2938 {
2939 caseString += " WHEN "_L1;
2940 QStringList conditions;
2941 if ( i > 0 )
2942 {
2943 const QVariant bottomZoom = stops.value( i ).toList().value( 0 );
2944 conditions << u"@vector_tile_zoom > %1"_s.arg( bottomZoom.toString() );
2945 }
2946 if ( i < stops.length() - 1 )
2947 {
2948 const QVariant topZoom = stops.value( i + 1 ).toList().value( 0 );
2949 conditions << u"@vector_tile_zoom <= %1"_s.arg( topZoom.toString() );
2950 }
2951
2952 const QVariantList values = stops.value( i ).toList().value( 1 ).toList();
2953 QStringList valuesFixed;
2954 bool ok = false;
2955 for ( const QVariant &value : values )
2956 {
2957 const double number = value.toDouble( &ok );
2958 if ( ok )
2959 valuesFixed << QString::number( number * multiplier );
2960 }
2961
2962 // top zoom and value
2963 caseString += u"%1 THEN array(%3)"_s.arg( conditions.join( " AND "_L1 ), valuesFixed.join( ',' ) );
2964 }
2965 caseString += " END"_L1;
2966 return caseString;
2967}
2968
2970 double base, const QVariantList &stops, double multiplier, QgsMapBoxGlStyleConversionContext &context, InterpolationType type, double x1, double y1, double x2, double y2
2971)
2972{
2973 QString caseString = u"CASE "_s;
2974
2975 for ( int i = 0; i < stops.length() - 1; ++i )
2976 {
2977 // bottom zoom and value
2978 const QVariant bz = stops.value( i ).toList().value( 0 );
2979 const QVariant bv = stops.value( i ).toList().value( 1 );
2980 if ( bz.userType() == QMetaType::Type::QVariantList || bz.userType() == QMetaType::Type::QStringList )
2981 {
2982 context.pushWarning( QObject::tr( "%1: Expressions in interpolation function are not supported, skipping." ).arg( context.layerId() ) );
2983 return QString();
2984 }
2985
2986 // top zoom and value
2987 const QVariant tz = stops.value( i + 1 ).toList().value( 0 );
2988 const QVariant tv = stops.value( i + 1 ).toList().value( 1 );
2989 if ( tz.userType() == QMetaType::Type::QVariantList || tz.userType() == QMetaType::Type::QStringList )
2990 {
2991 context.pushWarning( QObject::tr( "%1: Expressions in interpolation function are not supported, skipping." ).arg( context.layerId() ) );
2992 return QString();
2993 }
2994
2995 const QString lowerComparator = i == 0 ? u">="_s : u">"_s;
2996
2997 caseString += QStringLiteral(
2998 "WHEN @vector_tile_zoom %1 %2 AND @vector_tile_zoom <= %3 "
2999 "THEN %4 "
3000 )
3001 .arg( lowerComparator, bz.toString(), tz.toString(), interpolateExpression( bz.toDouble(), tz.toDouble(), bv, tv, base, multiplier, x1, y1, x2, y2, type, &context ) );
3002 }
3003
3004 const QVariant z = stops.last().toList().value( 0 );
3005 const QVariant v = stops.last().toList().value( 1 );
3006 QString vStr = v.toString();
3007 if ( ( QMetaType::Type ) v.userType() == QMetaType::QVariantList )
3008 {
3009 vStr = parseExpression( v.toList(), context );
3010 caseString += QStringLiteral(
3011 "WHEN @vector_tile_zoom > %1 "
3012 "THEN ( ( %2 ) * %3 ) END"
3013 )
3014 .arg( z.toString() )
3015 .arg( vStr )
3016 .arg( multiplier );
3017 }
3018 else
3019 {
3020 caseString += QStringLiteral(
3021 "WHEN @vector_tile_zoom > %1 "
3022 "THEN %2 END"
3023 )
3024 .arg( z.toString() )
3025 .arg( v.toDouble() * multiplier );
3026 }
3027
3028 return caseString;
3029}
3030
3031QString QgsMapBoxGlStyleConverter::parseStringStops( const QVariantList &stops, QgsMapBoxGlStyleConversionContext &context, const QVariantMap &conversionMap, QString *defaultString )
3032{
3033 QString caseString = u"CASE "_s;
3034
3035 for ( int i = 0; i < stops.length() - 1; ++i )
3036 {
3037 // bottom zoom and value
3038 const QVariant bz = stops.value( i ).toList().value( 0 );
3039 const QString bv = stops.value( i ).toList().value( 1 ).toString();
3040 if ( bz.userType() == QMetaType::Type::QVariantList || bz.userType() == QMetaType::Type::QStringList )
3041 {
3042 context.pushWarning( QObject::tr( "%1: Expressions in interpolation function are not supported, skipping." ).arg( context.layerId() ) );
3043 return QString();
3044 }
3045
3046 // top zoom
3047 const QVariant tz = stops.value( i + 1 ).toList().value( 0 );
3048 if ( tz.userType() == QMetaType::Type::QVariantList || tz.userType() == QMetaType::Type::QStringList )
3049 {
3050 context.pushWarning( QObject::tr( "%1: Expressions in interpolation function are not supported, skipping." ).arg( context.layerId() ) );
3051 return QString();
3052 }
3053
3054 caseString += QStringLiteral(
3055 "WHEN @vector_tile_zoom > %1 AND @vector_tile_zoom <= %2 "
3056 "THEN %3 "
3057 )
3058 .arg( bz.toString(), tz.toString(), QgsExpression::quotedValue( conversionMap.value( bv, bv ) ) );
3059 }
3060 caseString += u"ELSE %1 END"_s.arg( QgsExpression::quotedValue( conversionMap.value( stops.constLast().toList().value( 1 ).toString(), stops.constLast().toList().value( 1 ) ) ) );
3061 if ( defaultString )
3062 *defaultString = stops.constLast().toList().value( 1 ).toString();
3063 return caseString;
3064}
3065
3067{
3068 QString caseString = u"CASE "_s;
3069
3070 bool isExpression = false;
3071 for ( int i = 0; i < stops.length() - 1; ++i )
3072 {
3073 // bottom zoom and value
3074 const QVariant bz = stops.value( i ).toList().value( 0 );
3075 if ( bz.userType() == QMetaType::Type::QVariantList || bz.userType() == QMetaType::Type::QStringList )
3076 {
3077 context.pushWarning( QObject::tr( "%1: Lists in label interpolation function are not supported, skipping." ).arg( context.layerId() ) );
3078 return QString();
3079 }
3080
3081 // top zoom
3082 const QVariant tz = stops.value( i + 1 ).toList().value( 0 );
3083 if ( tz.userType() == QMetaType::Type::QVariantList || tz.userType() == QMetaType::Type::QStringList )
3084 {
3085 context.pushWarning( QObject::tr( "%1: Lists in label interpolation function are not supported, skipping." ).arg( context.layerId() ) );
3086 return QString();
3087 }
3088
3089 QString fieldPart = processLabelField( stops.constLast().toList().value( 1 ).toString(), isExpression );
3090 if ( fieldPart.isEmpty() )
3091 fieldPart = u"''"_s;
3092 else if ( !isExpression )
3093 fieldPart = QgsExpression::quotedColumnRef( fieldPart );
3094
3095 caseString += QStringLiteral(
3096 "WHEN @vector_tile_zoom > %1 AND @vector_tile_zoom < %2 "
3097 "THEN %3 "
3098 )
3099 .arg( bz.toString(), tz.toString(), fieldPart );
3100 }
3101
3102 {
3103 const QVariant bz = stops.constLast().toList().value( 0 );
3104 if ( bz.userType() == QMetaType::Type::QVariantList || bz.userType() == QMetaType::Type::QStringList )
3105 {
3106 context.pushWarning( QObject::tr( "%1: Lists in label interpolation function are not supported, skipping." ).arg( context.layerId() ) );
3107 return QString();
3108 }
3109
3110 QString fieldPart = processLabelField( stops.constLast().toList().value( 1 ).toString(), isExpression );
3111 if ( fieldPart.isEmpty() )
3112 fieldPart = u"''"_s;
3113 else if ( !isExpression )
3114 fieldPart = QgsExpression::quotedColumnRef( fieldPart );
3115
3116 caseString += QStringLiteral(
3117 "WHEN @vector_tile_zoom >= %1 "
3118 "THEN %3 "
3119 )
3120 .arg( bz.toString(), fieldPart );
3121 }
3122
3123 QString defaultPart = processLabelField( stops.constFirst().toList().value( 1 ).toString(), isExpression );
3124 if ( defaultPart.isEmpty() )
3125 defaultPart = u"''"_s;
3126 else if ( !isExpression )
3127 defaultPart = QgsExpression::quotedColumnRef( defaultPart );
3128 caseString += u"ELSE %1 END"_s.arg( defaultPart );
3129
3130 return caseString;
3131}
3132
3134 const QVariantList &json, QgsMapBoxGlStyleConverter::PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier, int maxOpacity, QColor *defaultColor, double *defaultNumber
3135)
3136{
3137 const QString method = json.value( 0 ).toString();
3138 if ( method == "interpolate"_L1 )
3139 {
3140 return parseInterpolateListByZoom( json, type, context, multiplier, maxOpacity, defaultColor, defaultNumber );
3141 }
3142 else if ( method == "match"_L1 )
3143 {
3144 return parseMatchList( json, type, context, multiplier, maxOpacity, defaultColor, defaultNumber );
3145 }
3146 else if ( method == "step"_L1 )
3147 {
3148 return parseStepList( json, type, context, multiplier, maxOpacity, defaultColor, defaultNumber );
3149 }
3150 else
3151 {
3152 return QgsProperty::fromExpression( parseExpression( json, context, type == PropertyType::Color ) );
3153 }
3154}
3155
3157 const QVariantList &json, QgsMapBoxGlStyleConverter::PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier, int maxOpacity, QColor *defaultColor, double *defaultNumber
3158)
3159{
3160 const QString attribute = parseExpression( json.value( 1 ).toList(), context );
3161 if ( attribute.isEmpty() )
3162 {
3163 context.pushWarning( QObject::tr( "%1: Could not interpret match list" ).arg( context.layerId() ) );
3164 return QgsProperty();
3165 }
3166
3167 QString caseString = u"CASE "_s;
3168
3169 for ( int i = 2; i < json.length() - 1; i += 2 )
3170 {
3171 QVariantList keys;
3172 QVariant variantKeys = json.value( i );
3173 if ( variantKeys.userType() == QMetaType::Type::QVariantList || variantKeys.userType() == QMetaType::Type::QStringList )
3174 keys = variantKeys.toList();
3175 else
3176 keys = { variantKeys };
3177
3178 QStringList matchString;
3179 for ( const QVariant &key : keys )
3180 {
3181 matchString << QgsExpression::quotedValue( key );
3182 }
3183
3184 const QVariant value = json.value( i + 1 );
3185
3186 QString valueString;
3187 switch ( type )
3188 {
3190 {
3191 if ( value.userType() == QMetaType::Type::QVariantList || value.userType() == QMetaType::Type::QStringList )
3192 {
3193 valueString = parseMatchList( value.toList(), PropertyType::Color, context, multiplier, maxOpacity, defaultColor, defaultNumber ).asExpression();
3194 }
3195 else
3196 {
3197 const QColor color = parseColor( value, context );
3198 valueString = QgsExpression::quotedString( color.name() );
3199 }
3200 break;
3201 }
3202
3204 {
3205 const double v = value.toDouble() * multiplier;
3206 valueString = QString::number( v );
3207 break;
3208 }
3209
3211 {
3212 const double v = value.toDouble() * maxOpacity;
3213 valueString = QString::number( v );
3214 break;
3215 }
3216
3218 {
3219 valueString = u"array(%1,%2)"_s.arg( value.toList().value( 0 ).toDouble() * multiplier, value.toList().value( 0 ).toDouble() * multiplier );
3220 break;
3221 }
3222
3224 {
3225 if ( value.toList().count() == 2 && value.toList().first().toString() == "literal"_L1 )
3226 {
3227 valueString = u"array(%1)"_s.arg( value.toList().at( 1 ).toStringList().join( ',' ) );
3228 }
3229 else
3230 {
3231 valueString = u"array(%1)"_s.arg( value.toStringList().join( ',' ) );
3232 }
3233 break;
3234 }
3235
3237 {
3238 if ( value.toList().count() == 2 && value.toList().first().toString() == "literal"_L1 )
3239 {
3240 QStringList dashValues = value.toList().at( 1 ).toStringList();
3241 if ( dashValues.length() % 2 == 1 )
3242 {
3243 dashValues << u"0"_s;
3244 }
3245 valueString = u"array(%1)"_s.arg( dashValues.join( ',' ) );
3246 }
3247 else
3248 {
3249 QStringList dashValues = value.toStringList();
3250 if ( dashValues.length() % 2 == 1 )
3251 {
3252 dashValues << u"0"_s;
3253 }
3254 valueString = u"array(%1)"_s.arg( dashValues.join( ',' ) );
3255 }
3256 break;
3257 }
3258 }
3259
3260 if ( matchString.count() == 1 )
3261 {
3262 caseString += u"WHEN %1 IS %2 THEN %3 "_s.arg( attribute, matchString.at( 0 ), valueString );
3263 }
3264 else
3265 {
3266 caseString += u"WHEN %1 IN (%2) THEN %3 "_s.arg( attribute, matchString.join( ',' ), valueString );
3267 }
3268 }
3269
3270 QVariant lastValue = json.constLast();
3271 QString elseValue;
3272
3273 switch ( lastValue.userType() )
3274 {
3275 case QMetaType::Type::QVariantList:
3276 case QMetaType::Type::QStringList:
3277 elseValue = parseValueList( lastValue.toList(), type, context, multiplier, maxOpacity, defaultColor, defaultNumber ).asExpression();
3278 break;
3279
3280 default:
3281 {
3282 switch ( type )
3283 {
3285 {
3286 const QColor color = parseColor( lastValue, context );
3287 if ( defaultColor )
3288 *defaultColor = color;
3289
3290 elseValue = QgsExpression::quotedString( color.name() );
3291 break;
3292 }
3293
3295 {
3296 const double v = json.constLast().toDouble() * multiplier;
3297 if ( defaultNumber )
3298 *defaultNumber = v;
3299 elseValue = QString::number( v );
3300 break;
3301 }
3302
3304 {
3305 const double v = json.constLast().toDouble() * maxOpacity;
3306 if ( defaultNumber )
3307 *defaultNumber = v;
3308 elseValue = QString::number( v );
3309 break;
3310 }
3311
3313 {
3314 elseValue = u"array(%1,%2)"_s.arg( json.constLast().toList().value( 0 ).toDouble() * multiplier ).arg( json.constLast().toList().value( 0 ).toDouble() * multiplier );
3315 break;
3316 }
3317
3319 {
3320 if ( json.constLast().toList().count() == 2 && json.constLast().toList().first().toString() == "literal"_L1 )
3321 {
3322 elseValue = u"array(%1)"_s.arg( json.constLast().toList().at( 1 ).toStringList().join( ',' ) );
3323 }
3324 else
3325 {
3326 elseValue = u"array(%1)"_s.arg( json.constLast().toStringList().join( ',' ) );
3327 }
3328 break;
3329 }
3330
3332 {
3333 if ( json.constLast().toList().count() == 2 && json.constLast().toList().first().toString() == "literal"_L1 )
3334 {
3335 QStringList dashValues = json.constLast().toList().at( 1 ).toStringList();
3336 if ( dashValues.length() % 2 == 1 )
3337 {
3338 dashValues << u"0"_s;
3339 }
3340 elseValue = u"array(%1)"_s.arg( dashValues.join( ',' ) );
3341 }
3342 else
3343 {
3344 QStringList dashValues = json.constLast().toStringList();
3345 if ( dashValues.length() % 2 == 1 )
3346 {
3347 dashValues << u"0"_s;
3348 }
3349 elseValue = u"array(%1)"_s.arg( dashValues.join( ',' ) );
3350 }
3351 break;
3352 }
3353 }
3354 break;
3355 }
3356 }
3357
3358 caseString += u"ELSE %1 END"_s.arg( elseValue );
3359 return QgsProperty::fromExpression( caseString );
3360}
3361
3363 const QVariantList &json, PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier, int maxOpacity, QColor *defaultColor, double *defaultNumber
3364)
3365{
3366 const QString expression = parseExpression( json.value( 1 ).toList(), context );
3367 if ( expression.isEmpty() )
3368 {
3369 context.pushWarning( QObject::tr( "%1: Could not interpret step list" ).arg( context.layerId() ) );
3370 return QgsProperty();
3371 }
3372
3373 QString caseString = u"CASE "_s;
3374
3375
3376 for ( int i = json.length() - 2; i > 0; i -= 2 )
3377 {
3378 const QVariant stepValue = json.value( i + 1 );
3379
3380 QString valueString;
3381 if ( stepValue.canConvert<QVariantList>() && ( stepValue.toList().count() != 2 || type != PropertyType::Point ) && type != PropertyType::NumericArray && type != PropertyType::DashArray )
3382 {
3383 valueString = parseValueList( stepValue.toList(), type, context, multiplier, maxOpacity, defaultColor, defaultNumber ).expressionString();
3384 }
3385 else
3386 {
3387 switch ( type )
3388 {
3390 {
3391 const QColor color = parseColor( stepValue, context );
3392 valueString = QgsExpression::quotedString( color.name() );
3393 break;
3394 }
3395
3397 {
3398 const double v = stepValue.toDouble() * multiplier;
3399 valueString = QString::number( v );
3400 break;
3401 }
3402
3404 {
3405 const double v = stepValue.toDouble() * maxOpacity;
3406 valueString = QString::number( v );
3407 break;
3408 }
3409
3411 {
3412 valueString = u"array(%1,%2)"_s.arg( stepValue.toList().value( 0 ).toDouble() * multiplier ).arg( stepValue.toList().value( 0 ).toDouble() * multiplier );
3413 break;
3414 }
3415
3417 {
3418 if ( stepValue.toList().count() == 2 && stepValue.toList().first().toString() == "literal"_L1 )
3419 {
3420 valueString = u"array(%1)"_s.arg( stepValue.toList().at( 1 ).toStringList().join( ',' ) );
3421 }
3422 else
3423 {
3424 valueString = u"array(%1)"_s.arg( stepValue.toStringList().join( ',' ) );
3425 }
3426 break;
3427 }
3428
3430 {
3431 if ( stepValue.toList().count() == 2 && stepValue.toList().first().toString() == "literal"_L1 )
3432 {
3433 QStringList dashValues = stepValue.toList().at( 1 ).toStringList();
3434 if ( dashValues.length() % 2 == 1 )
3435 {
3436 dashValues << u"0"_s;
3437 }
3438 valueString = u"array(%1)"_s.arg( dashValues.join( ',' ) );
3439 }
3440 else
3441 {
3442 QStringList dashValues = stepValue.toStringList();
3443 if ( dashValues.length() % 2 == 1 )
3444 {
3445 dashValues << u"0"_s;
3446 }
3447 valueString = u"array(%1)"_s.arg( dashValues.join( ',' ) );
3448 }
3449 break;
3450 }
3451 }
3452 }
3453
3454 if ( i > 1 )
3455 {
3456 const QString stepKey = QgsExpression::quotedValue( json.value( i ) );
3457 caseString += u" WHEN %1 >= %2 THEN (%3) "_s.arg( expression, stepKey, valueString );
3458 }
3459 else
3460 {
3461 caseString += u"ELSE (%1) END"_s.arg( valueString );
3462 }
3463 }
3464 return QgsProperty::fromExpression( caseString );
3465}
3466
3468 const QVariantList &json, PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier, int maxOpacity, QColor *defaultColor, double *defaultNumber
3469)
3470{
3471 if ( json.value( 0 ).toString() != "interpolate"_L1 )
3472 {
3473 context.pushWarning( QObject::tr( "%1: Could not interpret value list" ).arg( context.layerId() ) );
3474 return QgsProperty();
3475 }
3476
3477 const QVariantList parts = json.value( 1 ).toList();
3478 const QString technique = parts.value( 0 ).toString();
3479 InterpolationType interpolationType = InterpolationType::Linear;
3480 QVariantMap props;
3481
3482 if ( technique == "linear"_L1 )
3483 {
3484 props.insert( u"base"_s, 1 );
3485 interpolationType = InterpolationType::Linear;
3486 }
3487 else if ( technique == "exponential"_L1 )
3488 {
3489 props.insert( u"base"_s, parts.value( 1 ).toDouble() );
3490 interpolationType = InterpolationType::Exponential;
3491 }
3492 else if ( technique == "cubic-bezier"_L1 )
3493 {
3494 interpolationType = InterpolationType::CubicBezier;
3495
3496 props.insert( u"x1"_s, parts.value( 1 ).toDouble() );
3497 props.insert( u"y1"_s, parts.value( 2 ).toDouble() );
3498 props.insert( u"x2"_s, parts.value( 3 ).toDouble() );
3499 props.insert( u"y2"_s, parts.value( 4 ).toDouble() );
3500 }
3501 else
3502 {
3503 context.pushWarning( QObject::tr( "%1: Skipping not implemented interpolation method %2" ).arg( context.layerId(), technique ) );
3504 return QgsProperty();
3505 }
3506
3507 if ( json.value( 2 ).toList().value( 0 ).toString() != "zoom"_L1 )
3508 {
3509 context.pushWarning( QObject::tr( "%1: Skipping not implemented interpolation input %2" ).arg( context.layerId(), json.value( 2 ).toString() ) );
3510 return QgsProperty();
3511 }
3512
3513 // Convert stops into list of lists
3514 QVariantList stops;
3515 for ( int i = 3; i < json.length(); i += 2 )
3516 {
3517 stops.push_back( QVariantList() << json.value( i ).toString() << json.value( i + 1 ) );
3518 }
3519
3520 props.insert( u"stops"_s, stops );
3521
3522 switch ( type )
3523 {
3525 return parseInterpolateColorByZoom( props, context, defaultColor, interpolationType );
3526
3528 return parseInterpolateByZoom( props, context, multiplier, defaultNumber, interpolationType );
3529
3531 return parseInterpolateOpacityByZoom( props, maxOpacity, &context, interpolationType );
3532
3534 return parseInterpolatePointByZoom( props, context, multiplier, nullptr, interpolationType );
3535
3538 context.pushWarning( QObject::tr( "%1: Skipping unsupported numeric array in interpolate" ).arg( context.layerId() ) );
3539 return QgsProperty();
3540 }
3541 return QgsProperty();
3542}
3543
3545{
3546 if ( ( QMetaType::Type ) colorExpression.userType() == QMetaType::QVariantList )
3547 {
3548 return parseExpression( colorExpression.toList(), context, true );
3549 }
3550 return parseValue( colorExpression, context, true );
3551}
3552
3554{
3555 if ( color.userType() != QMetaType::Type::QString )
3556 {
3557 context.pushWarning( QObject::tr( "%1: Could not parse non-string color %2, skipping" ).arg( context.layerId(), color.toString() ) );
3558 return QColor();
3559 }
3560
3561 return QgsSymbolLayerUtils::parseColor( color.toString() );
3562}
3563
3564void QgsMapBoxGlStyleConverter::colorAsHslaComponents( const QColor &color, int &hue, int &saturation, int &lightness, int &alpha )
3565{
3566 hue = std::max( 0, color.hslHue() );
3567 saturation = color.hslSaturation() / 255.0 * 100;
3568 lightness = color.lightness() / 255.0 * 100;
3569 alpha = color.alpha();
3570}
3571
3573 double zoomMin, double zoomMax, QVariant valueMin, QVariant valueMax, double base, double multiplier, double x1, double y1, double x2, double y2, InterpolationType type, QgsMapBoxGlStyleConversionContext *contextPtr
3574)
3575{
3577 if ( contextPtr )
3578 {
3579 context = *contextPtr;
3580 }
3581
3582 // special case where min = max !
3583 if ( valueMin.canConvert( QMetaType::Double ) && valueMax.canConvert( QMetaType::Double ) )
3584 {
3585 bool minDoubleOk = true;
3586 const double min = valueMin.toDouble( &minDoubleOk );
3587 bool maxDoubleOk = true;
3588 const double max = valueMax.toDouble( &maxDoubleOk );
3589 if ( minDoubleOk && maxDoubleOk && qgsDoubleNear( min, max ) )
3590 {
3591 return QString::number( min * multiplier );
3592 }
3593 }
3594
3595 QString minValueExpr = valueMin.toString();
3596 QString maxValueExpr = valueMax.toString();
3597 if ( valueMin.userType() == QMetaType::Type::QVariantList )
3598 {
3599 minValueExpr = parseExpression( valueMin.toList(), context );
3600 }
3601 if ( valueMax.userType() == QMetaType::Type::QVariantList )
3602 {
3603 maxValueExpr = parseExpression( valueMax.toList(), context );
3604 }
3605
3606 QString expression;
3607 if ( minValueExpr == maxValueExpr )
3608 {
3609 expression = minValueExpr;
3610 }
3611 else
3612 {
3613 switch ( type )
3614 {
3616 expression = u"scale_linear(@vector_tile_zoom,%1,%2,%3,%4)"_s.arg( zoomMin ).arg( zoomMax ).arg( minValueExpr ).arg( maxValueExpr );
3617 break;
3619 if ( base == 1 )
3620 {
3621 expression = u"scale_linear(@vector_tile_zoom,%1,%2,%3,%4)"_s.arg( zoomMin ).arg( zoomMax ).arg( minValueExpr ).arg( maxValueExpr );
3622 }
3623 else
3624 {
3625 expression = u"scale_exponential(@vector_tile_zoom,%1,%2,%3,%4,%5)"_s.arg( zoomMin ).arg( zoomMax ).arg( minValueExpr ).arg( maxValueExpr ).arg( base );
3626 }
3627 break;
3628
3630 expression = u"scale_cubic_bezier(@vector_tile_zoom,%1,%2,%3,%4,%5,%6,%7,%8)"_s.arg( zoomMin ).arg( zoomMax ).arg( minValueExpr ).arg( maxValueExpr ).arg( x1 ).arg( y1 ).arg( x2 ).arg( y2 );
3631 break;
3632 }
3633 }
3634
3635 if ( multiplier != 1 )
3636 return u"(%1) * %2"_s.arg( expression ).arg( multiplier );
3637 else
3638 return expression;
3639}
3640
3641Qt::PenCapStyle QgsMapBoxGlStyleConverter::parseCapStyle( const QString &style )
3642{
3643 if ( style == "round"_L1 )
3644 return Qt::RoundCap;
3645 else if ( style == "square"_L1 )
3646 return Qt::SquareCap;
3647 else
3648 return Qt::FlatCap; // "butt" is default
3649}
3650
3651Qt::PenJoinStyle QgsMapBoxGlStyleConverter::parseJoinStyle( const QString &style )
3652{
3653 if ( style == "bevel"_L1 )
3654 return Qt::BevelJoin;
3655 else if ( style == "round"_L1 )
3656 return Qt::RoundJoin;
3657 else
3658 return Qt::MiterJoin; // "miter" is default
3659}
3660
3661QString QgsMapBoxGlStyleConverter::parseExpression( const QVariantList &expression, QgsMapBoxGlStyleConversionContext &context, bool colorExpected )
3662{
3663 QString op = expression.value( 0 ).toString();
3664 if ( ( op == "%"_L1 || op == "/"_L1 || op == "-"_L1 || op == "^"_L1 ) && expression.size() >= 3 )
3665 {
3666 if ( expression.size() != 3 )
3667 {
3668 context.pushWarning( QObject::tr( "%1: Operator %2 requires exactly two operands, skipping extra operands" ).arg( context.layerId() ).arg( op ) );
3669 }
3670 QString v1 = parseValue( expression.value( 1 ), context, colorExpected );
3671 QString v2 = parseValue( expression.value( 2 ), context, colorExpected );
3672 return u"(%1 %2 %3)"_s.arg( v1, op, v2 );
3673 }
3674 else if ( ( op == "*"_L1 || op == "+"_L1 ) && expression.size() >= 3 )
3675 {
3676 QStringList operands;
3677 std::transform( std::next( expression.begin() ), expression.end(), std::back_inserter( operands ), [&context, colorExpected]( const QVariant &val ) {
3678 return parseValue( val, context, colorExpected );
3679 } );
3680 return u"(%1)"_s.arg( operands.join( u" %1 "_s.arg( op ) ) );
3681 }
3682 else if ( op == "to-number"_L1 )
3683 {
3684 return u"to_real(%1)"_s.arg( parseValue( expression.value( 1 ), context ) );
3685 }
3686 else if ( op == "sqrt"_L1 )
3687 {
3688 return u"sqrt(%1)"_s.arg( parseValue( expression.value( 1 ), context ) );
3689 }
3690 else if ( op == "literal"_L1 )
3691 {
3692 return expression.value( 1 ).toString();
3693 }
3694 else if ( op == "all"_L1 || op == "any"_L1 || op == "none"_L1 )
3695 {
3696 QStringList parts;
3697 for ( int i = 1; i < expression.size(); ++i )
3698 {
3699 const QString part = parseValue( expression.at( i ), context );
3700 if ( part.isEmpty() )
3701 {
3702 context.pushWarning( QObject::tr( "%1: Skipping unsupported expression" ).arg( context.layerId() ) );
3703 return QString();
3704 }
3705 parts << part;
3706 }
3707
3708 if ( op == "none"_L1 )
3709 return u"NOT (%1)"_s.arg( parts.join( ") AND NOT ("_L1 ) );
3710
3711 QString operatorString;
3712 if ( op == "all"_L1 )
3713 operatorString = u") AND ("_s;
3714 else if ( op == "any"_L1 )
3715 operatorString = u") OR ("_s;
3716
3717 return u"(%1)"_s.arg( parts.join( operatorString ) );
3718 }
3719 else if ( op == '!' )
3720 {
3721 // ! inverts next expression's meaning
3722 QVariantList contraJsonExpr = expression.value( 1 ).toList();
3723 contraJsonExpr[0] = QString( op + contraJsonExpr[0].toString() );
3724 // ['!', ['has', 'level']] -> ['!has', 'level']
3725 return parseKey( contraJsonExpr, context );
3726 }
3727 else if ( op == "=="_L1 || op == "!="_L1 || op == ">="_L1 || op == '>' || op == "<="_L1 || op == '<' )
3728 {
3729 // use IS and NOT IS instead of = and != because they can deal with NULL values
3730 if ( op == "=="_L1 )
3731 op = u"IS"_s;
3732 else if ( op == "!="_L1 )
3733 op = u"IS NOT"_s;
3734 return u"%1 %2 %3"_s.arg( parseKey( expression.value( 1 ), context ), op, parseValue( expression.value( 2 ), context ) );
3735 }
3736 else if ( op == "has"_L1 )
3737 {
3738 return parseKey( expression.value( 1 ), context ) + u" IS NOT NULL"_s;
3739 }
3740 else if ( op == "!has"_L1 )
3741 {
3742 return parseKey( expression.value( 1 ), context ) + u" IS NULL"_s;
3743 }
3744 else if ( op == "in"_L1 || op == "!in"_L1 )
3745 {
3746 const QString key = parseKey( expression.value( 1 ), context );
3747 QStringList parts;
3748
3749 QVariantList values = expression.mid( 2 );
3750 if ( expression.size() == 3 && expression.at( 2 ).userType() == QMetaType::Type::QVariantList && expression.at( 2 ).toList().count() > 1 && expression.at( 2 ).toList().at( 0 ).toString() == "literal"_L1 )
3751 {
3752 values = expression.at( 2 ).toList().at( 1 ).toList();
3753 }
3754
3755 for ( const QVariant &value : std::as_const( values ) )
3756 {
3757 const QString part = parseValue( value, context );
3758 if ( part.isEmpty() )
3759 {
3760 context.pushWarning( QObject::tr( "%1: Skipping unsupported expression" ).arg( context.layerId() ) );
3761 return QString();
3762 }
3763 parts << part;
3764 }
3765
3766 if ( parts.size() == 1 )
3767 {
3768 if ( op == "in"_L1 )
3769 return u"%1 IS %2"_s.arg( key, parts.at( 0 ) );
3770 else
3771 return u"(%1 IS NULL OR %1 IS NOT %2)"_s.arg( key, parts.at( 0 ) );
3772 }
3773 else
3774 {
3775 if ( op == "in"_L1 )
3776 return u"%1 IN (%2)"_s.arg( key, parts.join( ", "_L1 ) );
3777 else
3778 return u"(%1 IS NULL OR %1 NOT IN (%2))"_s.arg( key, parts.join( ", "_L1 ) );
3779 }
3780 }
3781 else if ( op == "get"_L1 )
3782 {
3783 return parseKey( expression.value( 1 ), context );
3784 }
3785 else if ( op == "match"_L1 )
3786 {
3787 const QString attribute = expression.value( 1 ).toList().value( 1 ).toString();
3788
3789 if ( expression.size() == 5
3790 && expression.at( 3 ).userType() == QMetaType::Type::Bool
3791 && expression.at( 3 ).toBool() == true
3792 && expression.at( 4 ).userType() == QMetaType::Type::Bool
3793 && expression.at( 4 ).toBool() == false )
3794 {
3795 // simple case, make a nice simple expression instead of a CASE statement
3796 if ( expression.at( 2 ).userType() == QMetaType::Type::QVariantList || expression.at( 2 ).userType() == QMetaType::Type::QStringList )
3797 {
3798 QStringList parts;
3799 for ( const QVariant &p : expression.at( 2 ).toList() )
3800 {
3801 parts << parseValue( p, context );
3802 }
3803
3804 if ( parts.size() > 1 )
3805 return u"%1 IN (%2)"_s.arg( QgsExpression::quotedColumnRef( attribute ), parts.join( ", " ) );
3806 else
3807 return QgsExpression::createFieldEqualityExpression( attribute, expression.at( 2 ).toList().value( 0 ) );
3808 }
3809 else if ( expression.at( 2 ).userType() == QMetaType::Type::QString
3810 || expression.at( 2 ).userType() == QMetaType::Type::Int
3811 || expression.at( 2 ).userType() == QMetaType::Type::Double
3812 || expression.at( 2 ).userType() == QMetaType::Type::LongLong )
3813 {
3814 return QgsExpression::createFieldEqualityExpression( attribute, expression.at( 2 ) );
3815 }
3816 else
3817 {
3818 context.pushWarning( QObject::tr( "%1: Skipping unsupported expression" ).arg( context.layerId() ) );
3819 return QString();
3820 }
3821 }
3822 else
3823 {
3824 QString caseString = u"CASE "_s;
3825 for ( int i = 2; i < expression.size() - 2; i += 2 )
3826 {
3827 if ( expression.at( i ).userType() == QMetaType::Type::QVariantList || expression.at( i ).userType() == QMetaType::Type::QStringList )
3828 {
3829 QStringList parts;
3830 for ( const QVariant &p : expression.at( i ).toList() )
3831 {
3832 parts << QgsExpression::quotedValue( p );
3833 }
3834
3835 if ( parts.size() > 1 )
3836 caseString += u"WHEN %1 IN (%2) "_s.arg( QgsExpression::quotedColumnRef( attribute ), parts.join( ", " ) );
3837 else
3838 caseString += u"WHEN %1 "_s.arg( QgsExpression::createFieldEqualityExpression( attribute, expression.at( i ).toList().value( 0 ) ) );
3839 }
3840 else if ( expression.at( i ).userType() == QMetaType::Type::QString
3841 || expression.at( i ).userType() == QMetaType::Type::Int
3842 || expression.at( i ).userType() == QMetaType::Type::Double
3843 || expression.at( i ).userType() == QMetaType::Type::LongLong )
3844 {
3845 caseString += u"WHEN (%1) "_s.arg( QgsExpression::createFieldEqualityExpression( attribute, expression.at( i ) ) );
3846 }
3847
3848 caseString += u"THEN %1 "_s.arg( parseValue( expression.at( i + 1 ), context, colorExpected ) );
3849 }
3850 caseString += u"ELSE %1 END"_s.arg( parseValue( expression.last(), context, colorExpected ) );
3851 return caseString;
3852 }
3853 }
3854 else if ( op == "to-string"_L1 )
3855 {
3856 return u"to_string(%1)"_s.arg( parseExpression( expression.value( 1 ).toList(), context ) );
3857 }
3858 else if ( op == "to-boolean"_L1 )
3859 {
3860 return u"to_bool(%1)"_s.arg( parseExpression( expression.value( 1 ).toList(), context ) );
3861 }
3862 else if ( op == "case"_L1 )
3863 {
3864 QString caseString = u"CASE"_s;
3865 for ( int i = 1; i < expression.size() - 2; i += 2 )
3866 {
3867 const QString condition = parseExpression( expression.value( i ).toList(), context );
3868 const QString value = parseValue( expression.value( i + 1 ), context, colorExpected );
3869 caseString += u" WHEN (%1) THEN %2"_s.arg( condition, value );
3870 }
3871 const QString value = parseValue( expression.constLast(), context, colorExpected );
3872 caseString += u" ELSE %1 END"_s.arg( value );
3873 return caseString;
3874 }
3875 else if ( op == "zoom"_L1 && expression.count() == 1 )
3876 {
3877 return u"@vector_tile_zoom"_s;
3878 }
3879 else if ( op == "coalesce"_L1 )
3880 {
3881 QString coalesceString = u"coalesce("_s;
3882 for ( int i = 1; i < expression.size(); i++ )
3883 {
3884 if ( i > 1 )
3885 coalesceString += ", "_L1;
3886 coalesceString += parseValue( expression.value( i ), context );
3887 }
3888 coalesceString += ')'_L1;
3889 return coalesceString;
3890 }
3891 else if ( op == "concat"_L1 )
3892 {
3893 QString concatString = u"concat("_s;
3894 for ( int i = 1; i < expression.size(); i++ )
3895 {
3896 if ( i > 1 )
3897 concatString += ", "_L1;
3898 concatString += parseValue( expression.value( i ), context );
3899 }
3900 concatString += ')'_L1;
3901 return concatString;
3902 }
3903 else if ( op == "length"_L1 )
3904 {
3905 return u"length(%1)"_s.arg( parseExpression( expression.value( 1 ).toList(), context ) );
3906 }
3907 else if ( op == "step"_L1 )
3908 {
3909 const QString stepExpression = parseExpression( expression.value( 1 ).toList(), context );
3910 if ( stepExpression.isEmpty() )
3911 {
3912 context.pushWarning( QObject::tr( "%1: Could not interpret step list" ).arg( context.layerId() ) );
3913 return QString();
3914 }
3915
3916 QString caseString = u"CASE "_s;
3917
3918 for ( int i = expression.length() - 2; i > 0; i -= 2 )
3919 {
3920 const QString stepValue = parseValue( expression.value( i + 1 ), context, colorExpected );
3921 if ( i > 1 )
3922 {
3923 const QString stepKey = QgsExpression::quotedValue( expression.value( i ) );
3924 caseString += u" WHEN %1 >= %2 THEN (%3) "_s.arg( stepExpression, stepKey, stepValue );
3925 }
3926 else
3927 {
3928 caseString += u"ELSE (%1) END"_s.arg( stepValue );
3929 }
3930 }
3931 return caseString;
3932 }
3933 else if ( op == "pitch"_L1 )
3934 {
3935 return u"0"_s;
3936 }
3937 else if ( op == "slice"_L1 )
3938 {
3939 // ["slice", input, startIndex, endIndex?] returns a substring/sublist of input.
3940 // MapBox indices are 0-based and endIndex is exclusive, while QGIS substr() is
3941 // 1-based and takes a length
3942 const QString inputExpression = parseValue( expression.value( 1 ), context );
3943 if ( inputExpression.isEmpty() )
3944 {
3945 context.pushWarning( QObject::tr( "%1: Could not interpret slice list" ).arg( context.layerId() ) );
3946 return QString();
3947 }
3948
3949 // When the indices are constant integers
3950 // we can fold the index arithmetic at conversion time
3951 const auto constantInt = []( const QVariant &value, int &result ) -> bool {
3952 switch ( value.userType() )
3953 {
3954 case QMetaType::Int:
3955 case QMetaType::UInt:
3956 case QMetaType::LongLong:
3957 case QMetaType::ULongLong:
3958 {
3959 bool ok = false;
3960 result = value.toInt( &ok );
3961 return ok;
3962 }
3963 default:
3964 return false;
3965 }
3966 };
3967
3968 int startValue = 0;
3969 const bool startIsConstant = constantInt( expression.value( 2 ), startValue );
3970 const QString startExpression = parseValue( expression.value( 2 ), context );
3971 const QString startOffset = startIsConstant ? QString::number( startValue + 1 ) : u"(%1) + 1"_s.arg( startExpression );
3972
3973 if ( expression.size() > 3 )
3974 {
3975 int endValue = 0;
3976 const bool endIsConstant = constantInt( expression.value( 3 ), endValue );
3977 const QString endExpression = parseValue( expression.value( 3 ), context );
3978 const QString length = ( startIsConstant && endIsConstant ) ? QString::number( endValue - startValue ) : u"(%1) - (%2)"_s.arg( endExpression, startExpression );
3979 return u"substr(%1, %2, %3)"_s.arg( inputExpression, startOffset, length );
3980 }
3981 else
3982 {
3983 return u"substr(%1, %2)"_s.arg( inputExpression, startOffset );
3984 }
3985 }
3986 else
3987 {
3988 context.pushWarning( QObject::tr( "%1: Skipping unsupported expression \"%2\"" ).arg( context.layerId(), op ) );
3989 return QString();
3990 }
3991}
3992
3993QImage QgsMapBoxGlStyleConverter::retrieveSprite( const QString &name, QgsMapBoxGlStyleConversionContext &context, QSize &spriteSize )
3994{
3995 QImage spriteImage;
3996
3997 if ( name.isEmpty() )
3998 {
3999 return QImage();
4000 }
4001
4002 QString category;
4003 QString actualName = name;
4004 const int categorySeparator = name.indexOf( ':' );
4005 if ( categorySeparator > 0 )
4006 {
4007 category = name.left( categorySeparator );
4008 if ( context.spriteCategories().contains( category ) )
4009 {
4010 actualName = name.mid( categorySeparator + 1 );
4011 spriteImage = context.spriteImage( category );
4012 }
4013 else
4014 {
4015 category.clear();
4016 }
4017 }
4018
4019 if ( category.isEmpty() )
4020 {
4021 spriteImage = context.spriteImage();
4022 }
4023
4024 if ( spriteImage.isNull() )
4025 {
4026 context.pushWarning( QObject::tr( "%1: Could not retrieve sprite '%2'" ).arg( context.layerId(), name ) );
4027 return QImage();
4028 }
4029
4030 const QVariantMap spriteDefinition = context.spriteDefinitions( category ).value( actualName ).toMap();
4031 if ( spriteDefinition.size() == 0 )
4032 {
4033 context.pushWarning( QObject::tr( "%1: Could not retrieve sprite '%2'" ).arg( context.layerId(), name ) );
4034 return QImage();
4035 }
4036
4037 const QImage sprite
4038 = spriteImage.copy( spriteDefinition.value( u"x"_s ).toInt(), spriteDefinition.value( u"y"_s ).toInt(), spriteDefinition.value( u"width"_s ).toInt(), spriteDefinition.value( u"height"_s ).toInt() );
4039 if ( sprite.isNull() )
4040 {
4041 context.pushWarning( QObject::tr( "%1: Could not retrieve sprite '%2'" ).arg( context.layerId(), name ) );
4042 return QImage();
4043 }
4044
4045 spriteSize = sprite.size() / spriteDefinition.value( u"pixelRatio"_s ).toDouble() * context.pixelSizeConversionFactor();
4046 return sprite;
4047}
4048
4050 const QVariant &value, QgsMapBoxGlStyleConversionContext &context, QSize &spriteSize, QString &spriteProperty, QString &spriteSizeProperty
4051)
4052{
4053 QString spritePath;
4054
4055 auto prepareBase64 = []( const QImage &sprite ) {
4056 QString path;
4057 if ( !sprite.isNull() )
4058 {
4059 QByteArray blob;
4060 QBuffer buffer( &blob );
4061 buffer.open( QIODevice::WriteOnly );
4062 sprite.save( &buffer, "PNG" );
4063 buffer.close();
4064 const QByteArray encoded = blob.toBase64();
4065 path = QString( encoded );
4066 path.prepend( "base64:"_L1 );
4067 }
4068 return path;
4069 };
4070
4071 switch ( value.userType() )
4072 {
4073 case QMetaType::Type::QString:
4074 {
4075 QString spriteName = value.toString();
4076 const thread_local QRegularExpression fieldNameMatch( u"{([^}]+)}"_s );
4077 QRegularExpressionMatch match = fieldNameMatch.match( spriteName );
4078 if ( match.hasMatch() )
4079 {
4080 const QString fieldName = match.captured( 1 );
4081 spriteProperty = u"CASE"_s;
4082 spriteSizeProperty = u"CASE"_s;
4083
4084 spriteName.replace( "(", "\\("_L1 );
4085 spriteName.replace( ")", "\\)"_L1 );
4086 spriteName.replace( fieldNameMatch, u"([^\\/\\\\]+)"_s );
4087 const QRegularExpression fieldValueMatch( spriteName );
4088 const QStringList spriteNames = context.spriteDefinitions().keys();
4089 for ( const QString &name : spriteNames )
4090 {
4091 match = fieldValueMatch.match( name );
4092 if ( match.hasMatch() )
4093 {
4094 QSize size;
4095 QString path;
4096 const QString fieldValue = match.captured( 1 );
4097 const QImage sprite = retrieveSprite( name, context, size );
4098 path = prepareBase64( sprite );
4099 if ( spritePath.isEmpty() && !path.isEmpty() )
4100 {
4101 spritePath = path;
4102 spriteSize = size;
4103 }
4104
4105 spriteProperty += u" WHEN \"%1\" = '%2' THEN '%3'"_s.arg( fieldName, fieldValue, path );
4106 spriteSizeProperty += u" WHEN \"%1\" = '%2' THEN %3"_s.arg( fieldName ).arg( fieldValue ).arg( size.width() );
4107 }
4108 }
4109
4110 spriteProperty += " END"_L1;
4111 spriteSizeProperty += " END"_L1;
4112 }
4113 else
4114 {
4115 spriteProperty.clear();
4116 spriteSizeProperty.clear();
4117 const QImage sprite = retrieveSprite( spriteName, context, spriteSize );
4118 spritePath = prepareBase64( sprite );
4119 }
4120 break;
4121 }
4122
4123 case QMetaType::Type::QVariantMap:
4124 {
4125 const QVariantList stops = value.toMap().value( u"stops"_s ).toList();
4126 if ( stops.size() == 0 )
4127 break;
4128
4129 QString path;
4130 QSize size;
4131 QImage sprite;
4132
4133 sprite = retrieveSprite( stops.value( 0 ).toList().value( 1 ).toString(), context, spriteSize );
4134 spritePath = prepareBase64( sprite );
4135
4136 spriteProperty = u"CASE WHEN @vector_tile_zoom < %1 THEN '%2'"_s.arg( stops.value( 0 ).toList().value( 0 ).toString() ).arg( spritePath );
4137 spriteSizeProperty = u"CASE WHEN @vector_tile_zoom < %1 THEN %2"_s.arg( stops.value( 0 ).toList().value( 0 ).toString() ).arg( spriteSize.width() );
4138
4139 for ( int i = 0; i < stops.size() - 1; ++i )
4140 {
4141 ;
4142 sprite = retrieveSprite( stops.value( 0 ).toList().value( 1 ).toString(), context, size );
4143 path = prepareBase64( sprite );
4144
4145 spriteProperty += QStringLiteral(
4146 " WHEN @vector_tile_zoom >= %1 AND @vector_tile_zoom < %2 "
4147 "THEN '%3'"
4148 )
4149 .arg( stops.value( i ).toList().value( 0 ).toString(), stops.value( i + 1 ).toList().value( 0 ).toString(), path );
4150 spriteSizeProperty += QStringLiteral(
4151 " WHEN @vector_tile_zoom >= %1 AND @vector_tile_zoom < %2 "
4152 "THEN %3"
4153 )
4154 .arg( stops.value( i ).toList().value( 0 ).toString(), stops.value( i + 1 ).toList().value( 0 ).toString() )
4155 .arg( size.width() );
4156 }
4157 sprite = retrieveSprite( stops.last().toList().value( 1 ).toString(), context, size );
4158 path = prepareBase64( sprite );
4159
4160 spriteProperty += QStringLiteral(
4161 " WHEN @vector_tile_zoom >= %1 "
4162 "THEN '%2' END"
4163 )
4164 .arg( stops.last().toList().value( 0 ).toString() )
4165 .arg( path );
4166 spriteSizeProperty += QStringLiteral(
4167 " WHEN @vector_tile_zoom >= %1 "
4168 "THEN %2 END"
4169 )
4170 .arg( stops.last().toList().value( 0 ).toString() )
4171 .arg( size.width() );
4172 break;
4173 }
4174
4175 case QMetaType::Type::QVariantList:
4176 {
4177 const QVariantList json = value.toList();
4178 const QString method = json.value( 0 ).toString();
4179
4180 if ( method == "match"_L1 )
4181 {
4182 const QString attribute = parseExpression( json.value( 1 ).toList(), context );
4183 if ( attribute.isEmpty() )
4184 {
4185 context.pushWarning( QObject::tr( "%1: Could not interpret match list" ).arg( context.layerId() ) );
4186 break;
4187 }
4188
4189 spriteProperty = u"CASE"_s;
4190 spriteSizeProperty = u"CASE"_s;
4191
4192 for ( int i = 2; i < json.length() - 1; i += 2 )
4193 {
4194 const QVariant matchKey = json.value( i );
4195 const QVariant matchValue = json.value( i + 1 );
4196 QString matchString;
4197 switch ( matchKey.userType() )
4198 {
4199 case QMetaType::Type::QVariantList:
4200 case QMetaType::Type::QStringList:
4201 {
4202 const QVariantList keys = matchKey.toList();
4203 QStringList matchStringList;
4204 for ( const QVariant &key : keys )
4205 {
4206 matchStringList << QgsExpression::quotedValue( key );
4207 }
4208 matchString = matchStringList.join( ',' );
4209 break;
4210 }
4211
4212 case QMetaType::Type::Bool:
4213 case QMetaType::Type::QString:
4214 case QMetaType::Type::Int:
4215 case QMetaType::Type::LongLong:
4216 case QMetaType::Type::Double:
4217 {
4218 matchString = QgsExpression::quotedValue( matchKey );
4219 break;
4220 }
4221
4222 default:
4223 context.pushWarning( QObject::tr( "%1: Skipping unsupported sprite type (%2)." ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( value.userType() ) ) ) );
4224 break;
4225 }
4226
4227 const QImage sprite = retrieveSprite( matchValue.toString(), context, spriteSize );
4228 spritePath = prepareBase64( sprite );
4229
4230 spriteProperty += QStringLiteral(
4231 " WHEN %1 IN (%2) "
4232 "THEN '%3'"
4233 )
4234 .arg( attribute, matchString, spritePath );
4235
4236 spriteSizeProperty += QStringLiteral(
4237 " WHEN %1 IN (%2) "
4238 "THEN %3"
4239 )
4240 .arg( attribute, matchString )
4241 .arg( spriteSize.width() );
4242 }
4243
4244 if ( !json.constLast().toString().isEmpty() )
4245 {
4246 const QImage sprite = retrieveSprite( json.constLast().toString(), context, spriteSize );
4247 spritePath = prepareBase64( sprite );
4248 }
4249 else
4250 {
4251 spritePath = QString();
4252 }
4253
4254 spriteProperty += u" ELSE '%1' END"_s.arg( spritePath );
4255 spriteSizeProperty += u" ELSE %3 END"_s.arg( spriteSize.width() );
4256 break;
4257 }
4258 else if ( method == "step"_L1 )
4259 {
4260 const QString expression = parseExpression( json.value( 1 ).toList(), context );
4261 if ( expression.isEmpty() )
4262 {
4263 context.pushWarning( QObject::tr( "%1: Could not interpret step list" ).arg( context.layerId() ) );
4264 break;
4265 }
4266
4267 spriteProperty = u"CASE"_s;
4268 spriteSizeProperty = u"CASE"_s;
4269 for ( int i = json.length() - 2; i > 2; i -= 2 )
4270 {
4271 const QString stepKey = QgsExpression::quotedValue( json.value( i ) );
4272 const QString stepValue = json.value( i + 1 ).toString();
4273
4274 const QImage sprite = retrieveSprite( stepValue, context, spriteSize );
4275 spritePath = prepareBase64( sprite );
4276
4277 spriteProperty += u" WHEN %1 >= %2 THEN '%3' "_s.arg( expression, stepKey, spritePath );
4278 spriteSizeProperty += u" WHEN %1 >= %2 THEN %3 "_s.arg( expression ).arg( stepKey ).arg( spriteSize.width() );
4279 }
4280
4281 const QImage sprite = retrieveSprite( json.at( 2 ).toString(), context, spriteSize );
4282 spritePath = prepareBase64( sprite );
4283
4284 spriteProperty += u"ELSE '%1' END"_s.arg( spritePath );
4285 spriteSizeProperty += u"ELSE %3 END"_s.arg( spriteSize.width() );
4286 break;
4287 }
4288 else if ( method == "case"_L1 )
4289 {
4290 spriteProperty = u"CASE"_s;
4291 spriteSizeProperty = u"CASE"_s;
4292 for ( int i = 1; i < json.length() - 2; i += 2 )
4293 {
4294 const QString caseExpression = parseExpression( json.value( i ).toList(), context );
4295 const QString caseValue = json.value( i + 1 ).toString();
4296
4297 const QImage sprite = retrieveSprite( caseValue, context, spriteSize );
4298 spritePath = prepareBase64( sprite );
4299
4300 spriteProperty += u" WHEN %1 THEN '%2' "_s.arg( caseExpression, spritePath );
4301 spriteSizeProperty += u" WHEN %1 THEN %2 "_s.arg( caseExpression ).arg( spriteSize.width() );
4302 }
4303 const QImage sprite = retrieveSprite( json.last().toString(), context, spriteSize );
4304 spritePath = prepareBase64( sprite );
4305
4306 spriteProperty += u"ELSE '%1' END"_s.arg( spritePath );
4307 spriteSizeProperty += u"ELSE %3 END"_s.arg( spriteSize.width() );
4308 break;
4309 }
4310 else
4311 {
4312 context.pushWarning( QObject::tr( "%1: Could not interpret sprite value list with method %2" ).arg( context.layerId(), method ) );
4313 break;
4314 }
4315 }
4316
4317 default:
4318 context.pushWarning( QObject::tr( "%1: Skipping unsupported sprite type (%2)." ).arg( context.layerId(), QMetaType::typeName( static_cast<QMetaType::Type>( value.userType() ) ) ) );
4319 break;
4320 }
4321
4322 return spritePath;
4323}
4324
4325QString QgsMapBoxGlStyleConverter::parseValue( const QVariant &value, QgsMapBoxGlStyleConversionContext &context, bool colorExpected )
4326{
4327 QColor c;
4328 switch ( value.userType() )
4329 {
4330 case QMetaType::Type::QVariantList:
4331 case QMetaType::Type::QStringList:
4332 return parseExpression( value.toList(), context, colorExpected );
4333
4334 case QMetaType::Type::Bool:
4335 case QMetaType::Type::QString:
4336 if ( colorExpected )
4337 {
4338 QColor c = parseColor( value, context );
4339 if ( c.isValid() )
4340 {
4341 return parseValue( c, context );
4342 }
4343 }
4344 return QgsExpression::quotedValue( value );
4345
4346 case QMetaType::Type::Int:
4347 case QMetaType::Type::LongLong:
4348 case QMetaType::Type::Double:
4349 return value.toString();
4350
4351 case QMetaType::Type::QColor:
4352 c = value.value<QColor>();
4353 return QString( "color_rgba(%1,%2,%3,%4)" ).arg( c.red() ).arg( c.green() ).arg( c.blue() ).arg( c.alpha() );
4354
4355 default:
4356 context.pushWarning( QObject::tr( "%1: Skipping unsupported expression part" ).arg( context.layerId() ) );
4357 break;
4358 }
4359 return QString();
4360}
4361
4362QString QgsMapBoxGlStyleConverter::parseKey( const QVariant &value, QgsMapBoxGlStyleConversionContext &context )
4363{
4364 if ( value.toString() == "$type"_L1 )
4365 {
4366 return u"_geom_type"_s;
4367 }
4368 if ( value.toString() == "level"_L1 )
4369 {
4370 return u"level"_s;
4371 }
4372 else if ( ( value.userType() == QMetaType::Type::QVariantList && value.toList().size() == 1 ) || value.userType() == QMetaType::Type::QStringList )
4373 {
4374 if ( value.toList().size() > 1 )
4375 return value.toList().at( 1 ).toString();
4376 else
4377 {
4378 QString valueString = value.toList().value( 0 ).toString();
4379 if ( valueString == "geometry-type"_L1 )
4380 {
4381 return u"_geom_type"_s;
4382 }
4383 return valueString;
4384 }
4385 }
4386 else if ( value.userType() == QMetaType::Type::QVariantList && value.toList().size() > 1 )
4387 {
4388 return parseExpression( value.toList(), context );
4389 }
4390 return QgsExpression::quotedColumnRef( value.toString() );
4391}
4392
4393QString QgsMapBoxGlStyleConverter::processLabelField( const QString &string, bool &isExpression )
4394{
4395 // {field_name} is permitted in string -- if multiple fields are present, convert them to an expression
4396 // but if single field is covered in {}, return it directly
4397 const thread_local QRegularExpression singleFieldRx( u"^{([^}]+)}$"_s );
4398 const QRegularExpressionMatch match = singleFieldRx.match( string );
4399 if ( match.hasMatch() )
4400 {
4401 isExpression = false;
4402 return match.captured( 1 );
4403 }
4404
4405 const thread_local QRegularExpression multiFieldRx( u"(?={[^}]+})"_s );
4406 const QStringList parts = string.split( multiFieldRx );
4407 if ( parts.size() > 1 )
4408 {
4409 isExpression = true;
4410
4411 QStringList res;
4412 for ( const QString &part : parts )
4413 {
4414 if ( part.isEmpty() )
4415 continue;
4416
4417 if ( !part.contains( '{' ) )
4418 {
4419 res << QgsExpression::quotedValue( part );
4420 continue;
4421 }
4422
4423 // part will start at a {field} reference
4424 const QStringList split = part.split( '}' );
4425 res << QgsExpression::quotedColumnRef( split.at( 0 ).mid( 1 ) );
4426 if ( !split.at( 1 ).isEmpty() )
4427 res << QgsExpression::quotedValue( split.at( 1 ) );
4428 }
4429 return u"concat(%1)"_s.arg( res.join( ',' ) );
4430 }
4431 else
4432 {
4433 isExpression = false;
4434 return string;
4435 }
4436}
4437
4439{
4440 return mRenderer ? mRenderer->clone() : nullptr;
4441}
4442
4444{
4445 return mLabeling ? mLabeling->clone() : nullptr;
4446}
4447
4448QList<QgsMapBoxGlStyleAbstractSource *> QgsMapBoxGlStyleConverter::sources()
4449{
4450 return mSources;
4451}
4452
4453QList<QgsMapBoxGlStyleRasterSubLayer> QgsMapBoxGlStyleConverter::rasterSubLayers() const
4454{
4455 return mRasterSubLayers;
4456}
4457
4459{
4460 QList<QgsMapLayer *> subLayers;
4461 for ( const QgsMapBoxGlStyleRasterSubLayer &subLayer : mRasterSubLayers )
4462 {
4463 const QString sourceName = subLayer.source();
4464 std::unique_ptr< QgsRasterLayer > rl;
4465 for ( const QgsMapBoxGlStyleAbstractSource *source : mSources )
4466 {
4467 if ( source->type() == Qgis::MapBoxGlStyleSourceType::Raster && source->name() == sourceName )
4468 {
4469 const QgsMapBoxGlStyleRasterSource *rasterSource = qgis::down_cast< const QgsMapBoxGlStyleRasterSource * >( source );
4470 rl.reset( rasterSource->toRasterLayer() );
4471 rl->pipe()->setDataDefinedProperties( subLayer.dataDefinedProperties() );
4472 break;
4473 }
4474 }
4475
4476 if ( rl )
4477 {
4478 subLayers.append( rl.release() );
4479 }
4480 }
4481 return subLayers;
4482}
4483
4484
4486{
4487 std::unique_ptr< QgsMapBoxGlStyleConversionContext > tmpContext;
4488 if ( !context )
4489 {
4490 tmpContext = std::make_unique< QgsMapBoxGlStyleConversionContext >();
4491 context = tmpContext.get();
4492 }
4493
4494 auto typeFromString = [context]( const QString &string, const QString &name ) -> Qgis::MapBoxGlStyleSourceType {
4495 if ( string.compare( "vector"_L1, Qt::CaseInsensitive ) == 0 )
4497 else if ( string.compare( "raster"_L1, Qt::CaseInsensitive ) == 0 )
4499 else if ( string.compare( "raster-dem"_L1, Qt::CaseInsensitive ) == 0 )
4501 else if ( string.compare( "geojson"_L1, Qt::CaseInsensitive ) == 0 )
4503 else if ( string.compare( "image"_L1, Qt::CaseInsensitive ) == 0 )
4505 else if ( string.compare( "video"_L1, Qt::CaseInsensitive ) == 0 )
4507 context->pushWarning( QObject::tr( "Invalid source type \"%1\" for source \"%2\"" ).arg( string, name ) );
4509 };
4510
4511 for ( auto it = sources.begin(); it != sources.end(); ++it )
4512 {
4513 const QString name = it.key();
4514 const QVariantMap jsonSource = it.value().toMap();
4515 const QString typeString = jsonSource.value( u"type"_s ).toString();
4516
4517 const Qgis::MapBoxGlStyleSourceType type = typeFromString( typeString, name );
4518
4519 switch ( type )
4520 {
4522 parseRasterSource( jsonSource, name, context );
4523 break;
4530 QgsDebugError( u"Ignoring vector tile style source %1 (%2)"_s.arg( name, qgsEnumValueToKey( type ) ) );
4531 continue;
4532 }
4533 }
4534}
4535
4536void QgsMapBoxGlStyleConverter::parseRasterSource( const QVariantMap &source, const QString &name, QgsMapBoxGlStyleConversionContext *context )
4537{
4538 std::unique_ptr< QgsMapBoxGlStyleConversionContext > tmpContext;
4539 if ( !context )
4540 {
4541 tmpContext = std::make_unique< QgsMapBoxGlStyleConversionContext >();
4542 context = tmpContext.get();
4543 }
4544
4545 auto raster = std::make_unique< QgsMapBoxGlStyleRasterSource >( name );
4546 if ( raster->setFromJson( source, context ) )
4547 mSources.append( raster.release() );
4548}
4549
4550bool QgsMapBoxGlStyleConverter::numericArgumentsOnly( const QVariant &bottomVariant, const QVariant &topVariant, double &bottom, double &top )
4551{
4552 if ( bottomVariant.canConvert( QMetaType::Double ) && topVariant.canConvert( QMetaType::Double ) )
4553 {
4554 bool bDoubleOk, tDoubleOk;
4555 bottom = bottomVariant.toDouble( &bDoubleOk );
4556 top = topVariant.toDouble( &tDoubleOk );
4557 return ( bDoubleOk && tDoubleOk );
4558 }
4559 return false;
4560}
4561
4562//
4563// QgsMapBoxGlStyleConversionContext
4564//
4566{
4567 QgsDebugError( warning );
4568 mWarnings << warning;
4569}
4570
4572{
4573 return mTargetUnit;
4574}
4575
4580
4582{
4583 return mSizeConversionFactor;
4584}
4585
4587{
4588 mSizeConversionFactor = sizeConversionFactor;
4589}
4590
4592{
4593 return mSpriteImage.keys();
4594}
4595
4596QImage QgsMapBoxGlStyleConversionContext::spriteImage( const QString &category ) const
4597{
4598 return mSpriteImage.contains( category ) ? mSpriteImage[category] : QImage();
4599}
4600
4601QVariantMap QgsMapBoxGlStyleConversionContext::spriteDefinitions( const QString &category ) const
4602{
4603 return mSpriteDefinitions.contains( category ) ? mSpriteDefinitions[category] : QVariantMap();
4604}
4605
4606void QgsMapBoxGlStyleConversionContext::setSprites( const QImage &image, const QVariantMap &definitions, const QString &category )
4607{
4608 mSpriteImage[category] = image;
4609 mSpriteDefinitions[category] = definitions;
4610}
4611
4612void QgsMapBoxGlStyleConversionContext::setSprites( const QImage &image, const QString &definitions, const QString &category )
4613{
4614 setSprites( image, QgsJsonUtils::parseJson( definitions ).toMap(), category );
4615}
4616
4618{
4619 return mLayerId;
4620}
4621
4623{
4624 mLayerId = value;
4625}
4626
4627//
4628// QgsMapBoxGlStyleAbstractSource
4629//
4633
4635{
4636 return mName;
4637}
4638
4640
4641//
4642// QgsMapBoxGlStyleRasterSource
4643//
4644
4648
4653
4655{
4656 mAttribution = json.value( u"attribution"_s ).toString();
4657
4658 const QString scheme = json.value( u"scheme"_s, u"xyz"_s ).toString();
4659 if ( scheme.compare( "xyz"_L1 ) == 0 )
4660 {
4661 // xyz scheme is supported
4662 }
4663 else
4664 {
4665 context->pushWarning( QObject::tr( "%1 scheme is not supported for raster source %2" ).arg( scheme, name() ) );
4666 return false;
4667 }
4668
4669 mMinZoom = json.value( u"minzoom"_s, u"0"_s ).toInt();
4670 mMaxZoom = json.value( u"maxzoom"_s, u"22"_s ).toInt();
4671 mTileSize = json.value( u"tileSize"_s, u"512"_s ).toInt();
4672
4673 const QVariantList tiles = json.value( u"tiles"_s ).toList();
4674 for ( const QVariant &tile : tiles )
4675 {
4676 mTiles.append( tile.toString() );
4677 }
4678
4679 return true;
4680}
4681
4683{
4684 QVariantMap parts;
4685 parts.insert( u"type"_s, u"xyz"_s );
4686 parts.insert( u"url"_s, mTiles.value( 0 ) );
4687
4688 if ( mTileSize == 256 )
4689 parts.insert( u"tilePixelRation"_s, u"1"_s );
4690 else if ( mTileSize == 512 )
4691 parts.insert( u"tilePixelRation"_s, u"2"_s );
4692
4693 parts.insert( u"zmax"_s, QString::number( mMaxZoom ) );
4694 parts.insert( u"zmin"_s, QString::number( mMinZoom ) );
4695
4696 auto rl = std::make_unique< QgsRasterLayer >( QgsProviderRegistry::instance()->encodeUri( u"wms"_s, parts ), name(), u"wms"_s );
4697 return rl.release();
4698}
4699
4700//
4701// QgsMapBoxGlStyleRasterSubLayer
4702//
4704 : mId( id )
4705 , mSource( source )
4706{}
@ BelowLine
Labels can be placed below a line feature. Unless MapOrientation is also specified this mode respects...
Definition qgis.h:1385
@ 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
@ CentralPoint
Place symbols at the mid point of the line.
Definition qgis.h:3316
@ OverPoint
Arranges candidates over a point (or centroid of a polygon), or at a preset offset from the point....
Definition qgis.h:1276
@ Curved
Arranges candidates following the curvature of a line feature. Applies to line layers only.
Definition qgis.h:1278
@ Horizontal
Arranges horizontal candidates scattered throughout a polygon feature. Applies to polygon layers only...
Definition qgis.h:1279
@ 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
GeometryType
The geometry types are used to group Qgis::WkbType in a coarse way.
Definition qgis.h:379
@ Point
Points.
Definition qgis.h:380
@ Line
Lines.
Definition qgis.h:381
@ Polygon
Polygons.
Definition qgis.h:382
@ Center
Center align.
Definition qgis.h:1447
@ FollowPlacement
Alignment follows placement of label, e.g., labels to the left of a feature will be drawn with right ...
Definition qgis.h:1449
RenderUnit
Rendering size units.
Definition qgis.h:5607
MapBoxGlStyleSourceType
Available MapBox GL style source types.
Definition qgis.h:4722
@ Vector
Vector source.
Definition qgis.h:4723
@ RasterDem
Raster DEM source.
Definition qgis.h:4725
@ Raster
Raster source.
Definition qgis.h:4724
@ Unknown
Other/unknown source type.
Definition qgis.h:4729
@ GeoJson
GeoJSON source.
Definition qgis.h:4726
@ Viewport
Relative to the whole viewport/output device.
Definition qgis.h:3391
@ AllowOverlapAtNoCost
Labels may freely overlap other labels, at no cost.
Definition qgis.h:1238
void setPenJoinStyle(Qt::PenJoinStyle style)
Sets the pen join style used to render the line (e.g.
void setPenCapStyle(Qt::PenCapStyle style)
Sets the pen cap style used to render the line (e.g.
static QgsFontManager * fontManager()
Returns the application font manager, which manages available fonts and font installation for the QGI...
A paint effect which blurs a source picture, using a number of different blur methods.
void setBlurUnit(const Qgis::RenderUnit unit)
Sets the units used for the blur level (radius).
@ StackBlur
Stack blur, a fast but low quality blur. Valid blur level values are between 0 - 16.
void setBlurMethod(const BlurMethod method)
Sets the blur method (algorithm) to use for performing the blur.
void setBlurLevel(const double level)
Sets blur level (radius).
A paint effect which consists of a stack of other chained paint effects.
void appendEffect(QgsPaintEffect *effect)
Appends an effect to the end of the stack.
static QString quotedValue(const QVariant &value)
Returns a string representation of a literal value, including appropriate quotations where required.
static QString quotedString(QString text)
Returns a quoted version of a string (in single quotes).
static QString createFieldEqualityExpression(const QString &fieldName, const QVariant &value, QMetaType::Type fieldType=QMetaType::Type::UnknownType)
Create an expression allowing to evaluate if a field is equal to a value.
static QString quotedColumnRef(QString name)
Returns a quoted column reference (in double quotes).
QString processFontFamilyName(const QString &name) const
Processes a font family name, applying any matching fontFamilyReplacements() to the name.
static QFont createFont(const QString &family, int pointSize=-1, int weight=-1, bool italic=false)
Creates a font with the specified family.
static bool fontFamilyHasStyle(const QString &family, const QString &style)
Check whether font family on system has specific style.
static QVariant parseJson(const std::string &jsonString)
Converts JSON jsonString to a QVariant, in case of parsing error an invalid QVariant is returned and ...
void setPlacementFlags(Qgis::LabelLinePlacementFlags flags)
Returns the line placement flags, which dictate how line labels can be placed above or below the line...
void setFactor(double factor)
Sets the obstacle factor, where 1.0 = default, < 1.0 more likely to be covered by labels,...
Contains general settings related to how labels are placed.
void setOverlapHandling(Qgis::LabelOverlapHandling handling)
Sets the technique used to handle overlapping labels.
void setAllowDegradedPlacement(bool allow)
Sets whether labels can be placed in inferior fallback positions if they cannot otherwise be placed.
void setQuadrant(Qgis::LabelQuadrantPosition quadrant)
Sets the quadrant in which to offset labels from the point.
Contains settings related to how the label engine removes candidate label positions and reduces the n...
void setAllowDuplicateRemoval(bool allow)
Sets whether duplicate label removal is permitted for this layer.
void setMinimumDistanceToDuplicateUnit(Qgis::RenderUnit unit)
Sets the unit for the minimum distance to labels with duplicate text.
virtual void setWidth(double width)
Sets the width of the line symbol layer.
void setOffset(double offset)
Sets the line's offset.
void setOffsetUnit(Qgis::RenderUnit unit)
Sets the unit for the line's offset.
Abstract base class for MapBox GL style sources.
QString name() const
Returns the source's name.
QgsMapBoxGlStyleAbstractSource(const QString &name)
Constructor for QgsMapBoxGlStyleAbstractSource.
Context for a MapBox GL style conversion operation.
void setLayerId(const QString &value)
Sets the layer ID of the layer currently being converted.
QStringList warnings() const
Returns a list of warning messages generated during the conversion.
void pushWarning(const QString &warning)
Pushes a warning message generated during the conversion.
QImage spriteImage(const QString &category=QString()) const
Returns the sprite image for a given category to use during conversion, or an invalid image if this i...
double pixelSizeConversionFactor() const
Returns the pixel size conversion factor, used to scale the original pixel sizes when converting styl...
void setTargetUnit(Qgis::RenderUnit targetUnit)
Sets the target unit type.
void setSprites(const QImage &image, const QVariantMap &definitions, const QString &category=QString())
Sets the sprite image and definitions JSON for a given category to use during conversion.
void setPixelSizeConversionFactor(double sizeConversionFactor)
Sets the pixel size conversion factor, used to scale the original pixel sizes when converting styles.
QVariantMap spriteDefinitions(const QString &category=QString()) const
Returns the sprite definitions for a given category to use during conversion.
Qgis::RenderUnit targetUnit() const
Returns the target unit type.
QString layerId() const
Returns the layer ID of the layer currently being converted.
QStringList spriteCategories() const
Returns the list of sprite categories to use during conversion, or an empty list of none is set.
void clearWarnings()
Clears the list of warning messages.
static QgsProperty parseInterpolateByZoom(const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, double *defaultNumber=nullptr, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential)
Parses a numeric value which is interpolated by zoom range.
static QString parseColorExpression(const QVariant &colorExpression, QgsMapBoxGlStyleConversionContext &context)
Converts an expression representing a color to a string (can be color string or an expression where a...
QgsVectorTileRenderer * renderer() const
Returns a new instance of a vector tile renderer representing the converted style,...
static QString parseExpression(const QVariantList &expression, QgsMapBoxGlStyleConversionContext &context, bool colorExpected=false)
Converts a MapBox GL expression to a QGIS expression.
InterpolationType
Interpolation types, for interpolated value conversion.
static QString parsePointStops(double base, const QVariantList &stops, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential, double x1=0, double y1=0, double x2=0, double y2=0)
Takes values from stops and uses either scale_linear() or scale_exp() functions to interpolate point/...
PropertyType
Property types, for interpolated value conversion.
@ DashArray
Dash array. Like numeric array, but must be even length array. Odd length arrays are considered as ha...
@ Numeric
Numeric property (e.g. line width, text size).
@ NumericArray
Numeric array for dash arrays or such.
QList< QgsMapBoxGlStyleAbstractSource * > sources()
Returns the list of converted sources.
QgsVectorTileLabeling * labeling() const
Returns a new instance of a vector tile labeling representing the converted style,...
QList< QgsMapBoxGlStyleRasterSubLayer > rasterSubLayers() const
Returns a list of raster sub layers contained in the style.
static Qt::PenJoinStyle parseJoinStyle(const QString &style)
Converts a value to Qt::PenJoinStyle enum from JSON value.
static QString interpolateExpression(double zoomMin, double zoomMax, QVariant valueMin, QVariant valueMax, double base, double multiplier=1, double x1=0, double y1=0, double x2=1, double y2=1, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential, QgsMapBoxGlStyleConversionContext *contextPtr=nullptr)
Generates an interpolation for values between valueMin and valueMax, scaled between the ranges zoomMi...
static QgsProperty parseInterpolateStringByZoom(const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, const QVariantMap &conversionMap, QString *defaultString=nullptr)
Interpolates a string by zoom.
static QgsProperty parseStepList(const QVariantList &json, PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, int maxOpacity=255, QColor *defaultColor=nullptr, double *defaultNumber=nullptr)
Parses and converts a match function value list.
static bool parseCircleLayer(const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &style, QgsMapBoxGlStyleConversionContext &context)
Parses a circle layer.
Result convert(const QVariantMap &style, QgsMapBoxGlStyleConversionContext *context=nullptr)
Converts a JSON style map, and returns the resultant status of the conversion.
static QgsProperty parseInterpolateListByZoom(const QVariantList &json, PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, int maxOpacity=255, QColor *defaultColor=nullptr, double *defaultNumber=nullptr)
Interpolates a list which starts with the interpolate function.
QList< QgsMapLayer * > createSubLayers() const
Returns a list of new map layers corresponding to sublayers of the style, e.g.
@ Success
Conversion was successful.
@ NoLayerList
No layer list was found in JSON input.
QgsMapBoxGlStyleConverter()
Constructor for QgsMapBoxGlStyleConverter.
static QgsProperty parseInterpolateColorByZoom(const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, QColor *defaultColor=nullptr, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential)
Parses a color value which is interpolated by zoom range.
static QImage retrieveSprite(const QString &name, QgsMapBoxGlStyleConversionContext &context, QSize &spriteSize)
Retrieves the sprite image with the specified name, taken from the specified context.
static QString parseLabelStops(const QVariantList &stops, QgsMapBoxGlStyleConversionContext &context)
Parses a list of interpolation stops containing label values.
void parseLayers(const QVariantList &layers, QgsMapBoxGlStyleConversionContext *context=nullptr)
Parse list of layers from JSON.
static QgsProperty parseInterpolateOpacityByZoom(const QVariantMap &json, int maxOpacity, QgsMapBoxGlStyleConversionContext *contextPtr=nullptr, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential)
Interpolates opacity with either scale_linear() or scale_exp() (depending on base value).
static QString retrieveSpriteAsBase64WithProperties(const QVariant &value, QgsMapBoxGlStyleConversionContext &context, QSize &spriteSize, QString &spriteProperty, QString &spriteSizeProperty)
Retrieves the sprite image with the specified name, taken from the specified context as a base64 enco...
void parseSources(const QVariantMap &sources, QgsMapBoxGlStyleConversionContext *context=nullptr)
Parse list of sources from JSON.
static QColor parseColor(const QVariant &color, QgsMapBoxGlStyleConversionContext &context)
Parses a color in one of these supported formats:
static bool parseSymbolLayerAsRenderer(const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &rendererStyle, QgsMapBoxGlStyleConversionContext &context)
Parses a symbol layer as a renderer.
static bool parseFillLayer(const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &style, QgsMapBoxGlStyleConversionContext &context, bool isBackgroundStyle=false)
Parses a fill layer.
static void parseSymbolLayer(const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &rendererStyle, bool &hasRenderer, QgsVectorTileBasicLabelingStyle &labelingStyle, bool &hasLabeling, QgsMapBoxGlStyleConversionContext &context)
Parses a symbol layer as renderer or labeling.
static QString parseOpacityStops(double base, const QVariantList &stops, int maxOpacity, QgsMapBoxGlStyleConversionContext &context, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential, double x1=0, double y1=0, double x2=1, double y2=1)
Takes values from stops and uses either scale_linear() or scale_exp() functions to interpolate alpha ...
static bool parseLineLayer(const QVariantMap &jsonLayer, QgsVectorTileBasicRendererStyle &style, QgsMapBoxGlStyleConversionContext &context)
Parses a line layer.
void parseRasterSource(const QVariantMap &source, const QString &name, QgsMapBoxGlStyleConversionContext *context=nullptr)
Parse a raster source from JSON.
static void colorAsHslaComponents(const QColor &color, int &hue, int &saturation, int &lightness, int &alpha)
Takes a QColor object and returns HSLA components in required format for QGIS color_hsla() expression...
static Qt::PenCapStyle parseCapStyle(const QString &style)
Converts a value to Qt::PenCapStyle enum from JSON value.
static QString parseStringStops(const QVariantList &stops, QgsMapBoxGlStyleConversionContext &context, const QVariantMap &conversionMap, QString *defaultString=nullptr)
Parses a list of interpolation stops containing string values.
static QgsProperty parseMatchList(const QVariantList &json, PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, int maxOpacity=255, QColor *defaultColor=nullptr, double *defaultNumber=nullptr)
Parses and converts a match function value list.
static QString parseStops(double base, const QVariantList &stops, double multiplier, QgsMapBoxGlStyleConversionContext &context, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential, double x1=0, double y1=0, double x2=1, double y2=1)
Parses a list of interpolation stops.
static QgsProperty parseValueList(const QVariantList &json, PropertyType type, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, int maxOpacity=255, QColor *defaultColor=nullptr, double *defaultNumber=nullptr)
Parses and converts a value list (e.g.
static QgsProperty parseInterpolatePointByZoom(const QVariantMap &json, QgsMapBoxGlStyleConversionContext &context, double multiplier=1, QPointF *defaultPoint=nullptr, QgsMapBoxGlStyleConverter::InterpolationType type=QgsMapBoxGlStyleConverter::InterpolationType::Exponential)
Interpolates a point/offset with either scale_linear() or scale_exp() (depending on base value).
static QString parseArrayStops(const QVariantList &stops, QgsMapBoxGlStyleConversionContext &context, double multiplier=1)
Takes numerical arrays from stops.
Encapsulates a MapBox GL style raster source.
Qgis::MapBoxGlStyleSourceType type() const override
Returns the source type.
QgsMapBoxGlStyleRasterSource(const QString &name)
Constructor for QgsMapBoxGlStyleRasterSource.
QgsRasterLayer * toRasterLayer() const
Returns a new raster layer representing the raster source, or nullptr if the source cannot be represe...
bool setFromJson(const QVariantMap &json, QgsMapBoxGlStyleConversionContext *context) override
Sets the source's state from a json map.
QStringList tiles() const
Returns the list of tile sources.
Encapsulates a MapBox GL style raster sub layer.
QString source() const
Returns the layer's source.
QString id() const
Returns the layer's ID.
QgsPropertyCollection & dataDefinedProperties()
Returns a reference to the layer's data defined properties.
QgsMapBoxGlStyleRasterSubLayer(const QString &id, const QString &source)
Constructor for QgsMapBoxGlStyleRasterSubLayer, with the given id and source.
Line symbol layer type which draws repeating marker symbols along a line feature.
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
bool setSubSymbol(QgsSymbol *symbol) override
Sets layer's subsymbol. takes ownership of the passed symbol.
virtual void setSize(double size)
Sets the symbol size.
void setOffsetUnit(Qgis::RenderUnit unit)
Sets the units for the symbol's offset.
void setAngle(double angle)
Sets the rotation angle for the marker.
void setOffset(QPointF offset)
Sets the marker's offset, which is the horizontal and vertical displacement which the rendered marker...
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units for the symbol's size.
A marker symbol type, for rendering Point and MultiPoint geometries.
void setEnabled(bool enabled)
Sets whether the effect is enabled.
Contains settings for how a map layer will be labeled.
double yOffset
Vertical offset of label.
const QgsLabelObstacleSettings & obstacleSettings() const
Returns the label obstacle settings.
const QgsLabelPlacementSettings & placementSettings() const
Returns the label placement settings.
void setFormat(const QgsTextFormat &format)
Sets the label text formatting settings, e.g., font settings, buffer settings, etc.
double xOffset
Horizontal offset of label.
Qgis::LabelPlacement placement
Label placement mode.
Qgis::LabelMultiLineAlignment multilineAlign
Horizontal alignment of multi-line labels.
int priority
Label priority.
double angleOffset
Label rotation, in degrees clockwise.
const QgsLabelThinningSettings & thinningSettings() const
Returns the label thinning settings.
void setThinningSettings(const QgsLabelThinningSettings &settings)
Sets the label thinning settings.
Qgis::RenderUnit offsetUnits
Units for offsets of label.
void setDataDefinedProperties(const QgsPropertyCollection &collection)
Sets the label's property collection, used for data defined overrides.
bool isExpression
true if this label is made from a expression string, e.g., FieldName || 'mm'
void setPlacementSettings(const QgsLabelPlacementSettings &settings)
Sets the label placement settings.
const QgsLabelLineSettings & lineSettings() const
Returns the label line settings, which contain settings related to how the label engine places and fo...
double dist
Distance from feature to the label.
Qgis::RenderUnit distUnits
Units the distance from feature to the label.
@ LinePlacementOptions
Line placement flags.
@ RemoveDuplicateLabelDistance
Minimum distance from labels for this feature to other labels with duplicate text.
QString fieldName
Name of field (or an expression) to use for label text.
int autoWrapLength
If non-zero, indicates that label text should be automatically wrapped to (ideally) the specified num...
const QgsLabelPointSettings & pointSettings() const
Returns the label point settings, which contain settings related to how the label engine places and f...
A grouped map of multiple QgsProperty objects, each referenced by an integer key value.
QVariant value(int key, const QgsExpressionContext &context, const QVariant &defaultValue=QVariant()) const final
Returns the calculated value of the property with the specified key from within the collection.
void setProperty(int key, const QgsProperty &property)
Adds a property to the collection and takes ownership of it.
bool isActive(int key) const final
Returns true if the collection contains an active property with the specified key.
QgsProperty property(int key) const final
Returns a matching property from the collection, if one exists.
A store for object properties.
QString asExpression() const
Returns an expression string representing the state of the property, or an empty string if the proper...
QString expressionString() const
Returns the expression used for the property value.
QVariant value(const QgsExpressionContext &context, const QVariant &defaultValue=QVariant(), bool *ok=nullptr) const
Calculates the current value of the property, including any transforms which are set for the property...
static QgsProperty fromExpression(const QString &expression, bool isActive=true)
Returns a new ExpressionBasedProperty created from the specified expression.
void setExpressionString(const QString &expression)
Sets the expression to use for the property value.
static QgsProperty fromValue(const QVariant &value, bool isActive=true)
Returns a new StaticProperty created from the specified value.
void setActive(bool active)
Sets whether the property is currently active.
static QgsProviderRegistry * instance(const QString &pluginPath=QString())
Means of accessing canonical single instance.
A fill symbol layer which fills polygons with a repeated raster image.
void setSizeUnit(Qgis::RenderUnit unit)
Sets the unit for the image's width and height.
void setOpacity(double opacity)
Sets the opacity for the raster image used in the fill.
void setImageFilePath(const QString &imagePath)
Sets the path to the raster image used for the fill.
void setWidth(double width)
Sets the width for scaling the image used in the fill.
void setCoordinateMode(Qgis::SymbolCoordinateReference mode)
Set the coordinate mode for fill.
Represents a raster layer.
Line symbol layer type which draws line sections using a raster image file.
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
Raster marker symbol layer class.
void setOpacity(double opacity)
Set the marker opacity.
void setPath(const QString &path)
Set the marker raster image path.
@ RendererOpacity
Raster renderer global opacity.
Renders polygons using a single fill and stroke color.
void setBrushStyle(Qt::BrushStyle style)
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
void setStrokeWidth(double strokeWidth)
void setStrokeStyle(Qt::PenStyle strokeStyle)
void setOffsetUnit(Qgis::RenderUnit unit)
Sets the unit for the fill's offset.
void setFillColor(const QColor &color) override
Sets the fill color for the symbol layer.
void setOffset(QPointF offset)
Sets an offset by which polygons will be translated during rendering.
void setStrokeColor(const QColor &strokeColor) override
Sets the stroke color for the symbol layer.
A simple line symbol layer, which renders lines using a line in a variety of styles (e....
void setPenCapStyle(Qt::PenCapStyle style)
Sets the pen cap style used to render the line (e.g.
void setUseCustomDashPattern(bool b)
Sets whether the line uses a custom dash pattern.
void setCustomDashVector(const QVector< qreal > &vector)
Sets the custom dash vector, which is the pattern of alternating drawn/skipped lengths used while ren...
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
void setPenJoinStyle(Qt::PenJoinStyle style)
Sets the pen join style used to render the line (e.g.
Simple marker symbol layer, consisting of a rendered shape with solid fill color and a stroke.
void setFillColor(const QColor &color) override
Sets the fill color for the symbol layer.
void setStrokeWidthUnit(Qgis::RenderUnit u)
Sets the unit for the width of the marker's stroke.
void setStrokeWidth(double w)
Sets the width of the marker's stroke.
void setStrokeColor(const QColor &color) override
Sets the marker's stroke color.
static QColor parseColor(const QString &colorStr, bool strictEval=false)
Attempts to parse a string as a color using a variety of common formats, including hex codes,...
@ File
Filename, eg for svg files.
@ CustomDash
Custom dash pattern.
@ Name
Name, eg shape name for simple markers.
@ Interval
Line marker interval.
@ LayerEnabled
Whether symbol layer is enabled.
virtual void setColor(const QColor &color)
Sets the "representative" color for the symbol layer.
void setDataDefinedProperties(const QgsPropertyCollection &collection)
Sets the symbol layer's property collection, used for data defined overrides.
void setPlacements(Qgis::MarkerLinePlacements placements)
Sets the placement of the symbols.
Container for settings relating to a text background object.
void setMarkerSymbol(QgsMarkerSymbol *symbol)
Sets the current marker symbol for the background shape.
void setSizeType(SizeType type)
Sets the method used to determine the size of the background shape (e.g., fixed size or buffer around...
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units used for the shape's size.
void setType(ShapeType type)
Sets the type of background shape to draw (e.g., square, ellipse, SVG).
void setEnabled(bool enabled)
Sets whether the text background will be drawn.
void setSize(QSizeF size)
Sets the size of the background shape.
void setColor(const QColor &color)
Sets the color for the buffer.
void setOpacity(double opacity)
Sets the buffer opacity.
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 setPaintEffect(QgsPaintEffect *effect)
Sets the current paint effect for the buffer.
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 setBackground(const QgsTextBackgroundSettings &backgroundSettings)
Sets the text's background settings.q.
void setNamedStyle(const QString &style)
Sets the named style for the font used for rendering text.
QFont font() const
Returns the font used for rendering text.
QgsTextBufferSettings & buffer()
Returns a reference to the text buffer settings.
Configuration of a single style within QgsVectorTileBasicLabeling.
void setLayerName(const QString &name)
Sets name of the sub-layer to render (empty layer means that all layers match).
void setMinZoomLevel(int minZoom)
Sets minimum zoom level index (negative number means no limit).
void setFilterExpression(const QString &expr)
Sets filter expression (empty filter means that all features match).
void setMaxZoomLevel(int maxZoom)
Sets maximum zoom level index (negative number means no limit).
void setStyleName(const QString &name)
Sets human readable name of this style.
void setGeometryType(Qgis::GeometryType geomType)
Sets type of the geometry that will be used (point / line / polygon).
void setLabelSettings(const QgsPalLayerSettings &settings)
Sets labeling configuration of this style.
void setEnabled(bool enabled)
Sets whether this style is enabled (used for rendering).
Definition of map rendering of a subset of vector tile data.
void setEnabled(bool enabled)
Sets whether this style is enabled (used for rendering).
void setMinZoomLevel(int minZoom)
Sets minimum zoom level index (negative number means no limit).
void setLayerName(const QString &name)
Sets name of the sub-layer to render (empty layer means that all layers match).
void setFilterExpression(const QString &expr)
Sets filter expression (empty filter means that all features match).
void setSymbol(QgsSymbol *sym)
Sets symbol for rendering. Takes ownership of the symbol.
void setStyleName(const QString &name)
Sets human readable name of this style.
void setMaxZoomLevel(int maxZoom)
Sets maximum zoom level index (negative number means no limit).
void setGeometryType(Qgis::GeometryType geomType)
Sets type of the geometry that will be used (point / line / polygon).
Base class for labeling configuration classes for vector tile layers.
virtual QgsVectorTileLabeling * clone() const =0SIP_FACTORY
Returns a new copy of the object.
Abstract base class for all vector tile renderer implementations.
virtual QgsVectorTileRenderer * clone() const =0
Returns a clone of the renderer.
As part of the API refactoring and improvements which landed in the Processing API was substantially reworked from the x version This was done in order to allow much of the underlying Processing framework to be ported into c
QString qgsDoubleToString(double a, int precision=17)
Returns a string representation of a double.
Definition qgis.h:7247
QString qgsEnumValueToKey(const T &value, bool *returnOk=nullptr)
Returns the value for the given key of an enum.
Definition qgis.h:7576
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
#define QgsDebugError(str)
Definition qgslogger.h:71
QList< QgsSymbolLayer * > QgsSymbolLayerList
Definition qgssymbol.h:30