当前位置: 首页 > article >正文

Qt实战:如何用QTableView+自定义Delegate打造可编辑表格(附SpinBox/ComboBox完整代码)

Qt实战用QTableView自定义Delegate实现高级表格编辑1. 理解Qt模型-视图-委托架构在Qt框架中模型-视图-委托(Model-View-Delegate, MVD)模式是实现数据与界面分离的核心设计。这种架构将数据管理、界面展示和用户交互三个关注点清晰地划分开来模型(Model)负责数据的存储和底层逻辑不关心数据如何显示视图(View)负责数据的可视化呈现不关心数据的具体内容委托(Delegate)负责数据的编辑方式和显示细节充当模型与视图之间的桥梁这种架构的优势在于同一数据模型可以同时用于多个不同视图视图可以自由切换不同的数据模型委托可以针对特定数据类型提供最优的编辑体验// 基本MVD组件初始化示例 QTableView *tableView new QTableView; QStandardItemModel *model new QStandardItemModel; tableView-setModel(model);2. 自定义委托的核心实现2.1 委托类的基本结构自定义委托需要继承自QStyledItemDelegate推荐或QItemDelegate并实现四个关键方法class CustomDelegate : public QStyledItemDelegate { public: // 创建编辑器控件 QWidget* createEditor(QWidget *parent, const QStyleOptionViewItem option, const QModelIndex index) const override; // 将模型数据设置到编辑器 void setEditorData(QWidget *editor, const QModelIndex index) const override; // 将编辑器数据保存回模型 void setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex index) const override; // 调整编辑器位置和大小 void updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem option, const QModelIndex index) const override; };2.2 SpinBox委托实现数值输入是表格编辑的常见需求以下是完整的SpinBox委托实现// SpinBoxDelegate.h class SpinBoxDelegate : public QStyledItemDelegate { Q_OBJECT public: explicit SpinBoxDelegate(QObject *parent nullptr); void setRange(int min, int max); void setStep(int step); protected: QWidget* createEditor(QWidget *parent, const QStyleOptionViewItem option, const QModelIndex index) const override; // 其他必要方法实现... private: int m_min 0; int m_max 100; int m_step 1; }; // SpinBoxDelegate.cpp QWidget* SpinBoxDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem /*option*/, const QModelIndex /*index*/) const { QSpinBox *editor new QSpinBox(parent); editor-setFrame(false); editor-setMinimum(m_min); editor-setMaximum(m_max); editor-setSingleStep(m_step); return editor; } void SpinBoxDelegate::setEditorData(QWidget *editor, const QModelIndex index) const { int value index.model()-data(index, Qt::EditRole).toInt(); QSpinBox *spinBox static_castQSpinBox*(editor); spinBox-setValue(value); } void SpinBoxDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex index) const { QSpinBox *spinBox static_castQSpinBox*(editor); spinBox-interpretText(); int value spinBox-value(); model-setData(index, value, Qt::EditRole); } void SpinBoxDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem option, const QModelIndex /*index*/) const { editor-setGeometry(option.rect); }2.3 ComboBox委托实现对于枚举值或有限选项的数据ComboBox是更合适的编辑控件// ComboBoxDelegate.h class ComboBoxDelegate : public QStyledItemDelegate { Q_OBJECT public: explicit ComboBoxDelegate(const QStringList items, QObject *parent nullptr); protected: QWidget* createEditor(QWidget *parent, const QStyleOptionViewItem option, const QModelIndex index) const override; // 其他必要方法实现... private: QStringList m_items; }; // ComboBoxDelegate.cpp QWidget* ComboBoxDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem /*option*/, const QModelIndex /*index*/) const { QComboBox *editor new QComboBox(parent); editor-setFrame(false); editor-addItems(m_items); return editor; } void ComboBoxDelegate::setEditorData(QWidget *editor, const QModelIndex index) const { QString value index.model()-data(index, Qt::EditRole).toString(); QComboBox *comboBox static_castQComboBox*(editor); int idx comboBox-findText(value); if (idx 0) comboBox-setCurrentIndex(idx); } void ComboBoxDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex index) const { QComboBox *comboBox static_castQComboBox*(editor); QString value comboBox-currentText(); model-setData(index, value, Qt::EditRole); }3. 模型与委托的协同工作3.1 模型的可编辑性配置要使委托正常工作模型必须明确标识哪些单元格是可编辑的。这通过重写flags()方法实现Qt::ItemFlags MyModel::flags(const QModelIndex index) const { if (!index.isValid()) return Qt::NoItemFlags; // 假设第1列可编辑其他列只读 if (index.column() 1) { return QAbstractTableModel::flags(index) | Qt::ItemIsEditable; } return QAbstractTableModel::flags(index); }3.2 数据验证与同步在模型端实现setData()方法时应该包含数据验证逻辑bool MyModel::setData(const QModelIndex index, const QVariant value, int role) { if (!index.isValid() || role ! Qt::EditRole) return false; // 验证数值范围 if (index.column() 1) { // 数值列 bool ok; int intValue value.toInt(ok); if (!ok || intValue 0 || intValue 100) return false; } // 更新数据并通知视图 m_data[index.row()][index.column()] value; emit dataChanged(index, index, {role}); return true; }4. 高级应用技巧4.1 条件式委托分配可以根据不同行或列的数据特征动态分配不同的委托// 根据列分配不同委托 tableView-setItemDelegateForColumn(0, new SpinBoxDelegate(this)); tableView-setItemDelegateForColumn(1, new ComboBoxDelegate({选项1,选项2}, this)); // 根据行数据动态选择委托 class DynamicDelegate : public QStyledItemDelegate { QWidget* createEditor(...) const override { if (index.data(Qt::UserRole1).toBool()) { return new SpinBoxDelegate(parent); } else { return new ComboBoxDelegate({A,B,C}, parent); } } };4.2 自定义编辑器样式委托可以完全控制编辑器的外观和行为QWidget* FancyDelegate::createEditor(...) const { QLineEdit *editor new QLineEdit(parent); editor-setStyleSheet(QLineEdit { background: #f0f0f0; }); editor-setFrame(false); // 添加输入验证器 QRegExpValidator *validator new QRegExpValidator(QRegExp([A-Za-z0-9_]), editor); editor-setValidator(validator); return editor; }4.3 编辑状态持久化通过重写editorEvent()方法可以自定义编辑行为的触发方式bool CustomDelegate::editorEvent(QEvent *event, QAbstractItemModel *model, const QStyleOptionViewItem option, const QModelIndex index) { if (event-type() QEvent::MouseButtonDblClick) { // 自定义双击行为 return true; } return QStyledItemDelegate::editorEvent(event, model, option, index); }5. 性能优化与调试5.1 委托创建的性能考虑频繁创建和销毁编辑器控件会影响性能可以考虑重用编辑器实例延迟加载复杂控件对大量数据使用轻量级委托// 在委托类中添加编辑器缓存 mutable QHashQModelIndex, QWidget* m_editorCache; QWidget* CachedDelegate::createEditor(...) const { if (m_editorCache.contains(index)) { return m_editorCache.value(index); } QWidget *editor createNewEditor(parent, option, index); m_editorCache.insert(index, editor); return editor; }5.2 常见问题排查问题现象可能原因解决方案点击单元格无反应模型未设置可编辑标志检查flags()方法实现编辑器显示但数据不保存setModelData未正确实现验证数据转换和模型更新逻辑编辑器位置偏移updateEditorGeometry未正确实现确保使用option.rect设置几何编辑器样式不一致未考虑系统样式使用QStyle绘制而非硬编码样式5.3 调试技巧使用qDebug()输出委托生命周期事件qDebug() Creating editor for row index.row() col index.column();检查模型索引有效性Q_ASSERT(index.isValid());验证数据角色if (role ! Qt::EditRole) { qWarning() Unexpected role: role; return QVariant(); }

