QGIS API Documentation 3.41.0-Master (af5edcb665c)
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, &QgsFilterLineEdit::onTextChanged );
41}
42
44{
45 mClearButtonVisible = visible;
46 updateClearIcon();
47}
48
50{
51 if ( visible && !mSearchAction )
52 {
53 const QIcon searchIcon = QgsApplication::getThemeIcon( "/search.svg" );
54 mSearchAction = new QAction( searchIcon, QString(), this );
55 mSearchAction->setCheckable( false );
56 addAction( mSearchAction, QLineEdit::LeadingPosition );
57 }
58 else if ( !visible && mSearchAction )
59 {
60 mSearchAction->deleteLater();
61 mSearchAction = nullptr;
62 }
63}
64
65void QgsFilterLineEdit::setDefaultValue( const QString &defaultValue )
66{
67 if ( defaultValue == mDefaultValue )
68 return;
69
70 mDefaultValue = defaultValue;
71 updateClearIcon();
72}
73
74void QgsFilterLineEdit::updateClearIcon()
75{
76 const bool showClear = shouldShowClear();
77 if ( showClear && !mClearAction )
78 {
79 mClearAction = new QAction( mClearIcon, QString(), this );
80 mClearAction->setCheckable( false );
81 addAction( mClearAction, QLineEdit::TrailingPosition );
82 connect( mClearAction, &QAction::triggered, this, &QgsFilterLineEdit::clearValue );
83 }
84 else if ( !showClear && mClearAction )
85 {
86 // pretty freakin weird... seems the deleteLater call on the mClearAction
87 // isn't sufficient to actually remove the action from the line edit, and
88 // a kind of "ghost" action gets left behind... resulting in duplicate
89 // clear actions appearing if later we re-create the action.
90 // in summary: don't remove this "removeAction" call!
91 removeAction( mClearAction );
92 mClearAction->deleteLater();
93 mClearAction = nullptr;
94 }
95}
96
97void QgsFilterLineEdit::focusInEvent( QFocusEvent *e )
98{
99 QLineEdit::focusInEvent( e );
100 if ( e->reason() == Qt::MouseFocusReason && ( isNull() || mSelectOnFocus ) )
101 {
102 mWaitingForMouseRelease = true;
103 }
104}
105
107{
108 QLineEdit::mouseReleaseEvent( e );
109 if ( mWaitingForMouseRelease )
110 {
111 mWaitingForMouseRelease = false;
112 selectAll();
113 }
114}
115
117{
118 switch ( mClearMode )
119 {
120 case ClearToNull:
121 setText( mNullValue );
122 selectAll();
123 break;
124
125 case ClearToDefault:
126 setText( mDefaultValue );
127 break;
128 }
129
130 setModified( true );
131 emit cleared();
132}
133
134void QgsFilterLineEdit::onTextChanged( const QString &text )
135{
136 updateClearIcon();
137
138 if ( isNull() )
139 {
140 setStyleSheet( QStringLiteral( "QLineEdit { font: italic; color: gray; } %1" ).arg( mStyleSheet ) );
141 emit valueChanged( QString() );
142 }
143 else
144 {
145 setStyleSheet( mStyleSheet );
146 emit valueChanged( text );
147 }
148}
149
150void QgsFilterLineEdit::updateBusySpinner()
151{
152 if ( !mBusySpinnerAction )
153 {
154 mBusySpinnerAction = addAction( mBusySpinnerAnimatedIcon->icon(), QLineEdit::TrailingPosition );
155 }
156 mBusySpinnerAction->setIcon( mBusySpinnerAnimatedIcon->icon() );
157}
158
160{
161 return mSelectOnFocus;
162}
163
164void QgsFilterLineEdit::setSelectOnFocus( bool selectOnFocus )
165{
166 if ( mSelectOnFocus == selectOnFocus )
167 return;
168
169 mSelectOnFocus = selectOnFocus;
171}
172
174{
175 return mShowSpinner;
176}
177
178void QgsFilterLineEdit::setShowSpinner( bool showSpinner )
179{
180 if ( showSpinner == mShowSpinner )
181 return;
182
183 if ( showSpinner )
184 {
185 if ( !mBusySpinnerAnimatedIcon )
186 mBusySpinnerAnimatedIcon = new QgsAnimatedIcon( QgsApplication::iconPath( QStringLiteral( "/mIconLoading.gif" ) ), this );
187
188 mBusySpinnerAnimatedIcon->connectFrameChanged( this, &QgsFilterLineEdit::updateBusySpinner );
189 }
190 else
191 {
192 mBusySpinnerAnimatedIcon->disconnectFrameChanged( this, &QgsFilterLineEdit::updateBusySpinner );
193 removeAction( mBusySpinnerAction );
194 mBusySpinnerAction = nullptr;
195 }
196
197 mShowSpinner = showSpinner;
198 emit showSpinnerChanged();
199}
200
201bool QgsFilterLineEdit::shouldShowClear() const
202{
203 if ( !isEnabled() || isReadOnly() || !mClearButtonVisible )
204 return false;
205
206 switch ( mClearMode )
207 {
208 case ClearToNull:
209 return !isNull();
210
211 case ClearToDefault:
212 return value() != mDefaultValue;
213 }
214 return false; //avoid warnings
215}
216
217bool QgsFilterLineEdit::event( QEvent *event )
218{
219 if ( event->type() == QEvent::ReadOnlyChange || event->type() == QEvent::EnabledChange )
220 updateClearIcon();
221
222 return QLineEdit::event( event );
223}
224
226{
227 mLineEditState.text = text();
228 mLineEditState.selectionStart = selectionStart();
229 mLineEditState.selectionLength = selectedText().length();
230 mLineEditState.cursorPosition = cursorPosition();
231 mLineEditState.hasStateStored = true;
232}
233
235{
236 setText( mLineEditState.text );
237 setCursorPosition( mLineEditState.cursorPosition );
238 if ( mLineEditState.selectionStart > -1 )
239 setSelection( mLineEditState.selectionStart, mLineEditState.selectionLength );
240 mLineEditState.hasStateStored = false;
241}
242
244void QgsSpinBoxLineEdit::focusInEvent( QFocusEvent *e )
245{
247 if ( isNull() )
248 {
249 clear();
250 }
251}
static const double UI_SCALE_FACTOR
UI scaling factor.
Definition qgis.h:5775
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.