a29ae57a7d
comments below. * New Flow Layout. This provides simple icon view-esque layout of items. Useful for icons for documents , applications or other tasks on the desktop for example. Supports non-equally sized items. Works well when used with the LayoutAnimator class to animate insertions and removals. * Re-wrote BoxLayout and removed old HBoxLayout,VBoxLayout classes which had a lot of code duplication. BoxLayout class now takes a direction argument in the constructor, ala. QBoxLayout. New BoxLayout class actually takes minimumSize() , maximumSize() of items into account. The Qt layout code for box and grid layouts is surprisingly sophisticated, so the results from BoxLayout probably will not be as good in certain situations but it should do for the panel. New BoxLayout also has support for LayoutAnimator * Fix Plasma::HBoxLayout and Plasma::VBoxLayout to use margin() rather than spacing() for the distance from the top and left margins respectively. * Fix Plasma::Applet::contentSize() to return the actual content size rather than a size hint. Added a new method contentSizeHint() which applets use to provide a hint about suitable content size. Existing implementations of contentSize() in applets need to be renamed to contentSizeHint(). The arguments and return type are the same as before. * Install the LayoutAnimator header so that applets can use it svn path=/trunk/KDE/kdebase/workspace/libs/plasma/; revision=707275
472 lines
13 KiB
C++
472 lines
13 KiB
C++
/*
|
|
* Copyright 2007 by Matias Valdenegro T. <mvaldenegro@informatica.utem.cl>
|
|
* Copyright 2007 by Robert Knight <robertknight@gmail.com>
|
|
*
|
|
* This program 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 program 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 General Public License for more details
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this program; if not, write to the
|
|
* Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "boxlayout.h"
|
|
|
|
#include <QtCore/QList>
|
|
#include <QtCore/QTimeLine>
|
|
|
|
#include <KDebug>
|
|
|
|
#include "layoutanimator.h"
|
|
|
|
namespace Plasma
|
|
{
|
|
|
|
class BoxLayout::Private
|
|
{
|
|
public:
|
|
BoxLayout *const q;
|
|
Direction direction;
|
|
QRectF geometry;
|
|
QList<LayoutItem*> children;
|
|
|
|
Private(BoxLayout *parent)
|
|
: q(parent)
|
|
, direction(LeftToRight)
|
|
{
|
|
}
|
|
|
|
// returns the component of 'size' in the expanding direction
|
|
// of this layout
|
|
qreal size(const QSizeF& size) const
|
|
{
|
|
switch (direction) {
|
|
case LeftToRight:
|
|
case RightToLeft:
|
|
return size.width();
|
|
case TopToBottom:
|
|
case BottomToTop:
|
|
return size.height();
|
|
default:
|
|
Q_ASSERT(false);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// returns the direction in which this layout expands
|
|
// or shrinks
|
|
Qt::Orientation expandingDirection() const
|
|
{
|
|
switch ( direction ) {
|
|
case LeftToRight:
|
|
case RightToLeft:
|
|
return Qt::Horizontal;
|
|
case TopToBottom:
|
|
case BottomToTop:
|
|
return Qt::Vertical;
|
|
default:
|
|
Q_ASSERT(false);
|
|
return Qt::Horizontal;
|
|
}
|
|
}
|
|
|
|
// returns the position from which layouting should
|
|
// begin depending on the direction of this layout
|
|
qreal startPos(const QRectF& geometry) const
|
|
{
|
|
switch ( direction ) {
|
|
case LeftToRight:
|
|
case TopToBottom:
|
|
return 0;
|
|
case RightToLeft:
|
|
return geometry.width();
|
|
case BottomToTop:
|
|
return geometry.height();
|
|
default:
|
|
Q_ASSERT(false);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// lays out an item
|
|
//
|
|
// 'geometry' the geometry of the layout
|
|
// 'item' the item whoose geometry should be altered
|
|
// 'pos' the position of the item (in the expanding direction of the layout)
|
|
// 'size' the size of the item (in the expanding direction of the layout)
|
|
//
|
|
// returns the position for the next item in the layout
|
|
//
|
|
qreal layoutItem(const QRectF& geometry , LayoutItem *item , const qreal pos , qreal size)
|
|
{
|
|
//qDebug() << "layoutItem: " << direction << "item size" << size;
|
|
|
|
QRectF newGeometry;
|
|
qreal newPos = 0;
|
|
|
|
qreal top = 0;
|
|
qreal height = 0;
|
|
|
|
QSizeF minSize = item->minimumSize();
|
|
QSizeF maxSize = item->maximumSize();
|
|
switch ( direction ) {
|
|
case LeftToRight:
|
|
case RightToLeft:
|
|
height = qBound(minSize.height(),geometry.height(),maxSize.height());
|
|
top = geometry.top();
|
|
break;
|
|
case TopToBottom:
|
|
case BottomToTop:
|
|
height = qBound(minSize.width(),geometry.width(),maxSize.width());
|
|
top = geometry.left();
|
|
break;
|
|
}
|
|
|
|
switch ( direction ) {
|
|
case LeftToRight:
|
|
newGeometry = QRectF(pos,top,size,height);
|
|
newPos = pos+size+q->spacing();
|
|
break;
|
|
case RightToLeft:
|
|
newGeometry = QRectF(geometry.width()-pos-size,top,
|
|
size,height);
|
|
newPos = pos-size-q->spacing();
|
|
break;
|
|
case TopToBottom:
|
|
newGeometry = QRectF(top,pos,height,size);
|
|
newPos = pos+size+q->spacing();
|
|
break;
|
|
case BottomToTop:
|
|
newGeometry = QRectF(top,geometry.height()-pos-size,
|
|
height,size);
|
|
newPos = pos-size-q->spacing();
|
|
break;
|
|
}
|
|
|
|
// qDebug() << "Item geometry: " << newGeometry;
|
|
|
|
if ( q->animator() )
|
|
q->animator()->setGeometry(item,newGeometry);
|
|
else
|
|
item->setGeometry(newGeometry);
|
|
|
|
return newPos;
|
|
}
|
|
|
|
enum SizeType
|
|
{
|
|
MinSize,
|
|
MaxSize,
|
|
HintSize
|
|
};
|
|
|
|
// this provides a + function which can be passed as the 'op'
|
|
// argument to calculateSize
|
|
static qreal sum(const qreal a , const qreal b)
|
|
{
|
|
return a+b;
|
|
}
|
|
|
|
// calcualtes a size hint or value for this layout
|
|
// 'sizeType' - The item size ( minimum , maximum , hint ) to use
|
|
// 'dir' - The direction component of the item size to use
|
|
// 'op' - A function to apply to the size of each item in the layout
|
|
// , usually qMax,qMin or sum
|
|
template <class T>
|
|
qreal calculateSize(SizeType sizeType , Qt::Orientation dir , T (*op)(T,T)) const
|
|
{
|
|
qreal value = 0;
|
|
for ( int i = 0 ; i < children.count() ; i++ ) {
|
|
|
|
QSizeF itemSize;
|
|
switch ( sizeType ) {
|
|
case MinSize:
|
|
itemSize = children[i]->minimumSize();
|
|
break;
|
|
case MaxSize:
|
|
itemSize = children[i]->maximumSize();
|
|
break;
|
|
case HintSize:
|
|
itemSize = children[i]->sizeHint();
|
|
break;
|
|
}
|
|
|
|
if ( dir == Qt::Horizontal ) {
|
|
value = op(value,itemSize.width());
|
|
} else {
|
|
value = op(value,itemSize.height());
|
|
}
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
// calculates a size hint or value for this layout
|
|
// 'calculateSizeType' specifies the value to be calculated
|
|
QSizeF calculateSize(SizeType calculateSizeType) const
|
|
{
|
|
QSizeF result;
|
|
|
|
const qreal totalMargin = q->margin() * 2;
|
|
const qreal totalSpacing = q->spacing() * (children.count()-1);
|
|
|
|
switch ( direction ) {
|
|
case LeftToRight:
|
|
case RightToLeft:
|
|
result = QSizeF(calculateSize(calculateSizeType,Qt::Horizontal,sum),
|
|
calculateSize(calculateSizeType,Qt::Vertical,qMax<qreal>));
|
|
|
|
result.rwidth() += totalMargin + totalSpacing;
|
|
result.rheight() += totalMargin;
|
|
|
|
break;
|
|
case TopToBottom:
|
|
case BottomToTop:
|
|
result = QSizeF(calculateSize(calculateSizeType,Qt::Horizontal,qMax<qreal>),
|
|
calculateSize(calculateSizeType,Qt::Vertical,sum));
|
|
|
|
result.rheight() += totalMargin + totalSpacing;
|
|
result.rwidth() += totalMargin;
|
|
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
};
|
|
|
|
|
|
BoxLayout::BoxLayout(Direction direction , LayoutItem *parent)
|
|
: Layout(parent),
|
|
d(new Private(this))
|
|
{
|
|
if (parent) {
|
|
parent->setLayout(this);
|
|
}
|
|
|
|
d->direction = direction;
|
|
}
|
|
|
|
void BoxLayout::setDirection(Direction direction)
|
|
{
|
|
d->direction = direction;
|
|
update();
|
|
}
|
|
BoxLayout::Direction BoxLayout::direction() const
|
|
{
|
|
return d->direction;
|
|
}
|
|
|
|
BoxLayout::~BoxLayout()
|
|
{
|
|
foreach (LayoutItem* item, d->children) {
|
|
item->unsetManagingLayout(this);
|
|
}
|
|
delete d;
|
|
}
|
|
|
|
Qt::Orientations BoxLayout::expandingDirections() const
|
|
{
|
|
switch ( d->direction ) {
|
|
case LeftToRight:
|
|
case RightToLeft:
|
|
return Qt::Horizontal;
|
|
case TopToBottom:
|
|
case BottomToTop:
|
|
return Qt::Vertical;
|
|
default:
|
|
Q_ASSERT(false);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
QRectF BoxLayout::geometry() const
|
|
{
|
|
return d->geometry;
|
|
}
|
|
|
|
int BoxLayout::count() const
|
|
{
|
|
return d->children.count();
|
|
}
|
|
|
|
void BoxLayout::insertItem(int index, LayoutItem *item)
|
|
{
|
|
if (!item) {
|
|
return;
|
|
}
|
|
|
|
item->setManagingLayout(this);
|
|
|
|
if ( index == -1 )
|
|
index = d->children.size();
|
|
|
|
d->children.insert(index, item);
|
|
|
|
if ( animator() )
|
|
animator()->setCurrentState(item,LayoutAnimator::InsertedState);
|
|
|
|
update();
|
|
}
|
|
|
|
void BoxLayout::addItem(LayoutItem *item)
|
|
{
|
|
insertItem(-1,item);
|
|
}
|
|
|
|
void BoxLayout::removeItem(LayoutItem *item)
|
|
{
|
|
if (!item) {
|
|
return;
|
|
}
|
|
|
|
item->unsetManagingLayout(this);
|
|
d->children.removeAll(item);
|
|
|
|
if ( animator() )
|
|
animator()->setCurrentState(item,LayoutAnimator::RemovedState);
|
|
|
|
update();
|
|
}
|
|
|
|
int BoxLayout::indexOf(LayoutItem *l) const
|
|
{
|
|
return d->children.indexOf(l);
|
|
}
|
|
|
|
LayoutItem *BoxLayout::itemAt(int i) const
|
|
{
|
|
return d->children[i];
|
|
}
|
|
|
|
LayoutItem *BoxLayout::takeAt(int i)
|
|
{
|
|
return d->children.takeAt(i);
|
|
|
|
update();
|
|
}
|
|
|
|
void BoxLayout::setGeometry(const QRectF& geo)
|
|
{
|
|
QRectF geometry = geo.adjusted(margin(),margin(),-margin(),-margin());
|
|
|
|
//qDebug() << "geo before " << geo << "after" << geometry << "margin" << margin();
|
|
//qDebug() << "Box layout beginning with geo" << geometry;
|
|
//qDebug() << "This box max size" << maximumSize();
|
|
|
|
QVector<qreal> sizes(count());
|
|
QVector<qreal> expansionSpace(count());
|
|
|
|
qreal available = d->size(geometry.size()) - spacing()*(d->children.count()-1);
|
|
qreal perItemSize = available / count();
|
|
|
|
// initial distribution of space to items
|
|
for ( int i = 0 ; i < sizes.count() ; i++ ) {
|
|
const LayoutItem *item = d->children[i];
|
|
|
|
const bool isExpanding = item->expandingDirections() & d->expandingDirection();
|
|
|
|
if ( isExpanding )
|
|
sizes[i] = perItemSize;
|
|
else
|
|
sizes[i] = d->size(item->sizeHint());
|
|
|
|
const qreal minItemSize = d->size(item->minimumSize());
|
|
const qreal maxItemSize = d->size(item->maximumSize());
|
|
|
|
// qDebug() << "Layout max item " << i << "size: " << maxItemSize;
|
|
|
|
sizes[i] = qMin( sizes[i] , maxItemSize );
|
|
sizes[i] = qMax( sizes[i] , minItemSize );
|
|
|
|
// qDebug() << "Available: " << available << "per item:" << perItemSize <<
|
|
// "Initial size: " << sizes[i];
|
|
|
|
if ( isExpanding )
|
|
expansionSpace[i] = maxItemSize-sizes[i];
|
|
else
|
|
expansionSpace[i] = 0;
|
|
|
|
// adjust the per-item size if the space was over or under used
|
|
if ( sizes[i] != perItemSize && i != sizes.count()-1 ) {
|
|
perItemSize = available / (sizes.count()-i-1);
|
|
}
|
|
|
|
available -= sizes[i];
|
|
}
|
|
|
|
// distribute out any remaining space to items which can still expand
|
|
//
|
|
// space is distributed equally amongst remaining items until we run
|
|
// out of space or items to expand
|
|
int expandable = sizes.count();
|
|
const qreal threshold = 1.0;
|
|
while ( available > threshold && expandable > 0 ) {
|
|
|
|
qreal extraSpace = available / expandable;
|
|
for ( int i = 0 ; i < sizes.count() ; i++ ) {
|
|
if ( expansionSpace[i] > threshold ) {
|
|
qreal oldSize = sizes[i];
|
|
|
|
sizes[i] += qMin(extraSpace,expansionSpace[i]);
|
|
|
|
expansionSpace[i] -= sizes[i]-oldSize;
|
|
available -= sizes[i]-oldSize;
|
|
} else {
|
|
expandable--;
|
|
}
|
|
}
|
|
}
|
|
|
|
// set items' geometry according to new sizes
|
|
qreal pos = d->startPos(geometry);
|
|
for ( int i = 0 ; i < sizes.count() ; i++ ) {
|
|
|
|
//QObject *obj = dynamic_cast<QObject*>(d->children[i]);
|
|
//if ( obj )
|
|
//qDebug() << "Item " << i << obj->metaObject()->className() << "size:" << sizes[i];
|
|
|
|
pos = d->layoutItem( geometry , d->children[i] , pos , sizes[i] );
|
|
}
|
|
|
|
d->geometry = geometry;
|
|
|
|
if ( animator() && animator()->timeLine() )
|
|
animator()->timeLine()->start();
|
|
}
|
|
|
|
|
|
QSizeF BoxLayout::maximumSize() const
|
|
{
|
|
return Layout::maximumSize();
|
|
}
|
|
QSizeF BoxLayout::minimumSize() const
|
|
{
|
|
return d->calculateSize(Private::MinSize);
|
|
}
|
|
QSizeF BoxLayout::sizeHint() const
|
|
{
|
|
return d->calculateSize(Private::HintSize);
|
|
}
|
|
|
|
HBoxLayout::HBoxLayout(LayoutItem *parent)
|
|
: BoxLayout(LeftToRight,parent)
|
|
{
|
|
}
|
|
|
|
VBoxLayout::VBoxLayout(LayoutItem *parent)
|
|
: BoxLayout(TopToBottom,parent)
|
|
{
|
|
}
|
|
|
|
|
|
} // Plasma namespace
|
|
|