相关文章:

Qt实战:如何用QTableView+自定义Delegate打造可编辑表格(附SpinBox/ComboBox完整代码)

Qt实战:用QTableView自定义Delegate实现高级表格编辑 1. 理解Qt模型-视图-委托架构 在Qt框架中,模型-视图-委托(Model-View-Delegate, MVD)模式是实现数据与界面分离的核心设计。这种架构将数据管理、界面展示和用户交互三个关注点清晰地划分开来&#x…...

GitHub万星推荐:10本让程序员代码质量飙升的实战书籍(含PDF资源获取方式)

GitHub万星推荐:10本让程序员代码质量飙升的实战书籍(含PDF资源获取方式) 在代码的世界里,质量从来不是偶然。当我第一次被团队负责人指出代码中的"坏味道"时,才真正意识到写出可维护的代码远比实现功能困难…...

如何实现十微秒级IP定位?详解ip2region的本地化解决方案

如何实现十微秒级IP定位?详解ip2region的本地化解决方案 【免费下载链接】ip2region Ip2region (2.0 - xdb) 是一个离线IP地址管理与定位框架,能够支持数十亿级别的数据段,并实现十微秒级的搜索性能。它为多种编程语言提供了xdb引擎实现。 …...

小说下载开源工具fanqienovel-downloader:构建你的离线阅读库

