summaryrefslogtreecommitdiffstats
path: root/src/widgets/qbuttongroup.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/widgets/qbuttongroup.cpp')
-rw-r--r--src/widgets/qbuttongroup.cpp682
1 files changed, 682 insertions, 0 deletions
diff --git a/src/widgets/qbuttongroup.cpp b/src/widgets/qbuttongroup.cpp
new file mode 100644
index 0000000..ebcc580
--- /dev/null
+++ b/src/widgets/qbuttongroup.cpp
@@ -0,0 +1,682 @@
+/****************************************************************************
+**
+** Implementation of QButtonGroup class
+**
+** Created : 950130
+**
+** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved.
+**
+** This file is part of the widgets module of the Qt GUI Toolkit.
+**
+** This file may be used under the terms of the GNU General
+** Public License versions 2.0 or 3.0 as published by the Free
+** Software Foundation and appearing in the files LICENSE.GPL2
+** and LICENSE.GPL3 included in the packaging of this file.
+** Alternatively you may (at your option) use any later version
+** of the GNU General Public License if such license has been
+** publicly approved by Trolltech ASA (or its successors, if any)
+** and the KDE Free Qt Foundation.
+**
+** Please review the following information to ensure GNU General
+** Public Licensing requirements will be met:
+** http://trolltech.com/products/qt/licenses/licensing/opensource/.
+** If you are unsure which license is appropriate for your use, please
+** review the following information:
+** http://trolltech.com/products/qt/licenses/licensing/licensingoverview
+** or contact the sales department at [email protected].
+**
+** This file may be used under the terms of the Q Public License as
+** defined by Trolltech ASA and appearing in the file LICENSE.QPL
+** included in the packaging of this file. Licensees holding valid Qt
+** Commercial licenses may use this file in accordance with the Qt
+** Commercial License Agreement provided with the Software.
+**
+** This file is provided "AS IS" with NO WARRANTY OF ANY KIND,
+** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR
+** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted
+** herein.
+**
+**********************************************************************/
+
+#include "qbuttongroup.h"
+#ifndef QT_NO_BUTTONGROUP
+#include "qbutton.h"
+#include "qptrlist.h"
+#include "qapplication.h"
+#include "qradiobutton.h"
+
+
+
+/*!
+ \class QButtonGroup qbuttongroup.h
+ \brief The QButtonGroup widget organizes QButton widgets in a group.
+
+ \ingroup organizers
+ \ingroup geomanagement
+ \ingroup appearance
+ \mainclass
+
+ A button group widget makes it easier to deal with groups of
+ buttons. Each button in a button group has a unique identifier.
+ The button group emits a clicked() signal with this identifier
+ when a button in the group is clicked. This makes a button group
+ particularly useful when you have several similar buttons and want
+ to connect all their clicked() signals to a single slot.
+
+ An \link setExclusive() exclusive\endlink button group switches
+ off all toggle buttons except the one that was clicked. A button
+ group is, by default, non-exclusive. Note that all radio buttons
+ that are inserted into a button group are mutually exclusive even
+ if the button group is non-exclusive. (See
+ setRadioButtonExclusive().)
+
+ There are two ways of using a button group:
+ \list
+ \i The button group is the parent widget of a number of buttons,
+ i.e. the button group is the parent argument in the button
+ constructor. The buttons are assigned identifiers 0, 1, 2, etc.,
+ in the order they are created. A QButtonGroup can display a frame
+ and a title because it inherits QGroupBox.
+ \i The button group is an invisible widget and the contained
+ buttons have some other parent widget. In this usage, each button
+ must be manually inserted, using insert(), into the button group
+ and given an identifier.
+ \endlist
+
+ A button can be removed from the group with remove(). A pointer to
+ a button with a given id can be obtained using find(). The id of a
+ button is available using id(). A button can be set \e on with
+ setButton(). The number of buttons in the group is returned by
+ count().
+
+ <img src=qbttngrp-m.png> <img src=qbttngrp-w.png>
+
+ \sa QPushButton, QCheckBox, QRadioButton
+*/
+
+/*!
+ \property QButtonGroup::exclusive
+ \brief whether the button group is exclusive
+
+ If this property is TRUE, then the buttons in the group are
+ toggled, and to untoggle a button you must click on another button
+ in the group. The default value is FALSE.
+*/
+
+/*!
+ \property QButtonGroup::radioButtonExclusive
+ \brief whether the radio buttons in the group are exclusive
+
+ If this property is TRUE (the default), the \link QRadioButton
+ radiobuttons\endlink in the group are treated exclusively.
+*/
+
+struct QButtonItem
+{
+ QButton *button;
+ int id;
+};
+
+
+class QButtonList: public QPtrList<QButtonItem>
+{
+public:
+ QButtonList() {}
+ ~QButtonList() {}
+};
+
+
+typedef QPtrListIterator<QButtonItem> QButtonListIt;
+
+
+/*!
+ Constructs a button group with no title.
+
+ The \a parent and \a name arguments are passed to the QWidget
+ constructor.
+*/
+
+QButtonGroup::QButtonGroup( QWidget *parent, const char *name )
+ : QGroupBox( parent, name )
+{
+ init();
+}
+
+/*!
+ Constructs a button group with the title \a title.
+
+ The \a parent and \a name arguments are passed to the QWidget
+ constructor.
+*/
+
+QButtonGroup::QButtonGroup( const QString &title, QWidget *parent,
+ const char *name )
+ : QGroupBox( title, parent, name )
+{
+ init();
+}
+
+/*!
+ Constructs a button group with no title. Child widgets will be
+ arranged in \a strips rows or columns (depending on \a
+ orientation).
+
+ The \a parent and \a name arguments are passed to the QWidget
+ constructor.
+*/
+
+QButtonGroup::QButtonGroup( int strips, Orientation orientation,
+ QWidget *parent, const char *name )
+ : QGroupBox( strips, orientation, parent, name )
+{
+ init();
+}
+
+/*!
+ Constructs a button group with title \a title. Child widgets will
+ be arranged in \a strips rows or columns (depending on \a
+ orientation).
+
+ The \a parent and \a name arguments are passed to the QWidget
+ constructor.
+*/
+
+QButtonGroup::QButtonGroup( int strips, Orientation orientation,
+ const QString &title, QWidget *parent,
+ const char *name )
+ : QGroupBox( strips, orientation, title, parent, name )
+{
+ init();
+}
+
+/*!
+ Initializes the button group.
+*/
+
+void QButtonGroup::init()
+{
+ buttons = new QButtonList;
+ Q_CHECK_PTR( buttons );
+ buttons->setAutoDelete( TRUE );
+ excl_grp = FALSE;
+ radio_excl = TRUE;
+}
+
+/*! \reimp */
+
+QButtonGroup::~QButtonGroup()
+{
+ QButtonList * tmp = buttons;
+ QButtonItem *bi = tmp->first();
+ buttons = 0;
+ while( bi ) {
+ bi->button->setGroup(0);
+ bi = tmp->next();
+ }
+ delete tmp;
+}
+
+bool QButtonGroup::isExclusive() const
+{
+ return excl_grp;
+}
+
+void QButtonGroup::setExclusive( bool enable )
+{
+ excl_grp = enable;
+}
+
+
+/*!
+ Inserts the \a button with the identifier \a id into the button
+ group. Returns the button identifier.
+
+ Buttons are normally inserted into a button group automatically by
+ passing the button group as the parent when the button is
+ constructed. So it is not necessary to manually insert buttons
+ that have this button group as their parent widget. An exception
+ is when you want custom identifiers instead of the default 0, 1,
+ 2, etc., or if you want the buttons to have some other parent.
+
+ The button is assigned the identifier \a id or an automatically
+ generated identifier. It works as follows: If \a id >= 0, this
+ identifier is assigned. If \a id == -1 (default), the identifier
+ is equal to the number of buttons in the group. If \a id is any
+ other negative integer, for instance -2, a unique identifier
+ (negative integer \<= -2) is generated. No button has an id of -1.
+
+ \sa find(), remove(), setExclusive()
+*/
+
+int QButtonGroup::insert( QButton *button, int id )
+{
+ if ( button->group() )
+ button->group()->remove( button );
+
+ static int seq_no = -2;
+ QButtonItem *bi = new QButtonItem;
+ Q_CHECK_PTR( bi );
+
+ if ( id < -1 )
+ bi->id = seq_no--;
+ else if ( id == -1 )
+ bi->id = buttons->count();
+ else
+ bi->id = id;
+
+ bi->button = button;
+ button->setGroup(this);
+ buttons->append( bi );
+
+ connect( button, SIGNAL(pressed()) , SLOT(buttonPressed()) );
+ connect( button, SIGNAL(released()), SLOT(buttonReleased()) );
+ connect( button, SIGNAL(clicked()) , SLOT(buttonClicked()) );
+ connect( button, SIGNAL(toggled(bool)) , SLOT(buttonToggled(bool)) );
+
+ if ( button->isToggleButton() && !button->isOn() &&
+ selected() && (selected()->focusPolicy() & TabFocus) != 0 )
+ button->setFocusPolicy( (FocusPolicy)(button->focusPolicy() &
+ ~TabFocus) );
+
+ return bi->id;
+}
+
+/*!
+ Returns the number of buttons in the group.
+*/
+int QButtonGroup::count() const
+{
+ return buttons->count();
+}
+
+/*!
+ Removes the \a button from the button group.
+
+ \sa insert()
+*/
+
+void QButtonGroup::remove( QButton *button )
+{
+ if ( !buttons )
+ return;
+
+ QButtonListIt it( *buttons );
+ QButtonItem *i;
+ while ( (i=it.current()) != 0 ) {
+ ++it;
+ if ( i->button == button ) {
+ buttons->remove( i );
+ button->setGroup(0);
+ button->disconnect( this );
+ return;
+ }
+ }
+}
+
+
+/*!
+ Returns the button with the specified identifier \a id, or 0 if
+ the button was not found.
+*/
+
+QButton *QButtonGroup::find( int id ) const
+{
+ // introduce a QButtonListIt if calling anything
+ for ( QButtonItem *i=buttons->first(); i; i=buttons->next() )
+ if ( i->id == id )
+ return i->button;
+ return 0;
+}
+
+
+/*!
+ \fn void QButtonGroup::pressed( int id )
+
+ This signal is emitted when a button in the group is \link
+ QButton::pressed() pressed\endlink. The \a id argument is the
+ button's identifier.
+
+ \sa insert()
+*/
+
+/*!
+ \fn void QButtonGroup::released( int id )
+
+ This signal is emitted when a button in the group is \link
+ QButton::released() released\endlink. The \a id argument is the
+ button's identifier.
+
+ \sa insert()
+*/
+
+/*!
+ \fn void QButtonGroup::clicked( int id )
+
+ This signal is emitted when a button in the group is \link
+ QButton::clicked() clicked\endlink. The \a id argument is the
+ button's identifier.
+
+ \sa insert()
+*/
+
+
+/*!
+ \internal
+ This slot is activated when one of the buttons in the group emits the
+ QButton::pressed() signal.
+*/
+
+void QButtonGroup::buttonPressed()
+{
+ // introduce a QButtonListIt if calling anything
+ int id = -1;
+ QButton *bt = (QButton *)sender(); // object that sent the signal
+ for ( register QButtonItem *i=buttons->first(); i; i=buttons->next() )
+ if ( bt == i->button ) { // button was clicked
+ id = i->id;
+ break;
+ }
+ if ( id != -1 )
+ emit pressed( id );
+}
+
+/*!
+ \internal
+ This slot is activated when one of the buttons in the group emits the
+ QButton::released() signal.
+*/
+
+void QButtonGroup::buttonReleased()
+{
+ // introduce a QButtonListIt if calling anything
+ int id = -1;
+ QButton *bt = (QButton *)sender(); // object that sent the signal
+ for ( register QButtonItem *i=buttons->first(); i; i=buttons->next() )
+ if ( bt == i->button ) { // button was clicked
+ id = i->id;
+ break;
+ }
+ if ( id != -1 )
+ emit released( id );
+}
+
+/*!
+ \internal
+ This slot is activated when one of the buttons in the group emits the
+ QButton::clicked() signal.
+*/
+
+void QButtonGroup::buttonClicked()
+{
+ // introduce a QButtonListIt if calling anything
+ int id = -1;
+ QButton *bt = ::qt_cast<QButton*>(sender()); // object that sent the signal
+#if defined(QT_CHECK_NULL)
+ Q_ASSERT( bt );
+#endif
+ for ( register QButtonItem *i=buttons->first(); i; i=buttons->next() ) {
+ if ( bt == i->button ) { // button was clicked
+ id = i->id;
+ break;
+ }
+ }
+ if ( id != -1 )
+ emit clicked( id );
+}
+
+
+/*!
+ \internal
+ This slot is activated when one of the buttons in the group emits the
+ QButton::toggled() signal.
+*/
+
+void QButtonGroup::buttonToggled( bool on )
+{
+ // introduce a QButtonListIt if calling anything
+ if ( !on || !excl_grp && !radio_excl )
+ return;
+ QButton *bt = ::qt_cast<QButton*>(sender()); // object that sent the signal
+#if defined(QT_CHECK_NULL)
+ Q_ASSERT( bt );
+ Q_ASSERT( bt->isToggleButton() );
+#endif
+
+ if ( !excl_grp && !::qt_cast<QRadioButton*>(bt) )
+ return;
+ QButtonItem * i = buttons->first();
+ bool hasTabFocus = FALSE;
+
+ while( i != 0 && hasTabFocus == FALSE ) {
+ if ( ( excl_grp || ::qt_cast<QRadioButton*>(i->button) ) &&
+ (i->button->focusPolicy() & TabFocus) )
+ hasTabFocus = TRUE;
+ i = buttons->next();
+ }
+
+ i = buttons->first();
+ while( i ) {
+ if ( bt != i->button &&
+ i->button->isToggleButton() &&
+ i->button->isOn() &&
+ ( excl_grp || ::qt_cast<QRadioButton*>(i->button) ) )
+ i->button->setOn( FALSE );
+ if ( ( excl_grp || ::qt_cast<QRadioButton*>(i->button) ) &&
+ i->button->isToggleButton() &&
+ hasTabFocus )
+ i->button->setFocusPolicy( (FocusPolicy)(i->button->focusPolicy() &
+ ~TabFocus) );
+ i = buttons->next();
+ }
+
+ if ( hasTabFocus )
+ bt->setFocusPolicy( (FocusPolicy)(bt->focusPolicy() | TabFocus) );
+}
+
+
+
+void QButtonGroup::setButton( int id )
+{
+ QButton * b = find( id );
+ if ( b )
+ b->setOn( TRUE );
+}
+
+void QButtonGroup::setRadioButtonExclusive( bool on)
+{
+ radio_excl = on;
+}
+
+
+/*!
+ Moves the keyboard focus according to \a key, and if appropriate
+ checks the new focus item.
+
+ This function does nothing unless the keyboard focus points to one
+ of the button group members and \a key is one of \c Key_Up, \c
+ Key_Down, \c Key_Left and \c Key_Right.
+*/
+
+void QButtonGroup::moveFocus( int key )
+{
+ QWidget * f = qApp->focusWidget();
+
+ QButtonItem * i;
+ i = buttons->first();
+ while( i && i->button != f )
+ i = buttons->next();
+
+ if ( !i || !i->button )
+ return;
+
+ QWidget * candidate = 0;
+ int bestScore = -1;
+
+ QPoint goal( f->mapToGlobal( f->geometry().center() ) );
+
+ i = buttons->first();
+ while( i && i->button ) {
+ if ( i->button != f &&
+ i->button->isEnabled() ) {
+ QPoint p(i->button->mapToGlobal(i->button->geometry().center()));
+ int score = (p.y() - goal.y())*(p.y() - goal.y()) +
+ (p.x() - goal.x())*(p.x() - goal.x());
+ bool betterScore = score < bestScore || !candidate;
+ switch( key ) {
+ case Key_Up:
+ if ( p.y() < goal.y() && betterScore ) {
+ if ( QABS( p.x() - goal.x() ) < QABS( p.y() - goal.y() ) ) {
+ candidate = i->button;
+ bestScore = score;
+ } else if ( i->button->x() == f->x() ) {
+ candidate = i->button;
+ bestScore = score/2;
+ }
+ }
+ break;
+ case Key_Down:
+ if ( p.y() > goal.y() && betterScore ) {
+ if ( QABS( p.x() - goal.x() ) < QABS( p.y() - goal.y() ) ) {
+ candidate = i->button;
+ bestScore = score;
+ } else if ( i->button->x() == f->x() ) {
+ candidate = i->button;
+ bestScore = score/2;
+ }
+ }
+ break;
+ case Key_Left:
+ if ( p.x() < goal.x() && betterScore ) {
+ if ( QABS( p.y() - goal.y() ) < QABS( p.x() - goal.x() ) ) {
+ candidate = i->button;
+ bestScore = score;
+ } else if ( i->button->y() == f->y() ) {
+ candidate = i->button;
+ bestScore = score/2;
+ }
+ }
+ break;
+ case Key_Right:
+ if ( p.x() > goal.x() && betterScore ) {
+ if ( QABS( p.y() - goal.y() ) < QABS( p.x() - goal.x() ) ) {
+ candidate = i->button;
+ bestScore = score;
+ } else if ( i->button->y() == f->y() ) {
+ candidate = i->button;
+ bestScore = score/2;
+ }
+ }
+ break;
+ }
+ }
+ i = buttons->next();
+ }
+
+ QButton *buttoncand = ::qt_cast<QButton*>(candidate);
+ if ( buttoncand && ::qt_cast<QButton*>(f) &&
+ ((QButton*)f)->isOn() &&
+ buttoncand->isToggleButton() &&
+ ( isExclusive() || ( ::qt_cast<QRadioButton*>(f) &&
+ ::qt_cast<QRadioButton*>(candidate)))) {
+ if ( f->focusPolicy() & TabFocus ) {
+ f->setFocusPolicy( (FocusPolicy)(f->focusPolicy() & ~TabFocus) );
+ candidate->setFocusPolicy( (FocusPolicy)(candidate->focusPolicy()|
+ TabFocus) );
+ }
+ buttoncand->setOn( TRUE );
+ buttoncand->animateClick();
+ buttoncand->animateTimeout(); // ### crude l&f hack
+ }
+
+ if ( candidate ) {
+ if (key == Key_Up || key == Key_Left)
+ QFocusEvent::setReason(QFocusEvent::Backtab);
+ else
+ QFocusEvent::setReason(QFocusEvent::Tab);
+ candidate->setFocus();
+ QFocusEvent::resetReason();
+ }
+}
+
+
+/*!
+ Returns the selected toggle button if exactly one is selected;
+ otherwise returns 0.
+
+ \sa selectedId()
+*/
+
+QButton * QButtonGroup::selected() const
+{
+ if ( !buttons )
+ return 0;
+ QButtonListIt it( *buttons );
+ QButtonItem *i;
+ QButton *candidate = 0;
+
+ while ( (i = it.current()) != 0 ) {
+ ++it;
+ if ( i->button && i->button->isToggleButton() && i->button->isOn() ) {
+ if ( candidate != 0 )
+ return 0;
+ candidate = i->button;
+ }
+ }
+ return candidate;
+}
+
+/*!
+ \property QButtonGroup::selectedId
+ \brief the selected toggle button
+
+ The toggle button is specified as an ID.
+
+ If no toggle button is selected, this property holds -1.
+
+ If setButton() is called on an exclusive group, the button with
+ the given id will be set to on and all the others will be set to
+ off.
+
+ \sa selected()
+*/
+
+int QButtonGroup::selectedId() const
+{
+ return id( selected() );
+}
+
+
+/*!
+ Returns the id of \a button, or -1 if \a button is not a member of
+ this group.
+
+ \sa selectedId();
+*/
+
+int QButtonGroup::id( QButton * button ) const
+{
+ QButtonItem *i = buttons->first();
+ while ( i && i->button != button )
+ i = buttons->next();
+ return i ? i->id : -1;
+}
+
+
+/*!
+ \reimp
+*/
+bool QButtonGroup::event( QEvent * e )
+{
+ if ( e->type() == QEvent::ChildInserted ) {
+ QChildEvent * ce = (QChildEvent *) e;
+ if ( radio_excl && ::qt_cast<QRadioButton*>(ce->child()) ) {
+ QButton * button = (QButton *) ce->child();
+ if ( button->isToggleButton() && !button->isOn() &&
+ selected() && (selected()->focusPolicy() & TabFocus) != 0 )
+ button->setFocusPolicy( (FocusPolicy)(button->focusPolicy() &
+ ~TabFocus) );
+ }
+ }
+ return QGroupBox::event( e );
+}
+#endif