Refactor QStatusBar to reduce memory allocation
Reading through the code contemplating what might have caused QTBUG-89141 brought up some opportunities for improvement. * updated coding style and variable names * use ranged for where possible and meaningful * replacing a QList of pointers to heap-allocated structs with a list of values Since the QList population code makes sure that we never have gaps (we only insert within the existing range), the test for null-entries is not needed, and was perhaps just precausion to avoid nullptr dereference. Task-number: QTBUG-89141 Change-Id: I4694d820427a221f1334d2428f50069751919aef Reviewed-by: Oliver Eftevaag <oliver.eftevaag@qt.io> Reviewed-by: Jan Arve Sæther <jan-arve.saether@qt.io>
This commit is contained in:
parent
9f13842fe6
commit
7166a82844
@ -69,15 +69,20 @@ class QStatusBarPrivate : public QWidgetPrivate
|
||||
public:
|
||||
QStatusBarPrivate() {}
|
||||
|
||||
struct SBItem {
|
||||
SBItem(QWidget* widget, int stretch, bool permanent)
|
||||
: s(stretch), w(widget), p(permanent) {}
|
||||
int s;
|
||||
QWidget * w;
|
||||
bool p;
|
||||
enum ItemCategory
|
||||
{
|
||||
Normal,
|
||||
Permanent
|
||||
};
|
||||
|
||||
QList<SBItem *> items;
|
||||
struct SBItem {
|
||||
QWidget *widget = nullptr;
|
||||
int stretch = 0;
|
||||
ItemCategory category = Normal;
|
||||
bool isPermanent() const { return category == Permanent; }
|
||||
};
|
||||
|
||||
QList<SBItem> items;
|
||||
QString tempItem;
|
||||
|
||||
QBoxLayout *box;
|
||||
@ -94,8 +99,8 @@ public:
|
||||
{
|
||||
int i = items.size() - 1;
|
||||
for (; i >= 0; --i) {
|
||||
SBItem *item = items.at(i);
|
||||
if (!(item && item->p))
|
||||
const SBItem &item = items.at(i);
|
||||
if (!item.isPermanent())
|
||||
break;
|
||||
}
|
||||
return i;
|
||||
@ -122,7 +127,7 @@ public:
|
||||
QRect QStatusBarPrivate::messageRect() const
|
||||
{
|
||||
Q_Q(const QStatusBar);
|
||||
bool rtl = q->layoutDirection() == Qt::RightToLeft;
|
||||
const bool rtl = q->layoutDirection() == Qt::RightToLeft;
|
||||
|
||||
int left = 6;
|
||||
int right = q->width() - 12;
|
||||
@ -136,17 +141,12 @@ QRect QStatusBarPrivate::messageRect() const
|
||||
}
|
||||
#endif
|
||||
|
||||
for (int i=0; i<items.size(); ++i) {
|
||||
QStatusBarPrivate::SBItem* item = items.at(i);
|
||||
if (!item)
|
||||
break;
|
||||
if (item->p && item->w->isVisible()) {
|
||||
if (item->p) {
|
||||
for (const auto &item : items) {
|
||||
if (item.isPermanent() && item.widget->isVisible()) {
|
||||
if (rtl)
|
||||
left = qMax(left, item->w->x() + item->w->width() + 2);
|
||||
left = qMax(left, item.widget->x() + item.widget->width() + 2);
|
||||
else
|
||||
right = qMin(right, item->w->x() - 2);
|
||||
}
|
||||
right = qMin(right, item.widget->x() - 2);
|
||||
break;
|
||||
}
|
||||
}
|
||||
@ -246,9 +246,6 @@ QStatusBar::QStatusBar(QWidget * parent)
|
||||
*/
|
||||
QStatusBar::~QStatusBar()
|
||||
{
|
||||
Q_D(QStatusBar);
|
||||
while (!d->items.isEmpty())
|
||||
delete d->items.takeFirst();
|
||||
}
|
||||
|
||||
|
||||
@ -298,7 +295,7 @@ int QStatusBar::insertWidget(int index, QWidget *widget, int stretch)
|
||||
return -1;
|
||||
|
||||
Q_D(QStatusBar);
|
||||
QStatusBarPrivate::SBItem* item = new QStatusBarPrivate::SBItem(widget, stretch, false);
|
||||
QStatusBarPrivate::SBItem item{widget, stretch, QStatusBarPrivate::Normal};
|
||||
|
||||
int idx = d->indexToLastNonPermanentWidget();
|
||||
if (Q_UNLIKELY(index < 0 || index > d->items.size() || (idx >= 0 && index > idx + 1))) {
|
||||
@ -363,7 +360,7 @@ int QStatusBar::insertPermanentWidget(int index, QWidget *widget, int stretch)
|
||||
return -1;
|
||||
|
||||
Q_D(QStatusBar);
|
||||
QStatusBarPrivate::SBItem* item = new QStatusBarPrivate::SBItem(widget, stretch, true);
|
||||
QStatusBarPrivate::SBItem item{widget, stretch, QStatusBarPrivate::Permanent};
|
||||
|
||||
int idx = d->indexToLastNonPermanentWidget();
|
||||
if (Q_UNLIKELY(index < 0 || index > d->items.size() || (idx >= 0 && index <= idx))) {
|
||||
@ -396,15 +393,11 @@ void QStatusBar::removeWidget(QWidget *widget)
|
||||
|
||||
Q_D(QStatusBar);
|
||||
bool found = false;
|
||||
QStatusBarPrivate::SBItem* item;
|
||||
for (int i = 0; i < d->items.size(); ++i) {
|
||||
item = d->items.at(i);
|
||||
if (!item)
|
||||
break;
|
||||
if (item->w == widget) {
|
||||
const auto &item = d->items.at(i);
|
||||
if (item.widget == widget) {
|
||||
d->items.removeAt(i);
|
||||
item->w->hide();
|
||||
delete item;
|
||||
item.widget->hide();
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
@ -495,25 +488,22 @@ void QStatusBar::reformat()
|
||||
|
||||
int maxH = fontMetrics().height();
|
||||
|
||||
int i;
|
||||
QStatusBarPrivate::SBItem* item;
|
||||
for (i=0,item=nullptr; i<d->items.size(); ++i) {
|
||||
item = d->items.at(i);
|
||||
if (!item || item->p)
|
||||
qsizetype i;
|
||||
for (i = 0; i < d->items.size(); ++i) {
|
||||
const auto &item = d->items.at(i);
|
||||
if (item.isPermanent())
|
||||
break;
|
||||
l->addWidget(item->w, item->s);
|
||||
int itemH = qMin(qSmartMinSize(item->w).height(), item->w->maximumHeight());
|
||||
l->addWidget(item.widget, item.stretch);
|
||||
int itemH = qMin(qSmartMinSize(item.widget).height(), item.widget->maximumHeight());
|
||||
maxH = qMax(maxH, itemH);
|
||||
}
|
||||
|
||||
l->addStretch(0);
|
||||
|
||||
for (item=nullptr; i<d->items.size(); ++i) {
|
||||
item = d->items.at(i);
|
||||
if (!item)
|
||||
break;
|
||||
l->addWidget(item->w, item->s);
|
||||
int itemH = qMin(qSmartMinSize(item->w).height(), item->w->maximumHeight());
|
||||
for (; i < d->items.size(); ++i) {
|
||||
const auto &item = d->items.at(i);
|
||||
l->addWidget(item.widget, item.stretch);
|
||||
int itemH = qMin(qSmartMinSize(item.widget).height(), item.widget->maximumHeight());
|
||||
maxH = qMax(maxH, itemH);
|
||||
}
|
||||
#if QT_CONFIG(sizegrip)
|
||||
@ -617,16 +607,14 @@ void QStatusBar::hideOrShow()
|
||||
Q_D(QStatusBar);
|
||||
bool haveMessage = !d->tempItem.isEmpty();
|
||||
|
||||
QStatusBarPrivate::SBItem* item = nullptr;
|
||||
for (int i=0; i<d->items.size(); ++i) {
|
||||
item = d->items.at(i);
|
||||
if (!item || item->p)
|
||||
for (const auto &item : qAsConst(d->items)) {
|
||||
if (item.isPermanent())
|
||||
break;
|
||||
if (haveMessage && item->w->isVisible()) {
|
||||
item->w->hide();
|
||||
item->w->setAttribute(Qt::WA_WState_ExplicitShowHide, false);
|
||||
} else if (!haveMessage && !item->w->testAttribute(Qt::WA_WState_ExplicitShowHide)) {
|
||||
item->w->show();
|
||||
if (haveMessage && item.widget->isVisible()) {
|
||||
item.widget->hide();
|
||||
item.widget->setAttribute(Qt::WA_WState_ExplicitShowHide, false);
|
||||
} else if (!haveMessage && !item.widget->testAttribute(Qt::WA_WState_ExplicitShowHide)) {
|
||||
item.widget->show();
|
||||
}
|
||||
}
|
||||
|
||||
@ -671,16 +659,15 @@ void QStatusBar::paintEvent(QPaintEvent *event)
|
||||
opt.initFrom(this);
|
||||
style()->drawPrimitive(QStyle::PE_PanelStatusBar, &opt, &p, this);
|
||||
|
||||
for (int i=0; i<d->items.size(); ++i) {
|
||||
QStatusBarPrivate::SBItem* item = d->items.at(i);
|
||||
if (item && item->w->isVisible() && (!haveMessage || item->p)) {
|
||||
QRect ir = item->w->geometry().adjusted(-2, -1, 2, 1);
|
||||
for (const auto &item : qAsConst(d->items)) {
|
||||
if (item.widget->isVisible() && (!haveMessage || item.isPermanent())) {
|
||||
QRect ir = item.widget->geometry().adjusted(-2, -1, 2, 1);
|
||||
if (event->rect().intersects(ir)) {
|
||||
QStyleOption opt(0);
|
||||
opt.rect = ir;
|
||||
opt.palette = palette();
|
||||
opt.state = QStyle::State_None;
|
||||
style()->drawPrimitive(QStyle::PE_FrameStatusBarItem, &opt, &p, item->w);
|
||||
style()->drawPrimitive(QStyle::PE_FrameStatusBarItem, &opt, &p, item.widget);
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -706,17 +693,13 @@ bool QStatusBar::event(QEvent *e)
|
||||
{
|
||||
Q_D(QStatusBar);
|
||||
|
||||
if (e->type() == QEvent::LayoutRequest
|
||||
) {
|
||||
switch (e->type()) {
|
||||
case QEvent::LayoutRequest: {
|
||||
// Calculate new strut height and call reformat() if it has changed
|
||||
int maxH = fontMetrics().height();
|
||||
|
||||
QStatusBarPrivate::SBItem* item = nullptr;
|
||||
for (int i=0; i<d->items.size(); ++i) {
|
||||
item = d->items.at(i);
|
||||
if (!item)
|
||||
break;
|
||||
int itemH = qMin(qSmartMinSize(item->w).height(), item->w->maximumHeight());
|
||||
for (const auto &item : qAsConst(d->items)) {
|
||||
const int itemH = qMin(qSmartMinSize(item.widget).height(), item.widget->maximumHeight());
|
||||
maxH = qMax(maxH, itemH);
|
||||
}
|
||||
|
||||
@ -729,18 +712,17 @@ bool QStatusBar::event(QEvent *e)
|
||||
reformat();
|
||||
else
|
||||
update();
|
||||
}
|
||||
if (e->type() == QEvent::ChildRemoved) {
|
||||
QStatusBarPrivate::SBItem* item = nullptr;
|
||||
for (int i=0; i<d->items.size(); ++i) {
|
||||
item = d->items.at(i);
|
||||
if (!item)
|
||||
break;
|
||||
if (item->w == ((QChildEvent*)e)->child()) {
|
||||
}
|
||||
case QEvent::ChildRemoved:
|
||||
for (int i = 0; i < d->items.size(); ++i) {
|
||||
const auto &item = d->items.at(i);
|
||||
if (item.widget == static_cast<QChildEvent *>(e)->child())
|
||||
d->items.removeAt(i);
|
||||
delete item;
|
||||
}
|
||||
}
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return QWidget::event(e);
|
||||
|
Loading…
Reference in New Issue
Block a user