kde-workspace/plasma/applets/quicklaunch/icongridlayout.cpp
Ivailo Monev efbde9e504 plasma: organize sub-directories
Signed-off-by: Ivailo Monev <xakepa10@gmail.com>
2015-12-24 22:48:47 +02:00

422 lines
12 KiB
C++

/***************************************************************************
* Copyright (C) 2010 - 2011 by Ingomar Wesp <ingomar@wesp.name> *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* 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 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 "icongridlayout.h"
// Qt
#include <QtCore/qnamespace.h>
#include <QtGlobal>
#include <QtCore/QList>
#include <QtCore/qsize.h>
#include <QtCore/qrect.h>
#include <QtGui/QGraphicsLayout>
#include <QtGui/QGraphicsLayoutItem>
#include <QtGui/QSizePolicy>
// KDE
#include <KDebug>
#include <KIconLoader>
// stdlib
#include <math.h>
namespace Quicklaunch {
const int IconGridLayout::DEFAULT_CELL_SPACING = 4;
IconGridLayout::IconGridLayout(QGraphicsLayoutItem *parent)
: QGraphicsLayout(parent),
m_items(),
m_mode(PreferRows),
m_cellSpacing(DEFAULT_CELL_SPACING),
m_maxSectionCount(0),
m_maxSectionCountForced(false),
m_rowCount(0),
m_columnCount(0),
m_rowHeights(),
m_columnWidths()
{
setContentsMargins(0, 0, 0, 0);
QSizePolicy sizePolicy(
QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
sizePolicy.setHeightForWidth(true);
sizePolicy.setHorizontalStretch(1);
sizePolicy.setVerticalStretch(1);
setSizePolicy(sizePolicy);
}
IconGridLayout::~IconGridLayout()
{
Q_FOREACH(QGraphicsLayoutItem *item, m_items) {
if (item->ownedByLayout()) {
delete item;
}
}
m_items.clear();
}
IconGridLayout::Mode IconGridLayout::mode() const
{
return m_mode;
}
void IconGridLayout::setMode(Mode mode)
{
if (mode == m_mode) {
return;
}
m_mode = mode;
updateGridParameters();
invalidate();
}
int IconGridLayout::cellSpacing() const
{
return m_cellSpacing;
}
void IconGridLayout::setCellSpacing(int cellSpacing)
{
cellSpacing = qMax(0, cellSpacing);
if (cellSpacing == m_cellSpacing) {
return;
}
m_cellSpacing = cellSpacing;
updateGridParameters();
invalidate();
}
int IconGridLayout::maxSectionCount() const
{
return m_maxSectionCount;
}
void IconGridLayout::setMaxSectionCount(int maxSectionCount)
{
if (m_maxSectionCount == maxSectionCount) {
return;
}
m_maxSectionCount = maxSectionCount;
updateGridParameters();
invalidate();
}
bool IconGridLayout::maxSectionCountForced() const
{
return m_maxSectionCountForced;
}
void IconGridLayout::setMaxSectionCountForced(bool enable)
{
if (m_maxSectionCountForced == enable) {
return;
}
m_maxSectionCountForced = enable;
updateGridParameters();
invalidate();
}
void IconGridLayout::addItem(QGraphicsLayoutItem *item)
{
m_items.append(item);
addChildLayoutItem(item);
item->setParentLayoutItem(this);
updateGridParameters();
invalidate();
}
void IconGridLayout::insertItem(int index, QGraphicsLayoutItem *item)
{
m_items.insert(index, item);
addChildLayoutItem(item);
item->setParentLayoutItem(this);
updateGridParameters();
invalidate();
}
void IconGridLayout::moveItem(int from, int to)
{
m_items.move(from, to);
invalidate();
}
int IconGridLayout::rowCount() const
{
return m_rowCount;
}
int IconGridLayout::columnCount() const
{
return m_columnCount;
}
int IconGridLayout::count() const
{
return m_items.size();
}
QGraphicsLayoutItem *IconGridLayout::itemAt(int index) const
{
return m_items[index];
}
QGraphicsLayoutItem *IconGridLayout::itemAt(int row, int column) const
{
return m_items[row * m_columnCount + column];
}
void IconGridLayout::removeAt(int index)
{
QGraphicsLayoutItem *item = m_items.takeAt(index);
item->setParentLayoutItem(0);
if (item->ownedByLayout()) {
delete item;
}
updateGridParameters();
invalidate();
}
void IconGridLayout::setGeometry(const QRectF &rect)
{
QGraphicsLayout::setGeometry(rect);
updateGridParameters();
qreal offsetLeft =
qMax<qreal>(
contentsRect().left(),
(contentsRect().width() - preferredWidth()) / 2);
qreal offsetTop =
qMax<qreal>(
contentsRect().top(),
(contentsRect().height() - preferredHeight()) / 2);
QPointF pos(offsetLeft, offsetTop);
QSizeF size;
int itemCount = m_items.size();
for (int i = 0; i < itemCount; i++) {
int row = i / m_columnCount;
int column = i % m_columnCount;
if (column == 0) {
size.setHeight(m_rowHeights.at(row));
if (row > 0) {
pos.rx() = offsetLeft;
pos.ry() += m_rowHeights.at(row-1) + m_cellSpacing;
}
} else {
pos.rx() += m_columnWidths.at(column-1) + m_cellSpacing;
}
size.setWidth(m_columnWidths.at(column));
m_items[i]->setGeometry(QRectF(pos, size));
}
}
QSizeF IconGridLayout::sizeHint(
Qt::SizeHint which, const QSizeF &constraint) const
{
Q_UNUSED(constraint);
switch(which) {
case Qt::PreferredSize: return m_preferredSizeHint;
case Qt::MinimumSize:
if (m_mode == PreferRows) {
return QSizeF(m_preferredSizeHint.width(), KIconLoader::SizeSmall);
}
else {
return QSizeF(KIconLoader::SizeSmall, m_preferredSizeHint.height());
}
default:
return QSizeF();
}
}
void IconGridLayout::computeGridParameters(
QList<int> &rowHeights, QList<int> &columnWidths,
QSizeF &preferredSize) const
{
columnWidths.clear();
rowHeights.clear();
const int itemCount = m_items.size();
if (itemCount == 0) {
preferredSize.setWidth(.0);
preferredSize.setHeight(.0);
return;
}
int rowCount;
int columnCount;
if (m_mode == PreferRows) {
const int height = int(contentsRect().height());
int minRowHeight = 0;
Q_FOREACH(QGraphicsLayoutItem *item, m_items) {
minRowHeight = qMax(minRowHeight, (int)item->minimumHeight());
}
if (m_maxSectionCount > 0 && m_maxSectionCountForced) {
rowCount = qMin(itemCount, m_maxSectionCount);
}
else {
rowCount = height / (minRowHeight + m_cellSpacing);
rowCount = qBound(1, rowCount, itemCount);
if (m_maxSectionCount > 0) {
rowCount = qMin(rowCount, m_maxSectionCount);
}
}
columnCount = ceil(double(itemCount) / rowCount);
// Determine row heights.
int maxRowHeight =
qMax(minRowHeight,
(height - (rowCount - 1) * m_cellSpacing) / rowCount);
for (int row = 0; row < rowCount; row++) {
int desiredRowHeight = 0;
for (int column = 0; column < columnCount; column++) {
int itemIndex = row * columnCount + column;
if (itemIndex >= itemCount) {
break;
}
desiredRowHeight =
qMax(desiredRowHeight, int(m_items.at(itemIndex)->preferredHeight()));
}
rowHeights.append(desiredRowHeight < maxRowHeight ? desiredRowHeight : maxRowHeight);
}
// Determine column widths.
for (int column = 0; column < columnCount; column++) {
int columnWidth = 0;
for (int row = 0; row < rowCount; row++) {
int itemIndex = row * columnCount + column;
if (itemIndex >= itemCount) {
break;
}
int preferredItemWidth =
int(m_items.at(itemIndex)->effectiveSizeHint(Qt::PreferredSize, QSizeF(-1, rowHeights.at(row))).width());
columnWidth = qMax(columnWidth, preferredItemWidth);
}
columnWidths.append(columnWidth);
}
} else { // m_mode == PreferColumns
const int width = int(contentsRect().width());
int minColumnWidth = 0;
Q_FOREACH(QGraphicsLayoutItem *item, m_items) {
minColumnWidth = qMax(minColumnWidth, (int)item->minimumWidth());
}
if (m_maxSectionCount > 0 && m_maxSectionCountForced) {
columnCount = qMin(itemCount, m_maxSectionCount);
}
else {
columnCount = width / (minColumnWidth + m_cellSpacing);
columnCount = qBound(1, columnCount, itemCount);
if (m_maxSectionCount > 0) {
columnCount = qMin(columnCount, m_maxSectionCount);
}
}
rowCount = ceil(double(itemCount) / columnCount);
// Determine column widths.
int maxColumnWidth =
qMax(minColumnWidth,
(width - (columnCount - 1) * m_cellSpacing) / columnCount);
for (int column = 0; column < columnCount; column++) {
int desiredColumnWidth = 0;
for (int row = 0; row < rowCount; row++) {
int itemIndex = row * columnCount + column;
if (itemIndex >= itemCount) {
break;
}
desiredColumnWidth =
qMax(desiredColumnWidth, int(m_items.at(itemIndex)->preferredWidth()));
}
columnWidths.append(desiredColumnWidth < maxColumnWidth ? desiredColumnWidth : maxColumnWidth);
}
// Determine row heights.
for (int row = 0; row < rowCount; row++) {
int rowHeight = 0;
for (int column = 0; column < columnCount; column++) {
int itemIndex = row * columnCount + column;
if (itemIndex >= itemCount) {
break;
}
int preferredItemHeight =
int(m_items.at(itemIndex)->effectiveSizeHint(Qt::PreferredSize, QSizeF(columnWidths.at(column), -1)).height());
rowHeight = qMax(rowHeight, preferredItemHeight);
}
rowHeights.append(rowHeight);
}
}
Q_ASSERT(rowCount > 0 && columnCount > 0);
Q_ASSERT(rowHeights.length() == rowCount && columnWidths.length() == columnCount);
preferredSize.setWidth((columnCount - 1) * m_cellSpacing);
preferredSize.setHeight((rowCount - 1) * m_cellSpacing);
for (int i = 0; i < columnCount; i++) {
preferredSize.rwidth() += columnWidths.at(i);
}
for (int i = 0; i < rowCount; i++) {
preferredSize.rheight() += rowHeights.at(i);
}
}
void IconGridLayout::updateGridParameters()
{
QSizeF newPreferredSize;
computeGridParameters(
m_rowHeights, m_columnWidths,
newPreferredSize);
m_rowCount = m_rowHeights.size();
m_columnCount = m_columnWidths.size();
if (newPreferredSize != m_preferredSizeHint) {
m_preferredSizeHint = newPreferredSize;
updateGeometry();
}
}
}