QGIS API Documentation 4.1.0-Master (5bf3c20f3c9)
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;
140
141 QgsExpressionBuilderDialog dlg( mLayer, exp, nullptr, u"generic"_s, context );
142
143 if ( dlg.exec() )
144 {
145 mLayer->updateExpressionField( index, dlg.expressionText() );
146 loadRows();
147 }
148}
149
150void QgsSourceFieldsProperties::attributeAdded( int idx )
151{
152 const bool sorted = mFieldsList->isSortingEnabled();
153 if ( sorted )
154 mFieldsList->setSortingEnabled( false );
155
156 const QgsFields &fields = mLayer->fields();
157 const int row = mFieldsList->rowCount();
158 mFieldsList->insertRow( row );
159 setRow( row, idx, fields.at( idx ) );
160 mFieldsList->setCurrentCell( row, idx );
161
162 const QColor expressionColor = QColor( 103, 0, 243, 44 );
163 const QColor joinColor = QColor( 0, 243, 79, 44 );
164 const QColor defaultColor = QColor( 252, 255, 79, 44 );
165
166 for ( int i = 0; i < mFieldsList->columnCount(); i++ )
167 {
168 if ( i == AttrConfigurationFlagsCol )
169 continue;
170
171 switch ( mLayer->fields().fieldOrigin( idx ) )
172 {
174 if ( i == 7 )
175 continue;
176 mFieldsList->item( row, i )->setBackground( expressionColor );
177 break;
178
180 mFieldsList->item( row, i )->setBackground( joinColor );
181 break;
182
183 default:
184 if ( defaultColor.isValid() )
185 mFieldsList->item( row, i )->setBackground( defaultColor );
186 break;
187 }
188 }
189
190 if ( sorted )
191 mFieldsList->setSortingEnabled( true );
192}
193
194
195void QgsSourceFieldsProperties::attributeDeleted( int idx )
196{
197 mFieldsList->removeRow( mIndexedWidgets.at( idx )->row() );
198 mIndexedWidgets.removeAt( idx );
199 for ( int i = idx; i < mIndexedWidgets.count(); i++ )
200 {
201 mIndexedWidgets.at( i )->setData( Qt::DisplayRole, i );
202 }
203}
204
205void QgsSourceFieldsProperties::setRow( int row, int idx, const QgsField &field )
206{
207 QTableWidgetItem *dataItem = new QTableWidgetItem();
208 dataItem->setData( Qt::DisplayRole, idx );
209 dataItem->setIcon( mLayer->fields().iconForField( idx, true ) );
210 mFieldsList->setItem( row, AttrIdCol, dataItem );
211
212 // in case we insert and not append reindex remaining widgets by 1
213 for ( int i = idx + 1; i < mIndexedWidgets.count(); i++ )
214 mIndexedWidgets.at( i )->setData( Qt::DisplayRole, i );
215 mIndexedWidgets.insert( idx, mFieldsList->item( row, 0 ) );
216
217 mFieldsList->setItem( row, AttrNameCol, new QTableWidgetItem( field.name() ) );
218 mFieldsList->setItem( row, AttrAliasCol, new QTableWidgetItem( field.alias() ) );
219 mFieldsList->setItem( row, AttrTypeCol, new QTableWidgetItem( field.friendlyTypeString() ) );
220 mFieldsList->setItem( row, AttrTypeNameCol, new QTableWidgetItem( field.typeName() ) );
221 mFieldsList->setItem( row, AttrLengthCol, new QTableWidgetItem( QString::number( field.length() ) ) );
222 mFieldsList->setItem( row, AttrPrecCol, new QTableWidgetItem( QString::number( field.precision() ) ) );
223
224 if ( mLayer->fields().fieldOrigin( idx ) == Qgis::FieldOrigin::Expression )
225 {
226 QWidget *expressionWidget = new QWidget;
227 expressionWidget->setLayout( new QHBoxLayout );
228 QToolButton *editExpressionButton = new QToolButton;
229 editExpressionButton->setProperty( "Index", idx );
230 editExpressionButton->setIcon( QgsApplication::getThemeIcon( u"/mIconExpression.svg"_s ) );
231 connect( editExpressionButton, &QAbstractButton::clicked, this, &QgsSourceFieldsProperties::updateExpression );
232 expressionWidget->layout()->setContentsMargins( 0, 0, 0, 0 );
233 expressionWidget->layout()->addWidget( editExpressionButton );
234 expressionWidget->layout()->addWidget( new QLabel( mLayer->expressionField( idx ) ) );
235 expressionWidget->setStyleSheet( "*[class~=\"QWidget\"] { background-color: rgba( 103, 0, 243, 0.12 ); } QToolButton { background-color: rgba( 203, 100, 243, 0.6 ); }" );
236 mFieldsList->setCellWidget( row, AttrCommentCol, expressionWidget );
237 }
238 else
239 {
240 mFieldsList->setItem( row, AttrCommentCol, new QTableWidgetItem( field.comment() ) );
241 }
242
243 QList<int> notEditableCols = QList<int>() << AttrIdCol << AttrNameCol << AttrAliasCol << AttrTypeCol << AttrTypeNameCol << AttrLengthCol << AttrPrecCol << AttrCommentCol;
244
245 const auto constNotEditableCols = notEditableCols;
246 for ( const int i : constNotEditableCols )
247 {
248 if ( notEditableCols[i] != AttrCommentCol || mLayer->fields().fieldOrigin( idx ) != Qgis::FieldOrigin::Expression )
249 mFieldsList->item( row, i )->setFlags( mFieldsList->item( row, i )->flags() & ~Qt::ItemIsEditable );
250 if ( notEditableCols[i] == AttrAliasCol )
251 mFieldsList->item( row, i )->setToolTip( tr( "Edit alias in the Form config tab" ) );
252 }
253 const bool canRenameFields = mLayer->isEditable() && ( mLayer->dataProvider()->capabilities() & Qgis::VectorProviderCapability::RenameAttributes ) && !mLayer->readOnly();
254 if ( canRenameFields )
255 mFieldsList->item( row, AttrNameCol )->setFlags( mFieldsList->item( row, AttrNameCol )->flags() | Qt::ItemIsEditable );
256 else
257 mFieldsList->item( row, AttrNameCol )->setFlags( mFieldsList->item( row, AttrNameCol )->flags() & ~Qt::ItemIsEditable );
258
259 // Flags
260 QgsCheckableComboBox *cb = new QgsCheckableComboBox( mFieldsList );
261 const QList<Qgis::FieldConfigurationFlag> flagList = qgsEnumList<Qgis::FieldConfigurationFlag>();
262 for ( const Qgis::FieldConfigurationFlag flag : flagList )
263 {
265 continue;
266
267 cb->addItemWithCheckState( QgsField::readableConfigurationFlag( flag ), mLayer->fieldConfigurationFlags( idx ).testFlag( flag ) ? Qt::Checked : Qt::Unchecked, QVariant::fromValue( flag ) );
268 }
269 mFieldsList->setCellWidget( row, AttrConfigurationFlagsCol, cb );
270}
271
273{
274 QgsDebugMsgLevel( "inserting attribute " + field.name() + " of type " + field.typeName(), 2 );
275 mLayer->beginEditCommand( tr( "Added attribute" ) );
276 if ( mLayer->addAttribute( field ) )
277 {
278 mLayer->endEditCommand();
279 return true;
280 }
281 else
282 {
283 mLayer->destroyEditCommand();
284 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() ) );
285 return false;
286 }
287}
288
290{
291 applyToLayer( mLayer );
292}
293
294void QgsSourceFieldsProperties::applyToLayer( QgsVectorLayer *layer )
295{
296 for ( int i = 0; i < mFieldsList->rowCount(); i++ )
297 {
298 const int idx = mFieldsList->item( i, AttrIdCol )->data( Qt::DisplayRole ).toInt();
300
301 QgsCheckableComboBox *cb = qobject_cast<QgsCheckableComboBox *>( mFieldsList->cellWidget( i, AttrConfigurationFlagsCol ) );
302 if ( cb )
303 {
304 QgsCheckableItemModel *model = cb->model();
305 for ( int r = 0; r < model->rowCount(); ++r )
306 {
307 const QModelIndex index = model->index( r, 0 );
308 const Qgis::FieldConfigurationFlag flag = model->data( index, Qt::UserRole ).value<Qgis::FieldConfigurationFlag>();
309 const bool active = model->data( index, Qt::CheckStateRole ).value<Qt::CheckState>() == Qt::Checked ? true : false;
310 flags.setFlag( flag, active );
311 }
312 layer->setFieldConfigurationFlags( idx, flags );
313 }
314 }
315}
316
317// slots
318
319void QgsSourceFieldsProperties::editingToggled()
320{
322 updateFieldRenamingStatus();
323}
324
325void QgsSourceFieldsProperties::addAttributeClicked()
326{
327 if ( !mLayer || !mLayer->dataProvider() )
328 {
329 return;
330 }
331
332 QgsAddAttrDialog dialog( mLayer, this );
333 if ( dialog.exec() == QDialog::Accepted )
334 {
335 addAttribute( dialog.field() );
336 loadRows();
337 }
338}
339
340void QgsSourceFieldsProperties::deleteAttributeClicked()
341{
342 QSet<int> providerFields;
343 QSet<int> expressionFields;
344 const auto constSelectedItems = mFieldsList->selectedItems();
345 for ( QTableWidgetItem *item : constSelectedItems )
346 {
347 if ( item->column() == 0 )
348 {
349 const int idx = mIndexedWidgets.indexOf( item );
350 if ( idx < 0 )
351 continue;
352
353 if ( mLayer->fields().fieldOrigin( idx ) == Qgis::FieldOrigin::Expression )
354 expressionFields << idx;
355 else
356 providerFields << idx;
357 }
358 }
359
360 if ( !expressionFields.isEmpty() )
361 mLayer->deleteAttributes( expressionFields.values() );
362
363 if ( !providerFields.isEmpty() )
364 {
365 mLayer->beginEditCommand( tr( "Deleted attributes" ) );
366 if ( mLayer->deleteAttributes( providerFields.values() ) )
367 mLayer->endEditCommand();
368 else
369 mLayer->destroyEditCommand();
370 }
371}
372
373void QgsSourceFieldsProperties::calculateFieldClicked()
374{
375 if ( !mLayer || !mLayer->dataProvider() )
376 {
377 return;
378 }
379
380 QgsFieldCalculator calc( mLayer, this );
381 if ( calc.exec() == QDialog::Accepted )
382 {
383 loadRows();
384 }
385}
386
387void QgsSourceFieldsProperties::saveLayerEditsClicked()
388{
389 mLayer->commitChanges( false );
390}
391
392void QgsSourceFieldsProperties::attributesListCellChanged( int row, int column )
393{
394 if ( column == AttrNameCol && mLayer && mLayer->isEditable() )
395 {
396 const int idx = mIndexedWidgets.indexOf( mFieldsList->item( row, AttrIdCol ) );
397
398 QTableWidgetItem *nameItem = mFieldsList->item( row, column );
399 //avoiding that something will be changed, just because this is triggered by simple re-sorting
400 if ( !nameItem || nameItem->text().isEmpty() || !mLayer->fields().exists( idx ) || mLayer->fields().at( idx ).name() == nameItem->text() )
401 return;
402
403 mLayer->beginEditCommand( tr( "Rename attribute" ) );
404 if ( mLayer->renameAttribute( idx, nameItem->text() ) )
405 {
406 mLayer->endEditCommand();
407 }
408 else
409 {
410 mLayer->destroyEditCommand();
411 QMessageBox::critical( this, tr( "Rename Field" ), tr( "Failed to rename field to '%1'. Is the field name unique?" ).arg( nameItem->text() ) );
412 }
413 }
414}
415
416void QgsSourceFieldsProperties::attributesListCellPressed( int /*row*/, int /*column*/ )
417{
419}
420
421//NICE FUNCTIONS
423{
424 QgsVectorDataProvider *provider = mLayer->dataProvider();
425 if ( !provider )
426 return;
427 const Qgis::VectorProviderCapabilities cap = provider->capabilities();
428
429 mToggleEditingButton->setEnabled( ( cap & Qgis::VectorProviderCapability::ChangeAttributeValues ) && !mLayer->readOnly() );
430
431 if ( mLayer->isEditable() )
432 {
433 mDeleteAttributeButton->setEnabled( cap & Qgis::VectorProviderCapability::DeleteAttributes );
434 mAddAttributeButton->setEnabled( cap & Qgis::VectorProviderCapability::AddAttributes );
435 mToggleEditingButton->setChecked( true );
436 mSaveLayerEditsButton->setEnabled( true );
437 mSaveLayerEditsButton->setChecked( true );
438 }
439 else
440 {
441 mToggleEditingButton->setChecked( false );
442 mAddAttributeButton->setEnabled( false );
443 mSaveLayerEditsButton->setEnabled( false );
444
445 // Enable delete button if items are selected
446 mDeleteAttributeButton->setEnabled( !mFieldsList->selectedItems().isEmpty() );
447 // and only if all selected items have their origin in an expression
448 const auto constSelectedItems = mFieldsList->selectedItems();
449 for ( QTableWidgetItem *item : constSelectedItems )
450 {
451 if ( item->column() == 0 )
452 {
453 const int idx = mIndexedWidgets.indexOf( item );
454 if ( mLayer->fields().fieldOrigin( idx ) != Qgis::FieldOrigin::Expression )
455 {
456 mDeleteAttributeButton->setEnabled( false );
457 break;
458 }
459 }
460 }
461 }
462}
@ AddAttributes
Allows addition of new attributes (fields).
Definition qgis.h:530
@ RenameAttributes
Supports renaming attributes (fields).
Definition qgis.h:542
@ DeleteAttributes
Allows deletion of attributes (fields).
Definition qgis.h:531
@ ChangeAttributeValues
Allows modification of attribute values.
Definition qgis.h:529
QFlags< VectorProviderCapability > VectorProviderCapabilities
Vector data provider capabilities.
Definition qgis.h:562
@ Expression
Field is calculated from an expression.
Definition qgis.h:1788
@ Join
Field originates from a joined layer.
Definition qgis.h:1786
FieldConfigurationFlag
Configuration flags for fields These flags are meant to be user-configurable and are not describing a...
Definition qgis.h:1800
@ NoFlag
No flag is defined.
Definition qgis.h:1801
QFlags< FieldConfigurationFlag > FieldConfigurationFlags
Configuration flags for fields These flags are meant to be user-configurable and are not describing a...
Definition qgis.h:1815
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:158
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:136
static QString readableConfigurationFlag(Qgis::FieldConfigurationFlag flag)
Returns the readable and translated value of the configuration flag.
Definition qgsfield.cpp:448
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:7123
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63