c#中使用miniExcel和fastreport实现付款审批单的批量打印
haoteby 2025-01-10 13:47 16 浏览
有这么一个需求,公司财务会根据银行回单来打印付款申批单,没有软件的情况下只能在Excel表格中把付款审批单的样子做出来,然后再把银行回单的内容填进去,基本上录入一个打印一个,没什么效率不说,也无法实现查重,所以有时候会有付款给付的重复的情况出现;现在我们要给他解决一下这问题,就是通过excel把表格内容导入到软件中,在进行保存时进行查重,重复的记录就不保存了,然后可以按需要多选或单选记录来进行打印,通过fastreport设计好的模版来保障打印质量的可靠稳定,在打印时可以预览打印内容,也可以不预览直接连续打印。
使用c#语言进行开发,选择.net4.8+winform,数据库使用的是SQL 2016,就一个数据表,用来存放审批单的内容;使用MiniExcel来实现读取excel文件并把内容加载到DataGridView中,使用Ef Core 3.1来进行数据查重和保存,通过fastreport来实现按模板打印,可以使用AntdUI来美化下界面,下面开工。
界面设计
大概如下,通过【打开Excel】来实现加载表格中的数据,搜索按钮则是查询数据库中已经存在的数据,保存按钮在保存时会检测是从excel中打开的数据还是从数据库中查询出来的,如果是数据库中查询出来的,就不进行保存,然后单选或多选记录后点击打印,就能实现按模板连续或单个打印了。
NuGet包引用
由于是在.net4.8中使用ef core,所以要引用的包有些多,好些包是自动安装的,需要引用Microsoft.EntityFrameworkCore和Microsoft.EntityFrameworkCore.SqlServer,这两个都选3.1.32版本,还需要引用MiniExcel,不然没法方便导入数据。
包引用完成后,编写数据实体类,使用的是中文表名,所以写法如下
public class CaiWu
{
}
[Table("付款审批表")]
public class ShenPi
{
[Key]
public int Id { get; set; }
public DateTime 日期 { get; set; }
public string 申请部门 { get; set; }
public string 项目内容 { get; set; }
public decimal 金额 { get; set; }
public string 收款单位 { get; set; }
public string 账号 { get; set; }
public string 开户行 { get; set; }
public string 付款方式 { get; set; }
public string 备注 { get; set; }
}
编写数据库上下文,jxc.connStr是数据库连接字符串,直接写或写配置文件中都可以
public class MyDbContext : DbContext
{
public MyDbContext()
{
}
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer(jxc.connStr);
}
/// <summary>
/// 付款审批表
/// </summary>
public DbSet<ShenPi> ShenPis { get; set; }
}
编写Excel导入的代码
public Boolean isExcel = true;
List<ShenPi> books = new List<ShenPi>();
点击打开Excel时
string fileName = "";
OpenFileDialog ofd = new OpenFileDialog();
ofd.Filter = "Excel文件|*.xlsx";
ofd.Multiselect = false;
var dr = ofd.ShowDialog();
if (dr == DialogResult.OK)
{
fileName = ofd.FileName;
}
else
{
return;
}
try
{
books = MiniExcel.Query<ShenPi>(fileName).ToList();
DG_show.DataSource = books;
isExcel = true;
}
catch (Exception ex)
{
MessageBox.Show("打开表格中的数据时出现以下错误:\r\n" + ex.Message.ToString(), jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
return;
}
使用miniExcel直接把表格数据转换成list<ShenPi>,在转换时会自动检查数据格式是否正确,不正确的话会报错,会提示第几行第几列出错了。
查询
查询不管是单条件还是多条件,在ef core中实现起来简直太简单了,几行语句搞定,如果你还是使用SQL语句来实现的话,那一定要学下ef core了,两种方法都写了。
//搜索数据库,字符串拼接的方法实现
//string sql;
//string name = "";
//name = txt_name_ck.Text.ToString().Trim();
//if (name.Length < 1)
//{
// sql = "select top 100 * from 付款审批表 order by id desc ";
//}
//else
//{
// sql = "select * from 付款审批表 where 项目内容 like '%" + name + "%' or 收款单位 like '%" + name + "%'";
//}
//DG_show.DataSource = jxc.GetTable(sql);
//isExcel = false;
//使用ef Core进行查询
string name = txt_name_ck.Text.Trim();
if (name.Length < 1)
{
using (var context = new MyDbContext())
{
books = context.ShenPis
.Take(100)
.ToList();
}
}
if (name.Length > 0)
{
using (var context = new MyDbContext())
{
books = context.ShenPis
.Where(x => x.申请部门.Contains(name) || x.项目内容.Contains(name))
.ToList();
}
}
DG_show.DataSource = books;
isExcel = false;
保存
保存最主要的是查重,以前传统方式实现时,想要批量保存的话,查重就好难搞,但是用ef core就很方便了。
int num = 0;
if (!isExcel)
{
MessageBox.Show("只有从Excel中加载的数据才需要保存哦!", jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
return;
}
if (books.Count < 1)
{
MessageBox.Show("当前表格中没有数据哦!", jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
return;
}
//保存数据
using (var context = new MyDbContext())
{
books.ForEach(book =>
{
var t = context.ShenPis.FirstOrDefault(p => p.项目内容 == book.项目内容 && p.账号 == book.账号 && p.金额 == book.金额 && p.日期 == book.日期);
if (t != null)
{
num += 1;
}
else
{
context.ShenPis.Add(book);
}
});
try
{
context.SaveChanges();
MessageBox.Show("数据已成功保存,其中有" + num.ToString() + "条重复记录已跳过!", jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
}
catch (Exception ex)
{
MessageBox.Show("保存数据时出现如下错误:\r\n" + ex.Message.ToString(), jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
return;
}
}
打印
打印主要通过fastreport来实现,自用的话从网上找个大家都懂的版本就行了,之所以不用.net8.和ef core8.0,就是因为找不下大家都懂的版本[捂脸],设计模板这个就不写了,自己按需要和大小画就行了。
点击打印时先循环一下表格,看哪些被选中了,然后再打印,变量是随意起的,主打一个能跑就行[呲牙]
//开始打印
for (int i = 0; i <= DG_show.Rows.Count - 1; i++)
{
if (DG_show.Rows[i].Selected)
{
var row = DG_show.Rows[i];
PrintExcel(row);
}
}
public void PrintExcel(DataGridViewRow row)
{
string bm, nr, je, skdw, zh, khh, fkfs, bz, rq;
string DaXie;
string path = Environment.CurrentDirectory + @"\report\" + "huidan.frx";
bm = row.Cells["申请部门"].Value.ToString();
nr = row.Cells["项目内容"].Value.ToString();
je = Convert.ToDecimal( row.Cells["金额"].Value).ToString("N2");
skdw = row.Cells["收款单位"].Value.ToString();
zh = row.Cells["账号"].Value.ToString();
khh = row.Cells["开户行"].Value.ToString();
bz = row.Cells["备注"].Value.ToString();
rq = Convert.ToDateTime(row.Cells["日期"].Value).ToString("D");
fkfs = row.Cells["付款方式"].Value.ToString();
DaXie = jxc.MoneyToChinese(Convert.ToDecimal(je));
Report rp = new Report();
rp.Clear();
rp.Load(path);
rp.SetParameterValue("申请部门", bm);
rp.SetParameterValue("项目内容", nr);
rp.SetParameterValue("金额", je);
rp.SetParameterValue("收款单位", skdw);
rp.SetParameterValue("账号", zh);
rp.SetParameterValue("开户行", khh);
if (fkfs == "现金")
{
rp.SetParameterValue("fs_xj", true);
}
else if (fkfs == "汇票")
{
rp.SetParameterValue("fs_hp", true);
}
else
{
rp.SetParameterValue("fs_yh", true);
}
rp.SetParameterValue("日期", rq);
rp.SetParameterValue("备注", bz);
rp.SetParameterValue("金额大写", DaXie);
rp.Prepare();
if (chk_view.Checked)
{
rp.ShowPrepared();
}
else
{
rp.Print();
}
//
}
最后的效果
相关推荐
- 能跑源码,还提供数据集:这里有一个入门企业级验证码识别项目
-
机器之心专栏作者:kerlomz网上关于验证码识别的开源项目众多,但大多是学术型文章或者仅仅是一个测试demo,那么企业级的验证码识别究竟是怎样的呢?1.前言网上关于验证么识别的开源项目众多,但大...
- kdj源码_kdj源码公式描述
-
N:=9;M1:=3;M2:=3;...
- QT实现抖动文字和滚动文字,附源码
-
前言不知道大家有没有发现今天的文章有什么不一样,哈哈,我自己胡拼乱凑弄了一个logo,好不好看就先不说了,最起码萌萌哒...当然这不是今天的重点,在做logo的时候,我原本想让文字动起来的,奈何技术有...
- 我试图通过这篇文章告诉你,这行源码有多牛逼。
-
你好呀,我是歪歪。这次给你盘一个特别有意思的源码,正如我标题说的那样:看懂这行源码之后,我不禁鼓起掌来,直呼祖师爷牛逼。...
- 想了解Python源代码加密吗?现总结如下5大加密混淆手段!
-
我们在进行...
- Android系统基础(03) Android系统源码下载
-
常规官方网站说明:Android源码官方网站为(google你懂的):https://source.android.com官网参考链接,对应的tag(tag是一种标签,我们可以根据tag来判断下载的...
- 真香,Python爬取B站弹幕原来如此简单,源码已附在文末
-
B站的弹幕区一直是人才圣地。今天我就用python来手把手教大家爬取B站排行榜热门视频,Python爬取视频也可以如此简单。...
- 最详细的 maven 教程,可以收藏_maven步骤
-
链接|cnblogs.com/hzg110/p/6936101.html正文目前所有的项目都在使用maven,可是一直没有时间去整理学习,这两天正好有时间,好好的整理一下。...
- Python黑科技-VIP视频破解源码分享
-
《利用Python制作自己的VIP视频解析软件》想看的电视剧更新了还要充VIP?喜欢的电影你是VIP还得付费?学了Python哪要这些花里胡哨的,打开我自己的VIP付费视频解析软件,想怎么看就怎么看!...
- 抖音无水印解析网站源码_抖音无水印解析平台
-
链接:https://share.weiyun.com/59Ah44S密码:hv4dm7上传到主机解压不用安装,直接打开域名就可以了原文地址:https://www.xigsc.com/post/...
- 「电脑知识」USBOS 3.0 v2022.1.24 超级PE启动维护工具标准增强版
-
前几天一直在发PE类工具就是为了制作U盘PE启动重装系统教程的,今天小编继续分享有一篇关于pe的之前小编发布过一款微PE工具箱,今天发布另外一个无任何流氓行为功能超级强大虽然体积大了一点,但是这个...
- 模版网站建设制作的八步流程_模板的网站
-
模版网站比较简单,一般我们按照如下流程就可以制作出来。 一、网站定位: 在建站之前,一定要了解你要建的网站是什么。你必须考虑你网站的标题(关键词)、网站描述以及你想要建立的网站。 二、选择域...
- 求职季必备,这几个免费的个人简历模板网站,你可千万不要错过!
-
晃晃悠悠又到了春招的季节,相信一定有很多小伙伴趁着这个金三银四求职季,四处投递简历。这时候一个亮眼优秀的简历,可以很好的祝你吸引HR的注意。今天就把我珍藏很久的5个免费简历模板网站分享给大家,简历模板...
- 简约时尚作品博客商店网站HTML5模板源码
-
Meduza是简约时尚和现代的博客HTML模板,带商店电商元素的博客页面。考虑所有的作品集网站需求页可以设计一个旅游网站。原生响应设计HTML5和CSS3(台式机、平板电脑、手机…)简单,干净的和专业...
- 13 款免费样机网站合集,UI设计、产品设计、VI设计全都有!
-
俗话说得好,人靠衣装,在作完设计后不少设计师都会为自己的作品套一个「样机」好让设计看过去更加高端大气上档次!今天,我就总结了无论是UI设计、包装设计、服装设计、品牌设计、logo设计,都能用到...