summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorHannah von Reth <hannah.vonreth@kdab.com>2016-03-29 09:50:50 +0200
committerHannah von Reth <hannah.vonreth@kdab.com>2016-04-05 20:32:26 +0000
commit1726352207273465c12852d027f97b99884b78e4 (patch)
treed058c5268ddb085345b6eddb190a54f6b981e8f2
parent65aff4dc422f53faa44e961850cb4f1e4501a8c7 (diff)
Make sure QStyleOption is correctly initialized.
Ensures that QStyleOption is correctly initialized. This prevents possible styling issues due to QStyleOption's reporting version 0, see qstyleoption_cast. This enables users to handle more cases in their QProxyStyle. For now the test is only used for QCommonStyle. Change-Id: I768db00b12b46890343fffe44e4f562762e9cf80 Reviewed-by: Marc Mutz <marc.mutz@kdab.com>
-rw-r--r--src/widgets/styles/qcommonstyle.cpp10
-rw-r--r--tests/auto/widgets/styles/qstyle/tst_qstyle.cpp107
2 files changed, 110 insertions, 7 deletions
diff --git a/src/widgets/styles/qcommonstyle.cpp b/src/widgets/styles/qcommonstyle.cpp
index 4b6e3ac5c4..7b0d8eacd5 100644
--- a/src/widgets/styles/qcommonstyle.cpp
+++ b/src/widgets/styles/qcommonstyle.cpp
@@ -784,10 +784,8 @@ static void drawArrow(const QStyle *style, const QStyleOptionToolButton *toolbut
default:
return;
}
- QStyleOption arrowOpt;
+ QStyleOption arrowOpt = *toolbutton;
arrowOpt.rect = rect;
- arrowOpt.palette = toolbutton->palette;
- arrowOpt.state = toolbutton->state;
style->drawPrimitive(pe, &arrowOpt, painter, widget);
}
#endif // QT_NO_TOOLBUTTON
@@ -3309,8 +3307,7 @@ void QCommonStyle::drawComplexControl(ComplexControl cc, const QStyleOptionCompl
mflags |= State_Sunken;
}
- QStyleOption tool(0);
- tool.palette = toolbutton->palette;
+ QStyleOption tool = *toolbutton;
if (toolbutton->subControls & SC_ToolButton) {
if (bflags & (State_Sunken | State_On | State_Raised)) {
tool.rect = button;
@@ -3379,8 +3376,7 @@ void QCommonStyle::drawComplexControl(ComplexControl cc, const QStyleOptionCompl
bool down = false;
QPixmap pm;
- QStyleOption tool(0);
- tool.palette = tb->palette;
+ QStyleOption tool = *tb;
if (tb->subControls & SC_TitleBarCloseButton && tb->titleBarFlags & Qt::WindowSystemMenuHint) {
ir = proxy()->subControlRect(CC_TitleBar, tb, SC_TitleBarCloseButton, widget);
down = tb->activeSubControls & SC_TitleBarCloseButton && (opt->state & State_Sunken);
diff --git a/tests/auto/widgets/styles/qstyle/tst_qstyle.cpp b/tests/auto/widgets/styles/qstyle/tst_qstyle.cpp
index dafa95fb25..46b6fcca82 100644
--- a/tests/auto/widgets/styles/qstyle/tst_qstyle.cpp
+++ b/tests/auto/widgets/styles/qstyle/tst_qstyle.cpp
@@ -133,6 +133,7 @@ private slots:
void testFrameOnlyAroundContents();
void testProxyCalled();
+ void testStyleOptionInit();
private:
void lineUpLayoutTest(QStyle *);
QWidget *testWidget;
@@ -856,5 +857,111 @@ void tst_QStyle::testProxyCalled()
}
}
+
+class TestStyleOptionInitProxy: public QProxyStyle
+{
+ Q_OBJECT
+public:
+ mutable bool invalidOptionsDetected;
+ explicit TestStyleOptionInitProxy(QStyle *style = Q_NULLPTR)
+ : QProxyStyle(style),
+ invalidOptionsDetected(false)
+ {}
+
+ void drawPrimitive(PrimitiveElement pe, const QStyleOption *opt, QPainter *p, const QWidget *w) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::PrimitiveElement>(pe, opt);
+ return QProxyStyle::drawPrimitive(pe, opt, p, w);
+ }
+
+ void drawControl(ControlElement element, const QStyleOption *opt, QPainter *p, const QWidget *w) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::ControlElement>(element, opt);
+ return QProxyStyle::drawControl(element, opt, p, w);
+ }
+
+ QRect subElementRect(SubElement subElement, const QStyleOption *option, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::SubElement>(subElement, option);
+ return QProxyStyle::subElementRect(subElement, option, widget);
+ }
+
+ void drawComplexControl(ComplexControl cc, const QStyleOptionComplex *opt, QPainter *p, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::ComplexControl>(cc, opt);
+ return QProxyStyle::drawComplexControl(cc, opt, p, widget);
+ }
+
+ QRect subControlRect(ComplexControl cc, const QStyleOptionComplex *opt, SubControl sc, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::ComplexControl>(cc, opt);
+ return QProxyStyle::subControlRect(cc, opt, sc, widget);
+ }
+
+ int pixelMetric(PixelMetric metric, const QStyleOption *option, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::PixelMetric>(metric, option);
+ return QProxyStyle::pixelMetric(metric, option, widget);
+ }
+
+ QSize sizeFromContents(ContentsType ct, const QStyleOption *opt, const QSize &contentsSize, const QWidget *w) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::ContentsType>(ct, opt);
+ return QProxyStyle::sizeFromContents(ct, opt, contentsSize, w);
+ }
+
+ int styleHint(StyleHint stylehint, const QStyleOption *opt, const QWidget *widget, QStyleHintReturn *returnData) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::StyleHint>(stylehint, opt);
+ return QProxyStyle::styleHint(stylehint, opt, widget, returnData);
+ }
+
+ QPixmap standardPixmap(StandardPixmap standardPixmap, const QStyleOption *opt, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::StandardPixmap>(standardPixmap, opt);
+ return QProxyStyle::standardPixmap(standardPixmap, opt, widget);
+ }
+
+ QIcon standardIcon(StandardPixmap standardIcon, const QStyleOption *option, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyleEnum<QStyle::StandardPixmap>(standardIcon, option);
+ return QProxyStyle::standardIcon(standardIcon, option, widget);
+ }
+
+ QPixmap generatedIconPixmap(QIcon::Mode iconMode, const QPixmap &pixmap, const QStyleOption *opt) const Q_DECL_OVERRIDE {
+ checkStyle(QString::asprintf("QIcon::Mode(%i)", iconMode).toLatin1(), opt);
+ return QProxyStyle::generatedIconPixmap(iconMode, pixmap, opt);
+ }
+
+ int layoutSpacing(QSizePolicy::ControlType control1, QSizePolicy::ControlType control2, Qt::Orientation orientation, const QStyleOption *option, const QWidget *widget) const Q_DECL_OVERRIDE {
+ checkStyle(QString::asprintf("QSizePolicy::ControlType(%i), QSizePolicy::ControlType(%i)", control1, control2).toLatin1(), option);
+ return QProxyStyle::layoutSpacing(control1, control2, orientation, option, widget);
+ }
+
+private:
+ void checkStyle(const QByteArray &info, const QStyleOption *opt) const {
+ if (opt && (opt->version == 0 || opt->styleObject == Q_NULLPTR) ) {
+ invalidOptionsDetected = true;
+ qWarning() << baseStyle()->metaObject()->className()
+ << "Invalid QStyleOption found for"
+ << info;
+ qWarning() << "Version:" << opt->version << "StyleObject:" << opt->styleObject;
+ }
+ }
+
+ template<typename MEnum>
+ void checkStyleEnum(MEnum element, const QStyleOption *opt) const {
+ static QMetaEnum _enum = QMetaEnum::fromType<MEnum>();
+ checkStyle(_enum.valueToKey(element), opt);
+ }
+};
+
+void tst_QStyle::testStyleOptionInit()
+{
+ QStringList keys = QStyleFactory::keys();
+ QVector<QStyle*> styles;
+ styles.reserve(keys.size() + 1);
+
+ styles << new QCommonStyle();
+
+ Q_FOREACH (QStyle *style, styles) {
+ TestStyleOptionInitProxy testStyle;
+ testStyle.setBaseStyle(style);
+ testAllFunctions(style);
+ QVERIFY(!testStyle.invalidOptionsDetected);
+ delete style;
+ }
+}
+
QTEST_MAIN(tst_QStyle)
#include "tst_qstyle.moc"