#include "qcompletionedit.h" #include #include #include #include QCompletionEdit::QCompletionEdit( QWidget *parent, const char *name ) : QLineEdit( parent, name ), aAdd( false ), caseSensitive( false ) { popup = new QVBox( 0, 0, WType_Popup ); popup->setFrameStyle( QFrame::Box | QFrame::Plain ); popup->setLineWidth( 1 ); popup->hide(); listbox = new QListBox( popup ); listbox->setFrameStyle( QFrame::NoFrame ); listbox->setLineWidth( 1 ); listbox->installEventFilter( this ); listbox->setHScrollBarMode( QScrollView::AlwaysOn ); listbox->setVScrollBarMode( QScrollView::AlwaysOn ); listbox->setCornerWidget( new QSizeGrip( listbox, "completion sizegrip" ) ); connect( this, SIGNAL( textChanged( const QString & ) ), this, SLOT( textDidChange( const QString & ) ) ); popup->setFocusProxy( listbox ); installEventFilter( this ); } bool QCompletionEdit::autoAdd() const { return aAdd; } QStringList QCompletionEdit::completionList() const { return compList; } void QCompletionEdit::setCompletionList( const QStringList &l ) { compList = l; } void QCompletionEdit::setAutoAdd( bool add ) { aAdd = add; } void QCompletionEdit::textDidChange( const QString &text ) { if ( text.isEmpty() ) { popup->close(); return; } updateListBox(); placeListBox(); } void QCompletionEdit::placeListBox() { if ( listbox->count() == 0 ) { popup->close(); return; } popup->resize( QMAX( listbox->sizeHint().width() + listbox->verticalScrollBar()->width() + 4, width() ), listbox->sizeHint().height() + listbox->horizontalScrollBar()->height() + 4 ); QPoint p( mapToGlobal( QPoint( 0, 0 ) ) ); if ( p.y() + height() + popup->height() <= QApplication::desktop()->height() ) popup->move( p.x(), p.y() + height() ); else popup->move( p.x(), p.y() - listbox->height() ); popup->show(); listbox->setCurrentItem( 0 ); listbox->setSelected( 0, true ); setFocus(); } void QCompletionEdit::updateListBox() { listbox->clear(); if ( compList.isEmpty() ) return; for ( QStringList::Iterator it = compList.begin(); it != compList.end(); ++it ) { if ( caseSensitive && (*it).left( text().length() ) == text() || !caseSensitive && (*it).left( text().length() ).lower() == text().lower() ) listbox->insertItem( *it ); } } bool QCompletionEdit::eventFilter( QObject *o, QEvent *e ) { if ( o == popup || o == listbox || o == listbox->viewport() ) { if ( e->type() == QEvent::KeyPress ) { QKeyEvent *ke = (QKeyEvent*)e; if ( ke->key() == Key_Enter || ke->key() == Key_Return || ke->key() == Key_Tab ) { if ( ke->key() == Key_Tab && listbox->count() > 1 && listbox->currentItem() < (int)listbox->count() - 1 ) { listbox->setCurrentItem( listbox->currentItem() + 1 ); return true; } popup->close(); setFocus(); blockSignals( true ); setText( listbox->currentText() ); blockSignals( false ); emit chosen( text() ); return true; } else if ( ke->key() == Key_Left || ke->key() == Key_Right || ke->key() == Key_Up || ke->key() == Key_Down || ke->key() == Key_Home || ke->key() == Key_End || ke->key() == Key_Prior || ke->key() == Key_Next ) { return false; } else if ( ke->key() == Key_Escape ) { popup->close(); setFocus(); } else if ( ke->key() != Key_Shift && ke->key() != Key_Control && ke->key() != Key_Alt ) { updateListBox(); if ( listbox->count() == 0 || text().length() == 0 ) { popup->close(); setFocus(); } QApplication::sendEvent( this, e ); return true; } } else if ( e->type() == QEvent::MouseButtonDblClick ) { popup->close(); setFocus(); blockSignals( true ); setText( listbox->currentText() ); blockSignals( false ); emit chosen( text() ); return true; } } else if ( o == this ) { if ( e->type() == QEvent::KeyPress ) { QKeyEvent *ke = (QKeyEvent*)e; if ( ke->key() == Key_Up || ke->key() == Key_Down || ke->key() == Key_Prior || ke->key() == Key_Next || ke->key() == Key_Return || ke->key() == Key_Enter || ke->key() == Key_Tab || ke->key() == Key_Escape ) { QApplication::sendEvent( listbox, e ); return true; } } } return QLineEdit::eventFilter( o, e ); } void QCompletionEdit::addCompletionEntry( const QString &entry ) { if ( compList.find( entry ) == compList.end() ) { compList << entry; compList.sort(); } } void QCompletionEdit::removeCompletionEntry( const QString &entry ) { QStringList::Iterator it = compList.find( entry ); if ( it != compList.end() ) compList.remove( it ); } void QCompletionEdit::setCaseSensitive( bool b ) { caseSensitive = b; } bool QCompletionEdit::isCaseSensitive() const { return caseSensitive; } void QCompletionEdit::clear() { QLineEdit::clear(); compList.clear(); } #include "qcompletionedit.moc"