diff options
Diffstat (limited to 'kdeui/klistviewsearchline.cpp')
-rw-r--r-- | kdeui/klistviewsearchline.cpp | 501 |
1 files changed, 501 insertions, 0 deletions
diff --git a/kdeui/klistviewsearchline.cpp b/kdeui/klistviewsearchline.cpp new file mode 100644 index 000000000..e3f9271eb --- /dev/null +++ b/kdeui/klistviewsearchline.cpp @@ -0,0 +1,501 @@ +/* This file is part of the KDE libraries + Copyright (c) 2003 Scott Wheeler <wheeler@kde.org> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License version 2 as published by the Free Software Foundation. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + Boston, MA 02110-1301, USA. +*/ + +#include "klistviewsearchline.h" + +#include <klistview.h> +#include <kiconloader.h> +#include <ktoolbar.h> +#include <ktoolbarbutton.h> +#include <kdebug.h> +#include <klocale.h> + +#include <qapplication.h> +#include <qtimer.h> +#include <qpopupmenu.h> +#include <qlabel.h> +#include <qheader.h> + +#define KLISTVIEWSEARCHLINE_ALLVISIBLECOLUMNS_ID 2004 + +class KListViewSearchLine::KListViewSearchLinePrivate +{ +public: + KListViewSearchLinePrivate() : + listView(0), + caseSensitive(false), + activeSearch(false), + keepParentsVisible(true), + queuedSearches(0) {} + + KListView *listView; + bool caseSensitive; + bool activeSearch; + bool keepParentsVisible; + QString search; + int queuedSearches; + QValueList<int> searchColumns; +}; + +//////////////////////////////////////////////////////////////////////////////// +// public methods +//////////////////////////////////////////////////////////////////////////////// + +KListViewSearchLine::KListViewSearchLine(QWidget *parent, KListView *listView, const char *name) : + KLineEdit(parent, name) +{ + d = new KListViewSearchLinePrivate; + + d->listView = listView; + + connect(this, SIGNAL(textChanged(const QString &)), + this, SLOT(queueSearch(const QString &))); + + if(listView) { + connect(listView, SIGNAL(destroyed()), + this, SLOT(listViewDeleted())); + + connect(listView, SIGNAL(itemAdded(QListViewItem *)), + this, SLOT(itemAdded(QListViewItem *))); + } + else + setEnabled(false); +} + +KListViewSearchLine::KListViewSearchLine(QWidget *parent, const char *name) : + KLineEdit(parent, name) +{ + d = new KListViewSearchLinePrivate; + + d->listView = 0; + + connect(this, SIGNAL(textChanged(const QString &)), + this, SLOT(queueSearch(const QString &))); + + setEnabled(false); +} + +KListViewSearchLine::~KListViewSearchLine() +{ + delete d; +} + +bool KListViewSearchLine::caseSensitive() const +{ + return d->caseSensitive; +} + +QValueList<int> KListViewSearchLine::searchColumns() const +{ + return d->searchColumns; +} + +bool KListViewSearchLine::keepParentsVisible() const +{ + return d->keepParentsVisible; +} + +KListView *KListViewSearchLine::listView() const +{ + return d->listView; +} + +//////////////////////////////////////////////////////////////////////////////// +// public slots +//////////////////////////////////////////////////////////////////////////////// + +void KListViewSearchLine::updateSearch(const QString &s) +{ + if(!d->listView) + return; + + d->search = s.isNull() ? text() : s; + + // If there's a selected item that is visible, make sure that it's visible + // when the search changes too (assuming that it still matches). + + QListViewItem *currentItem = 0; + + switch(d->listView->selectionMode()) + { + case KListView::NoSelection: + break; + case KListView::Single: + currentItem = d->listView->selectedItem(); + break; + default: + { + int flags = QListViewItemIterator::Selected | QListViewItemIterator::Visible; + for(QListViewItemIterator it(d->listView, flags); + it.current() && !currentItem; + ++it) + { + if(d->listView->itemRect(it.current()).isValid()) + currentItem = it.current(); + } + } + } + + if(d->keepParentsVisible) + checkItemParentsVisible(d->listView->firstChild()); + else + checkItemParentsNotVisible(); + + if(currentItem) + d->listView->ensureItemVisible(currentItem); +} + +void KListViewSearchLine::setCaseSensitive(bool cs) +{ + d->caseSensitive = cs; +} + +void KListViewSearchLine::setKeepParentsVisible(bool v) +{ + d->keepParentsVisible = v; +} + +void KListViewSearchLine::setSearchColumns(const QValueList<int> &columns) +{ + d->searchColumns = columns; +} + +void KListViewSearchLine::setListView(KListView *lv) +{ + if(d->listView) { + disconnect(d->listView, SIGNAL(destroyed()), + this, SLOT(listViewDeleted())); + + disconnect(d->listView, SIGNAL(itemAdded(QListViewItem *)), + this, SLOT(itemAdded(QListViewItem *))); + } + + d->listView = lv; + + if(lv) { + connect(d->listView, SIGNAL(destroyed()), + this, SLOT(listViewDeleted())); + + connect(d->listView, SIGNAL(itemAdded(QListViewItem *)), + this, SLOT(itemAdded(QListViewItem *))); + } + + setEnabled(bool(lv)); +} + +//////////////////////////////////////////////////////////////////////////////// +// protected members +//////////////////////////////////////////////////////////////////////////////// + +bool KListViewSearchLine::itemMatches(const QListViewItem *item, const QString &s) const +{ + if(s.isEmpty()) + return true; + + // If the search column list is populated, search just the columns + // specifified. If it is empty default to searching all of the columns. + + if(!d->searchColumns.isEmpty()) { + QValueList<int>::ConstIterator it = d->searchColumns.begin(); + for(; it != d->searchColumns.end(); ++it) { + if(*it < item->listView()->columns() && + item->text(*it).find(s, 0, d->caseSensitive) >= 0) + return true; + } + } + else { + for(int i = 0; i < item->listView()->columns(); i++) { + if(item->listView()->columnWidth(i) > 0 && + item->text(i).find(s, 0, d->caseSensitive) >= 0) + { + return true; + } + } + } + + return false; +} + +QPopupMenu *KListViewSearchLine::createPopupMenu() +{ + QPopupMenu *popup = KLineEdit::createPopupMenu(); + + if (d->listView->columns()>1) { + QPopupMenu *subMenu = new QPopupMenu(popup); + connect(subMenu, SIGNAL(activated(int)), this, SLOT(searchColumnsMenuActivated(int))); + + popup->insertSeparator(); + popup->insertItem(i18n("Search Columns"), subMenu); + + subMenu->insertItem(i18n("All Visible Columns"), KLISTVIEWSEARCHLINE_ALLVISIBLECOLUMNS_ID); + subMenu->insertSeparator(); + + bool allColumnsAreSearchColumns = true; + // TODO Make the entry order match the actual column order + QHeader* const header = d->listView->header(); + int visibleColumns = 0; + for(int i = 0; i < d->listView->columns(); i++) { + if(d->listView->columnWidth(i)>0) { + QString columnText = d->listView->columnText(i); + if(columnText.isEmpty()) { + int visiblePosition=1; + for(int j = 0; j < header->mapToIndex(i); j++) + if(d->listView->columnWidth(header->mapToSection(j))>0) + visiblePosition++; + columnText = i18n("Column number %1","Column No. %1").arg(visiblePosition); + } + subMenu->insertItem(columnText, visibleColumns); + if(d->searchColumns.isEmpty() || d->searchColumns.find(i) != d->searchColumns.end()) + subMenu->setItemChecked(visibleColumns, true); + else + allColumnsAreSearchColumns = false; + visibleColumns++; + } + } + subMenu->setItemChecked(KLISTVIEWSEARCHLINE_ALLVISIBLECOLUMNS_ID, allColumnsAreSearchColumns); + + // searchColumnsMenuActivated() relies on one possible "all" representation + if(allColumnsAreSearchColumns && !d->searchColumns.isEmpty()) + d->searchColumns.clear(); + } + + return popup; +} + +//////////////////////////////////////////////////////////////////////////////// +// protected slots +//////////////////////////////////////////////////////////////////////////////// + +void KListViewSearchLine::queueSearch(const QString &search) +{ + d->queuedSearches++; + d->search = search; + QTimer::singleShot(200, this, SLOT(activateSearch())); +} + +void KListViewSearchLine::activateSearch() +{ + --(d->queuedSearches); + + if(d->queuedSearches == 0) + updateSearch(d->search); +} + +//////////////////////////////////////////////////////////////////////////////// +// private slots +//////////////////////////////////////////////////////////////////////////////// + +void KListViewSearchLine::itemAdded(QListViewItem *item) const +{ + item->setVisible(itemMatches(item, text())); +} + +void KListViewSearchLine::listViewDeleted() +{ + d->listView = 0; + setEnabled(false); +} + +void KListViewSearchLine::searchColumnsMenuActivated(int id) +{ + if(id == KLISTVIEWSEARCHLINE_ALLVISIBLECOLUMNS_ID) { + if(d->searchColumns.isEmpty()) + d->searchColumns.append(0); + else + d->searchColumns.clear(); + } + else { + if(d->searchColumns.find(id) != d->searchColumns.end()) + d->searchColumns.remove(id); + else { + if(d->searchColumns.isEmpty()) { + for(int i = 0; i < d->listView->columns(); i++) { + if(i != id) + d->searchColumns.append(i); + } + } + else + d->searchColumns.append(id); + } + } + updateSearch(); +} + +//////////////////////////////////////////////////////////////////////////////// +// private methods +//////////////////////////////////////////////////////////////////////////////// + +void KListViewSearchLine::checkItemParentsNotVisible() +{ + QListViewItemIterator it(d->listView); + for(; it.current(); ++it) + { + QListViewItem *item = it.current(); + if(itemMatches(item, d->search)) + item->setVisible(true); + else + item->setVisible(false); + } +} + +#include <kdebug.h> + +/** Check whether \p item, its siblings and their descendents should be shown. Show or hide the items as necessary. + * + * \p item The list view item to start showing / hiding items at. Typically, this is the first child of another item, or the + * the first child of the list view. + * \p highestHiddenParent The highest (closest to root) ancestor of \p item which is hidden. If 0, all parents of + * \p item must be visible. + * \return \c true if an item which should be visible is found, \c false if all items found should be hidden. If this function + * returns true and \p highestHiddenParent was not 0, highestHiddenParent will have been shown. + */ +bool KListViewSearchLine::checkItemParentsVisible(QListViewItem *item, QListViewItem *highestHiddenParent) +{ + bool visible = false; + QListViewItem * first = item; + for(; item; item = item->nextSibling()) + { + //What we pass to our children as highestHiddenParent: + QListViewItem * hhp = highestHiddenParent ? highestHiddenParent : item->isVisible() ? 0L : item; + bool childMatch = false; + if(item->firstChild() && checkItemParentsVisible(item->firstChild(), hhp)) + childMatch = true; + // Should this item be shown? It should if any children should be, or if it matches. + if(childMatch || itemMatches(item, d->search)) + { + visible = true; + if (highestHiddenParent) + { + highestHiddenParent->setVisible(true); + // Calling setVisible on our ancestor will unhide all its descendents. Hide the ones + // before us that should not be shown. + for(QListViewItem *hide = first; hide != item; hide = hide->nextSibling()) + hide->setVisible(false); + highestHiddenParent = 0; + // If we matched, than none of our children matched, yet the setVisible() call on our + // ancestor unhid them, undo the damage: + if(!childMatch) + for(QListViewItem *hide = item->firstChild(); hide; hide = hide->nextSibling()) + hide->setVisible(false); + } + else + item->setVisible(true); + } + else + item->setVisible(false); + } + return visible; +} + +//////////////////////////////////////////////////////////////////////////////// +// KListViewSearchLineWidget +//////////////////////////////////////////////////////////////////////////////// + +class KListViewSearchLineWidget::KListViewSearchLineWidgetPrivate +{ +public: + KListViewSearchLineWidgetPrivate() : listView(0), searchLine(0), clearButton(0) {} + KListView *listView; + KListViewSearchLine *searchLine; + QToolButton *clearButton; +}; + +KListViewSearchLineWidget::KListViewSearchLineWidget(KListView *listView, + QWidget *parent, + const char *name) : + QHBox(parent, name) +{ + d = new KListViewSearchLineWidgetPrivate; + d->listView = listView; + + setSpacing(5); + + QTimer::singleShot(0, this, SLOT(createWidgets())); +} + +KListViewSearchLineWidget::~KListViewSearchLineWidget() +{ + delete d; +} + +KListViewSearchLine *KListViewSearchLineWidget::createSearchLine(KListView *listView) +{ + if(!d->searchLine) + d->searchLine = new KListViewSearchLine(this, listView); + return d->searchLine; +} + +void KListViewSearchLineWidget::createWidgets() +{ + positionInToolBar(); + + if(!d->clearButton) { + d->clearButton = new QToolButton(this); + QIconSet icon = SmallIconSet(QApplication::reverseLayout() ? "clear_left" : "locationbar_erase"); + d->clearButton->setIconSet(icon); + } + + d->clearButton->show(); + + QLabel *label = new QLabel(i18n("S&earch:"), this, "kde toolbar widget"); + + d->searchLine = createSearchLine(d->listView); + d->searchLine->show(); + + label->setBuddy(d->searchLine); + label->show(); + + connect(d->clearButton, SIGNAL(clicked()), d->searchLine, SLOT(clear())); +} + +KListViewSearchLine *KListViewSearchLineWidget::searchLine() const +{ + return d->searchLine; +} + +void KListViewSearchLineWidget::positionInToolBar() +{ + KToolBar *toolBar = dynamic_cast<KToolBar *>(parent()); + + if(toolBar) { + + // Here we have The Big Ugly. Figure out how many widgets are in the + // and do a hack-ish iteration over them to find this widget so that we + // can insert the clear button before it. + + int widgetCount = toolBar->count(); + + for(int index = 0; index < widgetCount; index++) { + int id = toolBar->idAt(index); + if(toolBar->getWidget(id) == this) { + toolBar->setItemAutoSized(id); + if(!d->clearButton) { + QString icon = QApplication::reverseLayout() ? "clear_left" : "locationbar_erase"; + d->clearButton = new KToolBarButton(icon, 2005, toolBar); + } + toolBar->insertWidget(2005, d->clearButton->width(), d->clearButton, index); + break; + } + } + } + + if(d->searchLine) + d->searchLine->show(); +} + +#include "klistviewsearchline.moc" |