File indexing completed on 2024-09-08 09:40:13
0001 /* 0002 This file is part of the KDE libraries 0003 SPDX-FileCopyrightText: 2003 Scott Wheeler <wheeler@kde.org> 0004 SPDX-FileCopyrightText: 2004 Gustavo Sverzut Barbieri <gsbarbieri@users.sourceforge.net> 0005 0006 SPDX-License-Identifier: LGPL-2.0-or-later 0007 */ 0008 0009 #include "klistwidgetsearchline.h" 0010 0011 #include <QApplication> 0012 #include <QEvent> 0013 #include <QKeyEvent> 0014 #include <QListWidget> 0015 #include <QTimer> 0016 0017 class KListWidgetSearchLinePrivate 0018 { 0019 public: 0020 KListWidgetSearchLinePrivate(KListWidgetSearchLine *parent) 0021 : q(parent) 0022 { 0023 } 0024 0025 void _k_listWidgetDeleted(); 0026 void _k_queueSearch(const QString &); 0027 void _k_activateSearch(); 0028 void _k_rowsInserted(const QModelIndex &, int, int); 0029 void _k_dataChanged(const QModelIndex &, const QModelIndex &); 0030 0031 void init(QListWidget *listWidget = nullptr); 0032 void updateHiddenState(int start, int end); 0033 0034 KListWidgetSearchLine *const q; 0035 QListWidget *listWidget = nullptr; 0036 Qt::CaseSensitivity caseSensitivity = Qt::CaseInsensitive; 0037 bool activeSearch = false; 0038 QString search; 0039 int queuedSearches = 0; 0040 }; 0041 0042 /****************************************************************************** 0043 * Public Methods * 0044 *****************************************************************************/ 0045 KListWidgetSearchLine::KListWidgetSearchLine(QWidget *parent, QListWidget *listWidget) 0046 : QLineEdit(parent) 0047 , d(new KListWidgetSearchLinePrivate(this)) 0048 0049 { 0050 d->init(listWidget); 0051 } 0052 0053 KListWidgetSearchLine::~KListWidgetSearchLine() 0054 { 0055 clear(); // returning items back to listWidget 0056 } 0057 0058 Qt::CaseSensitivity KListWidgetSearchLine::caseSensitive() const 0059 { 0060 return d->caseSensitivity; 0061 } 0062 0063 QListWidget *KListWidgetSearchLine::listWidget() const 0064 { 0065 return d->listWidget; 0066 } 0067 0068 /****************************************************************************** 0069 * Public Slots * 0070 *****************************************************************************/ 0071 void KListWidgetSearchLine::updateSearch(const QString &s) 0072 { 0073 d->search = s.isNull() ? text() : s; 0074 if (d->listWidget) { 0075 d->updateHiddenState(0, d->listWidget->count() - 1); 0076 } 0077 } 0078 0079 void KListWidgetSearchLine::clear() 0080 { 0081 // Show items back to QListWidget 0082 if (d->listWidget != nullptr) { 0083 for (int i = 0; i < d->listWidget->count(); ++i) { 0084 d->listWidget->item(i)->setHidden(false); 0085 } 0086 } 0087 0088 d->search = QString(); 0089 d->queuedSearches = 0; 0090 QLineEdit::clear(); 0091 } 0092 0093 void KListWidgetSearchLine::setCaseSensitivity(Qt::CaseSensitivity cs) 0094 { 0095 d->caseSensitivity = cs; 0096 } 0097 0098 void KListWidgetSearchLine::setListWidget(QListWidget *lw) 0099 { 0100 if (d->listWidget != nullptr) { 0101 disconnect(d->listWidget, SIGNAL(destroyed()), this, SLOT(_k_listWidgetDeleted())); 0102 d->listWidget->model()->disconnect(this); 0103 } 0104 0105 d->listWidget = lw; 0106 0107 if (lw != nullptr) { 0108 // clang-format off 0109 connect(d->listWidget, SIGNAL(destroyed()), this, SLOT(_k_listWidgetDeleted())); 0110 connect(d->listWidget->model(), SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(_k_rowsInserted(QModelIndex,int,int))); 0111 connect(d->listWidget->model(), SIGNAL(dataChanged(QModelIndex,QModelIndex)), this, SLOT(_k_dataChanged(QModelIndex,QModelIndex))); 0112 // clang-format on 0113 setEnabled(true); 0114 } else { 0115 setEnabled(false); 0116 } 0117 } 0118 0119 /****************************************************************************** 0120 * Protected Methods * 0121 *****************************************************************************/ 0122 bool KListWidgetSearchLine::itemMatches(const QListWidgetItem *item, const QString &s) const 0123 { 0124 if (s.isEmpty()) { 0125 return true; 0126 } 0127 0128 if (item == nullptr) { 0129 return false; 0130 } 0131 0132 return (item->text().indexOf(s, 0, caseSensitive() ? Qt::CaseSensitive : Qt::CaseInsensitive) >= 0); 0133 } 0134 0135 void KListWidgetSearchLinePrivate::init(QListWidget *_listWidget) 0136 { 0137 listWidget = _listWidget; 0138 0139 QObject::connect(q, SIGNAL(textChanged(QString)), q, SLOT(_k_queueSearch(QString))); 0140 0141 if (listWidget != nullptr) { 0142 // clang-format off 0143 QObject::connect(listWidget, SIGNAL(destroyed()), q, SLOT(_k_listWidgetDeleted())); 0144 QObject::connect(listWidget->model(), SIGNAL(rowsInserted(QModelIndex,int,int)), q, SLOT(_k_rowsInserted(QModelIndex,int,int))); 0145 QObject::connect(listWidget->model(), SIGNAL(dataChanged(QModelIndex,QModelIndex)), q, SLOT(_k_dataChanged(QModelIndex,QModelIndex))); 0146 // clang-format on 0147 q->setEnabled(true); 0148 } else { 0149 q->setEnabled(false); 0150 } 0151 q->setClearButtonEnabled(true); 0152 } 0153 0154 void KListWidgetSearchLinePrivate::updateHiddenState(int start, int end) 0155 { 0156 if (!listWidget) { 0157 return; 0158 } 0159 0160 QListWidgetItem *currentItem = listWidget->currentItem(); 0161 0162 // Remove Non-Matching items 0163 for (int index = start; index <= end; ++index) { 0164 QListWidgetItem *item = listWidget->item(index); 0165 if (!q->itemMatches(item, search)) { 0166 item->setHidden(true); 0167 0168 if (item == currentItem) { 0169 currentItem = nullptr; // It's not in listWidget anymore. 0170 } 0171 } else if (item->isHidden()) { 0172 item->setHidden(false); 0173 } 0174 } 0175 0176 if (listWidget->isSortingEnabled()) { 0177 listWidget->sortItems(); 0178 } 0179 0180 if (currentItem != nullptr) { 0181 listWidget->scrollToItem(currentItem); 0182 } 0183 } 0184 0185 bool KListWidgetSearchLine::event(QEvent *event) 0186 { 0187 if (event->type() == QEvent::KeyPress) { 0188 QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event); 0189 if (keyEvent->matches(QKeySequence::MoveToNextLine) || keyEvent->matches(QKeySequence::SelectNextLine) 0190 || keyEvent->matches(QKeySequence::MoveToPreviousLine) || keyEvent->matches(QKeySequence::SelectPreviousLine) 0191 || keyEvent->matches(QKeySequence::MoveToNextPage) || keyEvent->matches(QKeySequence::SelectNextPage) 0192 || keyEvent->matches(QKeySequence::MoveToPreviousPage) || keyEvent->matches(QKeySequence::SelectPreviousPage)) { 0193 if (d->listWidget) { 0194 QApplication::sendEvent(d->listWidget, event); 0195 return true; 0196 } 0197 } else if (keyEvent->key() == Qt::Key_Enter || keyEvent->key() == Qt::Key_Return) { 0198 if (d->listWidget) { 0199 QApplication::sendEvent(d->listWidget, event); 0200 return true; 0201 } 0202 } 0203 } 0204 return QLineEdit::event(event); 0205 } 0206 /****************************************************************************** 0207 * Protected Slots * 0208 *****************************************************************************/ 0209 void KListWidgetSearchLinePrivate::_k_queueSearch(const QString &s) 0210 { 0211 queuedSearches++; 0212 search = s; 0213 QTimer::singleShot(200, q, SLOT(_k_activateSearch())); 0214 } 0215 0216 void KListWidgetSearchLinePrivate::_k_activateSearch() 0217 { 0218 queuedSearches--; 0219 0220 if (queuedSearches <= 0) { 0221 q->updateSearch(search); 0222 queuedSearches = 0; 0223 } 0224 } 0225 0226 /****************************************************************************** 0227 * KListWidgetSearchLinePrivate Slots * 0228 *****************************************************************************/ 0229 void KListWidgetSearchLinePrivate::_k_listWidgetDeleted() 0230 { 0231 listWidget = nullptr; 0232 q->setEnabled(false); 0233 } 0234 0235 void KListWidgetSearchLinePrivate::_k_rowsInserted(const QModelIndex &parent, int start, int end) 0236 { 0237 if (parent.isValid()) { 0238 return; 0239 } 0240 0241 updateHiddenState(start, end); 0242 } 0243 0244 void KListWidgetSearchLinePrivate::_k_dataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight) 0245 { 0246 if (topLeft.parent().isValid()) { 0247 return; 0248 } 0249 0250 updateHiddenState(topLeft.row(), bottomRight.row()); 0251 } 0252 0253 #include "moc_klistwidgetsearchline.cpp"