使用Aspose.Cell控件实现Excel高难度报表的生成(二)

继续在上篇《使用Aspose.Cell控件实现Excel高难度报表的生成(一)》随笔基础上,研究探讨基于模板的Aspose.cell报表实现,其中提到了下面两种报表的界面,如下所示:

 或者这样的报表格式

  

首先来分析第一种报表,这个其实还是比较固定的二维表,我们只要绑定相关的信息即可,设计模板如下所示:

 
实际生成的报表如下所示:

 

实现的代码其实不复杂,如下所示:

         private DataTable GetTable(string sql)

        {
            Database db = DatabaseFactory.CreateDatabase();
            DbCommand command = db.GetSqlStringCommand(sql);
            return db.ExecuteDataSet(command).Tables[0];
        }
        private void btnAllMonthReport_Click(object sender, EventArgs e)
        {
            string sql = @"Select [LastCount] as LC, [LastMoney] as LM, [CurrentInCount] as CIC, [CurrentInMoney] as CIM, 
                           [CurrentOutCount] as COC, [CurrentOutMoney] as COM, [CurrentCount] as CC, [CurrentMoney] as CM, 
                           YearMonth,ItemName 
                           from TB_ReportMonthCheckOut ";
            DataTable dtBigType = GetTable(sql + " where ReportType =3");
            dtBigType.TableName = "BigType";
            if (dtBigType.Rows.Count == 0)
                return;

            DataTable dtItemType = GetTable(sql + " where ReportType =3");
            dtItemType.TableName = "ItemType";

            WorkbookDesigner designer = new WorkbookDesigner();
            string path = System.IO.Path.Combine(Application.StartupPath, "Report2-1.xls");
            designer.Open(path);
            designer.SetDataSource(dtBigType);
            designer.SetDataSource(dtItemType);
            designer.SetDataSource("YearMonth", dtBigType.Rows[0]["YearMonth"].ToString());
            designer.Process();

            //Save the excel file
            string fileToSave = FileDialogHelper.SaveExcel();
            if (File.Exists(fileToSave))
            {
                File.Delete(fileToSave);
            }
            designer.Save(fileToSave, FileFormatType.Excel2003);
            Process.Start(fileToSave);
        }

通过绑定两个不同的DataTable对象,然后引用他的属性即可,行会自动增加以适应实际的数据,并且对象变量&=$YearMonth也正常显示了,注意一点就是,所有使用变量的地方,都必须在一个独立的Excel单元格中,否则不能解析出来。另外上图的红色圆圈里面表示,汇总的函数,会自动根据行列的增加,自动调整引用,这真是我们需要的。 

 出库单的实现也差不多,实现代码如下所示:

            string TakeOutBill = Path.Combine(Application.StartupPath, "TakeOutBill.xls");
            WorkbookDesigner designer = new WorkbookDesigner();
            designer.Open(TakeOutBill);
            designer.SetDataSource("TakeOutDate", DateTime.Now.ToString("yyyy-MM-dd"));
            designer.SetDataSource("WareHouse", this.txtWareHouse.Text);
            designer.SetDataSource("Manager", this.txtCreator.Text);
            designer.SetDataSource("CostCenter", this.txtCostCenter.Text);
            designer.SetDataSource("Dept", this.txtDept.Text);

            string columns = "Start|int,ItemNo,ItemName,Specification,Unit,Price|decimal,Count|int";
            DataTable dt = DataTableHelper.CreateTable(columns);
            dt.TableName = "Detail";
            DataRow row = null;
            for (int i = 0; i < this.lvwDetail.Items.Count; i++)
            {
                PurchaseDetailInfo info = this.lvwDetail.Items[i].Tag as PurchaseDetailInfo;
                if (info != null)
                {
                    row = dt.NewRow();
                    row["Start"] = (i + 1);
                    row["ItemNo"] = info.ItemNo;
                    row["ItemName"] = info.ItemName;
                    row["Specification"] = info.Specification;
                    row["Unit"] = info.Unit;
                    row["Price"] = info.Price;
                    row["Count"] = Math.Abs(info.Quantity);
                    dt.Rows.Add(row);
                }
            }
            designer.SetDataSource(dt);
            designer.Process();
            string fileToSave = FileDialogHelper.SaveExcel();
            if (File.Exists(fileToSave))
            {
                File.Delete(fileToSave);
            }
            designer.Save(fileToSave, FileFormatType.Excel2003);
            Process.Start(fileToSave);

以上报表,其实实现思路基本都差不多,相对来时,还是比较容易的,接下来设计一个比较困难的报表,需要结合Aspose.Cell一些对象来动态创建行列,并设置单元格的变量,然后填入相应的对象构造报表,另外还需要注意单元格格式的变化,如下所示的这种报表

 

