Step 1: Create a report file
Create a report file (rpt_get_employee_info_dept_id_team_id.rpt).
Write a stored procedure (rsp_get_employee_info_by_dept_id_team_id) to retrieve
data. Set the stored procedure as crystal report data source. Design crystal
reports.
Step 2: Add Assemblies
Add the following references
to your web site. Version of the assembly is most important. I used assembly
version: 13.0.2000.0
CrystalDecisions.CrystalReports.Engine;
CrystalDecisions.Shared;
CrystalDecisions.Web
Step 3: Create a report base class
This is a custom class,
responsible to generate crystal report document. Some global variable is used
in this class for configuration purpose. Server name, database name, user id,
password is hard coded here for easily understand but I recommend, it should
come from web configuration file.
string ServerName = "MAHEDEE-PC"; //Database server name to which report connected string DataBaseName = "TestDB"; //Database name to which report connected string UserID = "sa"; //Database user name to which report connected string Password = "sa"; //Database user password to which report connected ReportDocument crReportDocument = new ReportDocument(); //Crystal report document object
In this class some methods are used for different purposes.
- public ReportDocument PFSubReportConnectionMainParameter(string ReportName, ArrayList ParamArraylist, string ReportFolder) - This is the base method to generate report document
- public void PassParameter(int ParameterIndex, string ParameterValue) - Set parameter value in crystal report on corresponding index
- private bool Logon(CrystalDecisions.CrystalReports.Engine.ReportDocument cr, string server, string database, string user_id, string password) – Cystal report login check to the server
- private bool ApplyLogon(CrystalDecisions.CrystalReports.Engine.ReportDocument cr, CrystalDecisions.Shared.ConnectionInfo ci) – Supporting method of “Logon” to login check.
using System; using System.Collections.Generic; using System.Linq; using System.Web; using CrystalDecisions.CrystalReports.Engine; using System.Collections; using CrystalDecisions.Shared; ////// The base class to generate report document /// public class ReportBase { string ServerName = "MAHEDEE-PC"; //Database server name to which report connected string DataBaseName = "TestDB"; //Database name to which report connected string UserID = "sa"; //Database user name to which report connected string Password = "sa"; //Database user password to which report connected ReportDocument crReportDocument = new ReportDocument(); public ReportBase() { } //////Base method to generate report document. /// public ReportDocument PFSubReportConnectionMainParameter(string ReportName, ArrayList ParamArraylist, string ReportFolder) { int ObjArrayA, ObjArrayB, Paraindex; string Paravalue; ObjArrayB = 0; string path = ReportFolder + @"\" + @ReportName; //Full path of report try { crReportDocument.Load(path); //Load crystal Report } catch (Exception ex) { string msg = "The report file " + path + " can not be loaded, ensure that the report exists or the path is correct." + "\nException:\n" + ex.Message + "\nSource:" + ex.Source + "\nStacktrace:" + ex.StackTrace; throw new Exception(msg); } //Ensure login to the database server if (!Logon(crReportDocument, ServerName, DataBaseName, UserID, Password)) { string msg = "Can not login to Report Server " + "\nDatabase Server: " + ServerName + "\nDatabase:\n" + DataBaseName + "\nDBUser:" + UserID + "\nDBPassword:" + Password; throw new Exception(msg); } Logon(crReportDocument, ServerName, DataBaseName, UserID, Password); //To Check Parameter Feild Array have the Same Amount of Parameter Feild in the Report int ParamArrayCount, ParameterFieldCount; //Getting Value from the Array if (ParamArraylist.Count != 0) { ParamArrayCount = (ParamArraylist.Count / 2); //Getting Value From the Report (Parameter and Formula Feild) ParameterFieldCount = crReportDocument.DataDefinition.ParameterFields.Count; //Parameter on The Report and Array List Parameter Amount is not the same ParamArrayCount = ParameterFieldCount; for (ObjArrayA = 0; ObjArrayA < ((ParamArraylist.Count / 2)); ObjArrayA++) { Paraindex = (int)ParamArraylist[ObjArrayB]; //Parameter index Paravalue = (string)ParamArraylist[ObjArrayB + 1]; //Paramter Value PassParameter(Paraindex, Paravalue); ObjArrayB = ObjArrayB + 2; } } return crReportDocument; } ////// Set parameter value in crystal report on corresponding index /// public void PassParameter(int ParameterIndex, string ParameterValue) { // ' // ' Declare the parameter related objects. // ' ParameterDiscreteValue crParameterDiscreteValue; ParameterFieldDefinitions crParameterFieldDefinitions; ParameterFieldDefinition crParameterFieldLocation; ParameterValues crParameterValues; crParameterFieldDefinitions = crReportDocument.DataDefinition.ParameterFields; crParameterFieldLocation = (ParameterFieldDefinition)crParameterFieldDefinitions[ParameterIndex]; crParameterValues = crParameterFieldLocation.CurrentValues; crParameterDiscreteValue = new CrystalDecisions.Shared.ParameterDiscreteValue(); crParameterDiscreteValue.Value = System.Convert.ToString(ParameterValue); crParameterValues.Add(crParameterDiscreteValue); crParameterFieldLocation.ApplyCurrentValues(crParameterValues); } //Check whether crytal report can login to the server private bool Logon(CrystalDecisions.CrystalReports.Engine.ReportDocument cr, string server, string database, string user_id, string password) { // Declare and instantiate a new connection info object. CrystalDecisions.Shared.ConnectionInfo ci; ci = new CrystalDecisions.Shared.ConnectionInfo(); ci.ServerName = server; ci.DatabaseName = database; ci.UserID = user_id; ci.Password = password;//password; // ci.IntegratedSecurity = false; // If the ApplyLogon function fails then return a false for this function. // We are applying logon information to the main report at this stage. if (!ApplyLogon(cr, ci)) { return false; } // Declare a subreport object. CrystalDecisions.CrystalReports.Engine.SubreportObject subobj; // Loop through all the report objects and locate subreports. // If a subreport is found then apply logon information to // the subreport. foreach (CrystalDecisions.CrystalReports.Engine.ReportObject obj in cr.ReportDefinition.ReportObjects) { if (obj.Kind == CrystalDecisions.Shared.ReportObjectKind.SubreportObject) { subobj = (CrystalDecisions.CrystalReports.Engine.SubreportObject)obj; if (!ApplyLogon(cr.OpenSubreport(subobj.SubreportName), ci)) { return false; } } } // Return True if the code runs to this stage. return true; } //////This function is called by the "Logon" function. It loops through the report tables and applies the connection information to each table. /// private bool ApplyLogon(CrystalDecisions.CrystalReports.Engine.ReportDocument cr, CrystalDecisions.Shared.ConnectionInfo ci) { // This function is called by the "Logon" function // It loops through the report tables and applies // the connection information to each table. // Declare the TableLogOnInfo object and a table object for use later. CrystalDecisions.Shared.TableLogOnInfo li; // For each table apply connection info. foreach (CrystalDecisions.CrystalReports.Engine.Table tbl in cr.Database.Tables) { li = tbl.LogOnInfo; li.ConnectionInfo.ServerName = ci.ServerName; li.ConnectionInfo.DatabaseName = ci.DatabaseName; li.ConnectionInfo.UserID = ci.UserID; li.ConnectionInfo.Password = ci.Password; tbl.ApplyLogOnInfo(li); tbl.Location = ci.DatabaseName + ".dbo." + tbl.Name; // Verify that the logon was successful. // If TestConnectivity returns false, correct table locations. if (!tbl.TestConnectivity()) { return false; } } return true; } }
Step 4: Create Report Viewer pages
In this project, I have created
a report viewer page – ViewReport.aspx. Here some steps need to follow.
a. 
Register Assembly – Register assembly
CrystalDecisions.Web to the page directive
<%@ Register Assembly="CrystalDecisions.Web,
Version=13.0.2000.0, Culture=neutral, PublicKeyToken=692fbea5521e1304" Namespace="CrystalDecisions.Web" TagPrefix="CR" %>
b. 
Add crystal report viewer – Add crystal report
viewer to view report. 
<CR:CrystalReportViewer ID="CrystalReportViewer1" runat="server" AutoDataBind="True" HasPrintButton="True" HasRefreshButton="True" ReuseParameterValuesOnRefresh="True" Height="50px" Width="350px" OnReportRefresh="CrystalReportViewer1_ReportRefresh" PrintMode="ActiveX" />
c. 
Manually config CSS file - Sometimes crystal
report shows blank page due to some technical problem so add the following css
link in between head tag.
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
d. 
A button (btnViewReport) is used to display
report
Source file of ViewReport.aspx
<%@ Page Language="C#" AutoEventWireup="true" CodeFile="ViewReport.aspx.cs" Inherits="ViewReport" %>
<%@ Register Assembly="CrystalDecisions.Web,
Version=13.0.2000.0, Culture=neutral, PublicKeyToken=692fbea5521e1304"
    Namespace="CrystalDecisions.Web" TagPrefix="CR" %>
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <title></title>
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
    <link href="/aspnet_client/System_Web/4_0_30319/CrystalReportWebFormViewer3/css/default.css"
        rel="stylesheet" type="text/css" />
 </head>