小说下载开源工具fanqienovel-downloader:构建你的离线阅读库 【免费下载链接】fanqienovel-downloader 下载番茄小说 项目地址: https://gitcode.com/gh_mirrors/fa/fanqienovel-downloader 在数字阅读日益普及的今天,网络波动、流量限制和平台访…...

DAMOYOLO-S用于内网穿透测试:在无公网IP服务器部署与访问

DAMOYOLO-S用于内网穿透测试:在无公网IP服务器部署与访问 1. 引言 很多朋友在本地服务器上跑通了DAMOYOLO-S模型,想给同事或者客户演示一下效果,结果发现服务器在公司内网或者家里,没有公网IP,外面根本访问不了。这时…...

PyTorch3D 0.7.2 + CUDA 11.7完整安装流程:手把手教你配置3D深度学习环境

PyTorch3D 0.7.2 CUDA 11.7完整安装流程:手把手教你配置3D深度学习环境 3D深度学习正在重塑计算机视觉和图形学的边界,而PyTorch3D作为Facebook Research开源的3D深度学习库,为开发者提供了强大的工具集。本文将带你从零开始,在W…...

CTF流量分析终极指南:3步完成复杂网络数据包解析

CTF流量分析终极指南:3步完成复杂网络数据包解析 【免费下载链接】CTF-NetA 项目地址: https://gitcode.com/gh_mirrors/ct/CTF-NetA CTF流量分析是网络安全竞赛中最具挑战性的环节之一,传统工具往往让新手望而却步。CTF-NetA作为一款专业的网络…...

FPGA仿真数据跨平台分析:从Vivado到MATLAB的无缝衔接

1. FPGA仿真数据分析的痛点与解决方案 做FPGA开发的朋友们应该都深有体会,仿真验证和数据分析就像一对难兄难弟。在Vivado里跑完仿真,看着那一堆波形图,总觉得少了点什么——没错,就是深度分析的能力。Vivado的波形查看器虽然直观…...

Android12 编译环境搭建避坑指南:从配置优化到实战调试

1. 硬件配置:别让机器性能成为你的绊脚石 第一次编译Android 12的经历让我记忆犹新——连续8小时的等待,最终却以内存溢出告终。这种痛苦我懂,所以先来聊聊硬件配置这个最基础也最容易踩坑的环节。 Android官方文档给出的最低配置要求就像汽车…...

Java开发必看:VO、DTO、DO、BO到底怎么用?5个真实项目案例解析

Java开发必看:VO、DTO、DO、BO到底怎么用?5个真实项目案例解析 在Java企业级开发中,对象模型的设计直接影响着代码的可维护性和扩展性。每当看到代码中充斥着各种以VO、DTO、DO、BO结尾的类名时,不少开发者都会产生这样的困惑&…...

