QWidget的isHidden和isVisible
QWidget的isHidden和isVisible
问题的出现
最近在写代码的时候,出现了一个问题,我新建了两个类,分别是Chapter2
和ConsecutiveCurtain
// 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.cpp
中show()
时,什么都没有显示。
我左思右想,发现了两种解决方法:
第一种方法是:
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()
之后,就能正常显示时,我发现了这两个函数isVisible
和isHidden
,然后我用这两个函数加到代码里做测试(得到正确的测试方法不知道走了多少弯路)
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的更多相关文章
- 微软出品自动化神器【Playwright+Java】系列(七) 之 元素的可操作性验证
前言 昨天在某平台发表了一篇这系列的文章,结果不但提示说有违禁词(java也算?),然后文章审核通过后,文章还找不到,不到去哪了,表示很郁闷,去反应未果,确实有点尴尬了. 元素的可操作性验证 关于AP ...
- GUI学习之四——QWidget控件学习总结
上一章将的QObject是PyQt里所有控件的基类,并不属于可视化的控件.这一章所讲的QWidget,是所有可视化控件的基类. QWidget包含下面几点特性 a.控件是用户界面的最小的元素 b.每个 ...
- QWidget窗口类
import sys from PyQt5.QtWidgets import QWidget, QApplication,QPushButton from PyQt5.QtGui import QIc ...
- QWidget类参考
QWidget类是所有用户界面对象的基类. 详情请见…… #include <qwidget.h> 继承QObject和QPaintDevice. 被QButton.QFrame.QDia ...
- PyQt5学习 (3)--QWidget(下)
层级关系.层级控制: 调整Z轴顺序 点击查看代码 label1 = QLabel(window) label1.setText("标签1") label1.resize(200, ...
- Qt:QWidget
0.说明 QWidget类是所有用户界面对象的基类. QWidget是用户界面的原子类.它接收鼠标.键盘和来自系统的其他事件,并在屏幕上将它们绘制出来.每个Widget都是矩形的,并按照Z-order ...
- QDialog QMainwindow QWidget QFrame不同时候用法.
继承关系:在Qt中所有的类都有一个共同的基类QObject ,QWidget直接继承与QPaintDevice类,QDialog.QMainWindow.QFrame直接继承QWidget 类. ...
- MainWindow、QWidget和QDialog的区别和选择(转载)
QApplication类用于管理应用程序范围内的资源,其构造函数需要main函数的argc和argv作为参数. Widget被创建时都是不可见的,widget中可容纳其他的widget. Qt中的w ...
- QWidget 实现 打破布局 或者 当前窗体内的 弹窗 (借助伪造实现)
but = QtWidgets.QToolButton(Dialog2) but.setText('**') but.setAutoRaise(True) layout.addWidget(but) ...
- [转载] - QWidget、QMainWindow、QDialog和QFrame的区别
继承关系:在Qt中所有的类都有一个共同的基类QObject ,QWidget直接继承与QPaintDevice类,QDialog.QMainWindow.QFrame直接继承QWidget 类. ...
随机推荐
- Qt/C++开发经验小技巧281-285
悬停窗体QDockWidget默认在标题栏右键会弹出悬停模块的显示隐藏菜单,如果需要去掉,会发现设置Qt::NoContextMenu或者事件过滤器拦截都是无效的,必须设置 dockWidget-&g ...
- Qt/C++编写监控实时显示和取流回放工具(回放支持切换进度)
一.前言 现在各个监控大厂做的设备,基本上都会支持通过rtsp直接取流显示,而且做的比较好的还支持通过rtsp回放取流,基本上都会约定一个字符串的规则,每个厂家都是不一样的规则,比如回放对应的rtsp ...
- springboot的yml文件中如何配置redis?
springboot的yml文件中如何配置redis? 解决方法: spring: #redis配置 redis: database: 0 timeout: 0 # Redis服务器地址 host: ...
- 移动端开源 IM 框架 MobileIMSDK v6.0 发布!
一.更新内容简介 本次为主要版本更新(本次更新内容见文末"MobileIMSDK v6.0更新内容 "一节),强势升级,将同时支持TCP.UDP.WebSocket三种协议,精心封 ...
- react之Lazy和Suspense(懒加载)
React.lazy React.lazy 函数能让你像渲染常规组件一样处理动态引入(的组件). 什么意思呢? 其实就是懒加载. 其原理就是利用es6 import()函数.这个import不是imp ...
- C++ STL学习笔记-C++ STL基础
仅自己回忆使用,若有侵权,联系删除 algorithm实用库函数 sort:迭代器类型必须为随机访问迭代器(first,last),应该支持< 运算符,可以自己写比较 nth_element() ...
- SpringBoot 项目模板-摆脱步步搭建
本文以一个简单的电商订单系统为例,源代码请访问: git clone https://github.com/e-commerce-sample/order-backendgit checkout a4 ...
- 无网环境Docker Rpm离线安装
总体思路:找一台可以联网的linux,下载docker的RPM依赖包而不进行安装(yum localinstall),将所有依赖的rpm环境打包好,再在无网环境中解压逐一安装(rpm: --force ...
- 自定义快捷命令程序(VC++加批处理)
一 概述 在看<从小工到专家-程序员修炼之道>时,看到建议使用Shell,很有感触.在很多时候,通过键盘操作,比鼠标的确会块很多,如果能用好shell命令(或批处理命令) ,的确能节省 ...
- matlib:图像旋转-缩放
需求 使用MATLAB尝试完成一个自定义的图像攻击软件,功能描述: 1)根据输入参数,完成旋转功能 2)根据输入参数,完成缩放功能 开始 旋转 参数:参数为正,顺时针旋转:参数为负,逆时针旋转 主要代 ...