VS2017集成FastReport.Net并将模板保存到数据库_.NET_编程开发_程序员俱乐部

中国优秀的程序员网站程序员频道CXYCLUB技术地图
热搜:
更多>>
 
您所在的位置: 程序员俱乐部 > 编程开发 > .NET > VS2017集成FastReport.Net并将模板保存到数据库

VS2017集成FastReport.Net并将模板保存到数据库

 2017/12/5 17:21:33  缥缈的尘埃  程序员俱乐部  我要评论(0)
  • 摘要:本着开发与实施分离的思想,设计一个通用的报表设计窗体显得尤为重要(下图为图一):要求与优点:I、报表设计窗体支持所有单据调用,一种单据支持多个打印模板。II、报表模板存储在数据库中。一是支持客户端设计及保存模板,二是一次修改所有客户端生效。III、点击保存是将模板保存在数据库中,点击另存为可将模板保存为文件。这样可以实现模板的复制。IV、预览与打印。已设计好的模板不需要每次都进入设计界面,直接预览或打印即可。开发环境:VS2017+SQLSERVER2014+FastReport.Net
  • 标签:模板 .net net 数据库 数据 FastReport

本着开发与实施分离的思想,设计一个通用的报表设计窗体显得尤为重要(下图为图一):

要求与优点:

  I、报表设计窗体支持所有单据调用,一种单据支持多个打印模板。

  II、报表模板存储在数据库中。一是支持客户端设计及保存模板,二是一次修改所有客户端生效。

  III、点击保存是将模板保存在数据库中,点击另存为可将模板保存为文件。这样可以实现模板的复制。

  IV、预览与打印。已设计好的模板不需要每次都进入设计界面,直接预览或打印即可。

开发环境:

VS2017+SQL SERVER 2014+FastReport.Net(2017.1.16)

由于篇幅较多,本次主要分享设计按钮的功能。闲话少说!

1、数据表设计。

CREATE TABLE [dbo].[AT_REPORT](
    [FORMID] [varchar](20) NOT NULL,
    [RPT_NO] [varchar](20) NOT NULL,
    [RPT_NAME] [varchar](50) NULL,
    [FILEDATA] [varbinary](max) NULL,
 CONSTRAINT [PK_AT_REPORT] PRIMARY KEY CLUSTERED 
(
    [FORMID] ASC,
    [RPT_NO] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON, FILLFACTOR = 90) ON [PRIMARY]
) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]

2、新增FastReportDesign窗体(图一):

  1.1引用FastReport库文件。

  

  1.2代码引用。

using System.Data.SqlClient;
using FastReport;
using FastReport.Utils;
using FastReport.Design;

2、定义属性及变量:

说明:FormID是单据ID,即哪种单据调用报表设计窗体则给此属性赋值。RptNo、RptName在点击图一报表种类时赋值。

public string FormID { get; set; } = "PRDT"; //单据ID
private string RptNo, RptName; //报表编号、名称
private DataTable RptTable; //数据表
private DataRow RptRow; //数据行(报表数据源)
private bool isSaveAs = false; //另存为

3、双击设计按钮:

说明:tvwRight是图一右边Treeview的名称。

//设计
        private void btnDes_Click(object sender, EventArgs e)
        {
            if (tvwRight.SelectedNode != null)
            {
                if (!string.IsNullOrEmpty(FormID) && !string.IsNullOrEmpty(RptNo))
                {
                    InitializeReport("DESIGN");
                }
                else
                {
                    MessageBox.Show("报表获取失败。", "信息", MessageBoxButtons.OK, MessageBoxIcon.Information);
                }
            }
            else
            {
                MessageBox.Show("请先选择报表。", "信息", MessageBoxButtons.OK, MessageBoxIcon.Information);
            }
        }

4、初始化方法:

注:mymeans.GetDataSet是自己写的类方法,主要是将SQL生成DataSet。

//初始化报表
        private void InitializeReport(string RptMode)
        {
            DataSet Ds = mymeans.GetDataSet("SELECT RPT_NO,RPT_NAME,FILEDATA FROM AT_REPORT WHERE FORMID='" + FormID + "' AND RPT_NO='" + RptNo + "'", "REPORT");
            RptTable = Ds.Tables[0];
            RptRow = RptTable.Rows[0];
            RegisterDesignerEvents();
            DesignReport(RptMode);
        }

5、注册事件:

