QWidget的isHidden和isVisible

问题的出现

最近在写代码的时候,出现了一个问题,我新建了两个类,分别是Chapter2ConsecutiveCurtain

// ConsecutiveCurtain.cpp
ConsecutiveCurtain::ConsecutiveCurtain(QWidget *parent)
:QWidget(parent)
{
number = 1;
this->setFixedSize(1920, 1080);
} void ConsecutiveCurtain::start()
{
if (m_curtainList.isEmpty()) {
qDebug() << "显示列表为空";
return;
} m_curtain = new Curtain(m_curtainList.at(0), this);
connect(m_curtain, &Curtain::signal_pageEnd,
this, &ConsecutiveCurtain::slot_nextPage, Qt::UniqueConnection);
m_curtain->start();
}
// chapter2.cpp
Chapter2::Chapter2(QWidget *parent)
:QWidget(parent)
{
m_curtains = new ConsecutiveCurtain(this);
this->start();
} void Chapter2::slot_Curtains1()
{
m_phone->close(); QStringList curtains;
curtains.push_back("chapter2_4.png");
curtains.push_back("chapter2_5.png");
curtains.push_back("chapter2_6.png");
curtains.push_back("chapter2_7.png"); m_curtains->setCurtainList(curtains); connect(m_curtains, &ConsecutiveCurtain::signal_consecutiveCurtainEnd,
this, &Chapter2::slot_Phone2); m_curtains->start();
m_curtains->show();
}

,这几个的结构如图所示:

但是在我新建ConsecutiveCurtain这个类的实例,并在chapter2.cppshow()时,什么都没有显示。

我左思右想,发现了两种解决方法:

第一种方法是:

ConsecutiveCurtain::ConsecutiveCurtain(QWidget *parent)
:QWidget(parent)
{
number = 1;
this->setFixedSize(1920, 1080);
this->hide();
}

ConsecutiveCurtain类里调用this->hide(),之后就可以正常显示了。

第二种方法是:

void ConsecutiveCurtain::start()
{
if (m_curtainList.isEmpty()) {
qDebug() << "显示列表为空";
return;
} m_curtain = new Curtain(m_curtainList.at(0), this);
connect(m_curtain, &Curtain::signal_pageEnd,
this, &ConsecutiveCurtain::slot_nextPage, Qt::UniqueConnection);
m_curtain->show();
m_curtain->start();
}

start()函数里调用m_curtain->show()

两种方法都可以显示出来,于是我开始纠结,为啥会显示。第二方法我还是能理解的,因为直接把m_curtain显示出来了。但是第一种方法的调用hide()函数后,就可以显示的问题,使我百思不得其解。

QWidget的show()函数

在我印象中,show()是会将调用者和调用者的子窗口也一起show出来,于是我就有了一个疑问,在chapter2::slot_Curtains1函数中,我调用了m_curtains->show(),按道理是会将Consecutive中的m_curtain一起show()出来的。

于是我去看QWidget的源码,发现所有的show()函数,都是调用的setVisible()这一个函数。