这个报表初看没有太多特别的地方,难点就是他的第一行列也是变化的,因此不能通过普通的方式构建二维表,然后绑定数据源的方式,要先加载模板文件,然后操作Excel对象,把第一行的各列头部补齐,然后给下一行各单元格填入对象公式,如&=BigType.DeptName 和&=BigType.TotalMoney等内容,实现的代码如下所示:

            string sql = @"Select [YearMonth], [DeptName], [ItemType], [TotalMoney]
                           from TB_ReportDeptCost ";
            DataTable dt = GetTable(sql);
            if (dt.Rows.Count == 0)
                return;

            List<string> itemTypeList = new List<string>();
            List<string> partList = new List<string>();
            foreach (DataRow row in dt.Rows)
            {
                string itemType = row["ItemType"].ToString();
                if (!itemTypeList.Contains(itemType))
                {
                    itemTypeList.Add(itemType);
                }

                string part = row["DeptName"].ToString();
                if (!partList.Contains(part))
                {
                    partList.Add(part);
                }
            }

            string columnString = "DeptName";
            for (int i = 0; i < itemTypeList.Count; i++)
            {
                columnString += string.Format(",TotalMoney{0}|decimal", i);
            }
            DataTable dtBigType = DataTableHelper.CreateTable(columnString);
            dtBigType.TableName = "BigType";
            foreach (string part in partList)
            {
                DataRow row = dtBigType.NewRow();
                row["DeptName"] = part;
                for (int i = 0; i < itemTypeList.Count; i++)
                {
                    string itemType = itemTypeList[i];
                    DataRow[] rowSelect = dt.Select(string.Format("DeptName='{0}' AND ItemType='{1}'", part, itemType));
                    if (rowSelect.Length > 0)
                    {
                        row["TotalMoney" + i.ToString()] = rowSelect[0]["TotalMoney"];
                    }
                }
                dtBigType.Rows.Add(row);
            }

            WorkbookDesigner designer = new WorkbookDesigner();
            string path = System.IO.Path.Combine(Application.StartupPath, "Report1.xls");
            designer.Open(path);

            Aspose.Cells.Worksheet w = designer.Workbook.Worksheets[0];
            //先设置标题项目:如大修件,日常备件等
            int rowIndex = 2;//第三行为标题
            Aspose.Cells.Style style = w.Cells[rowIndex + 1, 1].Style;//继承数字栏目的样式
            style.Number = 4;//对应格式是#,##0.00
            Aspose.Cells.Style boldStyle = w.Cells[rowIndex, 0].Style;//继承开始栏目的样式
            for (int i = 0; i < itemTypeList.Count; i++)
            {
                w.Cells[rowIndex, i + 1].PutValue(itemTypeList[i]);
                w.Cells[rowIndex, i + 1].Style = boldStyle;
                w.Cells[rowIndex + 1, i + 1].PutValue("&=BigType.TotalMoney" + i.ToString());
                w.Cells[rowIndex + 1, i + 1].Style = style;
            }

            //添加合计行
            w.Cells[rowIndex, itemTypeList.Count + 1].PutValue("合计");
            w.Cells[rowIndex, itemTypeList.Count + 1].Style = boldStyle;
            w.Cells[rowIndex + 1, itemTypeList.Count + 1].PutValue(string.Format("&=&=SUM(B{{r}}:{0}{{r}})", GetChar(itemTypeList.Count + 1)));
            w.Cells[rowIndex + 1, itemTypeList.Count + 1].Style = style;

            designer.SetDataSource(dtBigType);
            designer.SetDataSource("YearMonth", dt.Rows[0]["YearMonth"].ToString());
            designer.Process();

            //Save the excel file
            string fileToSave = FileDialogHelper.SaveExcel();
            if (File.Exists(fileToSave))
            {
                File.Delete(fileToSave);
            }
            designer.Save(fileToSave, FileFormatType.Excel2003);
            Process.Start(fileToSave);  

 本文转自博客园伍华聪的博客,原文链接:使用Aspose.Cell控件实现Excel高难度报表的生成(二),如需转载请自行联系原博主。

时间: 2024-10-20 01:15:59

使用Aspose.Cell控件实现Excel高难度报表的生成(二)的相关文章

使用Aspose.Cell控件实现Excel高难度报表的生成(三)

在之前几篇文章中,介绍了关于Apsose.cell这个强大的Excel操作控件的使用,相关文章如下: 使用Aspose.Cell控件实现Excel高难度报表的生成(一) 使用Aspose.Cell控件实现Excel高难度报表的生成(二) 使用Aspose.Cell控件实现多个Excel文件的合并 这几篇文章,都对Apose.Cell这个控件生成各种Excel的方式进行了阐述,对直接把DataTable或者IList生成Excel的操作,对通过模板方式实现自定义报表的各种方式,以及多个文件的合并的

