14. WPS表格使用JS宏查询报表明细之查询实现
csdh11 2024-12-10 13:12 19 浏览
本节继续 上一节 内容实现明细查询的代码。
先分析一下报表结构:
报表主要由三部分组成,如图14.0.。
左侧是“固定区域”,图14.0中的灰色区域,共5列:排序、类、款、项、科目名称。
中间是“指标区域”,图14.0中的蓝色区域,共4列:指标总金额、指标已用金额、指标可用金额、计划金额合计。
右侧是“支出区域”,图14.0中的橙色区域,共6列:工资福利支出、对个人和家庭补助支出、公用经费、部门预算项目、专项资金项目、其他项目。
【显示明细】按钮的使用方法,鼠标选中相应单元格,使之成为活动单元格,再点击显示明细按钮。
显示明细按钮的代码应做如下工作:
第一步、判断活动单元格是否在指定的区域内;
第二步、如果这个单元格在相应区域内:
a. 根据当前活动单元格的排序行,分析出是合计、类、款、项以及单位这些汇总行的哪一个(如图13.0);
b. 根据当前活动单元格的标题决定是否执行项目分类明细的查询;
第三步、使用当前工作薄的现有连接,创建新两个新的查询:
a. 查询源数据抽出有用的列形成明细表;
b. 按项目名称汇总出相关数据。
第一步细化说明及相关代码:
? 如果不在报表数据区域内,什么也不做
? 如果在“固定区域”,什么也不做
? 如果在“指标区域”,则查询出无项目分类的汇总数据(计划合计)
? 如果在“支出区域”,则查询出具体的项目分类汇总数据(项目分类为橙色列对应的明细 图14.0)
涉及到了查询表所覆盖的工作表区域,用到了Range区域对象QueryTable.ResultRange这个属性。
整个查询表使用的列是固定的,从B列到P列;使用的行不是固定的,从第4行开始,到ResultRange.End(xlDown).Row返回的行号。
要对QueryTable.ResultRange这个区域分类,在lib模块添加三个函数,分别获取到这三个区域。相关代码如下:
/* getFixedRange 获取左侧固定区域
* 传入报表返回的数据区域
* 返回C:F,4列,不包括排序列和标题行
*/
function getFixedRange(ResultRange){
return ResultRange.Offset(1,1).Resize(ResultRange.Rows.Count-1, 4)
}
/* getQuotaRange 获取指标区域
* 传入报表返回的数据区域
* 返回J:G,4列,包括了计划合计列,不包括标题行
*/
function getQuotaRange(ResultRange){
return ResultRange.Offset(1,5).Resize(ResultRange.Rows.Count-1, 4)
}
/* getDisbursedRange 获取支出区域
* 传入报表返回的数据区域
* 返回K:P,6列,不包括计划合计列与标题行
*/
function getDisbursedRange(ResultRange){
return ResultRange.Offset(1,9).Resize(ResultRange.Rows.Count-1, 6)
}
判断当前活动单元格是否在指标与支出区域的代码:
//显示明细按钮的单击事件
function CommandButton2_Click()
{
let ShReport = Application.ThisWorkbook.ActiveSheet
let DataRange = ShReport.QueryTables.Item(1).ResultRange
//分别获取指标区域与支出区域
let r1 = getQuotaRange(DataRange)
let r2 = getDisbursedRange(DataRange)
//得到当前活动单元格,如果是区域,则选择第一个单元格
let c1 = Selection.Cells(1)
/* 第一步 判断活动单元格是否在指定的区域内;*/
//交叉区域判断 判断所选单元格是否在金额数据汇总区域
if (!Intersect(c1, Union(r1, r2)))
{
alert("所选单元格不在金额汇总数据显示区域,无法显示明细数据,请重新选择。")
return 0
}
第二步细化说明及相关代码:
/* 第二步 a. 根据当前活动单元格的排序行分析出
* 是合计、类、款、项以及单位这些汇总行的哪一级
* 并将这些条件增加到where语句后面
*/
//定义一个在sql where条件里增加的条件
let AddWhere = ""
//定义一个单位单元格是否为空或者全部的布尔变量
let UnitValue = ShReport.Range("f1").Text
let BoolUnit = UnitValue == "0-全部" || UnitValue == ""
//取得当前活动单元格同一行排序单元格的值
let OrderValue = ShReport.Cells.Item(c1.Row, 2).Value2
/* 解析OrderValue,判断当前活动单元格位于什么汇总级别 */
if (OrderValue == '0')
{
//汇总合计行,什么也不用做,全部提取就好了
}
else
{ //根据排序值的长度来判断是哪一级的汇总
switch(OrderValue.length)
{
case 7: //类款项的汇总行
/* 查询的是 2010000 这种类汇总行 */
if (OrderValue.substr(3,4) == "0000")
{ // l示例:and (left([支出功能分类],3) = '201')
AddWhere = " and (left([支出功能分类],3) = '" + OrderValue.substr(0,3) +"')"
}
/* 查询的是 2010100 这种款汇总行 */
else if (OrderValue.substr(5,2) == "00")
{ // 示例:and (left([支出功能分类],5) = '20101')
AddWhere = " and (left([支出功能分类],5) = '" + OrderValue.substr(0,5) +"')"
}
/* 查询的是 2010201 这种项汇总行 */
else
{ // 示例:and (left([支出功能分类],7) = '2010101')
AddWhere = " and (left([支出功能分类],7) = '" + OrderValue.substr(0,7) +"')"
}
break
case 9://单位汇总行-缺款、项,只有类+单位的汇总行 201+101003
// 示例:and (left([支出功能分类],4) = '201-') and (left([单位],6) = '101003')
AddWhere = " and (left([支出功能分类],4) = '" + OrderValue.substr(0,3) +"-')"
AddWhere += BoolUnit ? " and (left([单位],6) = '" + OrderValue.substr(4) + "')" : ""
break
case 11://单位汇总行-缺项,只有类款+单位的汇总行 20102+101003
// 示例:and (left([支出功能分类],6) = '20102-') and (left([单位],6) = '101003')
AddWhere = " and (left([支出功能分类],6) = '" + OrderValue.substr(0,5) +"-')"
AddWhere += BoolUnit ? " and (left([单位],6) = '" + OrderValue.substr(6) + "')" : ""
break
case 13://单位汇总行 类款项+单位的汇总行 2010201+102001
// 示例:and (left([支出功能分类],8) = '2010201-') and (left([单位],6) = '101003')
AddWhere = " and (left([支出功能分类],8) = '" + OrderValue.substr(0,7) +"-')"
AddWhere += BoolUnit ? " and (left([单位],6) = '" + OrderValue.substr(8) + "')" : ""
break
default:
Console.log("未知错误1")
}
}
/* 第二步 b. 根据标题列决定是否显示项目分类明细的其中一个:
* 工资福利支出 对个人和家庭补助支出 公用经费
* 部门预算项目 专项资金项目 其他项目
*/
//取得当前活动单元格列标题的值
let TitleValue = ShReport.Cells.Item(4, c1.Column).Value2
//列举项目分类明细内容 如果标题是这里面的内容,要查询项目分类明细
const ProjectDetails = "工资福利支出 对个人和家庭补助支出 公用经费 部门预算项目 专项资金项目 其他项目"
//定义在where里追加的第二个条件
let AddWhere2 = ""
//查询的是项目分类明细
if (ProjectDetails.indexOf(TitleValue) > -1){
AddWhere2 = TitleValue == "其他项目" ?
" and ([项目类别] not in ('工资福利支出','对个人和家庭补助支出','公用经费','部门预算项目','专项资金项目')"
: " and ([项目类别] ='" + TitleValue + "')"
}
第三步、使用当前工作薄的现有连接,创建新两个新的查询:
/* 第三步、使用当前工作薄的现有连接,创建新两个新的查询 */
//数据查询参数设置
let ShDict = Application.Worksheets.Item("字典")
//获取报表上面的查询条件参数
let whereStr = getWhereStr(ShReport)
//使用where条件替换存储的支出查询SQL中的?
let sqlstr = ShDict.Range("d3").Value2.replace(/\?/g, whereStr + AddWhere + AddWhere2)
//使用当前工作薄的现有连接 cnstr
let cnstr = Application.ThisWorkbook.Connections.Item(1).OLEDBConnection.Connection
//支出查询数据存放工作表
let ShDetail = Application.Worksheets.Item("支出明细")
//先清除查询
for (let qt of ShDetail.QueryTables){
qt.ResultRange.Clear()
qt.Delete()
}
//查询明细 创建支出查询
let qDetail = ShDetail.QueryTables.Add(cnstr, ShDetail.Range("A3"), sqlstr)
qDetail.Refresh()
/*
*下面查询项目分类明细
*/
let ShProjectDetails = Application.Worksheets.Item("项目分类明细汇总")
// 项目分类明细汇总查询的存放工作表
sqlstr = ShDict.Range("d2").Value2.replace(/\?/g, whereStr + AddWhere + AddWhere2)
//先清除查询
for (let qt of ShProjectDetails.QueryTables){
qt.ResultRange.Clear()
qt.Delete()
}
//查询明细 创建项目明细查询
let qProjectDetails = ShProjectDetails.QueryTables.Add(cnstr, ShProjectDetails.Range("A3"), sqlstr)
qProjectDetails.Refresh()
alert("明细查询完成,请切换到【支出明细】和【项目分类明细汇总】工作表(Sheet)查看。")
}//END function CommandButton2_Click()
// CommandButton2_Click() 函数结束
至此,完成了明细查询的所有代码。下一节,将对main模块内CommandButton_Click()函数的代码进行适当的精简,使用函数替换其中的大部分内容,以优化阅读体验。
本节示例,附下载链接:
/*
本节使用的两个工作簿在下面的链接,下载后放到一个文件夹中。
将文件名分别重命名为database.et 和 报表.et
打开【报表.et】,操作参考第4节的图4.4:
在数据菜单中,【导入数据】-【编辑连接属性】
点击连接文件路径后面的【浏览】,找到下载的database.et文件,就可以使用了。
*/
//本次查询使用的数据库文件
database.et 文件的下载链接:https://kdocs.cn/l/cjquBVytt7DX
//本次查询使用的报表文件
报表.et文件的下载链接:https://kdocs.cn/l/cs6tGas5E2he
相关推荐
- 探索Java项目中日志系统最佳实践:从入门到精通
-
探索Java项目中日志系统最佳实践:从入门到精通在现代软件开发中,日志系统如同一位默默无闻却至关重要的管家,它记录了程序运行中的各种事件,为我们排查问题、监控性能和优化系统提供了宝贵的依据。在Java...
- 用了这么多年的java日志框架,你真的弄懂了吗?
-
在项目开发过程中,有一个必不可少的环节就是记录日志,相信只要是个程序员都用过,可是咱们自问下,用了这么多年的日志框架,你确定自己真弄懂了日志框架的来龙去脉嘛?下面笔者就详细聊聊java中常用日志框架的...
- 物理老师教你学Java语言(中篇)(物理专业学编程)
-
第四章物质的基本结构——类与对象...
- 一文搞定!Spring Boot3 定时任务操作全攻略
-
各位互联网大厂的后端开发小伙伴们,在使用SpringBoot3开发项目时,你是否遇到过定时任务实现的难题呢?比如任务调度时间不准确,代码报错却找不到方向,是不是特别头疼?如今,随着互联网业务规模...
- 你还不懂java的日志系统吗 ?(java的日志类)
-
一、背景在java的开发中,使用最多也绕不过去的一个话题就是日志,在程序中除了业务代码外,使用最多的就是打印日志。经常听到的这样一句话就是“打个日志调试下”,没错在日常的开发、调试过程中打印日志是常干...
- 谈谈枚举的新用法--java(java枚举的作用与好处)
-
问题的由来前段时间改游戏buff功能,干了一件愚蠢的事情,那就是把枚举和运算集合在一起,然后运行一段时间后buff就出现各种问题,我当时懵逼了!事情是这样的,做过游戏的都知道,buff,需要分类型,且...
- 你还不懂java的日志系统吗(javaw 日志)
-
一、背景在java的开发中,使用最多也绕不过去的一个话题就是日志,在程序中除了业务代码外,使用最多的就是打印日志。经常听到的这样一句话就是“打个日志调试下”,没错在日常的开发、调试过程中打印日志是常干...
- Java 8之后的那些新特性(三):Java System Logger
-
去年12月份log4j日志框架的一个漏洞,给Java整个行业造成了非常大的影响。这个事情也顺带把log4j这个日志框架推到了争议的最前线。在Java领域,log4j可能相对比较流行。而在log4j之外...
- Java开发中的日志管理:让程序“开口说话”
-
Java开发中的日志管理:让程序“开口说话”日志是程序员的朋友,也是程序的“嘴巴”。它能让程序在运行过程中“开口说话”,告诉我们它的状态、行为以及遇到的问题。在Java开发中,良好的日志管理不仅能帮助...
- OS X 效率启动器 Alfred 详解与使用技巧
-
问:为什么要在Mac上使用效率启动器类应用?答:在非特殊专业用户的环境下,(每天)用户一般可以在系统中进行上百次操作,可以是点击,也可以是拖拽,但这些只是过程,而我们的真正目的是想获得结果,也就是...
- Java中 高级的异常处理(java中异常处理的两种方式)
-
介绍异常处理是软件开发的一个关键方面,尤其是在Java中,这种语言以其稳健性和平台独立性而闻名。正确的异常处理不仅可以防止应用程序崩溃,还有助于调试并向用户提供有意义的反馈。...
- 【性能调优】全方位教你定位慢SQL,方法介绍下!
-
1.使用数据库自带工具...
- 全面了解mysql锁机制(InnoDB)与问题排查
-
MySQL/InnoDB的加锁,一直是一个常见的话题。例如,数据库如果有高并发请求,如何保证数据完整性?产生死锁问题如何排查并解决?下面是不同锁等级的区别表级锁:开销小,加锁快;不会出现死锁;锁定粒度...
- 看懂这篇文章,你就懂了数据库死锁产生的场景和解决方法
-
一、什么是死锁加锁(Locking)是数据库在并发访问时保证数据一致性和完整性的主要机制。任何事务都需要获得相应对象上的锁才能访问数据,读取数据的事务通常只需要获得读锁(共享锁),修改数据的事务需要获...
- 一周热门
- 最近发表
- 标签列表
-
- mydisktest_v298 (34)
- document.appendchild (35)
- 头像打包下载 (61)
- acmecadconverter_8.52绿色版 (39)
- word文档批量处理大师破解版 (36)
- server2016安装密钥 (33)
- mysql 昨天的日期 (37)
- parsevideo (33)
- 个人网站源码 (37)
- centos7.4下载 (33)
- mysql 查询今天的数据 (34)
- intouch2014r2sp1永久授权 (36)
- 先锋影音源资2019 (35)
- jdk1.8.0_191下载 (33)
- axure9注册码 (33)
- pts/1 (33)
- spire.pdf 破解版 (35)
- shiro jwt (35)
- sklearn中文手册pdf (35)
- itextsharp使用手册 (33)
- 凯立德2012夏季版懒人包 (34)
- 反恐24小时电话铃声 (33)
- 冒险岛代码查询器 (34)
- 128*128png图片 (34)
- jdk1.8.0_131下载 (34)