/**************************************************************************** ** ** Implementation of tqlayout classes ** ** Created : 960416 ** ** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. ** ** This file is part of the kernel module of the TQt 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 TQt 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 sales@trolltech.com. ** ** This file may be used under the terms of the Q Public License as ** defined by Trolltech ASA and appearing in the file LICENSE.TQPL ** included in the packaging of this file. Licensees holding valid TQt ** Commercial licenses may use this file in accordance with the TQt ** 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 "tqlayout.h" #ifndef TQT_NO_LAYOUT #include "tqapplication.h" #include "tqwidget.h" #include "tqmenubar.h" #include "tqptrlist.h" #include "tqsizepolicy.h" #include "tqlayoutengine_p.h" #ifdef USE_QT4 int menuBarHeightForWidth( QWidget *menubar_w, int w ) { #ifndef TQT_NO_MENUBAR TQMenuBar *menubar = static_cast(TQT_TQWIDGET(menubar_w)); if ( menubar && !menubar->isHidden() && !menubar->isTopLevel() ) return menubar->heightForWidth( TQMAX(w, menubar->minimumWidth()) ); else #endif return 0; } TQLayout::TQLayout( QWidget *tqparent, int margin, int spacing, const char *name ) : QLayout( tqparent ), autoNewChild(false) { setObjectName(QString::fromAscii(name)); setMargin(margin); if (spacing < 0) setSpacing(margin); else setSpacing(spacing); if ( tqparent ) tqparent->installEventFilter( this ); TQT_TQOBJECT_REQUIRED_INITIALIZATION(tqparent) } TQLayout::TQLayout( QLayout *parentLayout, int spacing, const char *name ) : QLayout(), autoNewChild(false) { setObjectName(QString::fromAscii(name)); setSpacing(spacing); if ( parentLayout ) parentLayout->addItem(this); if ( parentLayout ) parentLayout->installEventFilter( this ); TQT_TQOBJECT_REQUIRED_INITIALIZATION(parentLayout) } TQLayout::TQLayout( int spacing, const char *name ) : QLayout(), autoNewChild(false) { setObjectName(QString::fromAscii(name)); setSpacing(spacing); } TQ_SPExpandData TQLayout::expandingDirections() const { return (Qt::Orientations)TQSizePolicy::BothDirections; } TQLayout *TQLayout::tqlayout() { return this; } TQSize TQLayout::tqsizeHint() const { return sizeHint(); } TQSize TQLayout::tqminimumSize() const { return QLayout::minimumSize(); } TQSize TQLayout::tqmaximumSize() const { return QLayout::maximumSize(); } TQWidget *TQLayout::mainWidget() { return TQT_TQWIDGET(parentWidget()); } void TQLayout::tqinvalidate() { QLayout::invalidate(); } void TQLayout::invalidate() { tqinvalidate(); } TQRect TQLayout::tqgeometry() const { return QLayout::geometry(); } const char *TQLayout::tqname() const { if (dynamic_cast(static_cast(static_cast(this)))) { static_object_name = TQT_OBJECT_NAME_HANDLER(objectName()); return static_object_name.ascii(); } else { printf("[WARNING] Attempted to call TQLayout::tqname() on an object without a constructed TQLayout object or base object. Returning \"\"\n\r"); return ""; } } const char *TQLayout::name() const { if (dynamic_cast(static_cast(static_cast(this)))) { static_object_name = TQT_OBJECT_NAME_HANDLER(objectName()); return static_object_name.ascii(); } else { printf("[WARNING] Attempted to call TQLayout::tqname() on an object without a constructed TQLayout object or base object. Returning \"\"\n\r"); return ""; } } int TQLayout::tqalignment() const { return alignment(); } TQWidget *TQLayout::mainWidget() const { return TQT_TQWIDGET(parentWidget()); } void TQLayout::remove(QWidget *w) { removeWidget(w); } void TQLayout::add(QWidget *w) { addWidget(w); } void TQLayout::setResizeMode(SizeConstraint s) { setSizeConstraint(s); } void TQLayout::setResizeMode(ResizeMode s) { setResizeMode((SizeConstraint)s); } TQLayout::SizeConstraint TQLayout::resizeMode() const { return sizeConstraint(); } TQLayout::ResizeMode TQLayout::tqresizeMode() const { return (ResizeMode)sizeConstraint(); } void TQLayout::setAutoAdd(bool a) { autoNewChild = a; } bool TQLayout::autoAdd() const { return autoNewChild; } void TQLayout::tqsetAlignment( int a ) { return setAlignment((Qt::Alignment)a); } // Qt4 handler interface bool TQLayout::eventFilter( QObject *q, QEvent *e ) { return eventFilter(static_cast(q), static_cast(e)); } bool TQLayout::event( QEvent *e ) { return event(static_cast(e)); } void TQLayout::timerEvent(QTimerEvent *e) { timerEvent(static_cast(e)); } void TQLayout::childEvent(QChildEvent *e) { TQT_TQOBJECT_CHILDEVENT_CONDITIONAL childEvent(static_cast(e)); } void TQLayout::customEvent(QEvent *e) { customEvent(static_cast(e)); } bool TQLayout::eventFilter( TQObject *o, TQEvent *e ) { TQ_UNUSED(o); if (e->type() == TQEvent::Resize) { activate(); TQResizeEvent *r = (TQResizeEvent *)e; int mbh = 0; #ifndef TQT_NO_MENUBAR mbh = menuBarHeightForWidth( menuBar(), r->size().width() ); #endif // int b = marginImpl ? 0 : outsideBorder; int b = 0; setGeometry( TQRect( b, mbh + b, r->size().width() - 2 * b, r->size().height() - mbh - 2 * b ) ); } else if (e->type() == TQEvent::ChildInserted) { if (autoNewChild) { QChildEvent *c = (QChildEvent *)e; if (c->child()->isWidgetType()) { QWidget *w = (QWidget *)c->child(); if (!w->isWindow()) { // #if !defined(QT_NO_MENUBAR) && !defined(QT_NO_TOOLBAR) // if (qobject_cast(w) && !qobject_cast(w->parentWidget())) { // setMenuBar( (QMenuBar *)w ); // invalidate(); // } else // #endif // #ifndef QT_NO_SIZEGRIP // if (qobject_cast(w) ) { // //SizeGrip is handled by the dialog itself. // } else // #endif // addItem(QLayoutPrivate::createWidgetItem(this, w)); addWidget(w); } } } } else if (e->type() == TQEvent::LayoutHint) { // d->d = false; // if (parent()) { // if (static_cast(parent())->isVisible()) { activate(); // } // } } // return FALSE; return QLayout::eventFilter( o, e ); // return TQObject::eventFilter( o, e ); } TQLayoutIterator TQLayout::iterator() { return TQLayoutIterator(this,true); } /*! \compat Sets this layout's parent widget to a fixed size with width \a w and height \a h, stopping the user from resizing it, and also prevents the layout from resizing it, even if the layout's size hint should change. Does nothing if this is not a top-level layout (i.e., if parent()->isWidgetType()). As a special case, if both \a w and \a h are 0, then the layout's current sizeHint() is used. Use \c setResizeMode(Fixed) to stop the widget from being resized by the user, while still allowing the layout to resize it when the sizeHint() changes. Use \c setResizeMode(FreeResize) to allow the user to resize the widget, while preventing the layout from resizing it. */ void TQLayout::freeze(int w, int h) { // if (!isTopLevel()) // return; if (w <= 0 || h <= 0) { QSize s = totalSizeHint(); w = s.width(); h = s.height(); } setSizeConstraint(SetNoConstraint); // layout will not change min/max size QWidget *parent = parentWidget(); if (parent) parent->setFixedSize(w, h); } // Use the TQt virtual functions, not the built in Qt ones... // This requires that the base virtual Qt functions be reimplemented so as to point to the TQt virtual functions instead as shown below. // This way, when Trinity overrides a TQt virtual function, the calling Qt code will blithely use the overriden TQt function instead. QLayout *TQLayout::layout() { return tqlayout(); } void TQLayout::setGeometry(const TQRect &r) { return QLayout::setGeometry(r); } TQRect TQLayout::alignmentRect( const TQRect& qr ) const { return alignmentRect(qr); } void TQLayout::tqt_handle_qt_destroyed(QObject* obj) { emit destroyed(TQT_TQOBJECT(obj)); } TQLayoutIterator::TQLayoutIterator(QLayout *i) : layout(static_cast(i)), index(0) { } TQLayoutIterator::TQLayoutIterator(QLayout *i, bool) : layout(static_cast(i)), index(0) { } TQLayoutItem *TQLayoutIterator::operator++() { return static_cast(layout->itemAt(++index)); } TQLayoutItem *TQLayoutIterator::current() { return static_cast(layout->itemAt(index)); } TQLayoutItem *TQLayoutIterator::takeCurrent() { return static_cast(layout->takeAt(index)); } void TQLayoutIterator::deleteCurrent() { delete layout->takeAt(index); } TQSpacerItem *TQLayoutItem::tqspacerItem() { return static_cast(QLayoutItem::spacerItem()); } /*! This virtual function receives events to an object and should return TRUE if the event \a e was recognized and processed. The event() function can be reimplemented to customize the behavior of an object. \sa installEventFilter(), timerEvent(), TQApplication::sendEvent(), TQApplication::postEvent(), TQWidget::event() */ bool TQLayout::event( TQEvent *e ) { #if defined(TQT_CHECK_NULL) if ( e == 0 ) qWarning( "TQLayout::event: Null events are not permitted" ); #endif // if ( eventFilters ) { // try filters // if ( activate_filters(e) ) // stopped by a filter // return TRUE; // } // switch ( e->type() ) { // case TQEvent::Timer: // timerEvent( (TQTimerEvent*)e ); // return TRUE; // // case TQEvent::DeferredDelete: // delete this; // return TRUE; // // default: // if ( e->type() >= TQEvent::User ) { // customEvent( (TQCustomEvent*) e ); // return TRUE; // } // break; // } case TQEvent::ChildInserted: // case TQEvent::ChildRemoved: // Causes a recursion loop if uncommented childEvent( (TQChildEvent*)e ); return TRUE; default: return QLayout::event(e); } return FALSE; } #if 0 void TQGridLayout::tqsetAlignment( int a ) { setAlignment((Qt::AlignmentFlag)a); } #endif #if 0 // NOTE // The following functions are cloned from TQWidget // They are only required for TQLayout and TQGridLayout becase those classes do not inherit TQObject or TQWidget /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse move events for the widget. If mouse tracking is switched off, mouse move events only occur if a mouse button is pressed while the mouse is being moved. If mouse tracking is switched on, mouse move events occur even if no mouse button is pressed. TQMouseEvent::pos() reports the position of the mouse cursor, relative to this widget. For press and release events, the position is usually the same as the position of the last mouse move event, but it might be different if the user's hand shakes. This is a feature of the underlying window system, not TQt. \sa setMouseTracking(), mousePressEvent(), mouseReleaseEvent(), mouseDoubleClickEvent(), event(), TQMouseEvent */ void TQLayout::mouseMoveEvent( TQMouseEvent * e) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse press events for the widget. If you create new widgets in the mousePressEvent() the mouseReleaseEvent() may not end up where you expect, depending on the underlying window system (or X11 window manager), the widgets' location and maybe more. The default implementation implements the closing of popup widgets when you click outside the window. For other widget types it does nothing. \sa mouseReleaseEvent(), mouseDoubleClickEvent(), mouseMoveEvent(), event(), TQMouseEvent */ void TQLayout::mousePressEvent( TQMouseEvent * ) { // TQT_TQWIDGET_CONST(this)->mousePressEvent(e); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse release events for the widget. \sa mouseReleaseEvent(), mouseDoubleClickEvent(), mouseMoveEvent(), event(), TQMouseEvent */ void TQLayout::mouseReleaseEvent( TQMouseEvent * e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse double click events for the widget. The default implementation generates a normal mouse press event. Note that the widgets gets a mousePressEvent() and a mouseReleaseEvent() before the mouseDoubleClickEvent(). \sa mousePressEvent(), mouseReleaseEvent() mouseMoveEvent(), event(), TQMouseEvent */ void TQLayout::mouseDoubleClickEvent( TQMouseEvent *e ) { mousePressEvent( e ); // try mouse press event } #ifndef TQT_NO_WHEELEVENT /*! This event handler, for event \a e, can be reimplemented in a subclass to receive wheel events for the widget. If you reimplement this handler, it is very important that you \link TQWheelEvent ignore()\endlink the event if you do not handle it, so that the widget's tqparent can interpret it. The default implementation ignores the event. \sa TQWheelEvent::ignore(), TQWheelEvent::accept(), event(), TQWheelEvent */ void TQLayout::wheelEvent( TQWheelEvent *e ) { e->ignore(); } #endif /*! This event handler, for event \a e, can be reimplemented in a subclass to receive tablet events for the widget. If you reimplement this handler, it is very important that you \link TQTabletEvent ignore()\endlink the event if you do not handle it, so that the widget's tqparent can interpret it. The default implementation ignores the event. \sa TQTabletEvent::ignore(), TQTabletEvent::accept(), event(), TQTabletEvent */ void TQLayout::tabletEvent( TQTabletEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive key press events for the widget. A widget must call setFocusPolicy() to accept focus initially and have focus in order to receive a key press event. If you reimplement this handler, it is very important that you explicitly \link TQKeyEvent::ignore() ignore\endlink the event if you do not understand it, so that the widget's tqparent can interpret it; otherwise, the event will be implicitly accepted. Although top-level widgets are able to choose whether to accept or ignore unknown events because they have no tqparent widgets that could otherwise handle them, it is good practice to explicitly ignore events to make widgets as reusable as possible. The default implementation closes popup widgets if the user presses Esc. Otherwise the event is ignored. \sa keyReleaseEvent(), TQKeyEvent::ignore(), setFocusPolicy(), focusInEvent(), focusOutEvent(), event(), TQKeyEvent */ void TQLayout::keyPressEvent( TQKeyEvent * ) { // TQT_TQWIDGET_CONST(this)->keyPressEvent(e); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive key release events for the widget. A widget must \link setFocusPolicy() accept focus\endlink initially and \link hasFocus() have focus\endlink in order to receive a key release event. If you reimplement this handler, it is very important that you \link TQKeyEvent ignore()\endlink the release if you do not understand it, so that the widget's tqparent can interpret it. The default implementation ignores the event. \sa keyPressEvent(), TQKeyEvent::ignore(), setFocusPolicy(), focusInEvent(), focusOutEvent(), event(), TQKeyEvent */ void TQLayout::keyReleaseEvent( TQKeyEvent *e ) { e->ignore(); } /*! This event handler can be reimplemented in a subclass to receive keyboard focus events (focus received) for the widget. A widget normally must setFocusPolicy() to something other than \c NoFocus in order to receive focus events. (Note that the application programmer can call setFocus() on any widget, even those that do not normally accept focus.) The default implementation updates the widget (except for toplevel widgets that do not specify a focusPolicy() ). It also calls setMicroFocusHint(), hinting any system-specific input tools about the focus of the user's attention. \sa focusOutEvent(), setFocusPolicy(), keyPressEvent(), keyReleaseEvent(), event(), TQFocusEvent */ void TQLayout::focusInEvent( TQFocusEvent * ) { // TQT_TQWIDGET_CONST(this)->focusInEvent(e); } /*! This event handler can be reimplemented in a subclass to receive keyboard focus events (focus lost) for the widget. A widget normally must setFocusPolicy() to something other than \c NoFocus in order to receive focus events. (Note that the application programmer can call setFocus() on any widget, even those that do not normally accept focus.) The default implementation updates the widget (except for toplevel widgets that do not specify a focusPolicy() ). It also calls setMicroFocusHint(), hinting any system-specific input tools about the focus of the user's attention. \sa focusInEvent(), setFocusPolicy(), keyPressEvent(), keyReleaseEvent(), event(), TQFocusEvent */ void TQLayout::focusOutEvent( TQFocusEvent * ) { // TQT_TQWIDGET_CONST(this)->focusOutEvent(e); } /*! This event handler can be reimplemented in a subclass to receive widget enter events. An event is sent to the widget when the mouse cursor enters the widget. \sa leaveEvent(), mouseMoveEvent(), event() */ void TQLayout::enterEvent( TQEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive widget leave events. A leave event is sent to the widget when the mouse cursor leaves the widget. \sa enterEvent(), mouseMoveEvent(), event() */ void TQLayout::leaveEvent( TQEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive paint events. A paint event is a request to tqrepaint all or part of the widget. It can happen as a result of tqrepaint() or update(), or because the widget was obscured and has now been uncovered, or for many other reasons. Many widgets can simply tqrepaint their entire surface when asked to, but some slow widgets need to optimize by painting only the requested region: TQPaintEvent::region(). This speed optimization does not change the result, as painting is clipped to that region during event processing. TQListView and TQCanvas do this, for example. TQt also tries to speed up painting by merging multiple paint events into one. When update() is called several times or the window system sends several paint events, TQt merges these events into one event with a larger region (see TQRegion::unite()). tqrepaint() does not permit this optimization, so we suggest using update() when possible. When the paint event occurs, the update region has normally been erased, so that you're painting on the widget's background. There are a couple of exceptions and TQPaintEvent::erased() tells you whether the widget has been erased or not. The background can be set using setBackgroundMode(), setPaletteBackgroundColor() or setBackgroundPixmap(). The documentation for setBackgroundMode() elaborates on the background; we recommend reading it. \sa event(), tqrepaint(), update(), TQPainter, TQPixmap, TQPaintEvent */ void TQLayout::paintEvent( TQPaintEvent *e ) { } /*! This event handler can be reimplemented in a subclass to receive widget move events. When the widget receives this event, it is already at the new position. The old position is accessible through TQMoveEvent::oldPos(). \sa resizeEvent(), event(), move(), TQMoveEvent */ void TQLayout::moveEvent( TQMoveEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive widget resize events. When resizeEvent() is called, the widget already has its new tqgeometry. The old size is accessible through TQResizeEvent::oldSize(). The widget will be erased and receive a paint event immediately after processing the resize event. No drawing need be (or should be) done inside this handler. Widgets that have been created with the \c WNoAutoErase flag will not be erased. Nevertheless, they will receive a paint event for their entire area afterwards. Again, no drawing needs to be done inside this handler. The default implementation calls updateMask() if the widget has \link TQLayout::setAutoMask() automatic masking\endlink enabled. \sa moveEvent(), event(), resize(), TQResizeEvent, paintEvent() */ void TQLayout::resizeEvent( TQResizeEvent * ) { } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive widget close events. The default implementation calls e->accept(), which hides this widget. See the \l TQCloseEvent documentation for more details. \sa event(), hide(), close(), TQCloseEvent */ void TQLayout::closeEvent( TQCloseEvent *e ) { e->accept(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive widget context menu events. The default implementation calls e->ignore(), which rejects the context event. See the \l TQContextMenuEvent documentation for more details. \sa event(), TQContextMenuEvent */ void TQLayout::contextMenuEvent( TQContextMenuEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive Input Method composition events. This handler is called when the user begins entering text using an Input Method. The default implementation calls e->ignore(), which rejects the Input Method event. See the \l TQIMEvent documentation for more details. \sa event(), TQIMEvent */ void TQLayout::imStartEvent( TQIMEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive Input Method composition events. This handler is called when the user has entered some text using an Input Method. The default implementation calls e->ignore(), which rejects the Input Method event. See the \l TQIMEvent documentation for more details. \sa event(), TQIMEvent */ void TQLayout::imComposeEvent( TQIMEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive Input Method composition events. This handler is called when the user has finished inputting text via an Input Method. The default implementation calls e->ignore(), which rejects the Input Method event. See the \l TQIMEvent documentation for more details. \sa event(), TQIMEvent */ void TQLayout::imEndEvent( TQIMEvent *e ) { e->ignore(); } #ifndef TQT_NO_DRAGANDDROP /*! This event handler is called when a drag is in progress and the mouse enters this widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDragEnterEvent */ void TQLayout::dragEnterEvent( TQDragEnterEvent * ) { } /*! This event handler is called when a drag is in progress and the mouse enters this widget, and whenever it moves within the widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDragMoveEvent */ void TQLayout::dragMoveEvent( TQDragMoveEvent * ) { } /*! This event handler is called when a drag is in progress and the mouse leaves this widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDragLeaveEvent */ void TQLayout::dragLeaveEvent( TQDragLeaveEvent * ) { } /*! This event handler is called when the drag is dropped on this widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDropEvent */ void TQLayout::dropEvent( TQDropEvent * ) { } #endif // TQT_NO_DRAGANDDROP /*! This event handler can be reimplemented in a subclass to receive widget show events. Non-spontaneous show events are sent to widgets immediately before they are shown. The spontaneous show events of top-level widgets are delivered afterwards. \sa event(), TQShowEvent */ void TQLayout::showEvent( TQShowEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive widget hide events. Hide events are sent to widgets immediately after they have been hidden. \sa event(), TQHideEvent */ void TQLayout::hideEvent( TQHideEvent * ) { } #endif void TQLayout::timerEvent( TQTimerEvent * ) { } void TQLayout::childEvent( TQChildEvent * ) { } void TQLayout::customEvent( TQCustomEvent * ) { } #if 0 /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse move events for the widget. If mouse tracking is switched off, mouse move events only occur if a mouse button is pressed while the mouse is being moved. If mouse tracking is switched on, mouse move events occur even if no mouse button is pressed. TQMouseEvent::pos() reports the position of the mouse cursor, relative to this widget. For press and release events, the position is usually the same as the position of the last mouse move event, but it might be different if the user's hand shakes. This is a feature of the underlying window system, not TQt. \sa setMouseTracking(), mousePressEvent(), mouseReleaseEvent(), mouseDoubleClickEvent(), event(), TQMouseEvent */ void TQGridLayout::mouseMoveEvent( TQMouseEvent * e) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse press events for the widget. If you create new widgets in the mousePressEvent() the mouseReleaseEvent() may not end up where you expect, depending on the underlying window system (or X11 window manager), the widgets' location and maybe more. The default implementation implements the closing of popup widgets when you click outside the window. For other widget types it does nothing. \sa mouseReleaseEvent(), mouseDoubleClickEvent(), mouseMoveEvent(), event(), TQMouseEvent */ void TQGridLayout::mousePressEvent( TQMouseEvent * ) { // TQT_TQLAYOUT_CONST(this)->mousePressEvent(e); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse release events for the widget. \sa mouseReleaseEvent(), mouseDoubleClickEvent(), mouseMoveEvent(), event(), TQMouseEvent */ void TQGridLayout::mouseReleaseEvent( TQMouseEvent * e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive mouse double click events for the widget. The default implementation generates a normal mouse press event. Note that the widgets gets a mousePressEvent() and a mouseReleaseEvent() before the mouseDoubleClickEvent(). \sa mousePressEvent(), mouseReleaseEvent() mouseMoveEvent(), event(), TQMouseEvent */ void TQGridLayout::mouseDoubleClickEvent( TQMouseEvent *e ) { mousePressEvent( e ); // try mouse press event } #ifndef TQT_NO_WHEELEVENT /*! This event handler, for event \a e, can be reimplemented in a subclass to receive wheel events for the widget. If you reimplement this handler, it is very important that you \link TQWheelEvent ignore()\endlink the event if you do not handle it, so that the widget's tqparent can interpret it. The default implementation ignores the event. \sa TQWheelEvent::ignore(), TQWheelEvent::accept(), event(), TQWheelEvent */ void TQGridLayout::wheelEvent( TQWheelEvent *e ) { e->ignore(); } #endif /*! This event handler, for event \a e, can be reimplemented in a subclass to receive tablet events for the widget. If you reimplement this handler, it is very important that you \link TQTabletEvent ignore()\endlink the event if you do not handle it, so that the widget's tqparent can interpret it. The default implementation ignores the event. \sa TQTabletEvent::ignore(), TQTabletEvent::accept(), event(), TQTabletEvent */ void TQGridLayout::tabletEvent( TQTabletEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive key press events for the widget. A widget must call setFocusPolicy() to accept focus initially and have focus in order to receive a key press event. If you reimplement this handler, it is very important that you explicitly \link TQKeyEvent::ignore() ignore\endlink the event if you do not understand it, so that the widget's tqparent can interpret it; otherwise, the event will be implicitly accepted. Although top-level widgets are able to choose whether to accept or ignore unknown events because they have no tqparent widgets that could otherwise handle them, it is good practice to explicitly ignore events to make widgets as reusable as possible. The default implementation closes popup widgets if the user presses Esc. Otherwise the event is ignored. \sa keyReleaseEvent(), TQKeyEvent::ignore(), setFocusPolicy(), focusInEvent(), focusOutEvent(), event(), TQKeyEvent */ void TQGridLayout::keyPressEvent( TQKeyEvent * ) { // TQT_TQLAYOUT_CONST(this)->keyPressEvent(e); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive key release events for the widget. A widget must \link setFocusPolicy() accept focus\endlink initially and \link hasFocus() have focus\endlink in order to receive a key release event. If you reimplement this handler, it is very important that you \link TQKeyEvent ignore()\endlink the release if you do not understand it, so that the widget's tqparent can interpret it. The default implementation ignores the event. \sa keyPressEvent(), TQKeyEvent::ignore(), setFocusPolicy(), focusInEvent(), focusOutEvent(), event(), TQKeyEvent */ void TQGridLayout::keyReleaseEvent( TQKeyEvent *e ) { e->ignore(); } /*! This event handler can be reimplemented in a subclass to receive keyboard focus events (focus received) for the widget. A widget normally must setFocusPolicy() to something other than \c NoFocus in order to receive focus events. (Note that the application programmer can call setFocus() on any widget, even those that do not normally accept focus.) The default implementation updates the widget (except for toplevel widgets that do not specify a focusPolicy() ). It also calls setMicroFocusHint(), hinting any system-specific input tools about the focus of the user's attention. \sa focusOutEvent(), setFocusPolicy(), keyPressEvent(), keyReleaseEvent(), event(), TQFocusEvent */ void TQGridLayout::focusInEvent( TQFocusEvent * ) { // TQT_TQLAYOUT_CONST(this)->focusInEvent(e); } /*! This event handler can be reimplemented in a subclass to receive keyboard focus events (focus lost) for the widget. A widget normally must setFocusPolicy() to something other than \c NoFocus in order to receive focus events. (Note that the application programmer can call setFocus() on any widget, even those that do not normally accept focus.) The default implementation updates the widget (except for toplevel widgets that do not specify a focusPolicy() ). It also calls setMicroFocusHint(), hinting any system-specific input tools about the focus of the user's attention. \sa focusInEvent(), setFocusPolicy(), keyPressEvent(), keyReleaseEvent(), event(), TQFocusEvent */ void TQGridLayout::focusOutEvent( TQFocusEvent * ) { // TQT_TQLAYOUT_CONST(this)->focusOutEvent(e); } /*! This event handler can be reimplemented in a subclass to receive widget enter events. An event is sent to the widget when the mouse cursor enters the widget. \sa leaveEvent(), mouseMoveEvent(), event() */ void TQGridLayout::enterEvent( TQEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive widget leave events. A leave event is sent to the widget when the mouse cursor leaves the widget. \sa enterEvent(), mouseMoveEvent(), event() */ void TQGridLayout::leaveEvent( TQEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive paint events. A paint event is a request to tqrepaint all or part of the widget. It can happen as a result of tqrepaint() or update(), or because the widget was obscured and has now been uncovered, or for many other reasons. Many widgets can simply tqrepaint their entire surface when asked to, but some slow widgets need to optimize by painting only the requested region: TQPaintEvent::region(). This speed optimization does not change the result, as painting is clipped to that region during event processing. TQListView and TQCanvas do this, for example. TQt also tries to speed up painting by merging multiple paint events into one. When update() is called several times or the window system sends several paint events, TQt merges these events into one event with a larger region (see TQRegion::unite()). tqrepaint() does not permit this optimization, so we suggest using update() when possible. When the paint event occurs, the update region has normally been erased, so that you're painting on the widget's background. There are a couple of exceptions and TQPaintEvent::erased() tells you whether the widget has been erased or not. The background can be set using setBackgroundMode(), setPaletteBackgroundColor() or setBackgroundPixmap(). The documentation for setBackgroundMode() elaborates on the background; we recommend reading it. \sa event(), tqrepaint(), update(), TQPainter, TQPixmap, TQPaintEvent */ void TQGridLayout::paintEvent( TQPaintEvent *e ) { // // At least let Qt4 get a shot at painting it // QWidget::paintEvent(e); } /*! This event handler can be reimplemented in a subclass to receive widget move events. When the widget receives this event, it is already at the new position. The old position is accessible through TQMoveEvent::oldPos(). \sa resizeEvent(), event(), move(), TQMoveEvent */ void TQGridLayout::moveEvent( TQMoveEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive widget resize events. When resizeEvent() is called, the widget already has its new tqgeometry. The old size is accessible through TQResizeEvent::oldSize(). The widget will be erased and receive a paint event immediately after processing the resize event. No drawing need be (or should be) done inside this handler. Widgets that have been created with the \c WNoAutoErase flag will not be erased. Nevertheless, they will receive a paint event for their entire area afterwards. Again, no drawing needs to be done inside this handler. The default implementation calls updateMask() if the widget has \link TQGridLayout::setAutoMask() automatic masking\endlink enabled. \sa moveEvent(), event(), resize(), TQResizeEvent, paintEvent() */ void TQGridLayout::resizeEvent( TQResizeEvent * ) { } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive widget close events. The default implementation calls e->accept(), which hides this widget. See the \l TQCloseEvent documentation for more details. \sa event(), hide(), close(), TQCloseEvent */ void TQGridLayout::closeEvent( TQCloseEvent *e ) { e->accept(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive widget context menu events. The default implementation calls e->ignore(), which rejects the context event. See the \l TQContextMenuEvent documentation for more details. \sa event(), TQContextMenuEvent */ void TQGridLayout::contextMenuEvent( TQContextMenuEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive Input Method composition events. This handler is called when the user begins entering text using an Input Method. The default implementation calls e->ignore(), which rejects the Input Method event. See the \l TQIMEvent documentation for more details. \sa event(), TQIMEvent */ void TQGridLayout::imStartEvent( TQIMEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive Input Method composition events. This handler is called when the user has entered some text using an Input Method. The default implementation calls e->ignore(), which rejects the Input Method event. See the \l TQIMEvent documentation for more details. \sa event(), TQIMEvent */ void TQGridLayout::imComposeEvent( TQIMEvent *e ) { e->ignore(); } /*! This event handler, for event \a e, can be reimplemented in a subclass to receive Input Method composition events. This handler is called when the user has finished inputting text via an Input Method. The default implementation calls e->ignore(), which rejects the Input Method event. See the \l TQIMEvent documentation for more details. \sa event(), TQIMEvent */ void TQGridLayout::imEndEvent( TQIMEvent *e ) { e->ignore(); } #ifndef TQT_NO_DRAGANDDROP /*! This event handler is called when a drag is in progress and the mouse enters this widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDragEnterEvent */ void TQGridLayout::dragEnterEvent( TQDragEnterEvent * ) { } /*! This event handler is called when a drag is in progress and the mouse enters this widget, and whenever it moves within the widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDragMoveEvent */ void TQGridLayout::dragMoveEvent( TQDragMoveEvent * ) { } /*! This event handler is called when a drag is in progress and the mouse leaves this widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDragLeaveEvent */ void TQGridLayout::dragLeaveEvent( TQDragLeaveEvent * ) { } /*! This event handler is called when the drag is dropped on this widget. See the \link dnd.html Drag-and-drop documentation\endlink for an overview of how to provide drag-and-drop in your application. \sa TQTextDrag, TQImageDrag, TQDropEvent */ void TQGridLayout::dropEvent( TQDropEvent * ) { } #endif // TQT_NO_DRAGANDDROP /*! This event handler can be reimplemented in a subclass to receive widget show events. Non-spontaneous show events are sent to widgets immediately before they are shown. The spontaneous show events of top-level widgets are delivered afterwards. \sa event(), TQShowEvent */ void TQGridLayout::showEvent( TQShowEvent * ) { } /*! This event handler can be reimplemented in a subclass to receive widget hide events. Hide events are sent to widgets immediately after they have been hidden. \sa event(), TQHideEvent */ void TQGridLayout::hideEvent( TQHideEvent * ) { } void TQGridLayout::timerEvent( TQTimerEvent * ) { } void TQGridLayout::childEvent( TQChildEvent * ) { } void TQGridLayout::customEvent( TQCustomEvent * ) { } #endif TQMetaObject *TQLayout::tqmetaObject() const { return TQT_TQOBJECT_CONST(this)->tqmetaObject(); } #else // USE_QT4 #endif // USE_QT4 /* Three internal classes related to TQGridLayout: (1) TQGridBox is a TQLayoutItem with (row, column) information; (2) TQGridMultiBox is a TQGridBox with (torow, tocolumn) information; (3) TQGridLayoutData is the internal representation of a TQGridLayout. */ class TQGridBox { public: TQGridBox( TQLayoutItem *lit ) { item_ = lit; } TQGridBox( TQWidget *wid ) { item_ = new TQWidgetItem( wid ); } TQGridBox( int w, int h, TQSizePolicy::SizeType hData = TQSizePolicy::Minimum, TQSizePolicy::SizeType vData = TQSizePolicy::Minimum ) { item_ = TQT_TQLAYOUTITEM(new TQSpacerItem( w, h, hData, vData )); } ~TQGridBox() { delete item_; } QSize tqsizeHint() const { return item_->sizeHint(); } QSize tqminimumSize() const { return item_->minimumSize(); } QSize tqmaximumSize() const { return item_->maximumSize(); } TQ_SPExpandData expandingDirections() const { return item_->expandingDirections(); } bool isEmpty() const { return item_->isEmpty(); } bool hasHeightForWidth() const { return item_->hasHeightForWidth(); } int heightForWidth( int w ) const { return item_->heightForWidth(w); } void tqsetAlignment( int a ) { item_->tqsetAlignment( a ); } void setGeometry( const TQRect &r ) { item_->setGeometry( r ); } int tqalignment() const { return item_->tqalignment(); } TQLayoutItem *item() { return item_; } TQLayoutItem *takeItem() { TQLayoutItem *i = item_; item_ = 0; return i; } int hStretch() { return item_->widget() ? TQT_TQSIZEPOLICY_OBJECT(item_->widget()->sizePolicy()).horStretch() : 0; } int vStretch() { return item_->widget() ? TQT_TQSIZEPOLICY_OBJECT(item_->widget()->sizePolicy()).verStretch() : 0; } private: friend class TQGridLayoutData; friend class TQGridLayoutDataIterator; TQLayoutItem *item_; int row, col; }; class TQGridMultiBox { public: TQGridMultiBox( TQGridBox *box, int toRow, int toCol ) : box_( box ), torow( toRow ), tocol( toCol ) { } ~TQGridMultiBox() { delete box_; } TQGridBox *box() { return box_; } TQLayoutItem *takeItem() { return box_->takeItem(); } private: friend class TQGridLayoutData; friend class TQGridLayoutDataIterator; TQGridBox *box_; int torow, tocol; }; class TQGridLayoutData { public: TQGridLayoutData(); TQGridLayoutData( int nRows, int nCols ); ~TQGridLayoutData(); void add( TQGridBox*, int row, int col ); void add( TQGridBox*, int row1, int row2, int col1, int col2 ); QSize tqsizeHint( int ) const; QSize tqminimumSize( int ) const; QSize tqmaximumSize( int ) const; TQ_SPExpandData expanding( int spacing ); void distribute( TQRect, int ); inline int numRows() const { return rr; } inline int numCols() const { return cc; } inline void expand( int rows, int cols ) { setSize( TQMAX(rows, rr), TQMAX(cols, cc) ); } inline void setRowStretch( int r, int s ) { expand( r + 1, 0 ); rStretch[r] = s; setDirty(); } inline void setColStretch( int c, int s ) { expand( 0, c + 1 ); cStretch[c] = s; setDirty(); } inline int rowStretch( int r ) const { return rStretch[r]; } inline int colStretch( int c ) const { return cStretch[c]; } inline void setRowSpacing( int r, int s ) { expand( r + 1, 0 ); rSpacing[r] = s; setDirty(); } inline void setColSpacing( int c, int s ) { expand( 0, c + 1 ); cSpacing[c] = s; setDirty(); } inline int rowSpacing( int r ) const { return rSpacing[r]; } inline int colSpacing( int c ) const { return cSpacing[c]; } inline void setReversed( bool r, bool c ) { hReversed = c; vReversed = r; } inline bool horReversed() const { return hReversed; } inline bool verReversed() const { return vReversed; } inline void setDirty() { needRecalc = TRUE; hfw_width = -1; } inline bool isDirty() const { return needRecalc; } bool hasHeightForWidth( int space ); int heightForWidth( int, int, int ); int minimumHeightForWidth( int, int, int ); bool tqfindWidget( TQWidget* w, int *row, int *col ); inline void getNextPos( int &row, int &col ) { row = nextR; col = nextC; } inline uint count() const { return things.count() + ( multi ? multi->count() : 0 ); } TQRect cellGeometry( int row, int col ) const; private: void setNextPosAfter( int r, int c ); void recalcHFW( int w, int s ); void addHfwData ( TQGridBox *box, int width ); void init(); TQSize tqfindSize( TQCOORD TQLayoutStruct::*, int ) const; void addData( TQGridBox *b, bool r = TRUE, bool c = TRUE ); void setSize( int rows, int cols ); void setupLayoutData( int space ); void setupHfwLayoutData( int space ); int rr; int cc; TQMemArray rowData; TQMemArray colData; TQMemArray *hfwData; TQMemArray rStretch; TQMemArray cStretch; TQMemArray rSpacing; TQMemArray cSpacing; TQPtrList things; TQPtrList *multi; int hfw_width; int hfw_height; int hfw_minheight; int nextR; int nextC; uint hReversed : 1; uint vReversed : 1; uint needRecalc : 1; uint has_hfw : 1; uint addVertical : 1; friend class TQGridLayoutDataIterator; }; TQGridLayoutData::TQGridLayoutData() { init(); } TQGridLayoutData::TQGridLayoutData( int nRows, int nCols ) : rowData( 0 ), colData( 0 ) { init(); if ( nRows < 0 ) { nRows = 1; addVertical = FALSE; } if ( nCols < 0 ) { nCols = 1; addVertical = TRUE; } setSize( nRows, nCols ); } TQGridLayoutData::~TQGridLayoutData() { // must be cleared while the data is still in a stable state things.clear(); delete multi; delete hfwData; } void TQGridLayoutData::init() { addVertical = FALSE; setDirty(); multi = 0; rr = cc = 0; nextR = nextC = 0; hfwData = 0; things.setAutoDelete( TRUE ); hReversed = FALSE; vReversed = FALSE; } bool TQGridLayoutData::hasHeightForWidth( int spacing ) { setupLayoutData( spacing ); return has_hfw; } /* Assumes that setupLayoutData() has been called, and that qGeomCalc() has filled in colData with appropriate values. */ void TQGridLayoutData::recalcHFW( int w, int spacing ) { /* Go through all tqchildren, using colData and heightForWidth() and put the results in hfw_rowData. */ if ( !hfwData ) hfwData = new TQMemArray( rr ); setupHfwLayoutData( spacing ); TQMemArray &rData = *hfwData; int h = 0; int mh = 0; int n = 0; for ( int r = 0; r < rr; r++ ) { h += rData[r].tqsizeHint; mh += rData[r].tqminimumSize; if ( !rData[r].empty ) n++; } if ( n ) { h += ( n - 1 ) * spacing; mh += ( n - 1 ) * spacing; } hfw_width = w; hfw_height = TQMIN( TQLAYOUTSIZE_MAX, h ); hfw_minheight = TQMIN( TQLAYOUTSIZE_MAX, mh ); } int TQGridLayoutData::heightForWidth( int w, int margin, int spacing ) { setupLayoutData( spacing ); if ( !has_hfw ) return -1; if ( w + 2*margin != hfw_width ) { qGeomCalc( colData, 0, cc, 0, w+2*margin, spacing ); recalcHFW( w+2*margin, spacing ); } return hfw_height + 2*margin; } int TQGridLayoutData::minimumHeightForWidth( int w, int margin, int spacing ) { (void) heightForWidth( w, margin, spacing ); return has_hfw ? (hfw_minheight + 2*margin) : -1; } bool TQGridLayoutData::tqfindWidget( TQWidget* w, int *row, int *col ) { TQPtrListIterator it( things ); TQGridBox * box; while ( (box = it.current()) != 0 ) { ++it; if ( box->item()->widget() == w ) { if ( row ) *row = box->row; if ( col ) *col = box->col; return TRUE; } } if ( multi ) { TQPtrListIterator it( *multi ); TQGridMultiBox * mbox; while ( (mbox = it.current()) != 0 ) { ++it; box = mbox->box(); if ( box->item()->widget() == w ) { if ( row ) *row = box->row; if ( col ) *col = box->col; return TRUE; } } } return FALSE; } TQSize TQGridLayoutData::tqfindSize( TQCOORD TQLayoutStruct::*size, int spacer ) const { TQGridLayoutData *that = (TQGridLayoutData *)this; that->setupLayoutData( spacer ); int w = 0; int h = 0; int n = 0; for ( int r = 0; r < rr; r++ ) { h = h + rowData[r].*size; if ( !rowData[r].empty ) n++; } if ( n ) h += ( n - 1 ) * spacer; n = 0; for ( int c = 0; c < cc; c++ ) { w = w + colData[c].*size; if ( !colData[c].empty ) n++; } if ( n ) w += ( n - 1 ) * spacer; w = TQMIN( TQLAYOUTSIZE_MAX, w ); h = TQMIN( TQLAYOUTSIZE_MAX, h ); return TQSize( w, h ); } TQ_SPExpandData TQGridLayoutData::expanding( int spacing ) { setupLayoutData( spacing ); int ret = 0; for ( int r = 0; r < rr; r++ ) { if ( rowData[r].expansive ) { ret |= (int) TQSizePolicy::Vertically; break; } } for ( int c = 0; c < cc; c++ ) { if ( colData[c].expansive ) { ret |= (int) TQSizePolicy::Horizontally; break; } } return (TQ_SPExpandData) ret; } QSize TQGridLayoutData::tqsizeHint( int spacer ) const { return tqfindSize( &TQLayoutStruct::tqsizeHint, spacer ); } QSize TQGridLayoutData::tqmaximumSize( int spacer ) const { return tqfindSize( &TQLayoutStruct::tqmaximumSize, spacer ); } QSize TQGridLayoutData::tqminimumSize( int spacer ) const { return tqfindSize( &TQLayoutStruct::tqminimumSize, spacer ); } void TQGridLayoutData::setSize( int r, int c ) { if ( (int)rowData.size() < r ) { int newR = TQMAX( r, rr * 2 ); rowData.resize( newR ); rStretch.resize( newR ); rSpacing.resize( newR ); for ( int i = rr; i < newR; i++ ) { rowData[i].init(); rStretch[i] = 0; rSpacing[i] = 0; } } if ( (int)colData.size() < c ) { int newC = TQMAX( c, cc * 2 ); colData.resize( newC ); cStretch.resize( newC ); cSpacing.resize( newC ); for ( int i = cc; i < newC; i++ ) { colData[i].init(); cStretch[i] = 0; cSpacing[i] = 0; } } if ( hfwData && (int)hfwData->size() < r ) { delete hfwData; hfwData = 0; hfw_width = -1; } rr = r; cc = c; } void TQGridLayoutData::setNextPosAfter( int row, int col ) { if ( addVertical ) { if ( col > nextC || (col == nextC && row >= nextR) ) { nextR = row + 1; nextC = col; if ( nextR >= rr ) { nextR = 0; nextC++; } } } else { if ( row > nextR || (row == nextR && col >= nextC) ) { nextR = row; nextC = col + 1; if ( nextC >= cc ) { nextC = 0; nextR++; } } } } void TQGridLayoutData::add( TQGridBox *box, int row, int col ) { expand( row+1, col+1 ); box->row = row; box->col = col; things.append( box ); setDirty(); setNextPosAfter( row, col ); } void TQGridLayoutData::add( TQGridBox *box, int row1, int row2, int col1, int col2 ) { #ifdef TQT_CHECK_RANGE if ( row2 >= 0 && row2 < row1 ) qWarning( "TQGridLayout: Multi-cell fromRow greater than toRow" ); if ( col2 >= 0 && col2 < col1 ) qWarning( "TQGridLayout: Multi-cell fromCol greater than toCol" ); #endif if ( row1 == row2 && col1 == col2 ) { add( box, row1, col1 ); return; } expand( row2+1, col2+1 ); box->row = row1; box->col = col1; TQGridMultiBox *mbox = new TQGridMultiBox( box, row2, col2 ); if ( !multi ) { multi = new TQPtrList; multi->setAutoDelete( TRUE ); } multi->append( mbox ); setDirty(); if ( col2 < 0 ) col2 = cc - 1; setNextPosAfter( row2, col2 ); } void TQGridLayoutData::addData( TQGridBox *box, bool r, bool c ) { TQSize hint = box->tqsizeHint(); TQSize minS = box->tqminimumSize(); TQSize maxS = box->tqmaximumSize(); if ( c ) { if ( !cStretch[box->col] ) colData[box->col].stretch = TQMAX( colData[box->col].stretch, box->hStretch() ); colData[box->col].tqsizeHint = TQMAX( hint.width(), colData[box->col].tqsizeHint ); colData[box->col].tqminimumSize = TQMAX( minS.width(), colData[box->col].tqminimumSize ); qMaxExpCalc( colData[box->col].tqmaximumSize, colData[box->col].expansive, maxS.width(), box->expandingDirections() & TQSizePolicy::Horizontally ); } if ( r ) { if ( !rStretch[box->row] ) rowData[box->row].stretch = TQMAX( rowData[box->row].stretch, box->vStretch() ); rowData[box->row].tqsizeHint = TQMAX( hint.height(), rowData[box->row].tqsizeHint ); rowData[box->row].tqminimumSize = TQMAX( minS.height(), rowData[box->row].tqminimumSize ); qMaxExpCalc( rowData[box->row].tqmaximumSize, rowData[box->row].expansive, maxS.height(), box->expandingDirections() & TQSizePolicy::Vertically ); } if ( box->isEmpty() ) { if ( box->item()->widget() != 0 ) { /* Hidden widgets should behave exactly the same as if there were no widgets at all in the cell. We could have TQWidgetItem::tqmaximumSize() to return TQSize(TQLAYOUTSIZE_MAX, TQLAYOUTSIZE_MAX). However, that value is not suitable for TQBoxLayouts. So let's fix it here. */ if ( c ) colData[box->col].tqmaximumSize = TQLAYOUTSIZE_MAX; if ( r ) rowData[box->row].tqmaximumSize = TQLAYOUTSIZE_MAX; } } else { // Empty boxes (i.e. spacers) do not get borders. This is // hacky, but compatible. if ( c ) colData[box->col].empty = FALSE; if ( r ) rowData[box->row].empty = FALSE; } } static void distributeMultiBox( TQMemArray &chain, int spacing, int start, int end, int minSize, int tqsizeHint, TQMemArray &stretchArray, int stretch ) { int i; int w = 0; int wh = 0; int max = 0; for ( i = start; i <= end; i++ ) { w += chain[i].tqminimumSize; wh += chain[i].tqsizeHint; max += chain[i].tqmaximumSize; chain[i].empty = FALSE; if ( stretchArray[i] == 0 ) chain[i].stretch = TQMAX(chain[i].stretch,stretch); } w += spacing * ( end - start ); wh += spacing * ( end - start ); max += spacing * ( end - start ); if ( max < minSize ) { // implies w < minSize /* We must increase the maximum size of at least one of the items. qGeomCalc() will put the extra space in between the items. We must recover that extra space and put it somewhere. It does not really matter where, since the user can always specify stretch factors and avoid this code. */ qGeomCalc( chain, start, end - start + 1, 0, minSize, spacing ); int pos = 0; for ( i = start; i <= end; i++ ) { int nextPos = ( i == end ) ? minSize - 1 : chain[i + 1].pos; int realSize = nextPos - pos; if ( i != end ) realSize -= spacing; if ( chain[i].tqminimumSize < realSize ) chain[i].tqminimumSize = realSize; if ( chain[i].tqmaximumSize < chain[i].tqminimumSize ) chain[i].tqmaximumSize = chain[i].tqminimumSize; pos = nextPos; } } else if ( w < minSize ) { qGeomCalc( chain, start, end - start + 1, 0, minSize, spacing ); for ( i = start; i <= end; i++ ) { if ( chain[i].tqminimumSize < chain[i].size ) chain[i].tqminimumSize = chain[i].size; } } if ( wh < tqsizeHint ) { qGeomCalc( chain, start, end - start + 1, 0, tqsizeHint, spacing ); for ( i = start; i <= end; i++ ) { if ( chain[i].tqsizeHint < chain[i].size ) chain[i].tqsizeHint = chain[i].size; } } } //#define TQT_LAYOUT_DISABLE_CACHING void TQGridLayoutData::setupLayoutData( int spacing ) { #ifndef TQT_LAYOUT_DISABLE_CACHING if ( !needRecalc ) return; #endif has_hfw = FALSE; int i; for ( i = 0; i < rr; i++ ) rowData[i].init( rStretch[i], rSpacing[i] ); for ( i = 0; i < cc; i++ ) colData[i].init( cStretch[i], cSpacing[i] ); TQPtrListIterator it( things ); TQGridBox * box; while ( (box = it.current()) != 0 ) { ++it; addData( box ); has_hfw = has_hfw || box->item()->hasHeightForWidth(); } if ( multi ) { TQPtrListIterator it( *multi ); TQGridMultiBox * mbox; while ( (mbox = it.current()) != 0 ) { ++it; TQGridBox *box = mbox->box(); int r1 = box->row; int c1 = box->col; int r2 = mbox->torow; int c2 = mbox->tocol; if ( r2 < 0 ) r2 = rr - 1; if ( c2 < 0 ) c2 = cc - 1; TQSize hint = box->tqsizeHint(); TQSize min = box->tqminimumSize(); if ( box->hasHeightForWidth() ) has_hfw = TRUE; if ( r1 == r2 ) { addData( box, TRUE, FALSE ); } else { distributeMultiBox( rowData, spacing, r1, r2, min.height(), hint.height(), rStretch, box->vStretch() ); } if ( c1 == c2 ) { addData( box, FALSE, TRUE ); } else { distributeMultiBox( colData, spacing, c1, c2, min.width(), hint.width(), cStretch, box->hStretch() ); } } } for ( i = 0; i < rr; i++ ) rowData[i].expansive = rowData[i].expansive || rowData[i].stretch > 0; for ( i = 0; i < cc; i++ ) colData[i].expansive = colData[i].expansive || colData[i].stretch > 0; needRecalc = FALSE; } void TQGridLayoutData::addHfwData( TQGridBox *box, int width ) { TQMemArray &rData = *hfwData; if ( box->hasHeightForWidth() ) { int hint = box->heightForWidth( width ); rData[box->row].tqsizeHint = TQMAX( hint, rData[box->row].tqsizeHint ); rData[box->row].tqminimumSize = TQMAX( hint, rData[box->row].tqminimumSize ); } else { TQSize hint = box->tqsizeHint(); TQSize minS = box->tqminimumSize(); rData[box->row].tqsizeHint = TQMAX( hint.height(), rData[box->row].tqsizeHint ); rData[box->row].tqminimumSize = TQMAX( minS.height(), rData[box->row].tqminimumSize ); } } /* Similar to setupLayoutData(), but uses heightForWidth(colData) instead of tqsizeHint(). Assumes that setupLayoutData() and qGeomCalc(colData) has been called. */ void TQGridLayoutData::setupHfwLayoutData( int spacing ) { TQMemArray &rData = *hfwData; int i; for ( i = 0; i < rr; i++ ) { rData[i] = rowData[i]; rData[i].tqminimumSize = rData[i].tqsizeHint = 0; } TQPtrListIterator it( things ); TQGridBox * box; while ( (box=it.current()) != 0 ) { ++it; addHfwData( box, colData[box->col].size ); } if ( multi ) { TQPtrListIterator it( *multi ); TQGridMultiBox * mbox; while ( (mbox=it.current()) != 0 ) { ++it; TQGridBox *box = mbox->box(); int r1 = box->row; int c1 = box->col; int r2 = mbox->torow; int c2 = mbox->tocol; if ( r2 < 0 ) r2 = rr-1; if ( c2 < 0 ) c2 = cc-1; int w = colData[c2].pos + colData[c2].size - colData[c1].pos; if ( r1 == r2 ) { addHfwData( box, w ); } else { TQSize hint = box->tqsizeHint(); TQSize min = box->tqminimumSize(); if ( box->hasHeightForWidth() ) { int hfwh = box->heightForWidth( w ); if ( hfwh > hint.height() ) hint.setHeight( hfwh ); if ( hfwh > min.height() ) min.setHeight( hfwh ); } distributeMultiBox( rData, spacing, r1, r2, min.height(), hint.height(), rStretch, box->vStretch() ); } } } for ( i = 0; i < rr; i++ ) rData[i].expansive = rData[i].expansive || rData[i].stretch > 0; } void TQGridLayoutData::distribute( TQRect r, int spacing ) { bool visualHReversed = hReversed; if ( TQApplication::reverseLayout() ) visualHReversed = !visualHReversed; setupLayoutData( spacing ); qGeomCalc( colData, 0, cc, r.x(), r.width(), spacing ); TQMemArray *rDataPtr; if ( has_hfw ) { recalcHFW( r.width(), spacing ); qGeomCalc( *hfwData, 0, rr, r.y(), r.height(), spacing ); rDataPtr = hfwData; } else { qGeomCalc( rowData, 0, rr, r.y(), r.height(), spacing ); rDataPtr = &rowData; } TQMemArray &rData = *rDataPtr; TQPtrListIterator it( things ); TQGridBox * box; while ( (box=it.current()) != 0 ) { ++it; int x = colData[box->col].pos; int y = rData[box->row].pos; int w = colData[box->col].size; int h = rData[box->row].size; if ( visualHReversed ) x = r.left() + r.right() - x - w + 1; if ( vReversed ) y = r.top() + r.bottom() - y - h + 1; box->setGeometry( TQRect( x, y, w, h ) ); } if ( multi ) { TQPtrListIterator it( *multi ); TQGridMultiBox * mbox; while ( (mbox=it.current()) != 0 ) { ++it; TQGridBox *box = mbox->box(); int r2 = mbox->torow; int c2 = mbox->tocol; if ( r2 < 0 ) r2 = rr-1; if ( c2 < 0 ) c2 = cc-1; int x = colData[box->col].pos; int y = rData[box->row].pos; int x2p = colData[c2].pos + colData[c2].size; // x2+1 int y2p = rData[r2].pos + rData[r2].size; // y2+1 int w = x2p - x; int h = y2p - y; // this code is copied from above: if ( visualHReversed ) x = r.left() + r.right() - x - w + 1; if ( vReversed ) y = r.top() + r.bottom() - y - h + 1; box->setGeometry( TQRect( x, y, w, h ) ); } } } TQRect TQGridLayoutData::cellGeometry( int row, int col ) const { if ( row < 0 || row >= rr || col < 0 || col >= cc ) return TQRect(); const TQMemArray *rDataPtr; if ( has_hfw ) rDataPtr = hfwData; else rDataPtr = &rowData; return TQRect( colData[col].pos, (*rDataPtr)[row].pos, colData[col].size, (*rDataPtr)[row].size ); } class TQGridLayoutDataIterator : public TQGLayoutIterator { public: TQGridLayoutDataIterator( TQGridLayoutData *d ); uint count() const { return data->count(); } TQLayoutItem *current() { if ( multi ) { if ( !data->multi || idx >= (int)data->multi->count() ) return 0; return data->multi->at( idx )->box()->item(); } else { if ( idx >= (int)data->things.count() ) return 0; return data->things.at( idx )->item(); } } void toFirst() { multi = data->things.isEmpty(); idx = 0; } TQLayoutItem *next() { idx++; if ( !multi && idx >= (int)data->things.count() ) { multi = TRUE; idx = 0; } return current(); } TQLayoutItem *takeCurrent() { TQLayoutItem *item = 0; if ( multi ) { if ( !data->multi || idx >= (int)data->multi->count() ) return 0; TQGridMultiBox *b = data->multi->take( idx ); item = b->takeItem(); delete b; } else { if ( idx >= (int)data->things.count() ) return 0; TQGridBox *b = data->things.take( idx ); item = b->takeItem(); delete b; } return item; } private: TQGridLayoutData *data; bool multi; int idx; }; inline TQGridLayoutDataIterator::TQGridLayoutDataIterator( TQGridLayoutData *d ) : data( d ) { toFirst(); } /*! \class TQGridLayout \brief The TQGridLayout class lays out widgets in a grid. \ingroup geomanagement \ingroup appearance \mainclass TQGridLayout takes the space made available to it (by its tqparent tqlayout or by the mainWidget()), divides it up into rows and columns, and puts each widget it manages into the correct cell. Columns and rows behave identically; we will discuss columns, but there are equivalent functions for rows. Each column has a minimum width and a stretch factor. The minimum width is the greatest of that set using addColSpacing() and the minimum width of each widget in that column. The stretch factor is set using setColStretch() and determines how much of the available space the column will get over and above its necessary minimum. Normally, each managed widget or tqlayout is put into a cell of its own using addWidget(), addLayout() or by the \link TQLayout::setAutoAdd() auto-add facility\endlink. It is also possible for a widget to occupy multiple cells using addMultiCellWidget(). If you do this, TQGridLayout will guess how to distribute the size over the columns/rows (based on the stretch factors). To remove a widget from a tqlayout, call remove(). Calling TQWidget::hide() on a widget also effectively removes the widget from the tqlayout until TQWidget::show() is called. This illustration shows a fragment of a dialog with a five-column, three-row grid (the grid is shown overlaid in magenta): \img gridtqlayout.png Columns 0, 2 and 4 in this dialog fragment are made up of a TQLabel, a TQLineEdit, and a TQListBox. Columns 1 and 3 are placeholders made with addColSpacing(). Row 0 consists of three TQLabel objects, row 1 of three TQLineEdit objects and row 2 of three TQListBox objects. We used placeholder columns (1 and 3) to get the right amount of space between the columns. Note that the columns and rows are not equally wide or tall. If you want two columns to have the same width, you must set their minimum widths and stretch factors to be the same yourself. You do this using addColSpacing() and setColStretch(). If the TQGridLayout is not the top-level tqlayout (i.e. does not manage all of the widget's area and tqchildren), you must add it to its tqparent tqlayout when you create it, but before you do anything with it. The normal way to add a tqlayout is by calling parentLayout-\>addLayout(). Once you have added your tqlayout you can start putting widgets and other layouts into the cells of your grid tqlayout using addWidget(), addLayout() and addMultiCellWidget(). TQGridLayout also includes two margin widths: the border and the spacing. The border is the width of the reserved space along each of the TQGridLayout's four sides. The spacing is the width of the automatically allocated spacing between neighboring boxes. Both the border and the spacing are parameters of the constructor and default to 0. \sa TQGrid, \link tqlayout.html Layout Overview \endlink */ /* Returns TRUE if the widget \a w can be added to the tqlayout \a l; otherwise returns FALSE. */ static bool checkWidget( TQLayout *l, TQWidget *w ) { if ( !w ) { #if defined(TQT_CHECK_STATE) qWarning( "TQLayout: Cannot add null widget to %s/%s", l->className(), l->name() ); #endif return FALSE; } if ( w->parentWidget() != l->mainWidget() && l->mainWidget() ) { #if defined(TQT_CHECK_STATE) if ( w->parentWidget() ) qWarning( "TQLayout: Adding %s/%s (child of %s/%s) to tqlayout for " "%s/%s", w->className(), w->name(), w->parentWidget()->className(), w->parentWidget()->name(), l->mainWidget()->className(), l->mainWidget()->name() ); else qWarning( "TQLayout: Adding %s/%s (top-level widget) to tqlayout for" " %s/%s", w->className(), w->name(), l->mainWidget()->className(), l->mainWidget()->name() ); #endif return FALSE; } return TRUE; } // #ifdef USE_QT4 // // #else // USE_QT4 /*! \enum TQGridLayout::Corner This enum identifies which corner is the origin (0, 0) of the tqlayout. \value TopLeft the top-left corner \value TopRight the top-right corner \value BottomLeft the bottom-left corner \value BottomRight the bottom-right corner */ /*! Constructs a new TQGridLayout with \a nRows rows, \a nCols columns and tqparent widget, \a tqparent. \a tqparent may not be 0. The grid tqlayout is called \a name. \a margin is the number of pixels between the edge of the widget and its managed tqchildren. \a space is the default number of pixels between cells. If \a space is -1, the value of \a margin is used. */ TQGridLayout::TQGridLayout( TQWidget *tqparent, int nRows, int nCols, int margin, int space, const char *name ) : TQLayout( tqparent, margin, space, name ) { init( nRows, nCols ); } /*! Constructs a new grid that is placed inside \a parentLayout with \a nRows rows and \a nCols columns. If \a spacing is -1, this TQGridLayout inherits its tqparent's spacing(); otherwise \a spacing is used. The grid tqlayout is called \a name. This grid is placed according to \a parentLayout's default placement rules. */ TQGridLayout::TQGridLayout( TQLayout *parentLayout, int nRows, int nCols, int spacing, const char *name ) : TQLayout( parentLayout, spacing, name ) { init( nRows, nCols ); } /*! Constructs a new grid with \a nRows rows and \a nCols columns. If \a spacing is -1, this TQGridLayout inherits its tqparent's spacing(); otherwise \a spacing is used. The grid tqlayout is called \a name. You must insert this grid into another tqlayout. You can insert widgets and layouts into this tqlayout at any time, but laying out will not be performed before this is inserted into another tqlayout. */ TQGridLayout::TQGridLayout( int nRows, int nCols, int spacing, const char *name ) : TQLayout( spacing, name ) { init( nRows, nCols ); } /*! Destroys the grid tqlayout. Geometry management is terminated if this is a top-level grid. The tqlayout's widgets aren't destroyed. */ TQGridLayout::~TQGridLayout() { delete data; } /*! Returns the number of rows in this grid. */ int TQGridLayout::numRows() const { return data->numRows(); } /*! Returns the number of columns in this grid. */ int TQGridLayout::numCols() const { return data->numCols(); } /*! Returns the preferred size of this grid. */ TQSize TQGridLayout::tqsizeHint() const { return data->tqsizeHint( spacing() ) + TQSize( 2 * margin(), 2 * margin() ); } /*! Returns the minimum size needed by this grid. */ TQSize TQGridLayout::tqminimumSize() const { return data->tqminimumSize( spacing() ) + TQSize( 2 * margin(), 2 * margin() ); } /*! Returns the maximum size needed by this grid. */ TQSize TQGridLayout::tqmaximumSize() const { TQSize s = data->tqmaximumSize( spacing() ) + TQSize( 2 * margin(), 2 * margin() ); s = s.boundedTo( TQSize(TQLAYOUTSIZE_MAX, TQLAYOUTSIZE_MAX) ); if ( tqalignment() & TQt::AlignHorizontal_Mask ) s.setWidth( TQLAYOUTSIZE_MAX ); if ( tqalignment() & TQt::AlignVertical_Mask ) s.setHeight( TQLAYOUTSIZE_MAX ); return s; } /*! Returns TRUE if this tqlayout's preferred height depends on its width; otherwise returns FALSE. */ bool TQGridLayout::hasHeightForWidth() const { return ((TQGridLayout*)this)->data->hasHeightForWidth( spacing() ); } /*! Returns the tqlayout's preferred height when it is \a w pixels wide. */ int TQGridLayout::heightForWidth( int w ) const { TQGridLayout *that = (TQGridLayout*)this; return that->data->heightForWidth( w, margin(), spacing() ); } /*! \internal */ int TQGridLayout::minimumHeightForWidth( int w ) const { TQGridLayout *that = (TQGridLayout*)this; return that->data->minimumHeightForWidth( w, margin(), spacing() ); } /*! Searches for widget \a w in this tqlayout (not including child layouts). If \a w is found, it sets \c \a row and \c \a col to the row and column and returns TRUE; otherwise returns FALSE. Note: if a widget spans multiple rows/columns, the top-left cell is returned. */ bool TQGridLayout::tqfindWidget( TQWidget* w, int *row, int *col ) { return data->tqfindWidget( w, row, col ); } /*! Resizes managed widgets within the rectangle \a r. */ void TQGridLayout::setGeometry( const TQRect &r ) { if ( data->isDirty() || r != tqgeometry() ) { TQLayout::setGeometry( r ); TQRect cr = tqalignment() ? alignmentRect( r ) : r; TQRect s( cr.x() + margin(), cr.y() + margin(), cr.width() - 2 * margin(), cr.height() - 2 * margin() ); data->distribute( s, spacing() ); } } /*! Returns the tqgeometry of the cell with row \a row and column \a col in the grid. Returns an invalid rectangle if \a row or \a col is outside the grid. \warning in the current version of TQt this function does not return valid results until setGeometry() has been called, i.e. after the mainWidget() is visible. */ TQRect TQGridLayout::cellGeometry( int row, int col ) const { return data->cellGeometry( row, col ); } /*! Expands this grid so that it will have \a nRows rows and \a nCols columns. Will not shrink the grid. You should not need to call this function because TQGridLayout expands automatically as new items are inserted. */ void TQGridLayout::expand( int nRows, int nCols ) { data->expand( nRows, nCols ); } /*! Sets up the grid. */ void TQGridLayout::init( int nRows, int nCols ) { setSupportsMargin( TRUE ); data = new TQGridLayoutData( nRows, nCols ); } /*! \overload Adds \a item to the next free position of this tqlayout. */ void TQGridLayout::addItem( QLayoutItem *item ) { int r, c; data->getNextPos( r, c ); add( TQT_TQLAYOUTITEM(item), r, c ); } /*! Adds \a item at position \a row, \a col. The tqlayout takes ownership of the \a item. */ void TQGridLayout::addItem( QLayoutItem *item, int row, int col ) { add( TQT_TQLAYOUTITEM(item), row, col ); } /*! Adds \a item at position \a row, \a col. The tqlayout takes ownership of the \a item. */ void TQGridLayout::add( TQLayoutItem *item, int row, int col ) { TQGridBox *box = new TQGridBox( item ); data->add( box, row, col ); } /*! Adds the \a item to the cell grid, spanning multiple rows/columns. The cell will span from \a fromRow, \a fromCol to \a toRow, \a toCol. Alignment is specified by \a tqalignment, which is a bitwise OR of \l TQt::AlignmentFlags values. The default tqalignment is 0, which means that the widget fills the entire cell. */ void TQGridLayout::addMultiCell( QLayoutItem *item, int fromRow, int toRow, int fromCol, int toCol, int tqalignment ) { TQGridBox *b = new TQGridBox( TQT_TQLAYOUTITEM(item) ); b->tqsetAlignment( tqalignment ); data->add( b, fromRow, toRow, fromCol, toCol ); } /*! Adds the widget \a w to the cell grid at \a row, \a col. The top-left position is (0, 0) by default. Alignment is specified by \a tqalignment, which is a bitwise OR of \l TQt::AlignmentFlags values. The default tqalignment is 0, which means that the widget fills the entire cell. \list \i You should not call this if you have enabled the \link TQLayout::setAutoAdd() auto-add facility of the tqlayout\endlink. \i From TQt 3.0, the \a tqalignment parameter is interpreted more aggressively than in previous versions of TQt. A non-default tqalignment now indicates that the widget should not grow to fill the available space, but should be sized according to tqsizeHint(). \endlist \sa addMultiCellWidget() */ void TQGridLayout::addWidget( TQWidget *w, int row, int col, int tqalignment ) { if ( !checkWidget( this, w ) ) return; if ( row < 0 || col < 0 ) { #if defined(TQT_CHECK_STATE) qWarning( "TQGridLayout: Cannot add %s/%s to %s/%s at row %d col %d", w->className(), w->name(), className(), name(), row, col ); #endif return; } TQWidgetItem *b = new TQWidgetItem( w ); b->tqsetAlignment( tqalignment ); add( b, row, col ); } /*! Adds the widget \a w to the cell grid, spanning multiple rows/columns. The cell will span from \a fromRow, \a fromCol to \a toRow, \a toCol. Alignment is specified by \a tqalignment, which is a bitwise OR of \l TQt::AlignmentFlags values. The default tqalignment is 0, which means that the widget fills the entire cell. A non-zero tqalignment indicates that the widget should not grow to fill the available space but should be sized according to tqsizeHint(). \sa addWidget() */ void TQGridLayout::addMultiCellWidget( TQWidget *w, int fromRow, int toRow, int fromCol, int toCol, int tqalignment ) { TQGridBox *b = new TQGridBox( w ); b->tqsetAlignment( tqalignment ); data->add( b, fromRow, toRow, fromCol, toCol ); } /*! Places the \a tqlayout at position (\a row, \a col) in the grid. The top-left position is (0, 0). \a tqlayout becomes a child of the grid tqlayout. When a tqlayout is constructed with another tqlayout as its tqparent, you don't need to call addLayout(); the child tqlayout is automatically added to the tqparent tqlayout as it is constructed. \sa addMultiCellLayout() */ void TQGridLayout::addLayout( TQLayout *tqlayout, int row, int col ) { addChildLayout( tqlayout ); add( TQT_TQLAYOUTITEM(tqlayout), row, col ); } /*! Adds the tqlayout \a tqlayout to the cell grid, spanning multiple rows/columns. The cell will span from \a fromRow, \a fromCol to \a toRow, \a toCol. Alignment is specified by \a tqalignment, which is a bitwise OR of \l TQt::AlignmentFlags values. The default tqalignment is 0, which means that the widget fills the entire cell. A non-zero tqalignment indicates that the tqlayout should not grow to fill the available space but should be sized according to tqsizeHint(). \a tqlayout becomes a child of the grid tqlayout. \sa addLayout() */ void TQGridLayout::addMultiCellLayout( TQLayout *tqlayout, int fromRow, int toRow, int fromCol, int toCol, int tqalignment ) { addChildLayout( tqlayout ); TQGridBox *b = new TQGridBox( TQT_TQLAYOUTITEM(tqlayout) ); b->tqsetAlignment( tqalignment ); data->add( b, fromRow, toRow, fromCol, toCol ); } /*! Sets the stretch factor of row \a row to \a stretch. The first row is number 0. The stretch factor is relative to the other rows in this grid. Rows with a higher stretch factor take more of the available space. The default stretch factor is 0. If the stretch factor is 0 and no other row in this table can grow at all, the row may still grow. \sa rowStretch(), setRowSpacing(), setColStretch() */ void TQGridLayout::setRowStretch( int row, int stretch ) { data->setRowStretch( row, stretch ); } /*! Returns the stretch factor for row \a row. \sa setRowStretch() */ int TQGridLayout::rowStretch( int row ) const { return data->rowStretch( row ); } /*! Returns the stretch factor for column \a col. \sa setColStretch() */ int TQGridLayout::colStretch( int col ) const { return data->colStretch( col ); } /*! Sets the stretch factor of column \a col to \a stretch. The first column is number 0. The stretch factor is relative to the other columns in this grid. Columns with a higher stretch factor take more of the available space. The default stretch factor is 0. If the stretch factor is 0 and no other column in this table can grow at all, the column may still grow. \sa colStretch(), addColSpacing(), setRowStretch() */ void TQGridLayout::setColStretch( int col, int stretch ) { data->setColStretch( col, stretch ); } #if TQT_VERSION >= 0x040000 #error "Make add{Row,Col}Spacing() inline TQT_NO_COMPAT functions defined in terms of set{Row,Col}Spacing()" #endif /*! \obsolete Sets the minimum height of row \a row to \a minsize pixels. Use setRowSpacing() instead. */ void TQGridLayout::addRowSpacing( int row, int minsize ) { TQLayoutItem *b = TQT_TQLAYOUTITEM(new TQSpacerItem( 0, minsize )); add( b, row, 0 ); } /*! \obsolete Sets the minimum width of column \a col to \a minsize pixels. Use setColSpacing() instead. */ void TQGridLayout::addColSpacing( int col, int minsize ) { TQLayoutItem *b = TQT_TQLAYOUTITEM(new TQSpacerItem( minsize, 0 )); add( b, 0, col ); } /*! Sets the minimum height of row \a row to \a minSize pixels. \sa rowSpacing(), setColSpacing() */ void TQGridLayout::setRowSpacing( int row, int minSize ) { data->setRowSpacing( row, minSize ); } /*! Returns the row spacing for row \a row. \sa setRowSpacing() */ int TQGridLayout::rowSpacing( int row ) const { return data->rowSpacing( row ); } /*! Sets the minimum width of column \a col to \a minSize pixels. \sa colSpacing(), setRowSpacing() */ void TQGridLayout::setColSpacing( int col, int minSize ) { data->setColSpacing( col, minSize ); } /*! Returns the column spacing for column \a col. \sa setColSpacing() */ int TQGridLayout::colSpacing( int col ) const { return data->colSpacing( col ); } /*! Returns whether this tqlayout can make use of more space than tqsizeHint(). A value of \c Vertical or \c Horizontal means that it wants to grow in only one dimension, whereas \c BothDirections means that it wants to grow in both dimensions. */ TQ_SPExpandData TQGridLayout::expandingDirections() const { return data->expanding( spacing() ); } /*! Sets the grid's origin corner, i.e. position (0, 0), to \a c. */ void TQGridLayout::setOrigin( Corner c ) { data->setReversed( c == BottomLeft || c == BottomRight, c == TopRight || c == BottomRight ); } /*! Returns the corner that's used for the grid's origin, i.e. for position (0, 0). */ TQGridLayout::Corner TQGridLayout::origin() const { if ( data->horReversed() ) { return data->verReversed() ? BottomRight : TopRight; } else { return data->verReversed() ? BottomLeft : TopLeft; } } /*! Resets cached information. */ void TQGridLayout::tqinvalidate() { TQLayout::tqinvalidate(); TQLayout::setGeometry( TQRect() ); // for binary compatibility (?) data->setDirty(); } /*! \reimp */ // TQLayoutIterator TQGridLayout::iterator() // { // return TQLayoutIterator( new TQGridLayoutDataIterator(data) ); // } #ifdef USE_QT4 /*! \reimp */ int TQGridLayout::count() const { return data->count(); } /*! \reimp */ TQLayoutItem* TQGridLayout::itemAt(int index) const { TQGridLayoutDataIterator tqgli = TQGridLayoutDataIterator(data); for (int i=0; i= 0 && index < data->count() ? tqgli.current() : 0; } /*! \reimp */ TQLayoutItem* TQGridLayout::takeAt(int index) { if (index < 0 || index >= data->count()) return 0; TQGridLayoutDataIterator tqgli = TQGridLayoutDataIterator(data); for (int i=0; ihasHeightForWidth() ) { return item->heightForWidth( w ); } else { return item->sizeHint().height(); } } int mhfw( int w ) { if ( item->hasHeightForWidth() ) { return item->heightForWidth( w ); } else { return item->minimumSize().height(); } } int hStretch() { if ( stretch == 0 && item->widget() ) { return TQT_TQSIZEPOLICY_OBJECT(item->widget()->sizePolicy()).horStretch(); } else { return stretch; } } int vStretch() { if ( stretch == 0 && item->widget() ) { return TQT_TQSIZEPOLICY_OBJECT(item->widget()->sizePolicy()).verStretch(); } else { return stretch; } } TQLayoutItem *item; int stretch; bool magic; }; class TQBoxLayoutData { public: TQBoxLayoutData() : geomArray( 0 ), hfwWidth( -1 ), dirty( TRUE ) { list.setAutoDelete( TRUE ); } ~TQBoxLayoutData() { delete geomArray; } void setDirty() { delete geomArray; geomArray = 0; hfwWidth = -1; hfwHeight = -1; dirty = TRUE; } TQPtrList list; TQMemArray *geomArray; int hfwWidth; int hfwHeight; int hfwMinHeight; TQSize tqsizeHint; TQSize minSize; TQSize maxSize; TQ_SPExpandData expanding; uint hasHfw : 1; uint dirty : 1; }; class TQBoxLayoutIterator : public TQGLayoutIterator { public: TQBoxLayoutIterator( TQBoxLayoutData *d ) : data( d ), idx( 0 ) {} TQLayoutItem *current() { if ( idx >= int(data->list.count()) ) return 0; return data->list.at(idx)->item; } TQLayoutItem *next() { idx++; return current(); } TQLayoutItem *takeCurrent() { TQLayoutItem *item = 0; TQBoxLayoutItem *b = data->list.take( idx ); if ( b ) { item = b->item; b->item = 0; delete b; } data->setDirty(); return item; } private: TQBoxLayoutData *data; int idx; }; /*! \class TQBoxLayout \brief The TQBoxLayout class lines up child widgets horizontally or vertically. \ingroup geomanagement \ingroup appearance TQBoxLayout takes the space it gets (from its tqparent tqlayout or from the mainWidget()), divides it up into a row of boxes, and makes each managed widget fill one box. \img qhbox-m.png Horizontal box with five child widgets If the TQBoxLayout's orientation is \c Horizontal the boxes are placed in a row, with suitable sizes. Each widget (or other box) will get at least its minimum size and at most its maximum size. Any excess space is shared according to the stretch factors (more about that below). \img qvbox-m.png Vertical box with five child widgets If the TQBoxLayout's orientation is \c Vertical, the boxes are placed in a column, again with suitable sizes. The easiest way to create a TQBoxLayout is to use one of the convenience classes, e.g. TQHBoxLayout (for \c Horizontal boxes) or TQVBoxLayout (for \c Vertical boxes). You can also use the TQBoxLayout constructor directly, specifying its direction as \c LeftToRight, \c Down, \c RightToLeft or \c Up. If the TQBoxLayout is not the top-level tqlayout (i.e. it is not managing all of the widget's area and tqchildren), you must add it to its tqparent tqlayout before you can do anything with it. The normal way to add a tqlayout is by calling parentLayout-\>addLayout(). Once you have done this, you can add boxes to the TQBoxLayout using one of four functions: \list \i addWidget() to add a widget to the TQBoxLayout and set the widget's stretch factor. (The stretch factor is along the row of boxes.) \i addSpacing() to create an empty box; this is one of the functions you use to create nice and spacious dialogs. See below for ways to set margins. \i addStretch() to create an empty, stretchable box. \i addLayout() to add a box containing another TQLayout to the row and set that tqlayout's stretch factor. \endlist Use insertWidget(), insertSpacing(), insertStretch() or insertLayout() to insert a box at a specified position in the tqlayout. TQBoxLayout also includes two margin widths: \list \i setMargin() sets the width of the outer border. This is the width of the reserved space along each of the TQBoxLayout's four sides. \i setSpacing() sets the width between neighboring boxes. (You can use addSpacing() to get more space at a particular spot.) \endlist The margin defaults to 0. The spacing defaults to the same as the margin width for a top-level tqlayout, or to the same as the tqparent tqlayout. Both are parameters to the constructor. To remove a widget from a tqlayout, call remove(). Calling TQWidget::hide() on a widget also effectively removes the widget from the tqlayout until TQWidget::show() is called. You will almost always want to use TQVBoxLayout and TQHBoxLayout rather than TQBoxLayout because of their convenient constructors. \sa TQGrid \link tqlayout.html Layout Overview \endlink */ /*! \enum TQBoxLayout::Direction This type is used to determine the direction of a box tqlayout. \value LeftToRight Horizontal, from left to right \value RightToLeft Horizontal, from right to left \value TopToBottom Vertical, from top to bottom \value Down The same as \c TopToBottom \value BottomToTop Vertical, from bottom to top \value Up The same as \c BottomToTop */ static inline bool horz( TQBoxLayout::Direction dir ) { return dir == TQBoxLayout::RightToLeft || dir == TQBoxLayout::LeftToRight; } /*! Constructs a new TQBoxLayout with direction \a d and main widget \a tqparent. \a tqparent may not be 0. The \a margin is the number of pixels between the edge of the widget and its managed tqchildren. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1 the value of \a margin is used for \a spacing. \a name is the internal object name. \sa direction() */ TQBoxLayout::TQBoxLayout( TQWidget *tqparent, Direction d, int margin, int spacing, const char *name ) : TQLayout( tqparent, margin, spacing, name ) { data = new TQBoxLayoutData; dir = d; setSupportsMargin( TRUE ); } /*! Constructs a new TQBoxLayout called \a name, with direction \a d, and inserts it into \a parentLayout. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1, the tqlayout will inherit its tqparent's spacing(). */ TQBoxLayout::TQBoxLayout( TQLayout *parentLayout, Direction d, int spacing, const char *name ) : TQLayout( parentLayout, spacing, name ) { data = new TQBoxLayoutData; dir = d; setSupportsMargin( TRUE ); } /*! Constructs a new TQBoxLayout called \a name, with direction \a d. If \a spacing is -1, the tqlayout will inherit its tqparent's spacing(); otherwise \a spacing is used. You must insert this box into another tqlayout. */ TQBoxLayout::TQBoxLayout( Direction d, int spacing, const char *name ) : TQLayout( spacing, name ) { data = new TQBoxLayoutData; dir = d; setSupportsMargin( TRUE ); } /*! Destroys this box tqlayout. The tqlayout's widgets aren't destroyed. */ TQBoxLayout::~TQBoxLayout() { delete data; } /*! Returns the preferred size of this box tqlayout. */ TQSize TQBoxLayout::tqsizeHint() const { if ( data->dirty ) { TQBoxLayout *that = (TQBoxLayout*)this; that->setupGeom(); } return data->tqsizeHint + TQSize( 2 * margin(), 2 * margin() ); } /*! Returns the minimum size needed by this box tqlayout. */ TQSize TQBoxLayout::tqminimumSize() const { if ( data->dirty ) { TQBoxLayout *that = (TQBoxLayout*)this; that->setupGeom(); } return data->minSize + TQSize( 2 * margin(), 2 * margin() ); } /*! Returns the maximum size needed by this box tqlayout. */ TQSize TQBoxLayout::tqmaximumSize() const { if ( data->dirty ) { TQBoxLayout *that = (TQBoxLayout*)this; that->setupGeom(); } TQSize s = ( data->maxSize + TQSize(2 * margin(), 2 * margin()) ) .boundedTo(TQSize(TQLAYOUTSIZE_MAX, TQLAYOUTSIZE_MAX)); if ( tqalignment() & TQt::AlignHorizontal_Mask ) s.setWidth( TQLAYOUTSIZE_MAX ); if ( tqalignment() & TQt::AlignVertical_Mask ) s.setHeight( TQLAYOUTSIZE_MAX ); return s; } /*! Returns TRUE if this tqlayout's preferred height depends on its width; otherwise returns FALSE. */ bool TQBoxLayout::hasHeightForWidth() const { if ( data->dirty ) { TQBoxLayout *that = (TQBoxLayout*)this; that->setupGeom(); } return data->hasHfw; } /*! Returns the tqlayout's preferred height when it is \a w pixels wide. */ int TQBoxLayout::heightForWidth( int w ) const { if ( !hasHeightForWidth() ) return -1; w -= 2 * margin(); if ( w != data->hfwWidth ) { TQBoxLayout *that = (TQBoxLayout*)this; that->calcHfw( w ); } return data->hfwHeight + 2 * margin(); } /*! \internal */ int TQBoxLayout::minimumHeightForWidth( int w ) const { (void) heightForWidth( w ); return data->hasHfw ? (data->hfwMinHeight + 2 * margin() ) : -1; } /*! Resets cached information. */ void TQBoxLayout::tqinvalidate() { TQLayout::tqinvalidate(); data->setDirty(); } /*! \reimp */ int TQBoxLayout::count() const { return data->list.count(); } /*! \reimp */ TQLayoutItem* TQBoxLayout::itemAt(int index) const { return index >= 0 && index < data->list.count() ? data->list.at(index)->item : 0; } /*! \reimp */ TQLayoutItem* TQBoxLayout::takeAt(int index) { if (index < 0 || index >= data->list.count()) return 0; TQBoxLayoutItem *b = data->list.take(index); TQLayoutItem *item = b->item; b->item = 0; delete b; invalidate(); return item; } /*! \reimp */ // TQLayoutIterator TQBoxLayout::iterator() // { // return TQLayoutIterator( new TQBoxLayoutIterator(data) ); // } /*! Returns whether this tqlayout can make use of more space than tqsizeHint(). A value of \c Vertical or \c Horizontal means that it wants to grow in only one dimension, whereas \c BothDirections means that it wants to grow in both dimensions. */ TQ_SPExpandData TQBoxLayout::expandingDirections() const { if ( data->dirty ) { TQBoxLayout *that = (TQBoxLayout*)this; that->setupGeom(); } return data->expanding; } /*! Resizes managed widgets within the rectangle \a r. */ void TQBoxLayout::setGeometry( const TQRect &r ) { if ( !data->geomArray || r != tqgeometry() ) { TQLayout::setGeometry( r ); if ( !data->geomArray ) setupGeom(); TQRect cr = tqalignment() ? alignmentRect( r ) : r; TQRect s( cr.x() + margin(), cr.y() + margin(), cr.width() - 2 * margin(), cr.height() - 2 * margin() ); TQMemArray a = *data->geomArray; int pos = horz( dir ) ? s.x() : s.y(); int space = horz( dir ) ? s.width() : s.height(); int n = a.count(); if ( data->hasHfw && !horz(dir) ) { for ( int i = 0; i < n; i++ ) { TQBoxLayoutItem *box = data->list.at( i ); if ( box->item->hasHeightForWidth() ) a[i].tqsizeHint = a[i].tqminimumSize = box->item->heightForWidth( s.width() ); } } Direction visualDir = dir; if ( TQApplication::reverseLayout() ) { if ( dir == LeftToRight ) visualDir = RightToLeft; else if ( dir == RightToLeft ) visualDir = LeftToRight; } qGeomCalc( a, 0, n, pos, space, spacing() ); for ( int i = 0; i < n; i++ ) { TQBoxLayoutItem *box = data->list.at( i ); switch ( visualDir ) { case LeftToRight: box->item->setGeometry( TQRect(a[i].pos, s.y(), a[i].size, s.height()) ); break; case RightToLeft: box->item->setGeometry( TQRect(s.left() + s.right() - a[i].pos - a[i].size + 1, s.y(), a[i].size, s.height()) ); break; case TopToBottom: box->item->setGeometry( TQRect(s.x(), a[i].pos, s.width(), a[i].size) ); break; case BottomToTop: box->item->setGeometry( TQRect(s.x(), s.top() + s.bottom() - a[i].pos - a[i].size + 1, s.width(), a[i].size) ); } } } } /*! Adds \a item to the end of this box tqlayout. */ void TQBoxLayout::addItem( QLayoutItem *item ) { TQBoxLayoutItem *it = new TQBoxLayoutItem( static_cast(item) ); data->list.append( it ); tqinvalidate(); } /*! Inserts \a item into this box tqlayout at position \a index. If \a index is negative, the item is added at the end. \warning Does not call TQLayout::insertChildLayout() if \a item is a TQLayout. \sa addItem(), tqfindWidget() */ void TQBoxLayout::insertItem( int index, TQLayoutItem *item ) { if ( index < 0 ) // append index = data->list.count(); TQBoxLayoutItem *it = new TQBoxLayoutItem( item ); data->list.insert( index, it ); tqinvalidate(); } /*! Inserts a non-stretchable space at position \a index, with size \a size. If \a index is negative the space is added at the end. The box tqlayout has default margin and spacing. This function adds additional space. \sa insertStretch() */ void TQBoxLayout::insertSpacing( int index, int size ) { if ( index < 0 ) // append index = data->list.count(); // hack in TQGridLayoutData: spacers do not get insideSpacing TQLayoutItem *b; if ( horz( dir ) ) b = TQT_TQLAYOUTITEM(new TQSpacerItem( size, 0, TQSizePolicy::Fixed, TQSizePolicy::Minimum )); else b = TQT_TQLAYOUTITEM(new TQSpacerItem( 0, size, TQSizePolicy::Minimum, TQSizePolicy::Fixed )); TQBoxLayoutItem *it = new TQBoxLayoutItem( b ); it->magic = TRUE; data->list.insert( index, it ); tqinvalidate(); } /*! Inserts a stretchable space at position \a index, with zero minimum size and stretch factor \a stretch. If \a index is negative the space is added at the end. \sa insertSpacing() */ void TQBoxLayout::insertStretch( int index, int stretch ) { if ( index < 0 ) // append index = data->list.count(); // hack in TQGridLayoutData: spacers do not get insideSpacing TQLayoutItem *b; if ( horz( dir ) ) b = TQT_TQLAYOUTITEM(new TQSpacerItem( 0, 0, TQSizePolicy::Expanding, TQSizePolicy::Minimum )); else b = TQT_TQLAYOUTITEM(new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Expanding )); TQBoxLayoutItem *it = new TQBoxLayoutItem( b, stretch ); it->magic = TRUE; data->list.insert( index, it ); tqinvalidate(); } /*! Inserts \a tqlayout at position \a index, with stretch factor \a stretch. If \a index is negative, the tqlayout is added at the end. \a tqlayout becomes a child of the box tqlayout. \sa setAutoAdd(), insertWidget(), insertSpacing() */ void TQBoxLayout::insertLayout( int index, TQLayout *tqlayout, int stretch ) { if ( index < 0 ) // append index = data->list.count(); addChildLayout( tqlayout ); TQBoxLayoutItem *it = new TQBoxLayoutItem( TQT_TQLAYOUTITEM(tqlayout), stretch ); data->list.insert( index, it ); tqinvalidate(); } /*! Inserts \a widget at position \a index, with stretch factor \a stretch and tqalignment \a tqalignment. If \a index is negative, the widget is added at the end. The stretch factor applies only in the \link direction() direction \endlink of the TQBoxLayout, and is relative to the other boxes and widgets in this TQBoxLayout. Widgets and boxes with higher stretch factors grow more. If the stretch factor is 0 and nothing else in the TQBoxLayout has a stretch factor greater than zero, the space is distributed according to the TQWidget:sizePolicy() of each widget that's involved. Alignment is specified by \a tqalignment, which is a bitwise OR of \l TQt::AlignmentFlags values. The default tqalignment is 0, which means that the widget fills the entire cell. From TQt 3.0, the \a tqalignment parameter is interpreted more aggressively than in previous versions of TQt. A non-default tqalignment now indicates that the widget should not grow to fill the available space, but should be sized according to tqsizeHint(). \sa setAutoAdd(), insertLayout(), insertSpacing() */ void TQBoxLayout::insertWidget( int index, TQWidget *widget, int stretch, int tqalignment ) { if ( !checkWidget(this, widget) ) return; if ( index < 0 ) // append index = data->list.count(); TQWidgetItem *b = new TQWidgetItem( widget ); b->tqsetAlignment( tqalignment ); TQBoxLayoutItem *it = new TQBoxLayoutItem( b, stretch ); data->list.insert( index, it ); tqinvalidate(); } /*! Adds a non-stretchable space with size \a size to the end of this box tqlayout. TQBoxLayout provides default margin and spacing. This function adds additional space. \sa insertSpacing(), addStretch() */ void TQBoxLayout::addSpacing( int size ) { insertSpacing( -1, size ); } /*! Adds a stretchable space with zero minimum size and stretch factor \a stretch to the end of this box tqlayout. \sa addSpacing() */ void TQBoxLayout::addStretch( int stretch ) { insertStretch( -1, stretch ); } /*! Adds \a widget to the end of this box tqlayout, with a stretch factor of \a stretch and tqalignment \a tqalignment. The stretch factor applies only in the \link direction() direction \endlink of the TQBoxLayout, and is relative to the other boxes and widgets in this TQBoxLayout. Widgets and boxes with higher stretch factors grow more. If the stretch factor is 0 and nothing else in the TQBoxLayout has a stretch factor greater than zero, the space is distributed according to the TQWidget:sizePolicy() of each widget that's involved. Alignment is specified by \a tqalignment which is a bitwise OR of \l TQt::AlignmentFlags values. The default tqalignment is 0, which means that the widget fills the entire cell. From TQt 3.0, the \a tqalignment parameter is interpreted more aggressively than in previous versions of TQt. A non-default tqalignment now indicates that the widget should not grow to fill the available space, but should be sized according to tqsizeHint(). \sa insertWidget(), setAutoAdd(), addLayout(), addSpacing() */ void TQBoxLayout::addWidget( TQWidget *widget, int stretch, int tqalignment ) { insertWidget( -1, widget, stretch, tqalignment ); } /*! Adds \a tqlayout to the end of the box, with serial stretch factor \a stretch. When a tqlayout is constructed with another tqlayout as its tqparent, you don't need to call addLayout(); the child tqlayout is automatically added to the tqparent tqlayout as it is constructed. \sa insertLayout(), setAutoAdd(), addWidget(), addSpacing() */ void TQBoxLayout::addLayout( TQLayout *tqlayout, int stretch ) { insertLayout( -1, tqlayout, stretch ); } /*! Limits the perpendicular dimension of the box (e.g. height if the box is LeftToRight) to a minimum of \a size. Other constraints may increase the limit. */ void TQBoxLayout::addStrut( int size ) { TQLayoutItem *b; if ( horz( dir ) ) b = TQT_TQLAYOUTITEM(new TQSpacerItem( 0, size, TQSizePolicy::Fixed, TQSizePolicy::Minimum )); else b = TQT_TQLAYOUTITEM(new TQSpacerItem( size, 0, TQSizePolicy::Minimum, TQSizePolicy::Fixed )); TQBoxLayoutItem *it = new TQBoxLayoutItem( b ); it->magic = TRUE; data->list.append( it ); tqinvalidate(); } /*! Searches for widget \a w in this tqlayout (not including child layouts). Returns the index of \a w, or -1 if \a w is not found. */ int TQBoxLayout::tqfindWidget( TQWidget* w ) { const int n = data->list.count(); for ( int i = 0; i < n; i++ ) { if ( data->list.at(i)->item->widget() == w ) return i; } return -1; } /*! Sets the stretch factor for widget \a w to \a stretch and returns TRUE if \a w is found in this tqlayout (not including child layouts); otherwise returns FALSE. */ bool TQBoxLayout::setStretchFactor( TQWidget *w, int stretch ) { TQPtrListIterator it( data->list ); TQBoxLayoutItem *box; while ( (box=it.current()) != 0 ) { ++it; if ( box->item->widget() == w ) { box->stretch = stretch; tqinvalidate(); return TRUE; } } return FALSE; } /*! \overload Sets the stretch factor for the tqlayout \a l to \a stretch and returns TRUE if \a l is found in this tqlayout (not including child layouts); otherwise returns FALSE. */ bool TQBoxLayout::setStretchFactor( TQLayout *l, int stretch ) { TQPtrListIterator it( data->list ); TQBoxLayoutItem *box; while ( (box=it.current()) != 0 ) { ++it; if ( box->item->tqlayout() == l ) { box->stretch = stretch; tqinvalidate(); return TRUE; } } return FALSE; } /*! Sets the direction of this tqlayout to \a direction. */ void TQBoxLayout::setDirection( Direction direction ) { if ( dir == direction ) return; if ( horz(dir) != horz(direction) ) { //swap around the spacers (the "magic" bits) //#### a bit yucky, knows too much. //#### probably best to add access functions to tqspacerItem //#### or even a TQSpacerItem::flip() TQPtrListIterator it( data->list ); TQBoxLayoutItem *box; while ( (box=it.current()) != 0 ) { ++it; if ( box->magic ) { TQSpacerItem *sp = box->item->tqspacerItem(); if ( sp ) { if ( sp->expandingDirections() == TQSizePolicy::NoDirection ) { //spacing or strut TQSize s = sp->tqsizeHint(); sp->changeSize( s.height(), s.width(), horz(direction) ? TQSizePolicy::Fixed:TQSizePolicy::Minimum, horz(direction) ? TQSizePolicy::Minimum:TQSizePolicy::Fixed ); } else { //stretch if ( horz(direction) ) sp->changeSize( 0, 0, TQSizePolicy::Expanding, TQSizePolicy::Minimum ); else sp->changeSize( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); } } } } } dir = direction; tqinvalidate(); if ( mainWidget() ) { TQEvent *lh = new TQEvent( TQEvent::LayoutHint ); TQApplication::postEvent( mainWidget(), lh ); } } /* Initializes the data structure needed by qGeomCalc and recalculates max/min and size hint. */ void TQBoxLayout::setupGeom() { if ( !data->dirty ) return; int maxw = horz( dir ) ? 0 : TQLAYOUTSIZE_MAX; int maxh = horz( dir ) ? TQLAYOUTSIZE_MAX : 0; int minw = 0; int minh = 0; int hintw = 0; int hinth = 0; bool horexp = FALSE; bool verexp = FALSE; data->hasHfw = FALSE; delete data->geomArray; int n = data->list.count(); data->geomArray = new TQMemArray( n ); TQMemArray& a = *data->geomArray; bool first = TRUE; for ( int i = 0; i < n; i++ ) { TQBoxLayoutItem *box = data->list.at( i ); TQSize max = box->item->maximumSize(); TQSize min = box->item->minimumSize(); TQSize hint = box->item->sizeHint(); TQ_SPExpandData exp = box->item->expandingDirections(); bool empty = box->item->isEmpty(); // space before non-empties, except the first: int space = ( empty || first ) ? 0 : spacing(); bool ignore = empty && box->item->widget(); // ignore hidden widgets if ( horz(dir) ) { bool expand = exp & TQSizePolicy::Horizontally || box->stretch > 0; horexp = horexp || expand; maxw += max.width() + space; minw += min.width() + space; hintw += hint.width() + space; if ( !ignore ) qMaxExpCalc( maxh, verexp, max.height(), exp & TQSizePolicy::Vertically ); minh = TQMAX( minh, min.height() ); hinth = TQMAX( hinth, hint.height() ); a[i].tqsizeHint = hint.width(); a[i].tqmaximumSize = max.width(); a[i].tqminimumSize = min.width(); a[i].expansive = expand; a[i].stretch = box->stretch ? box->stretch : box->hStretch(); } else { bool expand = ( exp & TQSizePolicy::Vertically || box->stretch > 0 ); verexp = verexp || expand; maxh += max.height() + space; minh += min.height() + space; hinth += hint.height() + space; if ( !ignore ) qMaxExpCalc( maxw, horexp, max.width(), exp & TQSizePolicy::Horizontally ); minw = TQMAX( minw, min.width() ); hintw = TQMAX( hintw, hint.width() ); a[i].tqsizeHint = hint.height(); a[i].tqmaximumSize = max.height(); a[i].tqminimumSize = min.height(); a[i].expansive = expand; a[i].stretch = box->stretch ? box->stretch : box->vStretch(); } a[i].empty = empty; data->hasHfw = data->hasHfw || box->item->hasHeightForWidth(); first = first && empty; } data->minSize = TQSize( minw, minh ); data->maxSize = TQSize( maxw, maxh ).expandedTo( data->minSize ); data->expanding = (TQ_SPExpandData) ( (horexp ? TQSizePolicy::Horizontally : 0) | (verexp ? TQSizePolicy::Vertically : 0) ); data->tqsizeHint = TQSize( hintw, hinth ) .expandedTo( data->minSize ) .boundedTo( data->maxSize ); data->dirty = FALSE; } /* Calculates and stores the preferred height given the width \a w. */ void TQBoxLayout::calcHfw( int w ) { int h = 0; int mh = 0; if ( horz(dir) ) { TQMemArray &a = *data->geomArray; int n = a.count(); qGeomCalc( a, 0, n, 0, w, spacing() ); for ( int i = 0; i < n; i++ ) { TQBoxLayoutItem *box = data->list.at(i); h = TQMAX( h, box->hfw(a[i].size) ); mh = TQMAX( mh, box->mhfw(a[i].size) ); } } else { TQPtrListIterator it( data->list ); TQBoxLayoutItem *box; bool first = TRUE; while ( (box = it.current()) != 0 ) { ++it; bool empty = box->item->isEmpty(); h += box->hfw( w ); mh += box->mhfw( w ); if ( !first && !empty ) { h += spacing(); mh += spacing(); } first = first && empty; } } data->hfwWidth = w; data->hfwHeight = h; data->hfwMinHeight = mh; } /*! \fn TQBoxLayout::Direction TQBoxLayout::direction() const Returns the direction of the box. addWidget() and addSpacing() work in this direction; the stretch stretches in this direction. \sa TQBoxLayout::Direction addWidget() addSpacing() */ /*! \class TQHBoxLayout \brief The TQHBoxLayout class lines up widgets horizontally. \ingroup geomanagement \ingroup appearance \mainclass This class is used to construct horizontal box tqlayout objects. See \l TQBoxLayout for more details. The simplest use of the class is like this: \code TQBoxLayout * l = new TQHBoxLayout( widget ); l->setAutoAdd( TRUE ); new TQSomeWidget( widget ); new TQSomeOtherWidget( widget ); new TQAnotherWidget( widget ); \endcode or like this: \code TQBoxLayout * l = new TQHBoxLayout( widget ); l->addWidget( existingChildOfWidget ); l->addWidget( anotherChildOfWidget ); \endcode \img qhboxtqlayout.png TQHBox \sa TQVBoxLayout TQGridLayout \link tqlayout.html the Layout overview \endlink */ /*! Constructs a new top-level horizontal box called \a name, with tqparent \a tqparent. The \a margin is the number of pixels between the edge of the widget and its managed tqchildren. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1 the value of \a margin is used for \a spacing. */ TQHBoxLayout::TQHBoxLayout( TQWidget *tqparent, int margin, int spacing, const char *name ) : TQBoxLayout( tqparent, LeftToRight, margin, spacing, name ) { } /*! Constructs a new horizontal box called name \a name and adds it to \a parentLayout. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1, this TQHBoxLayout will inherit its tqparent's spacing(). */ TQHBoxLayout::TQHBoxLayout( TQLayout *parentLayout, int spacing, const char *name ) : TQBoxLayout( parentLayout, LeftToRight, spacing, name ) { } /*! Constructs a new horizontal box called name \a name. You must add it to another tqlayout. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1, this TQHBoxLayout will inherit its tqparent's spacing(). */ TQHBoxLayout::TQHBoxLayout( int spacing, const char *name ) : TQBoxLayout( LeftToRight, spacing, name ) { } /*! Destroys this box tqlayout. The tqlayout's widgets aren't destroyed. */ TQHBoxLayout::~TQHBoxLayout() { } /*! \class TQVBoxLayout \brief The TQVBoxLayout class lines up widgets vertically. \ingroup geomanagement \ingroup appearance \mainclass This class is used to construct vertical box tqlayout objects. See TQBoxLayout for more details. The simplest use of the class is like this: \code TQBoxLayout * l = new TQVBoxLayout( widget ); l->addWidget( aWidget ); l->addWidget( anotherWidget ); \endcode \img qvboxtqlayout.png TQVBox \sa TQHBoxLayout TQGridLayout \link tqlayout.html the Layout overview \endlink */ /*! Constructs a new top-level vertical box called \a name, with tqparent \a tqparent. The \a margin is the number of pixels between the edge of the widget and its managed tqchildren. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1 the value of \a margin is used for \a spacing. */ TQVBoxLayout::TQVBoxLayout( TQWidget *tqparent, int margin, int spacing, const char *name ) : TQBoxLayout( tqparent, TopToBottom, margin, spacing, name ) { } /*! Constructs a new vertical box called name \a name and adds it to \a parentLayout. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1, this TQVBoxLayout will inherit its tqparent's spacing(). */ TQVBoxLayout::TQVBoxLayout( TQLayout *parentLayout, int spacing, const char *name ) : TQBoxLayout( parentLayout, TopToBottom, spacing, name ) { } /*! Constructs a new vertical box called name \a name. You must add it to another tqlayout. The \a spacing is the default number of pixels between neighboring tqchildren. If \a spacing is -1, this TQVBoxLayout will inherit its tqparent's spacing(). */ TQVBoxLayout::TQVBoxLayout( int spacing, const char *name ) : TQBoxLayout( TopToBottom, spacing, name ) { } /*! Destroys this box tqlayout. The tqlayout's widgets aren't destroyed. */ TQVBoxLayout::~TQVBoxLayout() { } TQBoxLayout *TQBoxLayout::createTmpCopy() { TQBoxLayout *bl = new TQBoxLayout( direction() ); delete bl->data; bl->data = data; return bl; } #endif // TQT_NO_LAYOUT