Qt之菜单栏、工具栏、状态栏介绍及工具栏QAction的动态增删显示实现方式
目的
端应用程序或者编辑器基本都支持工具栏快捷功能的动态增删,即通过在菜单栏上打钩就可以在工具栏上看到相应功能的快捷按钮,取消打钩则在工具栏上就移除了该功能的快捷按钮。那么Qt如何实现这个功能,本篇目的就是记录实现此功能的方法及思路。
效果
先看下动态效果:
菜单栏动态添加动作到工具栏
介绍
首先,说下菜单栏,工具栏和状态栏区别:
- 菜单栏:一般在窗体标题的下方,有下拉选项,和可有多级子菜单。
- 工具栏:一般在菜单栏下方,可上下左右四个方向调整位置,默认在菜单栏下方(即上方向),方便操作,直接点击即可触发想要的工作。
- 状态栏:一般在窗体最下方,用于永久或者暂时显示某些状态信息等。
UI如下图所示:

Qt之QMenuBar(菜单栏)、QToolBar(工具栏)、QStatusBar(状态栏)操作说明
可在帮助里,选择索引,输入想查找的类,比如qmenubar,一般选择第一个结果(可根据需要选择其他),会弹出选择主题,选择库版本,会跳到对应的类介绍页

点击More...,会跳到Detailed Description,查看此类详细介绍,或者点击Public Functions查看此类公有成员方法。
QMenuBar(菜单栏)

QStatusBar(状态栏)

QToolBar(工具栏)


菜单栏对工具栏进行动作动态配置的实现思路
- 首先,菜单栏的子菜单和动作是已知并存在的
- 动作设置为可选择的
- 当点击动作时,触发
triggered(bool checked)信号 - 绑定槽,然后根据
checked状态,进行工具栏动态创建动作或者移除动作 - 当点击工具栏动作时,触发动作实际功能
示例
mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QToolBar>namespace Ui {
class MainWindow;
}class MainWindow : public QMainWindow
{Q_OBJECTpublic:explicit MainWindow(QWidget *parent = nullptr);~MainWindow();private:void setAction(QToolBar* pTB, QAction* pActSender, bool checked);private slots:void on_actionact11_triggered(bool checked);void on_actionact12_triggered(bool checked);void on_actionact21_triggered(bool checked);void on_actionact22_triggered(bool checked);void on_actionact31_triggered(bool checked);void on_actionact32_triggered(bool checked);void on_actionact33_triggered(bool checked);private:Ui::MainWindow *ui;QToolBar* m_pTB1;QToolBar* m_pTB2;QToolBar* m_pTB3;
};#endif // MAINWINDOW_H
mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"#include <QDebug>
#include <QMessageBox>MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);m_pTB1 = new QToolBar("tb1");m_pTB2 = new QToolBar("tb2");m_pTB3 = new QToolBar("tb3");addToolBar(m_pTB1);addToolBar(m_pTB2);addToolBar(m_pTB3);}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::setAction(QToolBar *pTB, QAction *pActSender, bool checked)
{if(checked){foreach (QAction* pAct, pTB->actions()) {if(pAct->text().compare(pActSender->text()) == 0){return;}}QAction* pActClone = new QAction(pActSender->text(), this);connect(pActClone, &QAction::triggered, this, [this, pActClone](){ui->statusBar->showMessage(QString("我是 %1").arg(pActClone->text()), 2000);QMessageBox::information(this, "提示", QString("我是 %1").arg(pActClone->text()));});pTB->insertAction(0, pActClone);}else {foreach (QAction* pAct, pTB->actions()) {if(pAct->text().compare(pActSender->text()) == 0){pTB->removeAction(pAct);return;}}}
}void MainWindow::on_actionact11_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB1, pActSender, checked);
}void MainWindow::on_actionact12_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB1, pActSender, checked);
}void MainWindow::on_actionact21_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB2, pActSender, checked);
}void MainWindow::on_actionact22_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB2, pActSender, checked);
}void MainWindow::on_actionact31_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB3, pActSender, checked);
}void MainWindow::on_actionact32_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB3, pActSender, checked);
}void MainWindow::on_actionact33_triggered(bool checked)
{QAction* pActSender = dynamic_cast<QAction*>(sender());setAction(m_pTB3, pActSender, checked);
}
ui
// ui的话,主要是添加一些菜单和动作,工具栏是代码实现的// 动作名称

