QGIS API Documentation 3.99.0-Master (d270888f95f)
Loading...
Searching...
No Matches
qgssourcefieldsproperties.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgssourcefieldsproperties.cpp
3 ---------------------
4 begin : July 2017
5 copyright : (C) 2017 by David Signer
6 email : david at opengis dot ch
7
8 ***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16
18
19#include "qgsaddattrdialog.h"
20#include "qgsapplication.h"
23#include "qgsgui.h"
24#include "qgsnative.h"
25#include "qgsproject.h"
26#include "qgsvectorlayer.h"
27
28#include <QString>
29
30#include "moc_qgssourcefieldsproperties.cpp"
31
32using namespace Qt::StringLiterals;
33
35 : QWidget( parent )
36 , mLayer( layer )
37{
38 if ( !layer )
39 return;
40
41 setupUi( this );
42 layout()->setContentsMargins( 0, 0, 0, 0 );
43
44 //button appearance
45 mAddAttributeButton->setIcon( QgsApplication::getThemeIcon( u"/mActionNewAttribute.svg"_s ) );
46 mDeleteAttributeButton->setIcon( QgsApplication::getThemeIcon( u"/mActionDeleteAttribute.svg"_s ) );
47 mToggleEditingButton->setIcon( QgsApplication::getThemeIcon( u"/mActionToggleEditing.svg"_s ) );
48 mCalculateFieldButton->setIcon( QgsApplication::getThemeIcon( u"/mActionCalculateField.svg"_s ) );
49 mSaveLayerEditsButton->setIcon( QgsApplication::getThemeIcon( u"/mActionSaveAllEdits.svg"_s ) );
50
51 //button signals
52 connect( mToggleEditingButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::toggleEditing );
53 connect( mAddAttributeButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::addAttributeClicked );
54 connect( mDeleteAttributeButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::deleteAttributeClicked );
55 connect( mCalculateFieldButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::calculateFieldClicked );
56 connect( mSaveLayerEditsButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::saveLayerEditsClicked );
57
58 //slots
59 connect( mLayer, &QgsVectorLayer::editingStarted, this, &QgsSourceFieldsProperties::editingToggled );
60 connect( mLayer, &QgsVectorLayer::editingStopped, this, &QgsSourceFieldsProperties::editingToggled );
61 connect( mLayer, &QgsVectorLayer::attributeAdded, this, &QgsSourceFieldsProperties::attributeAdded );
62 connect( mLayer, &QgsVectorLayer::attributeDeleted, this, &QgsSourceFieldsProperties::attributeDeleted );
63
64 //field list appearance
65 mFieldsList->setColumnCount( AttrColCount );
66 mFieldsList->setSelectionBehavior( QAbstractItemView::SelectRows );
67 mFieldsList->setDragDropMode( QAbstractItemView::DragOnly );
68 mFieldsList->setHorizontalHeaderItem( AttrIdCol, new QTableWidgetItem( tr( "Id" ) ) );
69 mFieldsList->setHorizontalHeaderItem( AttrNameCol, new QTableWidgetItem( tr( "Name" ) ) );
70 mFieldsList->setHorizontalHeaderItem( AttrTypeCol, new QTableWidgetItem( tr( "Type" ) ) );
71 mFieldsList->setHorizontalHeaderItem( AttrTypeNameCol, new QTableWidgetItem( tr( "Type name" ) ) );
72 mFieldsList->setHorizontalHeaderItem( AttrLengthCol, new QTableWidgetItem( tr( "Length" ) ) );
73 mFieldsList->setHorizontalHeaderItem( AttrPrecCol, new QTableWidgetItem( tr( "Precision" ) ) );
74 mFieldsList->setHorizontalHeaderItem( AttrCommentCol, new QTableWidgetItem( tr( "Comment" ) ) );
75 const auto configurationFlagsWi = new QTableWidgetItem( tr( "Configuration" ) );
76 configurationFlagsWi->setToolTip( tr( "Configures the field" ) );
77 mFieldsList->setHorizontalHeaderItem( AttrConfigurationFlagsCol, configurationFlagsWi );
78 mFieldsList->setHorizontalHeaderItem( AttrAliasCol, new QTableWidgetItem( tr( "Alias" ) ) );
79
80 mFieldsList->setSortingEnabled( true );
81 mFieldsList->sortByColumn( 0, Qt::AscendingOrder );
82 mFieldsList->setSelectionBehavior( QAbstractItemView::SelectRows );
83 mFieldsList->setSelectionMode( QAbstractItemView::ExtendedSelection );
84 mFieldsList->horizontalHeader()->setStretchLastSection( true );
85 mFieldsList->verticalHeader()->hide();
86
87 //load buttons and field list
89}
90
95
97{
98 disconnect( mFieldsList, &QTableWidget::cellChanged, this, &QgsSourceFieldsProperties::attributesListCellChanged );
99 const QgsFields &fields = mLayer->fields();
100
101 mIndexedWidgets.clear();
102 mFieldsList->setRowCount( 0 );
103
104 for ( int i = 0; i < fields.count(); ++i )
105 attributeAdded( i );
106
107 mFieldsList->resizeColumnsToContents();
108 connect( mFieldsList, &QTableWidget::cellChanged, this, &QgsSourceFieldsProperties::attributesListCellChanged );
109
110 connect( mFieldsList, &QTableWidget::cellPressed, this, &QgsSourceFieldsProperties::attributesListCellPressed );
111
113 updateFieldRenamingStatus();
114}
115
116void QgsSourceFieldsProperties::updateFieldRenamingStatus()
117{
118 const bool canRenameFields = mLayer->isEditable() && ( mLayer->dataProvider()->capabilities() & Qgis::VectorProviderCapability::RenameAttributes ) && !mLayer->readOnly();
119
120 for ( int row = 0; row < mFieldsList->rowCount(); ++row )
121 {
122 if ( canRenameFields )
123 mFieldsList->item( row, AttrNameCol )->setFlags( mFieldsList->item( row, AttrNameCol )->flags() | Qt::ItemIsEditable );
124 else
125 mFieldsList->item( row, AttrNameCol )->setFlags( mFieldsList->item( row, AttrNameCol )->flags() & ~Qt::ItemIsEditable );
126 }
127}
128
129void QgsSourceFieldsProperties::updateExpression()
130{
131 QToolButton *btn = qobject_cast<QToolButton *>( sender() );
132 Q_ASSERT( btn );
133
134 const int index = btn->property( "Index" ).toInt();
135
136 const QString exp = mLayer->expressionField( index );
137
138 QgsExpressionContext context;
141
142 QgsExpressionBuilderDialog dlg( mLayer, exp, nullptr, u"generic"_s, context );
143
144 if ( dlg.exec() )
145 {
146 mLayer->updateExpressionField( index, dlg.expressionText() );
147 loadRows();
148 }
149}
150
151void QgsSourceFieldsProperties::attributeAdded( int idx )
152{
153 const bool sorted = mFieldsList->isSortingEnabled();
154 if ( sorted )
155 mFieldsList->setSortingEnabled( false );
156
157 const QgsFields &fields = mLayer->fields();
158 const int row = mFieldsList->rowCount();
159 mFieldsList->insertRow( row );
160 setRow( row, idx, fields.at( idx ) );
161 mFieldsList->setCurrentCell( row, idx );
162
163 const QColor expressionColor = QColor( 103, 0, 243, 44 );
164 const QColor joinColor = QColor( 0, 243, 79, 44 );
165 const QColor defaultColor = QColor( 252, 255, 79, 44 );
166
167 for ( int i = 0; i < mFieldsList->columnCount(); i++ )
168 {
169 if ( i == AttrConfigurationFlagsCol )
170 continue;
171
172 switch ( mLayer->fields().fieldOrigin( idx ) )
173 {
175 if ( i == 7 )
176 continue;
177 mFieldsList->item( row, i )->setBackground( expressionColor );
178 break;
179
181 mFieldsList->item( row, i )->setBackground( joinColor );
182 break;
183
184 default:
185 if ( defaultColor.isValid() )
186 mFieldsList->item( row, i )->setBackground( defaultColor );
187 break;
188 }
189 }
190
191 if ( sorted )
192 mFieldsList->setSortingEnabled( true );
193}
194
195
196void QgsSourceFieldsProperties::attributeDeleted( int idx )
197{
198 mFieldsList->removeRow( mIndexedWidgets.at( idx )->row() );
199 mIndexedWidgets.removeAt( idx );
200 for ( int i = idx; i < mIndexedWidgets.count(); i++ )
201 {
202 mIndexedWidgets.at( i )->setData( Qt::DisplayRole, i );
203 }
204}
205
206void QgsSourceFieldsProperties::setRow( int row, int idx, const QgsField &field )
207{
208 QTableWidgetItem *dataItem = new QTableWidgetItem();
209 dataItem->setData( Qt::DisplayRole, idx );
210 dataItem->setIcon( mLayer->fields().iconForField( idx, true ) );
211 mFieldsList->setItem( row, AttrIdCol, dataItem );
212
213 // in case we insert and not append reindex remaining widgets by 1
214 for ( int i = idx + 1; i < mIndexedWidgets.count(); i++ )
215 mIndexedWidgets.at( i )->setData( Qt::DisplayRole, i );
216 mIndexedWidgets.insert( idx, mFieldsList->item( row, 0 ) );
217
218 mFieldsList->setItem( row, AttrNameCol, new QTableWidgetItem( field.name() ) );
219 mFieldsList->setItem( row, AttrAliasCol, new QTableWidgetItem( field.alias() ) );
220 mFieldsList->setItem( row, AttrTypeCol, new QTableWidgetItem( field.friendlyTypeString() ) );
221 mFieldsList->setItem( row, AttrTypeNameCol, new QTableWidgetItem( field.typeName() ) );
222 mFieldsList->setItem( row, AttrLengthCol, new QTableWidgetItem( QString::number( field.length() ) ) );
223 mFieldsList->setItem( row, AttrPrecCol, new QTableWidgetItem( QString::number( field.precision() ) ) );
224
225 if ( mLayer->fields().fieldOrigin( idx ) == Qgis::FieldOrigin::Expression )
226 {
227 QWidget *expressionWidget = new QWidget;
228 expressionWidget->setLayout( new QHBoxLayout );
229 QToolButton *editExpressionButton = new QToolButton;
230 editExpressionButton->setProperty( "Index", idx );
231 editExpressionButton->setIcon( QgsApplication::getThemeIcon( u"/mIconExpression.svg"_s ) );
232 connect( editExpressionButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::updateExpression );
233 expressionWidget->layout()->setContentsMargins( 0, 0, 0, 0 );
234 expressionWidget->layout()->addWidget( editExpressionButton );
235 expressionWidget->layout()->addWidget( new QLabel( mLayer->expressionField( idx ) ) );
236 expressionWidget->setStyleSheet( "*[class~=\"QWidget\"] { background-color: rgba( 103, 0, 243, 0.12 ); } QToolButton { background-color: rgba( 203, 100, 243, 0.6 ); }" );
237 mFieldsList->setCellWidget( row, AttrCommentCol, expressionWidget );
238 }
239 else
240 {
241 mFieldsList->setItem( row, AttrCommentCol, new QTableWidgetItem( field.comment() ) );
242 }
243
244 QList<int> notEditableCols = QList<int>()
245 << AttrIdCol
246 << AttrNameCol
247 << AttrAliasCol
248 << AttrTypeCol
251 << AttrPrecCol
253
254 const auto constNotEditableCols = notEditableCols;
255 for ( const int i : constNotEditableCols )
256 {
257 if ( notEditableCols[i] != AttrCommentCol || mLayer->fields().fieldOrigin( idx ) != Qgis::FieldOrigin::Expression )
258 mFieldsList->item( row, i )->setFlags( mFieldsList->item( row, i )->flags() & ~Qt::ItemIsEditable );
259 if ( notEditableCols[i] == AttrAliasCol )
260 mFieldsList->item( row, i )->setToolTip( tr( "Edit alias in the Form config tab" ) );
261 }
262 const bool canRenameFields = mLayer->isEditable() && ( mLayer->dataProvider()->capabilities() & Qgis::VectorProviderCapability::RenameAttributes ) && !mLayer->readOnly();
263 if ( canRenameFields )
264 mFieldsList->item( row, AttrNameCol )->setFlags( mFieldsList->item( row, AttrNameCol )->flags() | Qt::ItemIsEditable );
265 else
266 mFieldsList->item( row, AttrNameCol )->setFlags( mFieldsList->item( row, AttrNameCol )->flags() & ~Qt::ItemIsEditable );
267
268 // Flags
269 QgsCheckableComboBox *cb = new QgsCheckableComboBox( mFieldsList );
270 const QList<Qgis::FieldConfigurationFlag> flagList = qgsEnumList<Qgis::FieldConfigurationFlag>();
271 for ( const Qgis::FieldConfigurationFlag flag : flagList )
272 {
274 continue;
275
276 cb->addItemWithCheckState( QgsField::readableConfigurationFlag( flag ), mLayer->fieldConfigurationFlags( idx ).testFlag( flag ) ? Qt::Checked : Qt::Unchecked, QVariant::fromValue( flag ) );
277 }
278 mFieldsList->setCellWidget( row, AttrConfigurationFlagsCol, cb );
279}
280
282{
283 QgsDebugMsgLevel( "inserting attribute " + field.name() + " of type " + field.typeName(), 2 );
284 mLayer->beginEditCommand( tr( "Added attribute" ) );
285 if ( mLayer->addAttribute( field ) )
286 {
287 mLayer->endEditCommand();
288 return true;
289 }
290 else
291 {
292 mLayer->destroyEditCommand();
293 QMessageBox::critical( this, tr( "Add Field" ), tr( "Failed to add field '%1' of type '%2'. Is the field name unique?" ).arg( field.name(), field.typeName() ) );
294 return false;
295 }
296}
297
299{
300 applyToLayer( mLayer );
301}
302
303void QgsSourceFieldsProperties::applyToLayer( QgsVectorLayer *layer )
304{
305 for ( int i = 0; i < mFieldsList->rowCount(); i++ )
306 {
307 const int idx = mFieldsList->item( i, AttrIdCol )->data( Qt::DisplayRole ).toInt();
309
310 QgsCheckableComboBox *cb = qobject_cast<QgsCheckableComboBox *>( mFieldsList->cellWidget( i, AttrConfigurationFlagsCol ) );
311 if ( cb )
312 {
313 QgsCheckableItemModel *model = cb->model();
314 for ( int r = 0; r < model->rowCount(); ++r )
315 {
316 const QModelIndex index = model->index( r, 0 );
317 const Qgis::FieldConfigurationFlag flag = model->data( index, Qt::UserRole ).value<Qgis::FieldConfigurationFlag>();
318 const bool active = model->data( index, Qt::CheckStateRole ).value<Qt::CheckState>() == Qt::Checked ? true : false;
319 flags.setFlag( flag, active );
320 }
321 layer->setFieldConfigurationFlags( idx, flags );
322 }
323 }
324}
325
326// slots
327
328void QgsSourceFieldsProperties::editingToggled()
329{
331 updateFieldRenamingStatus();
332}
333
334void QgsSourceFieldsProperties::addAttributeClicked()
335{
336 if ( !mLayer || !mLayer->dataProvider() )
337 {
338 return;
339 }
340
341 QgsAddAttrDialog dialog( mLayer, this );
342 if ( dialog.exec() == QDialog::Accepted )
343 {
344 addAttribute( dialog.field() );
345 loadRows();
346 }
347}
348
349void QgsSourceFieldsProperties::deleteAttributeClicked()
350{
351 QSet<int> providerFields;
352 QSet<int> expressionFields;
353 const auto constSelectedItems = mFieldsList->selectedItems();
354 for ( QTableWidgetItem *item : constSelectedItems )
355 {
356 if ( item->column() == 0 )
357 {
358 const int idx = mIndexedWidgets.indexOf( item );
359 if ( idx < 0 )
360 continue;
361
362 if ( mLayer->fields().fieldOrigin( idx ) == Qgis::FieldOrigin::Expression )
363 expressionFields << idx;
364 else
365 providerFields << idx;
366 }
367 }
368
369 if ( !expressionFields.isEmpty() )
370 mLayer->deleteAttributes( expressionFields.values() );
371
372 if ( !providerFields.isEmpty() )
373 {
374 mLayer->beginEditCommand( tr( "Deleted attributes" ) );
375 if ( mLayer->deleteAttributes( providerFields.values() ) )
376 mLayer->endEditCommand();
377 else
378 mLayer->destroyEditCommand();
379 }
380}
381
382void QgsSourceFieldsProperties::calculateFieldClicked()
383{
384 if ( !mLayer || !mLayer->dataProvider() )
385 {
386 return;
387 }
388
389 QgsFieldCalculator calc( mLayer, this );
390 if ( calc.exec() == QDialog::Accepted )
391 {
392 loadRows();
393 }
394}
395
396void QgsSourceFieldsProperties::saveLayerEditsClicked()
397{
398 mLayer->commitChanges( false );
399}
400
401void QgsSourceFieldsProperties::attributesListCellChanged( int row, int column )
402{
403 if ( column == AttrNameCol && mLayer && mLayer->isEditable() )
404 {
405 const int idx = mIndexedWidgets.indexOf( mFieldsList->item( row, AttrIdCol ) );
406
407 QTableWidgetItem *nameItem = mFieldsList->item( row, column );
408 //avoiding that something will be changed, just because this is triggered by simple re-sorting
409 if ( !nameItem || nameItem->text().isEmpty() || !mLayer->fields().exists( idx ) || mLayer->fields().at( idx ).name() == nameItem->text() )
410 return;
411
412 mLayer->beginEditCommand( tr( "Rename attribute" ) );
413 if ( mLayer->renameAttribute( idx, nameItem->text() ) )
414 {
415 mLayer->endEditCommand();
416 }
417 else
418 {
419 mLayer->destroyEditCommand();
420 QMessageBox::critical( this, tr( "Rename Field" ), tr( "Failed to rename field to '%1'. Is the field name unique?" ).arg( nameItem->text() ) );
421 }
422 }
423}
424
425void QgsSourceFieldsProperties::attributesListCellPressed( int /*row*/, int /*column*/ )
426{
428}
429
430//NICE FUNCTIONS
432{
433 QgsVectorDataProvider *provider = mLayer->dataProvider();
434 if ( !provider )
435 return;
436 const Qgis::VectorProviderCapabilities cap = provider->capabilities();
437
438 mToggleEditingButton->setEnabled( ( cap & Qgis::VectorProviderCapability::ChangeAttributeValues ) && !mLayer->readOnly() );
439
440 if ( mLayer->isEditable() )
441 {
442 mDeleteAttributeButton->setEnabled( cap & Qgis::VectorProviderCapability::DeleteAttributes );
443 mAddAttributeButton->setEnabled( cap & Qgis::VectorProviderCapability::AddAttributes );
444 mToggleEditingButton->setChecked( true );
445 mSaveLayerEditsButton->setEnabled( true );
446 mSaveLayerEditsButton->setChecked( true );
447 }
448 else
449 {
450 mToggleEditingButton->setChecked( false );
451 mAddAttributeButton->setEnabled( false );
452 mSaveLayerEditsButton->setEnabled( false );
453
454 // Enable delete button if items are selected
455 mDeleteAttributeButton->setEnabled( !mFieldsList->selectedItems().isEmpty() );
456 // and only if all selected items have their origin in an expression
457 const auto constSelectedItems = mFieldsList->selectedItems();
458 for ( QTableWidgetItem *item : constSelectedItems )
459 {
460 if ( item->column() == 0 )
461 {
462 const int idx = mIndexedWidgets.indexOf( item );
463 if ( mLayer->fields().fieldOrigin( idx ) != Qgis::FieldOrigin::Expression )
464 {
465 mDeleteAttributeButton->setEnabled( false );
466 break;
467 }
468 }
469 }
470 }
471}
@ AddAttributes
Allows addition of new attributes (fields).
Definition qgis.h:523
@ RenameAttributes
Supports renaming attributes (fields).
Definition qgis.h:535
@ DeleteAttributes
Allows deletion of attributes (fields).
Definition qgis.h:524
@ ChangeAttributeValues
Allows modification of attribute values.
Definition qgis.h:522
QFlags< VectorProviderCapability > VectorProviderCapabilities
Vector data provider capabilities.
Definition qgis.h:555
@ Expression
Field is calculated from an expression.
Definition qgis.h:1767
@ Join
Field originates from a joined layer.
Definition qgis.h:1765
FieldConfigurationFlag
Configuration flags for fields These flags are meant to be user-configurable and are not describing a...
Definition qgis.h:1779
@ NoFlag
No flag is defined.
Definition qgis.h:1780
QFlags< FieldConfigurationFlag > FieldConfigurationFlags
Configuration flags for fields These flags are meant to be user-configurable and are not describing a...
Definition qgis.h:1794
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
QComboBox subclass which allows selecting multiple items.
QgsCheckableItemModel * model() const
Returns the custom item model which handles checking the items.
void addItemWithCheckState(const QString &text, Qt::CheckState state, const QVariant &userData=QVariant())
Adds an item to the combobox with the given text, check state (stored in the Qt::CheckStateRole) and ...
QStandardItemModel subclass which makes all items checkable by default.
QVariant data(const QModelIndex &index, int role=Qt::DisplayRole) const override
Returns the data stored under the given role for the item referred to by the index.
static QgsExpressionContextScope * projectScope(const QgsProject *project)
Creates a new scope which contains variables and functions relating to a QGIS project.
static QgsExpressionContextScope * globalScope()
Creates a new scope which contains variables and functions relating to the global QGIS context.
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:56
QString typeName() const
Gets the field type.
Definition qgsfield.cpp:167
QString name
Definition qgsfield.h:65
int precision
Definition qgsfield.h:62
int length
Definition qgsfield.h:61
QString friendlyTypeString() const
Returns a user friendly, translated representation of the field type.
Definition qgsfield.cpp:145
static QString readableConfigurationFlag(Qgis::FieldConfigurationFlag flag)
Returns the readable and translated value of the configuration flag.
Definition qgsfield.cpp:457
QString alias
Definition qgsfield.h:66
QString comment
Definition qgsfield.h:64
Container of fields for a vector layer.
Definition qgsfields.h:46
int count
Definition qgsfields.h:50
QgsField at(int i) const
Returns the field at particular index (must be in range 0..N-1).
void editingStopped()
Emitted when edited changes have been successfully written to the data provider.
void editingStarted()
Emitted when editing on this layer has started.
static QgsProject * instance()
Returns the QgsProject singleton instance.
QList< QTableWidgetItem * > mIndexedWidgets
QgsSourceFieldsProperties(QgsVectorLayer *layer, QWidget *parent=nullptr)
bool addAttribute(const QgsField &field)
Adds an attribute to the table (but does not commit it yet).
void setRow(int row, int idx, const QgsField &field)
Base class for vector data providers.
virtual Q_INVOKABLE Qgis::VectorProviderCapabilities capabilities() const
Returns flags containing the supported capabilities.
Represents a vector layer which manages a vector based dataset.
bool isEditable() const final
Returns true if the provider is in editing mode.
void setFieldConfigurationFlags(int index, Qgis::FieldConfigurationFlags flags)
Sets the configuration flags of the field at given index.
void attributeAdded(int idx)
Will be emitted, when a new attribute has been added to this vector layer.
void attributeDeleted(int idx)
Will be emitted, when an attribute has been deleted from this vector layer.
Qgis::FieldConfigurationFlags fieldConfigurationFlags(int index) const
Returns the configuration flags of the field at given index.
QgsVectorDataProvider * dataProvider() final
Returns the layer's data provider, it may be nullptr.
const QList< T > qgsEnumList()
Returns a list all enum entries.
Definition qgis.h:7057
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63