<body>
    <form id="form1" runat="server">
        <div>
            <table>
               
<tr>
                   
<td>
                        <asp:Button ID="btnViewReport" Text="View Report" runat="server" OnClick="btnViewReport_Click" /></td>
               
</tr>
            </table>
            <br />
            <div style="height: 500px; width: 500px;">
               
<CR:CrystalReportViewer ID="CrystalReportViewer1" runat="server" AutoDataBind="True" HasPrintButton="True" HasRefreshButton="True" ReuseParameterValuesOnRefresh="True" Height="50px" Width="350px" OnReportRefresh="CrystalReportViewer1_ReportRefresh" PrintMode="ActiveX" />
            </div>
        </div>
    </form>
</body>
</html>
Code behind of ViewReport.aspx
Some methods are used in code
behind file for different purpose. 
- protected void btnViewReport_Click(object sender, EventArgs e) – Click event handler of btnViewReport button. In this method two parameters of crystal report are used as hardcoded. In real application, it should come as input control dynamically.
- private void GetReportDocument() – Responsible to generate report document using ReportBase class.
- private void ViewCystalReport() - Set generated report document as Crystal Report viewer report source and display report in report viewer.
using CrystalDecisions.CrystalReports.Engine;
using CrystalDecisions.Web;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Drawing;
using System.Linq;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
public partial class ViewReport : System.Web.UI.Page
{
    