说明:FastReport设计器菜单保存及另存为功能,都是将设计模板保存成文件。由于我们需要将设计模板保存到数据库,所以需要屏蔽掉系统原有的功能自己写。另外需要说明的是,保存按钮不会弹出对话框,所以当点击另存为时,才会触发OpenSaveDialogEventHandler。(可参考FastReport自带范例CustomOpenSaveDialogs

//菜单事件注册
        private void RegisterDesignerEvents()
        {
            Config.DesignerSettings.CustomSaveDialog += new OpenSaveDialogEventHandler(DesignerSettings_CustomSaveDialog);
            Config.DesignerSettings.CustomSaveReport += new OpenSaveReportEventHandler(DesignerSettings_CustomSaveReport);
        }

 6、设计模板加载:

//设计报表
        private void DesignReport(string RptMode)
        {
            using (Report TargetReport = new Report())
            {
                TargetReport.FileName = RptName;
                if (RptRow["FILEDATA"].ToString().Length > 0)
                {
                    byte[] ReportBytes = (byte[])RptRow["FILEDATA"];
                    using (MemoryStream Stream = new MemoryStream(ReportBytes))
                    {
                        TargetReport.Load(Stream);
                    }
                }
                //操作方式:DESIGN-设计;PREVIEW-预览;PRINT-打印
                if (RptMode == "DESIGN")
                {
                    TargetReport.Design();
                }
                else if (RptMode == "PREVIEW")
                {
                    TargetReport.Prepare();
                    TargetReport.ShowPrepared();
                }
                else if (RptMode == "PRINT")
                {
                    TargetReport.Print();
                }
            }
        }

7、另存为对话框:

说明:isSaveAs为true时,表时点击的是另存为按钮。

//保存菜单:对话框
        private void DesignerSettings_CustomSaveDialog(object sender, OpenSaveDialogEventArgs e)
        {
            isSaveAs = true;
        }

8、保存委托函数:

//保存菜单:委托函数
        private void DesignerSettings_CustomSaveReport(object sender, OpenSaveReportEventArgs e)
        {
            SaveReport(e.Report);
        }

 9、报表模板保存:

说明:mymeans.ConOpen()是自己写的类方法,主要是连接数据库。

//保存报表
        private void SaveReport(Report TargetReport)
        {
            try
            {
                using (MemoryStream msStream = new MemoryStream())
                {
                    //保存
                    TargetReport.Save(msStream);
                    RptRow["FILEDATA"] = msStream.ToArray();
                    if (MyMeans.Con == null || MyMeans.Con.State != ConnectionState.Open)
                    {
                        mymeans.ConOpen();
                    }
                    SqlCommand Cmd = MyMeans.Con.CreateCommand();
                    Cmd.CommandText = "UPDATE AT_REPORT SET FILEDATA=@FILEDATA WHERE FORMID=@FORMID AND RPT_NO=@RPT_NO";
                    Cmd.Parameters.AddWithValue("@FILEDATA", msStream.ToArray());
                    Cmd.Parameters.AddWithValue("@FORMID", FormID);
                    Cmd.Parameters.AddWithValue("@RPT_NO", RptNo);
                    Cmd.ExecuteNonQuery();
                    //另存为
                    if(isSaveAs==true)
                    {
                        SaveFileDialog saveFileDialog = new SaveFileDialog();
                        //设置文件类型
                        saveFileDialog.Filter = "报表文件(*.frx)|*.frx|C# 文件(*.cs)|*.cs|FastReport VCL report(*.fr3)|*.fr3|RDL file(*rdl)|*.rdl";
                        //设置默认文件类型显示顺序  
                        saveFileDialog.FilterIndex = 1;
                        //是否自动在文件名中添加扩展名
                        saveFileDialog.AddExtension = true;
                        //是否记忆上次打开的目录
                        saveFileDialog.RestoreDirectory = true;
                        //设置默认文件名
                        saveFileDialog.FileName = RptName;
                        //按下确定选择的按钮  
                        if (saveFileDialog.ShowDialog() == DialogResult.OK)
                        {
                            //获得文件路径 
                            string localFilePath = saveFileDialog.FileName.ToString();
                            //内存流转字节数组
                            byte[] arrBuffer= new byte[msStream.Length];
                            msStream.Position = 0;
                            int intBuffer = msStream.Read(arrBuffer, 0, arrBuffer.Length);
                            msStream.Write(arrBuffer, 0, intBuffer);
                            //文件保存
                            FileStream fsStream = new FileStream(localFilePath, FileMode.Create);
                            msStream.WriteTo(fsStream);
                            //资源释放      
                            fsStream.Close();
                            fsStream = null;
                        }
                        //赋初始值
                        isSaveAs = false;
                    }
                }
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
            }
        }

好了,主要的功能就分享到此,希望对大家有一些帮助。

发表评论
用户名: 匿名