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上导致状态栏信息消失的问题。
热门推荐
广东面食三巨头:竹升面、云吞面、车仔面
广东美食节:竹升面&云吞面大比拼
天然代糖木糖醇:三重健康益处与安全使用指南
木糖醇:糖尿病患者的甜味替代品,需控制摄入量
大寒时节如何养生?四大方面助你温暖过冬
靠当地人推荐,“打听式旅游”成暑期新宠
可乐鸡翅的精准做法
12道低碳水化合物中式蔬菜食谱
超级简易改良版可乐鸡翅
零基础也能学会:微信公众号创建与运营全攻略
喀左美食探秘:从碗砣到烤全羊
科比演讲技巧大揭秘:如何像他一样充满感染力?
新南威尔士州寒假特辑:悉尼歌剧秀、中央海岸骑马等你来
红烧牛肉丸到冬瓜:8道家常红烧菜制作教程
老挝:女性主导社会,湄公河孕育文明,中老铁路带来新机遇
占巴塞:老挝的咖啡天堂与双瀑布奇观
70、80、90岁老人祝寿,这些鲜花最能表达祝福
遗嘱需要公证吗?详解遗嘱生效条件
自书遗嘱怎么证明有效
遗嘱公证的法律效力
腰椎磨损老化高发,科学防治守护脊柱健康
TikTok或被禁引发用户大迁移,小红书海外下载量暴增
中国企业加速布局东南亚,出海战略转向技术驱动
告别“烧钱获客”:提升用户粘性成平台经济新趋势
各大银行上线存量房贷利率查询,调整幅度因人而异
牛肉炖汤食材搭配完全指南:从基础到创新,一文掌握炖出完美牛肉汤的关键
喀左羊杂汤:一碗凝聚历史与文化的美味
喀左羊杂汤:一碗热汤里的历史记忆
喀左陈醋:百年匠心酿造,一滴传世美味
网约车系统架构设计与实现的实践总结