Skip to content

Instantly share code, notes, and snippets.

@oldmud0
Forked from pjwhams/qtdumper.cpp
Last active December 21, 2020 04:21
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save oldmud0/ea42631547a168b4f60a355c89f100da to your computer and use it in GitHub Desktop.
Save oldmud0/ea42631547a168b4f60a355c89f100da to your computer and use it in GitHub Desktop.
Dump size and position of Qt QWidgets and QLayouts
#include "widgetdumper.h"
#include <sstream>
#include <string>
#include <iostream>
#include <stdio.h>
#include <QtWidgets/QLayout>
#include <QtWidgets/QWidget>
#if _MSC_VER
#define snprintf _snprintf
#endif
std::string WidgetDumper::toString(const QSizePolicy::Policy &policy) {
switch (policy) {
case QSizePolicy::Fixed:
return "Fixed";
case QSizePolicy::Minimum:
return "Minimum";
case QSizePolicy::Maximum:
return "Maximum";
case QSizePolicy::Preferred:
return "Preferred";
case QSizePolicy::MinimumExpanding:
return "MinimumExpanding";
case QSizePolicy::Expanding:
return "Expanding";
case QSizePolicy::Ignored:
return "Ignored";
}
return "unknown";
}
std::string WidgetDumper::toString(const QSizePolicy &policy) {
return "(" + toString(policy.horizontalPolicy()) + ", " + toString(policy.verticalPolicy()) + ")";
}
std::string WidgetDumper::toString(QLayout::SizeConstraint constraint) {
switch (constraint) {
case QLayout::SetDefaultConstraint:
return "SetDefaultConstraint";
case QLayout::SetNoConstraint:
return "SetNoConstraint";
case QLayout::SetMinimumSize:
return "SetMinimumSize";
case QLayout::SetFixedSize:
return "SetFixedSize";
case QLayout::SetMaximumSize:
return "SetMaximumSize";
case QLayout::SetMinAndMaxSize:
return "SetMinAndMaxSize";
}
return "unknown";
}
std::string WidgetDumper::getWidgetInfo(const QWidget &w) {
const QRect &geom = w.geometry();
QSize hint = w.sizeHint();
char buf[1024];
snprintf(buf, 1023, "%s %p ('%s'), pos (%d, %d), size (%d x %d), hint (%d x %d) pol: %s %s\n",
w.metaObject()->className(), (void *) &w, w.objectName().toStdString().c_str(),
geom.x(), geom.y(), geom.width(), geom.height(),
hint.width(), hint.height(),
toString(w.sizePolicy()).c_str(),
(w.isVisible() ? "" : "**HIDDEN**")
);
return buf;
}
std::string WidgetDumper::getLayoutItemInfo(QLayoutItem *item) {
if (dynamic_cast<QWidgetItem *>(item)) {
auto *wi = dynamic_cast<QWidgetItem *>(item);
if (wi->widget()) {
return getWidgetInfo(*wi->widget());
}
}
else if (dynamic_cast<QSpacerItem *>(item)) {
auto *si = dynamic_cast<QSpacerItem *>(item);
QSize hint = si->sizeHint();
std::string constraint = "<no layout>";
if (si->layout()) {
constraint = toString(si->layout()->sizeConstraint());
}
std::stringstream ss;
ss << "SpacerItem hint (" << hint.width() << " x " << hint.height() << ") policy: "
<< toString(si->sizePolicy()) << " constraint: " << constraint << "\n";
return ss.str();
}
return "";
}
//------------------------------------------------------------------------
void WidgetDumper::dumpWidgetAndChildren(std::ostream &os, const QWidget *w, int level) {
std::string padding("");
for (int i = 0; i <= level; i++)
padding += " ";
QLayout *layout = w->layout();
QList<QWidget *> dumpedChildren;
if (layout && !layout->isEmpty()) {
os << padding << "Layout ";
QMargins margins = layout->contentsMargins();
os << " margin: (" << margins.left() << "," << margins.top()
<< "," << margins.right() << "," << margins.bottom() << "), constraint: "
<< toString(layout->sizeConstraint());
if (dynamic_cast<QBoxLayout *>(layout)) {
auto *boxLayout = dynamic_cast<QBoxLayout *>(layout);
os << " spacing: " << boxLayout->spacing();
}
os << ":\n";
int numItems = layout->count();
for (int i = 0; i < numItems; i++) {
QLayoutItem *layoutItem = layout->itemAt(i);
std::string itemInfo = getLayoutItemInfo(layoutItem);
os << padding << " " << itemInfo;
auto *wi = dynamic_cast<QWidgetItem *>(layoutItem);
if (wi && wi->widget()) {
dumpWidgetAndChildren(os, wi->widget(), level + 1);
dumpedChildren.push_back(wi->widget());
}
}
}
// now output any child widgets that weren't dumped as part of the layout
QList<QWidget *> widgets = w->findChildren<QWidget *>(QString(), Qt::FindDirectChildrenOnly);
QList<QWidget *> undumpedChildren;
foreach (QWidget *child, widgets) {
if (dumpedChildren.indexOf(child) == -1) {
undumpedChildren.push_back(child);
}
}
if (!undumpedChildren.empty()) {
os << padding << " non-layout children:\n";
foreach (QWidget *child, undumpedChildren) {
dumpWidgetAndChildren(os, child, level + 1);
}
}
}
//------------------------------------------------------------------------
void WidgetDumper::dumpWidgetHierarchy(const QWidget *w) {
std::ostringstream oss;
oss << getWidgetInfo(*w);
dumpWidgetAndChildren(oss, w, 0);
std::cout << oss.str();
}
#ifndef WIDGETDUMPER_H
#define WIDGETDUMPER_H
// https://gist.github.com/pjwhams/6ebc040db3ab55615eafd831e184e39c
#include <string>
#include <QLayoutItem>
namespace WidgetDumper {
std::string toString(const QSizePolicy::Policy& policy);
std::string toString(const QSizePolicy& policy);
std::string toString(QLayout::SizeConstraint constraint);
std::__cxx11::basic_string<char> getLayoutItemInfo(QLayoutItem *item);
std::string getWidgetInfo(const QWidget &w);
void dumpWidgetAndChildren(std::ostream &os, const QWidget *w, int level);
void dumpWidgetHierarchy(const QWidget *w);
};
#endif // WIDGETDUMPER_H
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment