一、前言

自从上次做完的图文报表,又新来了个需求需要实现个人信息报表,类似个人简历一样的格式,数据从数据库中取出来,然后一个人的信息就打印一张,传入查询的多个人员信息,自动分页打印个人信息报表,报表可以打印也可以导出到pdf文件等。根据之前的经验,这种需求依然是采用html方式去实现最方便最简单,要不然还要自己用painter绘制网格线和对应内容,不好把控,既然painter本身就支持富文本QTextDocument,真是一个天大的方便。核心就是用html中的表格table把内容组织好,个人建议只要是有行列要求的数据通通用table来组织会非常方便,包括控制边框粗细,内容的对齐方式,甚至直接img标签嵌入图片,图片可以是资源文件和本地文件。

Qt中的painter绘制非常灵活强大,接口丰富,但是对于很多初学者来说还是有一定的难度,尤其是各种奇奇怪怪的复杂格式,而这些格式用html确很好描述,比如控制行间距、字符间距等,此时可以用QTextDocument传入html格式内容交给QPainter绘制,非常完美、简单、强大,包括一些数学公式啥的。

void Form::paintEvent(QPaintEvent *event)
{
QPainter painter(this);
QTextDocument doc;
doc.setHtml(html);
//设置文本宽度
doc.setTextWidth(200);
//指定绘制区域
doc.drawContents(&painter, QRect(0, 0, 200, 70));
}

二、功能特点

  1. 组件同时集成了导出数据到csv、xls、pdf和打印数据。
  2. 所有操作全部提供静态方法无需new,数据和属性等各种参数设置采用结构体数据,极为方便。
  3. 同时支持QTableView、QTableWidget、QStandardItemModel、QSqlTableModel等数据源。
  4. 提供静态方法直接传入QTableView、QTableWidget控件,自动识别列名、列宽和数据内容。
  5. 每组功能都提供单独的完整的示例,注释详细,非常适合各阶段Qter程序员。
  6. 原创导出数据机制,不依赖任何office组件或者操作系统等第三方库,支持嵌入式linux。
  7. 速度超快,9个字段10万行数据只需要2秒钟完成。
  8. 只需要四个步骤即可开始急速导出海量数据比如100W条记录到Excel。
  9. 同时提供直接写入数据接口和多线程写入数据接口,不卡主界面。
  10. 可设置标题、副标题、表名。
  11. 可设置导出数据的字段名、列名、列宽。
  12. 可设置末尾列自动拉伸填充,默认拉伸更美观。
  13. 可设置是否启用校验过滤数据,启用后符合规则的数据特殊颜色显示。
  14. 可指定校验的列、校验规则、校验值、校验值数据类型。
  15. 校验规则支持 精确等于==、大于>、大于等于>=、小于<、小于等于<=、不等于!=、包含contains。
  16. 校验值数据类型支持 整型int、浮点型float、双精度型double,默认文本字符串类型。
  17. 可设置随机背景颜色及需要随机背景色的列集合。
  18. 支持分组输出数据,比如按照设备分组输出数据,方便查看。
  19. 可设置csv分隔符、行内容分隔符、子内容分隔符。
  20. 可设置边框宽度、自动填数据类型,默认自动数据类型开启。
  21. 可设置是否开启数据单元格样式,默认不开启,不开启可以节约大概30%的文件体积。
  22. 可设置横向排版、纸张边距等,比如导出到pdf以及打印数据。
  23. 提供图文混排导出数据到pdf以及打印示例,自动分页,支持多图。
  24. 提供一个打印样板中同时包括横向纵向排版示例。
  25. 提供静态函数将控件截图导出到pdf文件。
  26. 提供静态函数将图片转成pdf文件。
  27. 提供静态函数将csv文件转成xls文件,支持列宽表名等参数设置。
  28. 针对每列可分别设置字段对齐样式、内容对齐样式,包括左对齐、居中对齐、右对齐。
  29. 灵活性超高,可自由更改源码设置对齐方式、文字颜色、背景颜色等。
  30. 支持任意excel表格软件,包括但不限于excel2003-2021、wps、openoffice等。
  31. 纯Qt编写,支持任意Qt版本+任意编译器+任意系统。

三、体验地址

  1. 体验地址:https://pan.baidu.com/s/1ZxG-oyUKe286LPMPxOrO2A 提取码:o05q 文件名:bin_dataout.zip
  2. 国内站点:https://gitee.com/feiyangqingyun
  3. 国际站点:https://github.com/feiyangqingyun
  4. 个人主页:https://blog.csdn.net/feiyangqingyun
  5. 知乎主页:https://www.zhihu.com/people/feiyangqingyun/

四、效果图

五、相关代码

