Qt开发技术【下拉复选框 MultiSelectComboBox 自定义全选项】
继承ComboBox完成下拉复选框
自定义全选项
效果图


整个控件继承于QCombobox类。主要修改QLineEdit、QListWidget这两部分,QComboBox提供如下接口,可以将这两部分设置为新建的QLineEdit、QListWidget对象
CMultiSelectComboBox::CMultiSelectComboBox(QWidget* parent, bool bHasAllSelected): QComboBox(parent), m_bHasAllSelected(bHasAllSelected)
{m_pListWidget = new QListWidget();m_pLineEdit = new QLineEdit();m_pLineEdit->setReadOnly(true);setModel(m_pListWidget->model());setView(m_pListWidget);setLineEdit(m_pLineEdit);m_pLineEdit->installEventFilter(this);if (m_bHasAllSelected){AddItem(TransString2Unicode("全选"), ALL_SELECTED);}
}
完整代码
/*
** File name: MultiSelectComboBox.h
** Author:
** Date: 2024-12-26
** Brief: 多选下拉框控件
** Copyright (C) 1392019713@qq.com All rights reserved.
*/#pragma once#include <QComboBox>
#include <QListWidget>
#include <QLineEdit>
#include <QVariant>
#include <QEvent>
#include "QtGuiExportLib.h"class QTGUI_EXPORT CMultiSelectComboBox : public QComboBox
{
public:/** @brief 构造函数* @param parent 父控件* @param bHasAllSelected 是否显示全选选项*/explicit CMultiSelectComboBox(QWidget* parent = nullptr, bool bHasAllSelected = false);virtual ~CMultiSelectComboBox() = default;/** @brief 设置是否显示全选选项 需在设置数据前调用* @param bHasAllSelected 是否显示全选选项*/void SetHasAllSelected(bool bHasAllSelected = true);void AddItem(const QString& qstrItem, const QVariant& rVar = QVariant());void RemoveItem(const QVariant& rVar);void ClearItems();QList<QVariant> GetSelectedItemDatas() const;QString GetCurrentText() const;protected:virtual bool eventFilter(QObject* pObj, QEvent* pEvent) override;private:void SetSelectedCheckState(int nState);void SetAllSelected(bool bChecked);int GetCheckedCount() const;private Q_SLOTS:void SlotItemStateChanged(int);void SlotItemClicked(int nIndex);void SlotChecBoxClicked();private:bool m_bHasAllSelected;QListWidget* m_pListWidget;QLineEdit* m_pLineEdit;
};
#include "../Include/MultiSelectComboBox.h"
#include "../Include/Conversion.h"
#include <QCheckBox>
#include <QDebug>#define ALL_SELECTED "ALL_SELECT" CMultiSelectComboBox::CMultiSelectComboBox(QWidget* parent, bool bHasAllSelected): QComboBox(parent), m_bHasAllSelected(bHasAllSelected)
{m_pListWidget = new QListWidget();m_pLineEdit = new QLineEdit();m_pLineEdit->setReadOnly(true);setModel(m_pListWidget->model());setView(m_pListWidget);setLineEdit(m_pLineEdit);m_pLineEdit->installEventFilter(this);if (m_bHasAllSelected){AddItem(TransString2Unicode("全选"), ALL_SELECTED);}
}void CMultiSelectComboBox::SetHasAllSelected(bool bHasAllSelected)
{if (bHasAllSelected){AddItem(TransString2Unicode("全选"), ALL_SELECTED);}else{RemoveItem(ALL_SELECTED);}m_bHasAllSelected = bHasAllSelected;
}void CMultiSelectComboBox::AddItem(const QString& qstrItem, const QVariant& rVar)
{QListWidgetItem* pItem = new QListWidgetItem(m_pListWidget);QCheckBox* pCheckBox = new QCheckBox();pCheckBox->setText(qstrItem);pItem->setData(Qt::UserRole, rVar);m_pListWidget->addItem(pItem);m_pListWidget->setItemWidget(pItem, pCheckBox);connect(pCheckBox, &QCheckBox::stateChanged, this, &CMultiSelectComboBox::SlotItemStateChanged);connect(pCheckBox, &QCheckBox::clicked, this, &CMultiSelectComboBox::SlotChecBoxClicked);
}void CMultiSelectComboBox::RemoveItem(const QVariant& rVar)
{for (int i = 0; i < m_pListWidget->count(); i++){QListWidgetItem* pItem = m_pListWidget->item(i);if (!pItem){continue;}if (pItem->data(Qt::UserRole) == rVar){QWidget* pWidget = m_pListWidget->itemWidget(pItem);QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(pWidget);if (pCheckBox){delete pCheckBox;}m_pListWidget->removeItemWidget(pItem);m_pListWidget->takeItem(i);delete pItem;break;}}
}void CMultiSelectComboBox::ClearItems()
{m_pListWidget->clear();m_pLineEdit->clear();
}QList<QVariant> CMultiSelectComboBox::GetSelectedItemDatas() const
{QList<QVariant> lstVars;int i = 0;if (m_bHasAllSelected){i = 1;}for (; i < m_pListWidget->count(); i++){QListWidgetItem* pItem = m_pListWidget->item(i);if (!pItem){continue;}QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(m_pListWidget->itemWidget(pItem));if (!pCheckBox){continue;}if (pCheckBox->isChecked()){QVariant rVar = pItem->data(Qt::UserRole);if (rVar.isValid()){lstVars.append(rVar);}}}return lstVars;
}QString CMultiSelectComboBox::GetCurrentText() const
{if (!m_pLineEdit){return QString();}return m_pLineEdit->text();
}bool CMultiSelectComboBox::eventFilter(QObject* pObj, QEvent* pEvent)
{if (pObj == m_pLineEdit){if (pEvent->type() == QEvent::MouseButtonPress){showPopup();return true;}}return false;
}void CMultiSelectComboBox::SetSelectedCheckState(int nState)
{for (int i = 0; i < m_pListWidget->count(); i++){QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(m_pListWidget->itemWidget(m_pListWidget->item(i)));if (!pCheckBox){continue;}if (pCheckBox->isChecked()){pCheckBox->setCheckState(Qt::CheckState(nState));}}
}void CMultiSelectComboBox::SetAllSelected(bool bChecked)
{for (int i = 0; i < m_pListWidget->count(); i++){QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(m_pListWidget->itemWidget(m_pListWidget->item(i)));if (!pCheckBox){continue;}pCheckBox->setChecked(bChecked);}
}int CMultiSelectComboBox::GetCheckedCount() const
{int i = 0;if (m_bHasAllSelected){i = 1;}int nCount = 0;for (; i < m_pListWidget->count(); i++){QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(m_pListWidget->itemWidget(m_pListWidget->item(i)));if (!pCheckBox){continue;}if (pCheckBox->isChecked()){nCount++;}}return nCount;
}void CMultiSelectComboBox::SlotItemClicked(int nIndex)
{QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(m_pListWidget->itemWidget(m_pListWidget->item(nIndex)));if (!pCheckBox){return;}if (m_bHasAllSelected && nIndex == 0){SetAllSelected(pCheckBox->isChecked());}else if(!m_bHasAllSelected){pCheckBox->setChecked(pCheckBox->isChecked());}else {pCheckBox->setChecked(pCheckBox->isChecked());int nCheckedCount = GetCheckedCount();if (nCheckedCount == 0){SetAllSelected(false);}else if (nCheckedCount < m_pListWidget->count() - 1){SetSelectedCheckState(Qt::PartiallyChecked);}else{SetAllSelected(true);}}
}void CMultiSelectComboBox::SlotChecBoxClicked()
{int nIndex = 0;QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(sender());if (!pCheckBox){return;}for (int i = 0; i < m_pListWidget->count(); i++){if (m_pListWidget->itemWidget(m_pListWidget->item(i)) == pCheckBox){nIndex = i;break;}}SlotItemClicked(nIndex);
}void CMultiSelectComboBox::SlotItemStateChanged(int nState)
{QString qstrText;int i = 0;if (m_bHasAllSelected){i = 1;}for (; i < m_pListWidget->count(); i++){QCheckBox* pCheckBox = qobject_cast<QCheckBox*>(m_pListWidget->itemWidget(m_pListWidget->item(i)));if(!pCheckBox){continue;}if (pCheckBox->isChecked()){qstrText += pCheckBox->text() + TransString2Unicode("、");}}qstrText.chop(1);m_pLineEdit->setText(qstrText);m_pLineEdit->setToolTip(qstrText);
}
使用示例 可以提升控件也可以直接new
提升

