如何使用 Qt 在代码中访问升级小部件的 UI 元素?

Ric*_*ick 5 c++ qt qt-designer qt5

所以我创建了一个类DemoPromotedWidget,它继承自QWidget。

然后,我在设计器中创建了一个小部件,向所述小部件添加了一个名为 DemoLabel 的标签,并将该小部件提升为 DemoPromotedWidget 类...然后我将其移动到我的工具箱中,它现在位于“Scratchpad”下。

通常,使用 Qt 的设计器,我可以访问 UI 文件中相应类文件中的任何元素。但是,在 DemoPromotedWidget 中我不能。

无论如何,我是否可以访问 DemoPromotedWidget 内的 DemoLabel,以便我可以调用类似...

ui->DemoPromotedWidget->setText("Updated Text");
Run Code Online (Sandbox Code Playgroud)

这是我的 UI 设计器的图片,其中已添加 DemoPromotedWidget。 https://i.stack.imgur.com/l7cLX.png

DemoPromotedWidget.h

#ifndef DEMOPROMOTEDWIDGET_H
#define DEMOPROMOTEDWIDGET_H

#include <QWidget>

class DemoPromotedWidget : public QWidget
{
    Q_OBJECT
public:
    explicit DemoPromotedWidget(QWidget *parent = nullptr);

signals:

public slots:
};

#endif // DEMOPROMOTEDWIDGET_H
Run Code Online (Sandbox Code Playgroud)

DemoPromotedWidget.cpp

#include "demopromotedwidget.h"

DemoPromotedWidget::DemoPromotedWidget(QWidget *parent) : QWidget(parent)
{
    // How do I access DemoPromotedWidget->DemoLabel->setText()?????????????????????
}
Run Code Online (Sandbox Code Playgroud)

这是我的 mainwindow.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>711</width>
    <height>418</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="QWidget" name="centralWidget">
   <widget class="DemoPromotedWidget" name="DemoPromotedWidget" native="true">
    <property name="geometry">
     <rect>
      <x>160</x>
      <y>60</y>
      <width>381</width>
      <height>191</height>
     </rect>
    </property>
    <property name="styleSheet">
     <string notr="true">background-color: black;
text-align: center;</string>
    </property>
    <widget class="QLabel" name="DemoLabel">
     <property name="geometry">
      <rect>
       <x>0</x>
       <y>5</y>
       <width>381</width>
       <height>181</height>
      </rect>
     </property>
     <property name="styleSheet">
      <string notr="true">color: white;
text-align: center;</string>
     </property>
     <property name="text">
      <string>DemoLabel</string>
     </property>
     <property name="alignment">
      <set>Qt::AlignCenter</set>
     </property>
    </widget>
   </widget>
  </widget>
  <widget class="QMenuBar" name="menuBar">
   <property name="geometry">
    <rect>
     <x>0</x>
     <y>0</y>
     <width>711</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>DemoPromotedWidget</class>
   <extends>QWidget</extends>
   <header>demopromotedwidget.h</header>
   <container>1</container>
  </customwidget>
 </customwidgets>
 <resources/>
 <connections/>
</ui>
Run Code Online (Sandbox Code Playgroud)

eyl*_*esc 4

您不应该将类的名称与小部件的名称一起使用,因此要做的第一件事就是更改其名称:

在此输入图像描述

第一个选项:

另一方面, theQLabel内部的 thisDemoPromotedWidget并不意味着DemoPromotedWidget可以访问QLabel构造函数中的 the 和 less ,因为在该方法中它尚未添加到主窗口中,一个可能的解决方案是创建一个属性,您可以在其中为其分配QLabel

demopromotedwidget.h

#ifndef DEMOPROMOTEDWIDGET_H
#define DEMOPROMOTEDWIDGET_H

#include <QLabel>
#include <QWidget>

class DemoPromotedWidget : public QWidget
{
    Q_OBJECT
public:
    explicit DemoPromotedWidget(QWidget *parent = nullptr);
    QLabel *getLabel() const;
    void setLabel(QLabel *value);
private:
    QLabel *label;
};

#endif // DEMOPROMOTEDWIDGET_H
Run Code Online (Sandbox Code Playgroud)

演示推广widget.cpp

#include "demopromotedwidget.h"

DemoPromotedWidget::DemoPromotedWidget(QWidget *parent) : QWidget(parent)
{

}

QLabel *DemoPromotedWidget::getLabel() const
{
    return label;
}

void DemoPromotedWidget::setLabel(QLabel *value)
{
    label = value;
    if(label)
        label->setText("some text");
}
Run Code Online (Sandbox Code Playgroud)

主窗口.cpp

...
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    ui->widget->setLabel(ui->DemoLabel);
}
...
Run Code Online (Sandbox Code Playgroud)

第二个选项:

另一种更简单的方法,但只有使用 Qt Designer 添加小部件(QLabel)才能完成,在本例中使用方法showEvent()findChild()

demopromotedwidget.h

#ifndef DEMOPROMOTEDWIDGET_H
#define DEMOPROMOTEDWIDGET_H

#include <QLabel>
#include <QWidget>

class DemoPromotedWidget : public QWidget
{
    Q_OBJECT
public:
    explicit DemoPromotedWidget(QWidget *parent = nullptr);
private:
    QLabel *label;
protected:
    void showEvent(QShowEvent *event);
};

#endif // DEMOPROMOTEDWIDGET_H
Run Code Online (Sandbox Code Playgroud)

演示推广widget.cpp

#include "demopromotedwidget.h"

DemoPromotedWidget::DemoPromotedWidget(QWidget *parent) : QWidget(parent)
{
}

void DemoPromotedWidget::showEvent(QShowEvent *event)
{
    label = findChild<QLabel *>("DemoLabel");
    if(label)
        label->setText("some text");
    QWidget::showEvent(event);
}
Run Code Online (Sandbox Code Playgroud)

更新:

一种新方法是创建一个 QWidget 表单,其中包含.ui,.h.cpp

在此输入图像描述

在此输入图像描述

在此输入图像描述

正如我所说,创建一个新类,与您的初始代码不同,它具有 .ui,然后在 Qt Designer 的帮助下打开它并添加标签

在此输入图像描述

但这也是一个可以像在第一种方法中所做的那样进行推广的课程。

在此输入图像描述

完整的示例可以在以下链接中找到