You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
563 lines
16 KiB
563 lines
16 KiB
15 years ago
|
/* This file is part of the KDE libraries
|
||
|
|
||
|
Copyright (c) 2000,2001,2002 Carsten Pfeiffer <pfeiffer@kde.org>
|
||
|
Copyright (c) 2000 Stefan Schimanski <1Stein@gmx.de>
|
||
|
Copyright (c) 2000,2001,2002,2003,2004 Dawit Alemayehu <adawit@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 (LGPL) as published by the Free Software Foundation; either
|
||
|
version 2 of the License, or (at your option) any later version.
|
||
|
|
||
|
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 <qapplication.h>
|
||
|
#include <qcombobox.h>
|
||
|
#include <qevent.h>
|
||
|
#include <qstyle.h>
|
||
|
|
||
|
#include <kdebug.h>
|
||
|
#include <kconfig.h>
|
||
|
#include <knotifyclient.h>
|
||
|
#include <kglobalsettings.h>
|
||
|
|
||
|
#include "kcompletionbox.h"
|
||
|
|
||
|
class KCompletionBox::KCompletionBoxPrivate
|
||
|
{
|
||
|
public:
|
||
|
QWidget *m_parent; // necessary to set the focus back
|
||
|
QString cancelText;
|
||
|
bool tabHandling;
|
||
|
bool down_workaround;
|
||
|
bool upwardBox;
|
||
|
bool emitSelected;
|
||
|
};
|
||
|
|
||
|
KCompletionBox::KCompletionBox( QWidget *parent, const char *name )
|
||
|
:KListBox( parent, name, WType_Popup ), d(new KCompletionBoxPrivate)
|
||
|
{
|
||
|
|
||
|
d->m_parent = parent;
|
||
|
d->tabHandling = true;
|
||
|
d->down_workaround = false;
|
||
|
d->upwardBox = false;
|
||
|
d->emitSelected = true;
|
||
|
|
||
|
setColumnMode( 1 );
|
||
|
setLineWidth( 1 );
|
||
|
setFrameStyle( QFrame::Box | QFrame::Plain );
|
||
|
|
||
|
if ( parent )
|
||
|
setFocusProxy( parent );
|
||
|
else
|
||
|
setFocusPolicy( NoFocus );
|
||
|
|
||
|
setVScrollBarMode( Auto );
|
||
|
setHScrollBarMode( AlwaysOff );
|
||
|
|
||
|
connect( this, SIGNAL( doubleClicked( QListBoxItem * )),
|
||
|
SLOT( slotActivated( QListBoxItem * )) );
|
||
|
|
||
|
// grmbl, just QListBox workarounds :[ Thanks Volker.
|
||
|
connect( this, SIGNAL( currentChanged( QListBoxItem * )),
|
||
|
SLOT( slotCurrentChanged() ));
|
||
|
connect( this, SIGNAL( clicked( QListBoxItem * )),
|
||
|
SLOT( slotItemClicked( QListBoxItem * )) );
|
||
|
}
|
||
|
|
||
|
KCompletionBox::~KCompletionBox()
|
||
|
{
|
||
|
d->m_parent = 0L;
|
||
|
delete d;
|
||
|
}
|
||
|
|
||
|
QStringList KCompletionBox::items() const
|
||
|
{
|
||
|
QStringList list;
|
||
|
|
||
|
const QListBoxItem* currItem = firstItem();
|
||
|
|
||
|
while (currItem) {
|
||
|
list.append(currItem->text());
|
||
|
currItem = currItem->next();
|
||
|
}
|
||
|
|
||
|
return list;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::slotActivated( QListBoxItem *item )
|
||
|
{
|
||
|
if ( !item )
|
||
|
return;
|
||
|
|
||
|
hide();
|
||
|
emit activated( item->text() );
|
||
|
}
|
||
|
|
||
|
bool KCompletionBox::eventFilter( QObject *o, QEvent *e )
|
||
|
{
|
||
|
int type = e->type();
|
||
|
|
||
|
if ( o == d->m_parent ) {
|
||
|
if ( isVisible() ) {
|
||
|
if ( type == QEvent::KeyPress ) {
|
||
|
QKeyEvent *ev = static_cast<QKeyEvent *>( e );
|
||
|
switch ( ev->key() ) {
|
||
|
case Key_BackTab:
|
||
|
if ( d->tabHandling && (ev->state() == NoButton ||
|
||
|
(ev->state() & ShiftButton)) ) {
|
||
|
up();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
}
|
||
|
break;
|
||
|
case Key_Tab:
|
||
|
if ( d->tabHandling && (ev->state() == NoButton) ) {
|
||
|
down(); // Only on TAB!!
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
}
|
||
|
break;
|
||
|
case Key_Down:
|
||
|
down();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
case Key_Up:
|
||
|
// If there is no selected item and we've popped up above
|
||
|
// our parent, select the first item when they press up.
|
||
|
if ( selectedItem() ||
|
||
|
mapToGlobal( QPoint( 0, 0 ) ).y() >
|
||
|
d->m_parent->mapToGlobal( QPoint( 0, 0 ) ).y() )
|
||
|
up();
|
||
|
else
|
||
|
down();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
case Key_Prior:
|
||
|
pageUp();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
case Key_Next:
|
||
|
pageDown();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
case Key_Escape:
|
||
|
canceled();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
case Key_Enter:
|
||
|
case Key_Return:
|
||
|
if ( ev->state() & ShiftButton ) {
|
||
|
hide();
|
||
|
ev->accept(); // Consume the Enter event
|
||
|
return true;
|
||
|
}
|
||
|
break;
|
||
|
case Key_End:
|
||
|
if ( ev->state() & ControlButton )
|
||
|
{
|
||
|
end();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
}
|
||
|
case Key_Home:
|
||
|
if ( ev->state() & ControlButton )
|
||
|
{
|
||
|
home();
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
}
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else if ( type == QEvent::AccelOverride ) {
|
||
|
// Override any acceleartors that match
|
||
|
// the key sequences we use here...
|
||
|
QKeyEvent *ev = static_cast<QKeyEvent *>( e );
|
||
|
switch ( ev->key() ) {
|
||
|
case Key_Down:
|
||
|
case Key_Up:
|
||
|
case Key_Prior:
|
||
|
case Key_Next:
|
||
|
case Key_Escape:
|
||
|
case Key_Enter:
|
||
|
case Key_Return:
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
break;
|
||
|
case Key_Tab:
|
||
|
case Key_BackTab:
|
||
|
if ( ev->state() == NoButton ||
|
||
|
(ev->state() & ShiftButton))
|
||
|
{
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
}
|
||
|
break;
|
||
|
case Key_Home:
|
||
|
case Key_End:
|
||
|
if ( ev->state() & ControlButton )
|
||
|
{
|
||
|
ev->accept();
|
||
|
return true;
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// parent loses focus or gets a click -> we hide
|
||
|
else if ( type == QEvent::FocusOut || type == QEvent::Resize ||
|
||
|
type == QEvent::Close || type == QEvent::Hide ||
|
||
|
type == QEvent::Move ) {
|
||
|
hide();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// any mouse-click on something else than "this" makes us hide
|
||
|
else if ( type == QEvent::MouseButtonPress ) {
|
||
|
QMouseEvent *ev = static_cast<QMouseEvent *>( e );
|
||
|
if ( !rect().contains( ev->pos() )) // this widget
|
||
|
hide();
|
||
|
|
||
|
if ( !d->emitSelected && currentItem() && !::qt_cast<QScrollBar*>(o) )
|
||
|
{
|
||
|
emit highlighted( currentText() );
|
||
|
hide();
|
||
|
ev->accept(); // Consume the mouse click event...
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return KListBox::eventFilter( o, e );
|
||
|
}
|
||
|
|
||
|
|
||
|
void KCompletionBox::popup()
|
||
|
{
|
||
|
if ( count() == 0 )
|
||
|
hide();
|
||
|
else {
|
||
|
ensureCurrentVisible();
|
||
|
bool block = signalsBlocked();
|
||
|
blockSignals( true );
|
||
|
setCurrentItem( 0 );
|
||
|
blockSignals( block );
|
||
|
clearSelection();
|
||
|
if ( !isVisible() )
|
||
|
show();
|
||
|
else if ( size().height() != sizeHint().height() )
|
||
|
sizeAndPosition();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::sizeAndPosition()
|
||
|
{
|
||
|
int currentGeom = height();
|
||
|
QPoint currentPos = pos();
|
||
|
QRect geom = calculateGeometry();
|
||
|
resize( geom.size() );
|
||
|
|
||
|
int x = currentPos.x(), y = currentPos.y();
|
||
|
if ( d->m_parent ) {
|
||
|
if ( !isVisible() ) {
|
||
|
QRect screenSize = KGlobalSettings::desktopGeometry(d->m_parent);
|
||
|
|
||
|
QPoint orig = d->m_parent->mapToGlobal( QPoint(0, d->m_parent->height()) );
|
||
|
x = orig.x() + geom.x();
|
||
|
y = orig.y() + geom.y();
|
||
|
|
||
|
if ( x + width() > screenSize.right() )
|
||
|
x = screenSize.right() - width();
|
||
|
if (y + height() > screenSize.bottom() ) {
|
||
|
y = y - height() - d->m_parent->height();
|
||
|
d->upwardBox = true;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// Are we above our parent? If so we must keep bottom edge anchored.
|
||
|
if (d->upwardBox)
|
||
|
y += (currentGeom-height());
|
||
|
}
|
||
|
move( x, y);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::show()
|
||
|
{
|
||
|
d->upwardBox = false;
|
||
|
if ( d->m_parent ) {
|
||
|
sizeAndPosition();
|
||
|
qApp->installEventFilter( this );
|
||
|
}
|
||
|
|
||
|
// ### we shouldn't need to call this, but without this, the scrollbars
|
||
|
// are pretty b0rked.
|
||
|
//triggerUpdate( true );
|
||
|
|
||
|
// Workaround for I'm not sure whose bug - if this KCompletionBox' parent
|
||
|
// is in a layout, that layout will detect inserting new child (posted
|
||
|
// ChildInserted event), and will trigger relayout (post LayoutHint event).
|
||
|
// QWidget::show() sends also posted ChildInserted events for the parent,
|
||
|
// and later all LayoutHint events, which causes layout updating.
|
||
|
// The problem is, KCompletionBox::eventFilter() detects resizing
|
||
|
// of the parent, and calls hide() - and this hide() happen in the middle
|
||
|
// of show(), causing inconsistent state. I'll try to submit a Qt patch too.
|
||
|
qApp->sendPostedEvents();
|
||
|
KListBox::show();
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::hide()
|
||
|
{
|
||
|
if ( d->m_parent )
|
||
|
qApp->removeEventFilter( this );
|
||
|
d->cancelText = QString::null;
|
||
|
KListBox::hide();
|
||
|
}
|
||
|
|
||
|
QRect KCompletionBox::calculateGeometry() const
|
||
|
{
|
||
|
int x = 0, y = 0;
|
||
|
int ih = itemHeight();
|
||
|
int h = QMIN( 15 * ih, (int) count() * ih ) + 2*frameWidth();
|
||
|
|
||
|
int w = (d->m_parent) ? d->m_parent->width() : KListBox::minimumSizeHint().width();
|
||
|
w = QMAX( KListBox::minimumSizeHint().width(), w );
|
||
|
|
||
|
//If we're inside a combox, Qt by default makes the dropdown
|
||
|
// as wide as the combo, and gives the style a chance
|
||
|
// to adjust it. Do that here as well, for consistency
|
||
|
const QObject* combo;
|
||
|
if ( d->m_parent && (combo = d->m_parent->parent() ) &&
|
||
|
combo->inherits("QComboBox") )
|
||
|
{
|
||
|
const QComboBox* cb = static_cast<const QComboBox*>(combo);
|
||
|
|
||
|
//Expand to the combo width
|
||
|
w = QMAX( w, cb->width() );
|
||
|
|
||
|
QPoint parentCorner = d->m_parent->mapToGlobal(QPoint(0, 0));
|
||
|
QPoint comboCorner = cb->mapToGlobal(QPoint(0, 0));
|
||
|
|
||
|
//We need to adjust our horizontal position to also be WRT to the combo
|
||
|
x += comboCorner.x() - parentCorner.x();
|
||
|
|
||
|
//The same with vertical one
|
||
|
y += cb->height() - d->m_parent->height() +
|
||
|
comboCorner.y() - parentCorner.y();
|
||
|
|
||
|
//Ask the style to refine this a bit
|
||
|
QRect styleAdj = style().querySubControlMetrics(QStyle::CC_ComboBox,
|
||
|
cb, QStyle::SC_ComboBoxListBoxPopup,
|
||
|
QStyleOption(x, y, w, h));
|
||
|
//QCommonStyle returns QRect() by default, so this is what we get if the
|
||
|
//style doesn't implement this
|
||
|
if (!styleAdj.isNull())
|
||
|
return styleAdj;
|
||
|
|
||
|
}
|
||
|
return QRect(x, y, w, h);
|
||
|
}
|
||
|
|
||
|
QSize KCompletionBox::sizeHint() const
|
||
|
{
|
||
|
return calculateGeometry().size();
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::down()
|
||
|
{
|
||
|
int i = currentItem();
|
||
|
|
||
|
if ( i == 0 && d->down_workaround ) {
|
||
|
d->down_workaround = false;
|
||
|
setCurrentItem( 0 );
|
||
|
setSelected( 0, true );
|
||
|
emit highlighted( currentText() );
|
||
|
}
|
||
|
|
||
|
else if ( i < (int) count() - 1 )
|
||
|
setCurrentItem( i + 1 );
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::up()
|
||
|
{
|
||
|
if ( currentItem() > 0 )
|
||
|
setCurrentItem( currentItem() - 1 );
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::pageDown()
|
||
|
{
|
||
|
int i = currentItem() + numItemsVisible();
|
||
|
i = i > (int)count() - 1 ? (int)count() - 1 : i;
|
||
|
setCurrentItem( i );
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::pageUp()
|
||
|
{
|
||
|
int i = currentItem() - numItemsVisible();
|
||
|
i = i < 0 ? 0 : i;
|
||
|
setCurrentItem( i );
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::home()
|
||
|
{
|
||
|
setCurrentItem( 0 );
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::end()
|
||
|
{
|
||
|
setCurrentItem( count() -1 );
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::setTabHandling( bool enable )
|
||
|
{
|
||
|
d->tabHandling = enable;
|
||
|
}
|
||
|
|
||
|
bool KCompletionBox::isTabHandling() const
|
||
|
{
|
||
|
return d->tabHandling;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::setCancelledText( const QString& text )
|
||
|
{
|
||
|
d->cancelText = text;
|
||
|
}
|
||
|
|
||
|
QString KCompletionBox::cancelledText() const
|
||
|
{
|
||
|
return d->cancelText;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::canceled()
|
||
|
{
|
||
|
if ( !d->cancelText.isNull() )
|
||
|
emit userCancelled( d->cancelText );
|
||
|
if ( isVisible() )
|
||
|
hide();
|
||
|
}
|
||
|
|
||
|
class KCompletionBoxItem : public QListBoxItem
|
||
|
{
|
||
|
public:
|
||
|
//Returns true if dirty.
|
||
|
bool reuse( const QString& newText )
|
||
|
{
|
||
|
if ( text() == newText )
|
||
|
return false;
|
||
|
setText( newText );
|
||
|
return true;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
void KCompletionBox::insertItems( const QStringList& items, int index )
|
||
|
{
|
||
|
bool block = signalsBlocked();
|
||
|
blockSignals( true );
|
||
|
insertStringList( items, index );
|
||
|
blockSignals( block );
|
||
|
d->down_workaround = true;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::setItems( const QStringList& items )
|
||
|
{
|
||
|
bool block = signalsBlocked();
|
||
|
blockSignals( true );
|
||
|
|
||
|
QListBoxItem* item = firstItem();
|
||
|
if ( !item ) {
|
||
|
insertStringList( items );
|
||
|
}
|
||
|
else {
|
||
|
//Keep track of whether we need to change anything,
|
||
|
//so we can avoid a repaint for identical updates,
|
||
|
//to reduce flicker
|
||
|
bool dirty = false;
|
||
|
|
||
|
QStringList::ConstIterator it = items.constBegin();
|
||
|
const QStringList::ConstIterator itEnd = items.constEnd();
|
||
|
|
||
|
for ( ; it != itEnd; ++it) {
|
||
|
if ( item ) {
|
||
|
const bool changed = ((KCompletionBoxItem*)item)->reuse( *it );
|
||
|
dirty = dirty || changed;
|
||
|
item = item->next();
|
||
|
}
|
||
|
else {
|
||
|
dirty = true;
|
||
|
//Inserting an item is a way of making this dirty
|
||
|
insertItem( new QListBoxText( *it ) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//If there is an unused item, mark as dirty -> less items now
|
||
|
if ( item ) {
|
||
|
dirty = true;
|
||
|
}
|
||
|
|
||
|
QListBoxItem* tmp = item;
|
||
|
while ( (item = tmp ) ) {
|
||
|
tmp = item->next();
|
||
|
delete item;
|
||
|
}
|
||
|
|
||
|
if (dirty)
|
||
|
triggerUpdate( false );
|
||
|
}
|
||
|
|
||
|
if ( isVisible() && size().height() != sizeHint().height() )
|
||
|
sizeAndPosition();
|
||
|
|
||
|
blockSignals( block );
|
||
|
d->down_workaround = true;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::slotCurrentChanged()
|
||
|
{
|
||
|
d->down_workaround = false;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::slotItemClicked( QListBoxItem *item )
|
||
|
{
|
||
|
if ( item )
|
||
|
{
|
||
|
if ( d->down_workaround ) {
|
||
|
d->down_workaround = false;
|
||
|
emit highlighted( item->text() );
|
||
|
}
|
||
|
|
||
|
hide();
|
||
|
emit activated( item->text() );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::setActivateOnSelect(bool state)
|
||
|
{
|
||
|
d->emitSelected = state;
|
||
|
}
|
||
|
|
||
|
bool KCompletionBox::activateOnSelect() const
|
||
|
{
|
||
|
return d->emitSelected;
|
||
|
}
|
||
|
|
||
|
void KCompletionBox::virtual_hook( int id, void* data )
|
||
|
{ KListBox::virtual_hook( id, data ); }
|
||
|
|
||
|
#include "kcompletionbox.moc"
|