VS2017集成FastReport.Net並將模板保存到資料庫

来源:http://www.cnblogs.com/atomy/archive/2017/12/05/7986996.html
-Advertisement-
Play Games

本著開發與實施分離的思想,設計一個通用的報表設計窗體顯得尤為重要(下圖為圖一): 要求與優點: I、報表設計窗體支持所有單據調用,一種單據支持多個列印模板。 II、報表模板存儲在資料庫中。一是支持客戶端設計及保存模板,二是一次修改所有客戶端生效。 III、點擊保存是將模板保存在資料庫中,點擊另存為可 ...


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

要求與優點:

  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();
                            //文件保存
                            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);
            }
        }

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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 題目描述 永無鄉包含 n 座島,編號從 1 到 n,每座島都有自己的獨一無二的重要度,按照重要度可 以將這 n 座島排名,名次用 1 到 n 來表示。某些島之間由巨大的橋連接,通過橋可以從一個島 到達另一個島。如果從島 a 出發經過若幹座(含 0 座)橋可以到達島 b,則稱島 a 和島 b 是連 通 ...
  • S#語言的最全能類型——字元串(對應C#的String),可用於表示文本內容,如"S#公式是很有特色"等。S#的字元串輸入格式有三種:"xxxxx",@"xxxxx"和'xxxxx'。在S#語言設計時字元串的地位是很高的,系統把它也看成是“程式即數據、數據即程式”的全能表達方式之一。 ...
  • 最近要做個winform的東西,要在裡面集成一個類似Windows自帶畫圖的標尺功能,還要能在圖片上畫矩形框。在網上找了好久也沒找到寫好的控制項,無奈自己做了一個。 目前還有些bug,這裡先做個分享。(Ps:很少做winform的東西,做的不好,輕噴。另外歡迎指點。) 由於最後要做的東西的背景是黑的, ...
  • 東西不是很複雜,不過百度出來的,貌似都是一種,代碼太長了,複製都不想複製,來個簡易版本的吧,直接貼代碼。 ...
  • 本節將分析 代碼。 源代碼參考.NET Core 2.0.0 "WebHostBuilder" "WebHost" "Kestrel" 問題概要 1. Hosting中有哪2個ServiceProvider,各自如何創建,以及有哪些ServiceCollection。 1. 什麼時候執行Startu ...
  • Markdown是一種可以使用普通文本編輯器編寫的標記語言,通過簡單的標記語法,它可以使普通文本內容具有一定的格式。 ...
  • Dapper介紹 Dapper是.Net下的一個輕量級ORM框架.在小型工具向的項目下,使用Dapper會使資料庫操作層代碼更加優雅. Dapper的使用 在項目中使用引用Dapper非常簡單,你可以選擇兩種方式: 1.在NuGet引用 2.Github下載源代碼,添加到項目當中去 Dapper 項 ...
  • 開發C#的程式,寫到屬性property時,我們可以在Set方法中做一些簡單的規則驗證:如下麵,Insus.NET寫一個Age屬性,只允許用戶輸入10以內的數字: class AA { private int _Age; public int Age { get { return _Age; } s ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...