这些年的工作当中,最早是在8年前接触到FastReport这个报表工具,从名字上来看,直译过来就是快速报表,正所谓天下武功,唯快不破,FastReport报表早些年确实是制作报表的不二之选,8年前的工作中,涉入到了医疗行业,项目中要使用FastReport.net 打印基因检测报告,效果确实是满足了项目要求。

最新版FastReport.net支持.net core

鸡肋的FastReport.net网页版

回过来看,最近几年的工作中,报表也是一直使用的是FastReport.net ,但转到了BS端,即在网页上打印业务单据,基本上不复杂的效果完全满足要求,而且我还开发了基于ActiveX的控件,以解决浏览器上FastReport报表BS版本比较鸡肋的功能,FastReport.net的网页版本的打印是通过生成PDF文件,然后调用PDF的打印功能来实现打印,如果原生的浏览器打印功能,会存在着页脚被添加一串URL地址、翻页,定制页不能打印的问题。通过开发FastReport.net的ActiveX控件,可以实现用户自定义报表功能,网页直接打印功能,指定默认打印机功能,反正跟桌面版本体验一致的效果。但唯一的缺点是基于IE内核,因ActiveX是弥补IE的缺陷而生的一项技术,等将来有兴趣了,我或许会移植该程序以支持基于Chrome浏览器,让FastReport.net这个报表,不管任何浏览器都获得跟桌面版本一致的体验。

今天要分享给大家的是使用FastReport.net 2017版本而编写的一个demo程序,并提供了示例程序下载;为何使用FastReport.net 2017,那是因为购买的授权是这个版本,当时想着该版本支持 .net core,但实际情况是该版本只是一个过渡版本,对.net core的支持有问题,官方示例程序我是没有运行起来,于是我就变通一下,直接使用 webform方式来实现,因为该报表对webform实现支持是比较良好的,如果有朋友或公司购买了FastReport.net老版本的授权,但想要它支持.net core是行不通的,必须最新版本才行,因此可以像我这样变通解决网页版打印问题。

解决方法如下

1、单独新建一个webform的网站项目,只添加报表相关文件和代码,我们暂且将这个项目称之为FastReport.Print,它是一个BS版本的基于FastReport.net实现打印需求的网站项目,如果您的业务单据较多,可以建很多目录,同时将报表文件扔到相应目录即可,只需要在代码里面指定报表文件即可。

2、正式的网站项目,我们称之为XXXX.Portal,在Portal某个目录的页面上发起打印请求,该页面上放置一个打印按钮,名字叫:批量打印,触发批量打印事件后,从后台获取到数据,然后将打印数据按报表所需要的格式Post到FastReport.Print对应的打印页面即可。记住,一定是发送post请求,至于格式,你可以是任意的,只需要您自己能解析出来即可,将解析出来的数据,在Print项目的页面的C#代码里面,向FastReport.net注册数据源即可。

示例代码解读

下面是aspx页面代码

