翻譯|使用教程|編輯:龔雪|2024-11-27 10:41:48.117|閱讀 125 次
概述:本文主要介紹展示如何使用為窗口實現流程布局,歡迎下載最新版組件體驗~
# 界面/圖表報表/文檔/IDE等千款熱門軟控件火熱銷售中 >>
相關鏈接:
Qt 是目前最先進、最完整的跨平臺C++開發工具。它不僅完全實現了一次編寫,所有平臺無差別運行,更提供了幾乎所有開發過程中需要用到的工具。如今,Qt已被運用于超過70個行業、數千家企業,支持數百萬設備及應用。
本文將展示如何為不同的窗口大小排列小部件。
流程布局實現了處理不同窗口大小的布局,小部件的位置取決于應用程序窗口的寬度。
Flowlayout類主要使用和,而Window類使用和。
在上文中(點擊這里回顧>>),我們主要介紹了FlowLayout類定義、示例運行等,本文將繼續介紹FlowLayout類實現,請繼續關注哦~
 
 
Qt技術交流群:166830288 歡迎一起進群討論
我們從構造函數開始:
FlowLayout::FlowLayout(QWidget *parent, int margin, int hSpacing, int vSpacing)
: QLayout(parent), m_hSpace(hSpacing), m_vSpace(vSpacing)
{
setContentsMargins(margin, margin, margin, margin);
}
FlowLayout::FlowLayout(int margin, int hSpacing, int vSpacing)
: m_hSpace(hSpacing), m_vSpace(vSpacing)
{
setContentsMargins(margin, margin, margin, margin);
}
在構造函數中,我們調用setContentsMargins()來設置左、上、右和下邊距。默認情況下,QLayout使用當前樣式提供的值(參見)。
FlowLayout::~FlowLayout()
{
QLayoutItem *item;
while ((item = takeAt(0)))
delete item;
}
在這個例子中,我們重新實現了addItem(),它是一個純虛函數。當使用addItem() 時,布局項的所有權被轉移到布局,因此它是布局的責任來刪除它們。
void FlowLayout::addItem(QLayoutItem *item)
{
itemList.append(item);
}
addItem()用于向布局中添加項。
int FlowLayout::horizontalSpacing() const
{
if (m_hSpace >= 0) {
return m_hSpace;
} else {
return smartSpacing(QStyle::PM_LayoutHorizontalSpacing);
}
}
int FlowLayout::verticalSpacing() const
{
if (m_vSpace >= 0) {
return m_vSpace;
} else {
return smartSpacing(QStyle::PM_LayoutVerticalSpacing);
}
}
我們實現了horizontalSpacing()和verticalSpacing() 來獲取布局中小部件之間的間距,如果該值小于或等于0,則使用此值。如果沒有,將調用smartSpacing()來計算間距。
int FlowLayout::count() const
{
return itemList.size();
}
QLayoutItem *FlowLayout::itemAt(int index) const
{
return itemList.value(index);
}
QLayoutItem *FlowLayout::takeAt(int index)
{
if (index >= 0 && index < itemList.size())
return itemList.takeAt(index);
return nullptr;
}
然后實現count()來返回布局中的項數,為了在項目列表中導航,我們使用 itemAt() 和 takeAt() 從列表中刪除和返回項目。如果一個項目被刪除,剩下的項目將重新編號,這三個函數都是來自的純虛函數。
Qt::Orientations FlowLayout::expandingDirections() const
{
return { };
}
expandingDirections()返回s,其中布局可以使用比sizeHint()更多的空間。
bool FlowLayout::hasHeightForWidth() const
{
return true;
}
int FlowLayout::heightForWidth(int width) const
{
int height = doLayout(QRect(0, 0, width, 0), true);
return height;
}
為了調整到高度依賴于寬度的小部件,我們實現了heightForWidth()。函數hasHeightForWidth()被用來測試這個依賴關系,并且heightForWidth()將寬度傳遞給doLayout(),后者反過來使用寬度作為布局矩形的參數,即項目布局的邊界,該矩形不包括布局margin()。
void FlowLayout::setGeometry(const QRect &rect)
{
QLayout::setGeometry(rect);
doLayout(rect, false);
}
QSize FlowLayout::sizeHint() const
{
return minimumSize();
}
QSize FlowLayout::minimumSize() const
{
QSize size;
for (const QLayoutItem *item : std::as_const(itemList))
size = size.expandedTo(item->minimumSize());
const QMargins margins = contentsMargins();
size += QSize(margins.left() + margins.right(), margins.top() + margins.bottom());
return size;
}
setGeometry()通常用于執行實際的布局,即計算布局項的幾何形狀。在這個例子中,它調用了doLayout()并傳遞了布局矩形。
sizeHint()返回布局的首選大小,minimumSize()返回布局的最小大小。
int FlowLayout::doLayout(const QRect &rect, bool testOnly) const
{
int left, top, right, bottom;
getContentsMargins(&left, &top, &right, &bottom);
QRect effectiveRect = rect.adjusted(+left, +top, -right, -bottom);
int x = effectiveRect.x();
int y = effectiveRect.y();
int lineHeight = 0;
如果horizontalSpacing() 或 verticalSpacing()不返回默認值,則doLayout()處理布局,它使用getContentsMargins()來計算布局項的可用面積。
for (QLayoutItem *item : std::as_const(itemList)) {
const QWidget *wid = item->widget();
int spaceX = horizontalSpacing();
if (spaceX == -1)
spaceX = wid->style()->layoutSpacing(
QSizePolicy::PushButton, QSizePolicy::PushButton, Qt::Horizontal);
int spaceY = verticalSpacing();
if (spaceY == -1)
spaceY = wid->style()->layoutSpacing(
QSizePolicy::PushButton, QSizePolicy::PushButton, Qt::Vertical);
然后,它根據當前樣式為布局中的每個小部件設置適當的間距。
int nextX = x + item->sizeHint().width() + spaceX;
if (nextX - spaceX > effectiveRect.right() && lineHeight > 0) {
x = effectiveRect.x();
y = 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() + bottom;
}
然后通過將項目寬度和行高添加到初始x和y坐標來計算布局中每個項目的位置,這反過來又讓我們知道下一項是否適合當前行,或者是否必須向下移動到下一行,我們還根據小部件的高度找到當前行的高度。
int FlowLayout::smartSpacing(QStyle::PixelMetric pm) const
{
QObject *parent = this->parent();
if (!parent) {
return -1;
} else if (parent->isWidgetType()) {
QWidget *pw = static_cast<QWidget *>(parent);
return pw->style()->pixelMetric(pm, nullptr, pw);
} else {
return static_cast<QLayout *>(parent)->spacing();
}
}
smartSpacing()被設計為獲取頂級布局或子布局的默認間距,當父組件是QWidget時,頂級布局的默認間距將通過查詢樣式來確定。當父布局為時,子布局的默認間距將通過查詢父布局的間距來確定。
本站文章除注明轉載外,均為本站原創或翻譯。歡迎任何形式的轉載,但請務必注明出處、不得修改原文相關鏈接,如果存在內容上的異議請郵件反饋至chenjj@ke049m.cn
文章轉載自:慧都網