QGIS API Documentation 3.41.0-Master (3440c17df1d)
Loading...
Searching...
No Matches
qgsfilterlineedit.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsfilterlineedit.cpp
3 ------------------------
4 begin : October 27, 2012
5 copyright : (C) 2012 by Alexander Bruy
6 email : alexander dot bruy at gmail dot com
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 "qgsfilterlineedit.h"
19#include "moc_qgsfilterlineedit.cpp"
20#include "qgsapplication.h"
21#include "qgsanimatedicon.h"
22#include "qgis.h"
23
24#include <QAction>
25#include <QToolButton>
26#include <QStyle>
27#include <QFocusEvent>
28#include <QPainter>
29
30QgsFilterLineEdit::QgsFilterLineEdit( QWidget *parent, const QString &nullValue )
31 : QLineEdit( parent )
32 , mNullValue( nullValue )
33{
34 // icon size is about 2/3 height of text, but minimum size of 16
35 const int iconSize = std::floor( std::max( Qgis::UI_SCALE_FACTOR * fontMetrics().height() * 0.75, 16.0 ) );
36
37 mClearIcon.addPixmap( QgsApplication::getThemeIcon( "/mIconClearText.svg" ).pixmap( QSize( iconSize, iconSize ) ), QIcon::Normal, QIcon::On );
38 mClearIcon.addPixmap( QgsApplication::getThemeIcon( "/mIconClearTextHover.svg" ).pixmap( QSize( iconSize, iconSize ) ), QIcon::Selected, QIcon::On );
39
40 connect( this, &QLineEdit::textChanged, this,
41 &QgsFilterLineEdit::onTextChanged );
42}
43
45{
46 mClearButtonVisible = visible;
47 updateClearIcon();
48}
49
51{
52 if ( visible && !mSearchAction )
53 {
54 const QIcon searchIcon = QgsApplication::getThemeIcon( "/search.svg" );
55 mSearchAction = new QAction( searchIcon, QString(), this );
56 mSearchAction->setCheckable( false );
57 addAction( mSearchAction, QLineEdit::LeadingPosition );
58 }
59 else if ( !visible && mSearchAction )
60 {
61 mSearchAction->deleteLater();
62 mSearchAction = nullptr;
63 }
64}
65
66void QgsFilterLineEdit::setDefaultValue( const QString &defaultValue )
67{
68 if ( defaultValue == mDefaultValue )
69 return;
70
71 mDefaultValue = defaultValue;
72 updateClearIcon();
73}
74
75void QgsFilterLineEdit::updateClearIcon()
76{
77 const bool showClear = shouldShowClear();
78 if ( showClear && !mClearAction )
79 {
80 mClearAction = new QAction( mClearIcon, QString(), this );
81 mClearAction->setCheckable( false );
82 addAction( mClearAction, QLineEdit::TrailingPosition );
83 connect( mClearAction, &QAction::triggered, this, &QgsFilterLineEdit::clearValue );
84 }
85 else if ( !showClear && mClearAction )
86 {
87 // pretty freakin weird... seems the deleteLater call on the mClearAction
88 // isn't sufficient to actually remove the action from the line edit, and
89 // a kind of "ghost" action gets left behind... resulting in duplicate
90 // clear actions appearing if later we re-create the action.
91 // in summary: don't remove this "removeAction" call!
92 removeAction( mClearAction );
93 mClearAction->deleteLater();
94 mClearAction = nullptr;
95 }
96}
97
98void QgsFilterLineEdit::focusInEvent( QFocusEvent *e )
99{
100 QLineEdit::focusInEvent( e );
101 if ( e->reason() == Qt::MouseFocusReason && ( isNull() || mSelectOnFocus ) )
102 {
103 mWaitingForMouseRelease = true;
104 }
105}
106
108{
109 QLineEdit::mouseReleaseEvent( e );
110 if ( mWaitingForMouseRelease )
111 {
112 mWaitingForMouseRelease = false;
113 selectAll();
114 }
115}
116
118{
119 switch ( mClearMode )
120 {
121 case ClearToNull:
122 setText( mNullValue );
123 selectAll();
124 break;
125
126 case ClearToDefault:
127 setText( mDefaultValue );
128 break;
129 }
130
131 setModified( true );
132 emit cleared();
133}
134
135void QgsFilterLineEdit::onTextChanged( const QString &text )
136{
137 updateClearIcon();
138
139 if ( isNull() )
140 {
141 setStyleSheet( QStringLiteral( "QLineEdit { font: italic; color: gray; } %1" ).arg( mStyleSheet ) );
142 emit valueChanged( QString() );
143 }
144 else
145 {
146 setStyleSheet( mStyleSheet );
147 emit valueChanged( text );
148 }
149}
150
151void QgsFilterLineEdit::updateBusySpinner()
152{
153 if ( !mBusySpinnerAction )
154 {
155 mBusySpinnerAction = addAction( mBusySpinnerAnimatedIcon->icon(), QLineEdit::TrailingPosition );
156 }
157 mBusySpinnerAction->setIcon( mBusySpinnerAnimatedIcon->icon() );
158}
159
161{
162 return mSelectOnFocus;
163}
164
165void QgsFilterLineEdit::setSelectOnFocus( bool selectOnFocus )
166{
167 if ( mSelectOnFocus == selectOnFocus )
168 return;
169
170 mSelectOnFocus = selectOnFocus;
172}
173
175{
176 return mShowSpinner;
177}
178
179void QgsFilterLineEdit::setShowSpinner( bool showSpinner )
180{
181
182 if ( showSpinner == mShowSpinner )
183 return;
184
185 if ( showSpinner )
186 {
187 if ( !mBusySpinnerAnimatedIcon )
188 mBusySpinnerAnimatedIcon = new QgsAnimatedIcon( QgsApplication::iconPath( QStringLiteral( "/mIconLoading.gif" ) ), this );
189
190 mBusySpinnerAnimatedIcon->connectFrameChanged( this, &QgsFilterLineEdit::updateBusySpinner );
191 }
192 else
193 {
194 mBusySpinnerAnimatedIcon->disconnectFrameChanged( this, &QgsFilterLineEdit::updateBusySpinner );
195 removeAction( mBusySpinnerAction );
196 mBusySpinnerAction = nullptr;
197 }
198
199 mShowSpinner = showSpinner;
200 emit showSpinnerChanged();
201}
202
203bool QgsFilterLineEdit::shouldShowClear() const
204{
205 if ( !isEnabled() || isReadOnly() || !mClearButtonVisible )
206 return false;
207
208 switch ( mClearMode )
209 {
210 case ClearToNull:
211 return !isNull();
212
213 case ClearToDefault:
214 return value() != mDefaultValue;
215 }
216 return false; //avoid warnings
217}
218
219bool QgsFilterLineEdit::event( QEvent *event )
220{
221 if ( event->type() == QEvent::ReadOnlyChange || event->type() == QEvent::EnabledChange )
222 updateClearIcon();
223
224 return QLineEdit::event( event );
225}
226
228{
229 mLineEditState.text = text();
230 mLineEditState.selectionStart = selectionStart();
231 mLineEditState.selectionLength = selectedText().length();
232 mLineEditState.cursorPosition = cursorPosition();
233 mLineEditState.hasStateStored = true;
234}
235
237{
238 setText( mLineEditState.text );
239 setCursorPosition( mLineEditState.cursorPosition );
240 if ( mLineEditState.selectionStart > -1 )
241 setSelection( mLineEditState.selectionStart, mLineEditState.selectionLength );
242 mLineEditState.hasStateStored = false;
243}
244
246void QgsSpinBoxLineEdit::focusInEvent( QFocusEvent *e )
247{
249 if ( isNull() )
250 {
251 clear();
252 }
253}
static const double UI_SCALE_FACTOR
UI scaling factor.
Definition qgis.h:5667
Animated icon is keeping an animation running if there are listeners connected to frameChanged.
bool disconnectFrameChanged(const typename QtPrivate::FunctionPointer< Func1 >::Object *receiver, Func1 slot)
Convenience function to disconnect the same style that the frame change connection was established.
bool connectFrameChanged(const typename QtPrivate::FunctionPointer< Func1 >::Object *receiver, Func1 slot)
Connect a slot that will be notified repeatedly whenever a frame changes and which should request the...
QIcon icon() const
Gets the icons representation in the current frame.
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
static QString iconPath(const QString &iconFile)
Returns path to the desired icon file.
void focusInEvent(QFocusEvent *e) override
void restoreState()
Restores the current state of the line edit (selection and cursor position)
void storeState()
Stores the current state of the line edit (selection and cursor position)
void selectOnFocusChanged()
Will select all text when this widget receives the focus.
void showSpinnerChanged()
Show a spinner icon.
virtual void clearValue()
Clears the widget and resets it to the null value.
@ ClearToNull
Reset value to null.
@ ClearToDefault
Reset value to default value (see defaultValue() )
bool isNull() const
Determine if the current text represents null.
bool selectOnFocus() const
Will select all text when this widget receives the focus.
void setShowSearchIcon(bool visible)
Define if a search icon shall be shown on the left of the image when no text is entered.
void setShowClearButton(bool visible)
Sets whether the widget's clear button is visible.
QgsFilterLineEdit(QWidget *parent=nullptr, const QString &nullValue=QString())
Constructor for QgsFilterLineEdit.
void setSelectOnFocus(bool selectOnFocus)
Will select all text when this widget receives the focus.
void mouseReleaseEvent(QMouseEvent *e) override
void cleared()
Emitted when the widget is cleared.
void setDefaultValue(const QString &defaultValue)
Sets the default value for the widget.
void setShowSpinner(bool showSpinner)
Show a spinner icon.
bool event(QEvent *event) override
Reimplemented to enable/disable the clear action depending on read-only status.
void valueChanged(const QString &value)
Same as textChanged() but with support for null values.