<%@ Page Title="Home Page" Language="C#" MasterPageFile="~/Site.Master" AutoEventWireup="true" CodeBehind="Default.aspx.cs" Inherits="WebApplication1._Default" %>
<%@ Register assembly="FastReport.Web" namespace="FastReport.Web" tagprefix="cc2" %> <asp:Content ID="BodyContent" ContentPlaceHolderID="MainContent" runat="server"> <div>
<h3><a href="https://jhrs.com" target="_blank"> jhrs.com</a> 出品 FastReport 打印功能示例</h3>
</div>
<div class="row">
<cc2:WebReport ID="WebReport1" runat="server" ShowCsvExport="true" ShowExports="true" />
</div>
</asp:Content>

  页面代码是一个WebReport的标签,即FastReport的服务器端标签,下面是后台C#代码:

 public partial class _Default : Page
{
protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
string reportFile = Server.MapPath("/report/设备标牌.frx");
//这里为了演示,直接new一个集合,您应该从另外一个项目将数据post过来再反序列化。通过Request.Form["参数名"],将它反序列化即可。
List<SbData> list = new List<SbData>();
for (int i = ; i < ; i++)
{
list.Add(new SbData
{
标题= "帝国游戏集团游戏装备标牌",
使用科室 = "机要部" + i,
启用日期 = DateTime.Now.AddDays(i).ToString(),
型号 = "XH34534-" + i,
序列号 = "XLH-3452" + i,
序号 = (i + ).ToString(),
条码 = "BH20190302002" + i,
生产厂家 = "帝国" + i + "科技有限公司",
规格 = "GGX-1" + i,
设备名称 = "X射線髮射器",
责任人 = "趙佳仁" + i,
质保日期 = DateTime.Now.AddYears(i).ToString()
});
}
var dt = ToDataTable(list); WebReport1.RegisterData(dt, "设备管理标牌");
WebReport1.Report.Load(reportFile);
WebReport1.Prepare();
} } /// <summary>
/// 将泛型集合类转换成DataTable
/// </summary>
/// <typeparam name="T">集合项类型</typeparam>
/// <param name="list">集合</param>
/// <returns>数据集(表)</returns>
public DataTable ToDataTable<T>(IList<T> list)
{
List<string> propertyNameList = new List<string>(); DataTable result = new DataTable();
if (list.Count > )
{
PropertyInfo[] propertys = list[].GetType().GetProperties();
foreach (PropertyInfo pi in propertys)
{
if (propertyNameList.Count == )
{
result.Columns.Add(pi.Name, pi.PropertyType);
}
else
{
if (propertyNameList.Contains(pi.Name))
result.Columns.Add(pi.Name, pi.PropertyType);
}
} for (int i = ; i < list.Count; i++)
{
ArrayList tempList = new ArrayList();
foreach (PropertyInfo pi in propertys)
{
if (propertyNameList.Count == )
{
object obj = pi.GetValue(list[i], null);
tempList.Add(obj);
}
else
{
if (propertyNameList.Contains(pi.Name))
{
object obj = pi.GetValue(list[i], null);
tempList.Add(obj);
}
}
}
object[] array = tempList.ToArray();
result.LoadDataRow(array, true);
}
}
return result;
} } public class SbData
{
public string 标题 { get; set; }
public string 序号 { get; set; }
public string 设备名称 { get; set; }
public string 规格 { get; set; }
public string 型号 { get; set; }
public string 启用日期 { get; set; }
public string 序列号 { get; set; }
public string 生产厂家 { get; set; }
public string 条码 { get; set; }
public string 责任人 { get; set; }
public string 使用科室 { get; set; }
public string 质保日期 { get; set; }
}

上面的代码,是一个aspx页面后置代码,我们可以在Page_Load事件里面接收正式项目提交过来的打印数据,建议使用json,因为你只需要直接反序列化即可,方便。然后将获取到的值还要转为DataTable,因为FastReport注册数据源我这儿使用的是DataTable类型,至于其它类型,我没有去烟酒是否正常。

FastReport.net打印网页版运行效果一览

网页版的打印效果如下:

上面的报表,排版格式是很简单的,二维码FastReport自带,这里不介绍使用教程,这个示例只是一个很简单的应用,它的更多功能,可以看官方示例及源码。

基于asp.net 的Demo源码下载

以下提供的demo程序,下载后直接使用VS打开即可运行。

源码下载:点击原文链接进行下载