别再让慢查询拖垮你的报表了!用StarRocks物化视图5分钟搞定实时聚合加速

5分钟极速优化:用StarRocks物化视图拯救你的慢查询报表 凌晨三点的办公室,咖啡杯已经见底,而你的聚合查询还在转圈——这是多少数据工程师的噩梦。当千万级交易数据遇上实时决策需求,传统方法往往力不从心。今天我们要解锁的StarR…...

避坑指南:YOLOv5多摄像头实时检测中的常见问题与解决方案

YOLOv5多摄像头实时检测实战:从性能瓶颈到工业级优化 在智能安防、工业质检和无人零售等场景中,多摄像头实时目标检测已成为刚需。YOLOv5凭借其优异的精度和速度表现,成为众多开发者的首选框架。但当我们将单摄像头检测扩展到多路视频流处理时…...

【风控】PSI指标实战:从原理到业务落地的全流程解析

1. PSI指标:风控模型的"体温计" 当你去医院体检时,医生总会先给你量体温——这个简单数值能快速判断身体是否异常。在金融风控领域,PSI(Population Stability Index)就是这样一个"体温计"&#xf…...

2026指纹浏览器性能瓶颈分析与优化技巧

在 2026 年的多账号运营、跨境业务、数据采集等高频使用场景中,指纹浏览器的性能表现直接决定业务效率与稳定性。随着环境数量的增加、业务复杂度的提升,很多从业者都会遇到指纹浏览器卡顿、崩溃、启动缓慢、资源占用过高、网络延迟等性能瓶颈&#xff0…...

Snyk CLI:现代化依赖安全扫描与漏洞治理解决方案

Snyk CLI:现代化依赖安全扫描与漏洞治理解决方案 【免费下载链接】cli Snyk CLI scans and monitors your projects for security vulnerabilities. 项目地址: https://gitcode.com/gh_mirrors/cli6/cli 在当今快速迭代的软件开发环境中,依赖安全…...

大模型高效微调技术集合:从BitFit到LoRA,大模型微调不再难!

模型是什么? 简单来说,大语言模型是一种在海量文本数据上进行预训练的深度学习模型。由于其巨大的参数规模(通常在数十亿到数千亿之间)和广泛的训练数据,它具备了强大的自然语言理解和生成能力,能够执行问答…...

基于51单片机与Proteus的LCD交互式电话拨号仿真系统设计

1. 系统设计概述 这个项目本质上是在模拟传统电话机的核心交互功能。想象一下老式座机电话的拨号体验:按下物理按键时会有"滴"声反馈,LCD屏幕实时显示输入号码,还能通过功能键进行修改。我们使用51单片机作为"大脑"&…...

国家平台首发!全球首个机器人“灵巧操作+全身运动”真机数据集来了

点击下方卡片,关注“CVer”公众号AI/CV重磅干货,第一时间送达具身智能太火了!连续两年写入政府工作报告!今年两会上关于具身智能的提案也迎来爆发性增长。过去几年,人形机器人已经学会了很多“炫技”:跑步、…...

实战准备:为短视频安卓应用开发,用快马ai定制专属的android studio环境配置方案

作为一名安卓开发者,我深知项目启动前的环境配置有多重要。尤其是面对像“短视频播放应用”这样对性能、兼容性有特定要求的项目,一个“开箱即用”的优化环境能省去大量前期摸索和踩坑的时间。最近在准备一个类似项目时,我没有像往常一样手动…...

YOLOv12在复杂数据结构处理中的优化:以图数据表示目标关联关系

YOLOv12在复杂数据结构处理中的优化:以图数据表示目标关联关系 1. 引言 想象一下,你正在看一张照片:一个人手里拿着手机,站在一辆汽车旁边。传统的目标检测模型,比如我们熟悉的YOLO系列,能准确地告诉你“…...

CLUE模型构建方法、模型验证及土地利用变化情景预测实践技术应用

