QGIS API Documentation 4.0.0-Norrköping (1ddcee3d0e4)
Loading...
Searching...
No Matches
qgsvectorlayerlabelprovider.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsvectorlayerlabelprovider.cpp
3 --------------------------------------
4 Date : September 2015
5 Copyright : (C) 2015 by Martin Dobias
6 Email : wonder dot sk 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
17
18#include "callouts/qgscallout.h"
19#include "feature.h"
20#include "labelposition.h"
21#include "pal/layer.h"
23#include "qgsgeometry.h"
24#include "qgslabelingresults.h"
25#include "qgslabelsearchtree.h"
26#include "qgslinestring.h"
27#include "qgslogger.h"
28#include "qgsmarkersymbol.h"
29#include "qgsmaskidprovider.h"
30#include "qgsmultipolygon.h"
31#include "qgspallabeling.h"
32#include "qgspolygon.h"
33#include "qgsrenderer.h"
34#include "qgssymbol.h"
36#include "qgstextfragment.h"
37#include "qgstextlabelfeature.h"
38#include "qgstextrenderer.h"
39#include "qgsvectorlayer.h"
41
42#include <QPicture>
43#include <QString>
44#include <QTextDocument>
45#include <QTextFragment>
46
47using namespace Qt::StringLiterals;
48
49using namespace pal;
50
51QgsVectorLayerLabelProvider::QgsVectorLayerLabelProvider( QgsVectorLayer *layer, const QString &providerId, bool withFeatureLoop, const QgsPalLayerSettings *settings, const QString &layerName )
53 , mSettings( settings ? *settings : QgsPalLayerSettings() ) // TODO: all providers should have valid settings?
54 , mLayerGeometryType( layer->geometryType() )
55 , mRenderer( layer->renderer() )
56 , mFields( layer->fields() )
57 , mCrs( layer->crs() )
58{
59 mName = layerName.isEmpty() ? layer->id() : layerName;
60
61 if ( withFeatureLoop )
62 {
63 mSource = std::make_unique<QgsVectorLayerFeatureSource>( layer );
64 }
65
66 init();
67}
68
70 Qgis::GeometryType geometryType, const QgsFields &fields, const QgsCoordinateReferenceSystem &crs, const QString &providerId, const QgsPalLayerSettings *settings, QgsMapLayer *layer, const QString &layerName
71)
73 , mSettings( settings ? *settings : QgsPalLayerSettings() ) // TODO: all providers should have valid settings?
74 , mLayerGeometryType( geometryType )
75 , mFields( fields )
76 , mCrs( crs )
77{
78 mName = layerName.isEmpty() ? layer->id() : layerName;
79
80 init();
81}
82
84{
85 mPlacement = mSettings.placement;
86
87 mFlags = Flags();
88 if ( mSettings.drawLabels )
90 if ( mSettings.lineSettings().mergeLines() && !mSettings.lineSettings().addDirectionSymbol() )
92 if ( mSettings.centroidInside )
94
95 mPriority = 1 - mSettings.priority / 10.0; // convert 0..10 --> 1..0
96
98 {
99 //override obstacle type to treat any intersection of a label with the point symbol as a high cost conflict
101 }
102 else
103 {
104 mObstacleType = mSettings.obstacleSettings().type();
105 }
106
107 mUpsidedownLabels = mSettings.upsidedownLabels;
108}
109
110
115
116
117bool QgsVectorLayerLabelProvider::prepare( QgsRenderContext &context, QSet<QString> &attributeNames )
118{
119 const QgsMapSettings &mapSettings = mEngine->mapSettings();
120
121 return mSettings.prepare( context, attributeNames, mFields, mapSettings, mCrs );
122}
123
125{
127 mSettings.startRender( context );
128}
129
131{
133 mSettings.stopRender( context );
134}
135
137{
138 if ( !mSource )
139 {
140 // we have created the provider with "own feature loop" == false
141 // so it is assumed that prepare() has been already called followed by registerFeature() calls
142 return mLabels;
143 }
144
145 QSet<QString> attrNames;
146 if ( !prepare( ctx, attrNames ) )
147 return QList<QgsLabelFeature *>();
148
149 if ( mRenderer )
150 mRenderer->startRender( ctx, mFields );
151
152 QgsRectangle layerExtent = ctx.extent();
153 if ( mSettings.ct.isValid() && !mSettings.ct.isShortCircuited() )
154 {
155 QgsCoordinateTransform extentTransform = mSettings.ct;
156 extentTransform.setBallparkTransformsAreAppropriate( true );
157 layerExtent = extentTransform.transformBoundingBox( ctx.extent(), Qgis::TransformDirection::Reverse );
158 }
159
160 QgsFeatureRequest request;
161 request.setFilterRect( layerExtent );
162 request.setSubsetOfAttributes( attrNames, mFields );
163 QgsFeatureIterator fit = mSource->getFeatures( request );
164
166 ctx.expressionContext().appendScope( symbolScope );
167 QgsFeature fet;
168 while ( fit.nextFeature( fet ) )
169 {
170 QgsGeometry obstacleGeometry;
171 const QgsSymbol *symbol = nullptr;
172 if ( mRenderer )
173 {
174 QgsSymbolList symbols = mRenderer->originalSymbolsForFeature( fet, ctx );
175 if ( !symbols.isEmpty() && fet.geometry().type() == Qgis::GeometryType::Point )
176 {
177 //point feature, use symbol bounds as obstacle
178 obstacleGeometry = QgsVectorLayerLabelProvider::getPointObstacleGeometry( fet, ctx, symbols );
179 }
180 if ( !symbols.isEmpty() )
181 {
182 symbol = symbols.at( 0 );
183 symbolScope = QgsExpressionContextUtils::updateSymbolScope( symbol, symbolScope );
184 }
185 }
186 ctx.expressionContext().setFeature( fet );
187 registerFeature( fet, ctx, obstacleGeometry, symbol );
188 }
189
190 if ( ctx.expressionContext().lastScope() == symbolScope )
191 delete ctx.expressionContext().popScope();
192
193 if ( mRenderer )
194 mRenderer->stopRender( ctx );
195
196 return mLabels;
197}
198
199QList< QgsLabelFeature * > QgsVectorLayerLabelProvider::registerFeature( const QgsFeature &feature, QgsRenderContext &context, const QgsGeometry &obstacleGeometry, const QgsSymbol *symbol )
200{
201 std::vector< std::unique_ptr< QgsLabelFeature > > labels = mSettings.registerFeatureWithDetails( feature, context, obstacleGeometry, symbol );
202 QList< QgsLabelFeature * > res;
203 for ( auto &it : labels )
204 {
205 if ( it )
206 {
207 res << it.get();
208 mLabels << it.release();
209 }
210 }
211 return res;
212}
213
215{
216 if ( !fet.hasGeometry() || fet.geometry().type() != Qgis::GeometryType::Point )
217 return QgsGeometry();
218
219 bool isMultiPoint = fet.geometry().constGet()->nCoordinates() > 1;
220 std::unique_ptr< QgsAbstractGeometry > obstacleGeom;
221 if ( isMultiPoint )
222 obstacleGeom = std::make_unique< QgsMultiPolygon >();
223
224 // for each point
225 for ( int i = 0; i < fet.geometry().constGet()->nCoordinates(); ++i )
226 {
227 QRectF bounds;
228 QgsPoint p = fet.geometry().constGet()->vertexAt( QgsVertexId( i, 0, 0 ) );
229 double x = p.x();
230 double y = p.y();
231 double z = 0; // dummy variable for coordinate transforms
232
233 //transform point to pixels
234 if ( context.coordinateTransform().isValid() )
235 {
236 try
237 {
238 context.coordinateTransform().transformInPlace( x, y, z );
239 }
240 catch ( QgsCsException & )
241 {
242 return QgsGeometry();
243 }
244 }
245 context.mapToPixel().transformInPlace( x, y );
246
247 QPointF pt( x, y );
248 const auto constSymbols = symbols;
249 for ( QgsSymbol *symbol : constSymbols )
250 {
251 if ( symbol->type() == Qgis::SymbolType::Marker )
252 {
253 if ( bounds.isValid() )
254 bounds = bounds.united( static_cast< QgsMarkerSymbol * >( symbol )->bounds( pt, context, fet ) );
255 else
256 bounds = static_cast< QgsMarkerSymbol * >( symbol )->bounds( pt, context, fet );
257 }
258 }
259
260 //convert bounds to a geometry
261 QVector< double > bX;
262 bX << bounds.left() << bounds.right() << bounds.right() << bounds.left();
263 QVector< double > bY;
264 bY << bounds.top() << bounds.top() << bounds.bottom() << bounds.bottom();
265 auto boundLineString = std::make_unique< QgsLineString >( bX, bY );
266
267 //then transform back to map units
268 //TODO - remove when labeling is refactored to use screen units
269 for ( int i = 0; i < boundLineString->numPoints(); ++i )
270 {
271 QgsPointXY point = context.mapToPixel().toMapCoordinates( static_cast<int>( boundLineString->xAt( i ) ), static_cast<int>( boundLineString->yAt( i ) ) );
272 boundLineString->setXAt( i, point.x() );
273 boundLineString->setYAt( i, point.y() );
274 }
275 if ( context.coordinateTransform().isValid() )
276 {
277 try
278 {
279 boundLineString->transform( context.coordinateTransform(), Qgis::TransformDirection::Reverse );
280 }
281 catch ( QgsCsException & )
282 {
283 return QgsGeometry();
284 }
285 }
286 boundLineString->close();
287
288 if ( context.coordinateTransform().isValid() )
289 {
290 // coordinate transforms may have resulted in nan coordinates - if so, strip these out
291 boundLineString->filterVertices( []( const QgsPoint &point ) -> bool { return std::isfinite( point.x() ) && std::isfinite( point.y() ); } );
292 if ( !boundLineString->isRing() )
293 return QgsGeometry();
294 }
295
296 auto obstaclePolygon = std::make_unique< QgsPolygon >();
297 obstaclePolygon->setExteriorRing( boundLineString.release() );
298
299 if ( isMultiPoint )
300 {
301 static_cast<QgsMultiPolygon *>( obstacleGeom.get() )->addGeometry( obstaclePolygon.release() );
302 }
303 else
304 {
305 obstacleGeom = std::move( obstaclePolygon );
306 }
307 }
308
309 return QgsGeometry( std::move( obstacleGeom ) );
310}
311
313{
314 if ( !mSettings.drawLabels )
315 return;
316
317 // render callout
318 if ( mSettings.callout() && mSettings.callout()->drawOrder() == QgsCallout::OrderBelowAllLabels )
319 {
320 drawCallout( context, label );
321 }
322}
323
324void QgsVectorLayerLabelProvider::drawCallout( QgsRenderContext &context, pal::LabelPosition *label ) const
325{
326 bool enabled = mSettings.callout()->enabled();
328 {
329 context.expressionContext().setOriginalValueVariable( enabled );
331 }
332 if ( enabled )
333 {
334 QgsMapToPixel xform = context.mapToPixel();
335 xform.setMapRotation( 0, 0, 0 );
336 QPointF outPt = xform.transform( label->getX(), label->getY() ).toQPointF();
337 QgsPointXY outPt2 = xform.transform( label->getX() + label->getWidth(), label->getY() + label->getHeight() );
338 QRectF rect( outPt.x(), outPt.y(), outPt2.x() - outPt.x(), outPt2.y() - outPt.y() );
339
340 QgsGeometry g( QgsGeos::fromGeos( label->getFeaturePart()->feature()->geometry() ) );
341 g.transform( xform.transform() );
342 QgsCallout::QgsCalloutContext calloutContext;
344 calloutContext.originalFeatureCrs = label->getFeaturePart()->feature()->originalFeatureCrs();
345 mSettings.callout()->render( context, rect, label->getAlpha() * 180 / M_PI, g, calloutContext );
346
347 const QList< QgsCalloutPosition > renderedPositions = calloutContext.positions();
348
349 for ( QgsCalloutPosition position : renderedPositions )
350 {
351 position.layerID = mLayerId;
352 position.featureId = label->getFeaturePart()->featureId();
353 position.providerID = mProviderId;
354 mEngine->results()->mLabelSearchTree->insertCallout( position );
355 }
356 }
357}
358
360{
361 if ( !mSettings.drawLabels )
362 return;
363
364 QgsTextLabelFeature *lf = dynamic_cast<QgsTextLabelFeature *>( label->getFeaturePart()->feature() );
365 if ( !lf )
366 return;
367
368 // Copy to temp, editable layer settings
369 // these settings will be changed by any data defined values, then used for rendering label components
370 // settings may be adjusted during rendering of components
372
373 // apply any previously applied data defined settings for the label
374 const QMap< QgsPalLayerSettings::Property, QVariant > &ddValues = lf->dataDefinedValues();
375
376 //font
377 QFont dFont = lf->definedFont();
378 QgsDebugMsgLevel( u"PAL font tmpLyr: %1, Style: %2"_s.arg( tmpLyr.format().font().toString(), tmpLyr.format().font().styleName() ), 4 );
379 QgsDebugMsgLevel( u"PAL font definedFont: %1, Style: %2"_s.arg( dFont.toString(), dFont.styleName() ), 4 );
380
381 QgsTextFormat format = tmpLyr.format();
382 format.setFont( dFont );
383
384 // size has already been calculated and stored in the defined font - this calculated size
385 // is in pixels
386 format.setSize( dFont.pixelSize() );
388 tmpLyr.setFormat( format );
389
391 {
392 //calculate font alignment based on label quadrant
393 switch ( label->quadrant() )
394 {
399 break;
404 break;
409 break;
410 }
411 }
412
413 // update tmpLyr with any data defined text style values
414 QgsPalLabeling::dataDefinedTextStyle( tmpLyr, ddValues );
415
416 // update tmpLyr with any data defined text buffer values
417 QgsPalLabeling::dataDefinedTextBuffer( tmpLyr, ddValues );
418
419 // update tmpLyr with any data defined text mask values
420 QgsPalLabeling::dataDefinedTextMask( tmpLyr, ddValues );
421
422 // update tmpLyr with any data defined text formatting values
423 QgsPalLabeling::dataDefinedTextFormatting( tmpLyr, ddValues );
424
425 // update tmpLyr with any data defined shape background values
426 QgsPalLabeling::dataDefinedShapeBackground( tmpLyr, ddValues );
427
428 // update tmpLyr with any data defined drop shadow values
429 QgsPalLabeling::dataDefinedDropShadow( tmpLyr, ddValues );
430
431 // Render the components of a label in reverse order
432 // (backgrounds -> text)
433
434 // render callout
435 if ( mSettings.callout() && mSettings.callout()->drawOrder() == QgsCallout::OrderBelowIndividualLabels )
436 {
437 drawCallout( context, label );
438 }
439
441 {
442 QgsTextFormat format = tmpLyr.format();
443
444 if ( tmpLyr.format().background().enabled() && tmpLyr.format().background().type() != QgsTextBackgroundSettings::ShapeMarkerSymbol ) // background shadows not compatible with marker symbol backgrounds
445 {
447 }
448 else if ( tmpLyr.format().buffer().enabled() )
449 {
451 }
452 else
453 {
455 }
456
457 tmpLyr.setFormat( format );
458 }
459
460 if ( tmpLyr.format().background().enabled() )
461 {
462 drawLabelPrivate( label, context, tmpLyr, Qgis::TextComponent::Background );
463 }
464
465 if ( tmpLyr.format().buffer().enabled() )
466 {
467 drawLabelPrivate( label, context, tmpLyr, Qgis::TextComponent::Buffer );
468 }
469
470 drawLabelPrivate( label, context, tmpLyr, Qgis::TextComponent::Text );
471
472 // add to the results
473 QString labeltext = label->getFeaturePart()->feature()->labelText();
474 mEngine->results()->mLabelSearchTree->insertLabel( label, label->getFeaturePart()->featureId(), mLayerId, labeltext, dFont, false, lf->hasFixedPosition(), mProviderId );
475}
476
478{
479 QgsTextLabelFeature *lf = dynamic_cast<QgsTextLabelFeature *>( label->getFeaturePart()->feature() );
480 if ( !lf )
481 return;
482
483 QgsTextFormat format = mSettings.format();
484 if ( mSettings.drawLabels && mSettings.unplacedVisibility() != Qgis::UnplacedLabelVisibility::NeverShow && mEngine->engineSettings().flags() & Qgis::LabelingFlag::DrawUnplacedLabels )
485 {
487 format = tmpLyr.format();
488 format.setColor( mEngine->engineSettings().unplacedLabelColor() );
489 tmpLyr.setFormat( format );
490 drawLabelPrivate( label, context, tmpLyr, Qgis::TextComponent::Text );
491 }
492
493 // add to the results
494 QString labeltext = label->getFeaturePart()->feature()->labelText();
495 mEngine->results()->mLabelSearchTree->insertLabel( label, label->getFeaturePart()->featureId(), mLayerId, labeltext, format.font(), false, lf->hasFixedPosition(), mProviderId, true );
496}
497
499{
500 // NOTE: this is repeatedly called for multi-part labels
501 Qgis::TextComponents components;
502 switch ( drawType )
503 {
506 break;
507
510 break;
511
514 components = drawType;
515 break;
516 }
517
518 // features are pre-rotated but not scaled/translated,
519 // so we only disable rotation here. Ideally, they'd be
520 // also pre-scaled/translated, as suggested here:
521 // https://github.com/qgis/QGIS/issues/20071
522 QgsMapToPixel xform = context.mapToPixel();
523 xform.setMapRotation( 0, 0, 0 );
524
525 QPointF outPt = xform.transform( label->getX(), label->getY() ).toQPointF();
526
527 QgsTextRenderer::Component component;
528 component.dpiRatio = dpiRatio;
529 component.origin = outPt;
530 component.rotation = label->getAlpha();
531
532 if ( drawType == Qgis::TextComponent::Background )
533 {
534 // get rotated label's center point
535 QPointF centerPt( outPt );
536 QgsPointXY outPt2 = xform.transform( label->getX() + label->getWidth() / 2, label->getY() + label->getHeight() / 2 );
537
538 double xc = outPt2.x() - outPt.x();
539 double yc = outPt2.y() - outPt.y();
540
541 double angle = -component.rotation;
542 double xd = xc * std::cos( angle ) - yc * std::sin( angle );
543 double yd = xc * std::sin( angle ) + yc * std::cos( angle );
544
545 centerPt.setX( centerPt.x() + xd );
546 centerPt.setY( centerPt.y() + yd );
547
548 component.center = centerPt;
549
550 {
551 // label size has already been calculated using any symbology reference scale factor -- we need
552 // to temporarily remove the reference scale here or we'll be applying the scaling twice
553 QgsScopedRenderContextReferenceScaleOverride referenceScaleOverride( context, -1.0 );
554
555 // convert label size to render units
556 double labelWidthPx = context.convertToPainterUnits( label->getWidth(), Qgis::RenderUnit::MapUnits, QgsMapUnitScale() );
557 double labelHeightPx = context.convertToPainterUnits( label->getHeight(), Qgis::RenderUnit::MapUnits, QgsMapUnitScale() );
558
559 component.size = QSizeF( labelWidthPx, labelHeightPx );
560 }
561
562 QgsTextRenderer::drawBackground( context, component, tmpLyr.format(), QgsTextDocumentMetrics(), Qgis::TextLayoutMode::Labeling );
563 }
564
565 else if ( drawType == Qgis::TextComponent::Buffer || drawType == Qgis::TextComponent::Text )
566 {
567 // TODO: optimize access :)
568 QgsTextLabelFeature *lf = static_cast<QgsTextLabelFeature *>( label->getFeaturePart()->feature() );
569 QString txt = lf->text( label->getPartId() );
570
571 if ( auto *lMaskIdProvider = context.maskIdProvider() )
572 {
573 int maskId = lMaskIdProvider->maskId( label->getFeaturePart()->layer()->provider()->layerId(), label->getFeaturePart()->layer()->provider()->providerId() );
574 context.setCurrentMaskId( maskId );
575 }
576
577 const QgsTextDocument &precalculatedDocument = lf->document();
578 const QgsTextDocumentMetrics &precalculatedMetrics = lf->documentMetrics();
579 const QgsTextDocument *document = &precalculatedDocument;
580 const QgsTextDocumentMetrics *documentMetrics = &precalculatedMetrics;
581
582 // add the direction symbol if needed
583 // note that IF we do this, we can no longer use the original text document and metrics
584 // but have to re-calculate these with the newly added text!
585 std::optional< QgsTextDocument > newDocument;
586 std::optional< QgsTextDocumentMetrics > newDocumentMetrics;
587 if ( !txt.isEmpty() && tmpLyr.placement == Qgis::LabelPlacement::Line && tmpLyr.lineSettings().addDirectionSymbol() )
588 {
589 newDocument.emplace( *document );
590 bool prependSymb = false;
591 QString symb = tmpLyr.lineSettings().rightDirectionSymbol();
592
593 if ( label->isReversedFromLineDirection() )
594 {
595 prependSymb = true;
596 symb = tmpLyr.lineSettings().leftDirectionSymbol();
597 }
598
599 if ( tmpLyr.lineSettings().reverseDirectionSymbol() )
600 {
601 if ( symb == tmpLyr.lineSettings().rightDirectionSymbol() )
602 {
603 prependSymb = true;
604 symb = tmpLyr.lineSettings().leftDirectionSymbol();
605 }
606 else
607 {
608 prependSymb = false;
609 symb = tmpLyr.lineSettings().rightDirectionSymbol();
610 }
611 }
612
613 switch ( tmpLyr.lineSettings().directionSymbolPlacement() )
614 {
616 {
617 newDocument->insert( 0, QgsTextBlock( QgsTextFragment( symb ) ) );
618 break;
619 }
620
622 newDocument->append( QgsTextBlock( QgsTextFragment( symb ) ) );
623 break;
624
626 {
627 QgsTextBlock &block = newDocument.value()[0];
628 if ( prependSymb )
629 {
630 block.insert( 0, QgsTextFragment( symb ) );
631 }
632 else
633 {
634 block.append( QgsTextFragment( symb ) );
635 }
636 break;
637 }
638 }
639
640 QgsScopedRenderContextReferenceScaleOverride referenceScaleOverride( context, -1.0 );
641 newDocumentMetrics.emplace( QgsTextDocumentMetrics::calculateMetrics( newDocument.value(), tmpLyr.format(), context ) );
642 document = &newDocument.value();
643 documentMetrics = &newDocumentMetrics.value();
644 }
645
653
654 QgsTextRenderer::Component component;
655 component.origin = outPt;
656 component.rotation = label->getAlpha();
657
658 // If we are using non-curved, HTML formatted labels then we've already precalculated the text metrics.
659 // Otherwise we'll need to calculate them now.
660 switch ( tmpLyr.placement )
661 {
664 {
665 QgsTextDocument document;
666 const QgsTextCharacterFormat c = lf->characterFormat( label->getPartId() );
667 const QStringList multiLineList = QgsPalLabeling::splitToLines( txt, tmpLyr.wrapChar, tmpLyr.autoWrapLength, tmpLyr.useMaxLineLengthForAutoWrap );
668 for ( const QString &line : multiLineList )
669 {
670 document.append( QgsTextBlock::fromPlainText( line, c ) );
671 }
672
673 QgsScopedRenderContextReferenceScaleOverride referenceScaleOverride( context, -1.0 );
674 const QgsTextDocumentMetrics metrics = QgsTextDocumentMetrics::calculateMetrics( document, tmpLyr.format(), context );
675 QgsTextRenderer::drawTextInternal( components, context, tmpLyr.format(), component, document, metrics, hAlign, Qgis::TextVerticalAlignment::Top, Qgis::TextLayoutMode::Labeling );
676 break;
677 }
678
686 {
687 const double verticalAlignOffset = -documentMetrics->blockVerticalMargin( document->size() - 1 );
688
689 component.origin.ry() += verticalAlignOffset;
690
691 QgsTextRenderer::drawTextInternal( components, context, tmpLyr.format(), component, *document, *documentMetrics, hAlign, Qgis::TextVerticalAlignment::Top, Qgis::TextLayoutMode::Labeling );
692 break;
693 }
694 }
695 }
696 if ( label->nextPart() )
697 drawLabelPrivate( label->nextPart(), context, tmpLyr, drawType, dpiRatio );
698}
699
704
706{
707 mFields = fields;
708}
@ LabelLargestPartOnly
Place a label only on the largest part from the geometry.
Definition qgis.h:1291
@ OverPoint
Arranges candidates over a point (or centroid of a polygon), or at a preset offset from the point....
Definition qgis.h:1234
@ Curved
Arranges candidates following the curvature of a line feature. Applies to line layers only.
Definition qgis.h:1236
@ AroundPoint
Arranges candidates in a circle around a point (or centroid of a polygon). Applies to point or polygo...
Definition qgis.h:1233
@ Line
Arranges candidates parallel to a generalised line representing the feature or parallel to a polygon'...
Definition qgis.h:1235
@ Free
Arranges candidates scattered throughout a polygon feature. Candidates are rotated to respect the pol...
Definition qgis.h:1238
@ OrderedPositionsAroundPoint
Candidates are placed in predefined positions around a point. Preference is given to positions with g...
Definition qgis.h:1239
@ Horizontal
Arranges horizontal candidates scattered throughout a polygon feature. Applies to polygon layers only...
Definition qgis.h:1237
@ PerimeterCurved
Arranges candidates following the curvature of a polygon's boundary. Applies to polygon layers only.
Definition qgis.h:1240
@ OutsidePolygons
Candidates are placed outside of polygon boundaries. Applies to polygon layers only.
Definition qgis.h:1241
@ Labeling
Labeling-specific layout mode.
Definition qgis.h:3005
@ AboveRight
Above right.
Definition qgis.h:1323
@ BelowLeft
Below left.
Definition qgis.h:1327
@ Above
Above center.
Definition qgis.h:1322
@ BelowRight
Below right.
Definition qgis.h:1329
@ Right
Right middle.
Definition qgis.h:1326
@ AboveLeft
Above left.
Definition qgis.h:1321
@ Below
Below center.
Definition qgis.h:1328
@ Over
Center middle.
Definition qgis.h:1325
@ NeverShow
Never show unplaced labels, regardless of the engine setting.
Definition qgis.h:1183
@ DrawUnplacedLabels
Whether to render unplaced labels as an indicator/warning for users.
Definition qgis.h:2949
GeometryType
The geometry types are used to group Qgis::WkbType in a coarse way.
Definition qgis.h:379
@ Point
Points.
Definition qgis.h:380
@ Center
Center align.
Definition qgis.h:1405
@ FollowPlacement
Alignment follows placement of label, e.g., labels to the left of a feature will be drawn with right ...
Definition qgis.h:1407
@ MapUnits
Map units.
Definition qgis.h:5342
@ Pixels
Pixels.
Definition qgis.h:5343
@ Top
Align to top.
Definition qgis.h:3063
@ Marker
Marker symbol.
Definition qgis.h:637
QFlags< TextComponent > TextComponents
Text components.
Definition qgis.h:3032
TextHorizontalAlignment
Text horizontal alignment.
Definition qgis.h:3043
@ Justify
Justify align.
Definition qgis.h:3047
@ Center
Center align.
Definition qgis.h:3045
TextComponent
Text components.
Definition qgis.h:3019
@ Shadow
Drop shadow.
Definition qgis.h:3023
@ Buffer
Buffer component.
Definition qgis.h:3021
@ Text
Text component.
Definition qgis.h:3020
@ Background
Background shape.
Definition qgis.h:3022
@ Reverse
Reverse/inverse transform (from destination to source).
Definition qgis.h:2766
virtual int nCoordinates() const
Returns the number of nodes contained in the geometry.
virtual QgsPoint vertexAt(QgsVertexId id) const =0
Returns the point corresponding to a specified vertex id.
QgsLabelObstacleSettings::ObstacleType mObstacleType
Type of the obstacle of feature geometries.
QString mName
Name of the layer.
virtual void stopRender(QgsRenderContext &context)
To be called after rendering is complete.
QString mLayerId
Associated layer's ID, if applicable.
double mPriority
Default priority of labels. 0 = highest priority, 1 = lowest priority.
const QgsLabelingEngine * mEngine
Associated labeling engine.
QgsMapLayer * layer() const
Returns the associated layer, or nullptr if no layer is associated with the provider.
Flags mFlags
Flags altering drawing and registration of features.
virtual void startRender(QgsRenderContext &context)
To be called before rendering of labels begins.
Qgis::LabelPlacement mPlacement
Placement strategy.
@ MergeConnectedLines
Whether adjacent lines (with the same label text) should be merged.
@ DrawLabels
Whether the labels should be rendered.
@ CentroidMustBeInside
Whether location of centroid must be inside of polygons.
QString layerId() const
Returns ID of associated layer, or empty string if no layer is associated with the provider.
Qgis::UpsideDownLabelHandling mUpsidedownLabels
How to handle labels that would be upside down.
QString providerId() const
Returns provider ID - useful in case there is more than one label provider within a layer (e....
QgsAbstractLabelProvider(QgsMapLayer *layer, const QString &providerId=QString())
Construct the provider with default values.
QString mProviderId
Associated provider ID (one layer may have multiple providers, e.g. in rule-based labeling).
bool valueAsBool(int key, const QgsExpressionContext &context, bool defaultValue=false, bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as an boolean.
bool allFeaturePartsLabeled
true if all parts of associated feature were labeled
Definition qgscallout.h:251
QList< QgsCalloutPosition > positions() const
Returns the list of rendered callout positions.
Definition qgscallout.h:285
QgsCoordinateReferenceSystem originalFeatureCrs
Contains the CRS of the original feature associated with this callout.
Definition qgscallout.h:258
@ OrderBelowIndividualLabels
Render callouts below their individual associated labels, some callouts may be drawn over other label...
Definition qgscallout.h:109
@ OrderBelowAllLabels
Render callouts below all labels.
Definition qgscallout.h:108
bool enabled() const
Returns true if the callout is enabled.
Definition qgscallout.h:321
Represents a coordinate reference system (CRS).
Handles coordinate transforms between two coordinate systems.
void setBallparkTransformsAreAppropriate(bool appropriate)
Sets whether approximate "ballpark" results are appropriate for this coordinate transform.
void transformInPlace(double &x, double &y, double &z, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward) const
Transforms an array of x, y and z double coordinates in place, from the source CRS to the destination...
QgsRectangle transformBoundingBox(const QgsRectangle &rectangle, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward, bool handle180Crossover=false) const
Transforms a rectangle from the source CRS to the destination CRS.
bool isValid() const
Returns true if the coordinate transform is valid, ie both the source and destination CRS have been s...
Custom exception class for Coordinate Reference System related exceptions.
Single scope for storing variables and functions for use within a QgsExpressionContext.
static QgsExpressionContextScope * updateSymbolScope(const QgsSymbol *symbol, QgsExpressionContextScope *symbolScope=nullptr)
Updates a symbol scope related to a QgsSymbol to an expression context.
QgsExpressionContextScope * popScope()
Removes the last scope from the expression context and return it.
void setOriginalValueVariable(const QVariant &value)
Sets the original value variable value for the context.
QgsExpressionContextScope * lastScope()
Returns the last scope added to the context.
void appendScope(QgsExpressionContextScope *scope)
Appends a scope to the end of the context.
void setFeature(const QgsFeature &feature)
Convenience function for setting a feature for the context.
Wrapper for iterator of features from vector data provider or vector layer.
bool nextFeature(QgsFeature &f)
Fetch next feature and stores in f, returns true on success.
Wraps a request for features to a vector layer (or directly its vector data provider).
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
QgsFeatureRequest & setFilterRect(const QgsRectangle &rectangle)
Sets the rectangle from which features will be taken.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
QgsGeometry geometry
Definition qgsfeature.h:71
bool hasGeometry() const
Returns true if the feature has an associated geometry.
Container of fields for a vector layer.
Definition qgsfields.h:46
A geometry is the spatial representation of a feature.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
Qgis::GeometryType type
static std::unique_ptr< QgsAbstractGeometry > fromGeos(const GEOSGeometry *geos)
Create a geometry from a GEOSGeometry.
Definition qgsgeos.cpp:1562
Qgis::MultiPartLabelingBehavior multiPartBehavior() const
Returns the multipart labeling behavior.
QgsCoordinateReferenceSystem originalFeatureCrs() const
Returns the original layer CRS of the feature associated with the label.
GEOSGeometry * geometry() const
Gets access to the associated geometry.
bool hasFixedPosition() const
Whether the label should use a fixed position instead of being automatically placed.
QString labelText() const
Text of the label.
bool reverseDirectionSymbol() const
Returns true if direction symbols should be reversed.
DirectionSymbolPlacement directionSymbolPlacement() const
Returns the placement for direction symbols.
QString leftDirectionSymbol() const
Returns the string to use for left direction arrows.
@ SymbolLeftRight
Place direction symbols on left/right of label.
@ SymbolAbove
Place direction symbols on above label.
@ SymbolBelow
Place direction symbols on below label.
QString rightDirectionSymbol() const
Returns the string to use for right direction arrows.
bool addDirectionSymbol() const
Returns true if '<' or '>' (or custom strings set via leftDirectionSymbol and rightDirectionSymbol) w...
@ PolygonWhole
Avoid placing labels over ANY part of polygon. Where PolygonInterior will prefer to place labels with...
Base class for all map layer types.
Definition qgsmaplayer.h:83
Contains configuration for rendering maps.
Perform transforms between map coordinates and device coordinates.
void setMapRotation(double degrees, double cx, double cy)
Sets map rotation in degrees (clockwise).
QgsPointXY toMapCoordinates(int x, int y) const
Transforms device coordinates to map (world) coordinates.
QgsPointXY transform(const QgsPointXY &p) const
Transforms a point p from map (world) coordinates to device coordinates.
void transformInPlace(double &x, double &y) const
Transforms map coordinates to device coordinates.
Struct for storing maximum and minimum scales for measurements in map units.
A marker symbol type, for rendering Point and MultiPoint geometries.
Multi polygon geometry collection.
bool addGeometry(QgsAbstractGeometry *g) override
Adds a geometry and takes ownership. Returns true in case of success.
static QStringList splitToLines(const QString &text, const QString &wrapCharacter, int autoWrapLength=0, bool useMaxLineLengthWhenAutoWrapping=true)
Splits a text string to a list of separate lines, using a specified wrap character (wrapCharacter).
Contains settings for how a map layer will be labeled.
void setFormat(const QgsTextFormat &format)
Sets the label text formatting settings, e.g., font settings, buffer settings, etc.
QString wrapChar
Wrapping character string.
Qgis::LabelPlacement placement
Label placement mode.
QgsPropertyCollection & dataDefinedProperties()
Returns a reference to the label's property collection, used for data defined overrides.
Qgis::LabelMultiLineAlignment multilineAlign
Horizontal alignment of multi-line labels.
QgsCallout * callout() const
Returns the label callout renderer, responsible for drawing label callouts.
const QgsLabelLineSettings & lineSettings() const
Returns the label line settings, which contain settings related to how the label engine places and fo...
const QgsTextFormat & format() const
Returns the label text formatting settings, e.g., font settings, buffer settings, etc.
int autoWrapLength
If non-zero, indicates that label text should be automatically wrapped to (ideally) the specified num...
bool useMaxLineLengthForAutoWrap
If true, indicates that when auto wrapping label text the autoWrapLength length indicates the maximum...
Represents a 2D point.
Definition qgspointxy.h:62
double y
Definition qgspointxy.h:66
double x
Definition qgspointxy.h:65
QPointF toQPointF() const
Converts a point to a QPointF.
Definition qgspointxy.h:168
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:53
double x
Definition qgspoint.h:56
double y
Definition qgspoint.h:57
bool isActive(int key) const final
Returns true if the collection contains an active property with the specified key.
A rectangle specified with double values.
Contains information about the context of a rendering operation.
const QgsMaskIdProvider * maskIdProvider() const
Returns the mask id provider attached to the context.
double convertToPainterUnits(double size, Qgis::RenderUnit unit, const QgsMapUnitScale &scale=QgsMapUnitScale(), Qgis::RenderSubcomponentProperty property=Qgis::RenderSubcomponentProperty::Generic) const
Converts a size from the specified units to painter units (pixels).
QgsExpressionContext & expressionContext()
Gets the expression context.
const QgsRectangle & extent() const
When rendering a map layer, calling this method returns the "clipping" extent for the layer (in the l...
void setCurrentMaskId(int id)
Stores a mask id as the "current" one.
const QgsMapToPixel & mapToPixel() const
Returns the context's map to pixel transform, which transforms between map coordinates and device coo...
QgsCoordinateTransform coordinateTransform() const
Returns the current coordinate transform for the context.
Scoped object for temporary override of the symbologyReferenceScale property of a QgsRenderContext.
Abstract base class for all rendered symbols.
Definition qgssymbol.h:227
bool enabled() const
Returns whether the background is enabled.
ShapeType type() const
Returns the type of background shape (e.g., square, ellipse, SVG).
Represents a block of text consisting of one or more QgsTextFragment objects.
void insert(int index, const QgsTextFragment &fragment)
Inserts a fragment into the block, at the specified index.
static QgsTextBlock fromPlainText(const QString &text, const QgsTextCharacterFormat &format=QgsTextCharacterFormat())
Constructor for QgsTextBlock consisting of a plain text, and optional character format.
void append(const QgsTextFragment &fragment)
Appends a fragment to the block.
bool enabled() const
Returns whether the buffer is enabled.
Stores information relating to individual character formatting.
Contains pre-calculated metrics of a QgsTextDocument.
static QgsTextDocumentMetrics calculateMetrics(const QgsTextDocument &document, const QgsTextFormat &format, const QgsRenderContext &context, double scaleFactor=1.0, const QgsTextDocumentRenderContext &documentContext=QgsTextDocumentRenderContext())
Returns precalculated text metrics for a text document, when rendered using the given base format and...
double blockVerticalMargin(int blockIndex) const
Returns the vertical margin for the specified block index.
Represents a document consisting of one or more QgsTextBlock objects.
int size() const
Returns the number of blocks in the document.
void append(const QgsTextBlock &block)
Appends a block to the document.
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.
QgsTextBackgroundSettings & background()
Returns a reference to the text background settings.
QgsTextShadowSettings & shadow()
Returns a reference to the text drop shadow settings.
QFont font() const
Returns the font used for rendering text.
QgsTextBufferSettings & buffer()
Returns a reference to the text buffer settings.
Stores a fragment of document along with formatting overrides to be used when rendering the fragment.
Adds extra information to QgsLabelFeature for text labels.
QFont definedFont() const
Font to be used for rendering.
const QgsTextDocumentMetrics & documentMetrics() const
Returns the document metrics for the label.
const QgsTextDocument & document() const
Returns the document for the label.
QgsTextCharacterFormat characterFormat(int partId) const
Returns the character format corresponding to the specified label part.
const QMap< QgsPalLayerSettings::Property, QVariant > & dataDefinedValues() const
Gets data-defined values.
QString text(int partId) const
Returns the text component corresponding to a specified label part.
bool enabled() const
Returns whether the shadow is enabled.
void setShadowPlacement(QgsTextShadowSettings::ShadowPlacement placement)
Sets the placement for the drop shadow.
@ ShadowBuffer
Draw shadow under buffer.
@ ShadowShape
Draw shadow under background shape.
@ ShadowLowest
Draw shadow below all text components.
@ ShadowText
Draw shadow under text.
QgsTextShadowSettings::ShadowPlacement shadowPlacement() const
Returns the placement for the drop shadow.
void stopRender(QgsRenderContext &context) override
To be called after rendering is complete.
QgsCoordinateReferenceSystem mCrs
Layer's CRS.
virtual bool prepare(QgsRenderContext &context, QSet< QString > &attributeNames)
Prepare for registration of features.
void drawUnplacedLabel(QgsRenderContext &context, pal::LabelPosition *label) const override
Draw an unplaced label.
QList< QgsLabelFeature * > mLabels
List of generated.
QgsVectorLayerLabelProvider(QgsVectorLayer *layer, const QString &providerId, bool withFeatureLoop, const QgsPalLayerSettings *settings, const QString &layerName=QString())
Convenience constructor to initialize the provider from given vector layer.
void startRender(QgsRenderContext &context) override
To be called before rendering of labels begins.
const QgsPalLayerSettings & settings() const
Returns the layer's settings.
void drawLabelPrivate(pal::LabelPosition *label, QgsRenderContext &context, QgsPalLayerSettings &tmpLyr, Qgis::TextComponent drawType, double dpiRatio=1.0) const
Internal label drawing method.
Qgis::GeometryType mLayerGeometryType
Geometry type of layer.
static QgsGeometry getPointObstacleGeometry(QgsFeature &fet, QgsRenderContext &context, const QgsSymbolList &symbols)
Returns the geometry for a point feature which should be used as an obstacle for labels.
QgsPalLayerSettings mSettings
Layer's labeling configuration.
std::unique_ptr< QgsAbstractFeatureSource > mSource
Layer's feature source.
QList< QgsLabelFeature * > labelFeatures(QgsRenderContext &context) override
Returns list of label features (they are owned by the provider and thus deleted on its destruction).
virtual QList< QgsLabelFeature * > registerFeature(const QgsFeature &feature, QgsRenderContext &context, const QgsGeometry &obstacleGeometry=QgsGeometry(), const QgsSymbol *symbol=nullptr)
Register a feature for labeling as one or more QgsLabelFeature objects stored into mLabels.
void init()
initialization method - called from constructors
void setFields(const QgsFields &fields)
Sets fields of this label provider.
void drawLabel(QgsRenderContext &context, pal::LabelPosition *label) const override
Draw this label at the position determined by the labeling engine.
void drawLabelBackground(QgsRenderContext &context, pal::LabelPosition *label) const override
Draw the background for the specified label.
Represents a vector layer which manages a vector based dataset.
QgsFeatureId featureId() const
Returns the unique ID of the feature.
Definition feature.cpp:168
QgsLabelFeature * feature()
Returns the parent feature.
Definition feature.h:87
Layer * layer()
Returns the layer that feature belongs to.
Definition feature.cpp:163
LabelPosition is a candidate feature label position.
double getAlpha() const
Returns the angle to rotate text (in radians).
double getHeight() const
double getWidth() const
bool isReversedFromLineDirection() const
Returns true if the label direction is the reversed from the line or polygon ring direction.
Qgis::LabelQuadrantPosition quadrant() const
Returns the quadrant associated with this label position.
FeaturePart * getFeaturePart() const
Returns the feature corresponding to this labelposition.
int getPartId() const
double getX(int i=0) const
Returns the down-left x coordinate.
double getY(int i=0) const
Returns the down-left y coordinate.
LabelPosition * nextPart() const
Returns the next part of this label position (i.e.
QgsAbstractLabelProvider * provider() const
Returns pointer to the associated provider.
Definition layer.h:156
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
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63
QList< QgsSymbol * > QgsSymbolList
Definition qgsrenderer.h:51
Utility class for identifying a unique vertex within a geometry.
Definition qgsvertexid.h:34