QGIS API Documentation 3.28.0-Firenze (ed3ad0430f)
qgslabelinggui.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslabelinggui.cpp
3 Smart labeling for vector layers
4 -------------------
5 begin : June 2009
6 copyright : (C) Martin Dobias
7 email : wonder dot sk at gmail dot com
8
9 ***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
18#include "qgslabelinggui.h"
19#include "qgsvectorlayer.h"
20#include "qgsmapcanvas.h"
22#include "qgsproject.h"
23#include "qgsauxiliarystorage.h"
27#include "qgshelp.h"
28#include "qgsstylesavedialog.h"
29#include "qgscallout.h"
30#include "qgsapplication.h"
31#include "qgscalloutsregistry.h"
36
37#include <mutex>
38
39#include <QButtonGroup>
40#include <QMessageBox>
41
43
44QgsExpressionContext QgsLabelingGui::createExpressionContext() const
45{
46 QgsExpressionContext expContext;
50 if ( mCanvas )
51 expContext << QgsExpressionContextUtils::mapSettingsScope( mCanvas->mapSettings() );
52
53 if ( mLayer )
54 expContext << QgsExpressionContextUtils::layerScope( mLayer );
55
57
58 //TODO - show actual value
59 expContext.setOriginalValueVariable( QVariant() );
61
62 return expContext;
63}
64
65static bool _initCalloutWidgetFunction( const QString &name, QgsCalloutWidgetFunc f )
66{
68
69 QgsCalloutAbstractMetadata *abstractMetadata = registry->calloutMetadata( name );
70 if ( !abstractMetadata )
71 {
72 QgsDebugMsg( QStringLiteral( "Failed to find callout entry in registry: %1" ).arg( name ) );
73 return false;
74 }
75 QgsCalloutMetadata *metadata = dynamic_cast<QgsCalloutMetadata *>( abstractMetadata );
76 if ( !metadata )
77 {
78 QgsDebugMsg( QStringLiteral( "Failed to cast callout's metadata: " ) .arg( name ) );
79 return false;
80 }
81 metadata->setWidgetFunction( f );
82 return true;
83}
84
85void QgsLabelingGui::initCalloutWidgets()
86{
87 _initCalloutWidgetFunction( QStringLiteral( "simple" ), QgsSimpleLineCalloutWidget::create );
88 _initCalloutWidgetFunction( QStringLiteral( "manhattan" ), QgsManhattanLineCalloutWidget::create );
89 _initCalloutWidgetFunction( QStringLiteral( "curved" ), QgsCurvedLineCalloutWidget::create );
90 _initCalloutWidgetFunction( QStringLiteral( "balloon" ), QgsBalloonCalloutWidget::create );
91}
92
93void QgsLabelingGui::updateCalloutWidget( QgsCallout *callout )
94{
95 if ( !callout )
96 {
97 mCalloutStackedWidget->setCurrentWidget( pageDummy );
98 return;
99 }
100
101 if ( mCalloutStackedWidget->currentWidget() != pageDummy )
102 {
103 // stop updating from the original widget
104 if ( QgsCalloutWidget *pew = qobject_cast< QgsCalloutWidget * >( mCalloutStackedWidget->currentWidget() ) )
105 disconnect( pew, &QgsCalloutWidget::changed, this, &QgsLabelingGui::updatePreview );
106 }
107
109 if ( QgsCalloutAbstractMetadata *am = registry->calloutMetadata( callout->type() ) )
110 {
111 if ( QgsCalloutWidget *w = am->createCalloutWidget( mLayer ) )
112 {
113
114 QgsWkbTypes::GeometryType geometryType = mGeomType;
115 if ( mGeometryGeneratorGroupBox->isChecked() )
116 geometryType = mGeometryGeneratorType->currentData().value<QgsWkbTypes::GeometryType>();
117 else if ( mLayer )
118 geometryType = mLayer->geometryType();
119 w->setGeometryType( geometryType );
120 w->setCallout( callout );
121
122 w->setContext( context() );
123 mCalloutStackedWidget->addWidget( w );
124 mCalloutStackedWidget->setCurrentWidget( w );
125 // start receiving updates from widget
126 connect( w, &QgsCalloutWidget::changed, this, &QgsLabelingGui::updatePreview );
127 return;
128 }
129 }
130 // When anything is not right
131 mCalloutStackedWidget->setCurrentWidget( pageDummy );
132}
133
134void QgsLabelingGui::showObstacleSettings()
135{
136 QgsExpressionContext context = createExpressionContext();
137
138 QgsSymbolWidgetContext symbolContext;
139 symbolContext.setExpressionContext( &context );
140 symbolContext.setMapCanvas( mMapCanvas );
141
142 QgsLabelObstacleSettingsWidget *widget = new QgsLabelObstacleSettingsWidget( nullptr, mLayer );
143 widget->setDataDefinedProperties( mDataDefinedProperties );
144 widget->setSettings( mObstacleSettings );
145 widget->setGeometryType( mLayer ? mLayer->geometryType() : QgsWkbTypes::UnknownGeometry );
146 widget->setContext( symbolContext );
147
148 auto applySettings = [ = ]
149 {
150 mObstacleSettings = widget->settings();
151 const QgsPropertyCollection obstacleDataDefinedProperties = widget->dataDefinedProperties();
152 widget->updateDataDefinedProperties( mDataDefinedProperties );
153 emit widgetChanged();
154 };
155
157 if ( panel && panel->dockMode() )
158 {
159 connect( widget, &QgsLabelSettingsWidgetBase::changed, this, [ = ]
160 {
161 applySettings();
162 } );
163 panel->openPanel( widget );
164 }
165 else
166 {
167 QgsLabelSettingsWidgetDialog dialog( widget, this );
168
169 dialog.buttonBox()->addButton( QDialogButtonBox::Help );
170 connect( dialog.buttonBox(), &QDialogButtonBox::helpRequested, this, [ = ]
171 {
172 QgsHelp::openHelp( QStringLiteral( "style_library/label_settings.html#obstacles" ) );
173 } );
174
175 if ( dialog.exec() )
176 {
177 applySettings();
178 }
179 // reactivate button's window
180 activateWindow();
181 }
182}
183
184void QgsLabelingGui::showLineAnchorSettings()
185{
186 QgsExpressionContext context = createExpressionContext();
187
188 QgsSymbolWidgetContext symbolContext;
189 symbolContext.setExpressionContext( &context );
190 symbolContext.setMapCanvas( mMapCanvas );
191
192 QgsLabelLineAnchorWidget *widget = new QgsLabelLineAnchorWidget( nullptr, mLayer );
193 widget->setDataDefinedProperties( mDataDefinedProperties );
194 widget->setSettings( mLineSettings );
195 widget->setGeometryType( mLayer ? mLayer->geometryType() : QgsWkbTypes::UnknownGeometry );
196 widget->setContext( symbolContext );
197
198 auto applySettings = [ = ]
199 {
200 const QgsLabelLineSettings widgetSettings = widget->settings();
201 mLineSettings.setLineAnchorPercent( widgetSettings.lineAnchorPercent() );
202 mLineSettings.setAnchorType( widgetSettings.anchorType() );
203 mLineSettings.setAnchorClipping( widgetSettings.anchorClipping() );
204 mLineSettings.setAnchorTextPoint( widgetSettings.anchorTextPoint() );
205 const QgsPropertyCollection obstacleDataDefinedProperties = widget->dataDefinedProperties();
206 widget->updateDataDefinedProperties( mDataDefinedProperties );
207 emit widgetChanged();
208 };
209
211 if ( panel && panel->dockMode() )
212 {
213 connect( widget, &QgsLabelSettingsWidgetBase::changed, this, [ = ]
214 {
215 applySettings();
216 } );
217 panel->openPanel( widget );
218 }
219 else
220 {
221 QgsLabelSettingsWidgetDialog dialog( widget, this );
222
223 dialog.buttonBox()->addButton( QDialogButtonBox::Help );
224 connect( dialog.buttonBox(), &QDialogButtonBox::helpRequested, this, [ = ]
225 {
226 QgsHelp::openHelp( QStringLiteral( "style_library/label_settings.html#placement-for-line-layers" ) );
227 } );
228
229 if ( dialog.exec() )
230 {
231 applySettings();
232 }
233 // reactivate button's window
234 activateWindow();
235 }
236}
237
238QgsLabelingGui::QgsLabelingGui( QgsVectorLayer *layer, QgsMapCanvas *mapCanvas, const QgsPalLayerSettings &layerSettings, QWidget *parent, QgsWkbTypes::GeometryType geomType )
239 : QgsTextFormatWidget( mapCanvas, parent, QgsTextFormatWidget::Labeling, layer )
240 , mSettings( layerSettings )
241 , mMode( NoLabels )
242 , mCanvas( mapCanvas )
243{
244 mGeomType = geomType;
245 static std::once_flag initialized;
246 std::call_once( initialized, [ = ]( )
247 {
248 initCalloutWidgets();
249 } );
250
251 mFontMultiLineAlignComboBox->addItem( tr( "Left" ), static_cast< int >( Qgis::LabelMultiLineAlignment::Left ) );
252 mFontMultiLineAlignComboBox->addItem( tr( "Center" ), static_cast< int >( Qgis::LabelMultiLineAlignment::Center ) );
253 mFontMultiLineAlignComboBox->addItem( tr( "Right" ), static_cast< int >( Qgis::LabelMultiLineAlignment::Right ) );
254 mFontMultiLineAlignComboBox->addItem( tr( "Justify" ), static_cast< int >( Qgis::LabelMultiLineAlignment::Justify ) );
255
256 mCoordRotationUnitComboBox->addItem( QgsUnitTypes::toString( QgsUnitTypes::AngleDegrees ), QgsUnitTypes::AngleDegrees );
257 mCoordRotationUnitComboBox->addItem( QgsUnitTypes::toString( QgsUnitTypes::AngleRadians ), QgsUnitTypes::AngleRadians );
258 mCoordRotationUnitComboBox->addItem( QgsUnitTypes::toString( QgsUnitTypes::AngleGon ), QgsUnitTypes::AngleGon );
261 mCoordRotationUnitComboBox->addItem( QgsUnitTypes::toString( QgsUnitTypes::AngleTurn ), QgsUnitTypes::AngleTurn );
263 mCoordRotationUnitComboBox->addItem( QgsUnitTypes::toString( QgsUnitTypes::AngleMilNATO ), QgsUnitTypes::AngleMilNATO );
264
265 // connections for groupboxes with separate activation checkboxes (that need to honor data defined setting)
266 connect( mBufferDrawChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
267 connect( mBufferDrawDDBtn, &QgsPropertyOverrideButton::changed, this, &QgsLabelingGui::updateUi );
268 connect( mEnableMaskChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
269 connect( mShapeDrawChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
270 connect( mCalloutsDrawCheckBox, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
271 connect( mShadowDrawChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
272 connect( mDirectSymbChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
273 connect( mFormatNumChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
274 connect( mScaleBasedVisibilityChkBx, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
275 connect( mFontLimitPixelChkBox, &QAbstractButton::toggled, this, &QgsLabelingGui::updateUi );
276 connect( mGeometryGeneratorGroupBox, &QGroupBox::toggled, this, &QgsLabelingGui::updateGeometryTypeBasedWidgets );
277 connect( mGeometryGeneratorType, qOverload<int>( &QComboBox::currentIndexChanged ), this, &QgsLabelingGui::updateGeometryTypeBasedWidgets );
278 connect( mGeometryGeneratorExpressionButton, &QToolButton::clicked, this, &QgsLabelingGui::showGeometryGeneratorExpressionBuilder );
279 connect( mGeometryGeneratorGroupBox, &QGroupBox::toggled, this, &QgsLabelingGui::validateGeometryGeneratorExpression );
280 connect( mGeometryGenerator, &QgsCodeEditorExpression::textChanged, this, &QgsLabelingGui::validateGeometryGeneratorExpression );
281 connect( mGeometryGeneratorType, qOverload<int>( &QComboBox::currentIndexChanged ), this, &QgsLabelingGui::validateGeometryGeneratorExpression );
282 connect( mObstacleSettingsButton, &QAbstractButton::clicked, this, &QgsLabelingGui::showObstacleSettings );
283 connect( mLineAnchorSettingsButton, &QAbstractButton::clicked, this, &QgsLabelingGui::showLineAnchorSettings );
284
285 mFieldExpressionWidget->registerExpressionContextGenerator( this );
286
287 mMinScaleWidget->setMapCanvas( mCanvas );
288 mMinScaleWidget->setShowCurrentScaleButton( true );
289 mMaxScaleWidget->setMapCanvas( mCanvas );
290 mMaxScaleWidget->setShowCurrentScaleButton( true );
291
292 const QStringList calloutTypes = QgsApplication::calloutRegistry()->calloutTypes();
293 for ( const QString &type : calloutTypes )
294 {
295 mCalloutStyleComboBox->addItem( QgsApplication::calloutRegistry()->calloutMetadata( type )->icon(),
296 QgsApplication::calloutRegistry()->calloutMetadata( type )->visibleName(), type );
297 }
298
299 mGeometryGeneratorWarningLabel->setStyleSheet( QStringLiteral( "color: #FFC107;" ) );
300 mGeometryGeneratorWarningLabel->setTextInteractionFlags( Qt::TextBrowserInteraction );
301 connect( mGeometryGeneratorWarningLabel, &QLabel::linkActivated, this, [this]( const QString & link )
302 {
303 if ( link == QLatin1String( "#determineGeometryGeneratorType" ) )
304 determineGeometryGeneratorType();
305 } );
306
307 connect( mCalloutStyleComboBox, static_cast<void ( QComboBox::* )( int )>( &QComboBox::currentIndexChanged ), this, &QgsLabelingGui::calloutTypeChanged );
308
309 mLblNoObstacle1->installEventFilter( this );
310
311 setLayer( layer );
312}
313
314void QgsLabelingGui::setLayer( QgsMapLayer *mapLayer )
315{
316 mPreviewFeature = QgsFeature();
317
318 if ( ( !mapLayer || mapLayer->type() != QgsMapLayerType::VectorLayer ) && mGeomType == QgsWkbTypes::UnknownGeometry )
319 {
320 setEnabled( false );
321 return;
322 }
323
324 setEnabled( true );
325
326 QgsVectorLayer *layer = static_cast<QgsVectorLayer *>( mapLayer );
327 mLayer = layer;
328
329 mTextFormatsListWidget->setLayerType( mLayer ? mLayer->geometryType() : mGeomType );
330 mBackgroundMarkerSymbolButton->setLayer( mLayer );
331 mBackgroundFillSymbolButton->setLayer( mLayer );
332
333 // load labeling settings from layer
334 updateGeometryTypeBasedWidgets();
335
336 mFieldExpressionWidget->setLayer( mLayer );
338 if ( mLayer )
339 da.setSourceCrs( mLayer->crs(), QgsProject::instance()->transformContext() );
340 da.setEllipsoid( QgsProject::instance()->ellipsoid() );
341 mFieldExpressionWidget->setGeomCalculator( da );
342
343 mFieldExpressionWidget->setEnabled( mMode == Labels || !mLayer );
344 mLabelingFrame->setEnabled( mMode == Labels || !mLayer );
345
346 blockInitSignals( true );
347
348 mGeometryGenerator->setText( mSettings.geometryGenerator );
349 mGeometryGeneratorGroupBox->setChecked( mSettings.geometryGeneratorEnabled );
350 if ( !mSettings.geometryGeneratorEnabled )
351 mGeometryGeneratorGroupBox->setCollapsed( true );
352 mGeometryGeneratorType->setCurrentIndex( mGeometryGeneratorType->findData( mSettings.geometryGeneratorType ) );
353
354 updateWidgetForFormat( mSettings.format().isValid() ? mSettings.format() : QgsStyle::defaultTextFormatForProject( QgsProject::instance(), QgsStyle::TextFormatContext::Labeling ) );
355
356 mFieldExpressionWidget->setRow( -1 );
357 mFieldExpressionWidget->setField( mSettings.fieldName );
358 mCheckBoxSubstituteText->setChecked( mSettings.useSubstitutions );
359 mSubstitutions = mSettings.substitutions;
360
361 // populate placement options
362 mCentroidRadioWhole->setChecked( mSettings.centroidWhole );
363 mCentroidInsideCheckBox->setChecked( mSettings.centroidInside );
364 mFitInsidePolygonCheckBox->setChecked( mSettings.fitInPolygonOnly );
365 mLineDistanceSpnBx->setValue( mSettings.dist );
366 mLineDistanceUnitWidget->setUnit( mSettings.distUnits );
367 mLineDistanceUnitWidget->setMapUnitScale( mSettings.distMapUnitScale );
368 mOffsetTypeComboBox->setCurrentIndex( mOffsetTypeComboBox->findData( static_cast< int >( mSettings.offsetType ) ) );
369 mQuadrantBtnGrp->button( static_cast<int>( mSettings.quadOffset ) )->setChecked( true );
370 mPointOffsetXSpinBox->setValue( mSettings.xOffset );
371 mPointOffsetYSpinBox->setValue( mSettings.yOffset );
372 mPointOffsetUnitWidget->setUnit( mSettings.offsetUnits );
373 mPointOffsetUnitWidget->setMapUnitScale( mSettings.labelOffsetMapUnitScale );
374 mPointAngleSpinBox->setValue( mSettings.angleOffset );
375 chkLineAbove->setChecked( mSettings.lineSettings().placementFlags() & QgsLabeling::LinePlacementFlag::AboveLine );
376 chkLineBelow->setChecked( mSettings.lineSettings().placementFlags() & QgsLabeling::LinePlacementFlag::BelowLine );
377 chkLineOn->setChecked( mSettings.lineSettings().placementFlags() & QgsLabeling::LinePlacementFlag::OnLine );
378 chkLineOrientationDependent->setChecked( !( mSettings.lineSettings().placementFlags() & QgsLabeling::LinePlacementFlag::MapOrientation ) );
379
380 mCheckAllowLabelsOutsidePolygons->setChecked( mSettings.polygonPlacementFlags() & QgsLabeling::PolygonPlacementFlag::AllowPlacementOutsideOfPolygon );
381
382 const int placementIndex = mPlacementModeComboBox->findData( static_cast< int >( mSettings.placement ) );
383 if ( placementIndex >= 0 )
384 {
385 mPlacementModeComboBox->setCurrentIndex( placementIndex );
386 }
387 else
388 {
389 // use default placement for layer type
390 mPlacementModeComboBox->setCurrentIndex( 0 );
391 }
392
393 // Label repeat distance
394 mRepeatDistanceSpinBox->setValue( mSettings.repeatDistance );
395 mRepeatDistanceUnitWidget->setUnit( mSettings.repeatDistanceUnit );
396 mRepeatDistanceUnitWidget->setMapUnitScale( mSettings.repeatDistanceMapUnitScale );
397
398 mOverrunDistanceSpinBox->setValue( mSettings.lineSettings().overrunDistance() );
399 mOverrunDistanceUnitWidget->setUnit( mSettings.lineSettings().overrunDistanceUnit() );
400 mOverrunDistanceUnitWidget->setMapUnitScale( mSettings.lineSettings().overrunDistanceMapUnitScale() );
401
402 mPrioritySlider->setValue( mSettings.priority );
403 mChkNoObstacle->setChecked( mSettings.obstacleSettings().isObstacle() );
404
405 mObstacleSettings = mSettings.obstacleSettings();
406 mLineSettings = mSettings.lineSettings();
407
408 chkLabelPerFeaturePart->setChecked( mSettings.labelPerPart );
409
410 mComboOverlapHandling->setCurrentIndex( mComboOverlapHandling->findData( static_cast< int >( mSettings.placementSettings().overlapHandling() ) ) );
411 mCheckAllowDegradedPlacement->setChecked( mSettings.placementSettings().allowDegradedPlacement() );
412
413 chkMergeLines->setChecked( mSettings.lineSettings().mergeLines() );
414 mMinSizeSpinBox->setValue( mSettings.thinningSettings().minimumFeatureSize() );
415 mLimitLabelChkBox->setChecked( mSettings.thinningSettings().limitNumberOfLabelsEnabled() );
416 mLimitLabelSpinBox->setValue( mSettings.thinningSettings().maximumNumberLabels() );
417
418 // direction symbol(s)
419 mDirectSymbChkBx->setChecked( mSettings.lineSettings().addDirectionSymbol() );
420 mDirectSymbLeftLineEdit->setText( mSettings.lineSettings().leftDirectionSymbol() );
421 mDirectSymbRightLineEdit->setText( mSettings.lineSettings().rightDirectionSymbol() );
422 mDirectSymbRevChkBx->setChecked( mSettings.lineSettings().reverseDirectionSymbol() );
423
424 mDirectSymbBtnGrp->button( static_cast<int>( mSettings.lineSettings().directionSymbolPlacement() ) )->setChecked( true );
425 mUpsidedownBtnGrp->button( static_cast<int>( mSettings.upsidedownLabels ) )->setChecked( true );
426
427 // curved label max character angles
428 mMaxCharAngleInDSpinBox->setValue( mSettings.maxCurvedCharAngleIn );
429 // lyr.maxCurvedCharAngleOut must be negative, but it is shown as positive spinbox in GUI
430 mMaxCharAngleOutDSpinBox->setValue( std::fabs( mSettings.maxCurvedCharAngleOut ) );
431
432 wrapCharacterEdit->setText( mSettings.wrapChar );
433 mAutoWrapLengthSpinBox->setValue( mSettings.autoWrapLength );
434 mAutoWrapTypeComboBox->setCurrentIndex( mSettings.useMaxLineLengthForAutoWrap ? 0 : 1 );
435
436 if ( mFontMultiLineAlignComboBox->findData( static_cast< int >( mSettings.multilineAlign ) ) != -1 )
437 {
438 mFontMultiLineAlignComboBox->setCurrentIndex( mFontMultiLineAlignComboBox->findData( static_cast< int >( mSettings.multilineAlign ) ) );
439 }
440 else
441 {
442 // the default pal layer settings for multiline alignment is to follow label placement, which isn't always available
443 // revert to left alignment in such case
444 mFontMultiLineAlignComboBox->setCurrentIndex( 0 );
445 }
446
447 chkPreserveRotation->setChecked( mSettings.preserveRotation );
448
449 mCoordRotationUnitComboBox->setCurrentIndex( 0 );
450 if ( mCoordRotationUnitComboBox->findData( static_cast< unsigned int >( mSettings.rotationUnit() ) ) >= 0 )
451 mCoordRotationUnitComboBox->setCurrentIndex( mCoordRotationUnitComboBox->findData( static_cast< unsigned int >( mSettings.rotationUnit() ) ) );
452
453 mScaleBasedVisibilityChkBx->setChecked( mSettings.scaleVisibility );
454 mMinScaleWidget->setScale( mSettings.minimumScale );
455 mMaxScaleWidget->setScale( mSettings.maximumScale );
456
457 mFormatNumChkBx->setChecked( mSettings.formatNumbers );
458 mFormatNumDecimalsSpnBx->setValue( mSettings.decimals );
459 mFormatNumPlusSignChkBx->setChecked( mSettings.plusSign );
460
461 // set pixel size limiting checked state before unit choice so limiting can be
462 // turned on as a default for map units, if minimum trigger value of 0 is used
463 mFontLimitPixelChkBox->setChecked( mSettings.fontLimitPixelSize );
464 mMinPixelLimit = mSettings.fontMinPixelSize; // ignored after first settings save
465 mFontMinPixelSpinBox->setValue( mSettings.fontMinPixelSize == 0 ? 3 : mSettings.fontMinPixelSize );
466 mFontMaxPixelSpinBox->setValue( mSettings.fontMaxPixelSize );
467
468 mZIndexSpinBox->setValue( mSettings.zIndex );
469
470 mDataDefinedProperties = mSettings.dataDefinedProperties();
471
472 // callout settings, to move to custom widget when multiple styles exist
473 if ( auto *lCallout = mSettings.callout() )
474 {
475 whileBlocking( mCalloutsDrawCheckBox )->setChecked( lCallout->enabled() );
476 whileBlocking( mCalloutStyleComboBox )->setCurrentIndex( mCalloutStyleComboBox->findData( lCallout->type() ) );
477 updateCalloutWidget( lCallout );
478 }
479 else
480 {
481 std::unique_ptr< QgsCallout > defaultCallout( QgsCalloutRegistry::defaultCallout() );
482 whileBlocking( mCalloutStyleComboBox )->setCurrentIndex( mCalloutStyleComboBox->findData( defaultCallout->type() ) );
483 whileBlocking( mCalloutsDrawCheckBox )->setChecked( false );
484 updateCalloutWidget( defaultCallout.get() );
485 }
486
487 updatePlacementWidgets();
488 updateLinePlacementOptions();
489
490 // needs to come before data defined setup, so connections work
491 blockInitSignals( false );
492
493 // set up data defined toolbuttons
494 // do this after other widgets are configured, so they can be enabled/disabled
495 populateDataDefinedButtons();
496
497 updateUi(); // should come after data defined button setup
498}
499
500void QgsLabelingGui::setSettings( const QgsPalLayerSettings &settings )
501{
502 mSettings = settings;
503 setLayer( mLayer );
504}
505
506void QgsLabelingGui::blockInitSignals( bool block )
507{
508 chkLineAbove->blockSignals( block );
509 chkLineBelow->blockSignals( block );
510 mPlacementModeComboBox->blockSignals( block );
511}
512
513void QgsLabelingGui::setLabelMode( LabelMode mode )
514{
515 mMode = mode;
516 mFieldExpressionWidget->setEnabled( mMode == Labels );
517 mLabelingFrame->setEnabled( mMode == Labels );
518}
519
520QgsPalLayerSettings QgsLabelingGui::layerSettings()
521{
523
524 // restore properties which aren't exposed in GUI
525 lyr.setUnplacedVisibility( mSettings.unplacedVisibility() );
526
527 lyr.drawLabels = ( mMode == Labels ) || !mLayer;
528
529 bool isExpression;
530 lyr.fieldName = mFieldExpressionWidget->currentField( &isExpression );
531 lyr.isExpression = isExpression;
532
533 lyr.dist = 0;
534
535 QgsLabeling::PolygonPlacementFlags polygonPlacementFlags = QgsLabeling::PolygonPlacementFlag::AllowPlacementInsideOfPolygon;
536 if ( mCheckAllowLabelsOutsidePolygons->isChecked() )
537 polygonPlacementFlags |= QgsLabeling::PolygonPlacementFlag::AllowPlacementOutsideOfPolygon;
538 lyr.setPolygonPlacementFlags( polygonPlacementFlags );
539
540 lyr.centroidWhole = mCentroidRadioWhole->isChecked();
541 lyr.centroidInside = mCentroidInsideCheckBox->isChecked();
542 lyr.fitInPolygonOnly = mFitInsidePolygonCheckBox->isChecked();
543 lyr.dist = mLineDistanceSpnBx->value();
544 lyr.distUnits = mLineDistanceUnitWidget->unit();
545 lyr.distMapUnitScale = mLineDistanceUnitWidget->getMapUnitScale();
546 lyr.offsetType = static_cast< Qgis::LabelOffsetType >( mOffsetTypeComboBox->currentData().toInt() );
547 if ( mQuadrantBtnGrp )
548 {
549 lyr.quadOffset = static_cast< Qgis::LabelQuadrantPosition >( mQuadrantBtnGrp->checkedId() );
550 }
551 lyr.xOffset = mPointOffsetXSpinBox->value();
552 lyr.yOffset = mPointOffsetYSpinBox->value();
553 lyr.offsetUnits = mPointOffsetUnitWidget->unit();
554 lyr.labelOffsetMapUnitScale = mPointOffsetUnitWidget->getMapUnitScale();
555 lyr.angleOffset = mPointAngleSpinBox->value();
556
557 QgsLabeling::LinePlacementFlags linePlacementFlags = QgsLabeling::LinePlacementFlags();
558 if ( chkLineAbove->isChecked() )
559 linePlacementFlags |= QgsLabeling::LinePlacementFlag::AboveLine;
560 if ( chkLineBelow->isChecked() )
561 linePlacementFlags |= QgsLabeling::LinePlacementFlag::BelowLine;
562 if ( chkLineOn->isChecked() )
563 linePlacementFlags |= QgsLabeling::LinePlacementFlag::OnLine;
564 if ( ! chkLineOrientationDependent->isChecked() )
565 linePlacementFlags |= QgsLabeling::LinePlacementFlag::MapOrientation;
566 lyr.lineSettings().setPlacementFlags( linePlacementFlags );
567
568 lyr.placement = static_cast< Qgis::LabelPlacement >( mPlacementModeComboBox->currentData().toInt() );
569
570 lyr.repeatDistance = mRepeatDistanceSpinBox->value();
571 lyr.repeatDistanceUnit = mRepeatDistanceUnitWidget->unit();
572 lyr.repeatDistanceMapUnitScale = mRepeatDistanceUnitWidget->getMapUnitScale();
573
574 lyr.lineSettings().setOverrunDistance( mOverrunDistanceSpinBox->value() );
575 lyr.lineSettings().setOverrunDistanceUnit( mOverrunDistanceUnitWidget->unit() );
576 lyr.lineSettings().setOverrunDistanceMapUnitScale( mOverrunDistanceUnitWidget->getMapUnitScale() );
577
578 lyr.priority = mPrioritySlider->value();
579
580 mObstacleSettings.setIsObstacle( mChkNoObstacle->isChecked() || mMode == ObstaclesOnly );
581 lyr.setObstacleSettings( mObstacleSettings );
582
583 lyr.lineSettings().setLineAnchorPercent( mLineSettings.lineAnchorPercent() );
584 lyr.lineSettings().setAnchorType( mLineSettings.anchorType() );
585 lyr.lineSettings().setAnchorClipping( mLineSettings.anchorClipping() );
586 lyr.lineSettings().setAnchorTextPoint( mLineSettings.anchorTextPoint() );
587
588 lyr.labelPerPart = chkLabelPerFeaturePart->isChecked();
589 lyr.placementSettings().setOverlapHandling( static_cast< Qgis::LabelOverlapHandling>( mComboOverlapHandling->currentData().toInt() ) );
590 lyr.placementSettings().setAllowDegradedPlacement( mCheckAllowDegradedPlacement->isChecked() );
591
592 lyr.lineSettings().setMergeLines( chkMergeLines->isChecked() );
593
594 lyr.scaleVisibility = mScaleBasedVisibilityChkBx->isChecked();
595 lyr.minimumScale = mMinScaleWidget->scale();
596 lyr.maximumScale = mMaxScaleWidget->scale();
597 lyr.useSubstitutions = mCheckBoxSubstituteText->isChecked();
598 lyr.substitutions = mSubstitutions;
599
600 lyr.setFormat( format( false ) );
601
602 // format numbers
603 lyr.formatNumbers = mFormatNumChkBx->isChecked();
604 lyr.decimals = mFormatNumDecimalsSpnBx->value();
605 lyr.plusSign = mFormatNumPlusSignChkBx->isChecked();
606
607 // direction symbol(s)
608 lyr.lineSettings().setAddDirectionSymbol( mDirectSymbChkBx->isChecked() );
609 lyr.lineSettings().setLeftDirectionSymbol( mDirectSymbLeftLineEdit->text() );
610 lyr.lineSettings().setRightDirectionSymbol( mDirectSymbRightLineEdit->text() );
611 lyr.lineSettings().setReverseDirectionSymbol( mDirectSymbRevChkBx->isChecked() );
612 if ( mDirectSymbBtnGrp )
613 {
614 lyr.lineSettings().setDirectionSymbolPlacement( static_cast< QgsLabelLineSettings::DirectionSymbolPlacement >( mDirectSymbBtnGrp->checkedId() ) );
615 }
616 if ( mUpsidedownBtnGrp )
617 {
618 lyr.upsidedownLabels = static_cast< Qgis::UpsideDownLabelHandling >( mUpsidedownBtnGrp->checkedId() );
619 }
620
621 lyr.maxCurvedCharAngleIn = mMaxCharAngleInDSpinBox->value();
622 // lyr.maxCurvedCharAngleOut must be negative, but it is shown as positive spinbox in GUI
623 lyr.maxCurvedCharAngleOut = -mMaxCharAngleOutDSpinBox->value();
624
625
626 lyr.thinningSettings().setMinimumFeatureSize( mMinSizeSpinBox->value() );
627 lyr.thinningSettings().setLimitNumberLabelsEnabled( mLimitLabelChkBox->isChecked() );
628 lyr.thinningSettings().setMaximumNumberLabels( mLimitLabelSpinBox->value() );
629 lyr.fontLimitPixelSize = mFontLimitPixelChkBox->isChecked();
630 lyr.fontMinPixelSize = mFontMinPixelSpinBox->value();
631 lyr.fontMaxPixelSize = mFontMaxPixelSpinBox->value();
632 lyr.wrapChar = wrapCharacterEdit->text();
633 lyr.autoWrapLength = mAutoWrapLengthSpinBox->value();
634 lyr.useMaxLineLengthForAutoWrap = mAutoWrapTypeComboBox->currentIndex() == 0;
635 lyr.multilineAlign = static_cast< Qgis::LabelMultiLineAlignment >( mFontMultiLineAlignComboBox->currentData().toInt() );
636 lyr.preserveRotation = chkPreserveRotation->isChecked();
637 lyr.setRotationUnit( static_cast< QgsUnitTypes::AngleUnit >( mCoordRotationUnitComboBox->currentData().toInt() ) );
638 lyr.geometryGenerator = mGeometryGenerator->text();
639 lyr.geometryGeneratorType = mGeometryGeneratorType->currentData().value<QgsWkbTypes::GeometryType>();
640 lyr.geometryGeneratorEnabled = mGeometryGeneratorGroupBox->isChecked();
641
642 lyr.layerType = mLayer ? mLayer->geometryType() : mGeomType;
643
644 lyr.zIndex = mZIndexSpinBox->value();
645
646 lyr.setDataDefinedProperties( mDataDefinedProperties );
647
648 // callout settings
649 const QString calloutType = mCalloutStyleComboBox->currentData().toString();
650 std::unique_ptr< QgsCallout > callout;
651 if ( QgsCalloutWidget *pew = qobject_cast< QgsCalloutWidget * >( mCalloutStackedWidget->currentWidget() ) )
652 {
653 callout.reset( pew->callout()->clone() );
654 }
655 if ( !callout )
656 callout.reset( QgsApplication::calloutRegistry()->createCallout( calloutType ) );
657
658 callout->setEnabled( mCalloutsDrawCheckBox->isChecked() );
659 lyr.setCallout( callout.release() );
660
661 return lyr;
662}
663
664void QgsLabelingGui::syncDefinedCheckboxFrame( QgsPropertyOverrideButton *ddBtn, QCheckBox *chkBx, QFrame *f )
665{
666 f->setEnabled( chkBx->isChecked() || ddBtn->isActive() );
667}
668
669bool QgsLabelingGui::eventFilter( QObject *object, QEvent *event )
670{
671 if ( object == mLblNoObstacle1 )
672 {
673 if ( event->type() == QEvent::MouseButtonPress && qgis::down_cast< QMouseEvent * >( event )->button() == Qt::LeftButton )
674 {
675 // clicking the obstacle label toggles the checkbox, just like a "normal" checkbox label...
676 mChkNoObstacle->setChecked( !mChkNoObstacle->isChecked() );
677 return true;
678 }
679 return false;
680 }
681 return QgsTextFormatWidget::eventFilter( object, event );
682}
683
684void QgsLabelingGui::updateUi()
685{
686 // enable/disable inline groupbox-like setups (that need to honor data defined setting)
687
688 syncDefinedCheckboxFrame( mBufferDrawDDBtn, mBufferDrawChkBx, mBufferFrame );
689 syncDefinedCheckboxFrame( mEnableMaskDDBtn, mEnableMaskChkBx, mMaskFrame );
690 syncDefinedCheckboxFrame( mShapeDrawDDBtn, mShapeDrawChkBx, mShapeFrame );
691 syncDefinedCheckboxFrame( mShadowDrawDDBtn, mShadowDrawChkBx, mShadowFrame );
692 syncDefinedCheckboxFrame( mCalloutDrawDDBtn, mCalloutsDrawCheckBox, mCalloutFrame );
693
694 syncDefinedCheckboxFrame( mDirectSymbDDBtn, mDirectSymbChkBx, mDirectSymbFrame );
695 syncDefinedCheckboxFrame( mFormatNumDDBtn, mFormatNumChkBx, mFormatNumFrame );
696 syncDefinedCheckboxFrame( mScaleBasedVisibilityDDBtn, mScaleBasedVisibilityChkBx, mScaleBasedVisibilityFrame );
697 syncDefinedCheckboxFrame( mFontLimitPixelDDBtn, mFontLimitPixelChkBox, mFontLimitPixelFrame );
698
699 chkMergeLines->setEnabled( !mDirectSymbChkBx->isChecked() );
700 if ( mDirectSymbChkBx->isChecked() )
701 {
702 chkMergeLines->setToolTip( tr( "This option is not compatible with line direction symbols." ) );
703 }
704 else
705 {
706 chkMergeLines->setToolTip( QString() );
707 }
708}
709
710void QgsLabelingGui::setFormatFromStyle( const QString &name, QgsStyle::StyleEntity type, const QString &stylePath )
711{
712 QgsStyle *style = QgsProject::instance()->styleSettings()->styleAtPath( stylePath );
713
714 if ( !style )
715 style = QgsStyle::defaultStyle();
716
717 switch ( type )
718 {
726 {
727 QgsTextFormatWidget::setFormatFromStyle( name, type, stylePath );
728 return;
729 }
730
732 {
733 if ( !style->labelSettingsNames().contains( name ) )
734 return;
735
736 QgsPalLayerSettings settings = style->labelSettings( name );
737 if ( settings.fieldName.isEmpty() )
738 {
739 // if saved settings doesn't have a field name stored, retain the current one
740 bool isExpression;
741 settings.fieldName = mFieldExpressionWidget->currentField( &isExpression );
742 settings.isExpression = isExpression;
743 }
744 setSettings( settings );
745 break;
746 }
747 }
748}
749
750void QgsLabelingGui::setContext( const QgsSymbolWidgetContext &context )
751{
752 if ( QgsCalloutWidget *cw = qobject_cast< QgsCalloutWidget * >( mCalloutStackedWidget->currentWidget() ) )
753 {
754 cw->setContext( context );
755 }
757}
758
759void QgsLabelingGui::saveFormat()
760{
762 saveDlg.setDefaultTags( mTextFormatsListWidget->currentTagFilter() );
763 if ( !saveDlg.exec() )
764 return;
765
766 if ( saveDlg.name().isEmpty() )
767 return;
768
769 QgsStyle *style = saveDlg.destinationStyle();
770 if ( !style )
771 return;
772
773 switch ( saveDlg.selectedType() )
774 {
776 {
777 // check if there is no format with same name
778 if ( style->textFormatNames().contains( saveDlg.name() ) )
779 {
780 const int res = QMessageBox::warning( this, tr( "Save Text Format" ),
781 tr( "Format with name '%1' already exists. Overwrite?" )
782 .arg( saveDlg.name() ),
783 QMessageBox::Yes | QMessageBox::No );
784 if ( res != QMessageBox::Yes )
785 {
786 return;
787 }
788 style->removeTextFormat( saveDlg.name() );
789 }
790 const QStringList symbolTags = saveDlg.tags().split( ',' );
791
792 const QgsTextFormat newFormat = format();
793 style->addTextFormat( saveDlg.name(), newFormat );
794 style->saveTextFormat( saveDlg.name(), newFormat, saveDlg.isFavorite(), symbolTags );
795 break;
796 }
797
799 {
800 // check if there is no settings with same name
801 if ( style->labelSettingsNames().contains( saveDlg.name() ) )
802 {
803 const int res = QMessageBox::warning( this, tr( "Save Label Settings" ),
804 tr( "Label settings with the name '%1' already exist. Overwrite?" )
805 .arg( saveDlg.name() ),
806 QMessageBox::Yes | QMessageBox::No );
807 if ( res != QMessageBox::Yes )
808 {
809 return;
810 }
811 style->removeLabelSettings( saveDlg.name() );
812 }
813 const QStringList symbolTags = saveDlg.tags().split( ',' );
814
815 const QgsPalLayerSettings newSettings = layerSettings();
816 style->addLabelSettings( saveDlg.name(), newSettings );
817 style->saveLabelSettings( saveDlg.name(), newSettings, saveDlg.isFavorite(), symbolTags );
818 break;
819 }
820
827 break;
828 }
829}
830
831void QgsLabelingGui::updateGeometryTypeBasedWidgets()
832{
833 QgsWkbTypes::GeometryType geometryType = mGeomType;
834
835 if ( mGeometryGeneratorGroupBox->isChecked() )
836 geometryType = mGeometryGeneratorType->currentData().value<QgsWkbTypes::GeometryType>();
837 else if ( mLayer )
838 geometryType = mLayer->geometryType();
839
840 // show/hide options based upon geometry type
841 chkMergeLines->setVisible( geometryType == QgsWkbTypes::LineGeometry );
842 mDirectSymbolsFrame->setVisible( geometryType == QgsWkbTypes::LineGeometry );
843 mMinSizeFrame->setVisible( geometryType != QgsWkbTypes::PointGeometry );
844 mPolygonFeatureOptionsFrame->setVisible( geometryType == QgsWkbTypes::PolygonGeometry );
845
846
847 const Qgis::LabelPlacement prevPlacement = static_cast< Qgis::LabelPlacement >( mPlacementModeComboBox->currentData().toInt() );
848 mPlacementModeComboBox->clear();
849
850 switch ( geometryType )
851 {
853 mPlacementModeComboBox->addItem( tr( "Cartographic" ), static_cast< int >( Qgis::LabelPlacement::OrderedPositionsAroundPoint ) );
854 mPlacementModeComboBox->addItem( tr( "Around Point" ), static_cast< int >( Qgis::LabelPlacement::AroundPoint ) );
855 mPlacementModeComboBox->addItem( tr( "Offset from Point" ), static_cast< int >( Qgis::LabelPlacement::OverPoint ) );
856 break;
857
859 mPlacementModeComboBox->addItem( tr( "Parallel" ), static_cast< int >( Qgis::LabelPlacement::Line ) );
860 mPlacementModeComboBox->addItem( tr( "Curved" ), static_cast< int >( Qgis::LabelPlacement::Curved ) );
861 mPlacementModeComboBox->addItem( tr( "Horizontal" ), static_cast< int >( Qgis::LabelPlacement::Horizontal ) );
862 break;
863
865 mPlacementModeComboBox->addItem( tr( "Offset from Centroid" ), static_cast< int >( Qgis::LabelPlacement::OverPoint ) );
866 mPlacementModeComboBox->addItem( tr( "Around Centroid" ), static_cast< int >( Qgis::LabelPlacement::AroundPoint ) );
867 mPlacementModeComboBox->addItem( tr( "Horizontal" ), static_cast< int >( Qgis::LabelPlacement::Horizontal ) );
868 mPlacementModeComboBox->addItem( tr( "Free (Angled)" ), static_cast< int >( Qgis::LabelPlacement::Free ) );
869 mPlacementModeComboBox->addItem( tr( "Using Perimeter" ), static_cast< int >( Qgis::LabelPlacement::Line ) );
870 mPlacementModeComboBox->addItem( tr( "Using Perimeter (Curved)" ), static_cast< int >( Qgis::LabelPlacement::PerimeterCurved ) );
871 mPlacementModeComboBox->addItem( tr( "Outside Polygons" ), static_cast< int >( Qgis::LabelPlacement::OutsidePolygons ) );
872 break;
873
875 break;
877 qFatal( "unknown geometry type unexpected" );
878 }
879
880 if ( mPlacementModeComboBox->findData( static_cast< int >( prevPlacement ) ) != -1 )
881 {
882 mPlacementModeComboBox->setCurrentIndex( mPlacementModeComboBox->findData( static_cast< int >( prevPlacement ) ) );
883 }
884
885 if ( geometryType == QgsWkbTypes::PointGeometry || geometryType == QgsWkbTypes::PolygonGeometry )
886 {
887 // follow placement alignment is only valid for point or polygon layers
888 if ( mFontMultiLineAlignComboBox->findData( static_cast< int >( Qgis::LabelMultiLineAlignment::FollowPlacement ) ) == -1 )
889 mFontMultiLineAlignComboBox->addItem( tr( "Follow Label Placement" ), static_cast< int >( Qgis::LabelMultiLineAlignment::FollowPlacement ) );
890 }
891 else
892 {
893 const int idx = mFontMultiLineAlignComboBox->findData( static_cast< int >( Qgis::LabelMultiLineAlignment::FollowPlacement ) );
894 if ( idx >= 0 )
895 mFontMultiLineAlignComboBox->removeItem( idx );
896 }
897
898 updatePlacementWidgets();
899 updateLinePlacementOptions();
900}
901
902void QgsLabelingGui::showGeometryGeneratorExpressionBuilder()
903{
904 QgsExpressionBuilderDialog expressionBuilder( mLayer );
905
906 expressionBuilder.setExpressionText( mGeometryGenerator->text() );
907 expressionBuilder.setExpressionContext( createExpressionContext() );
908
909 if ( expressionBuilder.exec() )
910 {
911 mGeometryGenerator->setText( expressionBuilder.expressionText() );
912 }
913}
914
915void QgsLabelingGui::validateGeometryGeneratorExpression()
916{
917 bool valid = true;
918
919 if ( mGeometryGeneratorGroupBox->isChecked() )
920 {
921 if ( !mPreviewFeature.isValid() && mLayer )
922 mLayer->getFeatures( QgsFeatureRequest().setLimit( 1 ) ).nextFeature( mPreviewFeature );
923
924 QgsExpression expression( mGeometryGenerator->text() );
925 QgsExpressionContext context = createExpressionContext();
926 context.setFeature( mPreviewFeature );
927
928 expression.prepare( &context );
929
930 if ( expression.hasParserError() )
931 {
932 mGeometryGeneratorWarningLabel->setText( expression.parserErrorString() );
933 valid = false;
934 }
935 else
936 {
937 const QVariant result = expression.evaluate( &context );
938 const QgsGeometry geometry = result.value<QgsGeometry>();
939 const QgsWkbTypes::GeometryType configuredGeometryType = mGeometryGeneratorType->currentData().value<QgsWkbTypes::GeometryType>();
940 if ( geometry.isNull() )
941 {
942 mGeometryGeneratorWarningLabel->setText( tr( "Result of the expression is not a geometry" ) );
943 valid = false;
944 }
945 else if ( geometry.type() != configuredGeometryType )
946 {
947 mGeometryGeneratorWarningLabel->setText( QStringLiteral( "<p>%1</p><p><a href=\"#determineGeometryGeneratorType\">%2</a></p>" ).arg(
948 tr( "Result of the expression does not match configured geometry type." ),
949 tr( "Change to %1" ).arg( QgsWkbTypes::geometryDisplayString( geometry.type() ) ) ) );
950 valid = false;
951 }
952 }
953 }
954
955 // The collapsible groupbox internally changes the visibility of this
956 // Work around by setting the visibility deferred in the next event loop cycle.
957 QTimer *timer = new QTimer();
958 connect( timer, &QTimer::timeout, this, [this, valid]()
959 {
960 mGeometryGeneratorWarningLabel->setVisible( !valid );
961 } );
962 connect( timer, &QTimer::timeout, timer, &QTimer::deleteLater );
963 timer->start( 0 );
964}
965
966void QgsLabelingGui::determineGeometryGeneratorType()
967{
968 if ( !mPreviewFeature.isValid() && mLayer )
969 mLayer->getFeatures( QgsFeatureRequest().setLimit( 1 ) ).nextFeature( mPreviewFeature );
970
971 QgsExpression expression( mGeometryGenerator->text() );
972 QgsExpressionContext context = createExpressionContext();
973 context.setFeature( mPreviewFeature );
974
975 expression.prepare( &context );
976 const QgsGeometry geometry = expression.evaluate( &context ).value<QgsGeometry>();
977
978 mGeometryGeneratorType->setCurrentIndex( mGeometryGeneratorType->findData( geometry.type() ) );
979}
980
981void QgsLabelingGui::calloutTypeChanged()
982{
983 const QString newCalloutType = mCalloutStyleComboBox->currentData().toString();
984 QgsCalloutWidget *pew = qobject_cast< QgsCalloutWidget * >( mCalloutStackedWidget->currentWidget() );
985 if ( pew )
986 {
987 if ( pew->callout() && pew->callout()->type() == newCalloutType )
988 return;
989 }
990
991 // get creation function for new callout from registry
993 QgsCalloutAbstractMetadata *am = registry->calloutMetadata( newCalloutType );
994 if ( !am ) // check whether the metadata is assigned
995 return;
996
997 // change callout to a new one (with different type)
998 // base new callout on existing callout's properties
999 const std::unique_ptr< QgsCallout > newCallout( am->createCallout( pew && pew->callout() ? pew->callout()->properties( QgsReadWriteContext() ) : QVariantMap(), QgsReadWriteContext() ) );
1000 if ( !newCallout )
1001 return;
1002
1003 updateCalloutWidget( newCallout.get() );
1004 updatePreview();
1005}
1006
1007
1008//
1009// QgsLabelSettingsDialog
1010//
1011
1012QgsLabelSettingsDialog::QgsLabelSettingsDialog( const QgsPalLayerSettings &settings, QgsVectorLayer *layer, QgsMapCanvas *mapCanvas, QWidget *parent,
1013 QgsWkbTypes::GeometryType geomType )
1014 : QDialog( parent )
1015{
1016 QVBoxLayout *vLayout = new QVBoxLayout();
1017 mWidget = new QgsLabelingGui( layer, mapCanvas, settings, nullptr, geomType );
1018 vLayout->addWidget( mWidget );
1019 mButtonBox = new QDialogButtonBox( QDialogButtonBox::Cancel | QDialogButtonBox::Help | QDialogButtonBox::Ok, Qt::Horizontal );
1020 connect( mButtonBox, &QDialogButtonBox::accepted, this, &QDialog::accept );
1021 connect( mButtonBox, &QDialogButtonBox::rejected, this, &QDialog::reject );
1022 connect( mButtonBox, &QDialogButtonBox::helpRequested, this, &QgsLabelSettingsDialog::showHelp );
1023 vLayout->addWidget( mButtonBox );
1024 setLayout( vLayout );
1025 setWindowTitle( tr( "Label Settings" ) );
1026}
1027
1028QDialogButtonBox *QgsLabelSettingsDialog::buttonBox() const
1029{
1030 return mButtonBox;
1031}
1032
1033void QgsLabelSettingsDialog::showHelp()
1034{
1035 QgsHelp::openHelp( QStringLiteral( "style_library/label_settings.html" ) );
1036}
1037
1038
1039
LabelOffsetType
Behavior modifier for label offset and distance, only applies in some label placement modes.
Definition: qgis.h:608
LabelPlacement
Placement modes which determine how label candidates are generated for a feature.
Definition: qgis.h:561
@ OverPoint
Arranges candidates over a point (or centroid of a polygon), or at a preset offset from the point....
@ Curved
Arranges candidates following the curvature of a line feature. Applies to line layers only.
@ AroundPoint
Arranges candidates in a circle around a point (or centroid of a polygon). Applies to point or polygo...
@ Line
Arranges candidates parallel to a generalised line representing the feature or parallel to a polygon'...
@ Free
Arranges candidates scattered throughout a polygon feature. Candidates are rotated to respect the pol...
@ OrderedPositionsAroundPoint
Candidates are placed in predefined positions around a point. Preference is given to positions with g...
@ Horizontal
Arranges horizontal candidates scattered throughout a polygon feature. Applies to polygon layers only...
@ PerimeterCurved
Arranges candidates following the curvature of a polygon's boundary. Applies to polygon layers only.
@ OutsidePolygons
Candidates are placed outside of polygon boundaries. Applies to polygon layers only....
LabelQuadrantPosition
Label quadrant positions.
Definition: qgis.h:622
LabelMultiLineAlignment
Text alignment for multi-line labels.
Definition: qgis.h:658
LabelOverlapHandling
Label overlap handling.
Definition: qgis.h:546
UpsideDownLabelHandling
Handling techniques for upside down labels.
Definition: qgis.h:643
static QgsCalloutRegistry * calloutRegistry()
Returns the application's callout registry, used for managing callout types.
Stores metadata about one callout renderer class.
virtual QgsCallout * createCallout(const QVariantMap &properties, const QgsReadWriteContext &context)=0
Create a callout of this type given the map of properties.
Convenience metadata class that uses static functions to create callouts and their widgets.
void setWidgetFunction(QgsCalloutWidgetFunc f)
Registry of available callout classes.
QgsCalloutAbstractMetadata * calloutMetadata(const QString &type) const
Returns the metadata for specified the specified callout type.
static QgsCallout * defaultCallout()
Create a new instance of a callout with default settings.
QStringList calloutTypes() const
Returns a list of all available callout types.
QgsCallout * createCallout(const QString &type, const QVariantMap &properties=QVariantMap(), const QgsReadWriteContext &context=QgsReadWriteContext()) const
Creates a new instance of a callout, given the callout type and properties.
Base class for widgets which allow control over the properties of callouts.
virtual QgsCallout * callout()=0
Returns the callout defined by the current settings in the widget.
void changed()
Should be emitted whenever configuration changes happened on this symbol layer configuration.
Abstract base class for callout renderers.
Definition: qgscallout.h:53
void setEnabled(bool enabled)
Sets whether the callout is enabled.
Definition: qgscallout.cpp:187
virtual QString type() const =0
Returns a unique string representing the callout type.
virtual QVariantMap properties(const QgsReadWriteContext &context) const
Returns the properties describing the callout encoded in a string format.
Definition: qgscallout.cpp:79
A general purpose distance and area calculator, capable of performing ellipsoid based calculations.
void setSourceCrs(const QgsCoordinateReferenceSystem &crs, const QgsCoordinateTransformContext &context)
Sets source spatial reference system crs.
bool setEllipsoid(const QString &ellipsoid)
Sets the ellipsoid by its acronym.
A generic dialog for building expression strings.
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.
static QgsExpressionContextScope * projectScope(const QgsProject *project)
Creates a new scope which contains variables and functions relating to a QGIS project.
static QgsExpressionContextScope * atlasScope(const QgsLayoutAtlas *atlas)
Creates a new scope which contains variables and functions relating to a QgsLayoutAtlas.
static QgsExpressionContextScope * mapSettingsScope(const QgsMapSettings &mapSettings)
Creates a new scope which contains variables and functions relating to a QgsMapSettings object.
static QgsExpressionContextScope * layerScope(const QgsMapLayer *layer)
Creates a new scope which contains variables and functions relating to a QgsMapLayer.
static QgsExpressionContextScope * globalScope()
Creates a new scope which contains variables and functions relating to the global QGIS context.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
void setOriginalValueVariable(const QVariant &value)
Sets the original value variable value for the context.
static const QString EXPR_SYMBOL_COLOR
Inbuilt variable name for symbol color variable.
void setFeature(const QgsFeature &feature)
Convenience function for setting a feature for the context.
void setHighlightedVariables(const QStringList &variableNames)
Sets the list of variable names within the context intended to be highlighted to the user.
static const QString EXPR_ORIGINAL_VALUE
Inbuilt variable name for value original value variable.
Class for parsing and evaluation of expressions (formerly called "search strings").
This class wraps a request for features to a vector layer (or directly its vector data provider).
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition: qgsfeature.h:56
A geometry is the spatial representation of a feature.
Definition: qgsgeometry.h:164
Q_GADGET bool isNull
Definition: qgsgeometry.h:166
QgsWkbTypes::GeometryType type
Definition: qgsgeometry.h:167
static void openHelp(const QString &key)
Opens help topic for the given help key using default system web browser.
Definition: qgshelp.cpp:38
A widget for customising label line anchor settings.
void updateDataDefinedProperties(QgsPropertyCollection &properties) override
Updates a data defined properties collection, correctly setting the values for any properties related...
QgsLabelLineSettings settings() const
Returns the line settings defined by the widget.
void setSettings(const QgsLabelLineSettings &settings)
Sets the line settings to show in the widget.
Contains settings related to how the label engine places and formats labels for line features (or pol...
void setPlacementFlags(QgsLabeling::LinePlacementFlags flags)
Returns the line placement flags, which dictate how line labels can be placed above or below the line...
void setLineAnchorPercent(double percent)
Sets the percent along the line at which labels should be placed.
void setDirectionSymbolPlacement(DirectionSymbolPlacement placement)
Sets the placement for direction symbols.
AnchorType anchorType() const
Returns the line anchor type, which dictates how the lineAnchorPercent() setting is handled.
void setAnchorTextPoint(AnchorTextPoint point)
Sets the line anchor text point, which dictates which part of the label text should be placed at the ...
void setLeftDirectionSymbol(const QString &symbol)
Sets the string to use for left direction arrows.
AnchorTextPoint anchorTextPoint() const
Returns the line anchor text point, which dictates which part of the label text should be placed at t...
void setMergeLines(bool merge)
Sets whether connected line features with identical label text should be merged prior to generating l...
DirectionSymbolPlacement
Placement options for direction symbols.
void setRightDirectionSymbol(const QString &symbol)
Sets the string to use for right direction arrows.
void setAnchorClipping(AnchorClipping clipping)
Sets the line anchor clipping mode, which dictates how line strings are clipped before calculating th...
void setOverrunDistanceMapUnitScale(const QgsMapUnitScale &scale)
Sets the map unit scale for label overrun distance.
double lineAnchorPercent() const
Returns the percent along the line at which labels should be placed.
void setAnchorType(AnchorType type)
Sets the line anchor type, which dictates how the lineAnchorPercent() setting is handled.
void setOverrunDistance(double distance)
Sets the distance which labels are allowed to overrun past the start or end of line features.
AnchorClipping anchorClipping() const
Returns the line anchor clipping mode, which dictates how line strings are clipped before calculating...
void setOverrunDistanceUnit(const QgsUnitTypes::RenderUnit &unit)
Sets the unit for label overrun distance.
void setReverseDirectionSymbol(bool reversed)
Sets whether the direction symbols should be reversed.
void setAddDirectionSymbol(bool enabled)
Sets whether '<' or '>' (or custom strings set via leftDirectionSymbol and rightDirectionSymbol) will...
A widget for customising label obstacle settings.
QgsLabelObstacleSettings settings() const
Returns the obstacle settings defined by the widget.
void updateDataDefinedProperties(QgsPropertyCollection &properties) override
Updates a data defined properties collection, correctly setting the values for any properties related...
void setSettings(const QgsLabelObstacleSettings &settings)
Sets the obstacle settings to show in the widget.
void setGeometryType(QgsWkbTypes::GeometryType type) override
Sets the geometry type of the features to customize the widget accordingly.
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 changed()
Emitted when any of the settings described by the widget are changed.
virtual void setContext(const QgsSymbolWidgetContext &context)
Sets the context in which the symbol widget is shown, e.g., the associated map canvas and expression ...
void setDataDefinedProperties(const QgsPropertyCollection &dataDefinedProperties)
Sets the current data defined properties to show in the widget.
virtual void setGeometryType(QgsWkbTypes::GeometryType type)
Sets the geometry type of the features to customize the widget accordingly.
QgsPropertyCollection dataDefinedProperties() const
Returns the current data defined properties state as specified in the widget.
A blocking dialog containing a QgsLabelSettingsWidgetBase.
void setMaximumNumberLabels(int number)
Sets the maximum number of labels which should be drawn for this layer.
void setLimitNumberLabelsEnabled(bool enabled)
Sets whether the the number of labels drawn for the layer should be limited.
void setMinimumFeatureSize(double size)
Sets the minimum feature size (in millimeters) for a feature to be labelled.
Map canvas is a class for displaying all GIS data types on a canvas.
Definition: qgsmapcanvas.h:90
Base class for all map layer types.
Definition: qgsmaplayer.h:73
QgsMapLayerType type
Definition: qgsmaplayer.h:80
Contains settings for how a map layer will be labeled.
bool fitInPolygonOnly
true if only labels which completely fit within a polygon are allowed.
double yOffset
Vertical offset of label.
QgsMapUnitScale labelOffsetMapUnitScale
Map unit scale for label offset.
int fontMaxPixelSize
Maximum pixel size for showing rendered map unit labels (1 - 10000).
void setObstacleSettings(const QgsLabelObstacleSettings &settings)
Sets the label obstacle settings.
const QgsLabelPlacementSettings & placementSettings() const
Returns the label placement settings.
double maxCurvedCharAngleIn
Maximum angle between inside curved label characters (valid range 20.0 to 60.0).
void setFormat(const QgsTextFormat &format)
Sets the label text formatting settings, e.g., font settings, buffer settings, etc.
double zIndex
Z-Index of label, where labels with a higher z-index are rendered on top of labels with a lower z-ind...
QString wrapChar
Wrapping character string.
Qgis::LabelOffsetType offsetType
Offset type for layer (only applies in certain placement modes)
double xOffset
Horizontal offset of label.
Qgis::LabelPlacement placement
Label placement mode.
bool drawLabels
Whether to draw labels for this layer.
bool fontLimitPixelSize
true if label sizes should be limited by pixel size.
double minimumScale
The minimum map scale (i.e.
QgsWkbTypes::GeometryType geometryGeneratorType
The type of the result geometry of the geometry generator.
QgsUnitTypes::RenderUnit offsetUnits
Units for offsets of label.
Qgis::LabelQuadrantPosition quadOffset
Sets the quadrant in which to offset labels from feature.
bool scaleVisibility
Set to true to limit label visibility to a range of scales.
double repeatDistance
Distance for repeating labels for a single feature.
bool geometryGeneratorEnabled
Defines if the geometry generator is enabled or not. If disabled, the standard geometry will be taken...
Qgis::LabelMultiLineAlignment multilineAlign
Horizontal alignment of multi-line labels.
bool centroidInside
true if centroid positioned labels must be placed inside their corresponding feature polygon,...
int priority
Label priority.
bool labelPerPart
true if every part of a multi-part feature should be labeled.
QgsUnitTypes::RenderUnit distUnits
Units the distance from feature to the label.
QgsUnitTypes::RenderUnit repeatDistanceUnit
Units for repeating labels for a single feature.
int fontMinPixelSize
Minimum pixel size for showing rendered map unit labels (1 - 1000).
double angleOffset
Label rotation, in degrees clockwise.
double maxCurvedCharAngleOut
Maximum angle between outside curved label characters (valid range -20.0 to -95.0)
const QgsLabelThinningSettings & thinningSettings() const
Returns the label thinning settings.
void setPolygonPlacementFlags(QgsLabeling::PolygonPlacementFlags flags)
Sets the polygon placement flags, which dictate how polygon labels can be placed.
void setRotationUnit(QgsUnitTypes::AngleUnit angleUnit)
Set unit for rotation of labels.
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'
bool preserveRotation
True if label rotation should be preserved during label pin/unpin operations.
bool plusSign
Whether '+' signs should be prepended to positive numeric labels.
QString geometryGenerator
The geometry generator expression. Null if disabled.
const QgsLabelLineSettings & lineSettings() const
Returns the label line settings, which contain settings related to how the label engine places and fo...
QgsMapUnitScale distMapUnitScale
Map unit scale for label feature distance.
QgsStringReplacementCollection substitutions
Substitution collection for automatic text substitution with labels.
int decimals
Number of decimal places to show for numeric labels.
double dist
Distance from feature to the label.
QgsMapUnitScale repeatDistanceMapUnitScale
Map unit scale for repeating labels for a single feature.
bool centroidWhole
true if feature centroid should be calculated from the whole feature, or false if only the visible pa...
Qgis::UpsideDownLabelHandling upsidedownLabels
Controls whether upside down labels are displayed and how they are handled.
QString fieldName
Name of field (or an expression) to use for label text.
bool formatNumbers
Set to true to format numeric label text as numbers (e.g.
QgsWkbTypes::GeometryType layerType
Geometry type of layers associated with these settings.
void setCallout(QgsCallout *callout)
Sets the label callout renderer, responsible for drawing label callouts.
double maximumScale
The maximum map scale (i.e.
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...
void setUnplacedVisibility(Qgis::UnplacedLabelVisibility visibility)
Sets the layer's unplaced label visibility.
bool useSubstitutions
True if substitutions should be applied.
Base class for any widget that can be shown as a inline panel.
void openPanel(QgsPanelWidget *panel)
Open a panel or dialog depending on dock mode setting If dock mode is true this method will emit the ...
static QgsPanelWidget * findParentPanel(QWidget *widget)
Traces through the parents of a widget to find if it is contained within a QgsPanelWidget widget.
bool dockMode()
Returns the dock mode state.
QgsStyle * styleAtPath(const QString &path)
Returns a reference to the style database associated with the project with matching file path.
static QgsProject * instance()
Returns the QgsProject singleton instance.
Definition: qgsproject.cpp:477
const QgsProjectStyleSettings * styleSettings() const
Returns the project's style settings, which contains settings and properties relating to how a QgsPro...
QgsCoordinateTransformContext transformContext
Definition: qgsproject.h:110
A grouped map of multiple QgsProperty objects, each referenced by a integer key value.
A button for controlling property overrides which may apply to a widget.
bool isActive() const
Returns true if the button has an active property.
void changed()
Emitted when property definition changes.
The class is used as a container of context for various read/write operations on other objects.
a dialog for setting properties of a newly saved style.
bool removeLabelSettings(const QString &name)
Removes label settings from the style.
Definition: qgsstyle.cpp:1053
bool saveLabelSettings(const QString &name, const QgsPalLayerSettings &settings, bool favorite, const QStringList &tags)
Adds label settings to the database.
Definition: qgsstyle.cpp:1017
QStringList textFormatNames() const
Returns a list of names of text formats in the style.
Definition: qgsstyle.cpp:2130
bool removeTextFormat(const QString &name)
Removes a text format from the style.
Definition: qgsstyle.cpp:977
StyleEntity
Enum for Entities involved in a style.
Definition: qgsstyle.h:179
@ LabelSettingsEntity
Label settings.
Definition: qgsstyle.h:185
@ TextFormatEntity
Text formats.
Definition: qgsstyle.h:184
@ SmartgroupEntity
Smart groups.
Definition: qgsstyle.h:183
@ Symbol3DEntity
3D symbol entity (since QGIS 3.14)
Definition: qgsstyle.h:187
@ SymbolEntity
Symbols.
Definition: qgsstyle.h:180
@ TagEntity
Tags.
Definition: qgsstyle.h:181
@ ColorrampEntity
Color ramps.
Definition: qgsstyle.h:182
@ LegendPatchShapeEntity
Legend patch shape (since QGIS 3.14)
Definition: qgsstyle.h:186
static QgsStyle * defaultStyle()
Returns default application-wide style.
Definition: qgsstyle.cpp:145
QStringList labelSettingsNames() const
Returns a list of names of label settings in the style.
Definition: qgsstyle.cpp:2194
static QgsTextFormat defaultTextFormatForProject(QgsProject *project, QgsStyle::TextFormatContext context=QgsStyle::TextFormatContext::Labeling)
Returns the default text format to use for new text based objects for the specified project,...
Definition: qgsstyle.cpp:1217
bool addTextFormat(const QString &name, const QgsTextFormat &format, bool update=false)
Adds a text format with the specified name to the style.
Definition: qgsstyle.cpp:337
QgsPalLayerSettings labelSettings(const QString &name) const
Returns the label settings with the specified name.
Definition: qgsstyle.cpp:2140
@ Labeling
Text format used in labeling.
bool saveTextFormat(const QString &name, const QgsTextFormat &format, bool favorite, const QStringList &tags)
Adds a text format to the database.
Definition: qgsstyle.cpp:941
bool addLabelSettings(const QString &name, const QgsPalLayerSettings &settings, bool update=false)
Adds label settings with the specified name to the style.
Definition: qgsstyle.cpp:358
Contains settings which reflect the context in which a symbol (or renderer) widget is shown,...
void setMapCanvas(QgsMapCanvas *canvas)
Sets the map canvas associated with the widget.
void setExpressionContext(QgsExpressionContext *context)
Sets the optional expression context used for the widget.
A widget for customizing text formatting settings.
virtual void setContext(const QgsSymbolWidgetContext &context)
Sets the context in which the widget is shown, e.g., the associated map canvas and expression context...
virtual void setFormatFromStyle(const QString &name, QgsStyle::StyleEntity type, const QString &stylePath)
Sets the current text settings from a style entry.
Container for all settings relating to text rendering.
Definition: qgstextformat.h:41
AngleUnit
Units of angles.
Definition: qgsunittypes.h:132
@ AngleGon
Gon/gradian.
Definition: qgsunittypes.h:135
@ AngleRadians
Square kilometers.
Definition: qgsunittypes.h:134
@ AngleDegrees
Degrees.
Definition: qgsunittypes.h:133
@ AngleMilNATO
Angular mil (NATO definition, 6400 mil = 2PI radians)
Definition: qgsunittypes.h:140
@ AngleMilliradiansSI
Angular milliradians (SI definition, 1/1000 of radian)
Definition: qgsunittypes.h:139
@ AngleTurn
Turn/revolutions.
Definition: qgsunittypes.h:138
@ AngleMinutesOfArc
Minutes of arc.
Definition: qgsunittypes.h:136
@ AngleSecondsOfArc
Seconds of arc.
Definition: qgsunittypes.h:137
static Q_INVOKABLE QString toString(QgsUnitTypes::DistanceUnit unit)
Returns a translated string representing a distance unit.
Represents a vector layer which manages a vector based data sets.
static QString geometryDisplayString(GeometryType type) SIP_HOLDGIL
Returns a display string for a geometry type.
GeometryType
The geometry types are used to group QgsWkbTypes::Type in a coarse way.
Definition: qgswkbtypes.h:141
@ VectorLayer
Vector layer.
QgsSignalBlocker< Object > whileBlocking(Object *object)
Temporarily blocks signals from a QObject while calling a single method from the object.
Definition: qgis.h:2453
QgsCalloutWidget *(* QgsCalloutWidgetFunc)(QgsVectorLayer *)
#define QgsDebugMsg(str)
Definition: qgslogger.h:38