QT窗口无法激活弹出问题排查记录

news/2024/9/25 23:45:16/

问题背景

问题环境

            操作系统: 银河麒麟V10SP1qt版本  : 5.12.12

碰见了一个问题应用最小化,然后激活程序窗口无法弹出

            这里描述一下代码的逻辑,使用QLocalServer实现一个单例进程,具体的功能就是在已存在一个程序A进程时,再启动这个程序A,新的程序A进程会被杀死,然后激活已存在的进程,使窗口弹出

跟踪代码发现走到了激活函数(如下所示)

            this->raise();this->activateWindow();
问题现象如下,确实是有激活效果,图标闪烁了,但是窗口在最小化的情况不会弹


排查过程

        首先,我想写个简易的demo来复现这个问题,但是下面所示的代码并不能复现出上面提到的问题,最小化窗口可以被激活弹出,所以我还是得用原来的代码进行排查

              MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow){ui->setupUi(this);//使用定时器触发执行槽函数QObject::connect(&m_timer, &QTimer::timeout, this,&MainWindow::slottimer);}MainWindow::~MainWindow(){delete ui;}//3秒钟执行一次void MainWindow::on_pushButton_clicked(){m_timer.start(3000);}//激活窗口void MainWindow::slottimer(){this->raise();this->activateWindow();}

为了排除操作系统的影响,我选择用同一份代码在不同系统上测试一下,看看效果

这里我测试了kylin、ubuntu都有这种问题,说明大概率和操作系统无关,uos因为操作系统镜像不好下载我就没测

和系统没关系,执行的时候也走了对应的函数,那么现在只能跟踪qt代码看看为什么没有弹出了,跟踪后有如下发现

            最后进入了QXcbWindow::requestActivateWindow()函数执行xcb_send_event(xcb_connection(), 0, xcbScreen()->root(),XCB_EVENT_MASK_STRUCTURE_NOTIFY | XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT,(const char *)&event);

可以看到激活窗口实际是调用xcb模块的功能

            xcb介绍,XCB是X协议的一个C语言绑定,它提供了一种更现代、更高效的方式来与X Window System进行交互。xcb官网  https://xcb.freedesktop.org/

我分别在使用wayland和X协议的系统上进行测试,最后代码都走到了xcb_send_event方法

        接下来我就开始尝试对xcb_send_event的传参进行修改,重点是修改第二和第四个传参,其实这一步修改就是瞎改,也没有什么修改逻辑,我做的尝试如下:

            0  ->  1XCB_EVENT_MASK_STRUCTURE_NOTIFY | XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT  在此基础上增加或者减少事件掩码

        在修改测试过程中,发现偶尔能弹出来,但概率很低,然后我再把参数都恢复,发现也能偶尔弹出来,此刻我感觉和修改xcb_send_event参数没有什么关系了

        再结合前面我自己写的demo可以正常弹出的情况,我觉得代码中使用QLocalServer实现单例进程并且杀死新进程的方法可能会造成激活窗口操作失效(纯猜测,但改变了我的解决思路)

        所以还是要从应用调用代码的角度去解决


解决方案

如上所述,激活窗口一次没有没有生效,是不是多执行几次激活是不是就可以了?

       所以我修改激活窗口的方式,在原先激活窗口的位置开启定时器(定时器触发时间间隔500Ms~2s即可),在槽函数中执行激活窗口操作,当窗口激活成功就停止定时器

       在测试中发现,定时器第一次触发激活窗口的现象只是任务栏的图标闪烁,但窗口不弹出,而在第二次触发时窗口从任务栏中弹出,一般情况下第二次触发就能弹出了


后记

      为什么第一次触发激活窗口没有成功?目前还没有找到根源,我猜测可能是因为起第二进程再杀死导致了一些时序错误或者影响了堆栈,后续还要验证

      如果单纯从解决窗口不弹出的角度来说,还有一个方法就是调用命令wmctrl,但也可能存在失效的问题

            QProcess myProcess;QString program = "wmctrl";QStringList arguments;arguments << "-a" << w->windowTitle();myProcess.start(program,arguments);myProcess.waitForFinished();


http://www.ppmy.cn/news/1530501.html

相关文章

使用 Internet 共享 (ICS) 方式分配ip

设备A使用dhcp的情况下&#xff0c;通过设备B分配ip并共享网络的方法。 启用网络共享&#xff08;ICS&#xff09;并配置 NAT Windows 自带的 Internet Connection Sharing (ICS) 功能可以简化 NAT 设置&#xff0c;允许共享一个网络连接给其他设备。 打开网络设置&#xff1…

Maven 项目无法下载某个依赖

问题描述 在使用 Maven 构建 Java 项目时&#xff0c;有时会遇到无法下载特定依赖的问题。这可能会影响项目的构建过程&#xff0c;导致构建失败。本文档旨在提供一系列步骤&#xff0c;帮助开发者定位并解决此类问题。 常见原因 依赖仓库配置错误&#xff1a;项目的 pom.xm…

使用Docker和Macvlan驱动程序模拟跨主机跨网段通信

以下是使用Docker和Macvlan驱动程序模拟跨主机跨网段通信的架构图&#xff1a; #mermaid-svg-b7wuGoTr6eQYSNHJ {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-b7wuGoTr6eQYSNHJ .error-icon{fill:#552222;}#mermai…

xpath应用大全

一、xpath在爬虫中的应用 1、/div 表示从根节点开始选取div节点 2、/span 表示从根节点开始选取span节点 3、//a 表示选取文档中所有a节点而不考虑其位置 4、class 表示选取名为class的属性 5、 . 表示选取当前节点 6、 .. 表示选取当前节点的父节点 7、/div/a 表示从根…

香港服务器PING测试有什么作用?

PING测试是一种常用的网络诊断工具&#xff0c;用于测试计算机与服务器之间的网络连通性和响应时间。对于香港服务器&#xff0c;进行PING测试有以下几个作用&#xff1a; 香港服务器PING测试的作用包括&#xff1a; 检查网络连通性&#xff1a;PING测试可以帮助确定从本地计算…

DataX实战:从MongoDB到MySQL的数据迁移--修改源码并测试打包

在现代数据驱动的业务环境中&#xff0c;数据迁移和集成是常见的需求。DataX&#xff0c;作为阿里云开源的数据集成工具&#xff0c;提供了强大的数据同步能力&#xff0c;支持多种数据源和目标端。本文将介绍如何使用DataX将数据从MongoDB迁移到MySQL。 环境准备 安装MongoDB…

在 Vue 项目中引用 assets 文件夹中的几种方式

在 Vue 项目中引用 assets 文件夹中的图片可以通过以下几种方式&#xff1a; 一、在模板中引用 在.vue文件的模板部分&#xff0c;可以使用相对路径来引用图片。例如&#xff1a; <template><img src"/assets/image.jpg" alt"描述图片的文本"&…

DataGrip在Windows和MacOS平台上的快捷键

0. 背景信息 No.说明1测试DataGrip版本号 : 2024.2.2 1. Windows下快捷键 2. MacOS下快捷键