QGIS API Documentation 3.41.0-Master (cea29feecf2)
Loading...
Searching...
No Matches
qgsaddattrdialog.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsaddattrdialog.h - description
3 -------------------
4 begin : January 2005
5 copyright : (C) 2005 by Marco Hugentobler
7 ***************************************************************************/
8
9/***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
18#include "qgsaddattrdialog.h"
19#include "moc_qgsaddattrdialog.cpp"
20#include "qgsvectorlayer.h"
22#include "qgslogger.h"
23#include "qgsfields.h"
24
25#include <QMessageBox>
26
27QgsAddAttrDialog::QgsAddAttrDialog( QgsVectorLayer *vlayer, QWidget *parent, Qt::WindowFlags fl )
28 : QDialog( parent, fl )
29 , mIsShapeFile( vlayer && vlayer->providerType() == QLatin1String( "ogr" ) && vlayer->storageType() == QLatin1String( "ESRI Shapefile" ) )
30{
31 setupUi( this );
32 connect( mTypeBox, static_cast<void ( QComboBox::* )( int )>( &QComboBox::currentIndexChanged ), this, &QgsAddAttrDialog::mTypeBox_currentIndexChanged );
33 connect( mLength, &QSpinBox::editingFinished, this, &QgsAddAttrDialog::mLength_editingFinished );
34
35 if ( !vlayer || !vlayer->dataProvider() )
36 return;
37
39
40 //fill data types into the combo box
41 const QList<QgsVectorDataProvider::NativeType> &typelist = vlayer->dataProvider()->nativeTypes();
42 for ( int i = 0; i < typelist.size(); i++ )
43 {
44 QgsDebugMsgLevel( QStringLiteral( "name:%1 type:%2 typeName:%3 length:%4-%5 prec:%6-%7" ).arg( typelist[i].mTypeDesc ).arg( typelist[i].mType ).arg( typelist[i].mTypeName ).arg( typelist[i].mMinLen ).arg( typelist[i].mMaxLen ).arg( typelist[i].mMinPrec ).arg( typelist[i].mMaxPrec ), 2 );
45
46 whileBlocking( mTypeBox )->addItem( QgsFields::iconForFieldType( typelist[i].mType, typelist[i].mSubType, typelist[i].mTypeName ), typelist[i].mTypeDesc );
47 mTypeBox->setItemData( i, static_cast<int>( typelist[i].mType ), Qt::UserRole );
48 mTypeBox->setItemData( i, typelist[i].mTypeName, Qt::UserRole + 1 );
49 mTypeBox->setItemData( i, typelist[i].mMinLen, Qt::UserRole + 2 );
50 mTypeBox->setItemData( i, typelist[i].mMaxLen, Qt::UserRole + 3 );
51 mTypeBox->setItemData( i, typelist[i].mMinPrec, Qt::UserRole + 4 );
52 mTypeBox->setItemData( i, typelist[i].mMaxPrec, Qt::UserRole + 5 );
53 }
54
55 //default values for field width and precision
56 mLength->setValue( 10 );
57 mLength->setClearValue( 10 );
58 mPrec->setValue( 3 );
59 mPrec->setClearValue( 3 );
60 mTypeBox_currentIndexChanged( 0 );
61
62 if ( mIsShapeFile )
63 mNameEdit->setMaxLength( 10 );
64
65 mNameEdit->setFocus();
66
67 if ( !( attributeEditCapabilities & Qgis::VectorDataProviderAttributeEditCapability::EditAlias ) )
68 {
69 mLabelAlias->hide();
70 mAliasEdit->hide();
71 }
72 if ( !( attributeEditCapabilities & Qgis::VectorDataProviderAttributeEditCapability::EditComment ) )
73 {
74 mLabelComment->hide();
75 mCommentEdit->hide();
76 }
77}
78
79void QgsAddAttrDialog::setIllegalFieldNames( const QSet<QString> &names )
80{
81 mIllegalFieldNames = names;
82}
83
84void QgsAddAttrDialog::mTypeBox_currentIndexChanged( int idx )
85{
86 mTypeName->setText( mTypeBox->itemData( idx, Qt::UserRole + 1 ).toString() );
87
88 mLength->setMinimum( mTypeBox->itemData( idx, Qt::UserRole + 2 ).toInt() );
89 mLength->setMaximum( mTypeBox->itemData( idx, Qt::UserRole + 3 ).toInt() );
90 mLength->setVisible( mLength->minimum() < mLength->maximum() );
91 mLengthLabel->setVisible( mLength->minimum() < mLength->maximum() );
92 if ( mLength->value() < mLength->minimum() )
93 mLength->setValue( mLength->minimum() );
94 if ( mLength->value() > mLength->maximum() )
95 mLength->setValue( mLength->maximum() );
96 setPrecisionMinMax();
97}
98
99void QgsAddAttrDialog::mLength_editingFinished()
100{
101 setPrecisionMinMax();
102}
103
104void QgsAddAttrDialog::setPrecisionMinMax()
105{
106 const int idx = mTypeBox->currentIndex();
107 const int minPrecType = mTypeBox->itemData( idx, Qt::UserRole + 4 ).toInt();
108 const int maxPrecType = mTypeBox->itemData( idx, Qt::UserRole + 5 ).toInt();
109 const bool precisionIsEnabled = minPrecType < maxPrecType;
110 mPrec->setEnabled( precisionIsEnabled );
111 mPrec->setVisible( precisionIsEnabled );
112 mPrecLabel->setVisible( precisionIsEnabled );
113
114 // Do not set min/max if it's disabled or we'll loose the default value,
115 // see https://github.com/qgis/QGIS/issues/26880 - QGIS saves integer field when
116 // I create a new real field through field calculator (Update field works as intended)
117 if ( precisionIsEnabled )
118 {
119 mPrec->setMinimum( minPrecType );
120 mPrec->setMaximum( std::max( minPrecType, std::min( maxPrecType, mLength->value() ) ) );
121 }
122}
123
124void QgsAddAttrDialog::accept()
125{
126 const QString newName = mNameEdit->text().trimmed();
127 if ( mIsShapeFile && newName.compare( QLatin1String( "shape" ), Qt::CaseInsensitive ) == 0 )
128 {
129 QMessageBox::warning( this, tr( "Add Field" ), tr( "Invalid field name. This field name is reserved and cannot be used." ) );
130 return;
131 }
132
133
134 for ( const QString &illegalName : std::as_const( mIllegalFieldNames ) )
135 {
136 if ( newName.compare( illegalName, Qt::CaseInsensitive ) == 0 )
137 {
138 QMessageBox::warning( this, tr( "Add Field" ), tr( "%1 is an illegal field name for this format and cannot be used." ).arg( newName ) );
139 return;
140 }
141 }
142
143 if ( mNameEdit->text().isEmpty() )
144 {
145 QMessageBox::warning( this, tr( "Add Field" ), tr( "No name specified. Please specify a name to create a new field." ) );
146 return;
147 }
148
149 QDialog::accept();
150}
151
153{
154 QgsDebugMsgLevel( QStringLiteral( "idx:%1 name:%2 type:%3 typeName:%4 length:%5 prec:%6 comment:%7" ).arg( mTypeBox->currentIndex() ).arg( mNameEdit->text() ).arg( mTypeBox->currentData( Qt::UserRole ).toInt() ).arg( mTypeBox->currentData( Qt::UserRole + 1 ).toString() ).arg( mLength->value() ).arg( mPrec->value() ).arg( mCommentEdit->text() ), 2 );
155
156 QgsField res = QgsField(
157 mNameEdit->text(),
158 ( QMetaType::Type ) mTypeBox->currentData( Qt::UserRole ).toInt(),
159 mTypeBox->currentData( Qt::UserRole + 1 ).toString(),
160 mLength->value(),
161 mPrec->isEnabled() ? mPrec->value() : 0,
162 mCommentEdit->text(),
163 static_cast<QMetaType::Type>( mTypeBox->currentData( Qt::UserRole ).toInt() ) == QMetaType::Type::QVariantMap ? QMetaType::Type::QString : QMetaType::Type::UnknownType
164 );
165
166 if ( !mAliasEdit->text().isEmpty() )
167 res.setAlias( mAliasEdit->text() );
168
169 return res;
170}
QFlags< VectorDataProviderAttributeEditCapability > VectorDataProviderAttributeEditCapabilities
Attribute editing capabilities which may be supported by vector data providers.
Definition qgis.h:566
QgsField field() const
Returns a field for the configured attribute.
void setIllegalFieldNames(const QSet< QString > &names)
Sets a list of field names which are considered illegal and should not be accepted by the dialog.
QgsAddAttrDialog(QgsVectorLayer *vlayer, QWidget *parent=nullptr, Qt::WindowFlags fl=QgsGuiUtils::ModalDialogFlags)
constructor
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:53
void setAlias(const QString &alias)
Sets the alias for the field (the friendly displayed name of the field ).
Definition qgsfield.cpp:296
static QIcon iconForFieldType(QMetaType::Type type, QMetaType::Type subType=QMetaType::Type::UnknownType, const QString &typeString=QString())
Returns an icon corresponding to a field type.
QList< QgsVectorDataProvider::NativeType > nativeTypes() const
Returns the names of the supported types.
virtual Qgis::VectorDataProviderAttributeEditCapabilities attributeEditCapabilities() const
Returns the provider's supported attribute editing capabilities.
Represents a vector layer which manages a vector based data sets.
QgsVectorDataProvider * dataProvider() FINAL
Returns the layer's data provider, it may be nullptr.
QgsSignalBlocker< Object > whileBlocking(Object *object)
Temporarily blocks signals from a QObject while calling a single method from the object.
Definition qgis.h:5928
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:39