main.cpp
#include "mainwindow.h"
#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);MainWindow w;w.show();return a.exec();
}
分析
以上示例,主要函数为:
private:void setAction(QToolBar* pTB, QAction* pActSender, bool checked);
void MainWindow::setAction(QToolBar *pTB, QAction *pActSender, bool checked)
{if(checked){foreach (QAction* pAct, pTB->actions()) {if(pAct->text().compare(pActSender->text()) == 0){return;}}QAction* pActClone = new QAction(pActSender->text(), this);connect(pActClone, &QAction::triggered, this, [this, pActClone](){ui->statusBar->showMessage(QString("我是 %1").arg(pActClone->text()), 2000);QMessageBox::information(this, "提示", QString("我是 %1").arg(pActClone->text()));});pTB->insertAction(0, pActClone);}else {foreach (QAction* pAct, pTB->actions()) {if(pAct->text().compare(pActSender->text()) == 0){pTB->removeAction(pAct);return;}}}
}
根据传入的参数,进行工具栏动态的创建。
此外:
ui->statusBar->showMessage(QString("我是 %1").arg(pActClone->text()), 2000);
上述代码是 暂时显示文本,时间是2000ms,之后会消失。
结论
学以致用。
相关文章:
Qt之菜单栏、工具栏、状态栏介绍及工具栏QAction的动态增删显示实现方式
目的 端应用程序或者编辑器基本都支持工具栏快捷功能的动态增删,即通过在菜单栏上打钩就可以在工具栏上看到相应功能的快捷按钮,取消打钩则在工具栏上就移除了该功能的快捷按钮。那么Qt如何实现这个功能,本篇目的就是记录实现此功能的方法及思…...
十四天学会C++之第八天:文件操作
1. 文件的打开和关闭 文件操作的基本概念。打开文件:使用fstream库打开文件以供读写。关闭文件:确保文件在使用完毕后正确关闭。 文件的打开和关闭:C 文件操作入门 在C编程中,文件操作是一项重要的任务,可以读取和写…...
基于(N-1)×(N-1)棋盘的解的情况推出N×N棋盘的解的情况的N皇后问题
N皇后问题是一个比较经典的问题,其主要目标是在NN的棋盘上,放置N个皇后,要求所有皇后之间不能互相攻击,即任意两个皇后不能处在同一行、同一列或同一对角线上。解决该问题可以采用递归的方式,基于(N-1)棋盘的解的情况推…...
Vue mixin混入
可以把多个组件中共有的配置提取出来构成一个混入。 一、配置混入 (一) 创建mixin.js 这里的名字可以自定义,但是为了方便识别,多数场景下都写mixin。 mixin.js 要创建在src目录下,与main.js平级: &…...
基于 FFmpeg 的跨平台视频播放器简明教程(十):在 Android 运行 FFmpeg
系列文章目录 基于 FFmpeg 的跨平台视频播放器简明教程(一):FFMPEG Conan 环境集成基于 FFmpeg 的跨平台视频播放器简明教程(二):基础知识和解封装(demux)基于 FFmpeg 的跨平台视频…...
正点原子嵌入式linux驱动开发——Linux LCD驱动
LCD是很常用的一个外设,通过LCD可以显示绚丽的图片、界面等,提交人机交互的效率。STM32MP1提供了一个LTDC接口用于连接RGB接口的液晶屏。本章就来学校一下如何在Linux下驱动LCD屏。 LCD和LTDC简介 LCD简介 这里在当时学习stm32裸机开发的时候就学过了…...
2-Java进阶知识总结-6-多线程
文章目录 多线程--基本概念并发和并行进程和线程多线程 多线程--实现方式一,继承Thread类方法介绍实现步骤注意事项 方式二,实现Runnable接口Thread构造方法实现步骤 方式三,实现Callable接口方法介绍实现步骤 三种多线程实现方法对比 多线程…...
openwrt下游设备在校园网(DLUT-LingShui)中使用ipv6网络
背景:校园网最多支持6台设备的无感认证,需要使用路由器(本人使用openwrt系统)为更多的设备提供网络,但校园网分配的ipv6地址子网为/128,不能为路由器下的设备分配全球ipv6地址,因此需要使用nat6转发下游设备的局域网ip…...
10个基于.Net开发的Windows开源软件项目
1、基于.NET的强大软件开发工具 一个基于.Net Core构建的简单、跨平台快速开发框架。JNPF开发平台前后端封装了上千个常用类,方便扩展;集成了代码生成器,支持前后端业务代码生成,满足快速开发,提升工作效率;…...
Java多线程秘籍,掌握这5种方法,让你的代码优化升级
介绍5种多线程方法,助您提高编码效率! 如果您的应用程序与那些能够同时处理多个任务的应用程序相比表现不佳,很可能是因为它是单线程的。解决这个问题的方法之一是采用多线程技术。 以下是一些可以考虑的方法: 线程(…...
npm install报错 缺少python
报错信息: Building:E:tolsnvmnodesnodeexe : ode emos ant-desig-we-eos odemodules node-gypbintnode-gp.s rebld -verbose -Libsass_ext --Libsas_cflags- lags --libsass_librarygyp info it worked if it ends with ok gyp verb cli [ gyp verb cliE: toolsnv…...
达梦:开启sql日志记录
前言 开启sql日志记录,可协助排查定位数据库问题。生产开启会有一定的性能消耗,建议打开 SQL 日志异步刷盘功能 1.配置sqllog.ini文件 sqllog.ini 用于 SQL 日志的配置,当且仅当 INI 参数 SVR_LOG1 时使用。 运行中的数据库实例,可…...
C语言开发,指针进阶,字符串查找,包含,拼接
文章目录 C语言开发,指针进阶。1.字符串与指针的关系2.指针获取字符串具体内容3.字符串比较,查找,包含,拼接4.字符串大小写 C语言开发,指针进阶。 1.字符串与指针的关系 // // Created by MagicBook on 2023-10-22. …...
PyCharm中文使用详解
PyCharm是一个Python IDE,可以帮助程序员节省时间,提高生产力。那么具体怎么用呢?本文介绍了PyCharm的安装、插件、外部工具、专业功能等,希望对大家有所帮助。 之前没有系统介绍过PyCharm。如何配置环境,如何DeBug&a…...
一键同步,无处不在的书签体验:探索多电脑Chrome书签同步插件
说在前面 平时大家都是怎么管理自己的浏览器书签数据的呢?有没有过公司和家里的电脑浏览器书签不同步的情况?有没有过电脑突然坏了但书签数据没有导出,导致书签数据丢失了?解决这些问题的方法有很多,我选择自己写个chr…...
在Go项目中二次封装Kafka客户端功能
1.摘要 在上一章节中,我利用Docker快速搭建了一个Kafka服务,并测试成功Kafka生产者和消费者功能,本章内容尝试在Go项目中对Kafka服务进行封装调用, 实现从Kafka自动接收消息并消费。 在本文中使用了Kafka的一个高性能开源库Sarama, Sarama是一个遵循MIT许可协议的Apache Kafk…...
CVE-2021-44228 Apache log4j 远程命令执行漏洞
一、漏洞原理 log4j(log for java)是由Java编写的可靠、灵活的日志框架,是Apache旗下的一个开源项目,使用Log4j,我们更加方便的记录了日志信息,它不但能控制日志输出的目的地,也能控制日志输出的内容格式;…...
前端跨域相关
注:前端配置跨域后服务器端(Nginx)也需要配置,否则接口无法访问 vue跨域 配置文件 /vue.config.js devServer: { port: 7100, proxy: { /api: { target: http://域名, changeOrigin: true, logLevel: debug, pathRewrite: { ^/…...
HTML笔记-狂神
1. 初识HTML 什么是HTML? Hyper Text Markup Language : 超文本标记语言 超文本包括:文字、图片、音频、视频、动画等 目前使用的是HTML5,使用 W3C标准 W3C标准包括: 结构化标准语言(HTML、XML) 表现标…...
python自动化测试工具selenium
概述 selenium是网页应用中最流行的自动化测试工具,可以用来做自动化测试或者浏览器爬虫等。官网地址为:Selenium。相对于另外一款web自动化测试工具QTP来说有如下优点: 免费开源轻量级,不同语言只需要一个体积很小的依赖包支持…...
STM32 定时器与 PWM 输出:电机调速、LED 呼吸灯实战
在嵌入式开发的世界里,有一个问题困扰着无数初学者:微控制器只能输出0V和3.3V(或5V)的数字信号,但现实世界中的设备——从电机的转速控制到LED的亮度调节——需要的却是连续的模拟信号。如何用数字引脚“模拟”出模拟电…...
17.在 React 中如何根据条件决定渲染哪个组件?
在 React 里,组件不是一上来就“全给你渲染出来”的。 很多时候,我们希望:界面要看情况说话——登录了看“欢迎回来”没登录就看“请先登录”加载中只给你个转圈圈请求失败再丢个错误提示这些“根据条件,决定渲染什么”的行为&…...
从 LLM 到 OpenClaw:七步看懂 Prompt、Memory、MCP、Skills、Agent
从 LLM 到 OpenClaw:七步看懂 Prompt、Memory、MCP、Skills、Agent 这两年 AI 术语越来越多:LLM、MCP、Agent、Skills、OpenClaw。 如果你不是技术背景,第一次看到这串词,基本都会懵。下面我用一个统一场景来讲:把 AI…...
百度飞桨PaddleOCR图片印章检测技术简介
百度飞桨PaddleOCR图片印章检测技术简介 全文链接 百度飞桨PaddleOCR图片印章检测技术简介 github仓库:使用PaddleOCR识别图片红色印章文字 red-seal-ocr 3.X和2.X区别较大,建议使用3.X版本。 PaddleX简介 PaddleX github地址PaddleX模型产线使用概览…...
千问3.5-9B参数调优:降低OpenClaw复杂任务token消耗
千问3.5-9B参数调优:降低OpenClaw复杂任务token消耗 1. 为什么需要关注token消耗? 去年冬天第一次用OpenClaw自动整理季度报告时,我被账单吓了一跳——连续运行3天的复杂任务消耗了价值200多美元的token。这让我意识到,在享受自…...
Element Plus:Vue 3企业级UI组件库的全方位解析与实践指南
Element Plus:Vue 3企业级UI组件库的全方位解析与实践指南 【免费下载链接】element-plus 🎉 A Vue.js 3 UI Library made by Element team 项目地址: https://gitcode.com/GitHub_Trending/el/element-plus Element Plus作为基于Vue 3的企业级UI…...
Bugtton:ATmega328P专用超低开销按钮消抖库
1. 项目概述Bugtton 是一款专为 ATmega328P 微控制器深度优化的轻量级按钮消抖库,其设计哲学直指嵌入式系统中一个被长期忽视却至关重要的性能瓶颈:空闲状态下的 CPU 周期开销。在传统 Arduino 风格的按钮处理方案中,digitalRead()函数因其通…...
前端必懂:开发环境、构建打包的核心差异,新手再也不踩坑
前端必懂:开发环境、构建打包的核心差异,新手再也不踩坑 文章目录前端必懂:开发环境、构建打包的核心差异,新手再也不踩坑一、先一句话打通核心逻辑(新手必记)二、开发环境:怎么舒服怎么写&…...
手把手教你开发电竞护航系统:从零到上线的小程序全流程
手把手教你开发电竞护航系统:从零到上线的小程序全流程 电竞产业近年来呈现爆发式增长,职业选手和游戏爱好者对专业服务的需求与日俱增。一款优秀的电竞护航小程序不仅能提供赛事资讯、战队管理、训练计划等基础功能,更能通过智能算法为玩家匹…...
物联网设备上高德地图离线地图加载慢?5秒内快速加载的终极解决方案
物联网设备高德地图离线加载优化实战:从2分钟到5秒的进阶方案 在智能电表、车载终端、工业传感器等物联网设备中,离线地图的快速加载直接影响着用户体验与系统响应效率。我们曾遇到一个典型场景:某共享单车智能锁通过4G模块上报位置时&#x…...