土地利用/土地覆盖数据是生态、环境和气象等领域众多模型的重要输入参数之一。基于遥感影像解译,可获取历史或当前任何一个区域的土地利用/土地覆盖数据,用于评估区域的生态环境变化、评价重大生态工程建设成效等。借助CLUE模型,实现对未来土…...

采样率自适应:SenseVoice-Small ONNX模型8k/16k/48k音频统一处理

采样率自适应:SenseVoice-Small ONNX模型8k/16k/48k音频统一处理 1. 引言:告别音频格式烦恼,一个模型通吃所有 你有没有遇到过这样的麻烦事?手头有一堆音频文件,有的是电话录音(8kHz)&#xf…...

LaTeX新手必看:VSCode中高效编辑与排版的5个实用技巧

LaTeX高效工作流:VSCode中提升排版效率的5个专业技巧 当你第一次在VSCode中成功编译出"Hello, world!"的LaTeX文档时,那种成就感可能很快就会被重复性操作和调试错误所消磨。作为一位每天与LaTeX打交道的学术工作者,我发现大多数新…...

ANSYS Fluent多相流避坑指南:Mixture和Eulerian模型选择的3个关键参数

ANSYS Fluent多相流模型深度解析:从参数阈值到实战优化 在工程仿真领域,多相流模拟一直是个令人又爱又恨的难题。当面对气泡柱、流化床或泥浆输送等复杂场景时,如何选择合适的模型往往成为项目成败的关键。本文将聚焦ANSYS Fluent中最易混淆的…...

VScode 集成Clangd打造智能Linux内核代码阅读环境

1. 为什么选择Clangd替代Global? 十年前我第一次接触Linux内核代码时,被庞大的代码量吓到了。当时用Global做索引,跳转功能时灵时不灵,经常跳到错误的定义位置。后来发现Clangd这个基于LLVM的Language Server,简直是代…...

如何在E-HPC集群上使用oneAPI加速LAMMPS编译?完整配置指南

基于oneAPI的LAMMPS高性能编译与优化实战指南 1. 高性能计算环境下的分子动力学模拟挑战 分子动力学模拟作为计算化学和材料科学的核心工具,其计算效率直接决定了科研项目的可行性和产出速度。LAMMPS(Large-scale Atomic/Molecular Massively Parallel Simulator)作…...

轻量级编辑器Notepad--全面测评:解决中文编码痛点与提升编码效率的最佳实践

轻量级编辑器Notepad--全面测评:解决中文编码痛点与提升编码效率的最佳实践 【免费下载链接】notepad-- 一个支持windows/linux/mac的文本编辑器,目标是做中国人自己的编辑器,来自中国。 项目地址: https://gitcode.com/GitHub_Trending/no…...

nlp_structbert_sentence-similarity_chinese-large 在AIGC内容审核中的应用:识别生成文本的语义重复

nlp_structbert_sentence-similarity_chinese-large 在AIGC内容审核中的应用:识别生成文本的语义重复 不知道你有没有发现,现在网上很多文章、营销文案,读起来感觉都差不多?开头一个吸引眼球的标题,中间几段看似有道理…...

DeepSeek-Coder-V2:开源代码智能模型的部署与应用实践

DeepSeek-Coder-V2:开源代码智能模型的部署与应用实践 【免费下载链接】DeepSeek-Coder-V2 项目地址: https://gitcode.com/GitHub_Trending/de/DeepSeek-Coder-V2 在当今软件开发领域,AI辅助编程已成为提升开发效率的关键技术。DeepSeek-Coder-…...

MATLAB小白也能画BODE图:手把手教你用sym2poly搞定复杂传递函数

MATLAB小白也能画BODE图:手把手教你用sym2poly搞定复杂传递函数 在控制系统分析与设计中,Bode图是工程师最常用的频率响应分析工具之一。它能直观展示系统在不同频率下的增益和相位特性,为稳定性分析和控制器设计提供重要依据。然而对于MATLA…...