Qt开发经验:鼠标悬停在QMenu上时导致状态栏信息消失
创作时间:
作者:
@小白创作中心
Qt开发经验:鼠标悬停在QMenu上时导致状态栏信息消失
引用
CSDN
1.
https://m.blog.csdn.net/chenai886/article/details/144055559
本文将围绕鼠标悬停在QMenu上时导致状态栏信息消失的情况,整合事件信息流,对流程进行详细解读,并通过代码剖析每个关键环节的行为。
1. 问题描述
当鼠标悬停在QMenu的空白区域、分隔符或禁用菜单项时:
- 当前的QAction被置为nullptr。
- QMenu调用previousAction->showStatusText("", ...),发送空字符串的QStatusTipEvent。
- 状态栏(QStatusBar)接收到该事件后清空状态栏内容。
2. 信息流的完整路径
整个信息流从鼠标事件开始,到状态栏显示或清空结束,共分为以下5个阶段:
- 鼠标移动事件触发(QMenu):
- 鼠标移动到QMenu内部,触发QMenu::mouseMoveEvent。
- 检测鼠标悬停位置,并调用QMenuPrivate::setCurrentAction更新当前QAction。
- 当前QAction状态更新(QMenuPrivate):
- 如果鼠标在空白区域,调用showStatusText("", ...),生成空字符串的QStatusTipEvent。
- 状态提示事件发送(QActionPrivate):
- 将QStatusTipEvent发送到QMainWindow或父控件。
- 事件捕获与处理(QMainWindow):
- QMainWindow::event捕获QEvent::StatusTip,并调用QStatusBar::showMessage。
- 状态栏显示更新(QStatusBar):
- 接收到空字符串消息,调用clearMessage清空状态栏。
3. 信息流的代码与详细解读
3.1 鼠标移动事件触发
当鼠标移动到QMenu内部时,QMenu::mouseMoveEvent被触发。
核心代码:QMenu::mouseMoveEvent
void QMenu::mouseMoveEvent(QMouseEvent *e) {
Q_D(QMenu);
QAction *action = d->actionAt(e->pos()); // 获取鼠标悬停的 QAction
if ((!action || action->isSeparator()) && !d->sloppyState.enabled()) {
d->setCurrentAction(action); // 鼠标在空白区域,action 为 nullptr
return;
}
d->setCurrentAction(action); // 鼠标悬停在有效菜单项上
}
- 行为分析:
- d->actionAt(e->pos()):检测鼠标当前位置是否有对应的QAction。
- 如果鼠标在空白区域或分隔符上,action == nullptr。
- 调用d->setCurrentAction(action),将当前激活的菜单项设置为nullptr。
3.2 当前QAction状态更新
QMenuPrivate::setCurrentAction是管理QAction激活状态的核心逻辑。
核心代码:QMenuPrivate::setCurrentAction
void QMenuPrivate::setCurrentAction(QAction *action, int popup, SelectionReason reason, bool activateFirst) {
QAction *previousAction = currentAction;
currentAction = action; // 更新当前 QAction
if (action) {
activateAction(action, QAction::Hover); // 激活当前 QAction
} else if (previousAction) {
// 当前 QAction 被清空,调用 previousAction 的 showStatusText
previousAction->d_func()->showStatusText(topCausedWidget(), QString());
}
}
- 行为分析:
- 如果currentAction == nullptr,说明鼠标离开了有效的菜单项。
- 调用previousAction->showStatusText("", ...),生成空字符串状态提示。
3.3 状态提示事件发送
QActionPrivate::showStatusText用于生成并分发QStatusTipEvent。
核心代码:QActionPrivate::showStatusText
bool QActionPrivate::showStatusText(QWidget *widget, const QString &str) {
if (QObject *object = widget ? widget : parent) {
QStatusTipEvent tip(str); // 创建状态提示事件
QApplication::sendEvent(object, &tip); // 发送事件到目标控件
return true;
}
return false;
}
- 行为分析:
- str是状态提示内容,此处为""(空字符串)。
- 使用QApplication::sendEvent分发QStatusTipEvent,通常发送到QMainWindow。
3.4 事件捕获与处理
QMainWindow是QStatusTipEvent的中继点,它捕获事件并调用状态栏的showMessage。
核心代码:QMainWindow::event
bool QMainWindow::event(QEvent *event) {
#if QT_CONFIG(statustip)
case QEvent::StatusTip:
#if QT_CONFIG(statusbar)
if (QStatusBar *sb = d->layout->statusBar())
sb->showMessage(static_cast<QStatusTipEvent *>(event)->tip()); // 更新状态栏
else
#endif
static_cast<QStatusTipEvent *>(event)->ignore(); // 忽略事件
return true;
#endif
return QMainWindow::event(event); // 其他事件交由父类处理
}
- 行为分析:
- 捕获QEvent::StatusTip类型事件。
- 调用QStatusBar::showMessage,将事件中的tip内容传递给状态栏。
- 如果tip == "",状态栏会显示空消息并清空内容。
3.5 状态栏显示更新
状态栏(QStatusBar)是信息流的终点,通过showMessage和clearMessage更新显示内容。
核心代码:QStatusBar::showMessage
void QStatusBar::showMessage(const QString &message, int timeout) {
Q_D(QStatusBar);
if (message.isEmpty()) {
clearMessage(); // 如果消息为空,清空状态栏
return;
}
d->tempItem = message; // 更新临时消息
hideOrShow(); // 根据消息状态更新状态栏显示
}
核心代码:QStatusBar::clearMessage
void QStatusBar::clearMessage() {
Q_D(QStatusBar);
d->tempItem.clear(); // 清空临时消息
hideOrShow(); // 隐藏状态栏
}
- 行为分析:
- 如果message是空字符串,调用clearMessage清空状态栏。
- 调用hideOrShow更新状态栏的可见性。
4. 问题的根本原因
- 鼠标悬停在空白区域或分隔符时:
- QMenu的currentAction被置为nullptr。
- 调用previousAction->showStatusText("", ...),生成空字符串状态提示。
- 状态栏响应空字符串事件:
- QMainWindow捕获QStatusTipEvent,将tip内容传递给状态栏。
- 如果tip == "",状态栏调用clearMessage清空显示内容。
5. 解决方案
方法1:拦截空字符串事件
通过事件过滤器,阻止空字符串的QStatusTipEvent被传递到状态栏。
bool MainWindow::eventFilter(QObject *obj, QEvent *event) {
if (event->type() == QEvent::StatusTip) {
QStatusTipEvent *se = static_cast<QStatusTipEvent *>(event);
if (se->tip().isEmpty()) {
return true; // 阻止空字符串事件
}
}
return QMainWindow::eventFilter(obj, event);
}
6. 总结
事件信息流的完整路径
- 鼠标移动到QMenu内部。
- QMenu更新currentAction,调用showStatusText("")。
- QActionPrivate发送空字符串的QStatusTipEvent。
- QMainWindow捕获事件并调用状态栏的showMessage。
- QStatusBar接收空字符串,调用clearMessage清空显示。
问题的根因
- 鼠标悬停在空白区域时,QMenu发送空字符串事件。
- 状态栏默认响应空字符串事件,清空内容。
通过优化QMenu和拦截空字符串事件,可以有效解决鼠标悬停在QMenu上导致状态栏信息消失的问题。
热门推荐
浪潮信息面临发展阵痛,能否重塑竞争力?
世界社会公正日:值得关注的6个社会问题
事件驱动架构详解:触发与响应构建高效系统
水果是如何影响血糖的?低糖水果推荐!
创新疫苗全程质量管理体系:山东三地疾控中心的实践探索
宛城之战爆发原因深度解析:曹操损失惨重之始末
宛城之战——张绣智胜曹操的致命一击
《从一到无穷大》:一本影响深远的科普经典
如何在研发团队中解决技术集成问题
三种食物助中老年人血糖稳如泰山
经典“双截龙”再添新作!《双截龙再临》大量游戏新情报公开
拉筋或能防血栓?这3个动作让血管“不再堵车”!
不吃主食能减肥吗?长肉“比较快”的几类主食,建议减肥的人少吃
狗狗乱咬东西怎么训?3周见效的科学纠正法
速生杨树的种植和管理!杨树几月份浇水最合适?怎么施肥长的快?
青杨:坚韧生命的象征
崔颢黄鹤楼诗千古绝唱,为何却鲜为人知?
唐朝的婚姻制度有何特点?女性在婚姻中的地位有显著提升
保障换电柜消防安全:热气溶胶灭火装置
六种常见植物油的特性与烹饪指南
精华和面霜顺序:保持肌肤健康的秘密
中国煤科实施系列生态环保科技创新项目,用实际行动守护绿水青山
显卡温度控制指南:如何防止GPU变成“烤肉”
股权溢价计算指标解析及其应用案例
蚊子灭绝会有什么后果?探讨蚊子消失对生态的影响
马桶二边尺寸留多少,家居设计中的黄金比例
如何分析新闻对市场的影响
健康的体脂率(最健康的体脂率)
如何选择在经济滞涨期的投资策略
艾森斯坦判别法:判断整系数多项式在有理数域是否可约的重要方法