QGIS API Documentation 3.43.0-Master (ac54a16a525)
qgsrelationreferencewidget.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsrelationreferencewidget.cpp
3 --------------------------------------
4 Date : 20.4.2013
5 Copyright : (C) 2013 Matthias Kuhn
6 Email : matthias at opengis dot ch
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15
17#include "moc_qgsrelationreferencewidget.cpp"
18
19#include <QPushButton>
20#include <QDialog>
21#include <QHBoxLayout>
22#include <QTimer>
23#include <QCompleter>
24
25#include "qgsattributeform.h"
26#include "qgsattributedialog.h"
27#include "qgsapplication.h"
29#include "qgsexpression.h"
30#include "qgsfields.h"
31#include "qgsgeometry.h"
32#include "qgshighlight.h"
33#include "qgsmapcanvas.h"
34#include "qgsmessagebar.h"
35#include "qgsvectorlayer.h"
38#include "qgsfeatureiterator.h"
40#include "qgsvectorlayerutils.h"
41
42
43bool qVariantListIsNull( const QVariantList &list )
44{
45 if ( list.isEmpty() )
46 return true;
47
48 for ( int i = 0; i < list.size(); ++i )
49 {
50 if ( !QgsVariantUtils::isNull( list.at( i ) ) )
51 return false;
52 }
53 return true;
54}
55
56
58 : QWidget( parent )
59{
60 mTopLayout = new QVBoxLayout( this );
61 mTopLayout->setContentsMargins( 0, 0, 0, 0 );
62
63 setSizePolicy( sizePolicy().horizontalPolicy(), QSizePolicy::Fixed );
64
65 setLayout( mTopLayout );
66
67 QHBoxLayout *editLayout = new QHBoxLayout();
68 editLayout->setContentsMargins( 0, 0, 0, 0 );
69 editLayout->setSpacing( 2 );
70
71 // Prepare the container and layout for the filter comboboxes
72 mChooserContainer = new QWidget;
73 editLayout->addWidget( mChooserContainer );
74 QHBoxLayout *chooserLayout = new QHBoxLayout;
75 chooserLayout->setContentsMargins( 0, 0, 0, 0 );
76 mFilterLayout = new QHBoxLayout;
77 mFilterLayout->setContentsMargins( 0, 0, 0, 0 );
78 mFilterContainer = new QWidget;
79 mFilterContainer->setLayout( mFilterLayout );
80 mChooserContainer->setLayout( chooserLayout );
81 chooserLayout->addWidget( mFilterContainer );
82
83 mComboBox = new QgsFeatureListComboBox();
84 mChooserContainer->layout()->addWidget( mComboBox );
85
86 // open form button
87 mOpenFormButton = new QToolButton();
88 mOpenFormButton->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mActionPropertyItem.svg" ) ) );
89 mOpenFormButton->setText( tr( "Open Related Feature Form" ) );
90 editLayout->addWidget( mOpenFormButton );
91
92 mAddEntryButton = new QToolButton();
93 mAddEntryButton->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mActionAdd.svg" ) ) );
94 mAddEntryButton->setText( tr( "Add New Entry" ) );
95 editLayout->addWidget( mAddEntryButton );
96
97 // highlight button
98 mHighlightFeatureButton = new QToolButton( this );
99 mHighlightFeatureButton->setPopupMode( QToolButton::MenuButtonPopup );
100 mHighlightFeatureAction = new QAction( QgsApplication::getThemeIcon( QStringLiteral( "/mActionHighlightFeature.svg" ) ), tr( "Highlight feature" ), this );
101 mScaleHighlightFeatureAction = new QAction( QgsApplication::getThemeIcon( QStringLiteral( "/mActionScaleHighlightFeature.svg" ) ), tr( "Scale and highlight feature" ), this );
102 mPanHighlightFeatureAction = new QAction( QgsApplication::getThemeIcon( QStringLiteral( "/mActionPanHighlightFeature.svg" ) ), tr( "Pan and highlight feature" ), this );
103 mHighlightFeatureButton->addAction( mHighlightFeatureAction );
104 mHighlightFeatureButton->addAction( mScaleHighlightFeatureAction );
105 mHighlightFeatureButton->addAction( mPanHighlightFeatureAction );
106 mHighlightFeatureButton->setDefaultAction( mHighlightFeatureAction );
107 editLayout->addWidget( mHighlightFeatureButton );
108
109 // map identification button
110 mMapIdentificationButton = new QToolButton( this );
111 mMapIdentificationButton->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mActionMapIdentification.svg" ) ) );
112 mMapIdentificationButton->setText( tr( "Select on Map" ) );
113 mMapIdentificationButton->setCheckable( true );
114 editLayout->addWidget( mMapIdentificationButton );
115
116 // remove foreign key button
117 mRemoveFKButton = new QToolButton( this );
118 mRemoveFKButton->setIcon( QgsApplication::getThemeIcon( QStringLiteral( "/mActionRemove.svg" ) ) );
119 mRemoveFKButton->setText( tr( "No Selection" ) );
120 editLayout->addWidget( mRemoveFKButton );
121
122 // add line to top layout
123 mTopLayout->addLayout( editLayout );
124
125 // embed form
126 mAttributeEditorFrame = new QgsCollapsibleGroupBox( this );
127 mAttributeEditorLayout = new QVBoxLayout( mAttributeEditorFrame );
128 mAttributeEditorFrame->setLayout( mAttributeEditorLayout );
129 mAttributeEditorFrame->setSizePolicy( mAttributeEditorFrame->sizePolicy().horizontalPolicy(), QSizePolicy::Expanding );
130 mTopLayout->addWidget( mAttributeEditorFrame );
131
132 // invalid label
133 mInvalidLabel = new QLabel( tr( "The relation is not valid. Please make sure your relation definitions are OK." ) );
134 mInvalidLabel->setWordWrap( true );
135 QFont font = mInvalidLabel->font();
136 font.setItalic( true );
137 mInvalidLabel->setStyleSheet( QStringLiteral( "QLabel { color: red; } " ) );
138 mInvalidLabel->setFont( font );
139 mTopLayout->addWidget( mInvalidLabel );
140
141 // default mode is combobox, no geometric relation and no embed form
142 mMapIdentificationButton->hide();
143 mHighlightFeatureButton->hide();
144 mAttributeEditorFrame->hide();
145 mInvalidLabel->hide();
146 mAddEntryButton->hide();
147
148 // connect buttons
149 connect( mOpenFormButton, &QAbstractButton::clicked, this, &QgsRelationReferenceWidget::openForm );
150 connect( mHighlightFeatureButton, &QToolButton::triggered, this, &QgsRelationReferenceWidget::highlightActionTriggered );
151 connect( mMapIdentificationButton, &QAbstractButton::clicked, this, &QgsRelationReferenceWidget::mapIdentification );
152 connect( mRemoveFKButton, &QAbstractButton::clicked, this, &QgsRelationReferenceWidget::deleteForeignKeys );
153 connect( mAddEntryButton, &QAbstractButton::clicked, this, &QgsRelationReferenceWidget::addEntry );
154 connect( mComboBox, &QComboBox::editTextChanged, this, &QgsRelationReferenceWidget::updateAddEntryButton );
155}
156
158{
159 deleteHighlight();
160 unsetMapTool();
161}
162
163void QgsRelationReferenceWidget::setRelation( const QgsRelation &relation, bool allowNullValue )
164{
165 mAllowNull = allowNullValue;
166 mRemoveFKButton->setVisible( allowNullValue && mReadOnlySelector );
167
168 if ( relation.isValid() )
169 {
170 mReferencedLayerId = relation.referencedLayerId();
171 mReferencedLayerName = relation.referencedLayer()->name();
173 mReferencedLayerProviderKey = relation.referencedLayer()->providerType();
174 mInvalidLabel->hide();
175
176 mRelation = relation;
177 mReferencingLayer = relation.referencingLayer();
178 mReferencedLayer = relation.referencedLayer();
179
180 const QList<QgsRelation::FieldPair> fieldPairs = relation.fieldPairs();
181 for ( const QgsRelation::FieldPair &fieldPair : fieldPairs )
182 {
183 mReferencedFields << fieldPair.referencedField();
184 }
185 if ( mComboBox )
186 {
187 mComboBox->setAllowNull( mAllowNull );
188 mComboBox->setSourceLayer( mReferencedLayer );
189 mComboBox->setIdentifierFields( mReferencedFields );
190 mComboBox->setFilterExpression( mFilterExpression );
191 mComboBox->setFetchLimit( mFetchLimit );
192 }
193 mAttributeEditorFrame->setObjectName( QStringLiteral( "referencing/" ) + relation.name() );
194
195 if ( mEmbedForm )
196 {
198 mAttributeEditorFrame->setTitle( mReferencedLayer->name() );
199 mReferencedAttributeForm = new QgsAttributeForm( relation.referencedLayer(), QgsFeature(), context, this );
200 mAttributeEditorLayout->addWidget( mReferencedAttributeForm );
201 }
202
203 connect( mReferencedLayer, &QgsVectorLayer::editingStarted, this, &QgsRelationReferenceWidget::updateAddEntryButton );
204 connect( mReferencedLayer, &QgsVectorLayer::editingStopped, this, &QgsRelationReferenceWidget::updateAddEntryButton );
205 updateAddEntryButton();
206 }
207 else
208 {
209 mInvalidLabel->show();
210 }
211
212 if ( mShown && isVisible() )
213 {
214 init();
215 }
216}
217
219{
220 if ( !editable )
221 {
222 unsetMapTool();
223 }
224
225 mFilterContainer->setEnabled( editable );
226 mComboBox->setEnabled( editable && !mReadOnlySelector );
227 mComboBox->setEditable( true );
228 mMapIdentificationButton->setEnabled( editable );
229 mRemoveFKButton->setEnabled( editable );
230 mIsEditable = editable;
231}
232
233void QgsRelationReferenceWidget::setForeignKey( const QVariant &value )
234{
235 setForeignKeys( QVariantList() << value );
236}
237
238void QgsRelationReferenceWidget::setForeignKeys( const QVariantList &values )
239{
240 if ( values.isEmpty() )
241 {
242 return;
243 }
244 if ( qVariantListIsNull( values ) )
245 {
247 return;
248 }
249
250 if ( !mReferencedLayer )
251 return;
252
253 mComboBox->setIdentifierValues( values );
254
255 if ( mEmbedForm || mChainFilters )
256 {
257 QgsFeatureRequest request = mComboBox->currentFeatureRequest();
258 mReferencedLayer->getFeatures( request ).nextFeature( mFeature );
259 }
260 if ( mChainFilters )
261 {
262 QVariant nullValue = QgsApplication::nullRepresentation();
263 const int count = std::min( mFilterComboBoxes.size(), mFilterFields.size() );
264 for ( int i = 0; i < count; i++ )
265 {
266 QVariant v = mFeature.attribute( mFilterFields[i] );
267 QString f = QgsVariantUtils::isNull( v ) ? nullValue.toString() : v.toString();
268 mFilterComboBoxes.at( i )->setCurrentIndex( mFilterComboBoxes.at( i )->findText( f ) );
269 }
270 }
271
272 mRemoveFKButton->setEnabled( mIsEditable );
273 highlightFeature( mFeature ); // TODO : make this async
274 updateAttributeEditorFrame( mFeature );
275
276 emitForeignKeysChanged( foreignKeys() );
277}
278
280{
281 // deactivate filter comboboxes
282 if ( mChainFilters && !mFilterComboBoxes.isEmpty() )
283 {
284 QComboBox *cb = mFilterComboBoxes.first();
285 cb->setCurrentIndex( 0 );
286 disableChainedComboBoxes( cb );
287 }
288
289 mComboBox->setIdentifierValuesToNull();
290 mRemoveFKButton->setEnabled( false );
291 updateAttributeEditorFrame( QgsFeature() );
292
293 emitForeignKeysChanged( foreignKeys() );
294}
295
297{
298 if ( !mEmbedForm )
299 // if there is no embedded form, everything is fine anyway
300 return true;
301
302 return mReferencedAttributeForm->save();
303}
304
306{
307 QgsFeature f;
308 if ( mReferencedLayer )
309 {
310 mReferencedLayer->getFeatures( mComboBox->currentFeatureRequest() ).nextFeature( f );
311 }
312 return f;
313}
314
316{
317 whileBlocking( mComboBox )->setIdentifierValuesToNull();
318 mRemoveFKButton->setEnabled( false );
319 updateAttributeEditorFrame( QgsFeature() );
320}
321
323{
324 QVariantList fkeys;
325 if ( fkeys.isEmpty() )
326 return QgsVariantUtils::createNullVariant( QMetaType::Type::Int );
327 else
328 return fkeys.at( 0 );
329}
330
332{
333 return mComboBox->identifierValues();
334}
335
337{
338 mEditorContext = context;
339 mCanvas = canvas;
340 mMessageBar = messageBar;
341
342 mMapToolIdentify.reset( new QgsMapToolIdentifyFeature( mCanvas ) );
343 mMapToolIdentify->setButton( mMapIdentificationButton );
344
345 if ( mEditorContext.cadDockWidget() )
346 {
347 mMapToolDigitize.reset( new QgsMapToolDigitizeFeature( mCanvas, mEditorContext.cadDockWidget() ) );
348 mMapToolDigitize->setButton( mAddEntryButton );
349 updateAddEntryButton();
350 }
351}
352
354{
355 if ( display )
356 {
357 setSizePolicy( sizePolicy().horizontalPolicy(), QSizePolicy::MinimumExpanding );
358 mTopLayout->setAlignment( Qt::AlignTop );
359 }
360
361 mAttributeEditorFrame->setVisible( display );
362 mEmbedForm = display;
363}
364
366{
367 mComboBox->setEnabled( !readOnly );
368 mRemoveFKButton->setVisible( mAllowNull && readOnly );
369 mReadOnlySelector = readOnly;
370}
371
373{
374 mHighlightFeatureButton->setVisible( allowMapIdentification );
375 mMapIdentificationButton->setVisible( allowMapIdentification );
376 mAllowMapIdentification = allowMapIdentification;
377}
378
379void QgsRelationReferenceWidget::setFilterFields( const QStringList &filterFields )
380{
381 mFilterFields = filterFields;
382}
383
385{
386 mOpenFormButton->setVisible( openFormButtonVisible );
387 mOpenFormButtonVisible = openFormButtonVisible;
388}
389
391{
392 mChainFilters = chainFilters;
393}
394
395void QgsRelationReferenceWidget::setFilterExpression( const QString &expression )
396{
397 mFilterExpression = expression;
398}
399
401{
402 Q_UNUSED( e )
403
404 mShown = true;
405 if ( !mInitialized )
406 init();
407}
408
410{
411 if ( mReferencedLayer )
412 {
413 QApplication::setOverrideCursor( Qt::WaitCursor );
414
415 QSet<QString> requestedAttrs;
416
417 if ( !mFilterFields.isEmpty() )
418 {
419 for ( const QString &fieldName : std::as_const( mFilterFields ) )
420 {
421 int idx = mReferencedLayer->fields().lookupField( fieldName );
422
423 if ( idx == -1 )
424 continue;
425
426 QComboBox *cb = new QComboBox();
427 cb->setProperty( "Field", fieldName );
428 cb->setProperty( "FieldAlias", mReferencedLayer->attributeDisplayName( idx ) );
429 mFilterComboBoxes << cb;
430 QVariantList uniqueValues = qgis::setToList( mReferencedLayer->uniqueValues( idx ) );
431 cb->addItem( mReferencedLayer->attributeDisplayName( idx ) );
432 QVariant nullValue = QgsApplication::nullRepresentation();
433 cb->addItem( nullValue.toString(), QgsVariantUtils::createNullVariant( mReferencedLayer->fields().at( idx ).type() ) );
434
435 std::sort( uniqueValues.begin(), uniqueValues.end(), qgsVariantLessThan );
436 const auto constUniqueValues = uniqueValues;
437 for ( const QVariant &v : constUniqueValues )
438 {
439 cb->addItem( v.toString(), v );
440 }
441
442 connect( cb, static_cast<void ( QComboBox::* )( int )>( &QComboBox::currentIndexChanged ), this, &QgsRelationReferenceWidget::filterChanged );
443
444 // Request this attribute for caching
445 requestedAttrs << fieldName;
446
447 mFilterLayout->addWidget( cb );
448 }
449
450 if ( mChainFilters )
451 {
452 QVariant nullValue = QgsApplication::nullRepresentation();
453
454 QgsFeature ft;
455 QgsFeatureIterator fit = mFilterExpression.isEmpty()
456 ? mReferencedLayer->getFeatures()
457 : mReferencedLayer->getFeatures( mFilterExpression );
458 while ( fit.nextFeature( ft ) )
459 {
460 const int count = std::min( mFilterComboBoxes.count(), mFilterFields.count() );
461 for ( int i = 0; i < count - 1; i++ )
462 {
463 QVariant cv = ft.attribute( mFilterFields.at( i ) );
464 QVariant nv = ft.attribute( mFilterFields.at( i + 1 ) );
465 QString cf = QgsVariantUtils::isNull( cv ) ? nullValue.toString() : cv.toString();
466 QString nf = QgsVariantUtils::isNull( nv ) ? nullValue.toString() : nv.toString();
467 mFilterCache[mFilterFields[i]][cf] << nf;
468 }
469 }
470
471 if ( !mFilterComboBoxes.isEmpty() )
472 {
473 QComboBox *cb = mFilterComboBoxes.first();
474 cb->setCurrentIndex( 0 );
475 disableChainedComboBoxes( cb );
476 }
477 }
478 }
479 else
480 {
481 mFilterContainer->hide();
482 }
483
484 mComboBox->setSourceLayer( mReferencedLayer );
485 mComboBox->setDisplayExpression( mReferencedLayer->displayExpression() );
486 mComboBox->setAllowNull( mAllowNull );
487 mComboBox->setIdentifierFields( mReferencedFields );
488 mComboBox->setFetchLimit( mFetchLimit );
489
490 if ( !mFilterExpression.isEmpty() )
491 mComboBox->setFilterExpression( mFilterExpression );
492
493 QVariant nullValue = QgsApplication::nullRepresentation();
494
495 if ( mChainFilters && mFeature.isValid() )
496 {
497 for ( int i = 0; i < mFilterFields.size(); i++ )
498 {
499 QVariant v = mFeature.attribute( mFilterFields[i] );
500 QString f = QgsVariantUtils::isNull( v ) ? nullValue.toString() : v.toString();
501 mFilterComboBoxes.at( i )->setCurrentIndex( mFilterComboBoxes.at( i )->findText( f ) );
502 }
503 }
504
505 // Only connect after iterating, to have only one iterator on the referenced table at once
506 connect( mComboBox, &QgsFeatureListComboBox::currentFeatureChanged, this, &QgsRelationReferenceWidget::comboReferenceChanged );
507 // To avoid wrongly signaling a foreign key change, handle model feature found state following feature gathering separately
508 connect( mComboBox, &QgsFeatureListComboBox::currentFeatureFoundChanged, this, &QgsRelationReferenceWidget::comboReferenceFoundChanged );
509
510 QApplication::restoreOverrideCursor();
511
512 mInitialized = true;
513 }
514}
515
516void QgsRelationReferenceWidget::highlightActionTriggered( QAction *action )
517{
518 if ( action == mHighlightFeatureAction )
519 {
520 highlightFeature();
521 }
522 else if ( action == mScaleHighlightFeatureAction )
523 {
524 highlightFeature( QgsFeature(), Scale );
525 }
526 else if ( action == mPanHighlightFeatureAction )
527 {
528 highlightFeature( QgsFeature(), Pan );
529 }
530}
531
533{
535
536 if ( !feat.isValid() )
537 return;
538
540 QgsAttributeDialog *attributeDialog = new QgsAttributeDialog( mReferencedLayer, new QgsFeature( feat ), true, this, true, context );
541 attributeDialog->show();
542}
543
544void QgsRelationReferenceWidget::highlightFeature( QgsFeature f, CanvasExtent canvasExtent )
545{
546 if ( !mCanvas )
547 return;
548
549 if ( !f.isValid() )
550 {
551 f = referencedFeature();
552 if ( !f.isValid() )
553 return;
554 }
555
556 if ( !f.hasGeometry() )
557 {
558 return;
559 }
560
561 QgsGeometry geom = f.geometry();
562
563 // scale or pan
564 if ( canvasExtent == Scale )
565 {
566 QgsRectangle featBBox = geom.boundingBox();
567 featBBox = mCanvas->mapSettings().layerToMapCoordinates( mReferencedLayer, featBBox );
568 QgsRectangle extent = mCanvas->extent();
569 if ( !extent.contains( featBBox ) )
570 {
571 extent.combineExtentWith( featBBox );
572 extent.scale( 1.1 );
573 mCanvas->setExtent( extent, true );
574 mCanvas->refresh();
575 }
576 }
577 else if ( canvasExtent == Pan )
578 {
579 QgsGeometry centroid = geom.centroid();
580 QgsPointXY center = centroid.asPoint();
581 center = mCanvas->mapSettings().layerToMapCoordinates( mReferencedLayer, center );
582 mCanvas->zoomByFactor( 1.0, &center ); // refresh is done in this method
583 }
584
585 // highlight
586 deleteHighlight();
587 mHighlight = new QgsHighlight( mCanvas, f, mReferencedLayer );
588 mHighlight->applyDefaultStyle();
589 mHighlight->show();
590
591 QTimer *timer = new QTimer( this );
592 timer->setSingleShot( true );
593 connect( timer, &QTimer::timeout, this, &QgsRelationReferenceWidget::deleteHighlight );
594 timer->start( 3000 );
595}
596
597void QgsRelationReferenceWidget::deleteHighlight()
598{
599 if ( mHighlight )
600 {
601 mHighlight->hide();
602 delete mHighlight;
603 }
604 mHighlight = nullptr;
605}
606
608{
609 if ( !mAllowMapIdentification || !mReferencedLayer )
610 return;
611
612 const QgsVectorLayerTools *tools = mEditorContext.vectorLayerTools();
613 if ( !tools )
614 return;
615 if ( !mCanvas )
616 return;
617
618 mMapToolIdentify->setLayer( mReferencedLayer );
619 setMapTool( mMapToolIdentify );
620
621 connect( mMapToolIdentify, qOverload<const QgsFeature &>( &QgsMapToolIdentifyFeature::featureIdentified ), this, &QgsRelationReferenceWidget::featureIdentified );
622
623 if ( mMessageBar )
624 {
625 QString title = tr( "Relation %1 for %2." ).arg( mRelation.name(), mReferencingLayer->name() );
626 QString msg = tr( "Identify a feature of %1 to be associated. Press &lt;ESC&gt; to cancel." ).arg( mReferencedLayer->name() );
627 mMessageBarItem = QgsMessageBar::createMessage( title, msg, this );
628 mMessageBar->pushItem( mMessageBarItem );
629 }
630}
631
632void QgsRelationReferenceWidget::comboReferenceChanged()
633{
634 mReferencedLayer->getFeatures( mComboBox->currentFeatureRequest() ).nextFeature( mFeature );
635 highlightFeature( mFeature );
636 updateAttributeEditorFrame( mFeature );
637
638 emitForeignKeysChanged( mComboBox->identifierValues() );
639}
640
641void QgsRelationReferenceWidget::comboReferenceFoundChanged( bool )
642{
643 mReferencedLayer->getFeatures( mComboBox->currentFeatureRequest() ).nextFeature( mFeature );
644 highlightFeature( mFeature );
645 updateAttributeEditorFrame( mFeature );
646}
647
648void QgsRelationReferenceWidget::updateAttributeEditorFrame( const QgsFeature &feature )
649{
650 mOpenFormButton->setEnabled( feature.isValid() );
651 // Check if we're running with an embedded frame we need to update
652 if ( mAttributeEditorFrame && mReferencedAttributeForm )
653 {
654 mReferencedAttributeForm->setFeature( feature );
655 }
656}
657
659{
660 return mAllowAddFeatures;
661}
662
664{
665 mAllowAddFeatures = allowAddFeatures;
666 updateAddEntryButton();
667}
668
670{
671 return mRelation;
672}
673
674void QgsRelationReferenceWidget::featureIdentified( const QgsFeature &feature )
675{
676 mComboBox->setCurrentFeature( feature );
677 mFeature = feature;
678
679 mRemoveFKButton->setEnabled( mIsEditable );
680 highlightFeature( feature );
681 updateAttributeEditorFrame( feature );
682 emitForeignKeysChanged( foreignKeys(), true );
683
684 unsetMapTool();
685}
686
687void QgsRelationReferenceWidget::setMapTool( QgsMapTool *mapTool )
688{
689 mCurrentMapTool = mapTool;
690 mCanvas->setMapTool( mapTool );
691
692 mWindowWidget = window();
693
694 mCanvas->window()->raise();
695 mCanvas->activateWindow();
696 mCanvas->setFocus();
697 connect( mapTool, &QgsMapTool::deactivated, this, &QgsRelationReferenceWidget::mapToolDeactivated );
698}
699
700void QgsRelationReferenceWidget::unsetMapTool()
701{
702 // deactivate map tools if activated
703 if ( mCurrentMapTool )
704 {
705 /* this will call mapToolDeactivated */
706 mCanvas->unsetMapTool( mCurrentMapTool );
707
708 if ( mCurrentMapTool == mMapToolDigitize )
709 {
710 disconnect( mCanvas, &QgsMapCanvas::keyPressed, this, &QgsRelationReferenceWidget::onKeyPressed );
711 disconnect( mMapToolDigitize, &QgsMapToolDigitizeFeature::digitizingCompleted, this, &QgsRelationReferenceWidget::entryAdded );
712 }
713 else
714 {
715 disconnect( mMapToolIdentify, qOverload<const QgsFeature &>( &QgsMapToolIdentifyFeature::featureIdentified ), this, &QgsRelationReferenceWidget::featureIdentified );
716 }
717 }
718}
719
720void QgsRelationReferenceWidget::onKeyPressed( QKeyEvent *e )
721{
722 if ( e->key() == Qt::Key_Escape )
723 {
724 unsetMapTool();
725 }
726}
727
728void QgsRelationReferenceWidget::mapToolDeactivated()
729{
730 if ( mWindowWidget )
731 {
732 mWindowWidget->raise();
733 mWindowWidget->activateWindow();
734 }
735
736 if ( mMessageBar && mMessageBarItem )
737 {
738 mMessageBar->popWidget( mMessageBarItem );
739 }
740 mMessageBarItem = nullptr;
741}
742
743void QgsRelationReferenceWidget::filterChanged()
744{
745 QVariant nullValue = QgsApplication::nullRepresentation();
746
747 QMap<QString, QString> filters;
748 QgsAttributeList attrs;
749
750 QComboBox *scb = qobject_cast<QComboBox *>( sender() );
751
752 Q_ASSERT( scb );
753
754 QgsFeature f;
755 QgsFeatureIds featureIds;
756 QString filterExpression = mFilterExpression;
757
758 // wrap the expression with parentheses as it might contain `OR`
759 if ( !filterExpression.isEmpty() )
760 filterExpression = QStringLiteral( " ( %1 ) " ).arg( filterExpression );
761
762 // comboboxes have to be disabled before building filters
763 if ( mChainFilters )
764 disableChainedComboBoxes( scb );
765
766 // build filters
767 const auto constMFilterComboBoxes = mFilterComboBoxes;
768 for ( QComboBox *cb : constMFilterComboBoxes )
769 {
770 if ( cb->currentIndex() != 0 )
771 {
772 const QString fieldName = cb->property( "Field" ).toString();
773
774 if ( cb->currentText() == nullValue.toString() )
775 {
776 filters[fieldName] = QStringLiteral( "\"%1\" IS NULL" ).arg( fieldName );
777 }
778 else
779 {
780 filters[fieldName] = QgsExpression::createFieldEqualityExpression( fieldName, cb->currentText() );
781 }
782 attrs << mReferencedLayer->fields().lookupField( fieldName );
783 }
784 }
785
786 if ( mChainFilters )
787 {
788 QComboBox *ccb = nullptr;
789 const auto constMFilterComboBoxes = mFilterComboBoxes;
790 for ( QComboBox *cb : constMFilterComboBoxes )
791 {
792 if ( !ccb )
793 {
794 if ( cb == scb )
795 ccb = cb;
796
797 continue;
798 }
799
800 if ( ccb->currentIndex() != 0 )
801 {
802 const QString fieldName = cb->property( "Field" ).toString();
803
804 cb->blockSignals( true );
805 cb->clear();
806 cb->addItem( cb->property( "FieldAlias" ).toString() );
807
808 // ccb = scb
809 // cb = scb + 1
810 QStringList texts;
811 const auto txts { mFilterCache[ccb->property( "Field" ).toString()][ccb->currentText()] };
812 for ( const QString &txt : txts )
813 {
814 QMap<QString, QString> filtersAttrs = filters;
815 filtersAttrs[fieldName] = QgsExpression::createFieldEqualityExpression( fieldName, txt );
816 QgsAttributeList subset = attrs;
817
818 QString expression = filterExpression;
819 if ( !expression.isEmpty() && !filtersAttrs.isEmpty() )
820 expression += QLatin1String( " AND " );
821
822 expression += filtersAttrs.isEmpty() ? QString() : QStringLiteral( " ( " );
823 expression += qgsMapJoinValues( filtersAttrs, QLatin1String( " AND " ) );
824 expression += filtersAttrs.isEmpty() ? QString() : QStringLiteral( " ) " );
825
826 subset << mReferencedLayer->fields().lookupField( fieldName );
827
828 QgsFeatureIterator it( mReferencedLayer->getFeatures( QgsFeatureRequest().setFilterExpression( expression ).setSubsetOfAttributes( subset ) ) );
829
830 bool found = false;
831 while ( it.nextFeature( f ) )
832 {
833 if ( !featureIds.contains( f.id() ) )
834 featureIds << f.id();
835
836 found = true;
837 }
838
839 // item is only provided if at least 1 feature exists
840 if ( found )
841 texts << txt;
842 }
843
844 texts.sort();
845 cb->addItems( texts );
846
847 cb->setEnabled( true );
848 cb->blockSignals( false );
849
850 ccb = cb;
851 }
852 }
853 }
854
855 if ( !filterExpression.isEmpty() && !filters.isEmpty() )
856 filterExpression += QLatin1String( " AND " );
857
858 filterExpression += filters.isEmpty() ? QString() : QStringLiteral( " ( " );
859 filterExpression += qgsMapJoinValues( filters, QLatin1String( " AND " ) );
860 filterExpression += filters.isEmpty() ? QString() : QStringLiteral( " ) " );
861
863}
864
865void QgsRelationReferenceWidget::addEntry()
866{
867 if ( !mReferencedLayer )
868 return;
869
870 const QgsVectorLayerTools *tools = mEditorContext.vectorLayerTools();
871 if ( !tools )
872 return;
873 if ( !mCanvas )
874 return;
875
876 // no geometry, skip the digitizing
877 if ( mReferencedLayer->geometryType() == Qgis::GeometryType::Unknown || mReferencedLayer->geometryType() == Qgis::GeometryType::Null )
878 {
879 QgsFeature f( mReferencedLayer->fields() );
880 entryAdded( f );
881 return;
882 }
883
884 mMapToolDigitize->setLayer( mReferencedLayer );
885 setMapTool( mMapToolDigitize );
886
887 connect( mMapToolDigitize, &QgsMapToolDigitizeFeature::digitizingCompleted, this, &QgsRelationReferenceWidget::entryAdded );
888 connect( mCanvas, &QgsMapCanvas::keyPressed, this, &QgsRelationReferenceWidget::onKeyPressed );
889
890 if ( mMessageBar )
891 {
892 QString title = tr( "Relation %1 for %2." ).arg( mRelation.name(), mReferencingLayer->name() );
893
894 QString displayString = QgsVectorLayerUtils::getFeatureDisplayString( mReferencingLayer, mComboBox->formFeature() );
895 QString msg = tr( "Link feature to %1 \"%2\" : Digitize the geometry for the new feature on layer %3. Press &lt;ESC&gt; to cancel." )
896 .arg( mReferencingLayer->name(), displayString, mReferencedLayer->name() );
897 mMessageBarItem = QgsMessageBar::createMessage( title, msg, this );
898 mMessageBar->pushItem( mMessageBarItem );
899 }
900}
901
902void QgsRelationReferenceWidget::entryAdded( const QgsFeature &feat )
903{
904 QgsFeature f( feat );
905 QgsAttributeMap attributes;
906
907 // if custom text is in the combobox and the displayExpression is simply a field, use the current text for the new feature
908 if ( mComboBox->itemText( mComboBox->currentIndex() ) != mComboBox->currentText() )
909 {
910 int fieldIdx = mReferencedLayer->fields().lookupField( mReferencedLayer->displayExpression() );
911
912 if ( fieldIdx != -1 )
913 {
914 attributes.insert( fieldIdx, mComboBox->currentText() );
915 }
916 }
917
918 if ( mEditorContext.vectorLayerTools()->addFeature( mReferencedLayer, attributes, f.geometry(), &f, this, false, true ) )
919 {
920 QVariantList attrs;
921 for ( const QString &fieldName : std::as_const( mReferencedFields ) )
922 attrs << f.attribute( fieldName );
923
924 setForeignKeys( attrs );
925
926 mAddEntryButton->setEnabled( false );
927 }
928
929 unsetMapTool();
930}
931
932void QgsRelationReferenceWidget::updateAddEntryButton()
933{
934 mAddEntryButton->setVisible( mAllowAddFeatures && mMapToolDigitize );
935 mAddEntryButton->setEnabled( mReferencedLayer && mReferencedLayer->isEditable() );
936}
937
938void QgsRelationReferenceWidget::disableChainedComboBoxes( const QComboBox *scb )
939{
940 QComboBox *ccb = nullptr;
941 const auto constMFilterComboBoxes = mFilterComboBoxes;
942 for ( QComboBox *cb : constMFilterComboBoxes )
943 {
944 if ( !ccb )
945 {
946 if ( cb == scb )
947 {
948 ccb = cb;
949 }
950
951 continue;
952 }
953
954 cb->setCurrentIndex( 0 );
955 if ( ccb->currentIndex() == 0 )
956 {
957 cb->setEnabled( false );
958 }
959
960 ccb = cb;
961 }
962}
963
964void QgsRelationReferenceWidget::emitForeignKeysChanged( const QVariantList &foreignKeys, bool force )
965{
966 if ( foreignKeys == mForeignKeys && force == false && qVariantListIsNull( foreignKeys ) == qVariantListIsNull( mForeignKeys ) )
967 return;
968
969 mForeignKeys = foreignKeys;
971 emit foreignKeyChanged( foreignKeys.at( 0 ) );
974}
975
977{
978 return mReferencedLayerName;
979}
980
981void QgsRelationReferenceWidget::setReferencedLayerName( const QString &relationLayerName )
982{
983 mReferencedLayerName = relationLayerName;
984}
985
987{
988 return mReferencedLayerId;
989}
990
991void QgsRelationReferenceWidget::setReferencedLayerId( const QString &relationLayerId )
992{
993 mReferencedLayerId = relationLayerId;
994}
995
997{
998 return mReferencedLayerProviderKey;
999}
1000
1001void QgsRelationReferenceWidget::setReferencedLayerProviderKey( const QString &relationProviderKey )
1002{
1003 mReferencedLayerProviderKey = relationProviderKey;
1004}
1005
1007{
1008 return mReferencedLayerDataSource;
1009}
1010
1011void QgsRelationReferenceWidget::setReferencedLayerDataSource( const QString &relationDataSource )
1012{
1013 const QgsPathResolver resolver { QgsProject::instance()->pathResolver() };
1014 mReferencedLayerDataSource = resolver.writePath( relationDataSource );
1015}
1016
1018{
1019 mComboBox->setFormFeature( formFeature );
1020}
1021
1023{
1024 mComboBox->setParentFormFeature( parentFormFeature );
1025}
void reset(T *p=nullptr)
Will reset the managed pointer to p.
@ Unknown
Unknown types.
@ Null
No geometry.
static QString nullRepresentation()
Returns the string used to represent the value NULL throughout QGIS.
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
An attribute table dialog for a vector layer.
void show()
Show the dialog non-blocking. Reparents this dialog to be a child of the dialog form.
Contains context information for attribute editor widgets.
QgsAdvancedDigitizingDockWidget * cadDockWidget() const
Returns the associated CAD dock widget (e.g.
@ Single
When showing a single feature (e.g. district information when looking at the form of a house)
const QgsVectorLayerTools * vectorLayerTools() const
Returns the associated vector layer tools.
@ Embed
A form was embedded as a widget on another form.
@ StandaloneDialog
A form was opened as a new dialog.
The attribute form widget for vector layer features.
void setFeature(const QgsFeature &feature)
Update all editors to correspond to a different feature.
bool save()
Save all the values from the editors to the layer.
A groupbox that collapses/expands when toggled and can save its collapsed and checked states.
static QString createFieldEqualityExpression(const QString &fieldName, const QVariant &value, QMetaType::Type fieldType=QMetaType::Type::UnknownType)
Create an expression allowing to evaluate if a field is equal to a value.
Wrapper for iterator of features from vector data provider or vector layer.
bool nextFeature(QgsFeature &f)
Fetch next feature and stores in f, returns true on success.
This offers a combobox with autocompleter that allows selecting features from a layer.
void setIdentifierValues(const QVariantList &identifierValues)
The identifier values of the currently selected feature.
void setDisplayExpression(const QString &displayExpression)
The display expression will be used to display features as well as the value to match the typed text ...
void setFilterExpression(const QString &filterExpression)
An additional expression to further restrict the available features.
void setIdentifierFields(const QStringList &identifierFields)
Field name that will be used to uniquely identify the current feature.
void setSourceLayer(QgsVectorLayer *sourceLayer)
The layer from which features should be listed.
void setParentFormFeature(const QgsFeature &feature)
Sets a parent attribute form feature to be used with the filter expression.
QgsFeature formFeature() const
Returns an attribute form feature to be used with the filter expression.
QgsFeatureRequest currentFeatureRequest() const
Shorthand for getting a feature request to query the currently selected feature.
void setIdentifierValuesToNull()
Sets the identifier values of the currently selected feature to NULL value(s).
void setFormFeature(const QgsFeature &feature)
Sets an attribute form feature to be used with the filter expression.
void setCurrentFeature(const QgsFeature &feature)
Sets the current index by using the given feature.
void currentFeatureFoundChanged(bool found)
Emitted when the feature picker model changes its feature found state.
void currentFeatureChanged()
Emitted when the current feature changes.
void setAllowNull(bool allowNull)
Determines if a NULL value should be available in the list.
void setFetchLimit(int fetchLimit)
Defines the feature request fetch limit If set to 0, no limit is applied when fetching.
Wraps a request for features to a vector layer (or directly its vector data provider).
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:58
QgsFeatureId id
Definition qgsfeature.h:66
QgsGeometry geometry
Definition qgsfeature.h:69
bool hasGeometry() const
Returns true if the feature has an associated geometry.
bool isValid() const
Returns the validity of this feature.
Q_INVOKABLE QVariant attribute(const QString &name) const
Lookup attribute value by attribute name.
QMetaType::Type type
Definition qgsfield.h:60
QgsField at(int i) const
Returns the field at particular index (must be in range 0..N-1).
Q_INVOKABLE int lookupField(const QString &fieldName) const
Looks up field's index from the field name.
A geometry is the spatial representation of a feature.
QgsPointXY asPoint() const
Returns the contents of the geometry as a 2-dimensional point.
QgsGeometry centroid() const
Returns the center of mass of a geometry.
QgsRectangle boundingBox() const
Returns the bounding box of the geometry.
Highlights features on the map.
void applyDefaultStyle()
Applies the default style from the user settings to the highlight.
Map canvas is a class for displaying all GIS data types on a canvas.
void setExtent(const QgsRectangle &r, bool magnified=false)
Sets the extent of the map canvas to the specified rectangle.
void zoomByFactor(double scaleFactor, const QgsPointXY *center=nullptr, bool ignoreScaleLock=false)
Zoom with the factor supplied.
void unsetMapTool(QgsMapTool *mapTool)
Unset the current map tool or last non zoom tool.
void keyPressed(QKeyEvent *e)
Emit key press event.
void setMapTool(QgsMapTool *mapTool, bool clean=false)
Sets the map tool currently being used on the canvas.
const QgsMapSettings & mapSettings() const
Gets access to properties used for map rendering.
QgsRectangle extent() const
Returns the current zoom extent of the map canvas.
void refresh()
Repaints the canvas map.
QString name
Definition qgsmaplayer.h:81
void editingStopped()
Emitted when edited changes have been successfully written to the data provider.
QString providerType() const
Returns the provider type (provider key) for this layer.
void editingStarted()
Emitted when editing on this layer has started.
QString publicSource(bool hidePassword=false) const
Gets a version of the internal layer definition that has sensitive bits removed (for example,...
QgsPointXY layerToMapCoordinates(const QgsMapLayer *layer, QgsPointXY point) const
transform point coordinates from layer's CRS to output CRS
This tool digitizes geometry of new point/line/polygon features on already existing vector layers.
void digitizingCompleted(const QgsFeature &feature)
Emitted whenever the digitizing has been successfully completed.
void setLayer(QgsMapLayer *vl)
Change the layer edited by the map tool.
A map tool to identify a feature on a chosen layer.
void featureIdentified(const QgsFeature &feature)
Emitted when a feature has been identified.
void setLayer(QgsVectorLayer *vl)
change the layer used by the map tool to identify
Abstract base class for all map tools.
Definition qgsmaptool.h:72
void deactivated()
Emitted when the map tool is deactivated.
void setButton(QAbstractButton *button)
Use this to associate a button to this maptool.
A bar for displaying non-blocking messages to the user.
bool popWidget(QgsMessageBarItem *item)
Remove the specified item from the bar, and display the next most recent one in the stack.
void pushItem(QgsMessageBarItem *item)
Display a message item on the bar, after hiding the currently visible one and putting it in a stack.
static QgsMessageBarItem * createMessage(const QString &text, QWidget *parent=nullptr)
Creates message bar item widget containing a message text to be displayed on the bar.
Resolves relative paths into absolute paths and vice versa.
Represents a 2D point.
Definition qgspointxy.h:60
static QgsProject * instance()
Returns the QgsProject singleton instance.
QgsPathResolver pathResolver() const
Returns path resolver object with considering whether the project uses absolute or relative paths and...
A rectangle specified with double values.
void scale(double scaleFactor, const QgsPointXY *c=nullptr)
Scale the rectangle around its center point.
bool contains(const QgsRectangle &rect) const
Returns true when rectangle contains other rectangle.
void combineExtentWith(const QgsRectangle &rect)
Expands the rectangle so that it covers both the original rectangle and the given rectangle.
void showEvent(QShowEvent *e) override
void setFilterExpression(const QString &filterExpression)
If not empty, will be used as filter expression.
QString filterExpression() const
Returns the currently set filter expression.
void setReferencedLayerProviderKey(const QString &referencedLayerProviderKey)
Set the data provider key of the referenced layer to referencedLayerProviderKey.
QString referencedLayerDataSource() const
Returns the public data source of the referenced layer.
bool allowAddFeatures() const
Determines if a button for adding new features should be shown.
Q_DECL_DEPRECATED void setForeignKey(const QVariant &value)
this sets the related feature using from the foreign key
QString referencedLayerProviderKey() const
Returns the data provider key of the referenced layer.
void setChainFilters(bool chainFilters)
Set if filters are chained.
QString referencedLayerId() const
Returns the id of the referenced layer.
void setEditorContext(const QgsAttributeEditorContext &context, QgsMapCanvas *canvas, QgsMessageBar *messageBar)
Sets the editor context.
void setReferencedLayerName(const QString &referencedLayerName)
Set the name of the referenced layer to referencedLayerName.
bool saveReferencedAttributeForm()
Trigger save of the embedded referenced attribute form.
void showIndeterminateState()
Sets the widget to display in an indeterminate "mixed value" state.
void openForm()
open the form of the related feature in a new dialog
void setReferencedLayerDataSource(const QString &referencedLayerDataSource)
Set the public data source of the referenced layer to referencedLayerDataSource.
void setParentFormFeature(const QgsFeature &parentFormFeature)
Set the current parent form feature.
void setFilterFields(const QStringList &filterFields)
Sets the fields for which filter comboboxes will be created.
void setAllowMapIdentification(bool allowMapIdentification)
bool allowMapIdentification()
determines if the widget offers the possibility to select the related feature on the map (using a ded...
Q_DECL_DEPRECATED void foreignKeyChanged(const QVariant &key)
Emitted when the foreign key changed.
QgsRelation relation() const
Returns the current relation, which might be invalid.
void setReferencedLayerId(const QString &referencedLayerId)
Set the id of the referenced layer to referencedLayerId.
bool chainFilters() const
Determines if the filters are chained.
QVariantList foreignKeys() const
returns the related feature foreign key
void setForeignKeys(const QVariantList &values)
Sets the related feature using the foreign keys.
Q_DECL_DEPRECATED QVariant foreignKey() const
returns the related feature foreign key
void foreignKeysChanged(const QVariantList &keys)
Emitted when the foreign keys changed.
void setRelation(const QgsRelation &relation, bool allowNullValue)
void setOpenFormButtonVisible(bool openFormButtonVisible)
QgsFeature referencedFeature() const
Returns the related feature (from the referenced layer) if no feature is related, it returns an inval...
void mapIdentification()
activate the map tool to select a new related feature on the map
void setAllowAddFeatures(bool allowAddFeatures)
Determines if a button for adding new features should be shown.
void deleteForeignKeys()
unset the currently related feature
QString referencedLayerName() const
Returns the name of the referenced layer.
void setFormFeature(const QgsFeature &formFeature)
Set the current form feature (from the referencing layer)
Defines a relation between matching fields of the two involved tables of a relation.
Definition qgsrelation.h:69
Represents a relationship between two vector layers.
Definition qgsrelation.h:44
QString name
Definition qgsrelation.h:50
QgsVectorLayer * referencedLayer
Definition qgsrelation.h:49
QList< QgsRelation::FieldPair > fieldPairs() const
Returns the field pairs which form this relation The first element of each pair are the field names o...
QgsVectorLayer * referencingLayer
Definition qgsrelation.h:48
QString referencedLayerId() const
Access the referenced (parent) layer's id.
static bool isNull(const QVariant &variant, bool silenceNullWarnings=false)
Returns true if the specified variant should be considered a NULL value.
static QVariant createNullVariant(QMetaType::Type metaType)
Helper method to properly create a null QVariant from a metaType Returns the created QVariant.
Used to handle basic editing operations on vector layers.
virtual bool addFeature(QgsVectorLayer *layer, const QgsAttributeMap &defaultValues=QgsAttributeMap(), const QgsGeometry &defaultGeometry=QgsGeometry(), QgsFeature *feature=nullptr, QWidget *parentWidget=nullptr, bool showModal=true, bool hideParent=false) const
This method should/will be called, whenever a new feature will be added to the layer.
static QString getFeatureDisplayString(const QgsVectorLayer *layer, const QgsFeature &feature)
QString attributeDisplayName(int index) const
Convenience function that returns the attribute alias if defined or the field name else.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const FINAL
Queries the layer for features specified in request.
bool isEditable() const FINAL
Returns true if the provider is in editing mode.
QString displayExpression
Q_INVOKABLE Qgis::GeometryType geometryType() const
Returns point, line or polygon.
QSet< QVariant > uniqueValues(int fieldIndex, int limit=-1) const FINAL
Calculates a list of unique values contained within an attribute in the layer.
bool qgsVariantLessThan(const QVariant &lhs, const QVariant &rhs)
Compares two QVariant values and returns whether the first is less than the second.
Definition qgis.cpp:129
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:6819
QString qgsMapJoinValues(const QMap< Key, Value > &map, const QString &separator)
Joins all the map values into a single string with each element separated by the given separator.
Definition qgis.h:6360
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:6818
QgsSignalBlocker< Object > whileBlocking(Object *object)
Temporarily blocks signals from a QObject while calling a single method from the object.
Definition qgis.h:6190
QMap< int, QVariant > QgsAttributeMap
QSet< QgsFeatureId > QgsFeatureIds
QList< int > QgsAttributeList
Definition qgsfield.h:27
bool qVariantListIsNull(const QVariantList &list)