利用Aspose.Cell控件导入Excel非强类型的数据

导入Excel的操作是非常常见的操作,可以使用Aspose.Cell.APOI.MyXls.OLEDB.Excel VBA等操作Excel文件,从而实现数据的导入,在导入数据的时候,如果是强类型的数据,那么这几种方式好像都表现差不多,正常操作能够导入数据.如果是非强类型的数据,那么就需要特别注意了,一般情况下,导入的DataTable数据列的类型是以第一行内容作为确定列类型的,本文介绍利用Aspose.Cell控件导入Excel非强类型的数据的操作. 什么是强类型的数据呢,就是在Excel表格中

利用Aspose.Word控件和Aspose.Cell控件,实现Word文档和Excel文档的模板化导出

我们知道,一般都导出的Word文档或者Excel文档,基本上分为两类,一类是动态生成全部文档的内容方式,一种是基于固定模板化的内容输出,后者在很多场合用的比较多,这也是企业报表规范化的一个体现. 我的博客介绍过几篇关于Aspose.Word控件和Aspose.Cell控件的使用操作,如下所示. <使用Aspose.Cell控件实现Excel高难度报表的生成(一)> <使用Aspose.Cell控件实现Excel高难度报表的生成(二)> <使用Aspose.Cell控件实现Ex

使用Aspose.Cell控件实现多个Excel文件的合并

之前有写过多篇关于使用Apose.Cell控件制作自定义模板报表和通用的导出Excel表格数据的操作,对这个控件的功能还是比较满意,而且也比较便利.忽然有一天,一个朋友说:你已经有生成基于自定义模板报表了,可是我每个单位都导出一张相同的报表的话,我岂不是要生成很多文件,而且对比查看也不方便,有没有更好的办法合并他们到一个文件里面呢?这样我看报表就方便很多了.本文主要介绍如何实现基于一个自定义报表模式,生成多个类似报表合并在一个文件中具体操作. 查询Apose.Cell控件的使用介绍,WorkBo

利用Aspose.Word控件实现Word文档的操作

原文:利用Aspose.Word控件实现Word文档的操作 Aspose系列的控件,功能都挺好,之前一直在我的Winform开发框架中用Aspose.Cell来做报表输出,可以实现多样化的报表设计及输出,由于一般输出的内容比较正规化或者多数是表格居多,所以一般使用Aspose.Cell来实现我想要的各种Excel报表输出.虽然一直也知道Aspose.Word是用来生成Word文档的,而且深信其也是一个很强大的控件,但一直没用用到,所以就不是很熟悉. 偶然一次机会,一个项目的报表功能指定需要导出为

关于控件导出Excel格式问题的新解决方案

今天弄了一下控件导出Excel,于是产生了格式问题了,一些日期如20091222的名称被显示为科学记数法. 所以,我搜了博客园.一大堆的文章都是复制来复制去的资料. 基本上都有这么一行闪亮的代码:文本:vnd.ms-excel.numberformat:@ 有深入一点的文章,会告诉你,这个东西要写在<td style="vnd.ms-excel.numberformat:@">xxx</td>里 于是这里就产生这两个分支方法: 1.直接就写上<td sty

Android编程获取控件宽和高的方法总结分析_Android

本文总结分析了Android编程获取控件宽和高的方法.分享给大家供大家参考,具体如下: 我们都知道在onCreate()里面获取控件的高度是0,这是为什么呢?我们来看一下示例: 首先我们自己写一个控件,这个控件非常简单: public class MyImageView extends ImageView { public MyImageView(Context context, AttributeSet attrs) { super(context, attrs); } public MyIm

calendar-我在日历控件里的每一天动态生成一个按钮,但是事件不生效是怎么回事

问题描述 我在日历控件里的每一天动态生成一个按钮,但是事件不生效是怎么回事 我在Calendar控件里的每一天动态生成一个LinkButton,显示农历且可以点击,但是Command事件不生效是怎么回事? 这是代码,农历能正常显示,但是LinkButton不能点击 //日的读取 protected void Calendar1_DayRender(object sender, DayRenderEventArgs e) { //农历 LinkButton lbChinaDate = new Li

导出CListCtrl控件到Excel文件

CListCtrl获得列的属性可以使用BOOL CListCtrl::GetColumn( int nCol, LVCOLUMN* pColumn ),nCol为需要获得获得属性值的列序号,pColumn 为 LVCOLUMN 结构体的指针.LVCOLUMN结构体中的 UINT mask 作为输入,决定返回哪些属性的值,如果 mask 的值包含 LVCF_TEXT ,则需要将字符串缓存的首地址指针置入结构体中的 LPTSTR pszText; 变量,缓存大小置入 int cchTextMax;