void QWidgetPrivate::setVisible(bool visible)
{
Q_Q(QWidget);
if (visible) { // show
// Designer uses a trick to make grabWidget work without showing
if (!q->isWindow() && q->parentWidget() && q->parentWidget()->isVisible()
&& !q->parentWidget()->testAttribute(Qt::WA_WState_Created))
q->parentWidget()->window()->d_func()->createRecursively(); //create toplevels but not children of non-visible parents
QWidget *pw = q->parentWidget();
if (!q->testAttribute(Qt::WA_WState_Created)
&& (q->isWindow() || pw->testAttribute(Qt::WA_WState_Created))) {
q->create();
}

这里的createRecursively则是去递归的show子窗口,这里是否创建子窗口有一个判断依据,那就是子窗口的isHidden是不是返回的false,也就是说子窗口是不是没有被隐藏。

void QWidgetPrivate::createRecursively()
{
Q_Q(QWidget);
q->create(0, true, true);
for (int i = 0; i < children.size(); ++i) {
QWidget *child = qobject_cast<QWidget *>(children.at(i));
if (child && !child->isHidden() && !child->isWindow() && !child->testAttribute(Qt::WA_WState_Created))
child->d_func()->createRecursively();
}
}

QWidget的isVisible和isHidden

在找寻为什么调用hide()之后,就能正常显示时,我发现了这两个函数isVisibleisHidden,然后我用这两个函数加到代码里做测试(得到正确的测试方法不知道走了多少弯路)

void Chapter2::slot_Curtains1()
{
m_phone->close(); qDebug() << m_curtains->isVisible(); QStringList curtains;
curtains.push_back("chapter2_4.png");
curtains.push_back("chapter2_5.png");
curtains.push_back("chapter2_6.png");
curtains.push_back("chapter2_7.png"); m_curtains->setCurtainList(curtains); connect(m_curtains, &ConsecutiveCurtain::signal_consecutiveCurtainEnd,
this, &Chapter2::slot_Phone2); m_curtains->start();
m_curtains->show(); qDebug() << m_curtains->isVisible();
}
void ConsecutiveCurtain::start()
{
if (m_curtainList.isEmpty()) {
qDebug() << "显示列表为空";
return;
} m_curtain = new Curtain(m_curtainList.at(0), this);
connect(m_curtain, &Curtain::signal_pageEnd,
this, &ConsecutiveCurtain::slot_nextPage, Qt::UniqueConnection);
qDebug() << "m_curtain visible :" << m_curtain->isVisible();
qDebug() << "m_curtain hidden :" << m_curtain->isHidden();
m_curtain->start();
}

在不调用hide()函数时,输出如下:

 true

 m_curtain visible : false

 m_curtain hidden : true

 true

调用hide()函数后,输出如下:

false

m_curtain visible : false

m_curtain hidden : false

true

然后我在查阅isHidden这个函数的手册时,发现如下解释:

到这里,我就恍然大悟:

如果不调用hide()函数,ConsecutiveCurtain的实例就不会隐藏,这样,m_curtain也就会被设置成隐藏,哪怕你show出父窗口,子窗口也会因为被隐藏这个原因,而不显示。

源码追溯

于是我又试着去源码里找这一部分的实现方法:

这是QWidget类的构造函数

我又溯源到init()函数…

这里我发现了一个q->setAttribute(Qt::WA_WState_Hidden);,这里相当于默认就给新建的窗口设置为隐藏,但是也没有找到帮助文档里所说的,于是我又继续往下寻找,我发现了一个setParent()函数

我找到这个setParent函数,

然后再setParent里找到了这个setParent_sys

最终,我在这个setParent_sys里找到了这一句代码,也就是说,如果新设置的parent是可见的,就将当前窗口设置成隐藏属性。

至此,就找出了所有问题的答案…
第一次看源码来找问题,虽然找了大半个下午,但是找到的那一瞬间真的是超级开心…

QWidget的isHidden和isVisible的更多相关文章

  1. 微软出品自动化神器【Playwright+Java】系列(七) 之 元素的可操作性验证

    前言 昨天在某平台发表了一篇这系列的文章,结果不但提示说有违禁词(java也算?),然后文章审核通过后,文章还找不到,不到去哪了,表示很郁闷,去反应未果,确实有点尴尬了. 元素的可操作性验证 关于AP ...

  2. GUI学习之四——QWidget控件学习总结

    上一章将的QObject是PyQt里所有控件的基类,并不属于可视化的控件.这一章所讲的QWidget,是所有可视化控件的基类. QWidget包含下面几点特性 a.控件是用户界面的最小的元素 b.每个 ...

  3. QWidget窗口类

    import sys from PyQt5.QtWidgets import QWidget, QApplication,QPushButton from PyQt5.QtGui import QIc ...

  4. QWidget类参考

    QWidget类是所有用户界面对象的基类. 详情请见…… #include <qwidget.h> 继承QObject和QPaintDevice. 被QButton.QFrame.QDia ...

  5. PyQt5学习 (3)--QWidget(下)

    层级关系.层级控制: 调整Z轴顺序 点击查看代码 label1 = QLabel(window) label1.setText("标签1") label1.resize(200, ...

  6. Qt:QWidget

    0.说明 QWidget类是所有用户界面对象的基类. QWidget是用户界面的原子类.它接收鼠标.键盘和来自系统的其他事件,并在屏幕上将它们绘制出来.每个Widget都是矩形的,并按照Z-order ...

  7. QDialog QMainwindow QWidget QFrame不同时候用法.

    继承关系:在Qt中所有的类都有一个共同的基类QObject ,QWidget直接继承与QPaintDevice类,QDialog.QMainWindow.QFrame直接继承QWidget 类.   ...

  8. MainWindow、QWidget和QDialog的区别和选择(转载)

    QApplication类用于管理应用程序范围内的资源,其构造函数需要main函数的argc和argv作为参数. Widget被创建时都是不可见的,widget中可容纳其他的widget. Qt中的w ...

  9. QWidget 实现 打破布局 或者 当前窗体内的 弹窗 (借助伪造实现)

    but = QtWidgets.QToolButton(Dialog2) but.setText('**') but.setAutoRaise(True) layout.addWidget(but) ...

  10. [转载] - QWidget、QMainWindow、QDialog和QFrame的区别

    继承关系:在Qt中所有的类都有一个共同的基类QObject ,QWidget直接继承与QPaintDevice类,QDialog.QMainWindow.QFrame直接继承QWidget 类.   ...

随机推荐

  1. consul discovery prefer-ip-address

    spring.cloud.consul.discovery.prefer-ip-address: 'true'

  2. Qt通用方法及类库6

    函数名 //判断是否是IP地址 static bool isIP(const QString &ip); //判断是否是MAC地址 static bool isMac(const QStrin ...

  3. Qt编写安防视频监控系统37-onvif预置位

    一.前言 预置位在视频监控系统中是不可或缺的存在,响应预置位功能的前提是要带预置位的云台球机,有些普通的云台球机其实不带预置位的,这个要检查清楚,硬件上不支持该功能的,你再怎么点也没反应.在这个视频监 ...

  4. [转]Spring Security打造一个简单Login登录页面,实现登录+跳转+注销+角色权限功能,核心代码不到100行!

    原文链接:Spring Security打造一个简单Login登录页面,实现登录+跳转+注销+角色权限功能,核心代码不到100行!

  5. [转]IDEA2020.2.3中创建JavaWeb工程的完整步骤记录

    原文链接: IDEA2020.2.3中创建JavaWeb工程的完整步骤记录

  6. MySql 建表出现的问题 : [ERR] 1064 - You have an error in your SQL syntax; check the manual.......

    使用 MySql 建表出现的问题 在使用 Navicat Premium 运行 sql 语句进行建表时,MySQL 报错如下: 1064 - You have an error in your SQL ...

  7. 微信团队分享:微信后端海量数据查询从1000ms降到100ms的技术实践

    本文由微信技术团队仇弈彬分享,原题"微信海量数据查询如何从1000ms降到100ms?",本文进行了内容修订和排版优化. 1.引言 微信的多维指标监控平台,具备自定义维度.指标的监 ...

  8. Solution Set -「AGC 007~009」C~F

    目录 「AGC 007C」Pushing Balls 「AGC 007D」Shik and Game 「AGC 007E」Shik and Travel ^ 「AGC 007F」Shik and Co ...

  9. WPF 加载外部字体

    例如将字体放入d:/Fonts 文件夹.然后就可以通过类似 btn.FontFamily = new FontFamily("file:///d:/Fonts/#Ashley"); ...

  10. DICOM 生成dcm文件

    using Dicom; using Dicom.Imaging; using Dicom.IO.Buffer; using System; using System.Collections.Gene ...