FemaLocalSoftware/flowlayout.cpp

140 lines
3.2 KiB
C++
Raw Permalink Normal View History

2024-12-10 08:02:54 +00:00
#include "flowlayout.h"
#include <QWidget>
2024-12-16 08:38:04 +00:00
#include <utility>
2024-12-10 08:02:54 +00:00
FlowLayout::FlowLayout(QWidget *parent, int margin, int hSpacing, int vSpacing)
2025-01-07 22:13:48 +00:00
: QLayout(parent)
, m_hSpacing(hSpacing)
, m_vSpacing(vSpacing)
{
2024-12-10 08:02:54 +00:00
setContentsMargins(margin, margin, margin, margin);
}
2025-01-07 22:13:48 +00:00
FlowLayout::~FlowLayout()
{
2024-12-10 08:02:54 +00:00
QLayoutItem *item;
while ((item = takeAt(0))) {
delete item;
}
}
2025-01-07 22:13:48 +00:00
void FlowLayout::clear()
{
2024-12-16 08:38:04 +00:00
while (QLayoutItem *item = takeAt(0)) {
if (item->widget()) {
item->widget()->hide();
}
delete item;
}
}
2025-01-07 22:13:48 +00:00
void FlowLayout::addItem(QLayoutItem *item)
{
2024-12-10 08:02:54 +00:00
m_items.append(item);
}
2025-01-07 22:13:48 +00:00
int FlowLayout::count() const
{
2024-12-10 08:02:54 +00:00
return m_items.size();
}
2025-01-07 22:13:48 +00:00
QLayoutItem *FlowLayout::itemAt(int index) const
{
2024-12-10 08:02:54 +00:00
return m_items.value(index);
}
2025-01-07 22:13:48 +00:00
QLayoutItem *FlowLayout::takeAt(int index)
{
2024-12-10 08:02:54 +00:00
return (index >= 0 && index < m_items.size()) ? m_items.takeAt(index) : nullptr;
}
2025-01-07 22:13:48 +00:00
Qt::Orientations FlowLayout::expandingDirections() const
{
2024-12-10 08:02:54 +00:00
return Qt::Orientation(0);
}
2025-01-07 22:13:48 +00:00
bool FlowLayout::hasHeightForWidth() const
{
2024-12-10 08:02:54 +00:00
return true;
}
2025-01-07 22:13:48 +00:00
int FlowLayout::heightForWidth(int width) const
{
2024-12-10 08:02:54 +00:00
return doLayout(QRect(0, 0, width, 0), true);
}
2025-01-07 22:13:48 +00:00
void FlowLayout::setGeometry(const QRect &rect)
{
2024-12-10 08:02:54 +00:00
QLayout::setGeometry(rect);
doLayout(rect, false);
}
2025-01-07 22:13:48 +00:00
QSize FlowLayout::sizeHint() const
{
2024-12-10 08:02:54 +00:00
return minimumSize();
}
2025-01-07 22:13:48 +00:00
QSize FlowLayout::minimumSize() const
{
2024-12-10 08:02:54 +00:00
QSize size;
2024-12-16 08:38:04 +00:00
for (const QLayoutItem *item : std::as_const(m_items)) {
2024-12-10 08:02:54 +00:00
size = size.expandedTo(item->minimumSize());
}
const QMargins margins = contentsMargins();
size += QSize(margins.left() + margins.right(), margins.top() + margins.bottom());
return size;
}
2025-01-07 22:13:48 +00:00
int FlowLayout::doLayout(const QRect &rect, bool testOnly) const
{
2024-12-10 08:02:54 +00:00
int x = rect.x();
int y = rect.y();
int lineHeight = 0;
2024-12-16 08:38:04 +00:00
for (QLayoutItem *item : std::as_const(m_items)) {
2024-12-10 08:02:54 +00:00
int spaceX = horizontalSpacing();
int spaceY = verticalSpacing();
int nextX = x + item->sizeHint().width() + spaceX;
if (nextX - spaceX > rect.right() && lineHeight > 0) {
x = rect.x();
y += lineHeight + spaceY;
nextX = x + item->sizeHint().width() + spaceX;
lineHeight = 0;
}
if (!testOnly) {
item->setGeometry(QRect(QPoint(x, y), item->sizeHint()));
}
x = nextX;
lineHeight = qMax(lineHeight, item->sizeHint().height());
}
return y + lineHeight - rect.y();
}
2025-01-07 22:13:48 +00:00
int FlowLayout::horizontalSpacing() const
{
2024-12-10 08:02:54 +00:00
return (m_hSpacing >= 0) ? m_hSpacing : smartSpacing(QStyle::PM_LayoutHorizontalSpacing);
}
2025-01-07 22:13:48 +00:00
int FlowLayout::verticalSpacing() const
{
2024-12-10 08:02:54 +00:00
return (m_vSpacing >= 0) ? m_vSpacing : smartSpacing(QStyle::PM_LayoutVerticalSpacing);
}
2025-01-07 22:13:48 +00:00
int FlowLayout::smartSpacing(QStyle::PixelMetric pm) const
{
2024-12-10 08:02:54 +00:00
QObject *parent = this->parent();
if (!parent) {
return -1;
} else if (parent->isWidgetType()) {
2025-01-07 22:13:48 +00:00
return static_cast<QWidget *>(parent)->style()->pixelMetric(pm,
nullptr,
static_cast<QWidget *>(parent));
2024-12-10 08:02:54 +00:00
} else {
return static_cast<QLayout *>(parent)->spacing();
}
}