使用FastReport.net 报表在网页上实现打印功能的更多相关文章

  1. 网页上facebook分享功能的具体实现

    1,一个链接: 参数是要分享的页面的链接 代码如下: <a style="width:35px; height:40px; position:relative; top:10px; l ...

  2. 在Winform开发中使用FastReport创建报表

    FastReport.Net是一款适用于Windows Forms, ASP.NET和MVC框架的功能齐全的报表分析解决方案.可用在Microsoft Visual Studio 2005到2015, ...

  3. 使用FastReport报表工具实现信封套打功能

    在较早期的报表套打的时候,我倾向于使用LODOP的ActiveX进行报表的打印或者套打,BS效果还是很不错的.之前利用它在Winform程序里面实现信封套打功能,详细参考<基于信封套打以及批量打 ...

  4. “此网页上的某个 Web 部件或 Web 表单控件无法显示或导入。找不到该类型,或该类型未注册为安全类型。”

    自从vs装了Resharper,看见提示总是手贱的想去改掉它.于是乎手一抖,把一个 可视web部件的命名空间给改了. 喏,从LibrarySharePoint.WebPart.LibraryAddEd ...

  5. [moka同学收藏]网页上的“返回上一页”的几种实现代码

    我们在制作网页的时候,经常在网页上要用到"返回上一页"的功能.这一功能在制作网页的时候会有多种编码方法,在此,笔者将比较常用的几种编码写作方法在下面列出来,供各位技术人员参考使用. ...

  6. 使用chrome查看网页上效果的实现方式

    使用chrome查看网页上效果的实现方式 chrome是一个极为强大的工具,很多时候,我们不知道一个效果怎么实现的,我们完全可以找到响应的网页,然后找到其html文件,和js文件,查看源码,获得其实现 ...

  7. css015 定位网页上的元素

    css015 定位网页上的元素 一.   定位属性的功能 1.         四中类型的定位 Position: absolute relative fixed static a. 绝对定位 绝对定 ...

  8. CSS3-基于浮动的布局,响应式WEB设计,定位网页上的元素,设计打印页面的css技术

    基于浮动的布局: 1.除非图片设置了宽度,否则始终应该要对浮动的图片设置一个宽度,这样可以让浏览器给其他内容腾出环绕的空间 2.当侧边栏的高度与主内容区的高度不一致的时候,可以用个margin进行调整 ...

  9. 如何获取网页上的LOGO

    一般公司网页上的图片都会禁止右键另存为,用截图工具接下来的图会带背景色,PS成背景透明有点费时间. 用Google Chrome 或Firefox 打开目标网页,右键点击审查元素,将鼠标放在图片上,一 ...

随机推荐

  1. 发现 TSplitter 在嵌套时不好用, 索性写了个替代品(处理MouseDown,MouseMove,MouseUp,然后设定控件的Left值就可以了)

    代替 TSplitter 的 TDirPanel 类: unit DirPanel; interface uses   Classes, Controls, Forms, ExtCtrls; type ...

  2. New,Getmem,ReallocMem联系与区别(转)

    procedure New(var P: Pointer);  {为一个指针变量分配内存,会自动计算指针所指数据结构需要空的空间大小} procedure GetMem(var P: Pointer; ...

  3. 模拟键盘发送文字(使用SendInput API函数)

    嗯...老生常谈的话题, 不过系统的总结了一下, 找了个相对简单的实现方式, 可以方便的发送任何文字 参考另一片文章: http://www.cnblogs.com/-clq/archive/2011 ...

  4. php身份证号的验证

    //身份证号验证 03 protected function checkIdCard(){ 04 if(empty($_POST['idcard'])){ 05 return false; 06 } ...

  5. Spring与IoC

    控制反转(IOC,Inversion of Control),是一个概念,是一种思想. 指将传统上由程序代码直接操控的对象调用权交给容器,通过容器来实现对象的装配和管理.控制反转就是对对象控制权的转移 ...

  6. hadoop之hive&hbase互操作

    大家都知道,hive的SQL操作非常方便,但是查询过程中需要启动MapReduce,无法做到实时响应. hbase是hadoop家族中的分布式数据库,与传统关系数据库不同,它底层采用列存储格式,扩展性 ...

  7. Java动态规划

    1. 介绍 动态规划典型的被用于优化递归算法,因为它们倾向于以指数的方式进行扩展.动态规划主要思想是将复杂问题(带有许多递归调用)分解为更小的子问题,然后将它们保存到内存中,这样我们就不必在每次使用它 ...

  8. element-ui源码之组件通信那些事

    最近在用element-ui重构前端项目,无意之中翻阅到一个比较好用的组件间通信方式,借助于vue的封装的发布-订阅消息模式与mixin语法.在开始之前先总结下vue常用的组件间通信方式,具体如下: ...

  9. Hive —— 安装部署

    一.安装Hive 1.1 下载并解压 下载所需版本的Hive,这里我下载版本为cdh5.15.2.下载地址:http://archive.cloudera.com/cdh5/cdh/5/ # 下载后进 ...

  10. [Vue 牛刀小试]:第十四章 - 编程式导航与实现组件与 Vue Router 之间的解耦

    一.前言 在上一章的学习中,通过举例说明,我们了解了 Vue Router 中命名路由.命名视图的使用方法,以及如何通过 query 查询参数传参,或者是采用 param 传参的方式实现路由间的参数传 ...