void DataReport::creatPersonReportHead(QStringList &list, const PersonReportData &reportData)
{
//表格开始
list << "<table border='0.0' cellspacing='0' cellpadding='6' style='font-size:15px;'>"; //标题
list << "<tr>";
list << QString("<td width='100%' align='center' style='font-size:25px;font-weight:bold;' colspan='%1'>%2</td>").arg(2).arg(reportData.title);
list << "</tr>"; //换行
list << "<br>"; //报表日期
list << "<tr>";
//可以自行修改左对齐右对齐,不要的内容为空就行
list << QString("<td align='left'>%1</td>").arg(QString("报表日期: %1").arg(reportData.time));
list << QString("<td align='right'>%1</td>").arg("");
list << "</tr>"; //表格结束
list << "</table>";
} void DataReport::creatPersonReportBody(QStringList &list, const PersonReportData &reportData, const QStringList &columnValue, int brCount)
{
//如果字段数量不够或者值数量不一致则不用处理
QStringList columnName = reportData.columnName;
int count = columnName.count() - 1;
if (count < 5 || count > columnValue.count()) {
return;
} //表格开始
list << "<table border='0.5' cellspacing='0' cellpadding='5' style='font-size:16px;'>"; //5行3列
list << "<tr style='vertical-align:middle;'>";
list << QString("<td width='%1' align='right'>%2</td>").arg(reportData.columnWidth1).arg(columnName.at(0));
list << QString("<td width='%1' align='left'>%2</td>").arg(reportData.columnWidth2).arg(columnValue.at(0)); //末尾数据是图片路径,将图片放到临时目录并缩放 QString image = QString("<img src='%1'>").arg(columnValue.last()); //头像占5行
list << QString("<td width='%1' align='center' rowspan='5'>%2</td>").arg(reportData.columnWidth3).arg(image);
list << "</tr>"; //头像左侧其他行
for (int i = 1; i <= 4; ++i) {
list << "<tr>";
list << QString("<td align='right'>%1</td>").arg(columnName.at(i));
list << QString("<td align='left'>%1</td>").arg(columnValue.at(i));
list << "</tr>";
} //没有头像的行,第二行占2列
for (int i = 5; i < count; ++i) {
list << "<tr>";
list << QString("<td align='right'>%1</td>").arg(columnName.at(i));
list << QString("<td align='left' colspan='2'>%1</td>").arg(columnValue.at(i));
list << "</tr>";
} //表格结束
list << "</table>"; //换行填充空行构成一页
for (int i = 0; i < brCount; ++i) {
list << "<br>";
}
} void frmDataReport::on_btnPersonReport_clicked()
{
//数据结构体
DataContent dataContent;
dataContent.pageMargin = QMargins(15, 20, 15, 20);
DataPrint::dataPrint->init();
DataPrint::dataPrint->setDataContent(dataContent); //还有很多参数可以设置,每次要变动的就是这个报表的数据
PersonReportData reportData;
reportData.title = "矿工个人信息";
//reportData.time = "xxxxxxxxx"; //拿到报表的html内容
QStringList list;
for (int i = 0; i < 15; ++i) {
//字段名称的数量和字段值的数量要一致
QStringList columnValue;
QString id = QString("#10%1").arg(i + 1, 3, 10, QChar('0'));
columnValue << id << "张美丽" << "女" << "中共党员" << "大学本科";
columnValue << "1988-02-10" << "已婚" << "正常" << "财务部" << "工程师";
columnValue << "熔化工段" << "井下工种" << "20年" << "财务技术人员" << "2020-01-01";
columnValue << "2025-12-31" << "362426190004362256" << "18066667777" << "上海市虹漕路321号";
columnValue << "刘德华" << "夫妻" << "13066668888";
//约定末尾这个是头像图片路径
columnValue << ":/image/head.jpg"; //创建报表头和报表体构成一页数据
DataReport::creatPersonReportHead(list, reportData);
//末尾参数是空行数,可以根据实际需求自行调整
DataReport::creatPersonReportBody(list, reportData, columnValue, 6);
} //将要打印的内容传给打印类结构体数据
dataContent.html = list.join("");
DataPrint::dataPrint->setDataContent(dataContent);
DataPrint::dataPrint->print();
}

