|
|
|
/* This file is part of the KDE libraries
|
|
|
|
Copyright (C) 2000 Max Judin <novaprint@mtu-net.ru>
|
|
|
|
Copyright (C) 2002,2003 Joseph Wenninger <jowenn@kde.org>
|
|
|
|
Copyright (C) 2005 Dominik Haumann <dhdev@gmx.de>
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Library General Public
|
|
|
|
License version 2 as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Library General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
#include "kdockwidget.h"
|
|
|
|
#include "kdockwidget_p.h"
|
|
|
|
#include "kdockwidget_private.h"
|
|
|
|
|
|
|
|
#include <tqpainter.h>
|
|
|
|
#include <tqcursor.h>
|
|
|
|
#include <kdebug.h>
|
|
|
|
#include <tqtimer.h>
|
|
|
|
#include <tqapplication.h>
|
|
|
|
|
|
|
|
#include <math.h> // need ceil
|
|
|
|
|
|
|
|
KDockSplitter::KDockSplitter(TQWidget *parent, const char *name, Orientation orient, int pos)
|
|
|
|
: TQWidget(parent, name)
|
|
|
|
{
|
|
|
|
m_dontRecalc=false;
|
|
|
|
divider = 0L;
|
|
|
|
child0 = 0L;
|
|
|
|
child1 = 0L;
|
|
|
|
fixedWidth0=-1;
|
|
|
|
fixedWidth1=-1;
|
|
|
|
fixedHeight0=-1;
|
|
|
|
fixedHeight1=-1;
|
|
|
|
|
|
|
|
m_orientation = orient;
|
|
|
|
mOpaqueResize = false;
|
|
|
|
mKeepSize = false;
|
|
|
|
setSeparatorPosInPercent( pos );
|
|
|
|
initialised = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::activate(TQWidget *c0, TQWidget *c1)
|
|
|
|
{
|
|
|
|
if ( c0 ) child0 = c0;
|
|
|
|
if ( c1 ) child1 = c1;
|
|
|
|
|
|
|
|
setupMinMaxSize();
|
|
|
|
|
|
|
|
if (divider) delete divider;
|
|
|
|
divider = new TQFrame(this, "pannerdivider");
|
|
|
|
divider->setFrameStyle(TQFrame::Panel | TQFrame::Raised);
|
|
|
|
divider->setLineWidth(1);
|
|
|
|
divider->raise();
|
|
|
|
|
|
|
|
if (m_orientation == Horizontal)
|
|
|
|
divider->setCursor(TQCursor(sizeVerCursor));
|
|
|
|
else
|
|
|
|
divider->setCursor(TQCursor(sizeHorCursor));
|
|
|
|
divider->installEventFilter(this);
|
|
|
|
|
|
|
|
initialised= true;
|
|
|
|
|
|
|
|
updateName();
|
|
|
|
divider->show();
|
|
|
|
|
|
|
|
// without this resize event, things will not work. why exactly? :(
|
|
|
|
resizeEvent(0);
|
|
|
|
|
|
|
|
|
|
|
|
KDockWidget* dw0 = (KDockWidget*) child0;
|
|
|
|
KDockWidget* dw1 = (KDockWidget*) child1;
|
|
|
|
|
|
|
|
// if fixed size is set, restore first, to restore xpos correctly
|
|
|
|
if( fixedWidth0 != -1 || fixedHeight0 != -1 ) restoreFromForcedFixedSize( dw0 );
|
|
|
|
if( fixedWidth1 != -1 || fixedHeight1 != -1 ) restoreFromForcedFixedSize( dw1 );
|
|
|
|
|
|
|
|
|
|
|
|
// now force fixed sizes, if they are set.
|
|
|
|
if( dw0->forcedFixedWidth() != -1 ) {
|
|
|
|
setForcedFixedWidth( dw0, dw0->forcedFixedWidth() );
|
|
|
|
}
|
|
|
|
else if( dw1->forcedFixedWidth() != -1 ) {
|
|
|
|
setForcedFixedWidth( dw1, dw1->forcedFixedWidth() );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( dw0->forcedFixedHeight() != -1 ) {
|
|
|
|
setForcedFixedHeight (dw0, dw0->forcedFixedHeight() );
|
|
|
|
}
|
|
|
|
else if( dw1->forcedFixedHeight() != -1 ) {
|
|
|
|
setForcedFixedHeight( dw1, dw1->forcedFixedHeight() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
void KDockSplitter::delayedResize()
|
|
|
|
{
|
|
|
|
kdDebug(282)<<"*********************** DELAYED RESIZE !!!!!!!!!!!!!!!"<<endl;
|
|
|
|
resizeEvent(0);
|
|
|
|
}*/
|
|
|
|
|
|
|
|
void KDockSplitter::setForcedFixedWidth(KDockWidget *dw,int w)
|
|
|
|
{
|
|
|
|
if (dw==child0)
|
|
|
|
{
|
|
|
|
if (fixedWidth0==-1) savedXPos=xpos;
|
|
|
|
if (w==fixedWidth0) return;
|
|
|
|
fixedWidth0=w;
|
|
|
|
setSeparatorPos(w*factor/width(),true);
|
|
|
|
// kdDebug(282)<<"Set forced fixed width for widget 0 :"<<w<<endl;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (fixedWidth1==-1) savedXPos=xpos;
|
|
|
|
if (w==fixedWidth1) return;
|
|
|
|
fixedWidth1=w;
|
|
|
|
setSeparatorPos((width()-w)*factor/width(),true);
|
|
|
|
// kdDebug(282)<<"Set forced fixed width for widget 1 :"<<w<<endl;
|
|
|
|
}
|
|
|
|
setupMinMaxSize();
|
|
|
|
if (divider) divider->hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::setForcedFixedHeight(KDockWidget *dw,int h)
|
|
|
|
{
|
|
|
|
if (dw==child0)
|
|
|
|
{
|
|
|
|
if (fixedHeight0==-1) savedXPos=xpos;
|
|
|
|
if (h==fixedHeight0) return;
|
|
|
|
fixedHeight0=h;
|
|
|
|
setSeparatorPos(h*factor/height(),true);
|
|
|
|
// // kdDebug(282)<<"Set forced fixed width for widget 0 :"<<h<<endl;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (fixedHeight1==-1) savedXPos=xpos;
|
|
|
|
if (h==fixedHeight1) return;
|
|
|
|
fixedHeight1=h;
|
|
|
|
setSeparatorPos((height()-h)*factor/height(),true);
|
|
|
|
// kdDebug(282)<<"Set forced fixed height for widget 1 :"<<h<<endl;
|
|
|
|
}
|
|
|
|
setupMinMaxSize();
|
|
|
|
if (divider) divider->hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::restoreFromForcedFixedSize(KDockWidget *dw)
|
|
|
|
{
|
|
|
|
if (divider) divider->show();
|
|
|
|
if (dw==child0)
|
|
|
|
{
|
|
|
|
fixedWidth0=-1;
|
|
|
|
fixedHeight0=-1;
|
|
|
|
setSeparatorPos(savedXPos,true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fixedWidth1=-1;
|
|
|
|
fixedHeight1=-1;
|
|
|
|
setSeparatorPos(savedXPos,true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void KDockSplitter::setupMinMaxSize()
|
|
|
|
{
|
|
|
|
// Set the minimum and maximum sizes for the KDockSplitter (this)
|
|
|
|
int minx, maxx, miny, maxy;
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
miny = child0->minimumHeight() + child1->minimumHeight() + 4;
|
|
|
|
maxy = child0->maximumHeight() + child1->maximumHeight() + 4;
|
|
|
|
minx = (child0->minimumWidth() > child1->minimumWidth()) ? child0->minimumWidth() : child1->minimumWidth();
|
|
|
|
maxx = (child0->maximumWidth() > child1->maximumWidth()) ? child0->maximumWidth() : child1->maximumWidth();
|
|
|
|
|
|
|
|
if (miny < 4) miny = 4;
|
|
|
|
if (maxy > 32000) maxy = 32000;
|
|
|
|
if (minx < 2) minx = 2;
|
|
|
|
if (maxx > 32000) maxx = 32000;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
minx = child0->minimumWidth() + child1->minimumWidth() + 4;
|
|
|
|
maxx = child0->maximumWidth() + child1->maximumWidth() + 4;
|
|
|
|
miny = (child0->minimumHeight() > child1->minimumHeight()) ? child0->minimumHeight() : child1->minimumHeight();
|
|
|
|
maxy = (child0->maximumHeight() > child1->maximumHeight()) ? child0->maximumHeight() : child1->maximumHeight();
|
|
|
|
|
|
|
|
if (miny < 2) miny = 2;
|
|
|
|
if (maxy > 32000) maxy = 32000;
|
|
|
|
if (minx < 4) minx = 4;
|
|
|
|
if (maxx > 32000) maxx = 32000;
|
|
|
|
}
|
|
|
|
|
|
|
|
setMinimumSize(minx, miny);
|
|
|
|
setMaximumSize(maxx, maxy);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::deactivate()
|
|
|
|
{
|
|
|
|
if (divider) delete divider;
|
|
|
|
divider = 0L;
|
|
|
|
initialised= false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int KDockSplitter::separatorPosInPercent()
|
|
|
|
{
|
|
|
|
return xpos / (factor/100);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::setSeparatorPosInPercent(int percent)
|
|
|
|
{
|
|
|
|
xpos = percent * (factor/100);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::setSeparatorPos(int pos, bool do_resize)
|
|
|
|
{
|
|
|
|
xpos = pos;
|
|
|
|
if (do_resize)
|
|
|
|
resizeEvent(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::setSeparatorPosX(int pos, bool do_resize)
|
|
|
|
{
|
|
|
|
savedXPos = pos;
|
|
|
|
setSeparatorPos( pos, do_resize );
|
|
|
|
}
|
|
|
|
|
|
|
|
int KDockSplitter::separatorPos() const
|
|
|
|
{
|
|
|
|
return xpos;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::resizeEvent(TQResizeEvent *ev)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// As already stated in the .h file we always have to differentiate
|
|
|
|
// between 6 modes.
|
|
|
|
// If we can cast child0->getWidget() or child1.getWidget() to
|
|
|
|
// KDockContainer* we *do* have a dockwidget around. For dockwidgets
|
|
|
|
// we have to take special care in the resizing routines, for example
|
|
|
|
// if mKeepSize is true and the dockcontainer is on the bottom or right,
|
|
|
|
// we always have to move the xpos splitter position. If there are no
|
|
|
|
// dockcontainers around, resizing is handeled like if child0 would
|
|
|
|
// be a dockcontainer.
|
|
|
|
//
|
|
|
|
|
|
|
|
// kdDebug(282)<<"ResizeEvent :"<< ((initialised) ? "initialised":"not initialised")<<", "<< ((ev) ? "real event":"")<<", "<<(isVisible() ?"visible":"")<<endl;
|
|
|
|
|
|
|
|
if (initialised) {
|
|
|
|
KDockContainer *dc = 0L;
|
|
|
|
KDockWidget *c0 = (KDockWidget*)child0;
|
|
|
|
KDockWidget *c1 = (KDockWidget*)child1;
|
|
|
|
bool stdHandling=false; // true: if closed or nonoverlap mode. false: overlap mode
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check whether this is a real resize event or a pseudo resize event
|
|
|
|
// Real resize events occure if the width() or height() changes. ev != 0L.
|
|
|
|
// Pseudo resize events occure if the dockwidget mode changes (overlaped,
|
|
|
|
// sticky or closed). ev == 0L.
|
|
|
|
//
|
|
|
|
if (ev && isVisible() && divider->isVisible()) {
|
|
|
|
// real resize event.
|
|
|
|
// kdDebug(282)<<"mKeepSize : "<< ((m_orientation == Horizontal) ? "Horizontal":"Vertical") <<endl;
|
|
|
|
|
|
|
|
if (mKeepSize) {
|
|
|
|
// keep the splitter on a fixed position. This may be a bit inaccurate, because
|
|
|
|
// xpos saves a proportional value, which means there might occur rounding errors.
|
|
|
|
// However, this works surprising well!
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
if (ev->oldSize().height() != ev->size().height()) {
|
|
|
|
if( (c1->getWidget()) && (dc=dynamic_cast<KDockContainer*>(c1->getWidget()))) {
|
|
|
|
// dockwidget is on the bottom. move xpos so that the size from child1 stays
|
|
|
|
xpos = (int)ceil(((double)factor) * checkValue(height() - child1->height() - 4) / height());
|
|
|
|
} else {
|
|
|
|
// xpos should not change, the docking is on the top
|
|
|
|
// checkValue is *fuzzy* here, it leads to ugly rounding bugs
|
|
|
|
// In truth, it is not needed, because it is called when calculating the "position".
|
|
|
|
xpos = qRound(((double)xpos) * ev->oldSize().height() / height());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (ev->oldSize().width() != width()) {
|
|
|
|
if( (c1->getWidget()) && (dc=dynamic_cast<KDockContainer*>(c1->getWidget()))) {
|
|
|
|
xpos = (int)ceil(((double)factor) * checkValue(width() - child1->width() - 4) / width());
|
|
|
|
} else {
|
|
|
|
// xpos should not change
|
|
|
|
// checkValue is *fuzzy* here, it leads to ugly rounding bugs
|
|
|
|
xpos = qRound(((double)xpos) * ev->oldSize().width() / width());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// dockwidget size proportional!
|
|
|
|
// Which means, xpos is always right (ratio value). Do nothing! :)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Maybe a multitabbartab was clicked, so force an update of the fixed
|
|
|
|
// values.
|
|
|
|
//
|
|
|
|
if ( isVisible()) {
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
if (fixedHeight0!=-1)
|
|
|
|
xpos = checkValue(fixedHeight0) * factor / height();
|
|
|
|
else if (fixedHeight1!=-1)
|
|
|
|
xpos = checkValue(height()-fixedHeight1) * factor / height();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (fixedWidth0!=-1)
|
|
|
|
xpos = checkValue(fixedWidth0) * factor / width();
|
|
|
|
else if (fixedWidth1!=-1)
|
|
|
|
xpos = checkValue(width()-fixedWidth1) * factor / width();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// else kdDebug(282)<<"Something else happened"<<endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
// --- debugging information ---
|
|
|
|
kdDebug(282) << "isVisible() is : " << isVisible() << endl;
|
|
|
|
kdDebug(282) << "Orientation : " << (m_orientation==Horizontal?"Horizontal":"Vertical")
|
|
|
|
<< endl;
|
|
|
|
kdDebug(282) << "Splitter visibility : " << divider->isVisible() << endl;;
|
|
|
|
kdDebug(282) << "Splitter procentual pos: " << xpos << endl;
|
|
|
|
if (c0->getWidget()) {
|
|
|
|
dc=dynamic_cast<KDockContainer*>(c0->getWidget());
|
|
|
|
kdDebug(282) << "Child 0 KDockContainer?: " << dc << endl;
|
|
|
|
}
|
|
|
|
if (c1->getWidget()) {
|
|
|
|
dc=dynamic_cast<KDockContainer*>(c1->getWidget());
|
|
|
|
kdDebug(282) << "Child 1 KDockContainer?: " << dc << endl;
|
|
|
|
}
|
|
|
|
kdDebug(282) << "Child0 : " << child0 << endl;
|
|
|
|
kdDebug(282) << "child1 : " << child1 << endl;
|
|
|
|
*/
|
|
|
|
|
|
|
|
//
|
|
|
|
// handle overlapped widgets only.
|
|
|
|
//
|
|
|
|
if( ( (m_orientation==Vertical) &&((fixedWidth0==-1) && (fixedWidth1==-1)) ) ||
|
|
|
|
( (m_orientation==Horizontal) &&((fixedHeight0==-1) && (fixedHeight1==-1)) ) ) {
|
|
|
|
if ((c0->getWidget()) && (dc=dynamic_cast<KDockContainer*>(c0->getWidget()))
|
|
|
|
&& (dc->isOverlapMode())) {
|
|
|
|
// child0 ist a KDockContainer
|
|
|
|
int position;
|
|
|
|
child0->show();
|
|
|
|
child0->raise();
|
|
|
|
divider->raise();
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
position = checkValueOverlapped( height() * xpos / factor, child0 );
|
|
|
|
child0->setGeometry(0, 0, width(), position);
|
|
|
|
child1->setGeometry(0, dc->m_nonOverlapSize, width(), height()-dc->m_nonOverlapSize);
|
|
|
|
divider->setGeometry(0, position, width(), 4);
|
|
|
|
} else {
|
|
|
|
position = checkValueOverlapped( width() * xpos / factor, child0 );
|
|
|
|
child0->setGeometry(0, 0, position, height());
|
|
|
|
child1->setGeometry(dc->m_nonOverlapSize, 0, width()-dc->m_nonOverlapSize, height());
|
|
|
|
divider->setGeometry(position, 0, 4, height());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((c1->getWidget()) && (dc=dynamic_cast<KDockContainer*>(c1->getWidget()))
|
|
|
|
&& (dc->isOverlapMode())) {
|
|
|
|
// child1 ist a KDockContainer
|
|
|
|
int position;
|
|
|
|
child1->show();
|
|
|
|
child1->raise();
|
|
|
|
divider->raise();
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
position = checkValueOverlapped( height() * xpos / factor, child1 );
|
|
|
|
child0->setGeometry(0, 0, width(), height()-dc->m_nonOverlapSize);
|
|
|
|
child1->setGeometry(0, position+4, width(), height()-position-4);
|
|
|
|
divider->setGeometry(0, position, width(), 4);
|
|
|
|
} else {
|
|
|
|
position = checkValueOverlapped( width() * xpos / factor, child1 );
|
|
|
|
child0->setGeometry(0, 0, width()-dc->m_nonOverlapSize, height());
|
|
|
|
child1->setGeometry(position+4, 0, width()-position-4, height());
|
|
|
|
divider->setGeometry(position, 0, 4, height());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else // no KDockContainer available, this means the mode cannot be overlapped
|
|
|
|
stdHandling=true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else // no KDockContainer available
|
|
|
|
stdHandling=true;
|
|
|
|
|
|
|
|
//
|
|
|
|
// stdHandling == true means either sticky mode (=nonoverlap mode) or
|
|
|
|
// closed mode. In both modes the widgets do *not* overlap, so we know
|
|
|
|
// the child0 and child1 adjoin.
|
|
|
|
//
|
|
|
|
if (stdHandling) {
|
|
|
|
int position = checkValue( (m_orientation == Vertical ? width() : height()) * xpos / factor );
|
|
|
|
int diff = 0;
|
|
|
|
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
if ((c1->getWidget()) && (dc=dynamic_cast<KDockContainer*>(c1->getWidget()))) {
|
|
|
|
// bottom is dockcontainer
|
|
|
|
if( divider->isVisible() ) {
|
|
|
|
child0->setGeometry(0, 0, width(), position);
|
|
|
|
child1->setGeometry(0, position+4, width(), height()-position-4);
|
|
|
|
} else {
|
|
|
|
child0->setGeometry(0, 0, width(), height()-dc->m_nonOverlapSize);
|
|
|
|
child1->setGeometry(0, height()-dc->m_nonOverlapSize, width(), height());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if( divider->isVisible() ) diff = 4;
|
|
|
|
child0->setGeometry(0, 0, width(), position);
|
|
|
|
child1->setGeometry(0, position+diff, width(), height()-position-diff);
|
|
|
|
}
|
|
|
|
divider->setGeometry(0, position, width(), 4);
|
|
|
|
} else {
|
|
|
|
if ((c1->getWidget()) && (dc=dynamic_cast<KDockContainer*>(c1->getWidget()))) {
|
|
|
|
// right is dockcontainer
|
|
|
|
if( divider->isVisible() ) {
|
|
|
|
child0->setGeometry(0, 0, position, height());
|
|
|
|
child1->setGeometry(position+4, 0, width()-position-4, height());
|
|
|
|
} else {
|
|
|
|
child0->setGeometry(0, 0, width()-dc->m_nonOverlapSize, height());
|
|
|
|
child1->setGeometry(width()-dc->m_nonOverlapSize, 0, width(), height());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if( divider->isVisible() ) diff = 4;
|
|
|
|
child0->setGeometry(0, 0, position, height());
|
|
|
|
child1->setGeometry(position+diff, 0, width()-position-diff, height());
|
|
|
|
}
|
|
|
|
divider->setGeometry(position, 0, 4, height());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int KDockSplitter::checkValueOverlapped(int position, TQWidget *overlappingWidget) const
|
|
|
|
{
|
|
|
|
if (initialised) {
|
|
|
|
if (m_orientation == Vertical) {
|
|
|
|
if (child0==overlappingWidget) {
|
|
|
|
if (position < child0->minimumWidth() || position > width())
|
|
|
|
position = child0->minimumWidth();
|
|
|
|
} else {
|
|
|
|
if (position > (width()-child1->minimumWidth()-4) || position < 0)
|
|
|
|
position = width()-child1->minimumWidth()-4;
|
|
|
|
}
|
|
|
|
} else {// orientation == Horizontal
|
|
|
|
if (child0==overlappingWidget) {
|
|
|
|
if (position < (child0->minimumHeight()) || position > height())
|
|
|
|
position = child0->minimumHeight();
|
|
|
|
} else {
|
|
|
|
if (position>(height()-child1->minimumHeight()-4) || position < 0)
|
|
|
|
position = height()-child1->minimumHeight()-4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
|
|
|
int KDockSplitter::checkValue( int position ) const
|
|
|
|
{
|
|
|
|
if (initialised) {
|
|
|
|
if (m_orientation == Vertical) {
|
|
|
|
if (position < child0->minimumWidth())
|
|
|
|
position = child0->minimumWidth();
|
|
|
|
if ((width()-4-position) < (child1->minimumWidth()))
|
|
|
|
position = width() - (child1->minimumWidth()) - 4;
|
|
|
|
} else {
|
|
|
|
if (position < (child0->minimumHeight()))
|
|
|
|
position = child0->minimumHeight();
|
|
|
|
if ((height()-4-position) < child1->minimumHeight())
|
|
|
|
position = height() - (child1->minimumHeight()) - 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (position < 0) position = 0;
|
|
|
|
|
|
|
|
if ((m_orientation == Vertical) && (position > width()))
|
|
|
|
position = width();
|
|
|
|
if ((m_orientation == Horizontal) && (position > height()))
|
|
|
|
position = height();
|
|
|
|
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KDockSplitter::eventFilter(TQObject *o, TQEvent *e)
|
|
|
|
{
|
|
|
|
TQMouseEvent *mev;
|
|
|
|
bool handled = false;
|
|
|
|
|
|
|
|
switch (e->type()) {
|
|
|
|
case TQEvent::MouseMove:
|
|
|
|
mev= (TQMouseEvent*)e;
|
|
|
|
child0->setUpdatesEnabled(mOpaqueResize);
|
|
|
|
child1->setUpdatesEnabled(mOpaqueResize);
|
|
|
|
if (m_orientation == Horizontal) {
|
|
|
|
if ((fixedHeight0!=-1) || (fixedHeight1!=-1))
|
|
|
|
{
|
|
|
|
handled=true; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mOpaqueResize) {
|
|
|
|
int position = checkValue( mapFromGlobal(mev->globalPos()).y() );
|
|
|
|
divider->move( 0, position );
|
|
|
|
} else {
|
|
|
|
int tmp_xpos = factor * checkValue( mapFromGlobal(mev->globalPos()).y() ) / height();
|
|
|
|
if (tmp_xpos != xpos) {
|
|
|
|
xpos = tmp_xpos;
|
|
|
|
resizeEvent(0);
|
|
|
|
divider->repaint(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((fixedWidth0!=-1) || (fixedWidth1!=-1))
|
|
|
|
{
|
|
|
|
handled=true; break;
|
|
|
|
}
|
|
|
|
if (!mOpaqueResize) {
|
|
|
|
int position = checkValue( mapFromGlobal(TQCursor::pos()).x() );
|
|
|
|
divider->move( position, 0 );
|
|
|
|
} else {
|
|
|
|
int tmp_xpos = factor * checkValue( mapFromGlobal( mev->globalPos()).x() ) / width();
|
|
|
|
if (tmp_xpos != xpos) {
|
|
|
|
xpos = tmp_xpos;
|
|
|
|
resizeEvent(0);
|
|
|
|
divider->repaint(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
handled= true;
|
|
|
|
break;
|
|
|
|
case TQEvent::MouseButtonRelease:
|
|
|
|
child0->setUpdatesEnabled(true);
|
|
|
|
child1->setUpdatesEnabled(true);
|
|
|
|
mev= (TQMouseEvent*)e;
|
|
|
|
if (m_orientation == Horizontal){
|
|
|
|
if ((fixedHeight0!=-1) || (fixedHeight1!=-1))
|
|
|
|
{
|
|
|
|
handled=true; break;
|
|
|
|
}
|
|
|
|
xpos = factor* checkValue( mapFromGlobal(mev->globalPos()).y() ) / height();
|
|
|
|
resizeEvent(0);
|
|
|
|
divider->repaint(true);
|
|
|
|
} else {
|
|
|
|
if ((fixedWidth0!=-1) || (fixedWidth1!=-1))
|
|
|
|
{
|
|
|
|
handled=true; break;
|
|
|
|
}
|
|
|
|
xpos = factor* checkValue( mapFromGlobal(mev->globalPos()).x() ) / width();
|
|
|
|
resizeEvent(0);
|
|
|
|
divider->repaint(true);
|
|
|
|
}
|
|
|
|
handled= true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return (handled) ? true : TQWidget::eventFilter( o, e );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KDockSplitter::event( TQEvent* e )
|
|
|
|
{
|
|
|
|
if ( e->type() == TQEvent::LayoutHint ){
|
|
|
|
// change children min/max size. This is needed, otherwise
|
|
|
|
// it is possible the divider get's out of bounds.
|
|
|
|
setupMinMaxSize();
|
|
|
|
resizeEvent(0);
|
|
|
|
}
|
|
|
|
return TQWidget::event(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
TQWidget* KDockSplitter::getAnother( TQWidget* w ) const
|
|
|
|
{
|
|
|
|
return ( w == child0 ) ? child1 : child0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::updateName()
|
|
|
|
{
|
|
|
|
if ( !initialised ) return;
|
|
|
|
|
|
|
|
TQString new_name = TQString( child0->name() ) + "," + child1->name();
|
|
|
|
parentWidget()->setName( new_name.latin1() );
|
|
|
|
parentWidget()->setCaption( child0->caption() + "," + child1->caption() );
|
|
|
|
parentWidget()->repaint( false );
|
|
|
|
|
|
|
|
((KDockWidget*)parentWidget())->firstName = child0->name();
|
|
|
|
((KDockWidget*)parentWidget())->lastName = child1->name();
|
|
|
|
((KDockWidget*)parentWidget())->splitterOrientation = m_orientation;
|
|
|
|
|
|
|
|
TQWidget* p = parentWidget()->parentWidget();
|
|
|
|
if ( p && p->inherits("KDockSplitter" ) )
|
|
|
|
((KDockSplitter*)p)->updateName();
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::setOpaqueResize(bool b)
|
|
|
|
{
|
|
|
|
mOpaqueResize = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KDockSplitter::opaqueResize() const
|
|
|
|
{
|
|
|
|
return mOpaqueResize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockSplitter::setKeepSize(bool b)
|
|
|
|
{
|
|
|
|
mKeepSize = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KDockSplitter::keepSize() const
|
|
|
|
{
|
|
|
|
return mKeepSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
KDockButton_Private::KDockButton_Private( TQWidget *parent, const char * name )
|
|
|
|
:TQPushButton( parent, name )
|
|
|
|
{
|
|
|
|
moveMouse = false;
|
|
|
|
setFocusPolicy( NoFocus );
|
|
|
|
}
|
|
|
|
|
|
|
|
KDockButton_Private::~KDockButton_Private()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockButton_Private::drawButton( TQPainter* p )
|
|
|
|
{
|
|
|
|
p->fillRect( 0,0, width(), height(), TQBrush(colorGroup().brush(TQColorGroup::Background)) );
|
|
|
|
p->drawPixmap( (width() - pixmap()->width()) / 2, (height() - pixmap()->height()) / 2, *pixmap() );
|
|
|
|
if ( moveMouse && !isDown() ){
|
|
|
|
p->setPen( white );
|
|
|
|
p->moveTo( 0, height() - 1 );
|
|
|
|
p->lineTo( 0, 0 );
|
|
|
|
p->lineTo( width() - 1, 0 );
|
|
|
|
|
|
|
|
p->setPen( colorGroup().dark() );
|
|
|
|
p->lineTo( width() - 1, height() - 1 );
|
|
|
|
p->lineTo( 0, height() - 1 );
|
|
|
|
}
|
|
|
|
if ( isOn() || isDown() ){
|
|
|
|
p->setPen( colorGroup().dark() );
|
|
|
|
p->moveTo( 0, height() - 1 );
|
|
|
|
p->lineTo( 0, 0 );
|
|
|
|
p->lineTo( width() - 1, 0 );
|
|
|
|
|
|
|
|
p->setPen( white );
|
|
|
|
p->lineTo( width() - 1, height() - 1 );
|
|
|
|
p->lineTo( 0, height() - 1 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockButton_Private::enterEvent( TQEvent * )
|
|
|
|
{
|
|
|
|
moveMouse = true;
|
|
|
|
repaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockButton_Private::leaveEvent( TQEvent * )
|
|
|
|
{
|
|
|
|
moveMouse = false;
|
|
|
|
repaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
KDockWidgetPrivate::KDockWidgetPrivate()
|
|
|
|
: TQObject()
|
|
|
|
,index(-1)
|
|
|
|
,splitPosInPercent(50)
|
|
|
|
,pendingFocusInEvent(false)
|
|
|
|
,blockHasUndockedSignal(false)
|
|
|
|
,pendingDtor(false)
|
|
|
|
,forcedWidth(-1)
|
|
|
|
,forcedHeight(-1)
|
|
|
|
,isContainer(false)
|
|
|
|
,container(0)
|
|
|
|
,resizePos(0,0)
|
|
|
|
,resizing(false)
|
|
|
|
{
|
|
|
|
#ifndef NO_KDE2
|
|
|
|
windowType = NET::Normal;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
_parent = 0L;
|
|
|
|
transient = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
KDockWidgetPrivate::~KDockWidgetPrivate()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void KDockWidgetPrivate::slotFocusEmbeddedWidget(TQWidget* w)
|
|
|
|
{
|
|
|
|
if (w) {
|
|
|
|
TQWidget* embeddedWdg = ((KDockWidget*)w)->getWidget();
|
|
|
|
if (embeddedWdg && ((embeddedWdg->focusPolicy() == TQWidget::ClickFocus) || (embeddedWdg->focusPolicy() == TQWidget::StrongFocus))) {
|
|
|
|
embeddedWdg->setFocus();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef NO_INCLUDE_MOCFILES // for Qt-only projects, because tmake doesn't take this name
|
|
|
|
#include "kdockwidget_private.moc"
|
|
|
|
#endif
|