C++ 命名空间中派生类中的 Qt 样式表(选择器)

Posted

技术标签:

【中文标题】C++ 命名空间中派生类中的 Qt 样式表(选择器)【英文标题】:Qt stylesheet in derived class in C++ namespace (selector) 【发布时间】:2014-11-30 03:18:37 【问题描述】:

我想将我的全局qss 样式表与派生类一起使用。我知道我必须覆盖 paintEvent(style sheet reference 或 here)。

void CustomWidget::paintEvent(QPaintEvent *) 
     QStyleOption opt;
     opt.init(this); // tried initFrom too, same result=>not working
     QPainter p(this);
     style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
 

但是,它似乎不起作用。使用CDerived:QWidget 和我面临的以下样式表行:

CDerived  background-color: black;  // no effect
QWidget   background-color: black;  // works

CDerived 实现 paintEvent 如上所述。我还有什么需要做的吗?

-- 编辑/解决方案--

感谢 JK 的提示,我已经弄清楚了。我上面的例子实际上并没有正确反映我的情况。 我真正的类驻留在 C++ 命名空间中(我的错误我错过了)。 所以我必须在 qss 中写 MyNamespace--CDerived。见“Widgets inside C++ namespaces”

在这里尝试了JK的简单示例后,我突然意识到自己的错误!

正确的一个:

MyNamespace--CDerived  background-color: black;  // works, use -- for ::

备注:相关问题 (a,b),但没有回答这个特定问题。我的派生类位于 C++ 命名空间中。

【问题讨论】:

我不知道是不是我,但我在这里找不到opt.init():qt-project.org/doc/qt-5/qstyleoption.html 另请参阅此处使用 Qt 样式表的示例:qt-project.org/doc/qt-5/stylesheet-examples.html 【参考方案1】:

这很奇怪……对我来说很好用:

untitled.pro:

#-------------------------------------------------
#
# Project created by QtCreator 2014-10-07T11:34:54
#
#-------------------------------------------------

QT       += core gui

greaterThan(QT_MAJOR_VERSION, 4): QT += widgets

TARGET = untitled
TEMPLATE = app


SOURCES += main.cpp\
        mainwindow.cpp \
    mywidget.cpp

HEADERS  += mainwindow.h \
    mywidget.h

FORMS    += mainwindow.ui

主窗口.h:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>

namespace Ui 
class MainWindow;


class MainWindow : public QMainWindow

    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

private:
    Ui::MainWindow *ui;
;

#endif // MAINWINDOW_H

主窗口.cpp:

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)

    ui->setupUi(this);


MainWindow::~MainWindow()

    delete ui;

主窗口.ui:

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>400</width>
    <height>300</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <property name="styleSheet">
   <string notr="true"/>
  </property>
  <widget class="QWidget" name="centralWidget">
   <widget class="MyWidget" name="widget" native="true">
    <property name="geometry">
     <rect>
      <x>70</x>
      <y>30</y>
      <width>201</width>
      <height>121</height>
     </rect>
    </property>
    <property name="styleSheet">
     <string notr="true"/>
    </property>
    <widget class="QPushButton" name="pushButton">
     <property name="geometry">
      <rect>
       <x>30</x>
       <y>20</y>
       <width>75</width>
       <height>23</height>
      </rect>
     </property>
     <property name="text">
      <string>PushButton</string>
     </property>
    </widget>
   </widget>
  </widget>
  <widget class="QMenuBar" name="menuBar">
   <property name="geometry">
    <rect>
     <x>0</x>
     <y>0</y>
     <width>400</width>
     <height>21</height>
    </rect>
   </property>
  </widget>
  <widget class="QToolBar" name="mainToolBar">
   <attribute name="toolBarArea">
    <enum>TopToolBarArea</enum>
   </attribute>
   <attribute name="toolBarBreak">
    <bool>false</bool>
   </attribute>
  </widget>
  <widget class="QStatusBar" name="statusBar"/>
 </widget>
 <layoutdefault spacing="6" margin="11"/>
 <customwidgets>
  <customwidget>
   <class>MyWidget</class>
   <extends>QWidget</extends>
   <header>mywidget.h</header>
   <container>1</container>
  </customwidget>
 </customwidgets>
 <resources/>
 <connections/>
</ui>

mywidget.h:

#ifndef MYWIDGET_H
#define MYWIDGET_H

#include <QWidget>

class MyWidget : public QWidget

    Q_OBJECT
public:
    explicit MyWidget(QWidget *parent = 0);

protected:
    void paintEvent(QPaintEvent *e);

;

#endif // MYWIDGET_H

mywidget.cpp:

#include "mywidget.h"

#include <QStyleOption>
#include <QPainter>

MyWidget::MyWidget(QWidget *parent) :
    QWidget(parent)



void MyWidget::paintEvent(QPaintEvent *e)

    Q_UNUSED(e)

    QStyleOption opt;
    opt.init(this); // tried initFrom too, same result=>not working
    QPainter p(this);
    style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);

main.cpp:

#include "mainwindow.h"
#include <QApplication>

int main(int argc, char *argv[])

    QApplication a(argc, argv);

    a.setStyleSheet("MyWidget  background-color: red; ");

    MainWindow w;
    w.show();

    return a.exec();

【讨论】:

是的,很奇怪,我正在 Win 7、Qt 5.3 上进行测试。但是,你能做一个简短的交叉检查吗?如果您“注释掉” MyWidget::paintEvent() 函数,它会删除红色吗?今天晚些时候我也会测试这样一个简单的例子,好主意。 @HorstWalter 是的,我试过了,如果你注释掉paintEvent,红色会被移除 感谢您的提示,我已经弄清楚了。这是一个命名空间问题。我错过了 a) 我的类在命名空间中 b) "::" 必须替换为 "--"。

以上是关于C++ 命名空间中派生类中的 Qt 样式表(选择器)的主要内容,如果未能解决你的问题,请参考以下文章

C++中派生类的构造函数怎么显式调用基类构造函数?

C ++中派生类的相等性测试[重复]

C ++继承:具有基类类型的虚函数中派生类类型的参数

C#中派生类调用基类构造函数用法分析

虚拟继承中派生类的大小

c++解决二义性的方法