最近折腾采用第三方报表控件(DevExpress),开始采用的是 文件(.repx)读取的方式进行报表设计器的加载。实现方式如下:
不设置文件复制到输出目录,是无法加载到该报表模块文件的。
1 //获取datatable数据 2 CreateDataTable(); 3 DataSet ds = new DataSet(); 4 ds.Tables.Add(dt); 5 //加载数据,用户自定义报表格式 6 DevExpress.XtraReports.UI.XtraReport report = new DevExpress.XtraReports.UI.XtraReport(); 7 report.LoadLayout(Application.StartupPath + @"\ReportFile\ReporTemplate.repx"); 8 report.DataSource = ds; 9 report.ShowDesignerDialog(); 10 report.Dispose();
做到这里表面上看,基本功能很简单的就实现了,但是我们应该考虑一个问题就是,用户在设置后,我们怎样保存,其实这个也比较简单,采用
1 report.SaveLayout(Application.StartupPath + @"\ReportFile\ReporTemplate1.repx");
保存过程中,注意文件名不能重复,因为本身的模版正在使用中,无法覆盖的。
基本功能就说到这里。说说这里面的问题吧, 用户保存模块文件过多,使用起来不方便,无疑造成用户或者服务器端文件不断增大承载比较重,直到一天崩溃为止。下面说说,通过模版二进制流的方式来完成该功能无疑是最佳选择。
设计思路:
1、数据库中设计一个表,来存储采用二进制byte生成的文件。最好是两个一个是初始化模版,一个是用户设计后的模版。
1 try 2 { 3 string filepath = Application.StartupPath.ToString(); 4 FileStream file = new FileStream(filepath + @"\ReportFile\ReporTemplate.repx", FileMode.Open, FileAccess.ReadWrite); 5 byte[] a = new byte[file.Length]; 6 file.Read(a, 0, (int)file.Length); 7 //这里是我将二进制转换成文本显示的 8 txtContent.Text = Convert.ToBase64String(a); 9 } 10 catch (Exception) 11 { 12 throw; 13 }
这样你就可以获取初始化的模版文件的二进制模版数据,保存到初始化的数据字段中,以后就不用这个了。
2、读取报表二进制数据,加载到模版报表中。如下:
1 using System; 2 using System.Collections.Generic; 3 using System.ComponentModel; 4 using System.Data; 5 using System.Drawing; 6 using System.Text; 7 using System.Windows.Forms; 8 using DevExpress.XtraEditors; 9 using System.IO; 10 using System.Linq; 11 using DevExpress.XtraReports.UI; 12 using DevExpress.XtraReports.UserDesigner; 13 using System.Drawing.Design; 14 using System.ComponentModel.Design; 15 using DevExpress.XtraReports;
1 XtraReport r; 2 private void btnRead_Click(object sender, EventArgs e) 3 { 4 r = new XtraReport(); 5 //二进制流读取转换 6 byte[] ff=Convert.FromBase64String(txtContent.Text.Trim()); 7 System.IO.MemoryStream ms = new MemoryStream(ff); 8 r.LoadLayout(ms); 9 10 XRDesignFormEx designForm = new XRDesignFormEx(); 11 12 //隐藏按钮 这段 13 //designForm.DesignPanel.SetCommandVisibility(new ReportCommand[]{ 14 // ReportCommand.NewReport, 15 // ReportCommand.SaveFileAs, 16 // ReportCommand.NewReportWizard, 17 // ReportCommand.OpenFile 18 //},new DevExpress.XtraReports.UserDesigner.c CommandVisibility.None); 19 20 21 22 //更改状态 23 designForm.ReportStateChanged += new ReportStateEventHandler(designForm_ReportStateChanged); 24 designForm.FormClosing += new FormClosingEventHandler(designForm_FormClosing); 25 // 加载报表. 26 designForm.OpenReport(r); 27 // 打开设计器 28 designForm.ShowDialog(); 29 designForm.Dispose(); 30 } 31 32 void designForm_FormClosing(object sender, FormClosingEventArgs e) 33 { 34 //在此处处理关闭设计器时的操作,主要用来自定义保存数据 35 36 System.IO.MemoryStream ms = new MemoryStream(); 37 r.SaveLayout(ms); 38 byte[] fff = ms.ToArray(); 39 txtContent.Text = Convert.ToBase64String(fff); 40 } 41 42 void designForm_ReportStateChanged(object sender, ReportStateEventArgs e) 43 { 44 //只要报表发生改变就立即将状态设置为保存 45 //避免系统默认保存对话框的出现 46 if (e.ReportState == ReportState.Changed) 47 { 48 ((XRDesignFormEx)sender).DesignPanel.ReportState = ReportState.Saved; 49 } 50 }
实现以上代码需要引用 DevExpress.XtraReports.v11.1.Design、DevExpress.XtraReports.v11.1.Extensions、DevExpress.XtraReports.v11.1 就可以了。
通过以上代码就可以实现了。多说一句,以下代码中,通过System.IO.MemoryStream 内存流的方式获取报表设计界面更改后的二进制数据模版的状态只。
1 r = new XtraReport(); 2 //二进制流读取转换 3 byte[] ff=Convert.FromBase64String(txtContent.Text.Trim()); 4 System.IO.MemoryStream ms = new MemoryStream(ff); 5 r.LoadLayout(ms);