2022-04-02 18:24:21 +05:00

135 lines
4.4 KiB
C++

/*
SPDX-FileCopyrightText: 2019 Marco Martin <mart@kde.org>
SPDX-License-Identifier: LGPL-2.0-or-later
*/
#include "abstractlayoutmanager.h"
#include "itemcontainer.h"
#include <cmath>
AbstractLayoutManager::AbstractLayoutManager(AppletsLayout *layout)
: QObject(layout)
, m_layout(layout)
{
}
AbstractLayoutManager::~AbstractLayoutManager()
{
}
AppletsLayout *AbstractLayoutManager::layout() const
{
return m_layout;
}
QSizeF AbstractLayoutManager::cellSize() const
{
return m_cellSize;
}
QSizeF AbstractLayoutManager::cellAlignedContainingSize(const QSizeF &size) const
{
return QSizeF(m_cellSize.width() * ceil(size.width() / m_cellSize.width()), m_cellSize.height() * ceil(size.height() / m_cellSize.height()));
}
void AbstractLayoutManager::setCellSize(const QSizeF &size)
{
m_cellSize = size;
}
QRectF AbstractLayoutManager::candidateGeometry(ItemContainer *item) const
{
const QRectF originalItemRect = QRectF(item->x(), item->y(), item->width(), item->height());
// TODO: a default minimum size
QSizeF minimumSize = QSize(m_layout->minimumItemWidth(), m_layout->minimumItemHeight());
if (item->layoutAttached()) {
minimumSize = QSizeF(qMax(minimumSize.width(), item->layoutAttached()->property("minimumWidth").toReal()),
qMax(minimumSize.height(), item->layoutAttached()->property("minimumHeight").toReal()));
}
const QRectF ltrRect = nextAvailableSpace(item, minimumSize, AppletsLayout::LeftToRight);
const QRectF rtlRect = nextAvailableSpace(item, minimumSize, AppletsLayout::RightToLeft);
const QRectF ttbRect = nextAvailableSpace(item, minimumSize, AppletsLayout::TopToBottom);
const QRectF bttRect = nextAvailableSpace(item, minimumSize, AppletsLayout::BottomToTop);
// Take the closest rect, unless the item prefers a particular positioning strategy
QMap<int, QRectF> distances;
if (!ltrRect.isEmpty()) {
const int dist =
item->preferredLayoutDirection() == AppletsLayout::LeftToRight ? 0 : QPointF(originalItemRect.center() - ltrRect.center()).manhattanLength();
distances[dist] = ltrRect;
}
if (!rtlRect.isEmpty()) {
const int dist =
item->preferredLayoutDirection() == AppletsLayout::RightToLeft ? 0 : QPointF(originalItemRect.center() - rtlRect.center()).manhattanLength();
distances[dist] = rtlRect;
}
if (!ttbRect.isEmpty()) {
const int dist =
item->preferredLayoutDirection() == AppletsLayout::TopToBottom ? 0 : QPointF(originalItemRect.center() - ttbRect.center()).manhattanLength();
distances[dist] = ttbRect;
}
if (!bttRect.isEmpty()) {
const int dist =
item->preferredLayoutDirection() == AppletsLayout::BottomToTop ? 0 : QPointF(originalItemRect.center() - bttRect.center()).manhattanLength();
distances[dist] = bttRect;
}
if (distances.isEmpty()) {
// Failure to layout, completely full
return originalItemRect;
} else {
return distances.first();
}
}
void AbstractLayoutManager::positionItem(ItemContainer *item)
{
// Give it a sane size if uninitialized: this may change size hints
if (item->width() <= 0 || item->height() <= 0) {
item->setSize(QSizeF(qMax(m_layout->minimumItemWidth(), m_layout->defaultItemWidth()), //
qMax(m_layout->minimumItemHeight(), m_layout->defaultItemHeight())));
}
QRectF candidate = candidateGeometry(item);
// Use setProperty to allow Behavior on to take effect
item->setProperty("x", candidate.topLeft().x());
item->setProperty("y", candidate.topLeft().y());
item->setSize(candidate.size());
}
void AbstractLayoutManager::positionItemAndAssign(ItemContainer *item)
{
releaseSpace(item);
positionItem(item);
assignSpace(item);
}
bool AbstractLayoutManager::assignSpace(ItemContainer *item)
{
if (assignSpaceImpl(item)) {
Q_EMIT layoutNeedsSaving();
return true;
} else {
return false;
}
}
void AbstractLayoutManager::releaseSpace(ItemContainer *item)
{
releaseSpaceImpl(item);
Q_EMIT layoutNeedsSaving();
}
void AbstractLayoutManager::layoutGeometryChanged(const QRectF &newGeometry, const QRectF &oldGeometry)
{
Q_UNUSED(newGeometry);
Q_UNUSED(oldGeometry);
// NOTE: Empty base implementation, don't put anything here
}
#include "moc_abstractlayoutmanager.cpp"