Qt数据库应用23-个人信息报表的更多相关文章

  1. SQL Server 数据库表的统计信息的更新

             最近在调整基础信息数据时,新增了几个客户类型,意想不到的事情发生了,在使用新增的客户类型作为 查询条件查询报表时,居然出现了超时的现象,但是用其他以前的客户类型查询就没有问题,用一个 ...

  2. 【mysql元数据库】使用information_schema.tables查询数据库和数据表信息

    概述 对于mysql和Infobright等数据库,information_schema数据库中的表都是只读的,不能进行更新.删除和插入等操作,也不能加触发器,因为它们实际只是一个视图,不是基本表,没 ...

  3. qt数据库多线程问题的解决(QSqlDatabase只能在创建它的线程中使用)

    Qt数据库由QSqlDatabase::addDatabase()生成的QSqlDatabase只能在创建它的线程中使用, 在多线程中共用连接或者在另外一个线程中创建query都是不支持的几乎国内没有 ...

  4. Spring Security教程(五):自定义过滤器从数据库从获取资源信息

    在之前的几篇security教程中,资源和所对应的权限都是在xml中进行配置的,也就在http标签中配置intercept-url,试想要是配置的对象不多,那还好,但是平常实际开发中都往往是非常多的资 ...

  5. Spring Security教程(二):通过数据库获得用户权限信息

    上一篇博客中,Spring Security教程(一):初识Spring Security,我把用户信息和权限信息放到了xml文件中,这是为了演示如何使用最小的配置就可以使用Spring Securi ...

  6. Qt添加驱动——Qt数据库之添加MySQL驱动插件

    Qt数据库之添加MySQL驱动插件(1) 现在可用的数据库驱动只有3种,在Qt中,我们需要自己编译其他数据库驱动的代码,让它们以插件的形式来使用.下面我们就以现在比较流行的MySQL数据库为例,说明一 ...

  7. Qt数据库_资料

    1. QT笔记_数据库总结(一)-rojian-ChinaUnix博客.html http://blog.chinaunix.net/uid-28194872-id-3631462.html (里面有 ...

  8. SQL Server 游标运用:查看数据库所有表大小信息

    一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 背景(Contexts) 实现代码(SQL Codes) 方法一:运用游标 方法二:运用系统存储过程 方法三:拼接SQL ...

  9. mongodb查看数据库和表的信息

    mongodb查看数据库和表的方法比较简单,在为这里推荐使用stats的方法,直观并且详细. 1.查看数据库 db.stats();1输出: { "db" : "siri ...

  10. SQL Server 游标运用:查看所有数据库所有表大小信息(Sizes of All Tables in All Database)

    原文:SQL Server 游标运用:查看所有数据库所有表大小信息(Sizes of All Tables in All Database) 一.本文所涉及的内容(Contents) 本文所涉及的内容 ...

随机推荐

  1. kotlin类和对象—>属性与字段

    1.声明属性,Kotlin 类中的属性既可以用关键字 var 声明为可变的,也可以用关键字 val 声明为只读的 class Address { var name: String = "Ho ...

  2. Exchange学习非常有用的网站

    Exchange学习非常有用的网站 https://docs.microsoft.com/zh-cn/exchange/plan-and-deploy/deployment-ref/network-p ...

  3. 在 KubeSphere 中使用 APISIX Ingress 网关接入自定义监控

    KubeSphere 3.2.0 发布了!为项目网关增配了整套监控及管理页面,同时引入了集群网关来提供集群层面全局的 Ingress 网关能力.当然,我们还是可以部署使用第三方 Ingress Con ...

  4. 带你一起看看nginx如何部署安装

    nginx部署安装 Linux安装 源码构建Nginx 管理器安装 windows安装 首先需要下载Nginx软件包 nginx软件官方下载地址: nginx官方下载连接 建议选择稳定的软件版本,如果 ...

  5. C++多线程应用

    一个进程就是一个程序,一个程序里不止一个功能,每个功能的实现就可以交给一个线程去完成.一个进程就像是一个工程,这个工程里,有设计,有监理,有施工,就相当于三个线程,各干各的又相互配合. https:/ ...

  6. C++ STL之map、multimap

    map和multimap是C++ STL(Standard Template Library)中的关联容器,它们提供键值对的存储和访问. map是一个有序关联容器,它存储一组键值对,其中每个键都是唯一 ...

  7. 剖析Air724UG的硬件设计,有大发现?04篇

    ​ 接下来分享第四部分. 5.4 功耗 5.4.1 模块工作电流 测试仪器:综测仪 R&S CMW500,程控电源 安捷伦 66319D 测试条件:VBAT=3.8V,环境温度 25℃,插入白 ...

  8. AT开发FOTA远程升级:Air780EP低功耗4G模组

    ​ 针对客户朋友的应用反馈,特本篇文章:基于Air780EP模组AT开发的FOTA远程升级指南. AT版本的远程升级主要是对AT固件版本进行升级,实际方式为通过合宙官方IoT平台升级或者使用自己搭建的 ...

  9. 程序员遇到bug时常见的30种反应

    开发应用程序是一项压力很大的工作,人无完人,工作中遇到bug是很正常的事,有些程序员会生气,沮丧,郁闷,甚至泄气,也有一些程序员则会比较淡定.如何进行修复bug的过程,是值得我们好好推敲的. 我想分享 ...

  10. Java Study For Six Day( 面向对象二)

    static(静态)关键字 用于修饰成员(成员变量和成员函数) 被修饰后的成员具备以下的特点 随着类的加载而加载 优先于对象存在 被所有的对象共享 可以被类名直接调用 静态注意事项 静态方法只能访问静 ...