百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术教程 > 正文

14. WPS表格使用JS宏查询报表明细之查询实现

csdh11 2024-12-10 13:12 22 浏览

本节继续 上一节 内容实现明细查询的代码。

先分析一下报表结构:

报表主要由三部分组成,如图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

相关推荐

IDEA界面太丑??尝试一下这几个插件

前言IntelliJIDEA主要用于支持Java、Scala、Groovy等语言的开发工具,同时具备支持目前主流的技术和框架,擅长于企业应用、移动应用和Web应用的开发。IntelliJi...

小巧 Vue 页面滚动进度条组件ScrollProgress

今天给大家分享一个轻量级Vue.js全屏滚动进度条组件VueScrollProgress。vue-scroll-progress一款基于vue.js构建的页面滚动进度条组件,...

基于vue实现可视化拖拽编辑器,页面生成工具,提升前端开发效率

项目介绍基于vue实现的可视化拖拽编辑器,实现页面生成工具,提升前端开发效率。可以基层到移动端项目作为自定义json直接生成UI页面。项目特点功能特点...

优秀 vue+heyui 后端管理系统HeyUI-Admin

今天再给小伙伴们推荐一款成熟的企业中后台管理系统HEYUI-Admin。heyui-admin基于vue.js和heyui组件库构建的后台管理系统。包含基础表单/表格功能,拓展组件(图表、富文本编辑...

响应式 Vue.js 前端组件化框架Xvue-UI

今天给小伙伴们推荐一款超不错的Vue轻量级组件框架XVueUI。xvue-ui基于vue2.x构建的响应式前端组件化框架。轻量级、易于上手,提供...

《基于SpringBoot+Vue的在线视频系统设计与实现》开题报告

【计算机毕业设计案例】基于SpringBoot+Vue的在线视频系统设计与实现_哔哩哔哩_bilibili...

超好用 Vue.js 图片裁切组件Vue-ImgCutter

今天给小伙伴们分享一个超棒的Vue图片任意裁剪插件VueImgCutter。vue-img-cutter基于vue2.x构建的轻量级剪切图片组件。支持移动图像、放大缩小图片、任意移动图片、固定比...

Vue 3 进阶用法:异步组件(vue 异步组件原理)

一、代码分割一个大型前端应用,如果所有代码都放在单一文件,体积会特别大,下载时间长,白屏时间久,用户体验差。...

源码补丁神器—patch-package(源码助手怎么用)

作者:张浩一、背景vue项目中使用vue-pdf第三方插件预览pdf,书写业务代码完美运行,pdf文件内容正常预览无问题。后期需求有变,业务需求增加电子签章功能。这个时候pdf文件的内容可以显示出...

经验分享:Vue2 项目升级 Vue3 + Element Plus,借助Deepseek手动升级

Vue3出来好久了,我开发的项目还在使用Vue2框架,一般情况下不考虑升级,但是最近需要接入工作流程引擎之类的,看了下Vue2生态下操作空间不是很好,那索性尝试升级Vue3吧。一番操作下来,升级成功,...

34K Star!史上最全JavaScript资源库 awesome-javascript

34KStar!史上最全JavaScript资源宝库大揭秘引言在GitHub上,有一个备受瞩目的JavaScript资源仓库,以其全面的内容和精心的分类,成为了众多开发者的必备参考。这个拥有超过...

基于 Vue.js 磁片栅格布局组件VueGridLayout

#头条创作挑战赛#今天给大家分享一个超优秀的vue.js拖拽栅格布局插件VueGridLayout。...

6款高颜值 Vue3 PC端UI组件库(vue3开发组件库)

马上到国庆了,还没学习或者想学习vue3的小伙伴们有安排上没?这次推荐几个比较流行的VUE3UI组件库,合理利用,又或者学习借鉴都是不错的选择。1、element-pluselement-plus...

高性能 vue.js+ztree 树形组件Vue-GiantTree

今天给大家分享一款超棒的Vue海量数据渲染树形组件VueGiantTree。vue-giant-tree基于ztree封装的Vue树形组件。轻松实现大数据高性能渲染,适合海量数据渲染场景。zTr...

【推荐】2024年推荐的6款开源免费 Vue 后台管理系统模板,建议收藏

前言在现今的软件开发领域,...