diff options
Diffstat (limited to 'tqtinterface/qt4/src/widgets/tqsplitter.cpp')
-rw-r--r-- | tqtinterface/qt4/src/widgets/tqsplitter.cpp | 1427 |
1 files changed, 1427 insertions, 0 deletions
diff --git a/tqtinterface/qt4/src/widgets/tqsplitter.cpp b/tqtinterface/qt4/src/widgets/tqsplitter.cpp new file mode 100644 index 0000000..2750a45 --- /dev/null +++ b/tqtinterface/qt4/src/widgets/tqsplitter.cpp @@ -0,0 +1,1427 @@ +/**************************************************************************** +** +** Implementation of TQSplitter class +** +** Created : 980105 +** +** Copyright (C) 2010 Timothy Pearson and (C) 1992-2008 Trolltech ASA. +** +** This file is part of the widgets 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 "tqsplitter.h" +#ifndef TQT_NO_SPLITTER + +#include "tqlayout.h" +#include "../kernel/tqlayoutengine_p.h" +#include "tqapplication.h" +#include "tqbitmap.h" +#include "tqdrawutil.h" +#include "tqmemarray.h" +#include "tqobjectlist.h" +#include "tqpainter.h" +#include "tqptrlist.h" +#include "tqstyle.h" + +class TQSplitterHandle : public TQWidget +{ + TQ_OBJECT +public: + TQSplitterHandle( Qt::Orientation o, + TQSplitter *tqparent, const char* name=0 ); + void setOrientation( Qt::Orientation o ); + Qt::Orientation orientation() const { return orient; } + + bool opaque() const { return s->opaqueResize(); } + + TQSize tqsizeHint() const; + + int id() const { return myId; } // d->list.at(id())->wid == this + void setId( int i ) { myId = i; } + +protected: + void paintEvent( TQPaintEvent * ); + void mouseMoveEvent( TQMouseEvent * ); + void mousePressEvent( TQMouseEvent * ); + void mouseReleaseEvent( TQMouseEvent * ); + +private: + Qt::Orientation orient; + bool opaq; + int myId; + + TQSplitter *s; +}; + +#include "tqsplitter.tqmoc" + +const uint Default = 2; + +static int mouseOffset; +static int opaqueOldPos = -1; // this assumes that there's only one mouse + +static TQPoint toggle( TQWidget *w, TQPoint pos ) +{ + TQSize minS = tqSmartMinSize( w ); + return -pos - TQPoint( minS.width(), minS.height() ); +} + +static bool isCollapsed( TQWidget *w ) +{ + return w->x() < 0 || w->y() < 0; +} + +static TQPoint topLeft( TQWidget *w ) +{ + if ( isCollapsed(w) ) { + return toggle( w, w->pos() ); + } else { + return w->pos(); + } +} + +static TQPoint bottomRight( TQWidget *w ) +{ + if ( isCollapsed(w) ) { + return toggle( w, w->pos() ) - TQPoint( 1, 1 ); + } else { + return w->tqgeometry().bottomRight(); + } +} + +TQSplitterHandle::TQSplitterHandle( Qt::Orientation o, TQSplitter *tqparent, + const char * name ) + : TQWidget( tqparent, name ) +{ + s = tqparent; + setOrientation( o ); +} + +TQSize TQSplitterHandle::tqsizeHint() const +{ + int hw = s->handleWidth(); + return parentWidget()->tqstyle().tqsizeFromContents( TQStyle::CT_Splitter, s, + TQSize(hw, hw) ) + .expandedTo( TQApplication::globalStrut() ); +} + +void TQSplitterHandle::setOrientation( Qt::Orientation o ) +{ + orient = o; +#ifndef TQT_NO_CURSOR + setCursor( o == Qt::Horizontal ? Qt::SplitHCursor : Qt::SplitVCursor ); +#endif +} + +void TQSplitterHandle::mouseMoveEvent( TQMouseEvent *e ) +{ + if ( !(e->state()&Qt::LeftButton) ) + return; + TQCOORD pos = s->pick( parentWidget()->mapFromGlobal(e->globalPos()) ) + - mouseOffset; + if ( opaque() ) { + s->moveSplitter( pos, id() ); + } else { + s->setRubberband( s->adjustPos(pos, id()) ); + } +} + +void TQSplitterHandle::mousePressEvent( TQMouseEvent *e ) +{ + if ( e->button() == Qt::LeftButton ) + mouseOffset = s->pick( e->pos() ); +} + +void TQSplitterHandle::mouseReleaseEvent( TQMouseEvent *e ) +{ + if ( !opaque() && e->button() == Qt::LeftButton ) { + TQCOORD pos = s->pick( parentWidget()->mapFromGlobal(e->globalPos()) ) + - mouseOffset; + s->setRubberband( -1 ); + s->moveSplitter( pos, id() ); + } +} + +void TQSplitterHandle::paintEvent( TQPaintEvent * ) +{ + TQPainter p( this ); + parentWidget()->tqstyle().tqdrawPrimitive( TQStyle::PE_Splitter, &p, rect(), + tqcolorGroup(), + (TQStyle::SFlags)(orientation() == Qt::Horizontal ? + TQStyle::Style_Horizontal : 0) ); +} + +class TQSplitterLayoutStruct : public TQt +{ +public: + TQCOORD sizer; + uint isHandle : 1; + uint collapsible : 2; + uint resizeMode : 2; + TQWidget *wid; + + TQSplitterLayoutStruct() + : sizer( -1 ), collapsible( Default ) { } + TQCOORD getSizer( Qt::Orientation orient ); +}; + +TQCOORD TQSplitterLayoutStruct::getSizer( Qt::Orientation orient ) +{ + if ( sizer == -1 ) { + TQSize s = wid->tqsizeHint(); + if ( !s.isValid() || wid->testWState(WState_Resized) ) + s = wid->size(); + sizer = ( orient == Qt::Horizontal ) ? s.width() : s.height(); + } + return sizer; +} + +class TQSplitterPrivate +{ +public: + TQSplitterPrivate() + : opaque( FALSE ), firstShow( TRUE ), tqchildrenCollapsible( TRUE ), + handleWidth( 0 ) { } + + TQPtrList<TQSplitterLayoutStruct> list; + bool opaque : 8; + bool firstShow : 8; + bool tqchildrenCollapsible : 8; + int handleWidth; +}; + + +/*! + \class TQSplitter + \brief The TQSplitter class implements a splitter widget. + + \ingroup organizers + \mainclass + + A splitter lets the user control the size of child widgets by + dragging the boundary between the tqchildren. Any number of widgets + may be controlled by a single splitter. + + To show a TQListBox, a TQListView and a TQTextEdit side by side: + \code + TQSplitter *split = new TQSplitter( tqparent ); + TQListBox *lb = new TQListBox( split ); + TQListView *lv = new TQListView( split ); + TQTextEdit *ed = new TQTextEdit( split ); + \endcode + + TQSplitter lays out its tqchildren horizontally (side by side); you + can use setOrientation(TQSplitter::Vertical) to lay out the + tqchildren vertically. + + By default, all widgets can be as large or as small as the user + wishes, between the \l tqminimumSizeHint() (or \l tqminimumSize()) + and \l tqmaximumSize() of the widgets. Use setResizeMode() to + specify that a widget should keep its size when the splitter is + resized, or set the stretch component of the \l sizePolicy. + + Although TQSplitter normally resizes the tqchildren only at the end + of a resize operation, if you call setOpaqueResize(TRUE) the + widgets are resized as often as possible. + + The initial distribution of size between the widgets is determined + by the initial size of each widget. You can also use setSizes() to + set the sizes of all the widgets. The function sizes() returns the + sizes set by the user. + + If you hide() a child its space will be distributed among the + other tqchildren. It will be reinstated when you show() it again. It + is also possible to reorder the widgets within the splitter using + moveToFirst() and moveToLast(). + + <img src=qsplitter-m.png> <img src=qsplitter-w.png> + + \sa TQTabBar +*/ + + +/*! + Constructs a horizontal splitter with the \a tqparent and \a name + arguments being passed on to the TQFrame constructor. +*/ + +TQSplitter::TQSplitter( TQWidget *tqparent, const char *name ) + : TQFrame( tqparent, name, TQt::WPaintUnclipped ) +{ + orient = Qt::Horizontal; + init(); +} + + +/*! + Constructs a splitter with orientation \a o with the \a tqparent and + \a name arguments being passed on to the TQFrame constructor. +*/ + +TQSplitter::TQSplitter( Qt::Orientation o, TQWidget *tqparent, const char *name ) + : TQFrame( tqparent, name, TQt::WPaintUnclipped ) +{ + orient = o; + init(); +} + + +/*! + Destroys the splitter and any tqchildren. +*/ + +TQSplitter::~TQSplitter() +{ + delete d; +} + + +void TQSplitter::init() +{ + d = new TQSplitterPrivate; + d->list.setAutoDelete( TRUE ); + TQSizePolicy sp( TQSizePolicy::Expanding, TQSizePolicy::Preferred ); + if ( orient == Qt::Vertical ) + sp.transpose(); + tqsetSizePolicy( sp ); + clearWState( TQt::WState_OwnSizePolicy ); +} + +/*! + \fn void TQSplitter::refresh() + + Updates the splitter's state. You should not need to call this + function. +*/ + + +/*! + \property TQSplitter::orientation + \brief the orientation of the splitter + + By default the orientation is horizontal (the widgets are side by + side). The possible orientations are \c Qt::Horizontal and + \c Qt::Vertical. +*/ + +void TQSplitter::setOrientation( Qt::Orientation o ) +{ + if ( orient == o ) + return; + + if ( !testWState( TQt::WState_OwnSizePolicy ) ) { + TQSizePolicy sp = tqsizePolicy(); + sp.transpose(); + tqsetSizePolicy( sp ); + clearWState( TQt::WState_OwnSizePolicy ); + } + + orient = o; + + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( s->isHandle ) + ((TQSplitterHandle*)s->wid)->setOrientation( o ); + s = d->list.next(); + } + recalc( isVisible() ); +} + +/*! + \property TQSplitter::tqchildrenCollapsible + \brief whether child widgets can be resized down to size 0 by the user + + By default, tqchildren are collapsible. It is possible to enable + and disable the collapsing of individual tqchildren; see + setCollapsible(). +*/ + +void TQSplitter::setChildrenCollapsible( bool collapse ) +{ + d->tqchildrenCollapsible = collapse; +} + +bool TQSplitter::tqchildrenCollapsible() const +{ + return d->tqchildrenCollapsible; +} + +/*! + Sets whether the child widget \a w is collapsible to \a collapse. + + By default, tqchildren are collapsible, meaning that the user can + resize them down to size 0, even if they have a non-zero + tqminimumSize() or tqminimumSizeHint(). This behavior can be changed + on a per-widget basis by calling this function, or globally for + all the widgets in the splitter by setting the \l + tqchildrenCollapsible property. + + \sa tqchildrenCollapsible +*/ + +void TQSplitter::setCollapsible( TQWidget *w, bool collapse ) +{ + tqfindWidget( w )->collapsible = collapse ? 1 : 0; +} + +/*! + \reimp +*/ +void TQSplitter::resizeEvent( TQResizeEvent * ) +{ + doResize(); +} + +TQSplitterLayoutStruct *TQSplitter::tqfindWidget( TQWidget *w ) +{ + processChildEvents(); + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( s->wid == w ) + return s; + s = d->list.next(); + } + return addWidget( w ); +} + +/* + Inserts the widget \a w at the end (or at the beginning if \a + prepend is TRUE) of the splitter's list of widgets. + + It is the responsibility of the caller to make sure that \a w is + not already in the splitter and to call recalcId() if needed. (If + \a prepend is TRUE, then recalcId() is very probably needed.) +*/ + +TQSplitterLayoutStruct *TQSplitter::addWidget( TQWidget *w, bool prepend ) +{ + TQSplitterLayoutStruct *s; + TQSplitterHandle *newHandle = 0; + if ( d->list.count() > 0 ) { + s = new TQSplitterLayoutStruct; + s->resizeMode = KeepSize; + TQString tmp = "qt_splithandle_"; + tmp += w->name(); + newHandle = new TQSplitterHandle( orientation(), this, tmp ); + s->wid = newHandle; + newHandle->setId( d->list.count() ); + s->isHandle = TRUE; + s->sizer = pick( newHandle->tqsizeHint() ); + if ( prepend ) + d->list.prepend( s ); + else + d->list.append( s ); + } + s = new TQSplitterLayoutStruct; + s->resizeMode = DefaultResizeMode; + s->wid = w; + s->isHandle = FALSE; + if ( prepend ) + d->list.prepend( s ); + else + d->list.append( s ); + if ( newHandle && isVisible() ) + newHandle->show(); // will trigger sending of post events + return s; +} + + +/*! + Tells the splitter that the child widget described by \a c has + been inserted or removed. +*/ + +void TQSplitter::childEvent( TQChildEvent *c ) +{ + if ( c->type() == TQEvent::ChildInserted ) { + if ( !c->child()->isWidgetType() ) + return; + + if ( ((TQWidget*)c->child())->testWFlags( TQt::WType_TopLevel ) ) + return; + + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( s->wid == c->child() ) + return; + s = d->list.next(); + } + addWidget( (TQWidget*)c->child() ); + recalc( isVisible() ); + } else if ( c->type() == TQEvent::ChildRemoved ) { + TQSplitterLayoutStruct *prev = 0; + if ( d->list.count() > 1 ) + prev = d->list.at( 1 ); // yes, this is correct + TQSplitterLayoutStruct *curr = d->list.first(); + while ( curr ) { + if ( curr->wid == c->child() ) { + d->list.removeRef( curr ); + if ( prev && prev->isHandle ) { + TQWidget *w = prev->wid; + d->list.removeRef( prev ); + delete w; // will call childEvent() + } + recalcId(); + doResize(); + return; + } + prev = curr; + curr = d->list.next(); + } + } +} + + +/*! + Displays a rubber band at position \a p. If \a p is negative, the + rubber band is removed. +*/ + +void TQSplitter::setRubberband( int p ) +{ + TQPainter paint( this ); + paint.setPen( Qt::gray ); + paint.setBrush( Qt::gray ); + paint.setRasterOp( TQt::XorROP ); + TQRect r = contentsRect(); + const int rBord = 3; // customizable? + int hw = handleWidth(); + if ( orient == Qt::Horizontal ) { + if ( opaqueOldPos >= 0 ) + paint.drawRect( opaqueOldPos + hw / 2 - rBord, r.y(), + 2 * rBord, r.height() ); + if ( p >= 0 ) + paint.drawRect( p + hw / 2 - rBord, r.y(), 2 * rBord, r.height() ); + } else { + if ( opaqueOldPos >= 0 ) + paint.drawRect( r.x(), opaqueOldPos + hw / 2 - rBord, + r.width(), 2 * rBord ); + if ( p >= 0 ) + paint.drawRect( r.x(), p + hw / 2 - rBord, r.width(), 2 * rBord ); + } + opaqueOldPos = p; +} + + +/*! + \reimp +*/ + +bool TQSplitter::event( TQEvent *e ) +{ + switch ( e->type() ) { + case TQEvent::Show: + if ( !d->firstShow ) + break; + d->firstShow = FALSE; + // fall through + case TQEvent::LayoutHint: + recalc( isVisible() ); + break; + default: + ; + } + return TQWidget::event( e ); +} + + +/*! + \obsolete + + Draws the splitter handle in the rectangle described by \a x, \a y, + \a w, \a h using painter \a p. + \sa TQStyle::tqdrawPrimitive() +*/ + +// ### Remove this in 4.0 + +void TQSplitter::drawSplitter( TQPainter *p, + TQCOORD x, TQCOORD y, TQCOORD w, TQCOORD h ) +{ + tqstyle().tqdrawPrimitive(TQStyle::PE_Splitter, p, TQRect(x, y, w, h), tqcolorGroup(), + (TQStyle::SFlags)(orientation() == Qt::Horizontal ? + TQStyle::Style_Horizontal : 0)); +} + + +/*! + Returns the ID of the widget to the right of or below the widget + \a w, or 0 if there is no such widget (i.e. it is either not in + this TQSplitter or \a w is at the end). +*/ + +int TQSplitter::idAfter( TQWidget* w ) const +{ + TQSplitterLayoutStruct *s = d->list.first(); + bool seen_w = FALSE; + while ( s ) { + if ( s->isHandle && seen_w ) + return d->list.at(); + if ( !s->isHandle && s->wid == w ) + seen_w = TRUE; + s = d->list.next(); + } + return 0; +} + + +/*! + Moves the left/top edge of the splitter handle with ID \a id as + close as possible to position \a p, which is the distance from the + left (or top) edge of the widget. + + For Arabic, Hebrew and other right-to-left languages the tqlayout is + reversed. \a p is then the distance from the right (or top) edge + of the widget. + + \sa idAfter() +*/ +void TQSplitter::moveSplitter( TQCOORD p, int id ) +{ + TQSplitterLayoutStruct *s = d->list.at( id ); + int farMin; + int min; + int max; + int farMax; + + p = adjustPos( p, id, &farMin, &min, &max, &farMax ); + int oldP = pick( s->wid->pos() ); + + if ( TQApplication::reverseLayout() && orient == Qt::Horizontal ) { + int q = p + s->wid->width(); + doMove( FALSE, q, id - 1, -1, (q > oldP), (p > max) ); + doMove( TRUE, q, id, -1, (q > oldP), (p < min) ); + } else { + doMove( FALSE, p, id, +1, (p < oldP), (p > max) ); + doMove( TRUE, p, id - 1, +1, (p < oldP), (p < min) ); + } + storeSizes(); +} + + +void TQSplitter::setGeo( TQWidget *w, int p, int s, bool splitterMoved ) +{ + TQRect r; + if ( orient == Qt::Horizontal ) { + if ( TQApplication::reverseLayout() && orient == Qt::Horizontal + && !splitterMoved ) + p = contentsRect().width() - p - s; + r.setRect( p, contentsRect().y(), s, contentsRect().height() ); + } else { + r.setRect( contentsRect().x(), p, contentsRect().width(), s ); + } + + /* + Hide the child widget, but without calling hide() so that the + splitter handle is still shown. + */ + if ( !w->isHidden() && s <= 0 && pick(tqSmartMinSize(w)) > 0 ) + r.moveTopLeft( toggle(w, r.topLeft()) ); + w->setGeometry( r ); +} + + +void TQSplitter::doMove( bool backwards, int pos, int id, int delta, bool upLeft, + bool mayCollapse ) +{ + if ( id < 0 || id >= (int) d->list.count() ) + return; + + TQSplitterLayoutStruct *s = d->list.at( id ); + TQWidget *w = s->wid; + + int nextId = backwards ? id - delta : id + delta; + + if ( w->isHidden() ) { + doMove( backwards, pos, nextId, delta, upLeft, TRUE ); + } else { + if ( s->isHandle ) { + int dd = s->getSizer( orient ); + int nextPos = backwards ? pos - dd : pos + dd; + int left = backwards ? pos - dd : pos; + setGeo( w, left, dd, TRUE ); + doMove( backwards, nextPos, nextId, delta, upLeft, mayCollapse ); + } else { + int dd = backwards ? pos - pick( topLeft(w) ) + : pick( bottomRight(w) ) - pos + 1; + if ( dd > 0 || (!isCollapsed(w) && !mayCollapse) ) { + dd = TQMAX( pick(tqSmartMinSize(w)), + TQMIN(dd, pick(w->tqmaximumSize())) ); + } else { + dd = 0; + } + setGeo( w, backwards ? pos - dd : pos, dd, TRUE ); + doMove( backwards, backwards ? pos - dd : pos + dd, nextId, delta, + upLeft, TRUE ); + } + } +} + +int TQSplitter::tqfindWidgetJustBeforeOrJustAfter( int id, int delta, int &collapsibleSize ) +{ + id += delta; + do { + TQWidget *w = d->list.at( id )->wid; + if ( !w->isHidden() ) { + if ( collapsible(d->list.at(id)) ) + collapsibleSize = pick( tqSmartMinSize(w) ); + return id; + } + id += 2 * delta; // go to previous (or next) widget, skip the handle + } while ( id >= 0 && id < (int)d->list.count() ); + + return -1; +} + +void TQSplitter::getRange( int id, int *farMin, int *min, int *max, int *farMax ) +{ + int n = d->list.count(); + if ( id <= 0 || id >= n - 1 ) + return; + + int collapsibleSizeBefore = 0; + int idJustBefore = tqfindWidgetJustBeforeOrJustAfter( id, -1, collapsibleSizeBefore ); + + int collapsibleSizeAfter = 0; + int idJustAfter = tqfindWidgetJustBeforeOrJustAfter( id, +1, collapsibleSizeAfter ); + + int minBefore = 0; + int minAfter = 0; + int maxBefore = 0; + int maxAfter = 0; + int i; + + for ( i = 0; i < id; i++ ) + addContribution( i, &minBefore, &maxBefore, i == idJustBefore ); + for ( i = id; i < n; i++ ) + addContribution( i, &minAfter, &maxAfter, i == idJustAfter ); + + TQRect r = contentsRect(); + int farMinVal; + int minVal; + int maxVal; + int farMaxVal; + + int smartMinBefore = TQMAX( minBefore, pick(r.size()) - maxAfter ); + int smartMaxBefore = TQMIN( maxBefore, pick(r.size()) - minAfter ); + + if ( orient == Qt::Vertical || !TQApplication::reverseLayout() ) { + minVal = pick( r.topLeft() ) + smartMinBefore; + maxVal = pick( r.topLeft() ) + smartMaxBefore; + + farMinVal = minVal; + if ( minBefore - collapsibleSizeBefore >= pick(r.size()) - maxAfter ) + farMinVal -= collapsibleSizeBefore; + farMaxVal = maxVal; + if ( pick(r.size()) - (minAfter - collapsibleSizeAfter) <= maxBefore ) + farMaxVal += collapsibleSizeAfter; + } else { + int hw = handleWidth(); + minVal = r.width() - smartMaxBefore - hw; + maxVal = r.width() - smartMinBefore - hw; + + farMinVal = minVal; + if ( pick(r.size()) - (minAfter - collapsibleSizeAfter) <= maxBefore ) + farMinVal -= collapsibleSizeAfter; + farMaxVal = maxVal; + if ( minBefore - collapsibleSizeBefore >= pick(r.size()) - maxAfter ) + farMaxVal += collapsibleSizeBefore; + } + + if ( farMin ) + *farMin = farMinVal; + if ( min ) + *min = minVal; + if ( max ) + *max = maxVal; + if ( farMax ) + *farMax = farMaxVal; +} + +/*! + Returns the valid range of the splitter with ID \a id in \a *min + and \a *max if \a min and \a max are not 0. + + \sa idAfter() +*/ + +void TQSplitter::getRange( int id, int *min, int *max ) +{ + getRange( id, min, 0, 0, max ); +} + + +/*! + Returns the closest legal position to \a pos of the widget with ID + \a id. + + \sa idAfter() +*/ + +int TQSplitter::adjustPos( int pos, int id ) +{ + int x, i, n, u; + return adjustPos( pos, id, &u, &n, &i, &x ); +} + +int TQSplitter::adjustPos( int pos, int id, int *farMin, int *min, int *max, + int *farMax ) +{ + const int Threshold = 40; + + getRange( id, farMin, min, max, farMax ); + + if ( pos >= *min ) { + if ( pos <= *max ) { + return pos; + } else { + int delta = pos - *max; + int width = *farMax - *max; + + if ( delta > width / 2 && delta >= TQMIN(Threshold, width) ) { + return *farMax; + } else { + return *max; + } + } + } else { + int delta = *min - pos; + int width = *min - *farMin; + + if ( delta > width / 2 && delta >= TQMIN(Threshold, width) ) { + return *farMin; + } else { + return *min; + } + } +} + +bool TQSplitter::collapsible( TQSplitterLayoutStruct *s ) +{ + if (pick(tqSmartMinSize(s->wid)) == 1) + return FALSE; + if ( s->collapsible != Default ) { + return (bool) s->collapsible; + } else { + return d->tqchildrenCollapsible; + } +} + +void TQSplitter::doResize() +{ + TQRect r = contentsRect(); + int n = d->list.count(); + TQMemArray<TQLayoutStruct> a( n ); + + for ( int pass = 0; pass < 2; pass++ ) { + int numAutoWithStretch = 0; + int numAutoWithoutStretch = 0; + + for ( int i = 0; i < n; i++ ) { + a[i].init(); + TQSplitterLayoutStruct *s = d->list.at( i ); + if ( s->wid->isHidden() || isCollapsed(s->wid) ) { + a[i].tqmaximumSize = 0; + } else if ( s->isHandle ) { + a[i].tqsizeHint = a[i].tqminimumSize = a[i].tqmaximumSize = s->sizer; + a[i].empty = FALSE; + } else { + int mode = s->resizeMode; + int stretch = 1; + + if ( mode == DefaultResizeMode ) { + TQSizePolicy p = s->wid->tqsizePolicy(); + int sizePolicyStretch = + pick( TQSize(p.horStretch(), p.verStretch()) ); + if ( sizePolicyStretch > 0 ) { + mode = Stretch; + stretch = sizePolicyStretch; + numAutoWithStretch++; + } else { + /* + Do things differently on the second pass, + if there's one. A second pass is necessary + if it was found out during the first pass + that all DefaultResizeMode items are + KeepSize items. In that case, we make them + all Stretch items instead, for a more TQt + 3.0-compatible behavior. + */ + mode = ( pass == 0 ) ? KeepSize : Stretch; + numAutoWithoutStretch++; + } + } + + a[i].tqminimumSize = pick( tqSmartMinSize(s->wid) ); + a[i].tqmaximumSize = pick( s->wid->tqmaximumSize() ); + a[i].empty = FALSE; + + if ( mode == Stretch ) { + if ( s->getSizer(orient) > 1 ) + stretch *= s->getSizer( orient ); + // TQMIN(): ad hoc work-around for tqlayout engine limitation + a[i].stretch = TQMIN( stretch, 8192 ); + a[i].tqsizeHint = a[i].tqminimumSize; + } else if ( mode == KeepSize ) { + a[i].tqsizeHint = s->getSizer( orient ); + } else { // mode == FollowSizeHint + a[i].tqsizeHint = pick( s->wid->tqsizeHint() ); + } + } + } + + // a second pass would yield the same results + if ( numAutoWithStretch > 0 || numAutoWithoutStretch == 0 ) + break; + } + + qGeomCalc( a, 0, n, pick( r.topLeft() ), pick( r.size() ), 0 ); + + for ( int i = 0; i < n; i++ ) { + TQSplitterLayoutStruct *s = d->list.at(i); + setGeo( s->wid, a[i].pos, a[i].size, FALSE ); + } +} + +void TQSplitter::recalc( bool update ) +{ + int fi = 2 * frameWidth(); + int maxl = fi; + int minl = fi; + int maxt = TQWIDGETSIZE_MAX; + int mint = fi; + int n = d->list.count(); + bool first = TRUE; + + /* + Splitter handles before the first visible widget or right + before a hidden widget must be hidden. + */ + for ( int i = 0; i < n; i++ ) { + TQSplitterLayoutStruct *s = d->list.at( i ); + if ( !s->isHandle ) { + TQSplitterLayoutStruct *p = 0; + if ( i > 0 ) + p = d->list.at( i - 1 ); + + // may trigger new recalc + if ( p && p->isHandle ) + p->wid->setHidden( first || s->wid->isHidden() ); + + if ( !s->wid->isHidden() ) + first = FALSE; + } + } + + bool empty = TRUE; + for ( int j = 0; j < n; j++ ) { + TQSplitterLayoutStruct *s = d->list.at( j ); + if ( !s->wid->isHidden() ) { + empty = FALSE; + if ( s->isHandle ) { + minl += s->getSizer( orient ); + maxl += s->getSizer( orient ); + } else { + TQSize minS = tqSmartMinSize( s->wid ); + minl += pick( minS ); + maxl += pick( s->wid->tqmaximumSize() ); + mint = TQMAX( mint, trans(minS) ); + int tm = trans( s->wid->tqmaximumSize() ); + if ( tm > 0 ) + maxt = TQMIN( maxt, tm ); + } + } + } + if ( empty ) { + if ( ::tqqt_cast<TQSplitter*>(parentWidget()) ) { + // nested splitters; be nice + maxl = maxt = 0; + } else { + // TQSplitter with no tqchildren yet + maxl = TQWIDGETSIZE_MAX; + } + } else { + maxl = TQMIN( maxl, TQWIDGETSIZE_MAX ); + } + if ( maxt < mint ) + maxt = mint; + + if ( orient == Qt::Horizontal ) { + setMaximumSize( maxl, maxt ); + setMinimumSize( minl, mint ); + } else { + setMaximumSize( maxt, maxl ); + setMinimumSize( mint, minl ); + } + if ( update ) + doResize(); + else + d->firstShow = TRUE; +} + +/*! + \enum TQSplitter::ResizeMode + + This enum type describes how TQSplitter will resize each of its + child widgets. + + \value Auto The widget will be resized according to the stretch + factors set in its sizePolicy(). + + \value Stretch The widget will be resized when the splitter + itself is resized. + + \value KeepSize TQSplitter will try to keep the widget's size + unchanged. + + \value FollowSizeHint TQSplitter will resize the widget when the + widget's size hint changes. +*/ + +/*! + Sets resize mode of widget \a w to \a mode. (The default is \c + Auto.) +*/ + +void TQSplitter::setResizeMode( TQWidget *w, ResizeMode mode ) +{ + tqfindWidget( w )->resizeMode = mode; +} + + +/*! + \property TQSplitter::opaqueResize + \brief whether resizing is opaque + + Opaque resizing is off by default. +*/ + +bool TQSplitter::opaqueResize() const +{ + return d->opaque; +} + + +void TQSplitter::setOpaqueResize( bool on ) +{ + d->opaque = on; +} + + +/*! + Moves widget \a w to the leftmost/top position. +*/ + +void TQSplitter::moveToFirst( TQWidget *w ) +{ + processChildEvents(); + bool found = FALSE; + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( s->wid == w ) { + found = TRUE; + TQSplitterLayoutStruct *p = d->list.prev(); + if ( p ) { // not already at first place + d->list.take(); // take p + d->list.take(); // take s + d->list.prepend( p ); + d->list.prepend( s ); + } + break; + } + s = d->list.next(); + } + if ( !found ) + addWidget( w, TRUE ); + recalcId(); +} + + +/*! + Moves widget \a w to the rightmost/bottom position. +*/ + +void TQSplitter::moveToLast( TQWidget *w ) +{ + processChildEvents(); + bool found = FALSE; + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( s->wid == w ) { + found = TRUE; + d->list.take(); // take s + TQSplitterLayoutStruct *p = d->list.current(); + if ( p ) { // the splitter handle after s + d->list.take(); // take p + d->list.append( p ); + } + d->list.append( s ); + break; + } + s = d->list.next(); + } + if ( !found ) + addWidget( w ); + recalcId(); +} + + +void TQSplitter::recalcId() +{ + int n = d->list.count(); + for ( int i = 0; i < n; i++ ) { + TQSplitterLayoutStruct *s = d->list.at( i ); + if ( s->isHandle ) + ((TQSplitterHandle*)s->wid)->setId( i ); + } +} + + +/*! + \reimp +*/ +TQSize TQSplitter::tqsizeHint() const +{ + constPolish(); + int l = 0; + int t = 0; + if ( !childrenListObject().isEmpty() ) { + TQObjectListIt it( childrenListObject() ); + TQObject * o; + + while( (o = it.current()) != 0 ) { + ++it; + if ( o->isWidgetType() && !((TQWidget*)o)->isHidden() ) { + TQSize s = ((TQWidget*)o)->tqsizeHint(); + if ( s.isValid() ) { + l += pick( s ); + t = TQMAX( t, trans( s ) ); + } + } + } + } + return orientation() == Qt::Horizontal ? TQSize( l, t ) : TQSize( t, l ); +} + + +/*! + \reimp +*/ + +TQSize TQSplitter::tqminimumSizeHint() const +{ + constPolish(); + int l = 0; + int t = 0; + if ( !childrenListObject().isEmpty() ) { + TQObjectListIt it( childrenListObject() ); + TQObject * o; + + while ( (o = it.current()) != 0 ) { + ++it; + if ( o->isWidgetType() && !((TQWidget*)o)->isHidden() ) { + TQSize s = tqSmartMinSize( (TQWidget*)o ); + if ( s.isValid() ) { + l += pick( s ); + t = TQMAX( t, trans( s ) ); + } + } + } + } + return orientation() == Qt::Horizontal ? TQSize( l, t ) : TQSize( t, l ); +} + + +void TQSplitter::storeSizes() +{ + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( !s->isHandle ) + s->sizer = pick( s->wid->size() ); + s = d->list.next(); + } +} + + +void TQSplitter::addContribution( int id, int *min, int *max, + bool mayCollapse ) +{ + TQSplitterLayoutStruct *s = d->list.at( id ); + if ( !s->wid->isHidden() ) { + if ( s->isHandle ) { + *min += s->getSizer( orient ); + *max += s->getSizer( orient ); + } else { + if ( mayCollapse || !isCollapsed(s->wid) ) + *min += pick( tqSmartMinSize(s->wid) ); + *max += pick( s->wid->tqmaximumSize() ); + } + } +} + + +/*! + Returns a list of the size parameters of all the widgets in this + splitter. + + If the splitter's orientation is horizontal, the list is a list of + widget widths; if the orientation is vertical, the list is a list + of widget heights. + + Giving the values to another splitter's setSizes() function will + produce a splitter with the same tqlayout as this one. + + Note that if you want to iterate over the list, you should iterate + over a copy, e.g. + \code + TQValueList<int> list = mySplitter.sizes(); + TQValueList<int>::Iterator it = list.begin(); + while( it != list.end() ) { + myProcessing( *it ); + ++it; + } + \endcode + + \sa setSizes() +*/ + +TQValueList<int> TQSplitter::sizes() const +{ + if ( !testWState(TQt::WState_Polished) ) + constPolish(); + + TQValueList<int> list; + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( !s->isHandle ) + list.append( isCollapsed(s->wid) ? 0 : pick(s->wid->size())); + s = d->list.next(); + } + return list; +} + +/*! + Sets the size parameters to the values given in the \a list. If + the splitter is horizontal, the values set the widths of each + widget going from left to right. If the splitter is vertical, the + values set the heights of each widget going from top to bottom. + Extra values in the \a list are ignored. + + If \a list tqcontains too few values, the result is undefined but + the program will still be well-behaved. + + Note that the values in \a list should be the height/width that + the widgets should be resized to. + + \sa sizes() +*/ + +void TQSplitter::setSizes( TQValueList<int> list ) +{ + processChildEvents(); + TQValueList<int>::Iterator it = list.begin(); + TQSplitterLayoutStruct *s = d->list.first(); + while ( s && it != list.end() ) { + if ( !s->isHandle ) { + s->sizer = TQMAX( *it, 0 ); + int smartMinSize = pick( tqSmartMinSize(s->wid) ); + // Make sure that we reset the collapsed state. + if ( s->sizer == 0 ) { + if ( collapsible(s) && smartMinSize > 0 ) { + s->wid->move( -1, -1 ); + } else { + s->sizer = smartMinSize; + s->wid->move( 0, 0 ); + } + } else { + if ( s->sizer < smartMinSize ) + s->sizer = smartMinSize; + s->wid->move( 0, 0 ); + } + ++it; + } + s = d->list.next(); + } + doResize(); +} + +/*! + \property TQSplitter::handleWidth + \brief the width of the splitter handle +*/ + +int TQSplitter::handleWidth() const +{ + if ( d->handleWidth > 0 ) { + return d->handleWidth; + } else { + return tqstyle().tqpixelMetric( TQStyle::PM_SplitterWidth, this ); + } +} + +void TQSplitter::setHandleWidth( int width ) +{ + d->handleWidth = width; + updateHandles(); +} + +/*! + Processes all posted child events, ensuring that the internal state of + the splitter is kept consistent. +*/ + +void TQSplitter::processChildEvents() +{ + TQApplication::sendPostedEvents( this, TQEvent::ChildInserted ); +} + +/*! + \reimp +*/ + +void TQSplitter::styleChange( TQStyle& old ) +{ + updateHandles(); + TQFrame::styleChange( old ); +} + +void TQSplitter::updateHandles() +{ + int hw = handleWidth(); + TQSplitterLayoutStruct *s = d->list.first(); + while ( s ) { + if ( s->isHandle ) + s->sizer = hw; + s = d->list.next(); + } + recalc( isVisible() ); +} + +#ifndef TQT_NO_TEXTSTREAM +/*! + \relates TQSplitter + + Writes the sizes and the hidden state of the widgets in the + splitter \a splitter to the text stream \a ts. + + \sa operator>>(), sizes(), TQWidget::isHidden() +*/ + +TQTextStream& operator<<( TQTextStream& ts, const TQSplitter& splitter ) +{ + TQSplitterLayoutStruct *s = splitter.d->list.first(); + bool first = TRUE; + ts << "["; + + while ( s != 0 ) { + if ( !s->isHandle ) { + if ( !first ) + ts << ","; + + if ( s->wid->isHidden() ) { + ts << "H"; + } else if ( isCollapsed(s->wid) ) { + ts << 0; + } else { + ts << s->getSizer( splitter.orientation() ); + } + first = FALSE; + } + s = splitter.d->list.next(); + } + ts << "]" << endl; + return ts; +} + +/*! + \relates TQSplitter + + Reads the sizes and the hidden state of the widgets in the + splitter \a splitter from the text stream \a ts. The sizes must + have been previously written by the operator<<() function. + + \sa operator<<(), setSizes(), TQWidget::hide() +*/ + +TQTextStream& operator>>( TQTextStream& ts, TQSplitter& splitter ) +{ +#undef SKIP_SPACES +#define SKIP_SPACES() \ + while ( line[i].isSpace() ) \ + i++ + + splitter.processChildEvents(); + TQSplitterLayoutStruct *s = splitter.d->list.first(); + TQString line = ts.readLine(); + int i = 0; + + SKIP_SPACES(); + if ( line[i] == '[' ) { + i++; + SKIP_SPACES(); + while ( line[i] != ']' ) { + while ( s != 0 && s->isHandle ) + s = splitter.d->list.next(); + if ( s == 0 ) + break; + + if ( line[i].upper() == 'H' ) { + s->wid->hide(); + i++; + } else { + s->wid->show(); + int dim = 0; + while ( line[i].digitValue() >= 0 ) { + dim *= 10; + dim += line[i].digitValue(); + i++; + } + s->sizer = dim; + if ( dim == 0 ) + splitter.setGeo( s->wid, 0, 0, FALSE ); + } + SKIP_SPACES(); + if ( line[i] == ',' ) { + i++; + } else { + break; + } + SKIP_SPACES(); + s = splitter.d->list.next(); + } + } + splitter.doResize(); + return ts; +} +#endif + +#endif |