QGIS API Documentation 3.43.0-Master (3ee7834ace6)
qgsvectorlayerproperties.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsdlgvectorlayerproperties.cpp
3 Unified property dialog for vector layers
4 -------------------
5 begin : 2004-01-28
6 copyright : (C) 2004 by Gary E.Sherman
7 email : sherman at mrcc.com
8***************************************************************************/
9
10/***************************************************************************
11 * *
12 * This program is free software; you can redistribute it and/or modify *
13 * it under the terms of the GNU General Public License as published by *
14 * the Free Software Foundation; either version 2 of the License, or *
15 * (at your option) any later version. *
16 * *
17 ***************************************************************************/
18
19#include <memory>
20#include <limits>
21
22#include "qgsactionmanager.h"
23#include "qgsjoindialog.h"
26#include "qgsapplication.h"
29#include "qgsdiagramwidget.h"
32#include "qgslabelingwidget.h"
33#include "qgsmapcanvas.h"
35#include "qgsmetadatawidget.h"
36#include "qgsproject.h"
37#include "qgsvectorlayer.h"
40#include "moc_qgsvectorlayerproperties.cpp"
43#include "qgsdatasourceuri.h"
44#include "qgsrenderer.h"
46#include "qgssettings.h"
48#include "qgsstyle.h"
49#include "qgsauxiliarystorage.h"
54#include "qgslabelinggui.h"
55#include "qgsmessagebar.h"
58#include "qgsmaskingwidget.h"
61#include "qgsproviderregistry.h"
63#include "qgslayertreemodel.h"
64#include "qgsmaptip.h"
65#include "qgsgui.h"
66#include "qgsnative.h"
69#include "qgsfileutils.h"
70#include "qgswebview.h"
71#include "qgswebframe.h"
72#include "qgsexpressionfinder.h"
73#if WITH_QTWEBKIT
74#include <QWebElement>
75#endif
76
77#include <QDesktopServices>
78#include <QMessageBox>
79#include <QDir>
80#include <QFile>
81#include <QFileDialog>
82#include <QFileInfo>
83#include <QFontDialog>
84#include <QComboBox>
85#include <QCheckBox>
86#include <QHeaderView>
87#include <QColorDialog>
88#include <QMenu>
89#include <QUrl>
90#include <QRegularExpressionValidator>
91
92
94 QgsMapCanvas *canvas,
95 QgsMessageBar *messageBar,
96 QgsVectorLayer *lyr,
97 QWidget *parent,
98 Qt::WindowFlags fl
99)
100 : QgsLayerPropertiesDialog( lyr, canvas, QStringLiteral( "VectorLayerProperties" ), parent, fl )
101 , mMessageBar( messageBar )
102 , mLayer( lyr )
103 , mOriginalSubsetSQL( lyr->subsetString() )
104{
105 setupUi( this );
106 connect( pbnQueryBuilder, &QPushButton::clicked, this, &QgsVectorLayerProperties::pbnQueryBuilder_clicked );
107 connect( pbnIndex, &QPushButton::clicked, this, &QgsVectorLayerProperties::pbnIndex_clicked );
108 connect( mCrsSelector, &QgsProjectionSelectionWidget::crsChanged, this, &QgsVectorLayerProperties::mCrsSelector_crsChanged );
109 connect( pbnUpdateExtents, &QPushButton::clicked, this, &QgsVectorLayerProperties::pbnUpdateExtents_clicked );
110 connect( mButtonAddJoin, &QPushButton::clicked, this, &QgsVectorLayerProperties::mButtonAddJoin_clicked );
111 connect( mButtonEditJoin, &QPushButton::clicked, this, &QgsVectorLayerProperties::mButtonEditJoin_clicked );
112 connect( mJoinTreeWidget, &QTreeWidget::itemDoubleClicked, this, &QgsVectorLayerProperties::mJoinTreeWidget_itemDoubleClicked );
113 connect( mButtonRemoveJoin, &QPushButton::clicked, this, &QgsVectorLayerProperties::mButtonRemoveJoin_clicked );
114 connect( mButtonAddWmsDimension, &QPushButton::clicked, this, &QgsVectorLayerProperties::mButtonAddWmsDimension_clicked );
115 connect( mButtonEditWmsDimension, &QPushButton::clicked, this, &QgsVectorLayerProperties::mButtonEditWmsDimension_clicked );
116 connect( mWmsDimensionsTreeWidget, &QTreeWidget::itemDoubleClicked, this, &QgsVectorLayerProperties::mWmsDimensionsTreeWidget_itemDoubleClicked );
117 connect( mButtonRemoveWmsDimension, &QPushButton::clicked, this, &QgsVectorLayerProperties::mButtonRemoveWmsDimension_clicked );
118 connect( mSimplifyDrawingGroupBox, &QGroupBox::toggled, this, &QgsVectorLayerProperties::mSimplifyDrawingGroupBox_toggled );
119 connect( buttonBox, &QDialogButtonBox::helpRequested, this, &QgsVectorLayerProperties::showHelp );
120
121 mProjectDirtyBlocker = std::make_unique<QgsProjectDirtyBlocker>( QgsProject::instance() );
122
123 // QgsOptionsDialogBase handles saving/restoring of geometry, splitter and current tab states,
124 // switching vertical tabs between icon/text to icon-only modes (splitter collapsed to left),
125 // and connecting QDialogButtonBox's accepted/rejected signals to dialog's accept/reject slots
126 initOptionsBase( false );
127
128 mBtnStyle = new QPushButton( tr( "Style" ), this );
129 QMenu *menuStyle = new QMenu( this );
130 mActionLoadStyle = new QAction( tr( "Load Style…" ), this );
131 connect( mActionLoadStyle, &QAction::triggered, this, &QgsVectorLayerProperties::loadStyle );
132
133 mActionSaveStyle = new QAction( tr( "Save Current Style…" ), this );
134 connect( mActionSaveStyle, &QAction::triggered, this, &QgsVectorLayerProperties::saveStyleAs );
135
136 mActionSaveMultipleStyles = new QAction( tr( "Save Multiple Styles…" ), this );
137 connect( mActionSaveMultipleStyles, &QAction::triggered, this, &QgsVectorLayerProperties::saveMultipleStylesAs );
138
139 mSourceGroupBox->hide();
140
141 mBtnStyle->setMenu( menuStyle );
142 connect( menuStyle, &QMenu::aboutToShow, this, &QgsVectorLayerProperties::aboutToShowStyleMenu );
143 buttonBox->addButton( mBtnStyle, QDialogButtonBox::ResetRole );
144
146
147 connect( buttonBox->button( QDialogButtonBox::Apply ), &QAbstractButton::clicked, this, &QgsVectorLayerProperties::apply );
148 connect( this, &QDialog::accepted, this, &QgsVectorLayerProperties::apply );
149 connect( this, &QDialog::rejected, this, &QgsVectorLayerProperties::rollback );
150
151 if ( mCanvas )
152 {
153 mContext = mCanvas->createExpressionContext();
154 }
155 else
156 {
161 }
162 mContext << QgsExpressionContextUtils::layerScope( mLayer );
163
164 mMapTipFieldComboBox->setLayer( lyr );
165 mDisplayExpressionWidget->setLayer( lyr );
166 mDisplayExpressionWidget->registerExpressionContextGenerator( this );
167 initMapTipPreview();
168
169 connect( mMapTipInsertFieldButton, &QAbstractButton::clicked, this, &QgsVectorLayerProperties::insertField );
170 connect( mMapTipInsertExpressionButton, &QAbstractButton::clicked, this, &QgsVectorLayerProperties::insertOrEditExpression );
171
172 if ( !mLayer )
173 return;
174
175 connect( mEnableMapTips, &QAbstractButton::toggled, mHtmlMapTipGroupBox, &QWidget::setEnabled );
176 mEnableMapTips->setChecked( mLayer->mapTipsEnabled() );
177
178 QVBoxLayout *layout = nullptr;
179
180 if ( mLayer->isSpatial() )
181 {
182 // Create the Labeling dialog tab
183 layout = new QVBoxLayout( labelingFrame );
184 layout->setContentsMargins( 0, 0, 0, 0 );
185 labelingDialog = new QgsLabelingWidget( mLayer, mCanvas, labelingFrame );
186 labelingDialog->layout()->setContentsMargins( 0, 0, 0, 0 );
187 connect( labelingDialog, &QgsLabelingWidget::auxiliaryFieldCreated, this, [=] { updateAuxiliaryStoragePage(); } );
188 layout->addWidget( labelingDialog );
189 labelingFrame->setLayout( layout );
190
191 // Create the masking dialog tab
192 layout = new QVBoxLayout( mMaskingFrame );
193 layout->setContentsMargins( 0, 0, 0, 0 );
194 mMaskingWidget = new QgsMaskingWidget( mMaskingFrame );
195 mMaskingWidget->setLayer( mLayer );
196 mMaskingWidget->layout()->setContentsMargins( 0, 0, 0, 0 );
197 layout->addWidget( mMaskingWidget );
198 mMaskingFrame->setLayout( layout );
199 }
200 else
201 {
202 labelingDialog = nullptr;
203 mOptsPage_Labels->setEnabled( false ); // disable labeling item
204 mOptsPage_Masks->setEnabled( false ); // disable masking item
205 mGeomGroupBox->setEnabled( false );
206 mGeomGroupBox->setVisible( false );
207 mCrsGroupBox->setEnabled( false );
208 mCrsGroupBox->setVisible( false );
209 }
210
211 // Create the Actions dialog tab
212 QVBoxLayout *actionLayout = new QVBoxLayout( actionOptionsFrame );
213 actionLayout->setContentsMargins( 0, 0, 0, 0 );
214 mActionDialog = new QgsAttributeActionDialog( *mLayer->actions(), actionOptionsFrame );
215 mActionDialog->layout()->setContentsMargins( 0, 0, 0, 0 );
216 actionLayout->addWidget( mActionDialog );
217
218 mSourceFieldsPropertiesDialog = new QgsSourceFieldsProperties( mLayer, mSourceFieldsFrame );
219 mSourceFieldsPropertiesDialog->layout()->setContentsMargins( 0, 0, 0, 0 );
220 mSourceFieldsFrame->setLayout( new QVBoxLayout( mSourceFieldsFrame ) );
221 mSourceFieldsFrame->layout()->setContentsMargins( 0, 0, 0, 0 );
222 mSourceFieldsFrame->layout()->addWidget( mSourceFieldsPropertiesDialog );
223
224 connect( mSourceFieldsPropertiesDialog, &QgsSourceFieldsProperties::toggleEditing, this, static_cast<void ( QgsVectorLayerProperties::* )()>( &QgsVectorLayerProperties::toggleEditing ) );
225
226 mAttributesFormPropertiesDialog = new QgsAttributesFormProperties( mLayer, mAttributesFormFrame );
227 mAttributesFormPropertiesDialog->layout()->setContentsMargins( 0, 0, 0, 0 );
228 mAttributesFormFrame->setLayout( new QVBoxLayout( mAttributesFormFrame ) );
229 mAttributesFormFrame->layout()->setContentsMargins( 0, 0, 0, 0 );
230 mAttributesFormFrame->layout()->addWidget( mAttributesFormPropertiesDialog );
231
232 // Metadata tab, before the syncToLayer
233 QVBoxLayout *metadataLayout = new QVBoxLayout( metadataFrame );
234 metadataLayout->setContentsMargins( 0, 0, 0, 0 );
235 mMetadataWidget = new QgsMetadataWidget( this, mLayer );
236 mMetadataWidget->layout()->setContentsMargins( 0, 0, 0, 0 );
237 mMetadataWidget->setMapCanvas( mCanvas );
238 metadataLayout->addWidget( mMetadataWidget );
239 metadataFrame->setLayout( metadataLayout );
240
241 QVBoxLayout *temporalLayout = new QVBoxLayout( temporalFrame );
242 temporalLayout->setContentsMargins( 0, 0, 0, 0 );
243 mTemporalWidget = new QgsVectorLayerTemporalPropertiesWidget( this, mLayer );
244 temporalLayout->addWidget( mTemporalWidget );
245
246 setMetadataWidget( mMetadataWidget, mOptsPage_Metadata );
247
248 mBtnMetadata = new QPushButton( tr( "Metadata" ), this );
249 QMenu *menuMetadata = new QMenu( this );
250 mActionLoadMetadata = menuMetadata->addAction( tr( "Load Metadata from File…" ), this, &QgsVectorLayerProperties::loadMetadataFromFile );
251 mActionSaveMetadataAs = menuMetadata->addAction( tr( "Save Metadata to File…" ), this, &QgsVectorLayerProperties::saveMetadataToFile );
252 menuMetadata->addSeparator();
253 menuMetadata->addAction( tr( "Save to Default Location" ), this, &QgsVectorLayerProperties::saveMetadataAsDefault );
254 menuMetadata->addAction( tr( "Restore from Default Location" ), this, &QgsVectorLayerProperties::loadDefaultMetadata );
255 mBtnMetadata->setMenu( menuMetadata );
256 buttonBox->addButton( mBtnMetadata, QDialogButtonBox::ResetRole );
257
258 mSelectionColorButton->setAllowOpacity( true );
259 mSelectionColorButton->setColorDialogTitle( tr( "Override Selection Color" ) );
260 if ( mCanvas )
261 {
262 mSelectionColorButton->setColor( mCanvas->selectionColor() );
263 mSelectionColorButton->setDefaultColor( mCanvas->selectionColor() );
264 }
265 connect( mRadioOverrideSelectionColor, &QRadioButton::toggled, mSelectionColorButton, &QWidget::setEnabled );
266 mSelectionColorButton->setEnabled( false );
267 connect( mRadioOverrideSelectionSymbol, &QRadioButton::toggled, mSelectionSymbolButton, &QWidget::setEnabled );
268 switch ( mLayer->geometryType() )
269 {
271 mSelectionSymbolButton->setSymbolType( Qgis::SymbolType::Marker );
272 break;
274 mSelectionSymbolButton->setSymbolType( Qgis::SymbolType::Line );
275 break;
277 mSelectionSymbolButton->setSymbolType( Qgis::SymbolType::Fill );
278 break;
279
282 break;
283 }
284 mSelectionSymbolButton->setEnabled( false );
285 mRadioDefaultSelectionColor->setChecked( true );
286
287 // Diagram tab, before the syncToLayer
288 QVBoxLayout *diagLayout = new QVBoxLayout( mDiagramFrame );
289 diagLayout->setContentsMargins( 0, 0, 0, 0 );
290 diagramPropertiesDialog = new QgsDiagramWidget( mLayer, mCanvas, mDiagramFrame );
291 diagramPropertiesDialog->layout()->setContentsMargins( 0, 0, 0, 0 );
292 connect( diagramPropertiesDialog, &QgsDiagramWidget::auxiliaryFieldCreated, this, [=] { updateAuxiliaryStoragePage(); } );
293 diagLayout->addWidget( diagramPropertiesDialog );
294 mDiagramFrame->setLayout( diagLayout );
295
296 syncToLayer();
297
298 if ( mLayer->dataProvider() )
299 {
300 //enable spatial index button group if supported by provider, or if one already exists
303 {
304 pbnIndex->setEnabled( false );
305 }
307 {
308 pbnIndex->setEnabled( false );
309 pbnIndex->setText( tr( "Spatial Index Exists" ) );
310 }
311
313 {
314 cboProviderEncoding->addItems( QgsVectorDataProvider::availableEncodings() );
315 QString enc = mLayer->dataProvider()->encoding();
316 int encindex = cboProviderEncoding->findText( enc );
317 if ( encindex < 0 )
318 {
319 cboProviderEncoding->insertItem( 0, enc );
320 encindex = 0;
321 }
322 cboProviderEncoding->setCurrentIndex( encindex );
323 }
324 else if ( mLayer->providerType() == QLatin1String( "ogr" ) )
325 {
326 // if OGR_L_TestCapability(OLCStringsAsUTF8) returns true, OGR provider encoding can be set to only UTF-8
327 // so make encoding box grayed out
328 cboProviderEncoding->addItem( mLayer->dataProvider()->encoding() );
329 cboProviderEncoding->setEnabled( false );
330 }
331 else
332 {
333 // other providers do not use mEncoding, so hide the group completely
334 mDataSourceEncodingFrame->hide();
335 }
336 }
337
338 mCrsSelector->setCrs( mLayer->crs() );
339
340 //insert existing join info
341 const QList<QgsVectorLayerJoinInfo> &joins = mLayer->vectorJoins();
342 for ( const QgsVectorLayerJoinInfo &join : joins )
343 {
344 addJoinToTreeWidget( join );
345 }
346
347 mOldJoins = mLayer->vectorJoins();
348
349 // Legend tab
350 mLegendWidget->setMapCanvas( mCanvas );
351 mLegendWidget->setLayer( mLayer );
352 mLegendConfigEmbeddedWidget->setLayer( mLayer );
353
354 //insert existing dimension info
355 QgsMapLayerServerProperties *serverProperties = static_cast<QgsMapLayerServerProperties *>( mLayer->serverProperties() );
356 const QList<QgsMapLayerServerProperties::WmsDimensionInfo> &wmsDims = serverProperties->wmsDimensions();
357 for ( const QgsMapLayerServerProperties::WmsDimensionInfo &dim : wmsDims )
358 {
359 addWmsDimensionInfoToTreeWidget( dim );
360 }
361
362 QString myStyle = QgsApplication::reportStyleSheet();
363 myStyle.append( QStringLiteral( "body { margin: 10px; }\n " ) );
364 teMetadataViewer->clear();
365 teMetadataViewer->document()->setDefaultStyleSheet( myStyle );
366 teMetadataViewer->setHtml( htmlMetadata() );
367 teMetadataViewer->setOpenLinks( false );
368 connect( teMetadataViewer, &QTextBrowser::anchorClicked, this, &QgsVectorLayerProperties::openUrl );
369 mMetadataFilled = true;
370
371 QgsSettings settings;
372 // if dialog hasn't been opened/closed yet, default to Styles tab, which is used most often
373 // this will be read by restoreOptionsBaseUi()
374 if ( !settings.contains( QStringLiteral( "/Windows/VectorLayerProperties/tab" ) ) )
375 {
376 settings.setValue( QStringLiteral( "Windows/VectorLayerProperties/tab" ), mOptStackedWidget->indexOf( mOptsPage_Style ) );
377 }
378
379 QList<QgsMapLayer *> dependencySources;
380 const QSet<QgsMapLayerDependency> constDependencies = mLayer->dependencies();
381 for ( const QgsMapLayerDependency &dep : constDependencies )
382 {
383 QgsMapLayer *layer = QgsProject::instance()->mapLayer( dep.layerId() );
384 if ( layer )
385 dependencySources << layer;
386 }
387
388 mLayersDependenciesTreeModel = new QgsLayerTreeFilterProxyModel( this );
389 mLayersDependenciesTreeModel->setLayerTreeModel( new QgsLayerTreeModel( QgsProject::instance()->layerTreeRoot(), mLayersDependenciesTreeModel ) );
390 mLayersDependenciesTreeModel->setCheckedLayers( dependencySources );
391 connect( QgsProject::instance(), &QObject::destroyed, this, [=] { mLayersDependenciesTreeView->setModel( nullptr ); } );
392 mLayersDependenciesTreeView->setModel( mLayersDependenciesTreeModel );
393
394 mRefreshSettingsWidget->setLayer( mLayer );
395 mMapLayerServerPropertiesWidget->setHasWfsTitle( true );
396 mMapLayerServerPropertiesWidget->setServerProperties( mLayer->serverProperties() );
397
398 // auxiliary layer
399 QMenu *menu = new QMenu( this );
400
401 mAuxiliaryLayerActionNew = new QAction( tr( "Create" ), this );
402 menu->addAction( mAuxiliaryLayerActionNew );
403 connect( mAuxiliaryLayerActionNew, &QAction::triggered, this, &QgsVectorLayerProperties::onAuxiliaryLayerNew );
404
405 mAuxiliaryLayerActionClear = new QAction( tr( "Clear" ), this );
406 menu->addAction( mAuxiliaryLayerActionClear );
407 connect( mAuxiliaryLayerActionClear, &QAction::triggered, this, &QgsVectorLayerProperties::onAuxiliaryLayerClear );
408
409 mAuxiliaryLayerActionDelete = new QAction( tr( "Delete" ), this );
410 menu->addAction( mAuxiliaryLayerActionDelete );
411 connect( mAuxiliaryLayerActionDelete, &QAction::triggered, this, &QgsVectorLayerProperties::onAuxiliaryLayerDelete );
412
413 mAuxiliaryLayerActionExport = new QAction( tr( "Export" ), this );
414 menu->addAction( mAuxiliaryLayerActionExport );
415 connect( mAuxiliaryLayerActionExport, &QAction::triggered, this, [=] { emit exportAuxiliaryLayer( mLayer->auxiliaryLayer() ); } );
416
417 mAuxiliaryStorageActions->setMenu( menu );
418
419 connect( mAuxiliaryStorageFieldsDeleteBtn, &QPushButton::clicked, this, &QgsVectorLayerProperties::onAuxiliaryLayerDeleteField );
420 connect( mAuxiliaryStorageFieldsAddBtn, &QPushButton::clicked, this, &QgsVectorLayerProperties::onAuxiliaryLayerAddField );
421
422 updateAuxiliaryStoragePage();
423
424 mOptsPage_Information->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#information-properties" ) );
425 mOptsPage_Source->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#source-properties" ) );
426 mOptsPage_Style->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#symbology-properties" ) );
427 mOptsPage_Labels->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#labels-properties" ) );
428 mOptsPage_Masks->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#masks-properties" ) );
429 mOptsPage_Diagrams->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#diagrams-properties" ) );
430 mOptsPage_SourceFields->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#fields-properties" ) );
431 mOptsPage_AttributesForm->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#attributes-form-properties" ) );
432 mOptsPage_Joins->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#joins-properties" ) );
433 mOptsPage_AuxiliaryStorage->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#auxiliary-storage-properties" ) );
434 mOptsPage_Actions->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#actions-properties" ) );
435 mOptsPage_Display->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#display-properties" ) );
436 mOptsPage_Rendering->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#rendering-properties" ) );
437 mOptsPage_Temporal->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#temporal-properties" ) );
438 mOptsPage_Variables->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#variables-properties" ) );
439 mOptsPage_Metadata->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#metadata-properties" ) );
440 mOptsPage_DataDependencies->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#dependencies-properties" ) );
441 mOptsPage_Legend->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#legend-properties" ) );
442 mOptsPage_Server->setProperty( "helpPage", QStringLiteral( "working_with_vector/vector_properties.html#qgis-server-properties" ) );
443
444
446
447 initialize();
448}
449
450void QgsVectorLayerProperties::toggleEditing()
451{
452 if ( !mLayer )
453 return;
454
455 emit toggleEditing( mLayer );
456
457 setPbnQueryBuilderEnabled();
458}
459
460void QgsVectorLayerProperties::insertField()
461{
462 // Convert the selected field to an expression and
463 // insert it into the action at the cursor position
464 if ( mMapTipFieldComboBox->currentField().isEmpty() )
465 return;
466 QString expression = QStringLiteral( "[%\"" );
467 expression += mMapTipFieldComboBox->currentField();
468 expression += QLatin1String( "\"%]" );
469
470 mMapTipWidget->insertText( expression );
471}
472
473void QgsVectorLayerProperties::insertOrEditExpression()
474{
475 // Get the linear indexes if the start and end of the selection
476 int selectionStart = mMapTipWidget->selectionStart();
477 int selectionEnd = mMapTipWidget->selectionEnd();
478 QString expression = QgsExpressionFinder::findAndSelectActiveExpression( mMapTipWidget );
479
480 QgsExpressionContext context = createExpressionContext();
481 QgsExpressionBuilderDialog exprDlg( mLayer, expression, this, QStringLiteral( "generic" ), context );
482
483 exprDlg.setWindowTitle( tr( "Insert Expression" ) );
484 if ( exprDlg.exec() == QDialog::Accepted && !exprDlg.expressionText().trimmed().isEmpty() )
485 mMapTipWidget->insertText( "[%" + exprDlg.expressionText().trimmed() + "%]" );
486 else // Restore the selection
487 mMapTipWidget->setLinearSelection( selectionStart, selectionEnd );
488}
489
491{
492 if ( !mSourceWidget )
493 {
494 mSourceWidget = QgsGui::sourceWidgetProviderRegistry()->createWidget( mLayer );
495 if ( mSourceWidget )
496 {
497 QHBoxLayout *layout = new QHBoxLayout();
498 layout->addWidget( mSourceWidget );
499 mSourceGroupBox->setLayout( layout );
500 if ( !mSourceWidget->groupTitle().isEmpty() )
501 mSourceGroupBox->setTitle( mSourceWidget->groupTitle() );
502
503 mSourceGroupBox->show();
504
505 connect( mSourceWidget, &QgsProviderSourceWidget::validChanged, this, [=]( bool isValid ) {
506 buttonBox->button( QDialogButtonBox::Apply )->setEnabled( isValid );
507 buttonBox->button( QDialogButtonBox::Ok )->setEnabled( isValid );
508 } );
509 }
510 }
511
512 if ( mSourceWidget )
513 {
514 mSourceWidget->setMapCanvas( mCanvas );
515 mSourceWidget->setSourceUri( mLayer->source() );
516 }
517
518 // populate the general information
519 mLayerOrigNameLineEdit->setText( mLayer->name() );
520 mBackupCrs = mLayer->crs();
521
522 //see if we are dealing with a pg layer here
523 mSubsetGroupBox->setEnabled( true );
524 txtSubsetSQL->setText( mLayer->subsetString() );
525 // if the user is allowed to type an adhoc query, the app will crash if the query
526 // is bad. For this reason, the sql box is disabled and the query must be built
527 // using the query builder, either by typing it in by hand or using the buttons, etc
528 // on the builder. If the ability to enter a query directly into the box is required,
529 // a mechanism to check it must be implemented.
530 txtSubsetSQL->setReadOnly( true );
531 txtSubsetSQL->setCaretWidth( 0 );
532 txtSubsetSQL->setCaretLineVisible( false );
533 setPbnQueryBuilderEnabled();
534 if ( mLayer->dataProvider() && !mLayer->dataProvider()->supportsSubsetString() )
535 {
536 // hide subset box entirely if not supported by data provider
537 mSubsetGroupBox->hide();
538 }
539
540 mDisplayExpressionWidget->setField( mLayer->displayExpression() );
541 mEnableMapTips->setChecked( mLayer->mapTipsEnabled() );
542 mMapTipWidget->setText( mLayer->mapTipTemplate() );
543
544 // set up the scale based layer visibility stuff....
545 mScaleRangeWidget->setScaleRange( mLayer->minimumScale(), mLayer->maximumScale() );
546 mScaleVisibilityGroupBox->setChecked( mLayer->hasScaleBasedVisibility() );
547 mScaleRangeWidget->setMapCanvas( mCanvas );
548
549 mUseReferenceScaleGroupBox->setChecked( mLayer->renderer() && mLayer->renderer()->referenceScale() > 0 );
550 mReferenceScaleWidget->setShowCurrentScaleButton( true );
551 mReferenceScaleWidget->setMapCanvas( mCanvas );
552 if ( mUseReferenceScaleGroupBox->isChecked() )
553 mReferenceScaleWidget->setScale( mLayer->renderer()->referenceScale() );
554 else if ( mCanvas )
555 mReferenceScaleWidget->setScale( mCanvas->scale() );
556
557 // get simplify drawing configuration
558 const QgsVectorSimplifyMethod &simplifyMethod = mLayer->simplifyMethod();
560 mSimplifyDrawingSpinBox->setValue( simplifyMethod.threshold() );
561 mSimplifyDrawingSpinBox->setClearValue( 1.0 );
562
563 QgsVectorLayerSelectionProperties *selectionProperties = qobject_cast<QgsVectorLayerSelectionProperties *>( mLayer->selectionProperties() );
564 if ( selectionProperties->selectionColor().isValid() )
565 {
566 mSelectionColorButton->setColor( selectionProperties->selectionColor() );
567 }
568 if ( QgsSymbol *symbol = selectionProperties->selectionSymbol() )
569 {
570 mSelectionSymbolButton->setSymbol( symbol->clone() );
571 }
572 switch ( selectionProperties->selectionRenderingMode() )
573 {
575 mRadioDefaultSelectionColor->setChecked( true );
576 break;
577
579 {
580 if ( selectionProperties->selectionColor().isValid() )
581 {
582 mRadioOverrideSelectionColor->setChecked( true );
583 }
584 else
585 {
586 mRadioDefaultSelectionColor->setChecked( true );
587 }
588 break;
589 }
590
592 if ( selectionProperties->selectionSymbol() )
593 {
594 mRadioOverrideSelectionSymbol->setChecked( true );
595 }
596 else
597 {
598 mRadioDefaultSelectionColor->setChecked( true );
599 }
600 break;
601 }
602
603 QString remark = QStringLiteral( " (%1)" ).arg( tr( "Not supported" ) );
604 const QgsVectorDataProvider *provider = mLayer->dataProvider();
605 if ( !( provider && ( provider->capabilities() & Qgis::VectorProviderCapability::SimplifyGeometries ) ) )
606 {
607 mSimplifyDrawingAtProvider->setChecked( false );
608 mSimplifyDrawingAtProvider->setEnabled( false );
609 if ( !mSimplifyDrawingAtProvider->text().endsWith( remark ) )
610 mSimplifyDrawingAtProvider->setText( mSimplifyDrawingAtProvider->text().append( remark ) );
611 }
612 else
613 {
614 mSimplifyDrawingAtProvider->setChecked( !simplifyMethod.forceLocalOptimization() );
615 mSimplifyDrawingAtProvider->setEnabled( mSimplifyDrawingGroupBox->isChecked() );
616 if ( mSimplifyDrawingAtProvider->text().endsWith( remark ) )
617 {
618 QString newText = mSimplifyDrawingAtProvider->text();
619 newText.chop( remark.size() );
620 mSimplifyDrawingAtProvider->setText( newText );
621 }
622 }
623
624 // disable simplification for point layers, now it is not implemented
625 if ( mLayer->geometryType() == Qgis::GeometryType::Point )
626 {
627 mSimplifyDrawingGroupBox->setChecked( false );
628 mSimplifyDrawingGroupBox->setEnabled( false );
629 }
630
631 // Default local simplification algorithm
632 mSimplifyAlgorithmComboBox->addItem( tr( "Distance" ), QVariant::fromValue( Qgis::VectorSimplificationAlgorithm::Distance ) );
633 mSimplifyAlgorithmComboBox->addItem( tr( "SnapToGrid" ), QVariant::fromValue( Qgis::VectorSimplificationAlgorithm::SnapToGrid ) );
634 mSimplifyAlgorithmComboBox->addItem( tr( "Visvalingam" ), QVariant::fromValue( Qgis::VectorSimplificationAlgorithm::Visvalingam ) );
635 mSimplifyAlgorithmComboBox->setCurrentIndex( mSimplifyAlgorithmComboBox->findData( QVariant::fromValue( simplifyMethod.simplifyAlgorithm() ) ) );
636
637 QStringList myScalesList = Qgis::defaultProjectScales().split( ',' );
638 myScalesList.append( QStringLiteral( "1:1" ) );
639 mSimplifyMaximumScaleComboBox->updateScales( myScalesList );
640 mSimplifyMaximumScaleComboBox->setScale( simplifyMethod.maximumScale() );
641
642 mForceRasterCheckBox->setChecked( mLayer->renderer() && mLayer->renderer()->forceRasterRender() );
643
644 mRefreshSettingsWidget->syncToLayer();
645 mMapLayerServerPropertiesWidget->sync();
646
647 mRefreshLayerNotificationCheckBox->setChecked( mLayer->isRefreshOnNotifyEnabled() );
648 mNotificationMessageCheckBox->setChecked( !mLayer->refreshOnNotifyMessage().isEmpty() );
649 mNotifyMessagValueLineEdit->setText( mLayer->refreshOnNotifyMessage() );
650
651
652 // load appropriate symbology page (V1 or V2)
653 updateSymbologyPage();
654
655 mActionDialog->init( *mLayer->actions(), mLayer->attributeTableConfig() );
656
657 if ( labelingDialog )
658 labelingDialog->adaptToLayer();
659
660 mSourceFieldsPropertiesDialog->init();
661 mAttributesFormPropertiesDialog->init();
662
663 // set initial state for variable editor
664 updateVariableEditor();
665
666 if ( diagramPropertiesDialog )
667 diagramPropertiesDialog->syncToOwnLayer();
668
669 // sync all plugin dialogs
670 for ( QgsMapLayerConfigWidget *page : std::as_const( mConfigWidgets ) )
671 {
672 page->syncToLayer( mLayer );
673 }
674
675 mMetadataWidget->setMetadata( &mLayer->metadata() );
676
677 mTemporalWidget->syncToLayer();
678
679 mLegendWidget->setLayer( mLayer );
680}
681
683{
684 if ( labelingDialog )
685 {
686 labelingDialog->writeSettingsToLayer();
687 }
688 mBackupCrs = mLayer->crs();
689 // apply legend settings
690 mLegendWidget->applyToLayer();
691 mLegendConfigEmbeddedWidget->applyToLayer();
692
693 // save metadata
694 mMetadataWidget->acceptMetadata();
695 mMetadataFilled = false;
696
697 // save masking settings
698 if ( mMaskingWidget )
699 mMaskingWidget->apply();
700
701 // set up the scale based layer visibility stuff....
702 mLayer->setScaleBasedVisibility( mScaleVisibilityGroupBox->isChecked() );
703 mLayer->setMaximumScale( mScaleRangeWidget->maximumScale() );
704 mLayer->setMinimumScale( mScaleRangeWidget->minimumScale() );
705
706 // provider-specific options
707 if ( mLayer->dataProvider() )
708 {
710 {
711 mLayer->setProviderEncoding( cboProviderEncoding->currentText() );
712 }
713 }
714
715 mLayer->setDisplayExpression( mDisplayExpressionWidget->asExpression() );
716 mLayer->setMapTipsEnabled( mEnableMapTips->isChecked() );
717 mLayer->setMapTipTemplate( mMapTipWidget->text() );
718
719 mLayer->actions()->clearActions();
720 const auto constActions = mActionDialog->actions();
721 for ( const QgsAction &action : constActions )
722 {
723 mLayer->actions()->addAction( action );
724 }
725 QgsAttributeTableConfig attributeTableConfig = mLayer->attributeTableConfig();
726 attributeTableConfig.update( mLayer->fields() );
727 attributeTableConfig.setActionWidgetStyle( mActionDialog->attributeTableWidgetStyle() );
728 QVector<QgsAttributeTableConfig::ColumnConfig> columns = attributeTableConfig.columns();
729
730 for ( int i = 0; i < columns.size(); ++i )
731 {
732 if ( columns.at( i ).type == QgsAttributeTableConfig::Action )
733 {
734 columns[i].hidden = !mActionDialog->showWidgetInAttributeTable();
735 }
736 }
737
738 attributeTableConfig.setColumns( columns );
739
740 mLayer->setAttributeTableConfig( attributeTableConfig );
741
742 mLayer->setName( mLayerOrigNameLineEdit->text() );
743
744 mAttributesFormPropertiesDialog->apply();
745 mSourceFieldsPropertiesDialog->apply();
746
747 // Update temporal properties
748 mTemporalWidget->saveTemporalProperties();
749
750 if ( mLayer->renderer() )
751 {
752 QgsRendererPropertiesDialog *dlg = static_cast<QgsRendererPropertiesDialog *>( widgetStackRenderers->currentWidget() );
753 dlg->apply();
754 }
755
756 //apply diagram settings
757 diagramPropertiesDialog->apply();
758
759 // apply all plugin dialogs
760 for ( QgsMapLayerConfigWidget *page : std::as_const( mConfigWidgets ) )
761 {
762 page->apply();
763 }
764
765 if ( mMapLayerServerPropertiesWidget->save() )
766 mMetadataFilled = false;
767
768 //layer simplify drawing configuration
770 if ( mSimplifyDrawingGroupBox->isChecked() )
771 {
773 if ( mSimplifyDrawingSpinBox->value() > 1 )
775 }
776 QgsVectorSimplifyMethod simplifyMethod = mLayer->simplifyMethod();
777 simplifyMethod.setSimplifyHints( simplifyHints );
778 simplifyMethod.setSimplifyAlgorithm( mSimplifyAlgorithmComboBox->currentData().value<Qgis::VectorSimplificationAlgorithm>() );
779 simplifyMethod.setThreshold( mSimplifyDrawingSpinBox->value() );
780 simplifyMethod.setForceLocalOptimization( !mSimplifyDrawingAtProvider->isChecked() );
781 simplifyMethod.setMaximumScale( mSimplifyMaximumScaleComboBox->scale() );
782 mLayer->setSimplifyMethod( simplifyMethod );
783
784 if ( mLayer->renderer() )
785 {
786 mLayer->renderer()->setForceRasterRender( mForceRasterCheckBox->isChecked() );
787 mLayer->renderer()->setReferenceScale( mUseReferenceScaleGroupBox->isChecked() ? mReferenceScaleWidget->scale() : -1 );
788 }
789
790 QgsVectorLayerSelectionProperties *selectionProperties = qobject_cast<QgsVectorLayerSelectionProperties *>( mLayer->selectionProperties() );
791 if ( mSelectionColorButton->color() != mSelectionColorButton->defaultColor() )
792 selectionProperties->setSelectionColor( mSelectionColorButton->color() );
793 else
794 selectionProperties->setSelectionColor( QColor() );
795 if ( QgsSymbol *symbol = mSelectionSymbolButton->symbol() )
796 selectionProperties->setSelectionSymbol( symbol->clone() );
797
798 if ( mRadioOverrideSelectionSymbol->isChecked() )
799 {
801 }
802 else if ( mRadioOverrideSelectionColor->isChecked() )
803 {
805 }
806 else
807 {
809 }
810
811 mRefreshSettingsWidget->saveToLayer();
812
813 mLayer->setRefreshOnNotifyEnabled( mRefreshLayerNotificationCheckBox->isChecked() );
814 mLayer->setRefreshOnNofifyMessage( mNotificationMessageCheckBox->isChecked() ? mNotifyMessagValueLineEdit->text() : QString() );
815
816 mOldJoins = mLayer->vectorJoins();
817
818 //save variables
819 QgsExpressionContextUtils::setLayerVariables( mLayer, mVariableEditor->variablesInActiveScope() );
820 updateVariableEditor();
821
822 // save dependencies
823 QSet<QgsMapLayerDependency> deps;
824 const auto checkedLayers = mLayersDependenciesTreeModel->checkedLayers();
825 for ( const QgsMapLayer *layer : checkedLayers )
826 deps << QgsMapLayerDependency( layer->id() );
827 if ( !mLayer->setDependencies( deps ) )
828 {
829 QMessageBox::warning( nullptr, tr( "Save Dependency" ), tr( "This configuration introduces a cycle in data dependencies and will be ignored." ) );
830 }
831
832 // Why is this here? Well, we if we're making changes to the layer's source then potentially
833 // we are changing the geometry type of the layer, or even going from spatial <-> non spatial types.
834 // So we need to ensure that anything from the dialog which sets things like renderer properties
835 // happens BEFORE we change the source, otherwise we might end up with a renderer which is not
836 // compatible with the new geometry type of the layer. (And likewise for other properties like
837 // fields!)
838 bool dialogNeedsResync = false;
839 if ( mSourceWidget )
840 {
841 const QString newSource = mSourceWidget->sourceUri();
842 if ( newSource != mLayer->source() )
843 {
844 mLayer->setDataSource( newSource, mLayer->name(), mLayer->providerType(), QgsDataProvider::ProviderOptions(), Qgis::DataProviderReadFlags() );
845
846 // resync dialog to layer's new state -- this allows any changed layer properties
847 // (such as a forced creation of a new renderer compatible with the new layer, new field configuration, etc)
848 // to show in the dialog correctly
849 dialogNeedsResync = true;
850 }
851 }
852 // now apply the subset string AFTER setting the layer's source. It's messy, but the subset string
853 // can form part of the layer's source, but it WON'T be present in the URI returned by the source widget!
854 // If we don't apply the subset string AFTER changing the source, then the subset string will be lost.
855 mSubsetGroupBox->setEnabled( true );
856 if ( txtSubsetSQL->text() != mLayer->subsetString() )
857 {
858 // set the subset sql for the layer
859 mLayer->setSubsetString( txtSubsetSQL->text() );
860 mMetadataFilled = false;
861 // need to resync the dialog, the subset string may have changed the layer's geometry type!
862 dialogNeedsResync = true;
863 }
864 mOriginalSubsetSQL = mLayer->subsetString();
865
866 if ( dialogNeedsResync )
867 syncToLayer();
868
869 mLayer->triggerRepaint();
870 // notify the project we've made a change
871 mProjectDirtyBlocker.reset();
873 mProjectDirtyBlocker = std::make_unique<QgsProjectDirtyBlocker>( QgsProject::instance() );
874}
875
877{
878 if ( mOldJoins != mLayer->vectorJoins() )
879 {
880 // need to undo changes in vector layer joins - they are applied directly to the layer (not in apply())
881 // so other parts of the properties dialog can use the fields from the joined layers
882
883 const auto constVectorJoins = mLayer->vectorJoins();
884 for ( const QgsVectorLayerJoinInfo &info : constVectorJoins )
885 mLayer->removeJoin( info.joinLayerId() );
886
887 for ( const QgsVectorLayerJoinInfo &info : std::as_const( mOldJoins ) )
888 mLayer->addJoin( info );
889 }
890
891 if ( mOriginalSubsetSQL != mLayer->subsetString() )
892 {
893 // need to undo changes in subset string - they are applied directly to the layer (not in apply())
894 // by QgsQueryBuilder::accept()
895
896 mLayer->setSubsetString( mOriginalSubsetSQL );
897 }
898
899 // Store it because QgsLayerPropertiesDialog::rollback() calls syncToLayer() which
900 // resets the backupCrs
901 const QgsCoordinateReferenceSystem backupCrs { mBackupCrs };
902
904
905 if ( backupCrs != mLayer->crs() )
906 mLayer->setCrs( backupCrs );
907}
908
909void QgsVectorLayerProperties::pbnQueryBuilder_clicked()
910{
911 // launch the query builder
913
914 // Set the sql in the query builder to the same in the prop dialog
915 // (in case the user has already changed it)
916 dialog->setSubsetString( txtSubsetSQL->text() );
917 // Open the query builder
918 if ( dialog->exec() )
919 {
920 // if the sql is changed, update it in the prop subset text box
921 txtSubsetSQL->setText( dialog->subsetString() );
922 //TODO If the sql is changed in the prop dialog, the layer extent should be recalculated
923
924 // The datasource for the layer needs to be updated with the new sql since this gets
925 // saved to the project file. This should happen at the map layer level...
926 }
927 // delete the query builder object
928 delete dialog;
929}
930
931void QgsVectorLayerProperties::pbnIndex_clicked()
932{
933 QgsVectorDataProvider *pr = mLayer->dataProvider();
934 if ( pr )
935 {
936 setCursor( Qt::WaitCursor );
937 bool errval = pr->createSpatialIndex();
938 setCursor( Qt::ArrowCursor );
939 if ( errval )
940 {
941 pbnIndex->setEnabled( false );
942 pbnIndex->setText( tr( "Spatial Index Exists" ) );
943 QMessageBox::information( this, tr( "Spatial Index" ), tr( "Creation of spatial index successful" ) );
944 }
945 else
946 {
947 QMessageBox::warning( this, tr( "Spatial Index" ), tr( "Creation of spatial index failed" ) );
948 }
949 }
950}
951
952QString QgsVectorLayerProperties::htmlMetadata()
953{
954 return mLayer->htmlMetadata();
955}
956
957void QgsVectorLayerProperties::mCrsSelector_crsChanged( const QgsCoordinateReferenceSystem &crs )
958{
959 QgsDatumTransformDialog::run( crs, QgsProject::instance()->crs(), this, mCanvas, tr( "Select Transformation for the vector layer" ) );
960 mLayer->setCrs( crs );
961 mMetadataFilled = false;
962 mMetadataWidget->crsChanged();
963}
964
965void QgsVectorLayerProperties::saveMultipleStylesAs()
966{
967 QgsMapLayerSaveStyleDialog dlg( mLayer );
968 dlg.setSaveOnlyCurrentStyle( false );
969 QgsSettings settings;
970
971 if ( dlg.exec() )
972 {
973 apply();
974
975 // Store the original style, that we can restore at the end
976 const QString originalStyle { mLayer->styleManager()->currentStyle() };
977 const QListWidget *stylesWidget { dlg.stylesWidget() };
978
979 // Collect selected (checked) styles for export/save
980 QStringList stylesSelected;
981 for ( int i = 0; i < stylesWidget->count(); i++ )
982 {
983 if ( stylesWidget->item( i )->checkState() == Qt::CheckState::Checked )
984 {
985 stylesSelected.push_back( stylesWidget->item( i )->text() );
986 }
987 }
988
989 if ( !stylesSelected.isEmpty() )
990 {
991 int styleIndex = 0;
992 for ( const QString &styleName : std::as_const( stylesSelected ) )
993 {
994 bool defaultLoadedFlag = false;
995
996 StyleType type = dlg.currentStyleType();
997 mLayer->styleManager()->setCurrentStyle( styleName );
998 switch ( type )
999 {
1000 case QML:
1001 case SLD:
1002 {
1003 QString message;
1004 const QString filePath { dlg.outputFilePath() };
1005 const QFileInfo fi { filePath };
1006 QString safePath { QString( filePath ).replace( fi.baseName(), QStringLiteral( "%1_%2" ).arg( fi.baseName(), QgsFileUtils::stringToSafeFilename( styleName ) ) ) };
1007 if ( styleIndex > 0 && stylesSelected.count() > 1 )
1008 {
1009 int i = 1;
1010 while ( QFile::exists( safePath ) )
1011 {
1012 const QFileInfo fi { safePath };
1013 safePath = QString( safePath ).replace( '.' + fi.completeSuffix(), QStringLiteral( "_%1.%2" ).arg( QString::number( i ), fi.completeSuffix() ) );
1014 i++;
1015 }
1016 }
1017 if ( type == QML )
1018 message = mLayer->saveNamedStyle( safePath, defaultLoadedFlag, dlg.styleCategories() );
1019 else
1020 message = mLayer->saveSldStyle( safePath, defaultLoadedFlag );
1021
1022 //reset if the default style was loaded OK only
1023 if ( defaultLoadedFlag )
1024 {
1025 syncToLayer();
1026 }
1027 else
1028 {
1029 //let the user know what went wrong
1030 QMessageBox::information( this, tr( "Save Style" ), message );
1031 }
1032
1033 break;
1034 }
1035 case DatasourceDatabase:
1036 {
1037 QString infoWindowTitle = QObject::tr( "Save style '%1' to DB (%2)" )
1038 .arg( styleName, mLayer->providerType() );
1039 QString msgError;
1040
1041 QgsMapLayerSaveStyleDialog::SaveToDbSettings dbSettings = dlg.saveToDbSettings();
1042
1043 // If a name is defined, we add _1 etc. else we use the style name
1044 QString name { dbSettings.name };
1045 if ( name.isEmpty() )
1046 {
1047 name = styleName;
1048 }
1049 else
1050 {
1051 name += QStringLiteral( "_%1" ).arg( styleName );
1052 QStringList ids, names, descriptions;
1053 mLayer->listStylesInDatabase( ids, names, descriptions, msgError );
1054 int i = 1;
1055 while ( names.contains( name ) )
1056 {
1057 name = QStringLiteral( "%1 %2" ).arg( name, QString::number( i ) );
1058 i++;
1059 }
1060 }
1061
1062 QString errorMessage;
1063 if ( QgsProviderRegistry::instance()->styleExists( mLayer->providerType(), mLayer->source(), dbSettings.name, errorMessage ) )
1064 {
1065 if ( QMessageBox::question( nullptr, QObject::tr( "Save style in database" ), QObject::tr( "A matching style already exists in the database for this layer. Do you want to overwrite it?" ), QMessageBox::Yes | QMessageBox::No ) == QMessageBox::No )
1066 {
1067 return;
1068 }
1069 }
1070 else if ( !errorMessage.isEmpty() )
1071 {
1072 QMessageBox::warning( this, infoWindowTitle, errorMessage );
1073 return;
1074 }
1075
1076 mLayer->saveStyleToDatabase( name, dbSettings.description, dbSettings.isDefault, dbSettings.uiFileContent, msgError, dlg.styleCategories() );
1077
1078 if ( !msgError.isNull() )
1079 {
1080 QMessageBox::warning( this, infoWindowTitle, msgError );
1081 }
1082 else
1083 {
1084 QMessageBox::information( this, infoWindowTitle, tr( "Style '%1' saved" ).arg( styleName ) );
1085 }
1086 break;
1087 }
1088 case UserDatabase:
1089 break;
1090 }
1091 styleIndex++;
1092 }
1093 // Restore original style
1094 mLayer->styleManager()->setCurrentStyle( originalStyle );
1095 }
1096 } // Nothing selected!
1097}
1098
1099void QgsVectorLayerProperties::aboutToShowStyleMenu()
1100{
1101 // this should be unified with QgsRasterLayerProperties::aboutToShowStyleMenu()
1102 QMenu *m = qobject_cast<QMenu *>( sender() );
1103 m->clear();
1104
1105 m->addAction( mActionLoadStyle );
1106 m->addAction( mActionSaveStyle );
1107
1108 // If we have multiple styles, offer an option to save them at once
1109 if ( mLayer->styleManager()->styles().count() > 1 )
1110 {
1111 mActionSaveStyle->setText( tr( "Save Current Style…" ) );
1112 m->addAction( mActionSaveMultipleStyles );
1113 }
1114 else
1115 {
1116 mActionSaveStyle->setText( tr( "Save Style…" ) );
1117 }
1118
1119 m->addSeparator();
1120 m->addAction( tr( "Save as Default" ), this, &QgsVectorLayerProperties::saveDefaultStyle );
1121 m->addAction( tr( "Restore Default" ), this, &QgsVectorLayerProperties::loadDefaultStyle );
1122
1123 // re-add style manager actions!
1124 m->addSeparator();
1126}
1127
1128void QgsVectorLayerProperties::mButtonAddJoin_clicked()
1129{
1130 if ( !mLayer )
1131 return;
1132
1133 QList<QgsMapLayer *> joinedLayers;
1134 const QList<QgsVectorLayerJoinInfo> &joins = mLayer->vectorJoins();
1135 joinedLayers.reserve( joins.size() );
1136 for ( int i = 0; i < joins.size(); ++i )
1137 {
1138 joinedLayers.append( joins[i].joinLayer() );
1139 }
1140
1141 QgsJoinDialog d( mLayer, joinedLayers );
1142 if ( d.exec() == QDialog::Accepted )
1143 {
1144 QgsVectorLayerJoinInfo info = d.joinInfo();
1145 //create attribute index if possible
1146 if ( d.createAttributeIndex() )
1147 {
1148 QgsVectorLayer *joinLayer = info.joinLayer();
1149 if ( joinLayer )
1150 {
1151 joinLayer->dataProvider()->createAttributeIndex( joinLayer->fields().indexFromName( info.joinFieldName() ) );
1152 }
1153 }
1154 mLayer->addJoin( info );
1155 addJoinToTreeWidget( info );
1156 setPbnQueryBuilderEnabled();
1157 mSourceFieldsPropertiesDialog->init();
1158 mAttributesFormPropertiesDialog->init();
1159 }
1160}
1161
1162void QgsVectorLayerProperties::mButtonEditJoin_clicked()
1163{
1164 QTreeWidgetItem *currentJoinItem = mJoinTreeWidget->currentItem();
1165 mJoinTreeWidget_itemDoubleClicked( currentJoinItem, 0 );
1166}
1167
1168void QgsVectorLayerProperties::mJoinTreeWidget_itemDoubleClicked( QTreeWidgetItem *item, int )
1169{
1170 if ( !mLayer || !item )
1171 {
1172 return;
1173 }
1174
1175 // if current item is a child item, we should use its parent to be able to edit join
1176 QTreeWidgetItem *currentJoinItem = item;
1177 if ( item->parent() )
1178 {
1179 currentJoinItem = item->parent();
1180 }
1181
1182
1183 QList<QgsMapLayer *> joinedLayers;
1184 QString joinLayerId = currentJoinItem->data( 0, Qt::UserRole ).toString();
1185 const QList<QgsVectorLayerJoinInfo> &joins = mLayer->vectorJoins();
1186 int j = -1;
1187 for ( int i = 0; i < joins.size(); ++i )
1188 {
1189 QgsVectorLayer *joinLayer = joins[i].joinLayer();
1190 if ( !joinLayer )
1191 continue; // invalid join (unresolved join layer)
1192
1193 if ( joinLayer->id() == joinLayerId )
1194 {
1195 j = i;
1196 }
1197 else
1198 {
1199 // remove already joined layers from possible list to be displayed in dialog
1200 joinedLayers.append( joinLayer );
1201 }
1202 }
1203 if ( j == -1 )
1204 {
1205 return;
1206 }
1207
1208 QgsJoinDialog d( mLayer, joinedLayers );
1209 d.setWindowTitle( tr( "Edit Vector Join" ) );
1210 d.setJoinInfo( joins[j] );
1211
1212 if ( d.exec() == QDialog::Accepted )
1213 {
1214 QgsVectorLayerJoinInfo info = d.joinInfo();
1215
1216 // remove old join
1217 mLayer->removeJoin( joinLayerId );
1218 int idx = mJoinTreeWidget->indexOfTopLevelItem( item );
1219 mJoinTreeWidget->takeTopLevelItem( idx );
1220
1221 // add the new edited
1222
1223 //create attribute index if possible
1224 if ( d.createAttributeIndex() )
1225 {
1226 QgsVectorLayer *joinLayer = info.joinLayer();
1227 if ( joinLayer )
1228 {
1229 joinLayer->dataProvider()->createAttributeIndex( joinLayer->fields().indexFromName( info.joinFieldName() ) );
1230 }
1231 }
1232 mLayer->addJoin( info );
1233 addJoinToTreeWidget( info, idx );
1234
1235 setPbnQueryBuilderEnabled();
1236 mSourceFieldsPropertiesDialog->init();
1237 mAttributesFormPropertiesDialog->init();
1238 }
1239}
1240
1241void QgsVectorLayerProperties::addJoinToTreeWidget( const QgsVectorLayerJoinInfo &join, const int insertIndex )
1242{
1243 QTreeWidgetItem *joinItem = new QTreeWidgetItem();
1244 joinItem->setFlags( Qt::ItemIsEnabled );
1245
1246 QgsVectorLayer *joinLayer = join.joinLayer();
1247 if ( !mLayer || !joinLayer )
1248 {
1249 return;
1250 }
1251
1252 joinItem->setText( 0, tr( "Join layer" ) );
1253 if ( mLayer->auxiliaryLayer() && mLayer->auxiliaryLayer()->id() == join.joinLayerId() )
1254 {
1255 return;
1256 }
1257
1258 joinItem->setText( 1, joinLayer->name() );
1259
1260 QFont f = joinItem->font( 0 );
1261 f.setBold( true );
1262 joinItem->setFont( 0, f );
1263 joinItem->setFont( 1, f );
1264
1265 joinItem->setData( 0, Qt::UserRole, join.joinLayerId() );
1266
1267 QTreeWidgetItem *childJoinField = new QTreeWidgetItem();
1268 childJoinField->setText( 0, tr( "Join field" ) );
1269 childJoinField->setText( 1, join.joinFieldName() );
1270 childJoinField->setFlags( Qt::ItemIsEnabled );
1271 joinItem->addChild( childJoinField );
1272
1273 QTreeWidgetItem *childTargetField = new QTreeWidgetItem();
1274 childTargetField->setText( 0, tr( "Target field" ) );
1275 childTargetField->setText( 1, join.targetFieldName() );
1276 joinItem->addChild( childTargetField );
1277
1278 QTreeWidgetItem *childMemCache = new QTreeWidgetItem();
1279 childMemCache->setText( 0, tr( "Cache join layer in virtual memory" ) );
1280 if ( join.isUsingMemoryCache() )
1281 childMemCache->setText( 1, QChar( 0x2714 ) );
1282 joinItem->addChild( childMemCache );
1283
1284 QTreeWidgetItem *childDynForm = new QTreeWidgetItem();
1285 childDynForm->setText( 0, tr( "Dynamic form" ) );
1286 if ( join.isDynamicFormEnabled() )
1287 childDynForm->setText( 1, QChar( 0x2714 ) );
1288 joinItem->addChild( childDynForm );
1289
1290 QTreeWidgetItem *childEditable = new QTreeWidgetItem();
1291 childEditable->setText( 0, tr( "Editable join layer" ) );
1292 if ( join.isEditable() )
1293 childEditable->setText( 1, QChar( 0x2714 ) );
1294 joinItem->addChild( childEditable );
1295
1296 QTreeWidgetItem *childUpsert = new QTreeWidgetItem();
1297 childUpsert->setText( 0, tr( "Upsert on edit" ) );
1298 if ( join.hasUpsertOnEdit() )
1299 childUpsert->setText( 1, QChar( 0x2714 ) );
1300 joinItem->addChild( childUpsert );
1301
1302 QTreeWidgetItem *childCascade = new QTreeWidgetItem();
1303 childCascade->setText( 0, tr( "Delete cascade" ) );
1304 if ( join.hasCascadedDelete() )
1305 childCascade->setText( 1, QChar( 0x2714 ) );
1306 joinItem->addChild( childCascade );
1307
1308 QTreeWidgetItem *childPrefix = new QTreeWidgetItem();
1309 childPrefix->setText( 0, tr( "Custom field name prefix" ) );
1310 childPrefix->setText( 1, join.prefix() );
1311 joinItem->addChild( childPrefix );
1312
1313 QTreeWidgetItem *childFields = new QTreeWidgetItem();
1314 childFields->setText( 0, tr( "Joined fields" ) );
1315 const QStringList *list = join.joinFieldNamesSubset();
1316 if ( list )
1317 childFields->setText( 1, QLocale().toString( list->count() ) );
1318 else
1319 childFields->setText( 1, tr( "all" ) );
1320 joinItem->addChild( childFields );
1321
1322 if ( insertIndex >= 0 )
1323 mJoinTreeWidget->insertTopLevelItem( insertIndex, joinItem );
1324 else
1325 mJoinTreeWidget->addTopLevelItem( joinItem );
1326
1327 mJoinTreeWidget->setCurrentItem( joinItem );
1328 mJoinTreeWidget->header()->setSectionResizeMode( QHeaderView::ResizeToContents );
1329}
1330
1331QgsExpressionContext QgsVectorLayerProperties::createExpressionContext() const
1332{
1333 return mContext;
1334}
1335
1336void QgsVectorLayerProperties::openPanel( QgsPanelWidget *panel )
1337{
1338 QDialog *dlg = new QDialog();
1339 QString key = QStringLiteral( "/UI/paneldialog/%1" ).arg( panel->panelTitle() );
1340 QgsSettings settings;
1341 dlg->restoreGeometry( settings.value( key ).toByteArray() );
1342 dlg->setWindowTitle( panel->panelTitle() );
1343 dlg->setLayout( new QVBoxLayout() );
1344 dlg->layout()->addWidget( panel );
1345 QDialogButtonBox *buttonBox = new QDialogButtonBox( QDialogButtonBox::Ok );
1346 connect( buttonBox, &QDialogButtonBox::accepted, dlg, &QDialog::accept );
1347 dlg->layout()->addWidget( buttonBox );
1348 dlg->exec();
1349 settings.setValue( key, dlg->saveGeometry() );
1350 panel->acceptPanel();
1351}
1352
1353void QgsVectorLayerProperties::mButtonRemoveJoin_clicked()
1354{
1355 QTreeWidgetItem *currentJoinItem = mJoinTreeWidget->currentItem();
1356 // if current item is a child item, we should use its parent to be able to remove join
1357 if ( currentJoinItem && currentJoinItem->parent() )
1358 {
1359 currentJoinItem = currentJoinItem->parent();
1360 }
1361
1362 if ( !mLayer || !currentJoinItem )
1363 {
1364 return;
1365 }
1366
1367 mLayer->removeJoin( currentJoinItem->data( 0, Qt::UserRole ).toString() );
1368 mJoinTreeWidget->takeTopLevelItem( mJoinTreeWidget->indexOfTopLevelItem( currentJoinItem ) );
1369 setPbnQueryBuilderEnabled();
1370 mSourceFieldsPropertiesDialog->init();
1371 mAttributesFormPropertiesDialog->init();
1372}
1373
1374
1375void QgsVectorLayerProperties::mButtonAddWmsDimension_clicked()
1376{
1377 if ( !mLayer )
1378 return;
1379
1380 // get wms dimensions name
1381 QStringList alreadyDefinedDimensions;
1382 QgsMapLayerServerProperties *serverProperties = static_cast<QgsMapLayerServerProperties *>( mLayer->serverProperties() );
1383 const QList<QgsMapLayerServerProperties::WmsDimensionInfo> &dims = serverProperties->wmsDimensions();
1384 for ( const QgsMapLayerServerProperties::WmsDimensionInfo &dim : dims )
1385 {
1386 alreadyDefinedDimensions << dim.name;
1387 }
1388
1389 QgsWmsDimensionDialog d( mLayer, alreadyDefinedDimensions );
1390 if ( d.exec() == QDialog::Accepted )
1391 {
1393 // save dimension
1394 serverProperties->addWmsDimension( info );
1395 addWmsDimensionInfoToTreeWidget( info );
1396 }
1397}
1398
1399void QgsVectorLayerProperties::mButtonEditWmsDimension_clicked()
1400{
1401 QTreeWidgetItem *currentWmsDimensionItem = mWmsDimensionsTreeWidget->currentItem();
1402 mWmsDimensionsTreeWidget_itemDoubleClicked( currentWmsDimensionItem, 0 );
1403}
1404
1405void QgsVectorLayerProperties::mWmsDimensionsTreeWidget_itemDoubleClicked( QTreeWidgetItem *item, int )
1406{
1407 if ( !mLayer || !item )
1408 {
1409 return;
1410 }
1411
1412 QString wmsDimName = item->data( 0, Qt::UserRole ).toString();
1413 QgsMapLayerServerProperties *serverProperties = static_cast<QgsMapLayerServerProperties *>( mLayer->serverProperties() );
1414 const QList<QgsMapLayerServerProperties::WmsDimensionInfo> &dims = serverProperties->wmsDimensions();
1415 QStringList alreadyDefinedDimensions;
1416 int j = -1;
1417 for ( int i = 0; i < dims.size(); ++i )
1418 {
1419 QString dimName = dims[i].name;
1420 if ( dimName == wmsDimName )
1421 {
1422 j = i;
1423 }
1424 else
1425 {
1426 alreadyDefinedDimensions << dimName;
1427 }
1428 }
1429 if ( j == -1 )
1430 {
1431 return;
1432 }
1433
1434 QgsWmsDimensionDialog d( mLayer, alreadyDefinedDimensions );
1435 d.setWindowTitle( tr( "Edit WMS Dimension" ) );
1436 d.setInfo( dims[j] );
1437
1438 if ( d.exec() == QDialog::Accepted )
1439 {
1441
1442 // remove old
1443 QgsMapLayerServerProperties *serverProperties = static_cast<QgsMapLayerServerProperties *>( mLayer->serverProperties() );
1444 serverProperties->removeWmsDimension( wmsDimName );
1445 int idx = mWmsDimensionsTreeWidget->indexOfTopLevelItem( item );
1446 mWmsDimensionsTreeWidget->takeTopLevelItem( idx );
1447
1448 // save new
1449 serverProperties->addWmsDimension( info );
1450 addWmsDimensionInfoToTreeWidget( info, idx );
1451 }
1452}
1453
1454void QgsVectorLayerProperties::addWmsDimensionInfoToTreeWidget( const QgsMapLayerServerProperties::WmsDimensionInfo &wmsDim, const int insertIndex )
1455{
1456 QTreeWidgetItem *wmsDimensionItem = new QTreeWidgetItem();
1457 wmsDimensionItem->setFlags( Qt::ItemIsEnabled );
1458
1459 wmsDimensionItem->setText( 0, tr( "Dimension" ) );
1460 wmsDimensionItem->setText( 1, wmsDim.name );
1461
1462 QFont f = wmsDimensionItem->font( 0 );
1463 f.setBold( true );
1464 wmsDimensionItem->setFont( 0, f );
1465 wmsDimensionItem->setFont( 1, f );
1466
1467 wmsDimensionItem->setData( 0, Qt::UserRole, wmsDim.name );
1468
1469 QTreeWidgetItem *childWmsDimensionField = new QTreeWidgetItem();
1470 childWmsDimensionField->setText( 0, tr( "Field" ) );
1471 childWmsDimensionField->setText( 1, wmsDim.fieldName );
1472 childWmsDimensionField->setFlags( Qt::ItemIsEnabled );
1473 wmsDimensionItem->addChild( childWmsDimensionField );
1474
1475 QTreeWidgetItem *childWmsDimensionEndField = new QTreeWidgetItem();
1476 childWmsDimensionEndField->setText( 0, tr( "End field" ) );
1477 childWmsDimensionEndField->setText( 1, wmsDim.endFieldName );
1478 childWmsDimensionEndField->setFlags( Qt::ItemIsEnabled );
1479 wmsDimensionItem->addChild( childWmsDimensionEndField );
1480
1481 QTreeWidgetItem *childWmsDimensionUnits = new QTreeWidgetItem();
1482 childWmsDimensionUnits->setText( 0, tr( "Units" ) );
1483 childWmsDimensionUnits->setText( 1, wmsDim.units );
1484 childWmsDimensionUnits->setFlags( Qt::ItemIsEnabled );
1485 wmsDimensionItem->addChild( childWmsDimensionUnits );
1486
1487 QTreeWidgetItem *childWmsDimensionUnitSymbol = new QTreeWidgetItem();
1488 childWmsDimensionUnitSymbol->setText( 0, tr( "Unit symbol" ) );
1489 childWmsDimensionUnitSymbol->setText( 1, wmsDim.unitSymbol );
1490 childWmsDimensionUnitSymbol->setFlags( Qt::ItemIsEnabled );
1491 wmsDimensionItem->addChild( childWmsDimensionUnitSymbol );
1492
1493 QTreeWidgetItem *childWmsDimensionDefaultValue = new QTreeWidgetItem();
1494 childWmsDimensionDefaultValue->setText( 0, tr( "Default display" ) );
1495 childWmsDimensionDefaultValue->setText( 1, QgsMapLayerServerProperties::wmsDimensionDefaultDisplayLabels().value( wmsDim.defaultDisplayType ) );
1496 childWmsDimensionDefaultValue->setFlags( Qt::ItemIsEnabled );
1497 wmsDimensionItem->addChild( childWmsDimensionDefaultValue );
1498
1499 QTreeWidgetItem *childWmsDimensionRefValue = new QTreeWidgetItem();
1500 childWmsDimensionRefValue->setText( 0, tr( "Reference value" ) );
1501 childWmsDimensionRefValue->setText( 1, wmsDim.referenceValue.toString() );
1502 childWmsDimensionRefValue->setFlags( Qt::ItemIsEnabled );
1503 wmsDimensionItem->addChild( childWmsDimensionRefValue );
1504
1505 if ( insertIndex >= 0 )
1506 mWmsDimensionsTreeWidget->insertTopLevelItem( insertIndex, wmsDimensionItem );
1507 else
1508 mWmsDimensionsTreeWidget->addTopLevelItem( wmsDimensionItem );
1509
1510 mWmsDimensionsTreeWidget->setCurrentItem( wmsDimensionItem );
1511 mWmsDimensionsTreeWidget->header()->setSectionResizeMode( QHeaderView::ResizeToContents );
1512}
1513
1514void QgsVectorLayerProperties::mButtonRemoveWmsDimension_clicked()
1515{
1516 QTreeWidgetItem *currentWmsDimensionItem = mWmsDimensionsTreeWidget->currentItem();
1517 if ( !mLayer || !currentWmsDimensionItem )
1518 {
1519 return;
1520 }
1521
1522 QgsMapLayerServerProperties *serverProperties = static_cast<QgsMapLayerServerProperties *>( mLayer->serverProperties() );
1523 serverProperties->removeWmsDimension( currentWmsDimensionItem->data( 0, Qt::UserRole ).toString() );
1524 mWmsDimensionsTreeWidget->takeTopLevelItem( mWmsDimensionsTreeWidget->indexOfTopLevelItem( currentWmsDimensionItem ) );
1525}
1526
1527
1528void QgsVectorLayerProperties::updateSymbologyPage()
1529{
1530 //find out the type of renderer in the vectorlayer, create a dialog with these settings and add it to the form
1531 delete mRendererDialog;
1532 mRendererDialog = nullptr;
1533
1534 if ( mLayer->renderer() )
1535 {
1536 mRendererDialog = new QgsRendererPropertiesDialog( mLayer, QgsStyle::defaultStyle(), true, this );
1537 mRendererDialog->setDockMode( false );
1538 QgsSymbolWidgetContext context;
1539 context.setMapCanvas( mCanvas );
1540 context.setMessageBar( mMessageBar );
1541 mRendererDialog->setContext( context );
1542 connect( mRendererDialog, &QgsRendererPropertiesDialog::showPanel, this, &QgsVectorLayerProperties::openPanel );
1543 connect( mRendererDialog, &QgsRendererPropertiesDialog::layerVariablesChanged, this, &QgsVectorLayerProperties::updateVariableEditor );
1544 connect( mRendererDialog, &QgsRendererPropertiesDialog::widgetChanged, this, [=] { updateAuxiliaryStoragePage(); } );
1545 }
1546 else
1547 {
1548 mOptsPage_Style->setEnabled( false ); // hide symbology item
1549 }
1550
1551 if ( mRendererDialog )
1552 {
1553 mRendererDialog->layout()->setContentsMargins( 0, 0, 0, 0 );
1554 widgetStackRenderers->addWidget( mRendererDialog );
1555 widgetStackRenderers->setCurrentWidget( mRendererDialog );
1556 widgetStackRenderers->currentWidget()->layout()->setContentsMargins( 0, 0, 0, 0 );
1557 }
1558}
1559
1560void QgsVectorLayerProperties::setPbnQueryBuilderEnabled()
1561{
1562 pbnQueryBuilder->setEnabled( mLayer && mLayer->dataProvider() && mLayer->dataProvider()->supportsSubsetString() && !mLayer->isEditable() );
1563
1564 if ( mLayer && mLayer->isEditable() )
1565 {
1566 pbnQueryBuilder->setToolTip( tr( "Stop editing mode to enable this." ) );
1567 }
1568}
1569
1570void QgsVectorLayerProperties::pbnUpdateExtents_clicked()
1571{
1572 mLayer->updateExtents( true ); // force update whatever options activated
1573 mMetadataFilled = false;
1574}
1575
1577{
1579
1580 if ( index == mOptStackedWidget->indexOf( mOptsPage_Information ) && !mMetadataFilled )
1581 {
1582 // set the metadata contents (which can be expensive)
1583 teMetadataViewer->clear();
1584 teMetadataViewer->setHtml( htmlMetadata() );
1585 mMetadataFilled = true;
1586 }
1587 else if ( index == mOptStackedWidget->indexOf( mOptsPage_SourceFields ) || index == mOptStackedWidget->indexOf( mOptsPage_Joins ) )
1588 {
1589 // store any edited attribute form field configuration to prevent loss of edits when adding/removing fields and/or joins
1590 mAttributesFormPropertiesDialog->store();
1591 }
1592
1593 resizeAlltabs( index );
1594}
1595
1596void QgsVectorLayerProperties::mSimplifyDrawingGroupBox_toggled( bool checked )
1597{
1598 const QgsVectorDataProvider *provider = mLayer->dataProvider();
1599 if ( !( provider && ( provider->capabilities() & Qgis::VectorProviderCapability::SimplifyGeometries ) != 0 ) )
1600 {
1601 mSimplifyDrawingAtProvider->setEnabled( false );
1602 }
1603 else
1604 {
1605 mSimplifyDrawingAtProvider->setEnabled( checked );
1606 }
1607}
1608
1609void QgsVectorLayerProperties::updateVariableEditor()
1610{
1611 QgsExpressionContext context;
1612 mVariableEditor->setContext( &context );
1613 mVariableEditor->context()->appendScope( QgsExpressionContextUtils::globalScope() );
1614 mVariableEditor->context()->appendScope( QgsExpressionContextUtils::projectScope( QgsProject::instance() ) );
1615 mVariableEditor->context()->appendScope( QgsExpressionContextUtils::layerScope( mLayer ) );
1616 mVariableEditor->reloadContext();
1617 mVariableEditor->setEditableScopeIndex( 2 );
1618}
1619
1620void QgsVectorLayerProperties::showHelp()
1621{
1622 const QVariant helpPage = mOptionsStackedWidget->currentWidget()->property( "helpPage" );
1623
1624 if ( helpPage.isValid() )
1625 {
1626 QgsHelp::openHelp( helpPage.toString() );
1627 }
1628 else
1629 {
1630 QgsHelp::openHelp( QStringLiteral( "working_with_vector/vector_properties.html" ) );
1631 }
1632}
1633
1634void QgsVectorLayerProperties::updateAuxiliaryStoragePage()
1635{
1636 const QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1637
1638 if ( alayer )
1639 {
1640 // set widgets to enable state
1641 mAuxiliaryStorageInformationGrpBox->setEnabled( true );
1642 mAuxiliaryStorageFieldsGrpBox->setEnabled( true );
1643
1644 // update key
1645 mAuxiliaryStorageKeyLineEdit->setText( alayer->joinInfo().targetFieldName() );
1646
1647 // update feature count
1648 const qlonglong features = alayer->featureCount();
1649 mAuxiliaryStorageFeaturesLineEdit->setText( QLocale().toString( features ) );
1650
1651 // update actions
1652 mAuxiliaryLayerActionClear->setEnabled( true );
1653 mAuxiliaryLayerActionDelete->setEnabled( true );
1654 mAuxiliaryLayerActionExport->setEnabled( true );
1655 mAuxiliaryLayerActionNew->setEnabled( false );
1656
1657 const QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1658 if ( alayer )
1659 {
1660 const int fields = alayer->auxiliaryFields().count();
1661 mAuxiliaryStorageFieldsLineEdit->setText( QLocale().toString( fields ) );
1662
1663 // add fields
1664 mAuxiliaryStorageFieldsTree->clear();
1665 for ( const QgsField &field : alayer->auxiliaryFields() )
1666 {
1668 QTreeWidgetItem *item = new QTreeWidgetItem();
1669
1670 item->setText( 0, prop.origin() );
1671 item->setText( 1, prop.name() );
1672 item->setText( 2, prop.comment() );
1673 item->setText( 3, field.typeName() );
1674 item->setText( 4, field.name() );
1675
1676 mAuxiliaryStorageFieldsTree->addTopLevelItem( item );
1677 }
1678 }
1679 }
1680 else
1681 {
1682 mAuxiliaryStorageInformationGrpBox->setEnabled( false );
1683 mAuxiliaryStorageFieldsGrpBox->setEnabled( false );
1684
1685 mAuxiliaryLayerActionClear->setEnabled( false );
1686 mAuxiliaryLayerActionDelete->setEnabled( false );
1687 mAuxiliaryLayerActionExport->setEnabled( false );
1688 mAuxiliaryLayerActionNew->setEnabled( true );
1689
1690 mAuxiliaryStorageFieldsTree->clear();
1691 mAuxiliaryStorageKeyLineEdit->setText( QString() );
1692 mAuxiliaryStorageFieldsLineEdit->setText( QString() );
1693 mAuxiliaryStorageFeaturesLineEdit->setText( QString() );
1694 }
1695}
1696
1697void QgsVectorLayerProperties::onAuxiliaryLayerNew()
1698{
1699 QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1700
1701 if ( alayer )
1702 return;
1703
1704 QgsNewAuxiliaryLayerDialog dlg( mLayer, this );
1705 if ( dlg.exec() == QDialog::Accepted )
1706 {
1707 updateAuxiliaryStoragePage();
1708 }
1709}
1710
1711void QgsVectorLayerProperties::onAuxiliaryLayerClear()
1712{
1713 QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1714
1715 if ( !alayer )
1716 return;
1717
1718 const QString msg = tr( "Are you sure you want to clear auxiliary data for %1?" ).arg( mLayer->name() );
1719 QMessageBox::StandardButton reply;
1720 reply = QMessageBox::question( this, "Clear Auxiliary Data", msg, QMessageBox::Yes | QMessageBox::No );
1721
1722 if ( reply == QMessageBox::Yes )
1723 {
1724 QApplication::setOverrideCursor( Qt::WaitCursor );
1725 alayer->clear();
1726 QApplication::restoreOverrideCursor();
1727 updateAuxiliaryStoragePage();
1728 mLayer->triggerRepaint();
1729 }
1730}
1731
1732void QgsVectorLayerProperties::onAuxiliaryLayerDelete()
1733{
1734 QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1735 if ( !alayer )
1736 return;
1737
1738 const QString msg = tr( "Are you sure you want to delete auxiliary storage for %1?" ).arg( mLayer->name() );
1739 QMessageBox::StandardButton reply;
1740 reply = QMessageBox::question( this, "Delete Auxiliary Storage", msg, QMessageBox::Yes | QMessageBox::No );
1741
1742 if ( reply == QMessageBox::Yes )
1743 {
1744 QApplication::setOverrideCursor( Qt::WaitCursor );
1745 QgsDataSourceUri uri( alayer->source() );
1746
1747 // delete each attribute to correctly update layer settings and data
1748 // defined buttons
1749 while ( alayer->auxiliaryFields().size() > 0 )
1750 {
1751 QgsField aField = alayer->auxiliaryFields()[0];
1752 deleteAuxiliaryField( alayer->fields().indexOf( aField.name() ) );
1753 }
1754
1755 mLayer->setAuxiliaryLayer(); // remove auxiliary layer
1757 QApplication::restoreOverrideCursor();
1758 updateAuxiliaryStoragePage();
1759 mLayer->triggerRepaint();
1760 }
1761}
1762
1763void QgsVectorLayerProperties::onAuxiliaryLayerDeleteField()
1764{
1765 QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1766 if ( !alayer )
1767 return;
1768
1769 QList<QTreeWidgetItem *> items = mAuxiliaryStorageFieldsTree->selectedItems();
1770 if ( items.count() < 1 )
1771 return;
1772
1773 // get auxiliary field name and index from item
1774 const QTreeWidgetItem *item = items[0];
1776 def.setOrigin( item->text( 0 ) );
1777 def.setName( item->text( 1 ) );
1778 def.setComment( item->text( 2 ) );
1779
1780 const QString fieldName = QgsAuxiliaryLayer::nameFromProperty( def );
1781
1782 const int index = mLayer->auxiliaryLayer()->fields().indexOf( fieldName );
1783 if ( index < 0 )
1784 return;
1785
1786 // should be only 1 field
1787 const QString msg = tr( "Are you sure you want to delete auxiliary field %1 for %2?" ).arg( item->text( 1 ), item->text( 0 ) );
1788
1789 QMessageBox::StandardButton reply;
1790 const QString title = QObject::tr( "Delete Auxiliary Field" );
1791 reply = QMessageBox::question( this, title, msg, QMessageBox::Yes | QMessageBox::No );
1792
1793 if ( reply == QMessageBox::Yes )
1794 {
1795 QApplication::setOverrideCursor( Qt::WaitCursor );
1796 deleteAuxiliaryField( index );
1797 mLayer->triggerRepaint();
1798 QApplication::restoreOverrideCursor();
1799 }
1800}
1801
1802void QgsVectorLayerProperties::onAuxiliaryLayerAddField()
1803{
1804 QgsAuxiliaryLayer *alayer = mLayer->auxiliaryLayer();
1805 if ( !alayer )
1806 return;
1807
1808 QgsNewAuxiliaryFieldDialog dlg( QgsPropertyDefinition(), mLayer, false );
1809 if ( dlg.exec() == QDialog::Accepted )
1810 {
1811 updateAuxiliaryStoragePage();
1812 }
1813}
1814
1815void QgsVectorLayerProperties::deleteAuxiliaryField( int index )
1816{
1817 if ( !mLayer->auxiliaryLayer() )
1818 return;
1819
1820 int key = mLayer->auxiliaryLayer()->propertyFromIndex( index );
1822
1823 if ( mLayer->auxiliaryLayer()->deleteAttribute( index ) )
1824 {
1825 mLayer->updateFields();
1826
1827 // immediately deactivate data defined button
1828 if ( key >= 0 && def.origin().compare( "labeling", Qt::CaseInsensitive ) == 0
1829 && labelingDialog
1830 && labelingDialog->labelingGui() )
1831 {
1832 labelingDialog->labelingGui()->deactivateField( static_cast<QgsPalLayerSettings::Property>( key ) );
1833 }
1834
1835 updateAuxiliaryStoragePage();
1836 mSourceFieldsPropertiesDialog->init();
1837 }
1838 else
1839 {
1840 const QString title = QObject::tr( "Delete Auxiliary Field" );
1841 const QString errors = mLayer->auxiliaryLayer()->commitErrors().join( QLatin1String( "\n " ) );
1842 const QString msg = QObject::tr( "Unable to remove auxiliary field (%1)" ).arg( errors );
1843 mMessageBar->pushMessage( title, msg, Qgis::MessageLevel::Warning );
1844 }
1845}
1846
1847bool QgsVectorLayerProperties::eventFilter( QObject *obj, QEvent *ev )
1848{
1849 // If the map tip preview container is resized, resize the map tip
1850 if ( obj == mMapTipPreviewContainer && ev->type() == QEvent::Resize )
1851 {
1852 resizeMapTip();
1853 }
1854 return QgsOptionsDialogBase::eventFilter( obj, ev );
1855}
1856
1857void QgsVectorLayerProperties::initMapTipPreview()
1858{
1859 // HTML editor and preview are in a splitter. By default, the editor takes 2/3 of the space
1860 mMapTipSplitter->setSizes( { 400, 200 } );
1861 // Event filter is used to resize the map tip when the container is resized
1862 mMapTipPreviewContainer->installEventFilter( this );
1863
1864 // Note: there's quite a bit of overlap between this and the code in QgsMapTip::showMapTip
1865 // Create the WebView
1866 mMapTipPreview = new QgsWebView( mMapTipPreviewContainer );
1867
1868#if WITH_QTWEBKIT
1869 mMapTipPreview->page()->setLinkDelegationPolicy( QWebPage::DelegateAllLinks ); //Handle link clicks by yourself
1870 mMapTipPreview->setContextMenuPolicy( Qt::NoContextMenu ); //No context menu is allowed if you don't need it
1871 connect( mMapTipPreview, &QWebView::loadFinished, this, &QgsVectorLayerProperties::resizeMapTip );
1872#endif
1873
1874 mMapTipPreview->page()->settings()->setAttribute( QWebSettings::DeveloperExtrasEnabled, true );
1875 mMapTipPreview->page()->settings()->setAttribute( QWebSettings::JavascriptEnabled, true );
1876 mMapTipPreview->page()->settings()->setAttribute( QWebSettings::LocalStorageEnabled, true );
1877
1878 // Disable scrollbars, avoid random resizing issues
1879 mMapTipPreview->page()->mainFrame()->setScrollBarPolicy( Qt::Horizontal, Qt::ScrollBarAlwaysOff );
1880 mMapTipPreview->page()->mainFrame()->setScrollBarPolicy( Qt::Vertical, Qt::ScrollBarAlwaysOff );
1881
1882 // Update the map tip preview when the expression or the map tip template changes
1883 connect( mMapTipWidget, &QgsCodeEditorHTML::textChanged, this, &QgsVectorLayerProperties::updateMapTipPreview );
1884 connect( mDisplayExpressionWidget, qOverload<const QString &>( &QgsFieldExpressionWidget::fieldChanged ), this, &QgsVectorLayerProperties::updateMapTipPreview );
1885}
1886
1887void QgsVectorLayerProperties::updateMapTipPreview()
1888{
1889 mMapTipPreview->setMaximumSize( mMapTipPreviewContainer->width(), mMapTipPreviewContainer->height() );
1890 const QString htmlContent = QgsMapTip::vectorMapTipPreviewText( mLayer, mCanvas, mMapTipWidget->text(), mDisplayExpressionWidget->asExpression() );
1891 mMapTipPreview->setHtml( htmlContent );
1892}
1893
1894void QgsVectorLayerProperties::resizeMapTip()
1895{
1896 // Ensure the map tip is not bigger than the container
1897 mMapTipPreview->setMaximumSize( mMapTipPreviewContainer->width(), mMapTipPreviewContainer->height() );
1898#if WITH_QTWEBKIT
1899 // Get the content size
1900 const QWebElement container = mMapTipPreview->page()->mainFrame()->findFirstElement(
1901 QStringLiteral( "#QgsWebViewContainer" )
1902 );
1903 const int width = container.geometry().width();
1904 const int height = container.geometry().height();
1905 mMapTipPreview->resize( width, height );
1906
1907 // Move the map tip to the center of the container
1908 mMapTipPreview->move( ( mMapTipPreviewContainer->width() - mMapTipPreview->width() ) / 2, ( mMapTipPreviewContainer->height() - mMapTipPreview->height() ) / 2 );
1909
1910#else
1911 mMapTipPreview->adjustSize();
1912#endif
1913}
@ SimplifyGeometries
Supports simplification of geometries on provider side according to a distance tolerance.
@ SelectEncoding
Allows user to select encoding.
@ CreateSpatialIndex
Allows creation of spatial index.
QFlags< VectorRenderingSimplificationFlag > VectorRenderingSimplificationFlags
Simplification flags for vector feature rendering.
Definition qgis.h:2938
@ Present
A valid spatial index exists for the source.
@ AntialiasingSimplification
The geometries can be rendered with 'AntiAliasing' disabled because of it is '1-pixel size'.
@ NoSimplification
No simplification can be applied.
@ GeometrySimplification
The geometries can be simplified using the current map2pixel context state.
VectorSimplificationAlgorithm
Simplification algorithms for vector features.
Definition qgis.h:2907
@ Distance
The simplification uses the distance between points to remove duplicate points.
@ SnapToGrid
The simplification uses a grid (similar to ST_SnapToGrid) to remove duplicate points.
@ Visvalingam
The simplification gives each point in a line an importance weighting, so that least important points...
@ Warning
Warning message.
Definition qgis.h:156
static QString defaultProjectScales()
A string with default project scales.
Definition qgis.cpp:253
@ Polygon
Polygons.
@ Unknown
Unknown types.
@ Null
No geometry.
QFlags< DataProviderReadFlag > DataProviderReadFlags
Flags which control data provider construction.
Definition qgis.h:450
QFlags< VectorProviderCapability > VectorProviderCapabilities
Vector data provider capabilities.
Definition qgis.h:500
@ Marker
Marker symbol.
@ Line
Line symbol.
@ Fill
Fill symbol.
@ CustomColor
Use default symbol with a custom selection color.
@ CustomSymbol
Use a custom symbol.
@ Default
Use default symbol and selection colors.
void clearActions()
Removes all actions.
QUuid addAction(Qgis::AttributeActionType type, const QString &name, const QString &command, bool capture=false)
Add an action with the given name and action details.
Utility class that encapsulates an action based on vector attributes.
Definition qgsaction.h:37
static QString reportStyleSheet(QgsApplication::StyleSheetType styleSheetType=QgsApplication::StyleSheetType::Qt)
Returns a css style sheet for reports, the styleSheetType argument determines what type of stylesheet...
void init(const QgsActionManager &action, const QgsAttributeTableConfig &attributeTableConfig)
QList< QgsAction > actions() const
QgsAttributeTableConfig::ActionWidgetStyle attributeTableWidgetStyle() const
This is a container for configuration of the attribute table.
@ Action
This column represents an action widget.
QVector< QgsAttributeTableConfig::ColumnConfig > columns() const
Gets the list with all columns and their configuration.
void update(const QgsFields &fields)
Update the configuration with the given fields.
void setActionWidgetStyle(ActionWidgetStyle actionWidgetStyle)
Set the style of the action widget.
void setColumns(const QVector< QgsAttributeTableConfig::ColumnConfig > &columns)
Set the list of columns visible in the attribute table.
void store()
Stores currently opened widget configuration.
void apply()
Applies the attribute from properties to the vector layer.
Class allowing to manage the auxiliary storage for a vector layer.
static QString nameFromProperty(const QgsPropertyDefinition &def, bool joined=false)
Returns the name of the auxiliary field for a property definition.
static QgsPropertyDefinition propertyDefinitionFromField(const QgsField &field)
Returns the property definition from an auxiliary field.
bool clear()
Deletes all features from the layer.
bool deleteAttribute(int attr) override
Removes attribute from the layer and commits changes.
QgsFields auxiliaryFields() const
Returns a list of all auxiliary fields currently managed by the layer.
int propertyFromIndex(int index) const
Returns the underlying property key for the field index.
QgsVectorLayerJoinInfo joinInfo() const
Returns information to use for joining with primary key and so on.
QgsPropertyDefinition propertyDefinitionFromIndex(int index) const
Returns the property definition for the underlying field index.
static bool deleteTable(const QgsDataSourceUri &uri)
Removes a table from the auxiliary storage.
This class represents a coordinate reference system (CRS).
virtual bool supportsSubsetString() const
Returns true if the provider supports setting of subset strings.
Class for storing the component parts of a RDBMS data source URI (e.g.
static bool run(const QgsCoordinateReferenceSystem &sourceCrs=QgsCoordinateReferenceSystem(), const QgsCoordinateReferenceSystem &destinationCrs=QgsCoordinateReferenceSystem(), QWidget *parent=nullptr, QgsMapCanvas *mapCanvas=nullptr, const QString &windowTitle=QString())
Runs the dialog (if required) prompting for the desired transform to use from sourceCrs to destinatio...
void syncToOwnLayer()
Updates the widget to reflect the layer's current diagram settings.
void apply() override
Saves the labeling configuration and immediately updates the map canvas to reflect the changes.
void auxiliaryFieldCreated()
Emitted when an auxiliary field is created.
A generic dialog for building expression strings.
static void setLayerVariables(QgsMapLayer *layer, const QVariantMap &variables)
Sets all layer context variables.
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...
static QString findAndSelectActiveExpression(QgsCodeEditor *editor, const QString &pattern=QString())
Find the expression under the cursor in the given editor and select it.
bool forceRasterRender() const
Returns whether the renderer must render as a raster.
void setForceRasterRender(bool forceRaster)
Sets whether the renderer should be rendered to a raster destination.
double referenceScale() const
Returns the symbology reference scale.
void setReferenceScale(double scale)
Sets the symbology reference scale.
virtual Qgis::SpatialIndexPresence hasSpatialIndex() const
Returns an enum value representing the presence of a valid spatial index on the source,...
void fieldChanged(const QString &fieldName)
Emitted when the currently selected field changes.
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:53
QString name
Definition qgsfield.h:62
int count
Definition qgsfields.h:50
Q_INVOKABLE int indexFromName(const QString &fieldName) const
Gets the field index from the field name.
Q_INVOKABLE int indexOf(const QString &fieldName) const
Gets the field index from the field name.
int size() const
Returns number of items.
static QString stringToSafeFilename(const QString &string)
Converts a string to a safe filename, replacing characters which are not safe for filenames with an '...
static QgsProviderSourceWidgetProviderRegistry * sourceWidgetProviderRegistry()
Returns the registry of provider source widget providers.
Definition qgsgui.cpp:120
static QgsSubsetStringEditorProviderRegistry * subsetStringEditorProviderRegistry()
Returns the registry of subset string editors of data providers.
Definition qgsgui.cpp:115
static void openHelp(const QString &key)
Opens help topic for the given help key using default system web browser.
Definition qgshelp.cpp:39
Master widget for configuration of labeling of a vector layer.
void adaptToLayer()
reload the settings shown in the dialog from the current layer
QgsLabelingGui * labelingGui()
Returns the labeling gui widget or nullptr if none.
void auxiliaryFieldCreated()
Emitted when an auxiliary field is created.
void writeSettingsToLayer()
save config to layer
Base class for "layer properties" dialogs, containing common utilities for handling functionality in ...
QPushButton * mBtnStyle
Style button.
void saveMetadataToFile()
Allows the user to save the layer's metadata as a file.
virtual void rollback()
Rolls back changes made to the layer.
void optionsStackedWidget_CurrentChanged(int index) override
QList< QgsMapLayerConfigWidget * > mConfigWidgets
Layer config widgets.
void setMetadataWidget(QgsMetadataWidget *widget, QWidget *page)
Sets the metadata widget and page associated with the dialog.
void loadDefaultStyle()
Reloads the default style for the layer.
void saveStyleAs()
Saves a style when appriate button is pressed.
void loadStyle()
Triggers a dialog to load a saved style.
QgsMapCanvas * mCanvas
Associated map canvas.
void loadDefaultMetadata()
Reloads the default layer metadata for the layer.
void loadMetadataFromFile()
Allows the user to load layer metadata from a file.
void saveDefaultStyle()
Saves the default style when appropriate button is pressed.
QPushButton * mBtnMetadata
Metadata button.
void initialize()
Initialize the dialog.
void saveMetadataAsDefault()
Saves the current layer metadata as the default for the layer.
void openUrl(const QUrl &url)
Handles opening a url from the dialog.
QgsLayerTreeFilterProxyModel is a sort filter proxy model to easily reproduce the legend/layer tree i...
QList< QgsMapLayer * > checkedLayers() const
Returns the checked layers.
void setLayerTreeModel(QgsLayerTreeModel *layerTreeModel)
Sets the layer tree model.
void setCheckedLayers(const QList< QgsMapLayer * > layers)
Initialize the list of checked layers.
The QgsLayerTreeModel class is model implementation for Qt item views framework.
Map canvas is a class for displaying all GIS data types on a canvas.
QColor selectionColor() const
Returns color for selected features.
QgsExpressionContext createExpressionContext() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
double scale() const
Returns the last reported scale of the canvas.
A panel widget that can be shown in the map style dock.
This class models dependencies with or between map layers.
The QgsMapLayerSaveStyleDialog class provides the UI to save the current style or multiple styles int...
Manages QGIS Server properties for a map layer.
void addStyleManagerActions(QMenu *m, QgsMapLayer *layer)
adds actions to the menu in accordance to the layer
static QgsMapLayerStyleGuiUtils * instance()
returns a singleton instance of this class
QString currentStyle() const
Returns name of the current style.
QStringList styles() const
Returns list of all defined style names.
bool setCurrentStyle(const QString &name)
Set a different style as the current style - will apply it to the layer.
void currentStyleChanged(const QString &currentName)
Emitted when the current style has been changed.
Base class for all map layer types.
Definition qgsmaplayer.h:76
QString name
Definition qgsmaplayer.h:80
void setRefreshOnNotifyEnabled(bool enabled)
Set whether provider notification is connected to triggerRepaint.
virtual QString saveSldStyle(const QString &uri, bool &resultFlag) const
Saves the properties of this layer to an SLD format file.
QString source() const
Returns the source for the layer.
QString providerType() const
Returns the provider type (provider key) for this layer.
void setMinimumScale(double scale)
Sets the minimum map scale (i.e.
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:83
QgsMapLayerServerProperties * serverProperties()
Returns QGIS Server Properties for the map layer.
virtual int listStylesInDatabase(QStringList &ids, QStringList &names, QStringList &descriptions, QString &msgError)
Lists all the style in db split into related to the layer and not related to.
void setDataSource(const QString &dataSource, const QString &baseName=QString(), const QString &provider=QString(), bool loadDefaultStyleFlag=false)
Updates the data source of the layer.
QString id
Definition qgsmaplayer.h:79
void triggerRepaint(bool deferredUpdate=false)
Will advise the map canvas (and any other interested party) that this layer requires to be repainted.
void setMaximumScale(double scale)
Sets the maximum map scale (i.e.
QgsLayerMetadata metadata
Definition qgsmaplayer.h:82
bool isRefreshOnNotifyEnabled() const
Returns true if the refresh on provider nofification is enabled.
virtual void saveStyleToDatabase(const QString &name, const QString &description, bool useAsDefault, const QString &uiFileContent, QString &msgError, QgsMapLayer::StyleCategories categories=QgsMapLayer::AllStyleCategories)
Saves named and sld style of the layer to the style table in the db.
void setScaleBasedVisibility(bool enabled)
Sets whether scale based visibility is enabled for the layer.
bool hasScaleBasedVisibility() const
Returns whether scale based visibility is enabled for the layer.
void setRefreshOnNofifyMessage(const QString &message)
Set the notification message that triggers repaint If refresh on notification is enabled,...
void setName(const QString &name)
Set the display name of the layer.
QString refreshOnNotifyMessage() const
Returns the message that should be notified by the provider to triggerRepaint.
void setMapTipsEnabled(bool enabled)
Enable or disable map tips for this layer.
double minimumScale() const
Returns the minimum map scale (i.e.
QgsMapLayerStyleManager * styleManager() const
Gets access to the layer's style manager.
virtual QString saveNamedStyle(const QString &uri, bool &resultFlag, StyleCategories categories=AllStyleCategories)
Save the properties of this layer as a named style (either as a .qml file on disk or as a record in t...
void setMapTipTemplate(const QString &mapTipTemplate)
The mapTip is a pretty, html representation for feature information.
bool mapTipsEnabled
Definition qgsmaplayer.h:90
double maximumScale() const
Returns the maximum map scale (i.e.
QString mapTipTemplate
Definition qgsmaplayer.h:89
void setCrs(const QgsCoordinateReferenceSystem &srs, bool emitSignal=true)
Sets layer's spatial reference system.
The QgsMapSettings class contains configuration for rendering of the map.
static QString vectorMapTipPreviewText(QgsMapLayer *layer, QgsMapCanvas *mapCanvas, const QString &mapTemplate, const QString &displayExpression)
Returns the html that would be displayed in a maptip for a given layer.
A bar for displaying non-blocking messages to the user.
void pushMessage(const QString &text, Qgis::MessageLevel level=Qgis::MessageLevel::Info, int duration=-1)
A convenience method for pushing a message with the specified text to the bar.
A wizard to edit metadata on a map layer.
void acceptMetadata()
Saves the metadata to the layer.
void crsChanged()
If the CRS is updated.
void setMapCanvas(QgsMapCanvas *canvas)
Sets a map canvas associated with the widget.
void setMetadata(const QgsAbstractMetadataBase *metadata)
Sets the metadata to display in the widget.
A dialog to create a new auxiliary field.
A dialog to create a new auxiliary layer.
void resizeAlltabs(int index)
Resizes all tabs when the dialog is resized.
QStackedWidget * mOptStackedWidget
void initOptionsBase(bool restoreUi=true, const QString &title=QString())
Set up the base ui connections for vertical tabs.
Property
Data definable properties.
Base class for any widget that can be shown as a inline panel.
QString panelTitle()
The title of the panel.
void acceptPanel()
Accept the panel.
static QgsProject * instance()
Returns the QgsProject singleton instance.
Q_INVOKABLE QgsMapLayer * mapLayer(const QString &layerId) const
Retrieve a pointer to a registered layer by layer ID.
void setDirty(bool b=true)
Flag the project as dirty (modified).
void crsChanged(const QgsCoordinateReferenceSystem &crs)
Emitted when the selected CRS is changed.
Definition for a property.
Definition qgsproperty.h:45
QString comment() const
Returns the comment of the property.
void setOrigin(const QString &origin)
Sets the origin of the property.
QString name() const
Returns the name of the property.
void setName(const QString &name)
Sets the name of the property.
QString origin() const
Returns the origin of the property.
void setComment(const QString &comment)
Sets comment of the property.
static QgsProviderRegistry * instance(const QString &pluginPath=QString())
Means of accessing canonical single instance.
QgsProviderSourceWidget * createWidget(QgsMapLayer *layer, QWidget *parent=nullptr)
Creates a new widget to configure the source of the specified layer.
virtual QString groupTitle() const
Returns an optional group title for the source settings, for use in layer properties dialogs.
void validChanged(bool isValid)
Emitted whenever the validation status of the widget changes.
virtual QString sourceUri() const =0
Returns the source URI as currently defined by the widget.
virtual void setMapCanvas(QgsMapCanvas *mapCanvas)
Sets a map canvas associated with the widget.
virtual void setSourceUri(const QString &uri)=0
Sets the source uri to show in the widget.
void showPanel(QgsPanelWidget *panel)
Emit when you require a panel to be show in the interface.
void setContext(const QgsSymbolWidgetContext &context)
Sets the context in which the dialog is shown, e.g., the associated map canvas and expression context...
void apply()
Apply the changes from the dialog to the layer.
void layerVariablesChanged()
Emitted when expression context variables on the associated vector layers have been changed.
void widgetChanged()
Emitted when something on the widget has changed.
void setDockMode(bool dockMode)
Set the widget in dock mode which tells the widget to emit panel widgets and not open dialogs.
static QMap< int, QString > wmsDimensionDefaultDisplayLabels()
Returns WMS Dimension default display labels.
bool addWmsDimension(const QgsServerWmsDimensionProperties::WmsDimensionInfo &wmsDimInfo)
Adds a QGIS Server WMS Dimension.
bool removeWmsDimension(const QString &wmsDimName)
Removes a QGIS Server WMS Dimension.
const QList< QgsServerWmsDimensionProperties::WmsDimensionInfo > wmsDimensions() const
Returns the QGIS Server WMS Dimension list.
This class is a composition of two QSettings instances:
Definition qgssettings.h:64
QVariant value(const QString &key, const QVariant &defaultValue=QVariant(), Section section=NoSection) const
Returns the value for setting key.
bool contains(const QString &key, QgsSettings::Section section=QgsSettings::NoSection) const
Returns true if there exists a setting called key; returns false otherwise.
void setValue(const QString &key, const QVariant &value, QgsSettings::Section section=QgsSettings::NoSection)
Sets the value of setting key to value.
static QgsStyle * defaultStyle(bool initialize=true)
Returns the default application-wide style.
Definition qgsstyle.cpp:146
Interface for a dialog that can edit subset strings.
virtual QString subsetString() const =0
Returns the subset string entered in the dialog.
virtual void setSubsetString(const QString &subsetString)=0
Sets a subset string into the dialog.
QgsSubsetStringEditorInterface * createDialog(QgsVectorLayer *layer, QWidget *parent=nullptr, Qt::WindowFlags fl=QgsGuiUtils::ModalDialogFlags)
Creates a new dialog to edit the subset string of the provided layer.
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 setMessageBar(QgsMessageBar *bar)
Sets the message bar associated with the widget.
Abstract base class for all rendered symbols.
Definition qgssymbol.h:231
This is the base class for vector data providers.
static QStringList availableEncodings()
Returns a list of available encodings.
virtual bool createSpatialIndex()
Creates a spatial index on the datasource (if supported by the provider type).
virtual Q_INVOKABLE Qgis::VectorProviderCapabilities capabilities() const
Returns flags containing the supported capabilities.
virtual bool createAttributeIndex(int field)
Create an attribute index on the datasource.
QString encoding() const
Returns the encoding which is used for accessing data.
Defines left outer join from our vector layer to some other vector layer.
bool hasCascadedDelete() const
Returns whether a feature deleted on the target layer has to impact the joined layer by deleting the ...
bool isDynamicFormEnabled() const
Returns whether the form has to be dynamically updated with joined fields when a feature is being cre...
bool hasUpsertOnEdit() const
Returns whether a feature created on the target layer has to impact the joined layer by creating a ne...
bool isEditable() const
Returns whether joined fields may be edited through the form of the target layer.
bool isUsingMemoryCache() const
Returns whether values from the joined layer should be cached in memory to speed up lookups.
QString prefix() const
Returns prefix of fields from the joined layer. If nullptr, joined layer's name will be used.
static QStringList joinFieldNamesSubset(const QgsVectorLayerJoinInfo &info, bool blocklisted=true)
Returns the list of field names to use for joining considering blocklisted fields and subset.
QString joinFieldName() const
Returns name of the field of joined layer that will be used for join.
QString targetFieldName() const
Returns name of the field of our layer that will be used for join.
QString joinLayerId() const
ID of the joined layer - may be used to resolve reference to the joined layer.
QgsVectorLayer * joinLayer() const
Returns joined layer (may be nullptr if the reference was set by layer ID and not resolved yet)
bool eventFilter(QObject *obj, QEvent *ev) override
QgsVectorLayerProperties(QgsMapCanvas *canvas, QgsMessageBar *messageBar, QgsVectorLayer *lyr=nullptr, QWidget *parent=nullptr, Qt::WindowFlags fl=QgsGuiUtils::ModalDialogFlags)
void optionsStackedWidget_CurrentChanged(int index) final
void toggleEditing(QgsMapLayer *)
void exportAuxiliaryLayer(QgsAuxiliaryLayer *layer)
Implementation of layer selection properties for vector layers.
void setSelectionSymbol(QgsSymbol *symbol)
Sets the symbol used to render selected features in the layer.
QgsSymbol * selectionSymbol() const
Returns the symbol used to render selected features in the layer.
Qgis::SelectionRenderingMode selectionRenderingMode() const
Returns the selection rendering mode to use for selected features in the layer.
QColor selectionColor() const
Returns the color to use for rendering selected features in the layer.
void setSelectionColor(const QColor &color)
Sets the color to use for rendering selected features in the layer.
void setSelectionRenderingMode(Qgis::SelectionRenderingMode mode)
Sets the selection rendering mode to use for selected features in the layer.
A widget for configuring the temporal properties for a vector layer.
void syncToLayer()
Updates the widget state to match the current layer state.
void saveTemporalProperties()
Save widget temporal properties inputs.
Represents a vector layer which manages a vector based data sets.
void setAuxiliaryLayer(QgsAuxiliaryLayer *layer=nullptr)
Sets the current auxiliary layer.
bool addJoin(const QgsVectorLayerJoinInfo &joinInfo)
Joins another vector layer to this layer.
QSet< QgsMapLayerDependency > dependencies() const FINAL
Gets the list of dependencies.
long long featureCount(const QString &legendKey) const
Number of features rendered with specified legend key.
void updateFields()
Will regenerate the fields property of this layer by obtaining all fields from the dataProvider,...
bool isSpatial() const FINAL
Returns true if this is a geometry layer and false in case of NoGeometry (table only) or UnknownGeome...
QString htmlMetadata() const FINAL
Obtain a formatted HTML string containing assorted metadata for this layer.
void setSimplifyMethod(const QgsVectorSimplifyMethod &simplifyMethod)
Sets the simplification settings for fast rendering of features.
QgsAuxiliaryLayer * auxiliaryLayer()
Returns the current auxiliary layer.
bool setDependencies(const QSet< QgsMapLayerDependency > &layers) FINAL
Sets the list of dependencies.
void setProviderEncoding(const QString &encoding)
Sets the text encoding of the data provider.
void setDisplayExpression(const QString &displayExpression)
Set the preview expression, used to create a human readable preview string.
bool removeJoin(const QString &joinLayerId)
Removes a vector layer join.
bool isEditable() const FINAL
Returns true if the provider is in editing mode.
QStringList commitErrors() const
Returns a list containing any error messages generated when attempting to commit changes to the layer...
virtual void updateExtents(bool force=false)
Update the extents for the layer.
QgsFeatureRenderer * renderer()
Returns the feature renderer used for rendering the features in the layer in 2D map views.
QString displayExpression
QgsVectorDataProvider * dataProvider() FINAL
Returns the layer's data provider, it may be nullptr.
QgsAttributeTableConfig attributeTableConfig() const
Returns the attribute table configuration object.
QgsActionManager * actions()
Returns all layer actions defined on this layer.
Q_INVOKABLE Qgis::GeometryType geometryType() const
Returns point, line or polygon.
void setAttributeTableConfig(const QgsAttributeTableConfig &attributeTableConfig)
Sets the attribute table configuration object.
virtual bool setSubsetString(const QString &subset)
Sets the string (typically sql) used to define a subset of the layer.
const QgsVectorSimplifyMethod & simplifyMethod() const
Returns the simplification settings for fast rendering of features.
const QList< QgsVectorLayerJoinInfo > vectorJoins() const
QgsMapLayerSelectionProperties * selectionProperties() override
Returns the layer's selection properties.
This class contains information how to simplify geometries fetched from a vector layer.
bool forceLocalOptimization() const
Gets where the simplification executes, after fetch the geometries from provider, or when supported,...
Qgis::VectorRenderingSimplificationFlags simplifyHints() const
Gets the simplification hints of the vector layer managed.
float maximumScale() const
Gets the maximum scale at which the layer should be simplified.
Qgis::VectorSimplificationAlgorithm simplifyAlgorithm() const
Gets the local simplification algorithm of the vector layer managed.
void setThreshold(float threshold)
Sets the simplification threshold of the vector layer managed.
void setForceLocalOptimization(bool localOptimization)
Sets where the simplification executes, after fetch the geometries from provider, or when supported,...
void setSimplifyHints(Qgis::VectorRenderingSimplificationFlags simplifyHints)
Sets the simplification hints of the vector layer managed.
float threshold() const
Gets the simplification threshold of the vector layer managed.
void setMaximumScale(float maximumScale)
Sets the maximum scale at which the layer should be simplified.
void setSimplifyAlgorithm(Qgis::VectorSimplificationAlgorithm simplifyAlgorithm)
Sets the local simplification algorithm of the vector layer managed.
The QgsWebView class is a collection of stubs to mimic the API of QWebView on systems where the real ...
Definition qgswebview.h:66
The QgsWmsDimensionDialog class provides an interface for WMS/OAPIF (WFS3) dimensions configuration A...
const QgsCoordinateReferenceSystem & crs
Setting options for creating vector data providers.
Setting to define QGIS Server WMS Dimension.