VS2017集成FastReport.Net並將模板保存到數據庫

本着開發與實施分離的思想,設計一個通用的報表設計窗體顯得尤其重要(下圖爲圖一):數據庫

要求與優勢:函數

  I、報表設計窗體支持全部單據調用,一種單據支持多個打印模板。spa

  II、報表模板存儲在數據庫中。一是支持客戶端設計及保存模板,二是一次修改全部客戶端生效。設計

  III、點擊保存是將模板保存在數據庫中,點擊另存爲可將模板保存爲文件。這樣能夠實現模板的複製。code

  IV、預覽與打印。已設計好的模板不須要每次都進入設計界面,直接預覽或打印便可。orm

開發環境:blog

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

因爲篇幅較多,本次主要分享設計按鈕的功能。閒話少說!資源

一、數據表設計。開發

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]

二、新增FastReportDesign窗體(圖一):

  1.1引用FastReport庫文件。

  

  1.2代碼引用。

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

二、定義屬性及變量:

說明: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; //另存爲

三、雙擊設計按鈕:

說明: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);
            }
        }

四、初始化方法:

注: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);
        }

五、註冊事件:

說明:FastReport設計器菜單保存及另存爲功能,都是將設計模板保存成文件。因爲咱們須要將設計模板保存到數據庫,因此須要屏蔽掉系統原有的功能本身寫。另外須要說明的是,保存按鈕不會彈出對話框,因此當點擊另存爲時,纔會觸發OpenSaveDialogEventHandler。(可參考FastReport自帶範例CustomOpenSaveDialogs)

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

 六、設計模板加載:

//設計報表
        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();
                }
            }
        }

七、另存爲對話框:

說明:isSaveAs爲true時,代表點擊的是另存爲按鈕。

//保存菜單:對話框
        private void DesignerSettings_CustomSaveDialog(object sender, OpenSaveDialogEventArgs e)
        {
            isSaveAs = true;
        }

八、保存委託函數:

//保存菜單:委託函數
        private void DesignerSettings_CustomSaveReport(object sender, OpenSaveReportEventArgs e)
        {
            SaveReport(e.Report);
        }

 九、報表模板保存:

說明:mymeans.ConOpen()是本身寫的類方法,主要是鏈接數據庫。

 
 
//保存報表
        private void SaveReport(Report TargetReport)
        {
            try
            {
                using (MemoryStream msStream = new MemoryStream())
                {
                    //解決困擾多時的屢次保存問題。
                    Config.DesignerSettings.CustomSaveDialog -= new OpenSaveDialogEventHandler(DesignerSettings_CustomSaveDialog);
                    Config.DesignerSettings.CustomSaveReport -= new OpenSaveReportEventHandler(DesignerSettings_CustomSaveReport);
                    //保存
                    TargetReport.Save(msStream);
                    RptRow["FILEDATA"] = msStream.ToArray();
                    if (FrameClass.MyMeans.Con == null || FrameClass.MyMeans.Con.State != ConnectionState.Open)
                    {
                        mymeans.ConOpen();
                    }
                    SqlCommand Cmd = FrameClass.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";
                        //設置默認文件類型顯示順序  
                        saveFileDialog.FilterIndex = 1;
                        //是否自動在文件名中添加擴展名
                        saveFileDialog.AddExtension = true;
                        //是否記憶上次打開的目錄
                        saveFileDialog.RestoreDirectory = true;
                        //設置默認文件名
                        saveFileDialog.FileName = RptName;
                        //按下肯定選擇的按鈕  
                        if (saveFileDialog.ShowDialog() == DialogResult.OK)
                        {
                            //得到文件路徑 
                            string localFilePath = saveFileDialog.FileName.ToString();
                            //文件保存
                            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);
            }
        }

好了,主要的功能就分享到此,但願對你們有一些幫助。

相關文章
相關標籤/搜索