    ArrayList ParameterArrayList = new ArrayList(); //Report parameter list
    ReportDocument ObjReportClientDocument = new ReportDocument(); //Report document
    protected void Page_Load(object sender, EventArgs e)
    {
    }
    protected void btnViewReport_Click(object sender, EventArgs e)
    {
        /*The report with two parameters. */
        ParameterArrayList.Add(0);
        ParameterArrayList.Add("1"); //Parameter 1 with input 1. This is actually dept id according to report parameter
        ParameterArrayList.Add(1); 
        ParameterArrayList.Add("1"); //Parameter 2 with input 1. This is actually team id according to report parameter
        GetReportDocument(); //Generate Report document
        ViewCystalReport();  //View report document in crystal report viewer
    }
    /*Generate report document*/
    private void GetReportDocument()
    {
        ReportBase objReportBase = new ReportBase();
        string sRptFolder = string.Empty;
        string sRptName = string.Empty;
        sRptName = "rpt_get_employee_info_dept_id_team_id.rpt"; //Report name
        sRptFolder = Server.MapPath("~/Reports");  //Report folder name
        ObjReportClientDocument = objReportBase.PFSubReportConnectionMainParameter(sRptName, ParameterArrayList, sRptFolder);
        
        //This section is for manipulating font and font size. This an optional section 
        foreach (Section oSection in ObjReportClientDocument.ReportDefinition.Sections)
        {
            foreach (ReportObject obj in oSection.ReportObjects)
            {
                FieldObject field;
                field = ObjReportClientDocument.ReportDefinition.ReportObjects[obj.Name] as FieldObject;
                if (field != null)
                {
                    Font oFon1 = new Font("Arial Narrow", field.Font.Size - 1F);
                    Font oFon2 = new Font("Arial", field.Font.Size - 1F);
                    if (oFon1 != null)
                    {
                        field.ApplyFont(oFon1);
                    }
                    else if (oFon2 != null)
                    {
                        field.ApplyFont(oFon2);
                    }
                }
            }
        }
    }
    protected void CrystalReportViewer1_ReportRefresh(object source, ViewerEventArgs e)
    {
        //OnInit(e);
        //ViewCystalReport();
    }
    ///    
    /// To view crystal report
    ///     
    private void ViewCystalReport()
    {
        //Set generated report document as Crystal Report viewer report source
        CrystalReportViewer1.ReportSource = ObjReportClientDocument;
    }
}
Now your report is
ready to view in asp.net. Just click on the “View Report” button, you will see
the report.
