QT笔记--》QMenu
文章目录
- 1、概要
- 1.1、基于 QAction 的 QMenu 实现模式
- 1.2、 通过按钮触发的 QMenu 实现(ui)
- 1.3、菜单触发显示自定义窗口(ui)
1、概要
在 Qt 界面开发中,QMenu 是构建交互菜单的核心组件,其实现方式直接影响菜单的灵活性与易用性。本文将聚焦两种主流实现模式 —— 基于
QAction 的联动模式和直接通过按钮触发的模式,深入解析它们的实现逻辑、适用场景及优劣势,帮助开发者根据实际需求选择最优方案。
1.1、基于 QAction 的 QMenu 实现模式
(1)、什么是 QAction?
QAction 是 Qt 中用于封装用户操作的对象,它可以关联菜单、工具栏等组件,集中管理操作的文本、图标、快捷键及触发逻辑。
在 QMenu 中,QAction 是菜单选项的 “灵魂”,菜单的显示与功能触发均围绕它展开。
(2)、实现步骤
//创建QAction对象
QAction *actionNew = new QAction(QIcon("new.png"), tr("新建(&N)"), this);
actionNew->setShortcut(QKeySequence::New);
connect(actionNew, &QAction::triggered, this, &MainWindow::onNewFile);
//构建 QMenu 并关联 QAction
QMenu *fileMenu = menuBar()->addMenu(tr("文件(&F)"));
fileMenu->addAction(actionNew);
fileMenu->addSeparator(); // 添加分隔线
1.2、 通过按钮触发的 QMenu 实现(ui)
(1)、实现效果
QMenu
(2)、具体实现
//mainwindow.ui
//mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QMenu>
#include <QMap>
#include <QDebug>
#include <QMouseEvent>
#include <QPoint>QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();public slots:void installMenuEventFilters(); //安装事件过滤器到所有菜单标题bool eventFilter(QObject* obj, QEvent* event); //事件过滤器:捕捉菜单标题的鼠标点击事件void onMenubar_even(QMenu* menu);
signals:void menuClicked(QMenu* menu);private:Ui::MainWindow *ui;QMap<QMenu*, QAction*> menuToActionMap;
};
#endif // MAINWINDOW_H
//mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);connect(this,&MainWindow::menuClicked,this,&MainWindow::onMenubar_even);installMenuEventFilters();
}MainWindow::~MainWindow(){delete ui;}void MainWindow::onMenubar_even(QMenu* menu)
{if (!menu) {qDebug() << "错误:接收到空菜单指针";return;}// 检查tabWidget是否存在if (!ui->tabWidget) {qDebug() << "错误:tabWidget未初始化";return;}QString objName = menu->objectName();if (objName == "menuA") {ui->tabWidget->setCurrentIndex(0);}else if(objName == "menuB"){ui->tabWidget->setCurrentIndex(1);;}else if(objName == "menuC"){ui->tabWidget->setCurrentIndex(0);}
}// 安装事件过滤器到所有菜单标题
void MainWindow::installMenuEventFilters() {// 遍历菜单栏中的所有动作for (QAction* action : ui->menubar->actions()) {QMenu* menu = action->menu();if (menu) {menuToActionMap[menu] = action;ui->menubar->installEventFilter(this);}}
}// 事件过滤器:捕获菜单标题的鼠标点击事件
bool MainWindow::eventFilter(QObject* obj, QEvent* event){if (obj == ui->menubar && event->type() == QEvent::MouseButtonRelease) {QMouseEvent* mouseEvent = static_cast<QMouseEvent*>(event);QPoint pos = mouseEvent->pos();for (auto it = menuToActionMap.begin(); it != menuToActionMap.end(); ++it) {QMenu* menu = it.key();QAction* action = it.value();QRect actionRect = ui->menubar->actionGeometry(action);if (actionRect.contains(pos)) {emit menuClicked(menu);return true;}}}return QMainWindow::eventFilter(obj, event);
}
1.3、菜单触发显示自定义窗口(ui)
(1)、功能描述
从技术实现角度,这种功能的核心是:
(a)、监听 QMenu 中特定菜单项(QAction 或自定义部件)的触发信号
(b)、在信号处理函数中调用 QWidget 的show()、exec()(模态)或popup()等方法显示窗口
(2)、实现效果
QMenu菜单触发显示自定义弹框
(3)、具体实现
//mainwindow.ui
//mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QDebug>
#include <QTextCursor>
#include <QListView>
#include <QListView>
#include <QAbstractItemView>
#include "doubleclickfoldwidget.h"QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECT
public:MainWindow(QWidget *parent = nullptr);~MainWindow();public slots: //QTextEditvoid on_fruitParty_triggered();private:Ui::MainWindow *ui;FruitPartyWidget *m_fruitPartyWidget; //初始化自定义窗口类
};
#endif // MAINWINDOW_H
MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);m_doubleClickFoldWidget=new DoubleClickFold();//初始化自定义窗口类
}
MainWindow::~MainWindow()
{delete ui;if(m_doubleClickFoldWidget){delete m_doubleClickFoldWidget;m_doubleClickFoldWidget=nullptr;}
}void MainWindow::on_fruitParty_triggered()
{m_fruitPartyWidget->show();
}
//自定义窗口类,实现了双击QLable折叠QWidget
//fruit_party_widget.ui
//fruit_party_widget.h
#include <QLabel>
#include <QMouseEvent>class FruitPartyWidget : public QWidget
{Q_OBJECT
public:explicit FruitPartyWidget(QWidget *parent = nullptr);~FruitPartyWidget();protected:bool eventFilter(QObject *obj, QEvent *event) override; // 重写事件过滤器,用于捕获子部件的事件
private slots:void toggleWidgetVisibility();// 切换窗口部件可见性的槽函数(折叠/展开)private:Ui::FruitPartyWidget *ui;
};
//fruit_party_widget.cpp
FruitPartyWidget::FruitPartyWidget(QWidget *parent) :QWidget(parent),ui(new Ui::DoubleClickFold)
{ui->setupUi(this);// 为 QLabel 安装事件过滤器以捕获点击事件ui->label->installEventFilter(this);
}
// 重写事件过滤器,用于捕获子部件的事件
bool FruitPartyWidget::eventFilter(QObject *obj, QEvent *event)
{ //判断事件来源是否为label,且事件类型为鼠标双击if (obj == ui->label && event->type() == QEvent::MouseButtonDblClick) {toggleWidgetVisibility();// 双击时触发窗口可见性切换(折叠/展开)return true;}return QObject::eventFilter(obj, event);// 未处理的事件交给父类默认处理
}
// 切换窗口部件可见性的槽函数(折叠/展开)
void FruitPartyWidget::toggleWidgetVisibility()
{bool isVisible = ui->widget->isVisible();ui->widget->setVisible(!isVisible);
}