ui->comboBox->SetHasAllSelected(true);ui->comboBox->AddItem("Sine Wave", 1);ui->comboBox->AddItem("Random Data", 2);ui->comboBox->AddItem("Custom Data", 3);ui->comboBox->AddItem("Custom Data 2", 4);ui->comboBox->RemoveItem(1);
如果此文帮助到你,动动小手点个赞可好
。
相关文章:
Qt开发技术【下拉复选框 MultiSelectComboBox 自定义全选项】
继承ComboBox完成下拉复选框 自定义全选项 效果图 整个控件继承于QCombobox类。主要修改QLineEdit、QListWidget这两部分,QComboBox提供如下接口,可以将这两部分设置为新建的QLineEdit、QListWidget对象 CMultiSelectComboBox::CMultiSelectComboBo…...
20_HTML5 SSE --[HTML5 API 学习之旅]
HTML5 Server-Sent Events (SSE) 是一种技术,它允许服务器向浏览器推送更新。与传统的轮询不同,SSE提供了真正的单向实时通信通道:服务器可以主动发送数据到客户端,而不需要客户端发起请求。这对于实现实时更新的应用非常有用&…...
jetson Orin nx + yolov8 TensorRT 加速量化 环境配置
参考【Jetson】Jetson Orin NX纯系统配置环境-CSDN博客 一 系统环境配置: 1.更换源: sudo vi /etc/apt/sources.list.d/nvidia-l4t-apt-source.list2.更新源: sudo apt upgradesudo apt updatesudo apt dist-upgrade sudo apt-get updat…...
Android Studio IDE环境配置
需要安装哪些东西: Java jdk Java Downloads | OracleAndroid Studio 下载 Android Studio 和应用工具 - Android 开发者 | Android DevelopersAndroid Sdk 现在的Android Studio版本安装时会自动安装,需要注意下安装的路径Android Studio插件…...
PTA 7-2 0/1背包问题(回溯法) 作者 王东 单位 贵州师范学院
0/1背包问题。给定一载重量为W的背包及n个重量为wi、价值为vi的物体,1≤i≤n,要求重量和恰好为W具有最大的价值。 输入格式: 第一行输入背包载重量W及背包个数n,再依次输入n行,每行为背包重量wi和价值vi。 输出格式: 第一行输出装入背包内…...
Matlab环形柱状图
数据准备: 名称 数值 Aa 21 Bb 23 Cc 35 Dd 47 保存为Excel文件后: % Load data from Excel file filename data.xlsx; % Ensure the file is in the current folder or provide full path dataTable readtable(filena…...
【AI大模型】探索GPT模型的奥秘:引领自然语言处理的新纪元
目录 🍔 GPT介绍 🍔 GPT的架构 🍔 GPT训练过程 3.1 无监督的预训练语言模型 3.2 有监督的下游任务fine-tunning 🍔 小结 学习目标 了解什么是GPT.掌握GPT的架构.掌握GPT的预训练任务. 🍔 GPT介绍 GPT是OpenAI公…...
5.Python爬虫相关
爬虫 爬虫原理 爬虫,又称网络爬虫,是一种自动获取网页内容的程序。它模拟人类浏览网页的行为,发送HTTP请求,获取网页源代码,再通过解析、提取等技术手段,获取所需数据。 HTTP请求与响应过程 爬虫首先向…...
Windows系统上配置eNSP环境的详细步骤
华为eNSP(Enterprise Network Simulation Platform)是一款针对华为数通网络设备的网络仿真平台,用于辅助工程师进行网络技术学习、方案验证和故障排查等工作。以下是在Windows系统上配置eNSP环境的详细步骤: 1. 准备工作 下载安…...
Database.NET——一款轻量级多数据库客户端工具
文章目录 Database.NET简介下载使用使用场景总结 Database.NET简介 Database.NET 是一个功能强大且易于使用的数据库管理工具,适用于多种数据库系统。它为开发者和数据库管理员提供了一个统一的界面,可以方便地管理和操作不同类型的数据库。 支持的数据…...
新浪微博C++面试题及参考答案
多态是什么?请详细解释其实现原理,例如通过虚函数表实现。 多态是面向对象编程中的一个重要概念,它允许不同的对象对同一消息或函数调用做出不同的响应,使得程序具有更好的可扩展性和灵活性。 在 C 中,多态主要通过虚函…...
计算机视觉目标检测-1
文章目录 摘要Abstract1.目标检测任务描述1.1 目标检测分类算法1.2 目标定位的简单实现思路1.2.1 回归位置 2.R-CNN2.1 目标检测-Overfeat模型2.1.1 滑动窗口 2.2 目标检测-RCNN模型2.2.1 非极大抑制(NMS) 2.3 目标检测评价指标 3.SPPNet3.1 spatial pyr…...
【物联网技术与应用】实验15:电位器传感器实验
实验15 电位器传感器实验 【实验介绍】 电位器可以帮助控制Arduino板上的LED闪烁的时间间隔。 【实验组件】 ● Arduino Uno主板* 1 ● 电位器模块* 1 ● USB电缆*1 ● 面包板* 1 ● 9V方型电池* 1 ● 跳线若干 【实验原理】 模拟电位器是模拟电子元件,模…...
java常用类(上)
笔上得来终觉浅,绝知此事要躬行 🔥 个人主页:星云爱编程 🔥 所属专栏:javase 🌷追光的人,终会万丈光芒 🎉欢迎大家点赞👍评论📝收藏⭐文章 目录 一、包装类 1.1包装类…...
包管理工具npm、yarn、pnpm、cnpm详解
1. 包管理工具 1.1 npm # 安装 $ node 自带 npm# 基本用法 npm install package # 安装包 npm install # 安装所有依赖 npm install -g package # 全局安装 npm uninstall package # 卸载包 npm update package # 更新包 npm run script #…...
CI/CD是什么?
CI/CD 定义 CI/CD 代表持续集成和持续部署(或持续交付)。它是一套实践和工具,旨在通过自动化构建、测试和部署来改进软件开发流程,使您能够更快、更可靠地交付代码更改。 持续集成 (CI):在共享存储库中自动构建、测试…...
[Java]合理封装第三方工具包(附视频)
-1.视频链接 视频版: 视频版会对本文章内容进行详细解释 [Java]合理封装第三方工具包_哔哩哔哩_bilibili 0.核心思想 对第三方工具方法进行封装,使其本地化,降低记忆和使用成本 1.背景 在我们的项目中,通常会引用一些第三方工具包,或者是使用jdk自带的一些工具类 例如: c…...
常规配置、整合IDEA
目录 Redis常规配置 tcp-keepalive security Jedis RedisTemplate 连接池技术 Lua脚本 Jedis集群 Redis应用问题&解决方案 缓存穿透 缓存击穿 缓存雪崩 分布式锁 Redis实现分布式锁 Redis新功能 ACL Redis常规配置 tcp-keepalive security redis.conf中…...
用Python写炸金花游戏
文章目录 **代码分解与讲解**1. **扑克牌的生成与洗牌**2. **给玩家发牌**3. **打印玩家的手牌**4. **定义牌的优先级**5. **判断牌型**6. **确定牌型优先级**7. **比较两手牌的大小**8. **计算每个玩家的牌型并找出赢家**9. **打印结果** 完整代码 以下游戏规则: 那…...
计算机的错误计算(一百九十二)
摘要 用两个大模型计算 csc(0.999), 其中,0.999是以弧度为单位的角度,结果保留5位有效数字。两个大模型均给出了 Python代码与答案。但是,答案是错误的。 例1. 计算 csc(0.999), 其中,0.999是以弧度为单位的角度,结…...
通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...
【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...
第25节 Node.js 断言测试
Node.js的assert模块主要用于编写程序的单元测试时使用,通过断言可以提早发现和排查出错误。 稳定性: 5 - 锁定 这个模块可用于应用的单元测试,通过 require(assert) 可以使用这个模块。 assert.fail(actual, expected, message, operator) 使用参数…...
聊一聊接口测试的意义有哪些?
目录 一、隔离性 & 早期测试 二、保障系统集成质量 三、验证业务逻辑的核心层 四、提升测试效率与覆盖度 五、系统稳定性的守护者 六、驱动团队协作与契约管理 七、性能与扩展性的前置评估 八、持续交付的核心支撑 接口测试的意义可以从四个维度展开,首…...
关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...
高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数
高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...
Java + Spring Boot + Mybatis 实现批量插入
在 Java 中使用 Spring Boot 和 MyBatis 实现批量插入可以通过以下步骤完成。这里提供两种常用方法:使用 MyBatis 的 <foreach> 标签和批处理模式(ExecutorType.BATCH)。 方法一:使用 XML 的 <foreach> 标签ÿ…...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...
【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...
免费数学几何作图web平台
光锐软件免费数学工具,maths,数学制图,数学作图,几何作图,几何,AR开发,AR教育,增强现实,软件公司,XR,MR,VR,虚拟仿真,虚拟现实,混合现实,教育科技产品,职业模拟培训,高保真VR场景,结构互动课件,元宇宙http://xaglare.c…...
