2023年2月

一直以来,我的博客几乎一直在介绍Winform相关的技术,其中包括Winform的分页控件,其实Web方面的开发一直也在从事,只是较少写这方面的内容而已,经过多年的积累及提炼,也逐渐形成了我自己特色的一套Web开发框架。最近对Web分页控件进行了优化提炼,本文就从Web分页控件进行介绍,让大家了解这个体系中的一部分,也希望提出宝贵的意见,共同探讨。

分页控件的主要特点有:

1)内置支持Oracle、SqlServer、SQlite、MySql、Access数据库,只需要指定数据库类型即可自动调整。

2)与列表控件(DataGrid)分开,支持更多的界面特性。使得DataGrid控件展现更加丰富。

3)列表支持字段排序(升序、降序),支持样式变化等。

4)分页界面可以设置页面数量为10、20、50、全部等数量级,更加方便数据显示控制。

5)分页控件分页按钮可以设定图片。

6)可以结合我的查询控件,快速实现数据查询分页显示。

7)分页控件支持参数化SQL语句。

下载地址为

http://www.patientmis.com/Download/Web分页控件例子Demo.rar

1、Web分页界面效果

无图无真相,首先我们来了解下分页控件的相关显示界面效果。

带有查询条件的效果如下所示,查询模块为了方便和分页控件结合,也是一个控件来的,构造起来很方便,后面再详细介绍。

当然,分页控件很多属性可以设置的,包括分页的按钮图片,由于列表的DataGrid是外部的控件,因此样式更可以自定义,只需要告诉分页控件,列表的控件ID即可,因此你可以根据项目需要,构造自己列表展示效果。下面列出我项目中常用到的两种皮肤界面。

1)浅蓝色冷色调分页效果

2)浅黄色暖色调分页效果

当然,用户可以根据需要,自己使用相应的界面效果样式。

2)分页控件具体使用代码

1)首先在页面上注册控件,如下代码所示

<%@ Register TagPrefix="cc1" Namespace="WHC.WebPager" Assembly="WebPager" %>

2)然后在页面主体内容中添加分页控件即可。其中dg为具体的DataGrid控件,使用的时候和分页控件联合使用。

 <cc1:PagerID="Pager1"runat="server"Height="30"BindControlID="dg"ImageBasePath="../Images"FirstImagePath="btn_sy.gif"NextImagePath="btn_xyy.gif"PreviousImagePath="btn_syy.gif"LastImagePath="btn_wy.gif"GoImagePath="btn_go.gif"ExportImagePath="btn_export.gif">
                    </cc1:Pager>

3)DataGrid控件的参考代码。

                    <asp:DataGridID="dg"runat="server"Width="100%"CssClass="dg"AutoGenerateColumns="False"PageSize="20"AllowSorting="True"DataKeyField="ID"Height="0px"CellPadding="4"ForeColor="#333333"GridLines="None"OnItemCreated="dg_ItemCreated"OnSortCommand="dg_SortCommand"OnItemDataBound="dg_ItemDataBound">
                        <EditItemStyleCssClass="EditItem"></EditItemStyle>
                        <AlternatingItemStyleCssClass="AlternatingItem"></AlternatingItemStyle>
                        <ItemStyleCssClass="Item"></ItemStyle>
                        <HeaderStyleCssClass="Header"></HeaderStyle>
                        <FooterStyleCssClass="Footer"></FooterStyle>
                        <Columns>
                            <asp:TemplateColumnHeaderImageUrl="Images/delete.GIF">
                                <HeaderStyleWrap="False"Width="60px"></HeaderStyle>
                                <ItemTemplate>
                                    <asp:CheckBoxrunat="server"ID="cbxDelete"></asp:CheckBox>
                                </ItemTemplate>
                            </asp:TemplateColumn>
                            <asp:BoundColumnDataField="ID"HeaderText="ID"SortExpression="ID"></asp:BoundColumn>
                            <asp:BoundColumnDataField="Name"HeaderText="名称"SortExpression="Name"></asp:BoundColumn>
                            <asp:BoundColumnDataField="City"HeaderText="城市"SortExpression="City"></asp:BoundColumn>
                            <asp:BoundColumnDataField="Age"HeaderText="年龄"SortExpression="Age"></asp:BoundColumn>
                            <asp:BoundColumnDataField="Birthday"HeaderText="出生日期"SortExpression="Birthday">
                            </asp:BoundColumn>
                            <asp:BoundColumnDataField="Man"HeaderText="男性"></asp:BoundColumn>
                            <asp:TemplateColumnHeaderImageUrl="Images/edititem.gif">
                                <HeaderStyleWrap="False"Width="60px"></HeaderStyle>
                                <ItemTemplate>
                                    <asp:HyperLinkID="lnkView"runat="server"Text="查看"NavigateUrl='<%#"Modify.aspx?type=view&id="+ Eval("ID") %>'>查看</asp:HyperLink>
                                </ItemTemplate>
                            </asp:TemplateColumn>
                            <asp:TemplateColumnHeaderImageUrl="Images/edititem.gif">
                                <HeaderStyleWrap="False"Width="60px"></HeaderStyle>
                                <ItemTemplate>
                                    <asp:HyperLinkID="lnkEdit"runat="server"Text="编辑"NavigateUrl='<%#"Modify.aspx?type=edit&id="+ Eval("ID") %>'>编辑</asp:HyperLink>
                                </ItemTemplate>
                            </asp:TemplateColumn>
                        </Columns>
                        <SelectedItemStyleBackColor="#D1DDF1"Font-Bold="True"ForeColor="#333333" />
                        <PagerStyleHorizontalAlign="Left"ForeColor="#003399"BackColor="#99CCCC"Mode="NumericPages">
                        </PagerStyle>
                    </asp:DataGrid>

页面的后台代码如下所示,主要就是设置分页控件的查询语句以及排序等属性,如下所示。

        protected void Page_Load(objectsender, EventArgs e)
{
if (!this.IsPostBack)
{
BindData();
}
}
private voidBindData()
{
try{this.Pager1.SortFieldNames = this.SortFieldNames;this.Pager1.SortFieldAscend = this.SortFieldAscend;this.Pager1.SQL = "SELECT * FROM test Order By ID";
}
catch(Exception ex)
{
Response.Write(
string.Format("<script>alert('{0}');</script>", ex.Message));
}
}

以上就是没有带查询条件的分页控件,当然我们可以自己添加一些条件控件来实现查询过滤数据的功能。

3、查询控件和分页控件的结合

前面介绍过,为了快速方便实现列表的查询分页,结合分页控件使用,还有一个叫做查询控件的东西,这个效果就是前面介绍的那样,只不过是制定数据库字段信息就可以了,不用像传统那样,自己部署各种条件控件在界面上,效果其实一样,如下所示。

我们来看这个界面的实现代码。首先在界面上放置两个控件的声明代码。

<%@ Register TagPrefix="cc1"Namespace="WHC.SearchControl"Assembly="SearchControl" %>
<%@ Register TagPrefix="cc2"Namespace="WHC.WebPager"Assembly="WebPager" %>

然后看页面的前台代码如下所示。

    <formid="Form1"method="post"runat="server">
    <divstyle="border: 1px solid #A8CFEB; background-color: #F1F6FF">
        <cc1:SearchControlID="SearchControl1"runat="server"Width="100%"InSQL="SELECT * FROM Test"SortFieldNames="ID"RowControls="3"SearchButtonImage="../Images/btn_search.gif"ResetButtonImage="../Images/btn_reset.gif"CssClass="content"></cc1:SearchControl>
    </div>
    <div>
        <br/>
        <asp:DataGridID="dg"runat="server"Width="100%"CssClass="dg"AutoGenerateColumns="False"BorderColor="#3366CC"BorderStyle="None"BorderWidth="1px"BackColor="White"CellPadding="4"AllowSorting="True"OnItemCreated="dg_ItemCreated"OnSortCommand="dg_SortCommand"OnItemDataBound="dg_ItemDataBound">
            <EditItemStyleCssClass="EditItem"></EditItemStyle>
            <AlternatingItemStyleCssClass="AlternatingItem"></AlternatingItemStyle>
            <ItemStyleCssClass="Item"></ItemStyle>
            <HeaderStyleCssClass="Header"></HeaderStyle>
            <FooterStyleCssClass="Footer"></FooterStyle>
            <Columns>
                <asp:BoundColumnDataField="ID"HeaderText="ID"SortExpression="ID">
                    <HeaderStyleWidth="60px" />
                </asp:BoundColumn>
                <asp:BoundColumnDataField="Name"HeaderText="名称"SortExpression="Name"></asp:BoundColumn>
                <asp:BoundColumnDataField="City"HeaderText="城市"SortExpression="City"></asp:BoundColumn>
                <asp:BoundColumnDataField="Age"HeaderText="年龄"SortExpression="Age"></asp:BoundColumn>
                <asp:BoundColumnDataField="Birthday"HeaderText="出生日期"SortExpression="Birthday">
                </asp:BoundColumn>
                <asp:BoundColumnDataField="Man"HeaderText="男性"></asp:BoundColumn>
                <asp:TemplateColumnHeaderImageUrl="Images/edititem.gif">
                    <HeaderStyleWrap="False"Width="60px"></HeaderStyle>
                    <ItemTemplate>
                        <asp:HyperLinkID="lnkView"runat="server"Text="查看"NavigateUrl='<%#"Modify.aspx?type=view&id="+ Eval("ID") %>'>查看</asp:HyperLink>
                    </ItemTemplate>
                </asp:TemplateColumn>
                <asp:TemplateColumnHeaderImageUrl="Images/edititem.gif">
                    <HeaderStyleWrap="False"Width="60px"></HeaderStyle>
                    <ItemTemplate>
                        <asp:HyperLinkID="lnkEdit"runat="server"Text="编辑"NavigateUrl='<%#"Modify.aspx?type=edit&id="+ Eval("ID") %>'>编辑</asp:HyperLink>
                    </ItemTemplate>
                </asp:TemplateColumn>
            </Columns>
        </asp:DataGrid>
    </div>
    <br/>
    <div>
        <cc2:PagerID="Pager1"runat="server"Height="30"BindControlID="dg"ImageBasePath="../Images"FirstImagePath="btn_sy.gif"NextImagePath="btn_xyy.gif"PreviousImagePath="btn_syy.gif"LastImagePath="btn_wy.gif"GoImagePath="btn_go.gif"ExportImagePath="btn_export.gif">
        </cc2:Pager>
    </div>
    </form>

后台主要代码如下所示。

        private void Page_Load(objectsender, EventArgs e)
{
this.SearchControl1.OutSQLValueChanged += newOutSQLChangedHandle(SearchControl1_OutSQLValueChanged);this.SearchControl1.OnAddNew += newAddNewHandler(SearchControl1_OnAddNew);this.SearchControl1.OnDelete += newDeleteHandler(SearchControl1_OnDelete);

WebControl appendButton
= CreateImageButton("btnAppended", "统计", "Images/btn_Statistics.gif");this.SearchControl1.AppendedButtons = newWebControl[] { appendButton };if (!Page.IsPostBack)
{
FieldInfo nameInfo
= new FieldInfo("Name", "姓名", FieldType.String);

FieldInfo cityInfo
= new FieldInfo("City", "城市", FieldType.String);
cityInfo.Width
= 100;
cityInfo.Items
= new ListItem[] { new ListItem("北京市", "北京"), new ListItem("广州"), new ListItem("成都") };
cityInfo.TargetFieldName
= "Area";
cityInfo.UseInItems
= false;//所有的时候,不限定内容,默认限定为列表内容 cityInfo.OnFillItem += new AddItemHandler(this.OnFillItem);

FieldInfo areaInfo
= new FieldInfo("Area", "地区", FieldType.String);
areaInfo.Items
= new ListItem[0];

FieldInfo manInfo
= new FieldInfo("Man", "是否男性", FieldType.Boolean);
manInfo.UseInItems
= false;

FieldInfo birthInfo
= new FieldInfo("Birthday", "出生日期", FieldType.DateTime);
FieldInfo ageInfo
= new FieldInfo("Age", "年龄", FieldType.Numeric);this.SearchControl1.SearchFields = newFieldInfo[] { nameInfo, cityInfo, areaInfo, manInfo, birthInfo, ageInfo };

BindData();
}
}
private voidBindData()
{
this.Pager1.SortFieldNames = this.SearchControl1.SortFieldNames;this.Pager1.SortFieldAscend = this.SearchControl1.SortFieldAscend;this.Pager1.PagerParameters = this.SearchControl1.PagerParameters;this.Pager1.SQL = this.SearchControl1.OutSQL;
}
private void SearchControl1_OutSQLValueChanged(objectsender, OutSQLChangedEventArgs e)
{
//测试输出 this.Response.Write(SearchControl1.OutSQL);

BindData();
this.Pager1.CurrentPage = 1;
}

当然实现列表的联动还有一个OFillItem的函数,代码如下所示。

        private void OnFillItem(DropDownList ddListControl, stringselItemValue)
{
ddListControl.Items.Clear();
if (selItemValue.Trim().Length > 0)
{
string sql = string.Format("select Area from CityArea where City ='{0}'", selItemValue);
Database database
=DatabaseFactory.CreateDatabase();
DbCommand command
=database.GetSqlStringCommand(sql);using (IDataReader reader =database.ExecuteReader(command))
{
while(reader.Read())
{
ddListControl.Items.Add(
new ListItem(reader["Area"].ToString()));
}
}
}

ddListControl.Items.Insert(
0, new ListItem("--所有--", string.Empty));
}

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

偶然一次机会,一个项目的报表功能指定需要导出为Word文档,因此寻找了很多篇文章,不过多数介绍的比较简单一点,于是也参考了官方的帮助介绍,终于满足了客户的需求。下面我由浅入深来介绍这个控件在实际业务中的使用过程吧。

1、二维表格的Word操作

日常中,常见的内容输出就是二维表格的方式,表头比较固定,内容每行一条,那么在实际的使用控件我们该如何操作呢,其实这个控件这方面介绍的文章很多,参考一下就能做出来了。其实介绍这个就是要说明书签的重要性,这个在Aspose.Cell控件也是如此,书签除了可以用来替换内容,还可以用来标记内容输入的开始位置等等功能。

首先我们在一个空白的Word文档中绘制一个表格头,然后再换行的开始插入一个标签引用,插入书签有两种方式,一种是在Word(2007、2010)的【插入】-【书签】中插入制定位置的书签引用,如下所示。

一种是在Word的自定义快速访问工具栏上添加其他命令,如下步骤所示

前者插入的书签是没有文字或者特别的标记,但是确实存在,后者会插入一个灰色块作为占位符,如下所示,我这这个二维表格的例子里面使用后者进行测试(两者同等效果的)

这样设计好Word模板后,下一步就是如何利用代码生成二维表格了。首先这里提示一下,就是我故意设置了每个表格单元格的宽度不同,所以也就要求生成的行要和头部对应,所以表格生成每行之前,肯定要获得对应列的样式属性的,否则就会对应不上了。下面看代码。

try{
Aspose.Words.Document doc
= newAspose.Words.Document(templateFile);
Aspose.Words.DocumentBuilder builder
= newAspose.Words.DocumentBuilder(doc);

DataTable nameList
= DataTableHelper.CreateTable("编号,姓名,时间");
DataRow row
= null;for (int i = 0; i < 50; i++)
{
row
=nameList.NewRow();
row[
"编号"] = i.ToString().PadLeft(4, '0');
row[
"姓名"] = "伍华聪" +i.ToString();
row[
"时间"] =DateTime.Now.ToString();
nameList.Rows.Add(row);
}

List
<double> widthList = new List<double>();for (int i = 0; i < nameList.Columns.Count; i++)
{
builder.MoveToCell(
0, 0, i, 0); //移动单元格 double width = builder.CellFormat.Width;//获取单元格宽度 widthList.Add(width);
}

builder.MoveToBookmark(
"table"); //开始添加值 for (var i = 0; i < nameList.Rows.Count; i++)
{
for (var j = 0; j < nameList.Columns.Count; j++)
{
builder.InsertCell();
//添加一个单元格 builder.CellFormat.Borders.LineStyle =LineStyle.Single;
builder.CellFormat.Borders.Color
=System.Drawing.Color.Black;
builder.CellFormat.Width
=widthList[j];
builder.CellFormat.VerticalMerge
=Aspose.Words.Tables.CellMerge.None;
builder.CellFormat.VerticalAlignment
= CellVerticalAlignment.Center;//垂直居中对齐 builder.ParagraphFormat.Alignment = ParagraphAlignment.Center;//水平居中对齐 builder.Write(nameList.Rows[i][j].ToString());
}
builder.EndRow();
}
doc.Range.Bookmarks[
"table"].Text = ""; //清掉标示 doc.Save(saveDocFile);if (MessageUtil.ShowYesNoAndTips("保存成功,是否打开文件?") ==System.Windows.Forms.DialogResult.Yes)
{
System.Diagnostics.Process.Start(saveDocFile);
}
}
catch(Exception ex)
{
LogHelper.Error(ex);
MessageUtil.ShowError(ex.Message);
return;
}

以上代码的步骤就是

1)创建Aspose.Words.Document 和 Aspose.Words.DocumentBuilder对象,然后生成数据的二维表格内容。

2)遍历模板表格,或者每一列的宽度,以备后用。

3)移动到表格的书签位置,然后开始录入数据,Word表格的每个Cell都要求制定样式和宽度,这样才能和表格头部吻合。

4)保存文件内容到新的文件里面即可。

输出的效果如下所示。

2、单元格合并的操作

常见的Word文件或者Excel文件中,都经常看到合并单元格的内容,因此这个部分也是非常常见的操作,必须掌握。

我们先看一个例子代码及效果。

                try{
Aspose.Words.Document doc
= newAspose.Words.Document(templateFile);
Aspose.Words.DocumentBuilder builder
= newAspose.Words.DocumentBuilder(doc);

builder.InsertCell();
builder.CellFormat.Borders.LineStyle
=LineStyle.Single;
builder.CellFormat.Borders.Color
=System.Drawing.Color.Black;
builder.CellFormat.VerticalMerge
=CellMerge.First;
builder.Write(
"Text in merged cells.");

builder.InsertCell();
builder.CellFormat.Borders.LineStyle
=LineStyle.Single;
builder.CellFormat.Borders.Color
=System.Drawing.Color.Black;
builder.CellFormat.VerticalMerge
=CellMerge.None;
builder.Write(
"Text in one cell");
builder.EndRow();

builder.InsertCell();
builder.CellFormat.Borders.LineStyle
=LineStyle.Single;
builder.CellFormat.Borders.Color
=System.Drawing.Color.Black;//This cell is vertically merged to the cell above and should be empty. builder.CellFormat.VerticalMerge =CellMerge.Previous;

builder.InsertCell();
builder.CellFormat.Borders.LineStyle
=LineStyle.Single;
builder.CellFormat.Borders.Color
=System.Drawing.Color.Black;
builder.CellFormat.VerticalMerge
=CellMerge.None;
builder.Write(
"Text in another cell");
builder.EndRow();

doc.Save(saveDocFile);
if (MessageUtil.ShowYesNoAndTips("保存成功,是否打开文件?") ==System.Windows.Forms.DialogResult.Yes)
{
System.Diagnostics.Process.Start(saveDocFile);
}
}
catch(Exception ex)
{
LogHelper.Error(ex);
MessageUtil.ShowError(ex.Message);
return;
}

他的效果如下

关于合并单元格的介绍,你还可以参考下这篇官方介绍:
http://www.aspose.com/docs/display/wordsnet/Working+with+Merged+Cells

如果上面的例子还不够明白,OK,我在介绍一个实际的例子,来说明合并单元格的操作模式。

实际文档生成如下所示:

文档的模板如下所示:

其实这个里面的“测试”内容是使用代码写入的,其实就是一行业务数据,用两行来展示,其中有些合并的单元格,这是一个实际项目的表格形式。我们注意到,每行有13个单元格,其中第一、第二、第十三列是合并列。和并列有一个特点,就是它的两个索引都有效,不过只是能使用第一个索引来对它进行操作复制,利用第二个没有用处的。

如第一个列是和并列,它应该有0、13这样的索引,第二列也是和并列,它也有1、14的索引,其他的类推。

了解这样的逻辑关系后,我们看实际操作的代码如下所示。

                try{
Aspose.Words.Document doc
= newAspose.Words.Document(templateFile);
Aspose.Words.DocumentBuilder builder
= newAspose.Words.DocumentBuilder(doc);List<double> widthList = new List<double>();for (int i = 0; i < 13; i++)
{
builder.MoveToCell(
0, 2, i, 0); //移动单元格 double width = builder.CellFormat.Width;//获取单元格宽度 widthList.Add(width);
}

builder.MoveToBookmark(
"table"); //开始添加值 Table table=builder.StartTable();
builder.RowFormat.HeadingFormat
= true;
builder.ParagraphFormat.Alignment
=ParagraphAlignment.Center;for (int j = 0; j < 26; j++)
{
builder.InsertCell();
//添加一个单元格 builder.CellFormat.Borders.LineStyle =LineStyle.Single;
builder.CellFormat.Borders.Color
=System.Drawing.Color.Black;int cellIndex = (j > 12) ? (j-13) : j; //位于第几个单元格 builder.CellFormat.Width =widthList[cellIndex];
builder.CellFormat.VerticalAlignment
= CellVerticalAlignment.Center;//垂直居中对齐 builder.ParagraphFormat.Alignment = ParagraphAlignment.Center;//水平居中对齐 builder.CellFormat.VerticalMerge=Aspose.Words.Tables.CellMerge.None;if (cellIndex == 0 || cellIndex == 1 || cellIndex == 12)
{
if (j > 12)
{
builder.CellFormat.VerticalMerge
=CellMerge.Previous;
}
else{
builder.CellFormat.VerticalMerge
=CellMerge.First;
}
}

builder.Write(
"测试" +j.ToString());if (cellIndex == 12)
{
builder.EndRow();
}
}
builder.EndTable();

doc.Save(saveDocFile);
if (MessageUtil.ShowYesNoAndTips("保存成功,是否打开文件?") ==System.Windows.Forms.DialogResult.Yes)
{
System.Diagnostics.Process.Start(saveDocFile);
}
}
catch(Exception ex)
{
LogHelper.Error(ex);
MessageUtil.ShowError(ex.Message);
return;
}

界面开发,无论对于Web开发,还是Winform开发,都需要耗费一定的时间,特别对于一个数据库字段比较多的界面,一般就需要在编辑界面上摆的更多的控件来做数据显示,每次碰到这个,都有点头痛,反复的机械操作让人挺累,也很烦,但是又必须这样做。

由于数据库字段和界面的排版都有一定的关联关系,因此可以通过代码生成工具Database2Sharp的数据库元数据,包含表名称、备注信息、字段列表,以及每个字段的名称、备注、类型等信息,构造一个基础的界面,把重复机械的部分给快速完成,这就是我所说的界面快速生成。当然,对于精致的界面,机械的生成肯定不能满足我们的需要,因此真正的界面需要在这个基础上修改完善一下,但是由于重复劳动部分,已经给工具处理掉了,因此,界面开发效率会大大提高。

1、基于Web界面的快速生成

基于EnterpriseLibray架构的代码生成,除了可以生成传统的业务层、数据访问层、数据访问接口层、实体层外,还同时生成了Web界面工程,Web界面工程已经添加了相关的引用程序集及必须的控件,生成后即可编译运行,一些表字段设计合理的话,甚至不用修改一行代码就能直接使用,如下所示。

生成后的项目界面截图如下所示。

以下是TB_ItemDetail表生成的界面,没有修改过任何东西,列表页面呈现的效果。

其中编辑及查看页面也生成了,直接单击连接就可以看到效果,如下图所示。

这些效果是很基本的界面布局了,如果需要更好看的,可以根据需要调整下每行的数量,以及显示方式。这样的效果,虽然不算最好看,但是已经帮我们把界面元素弄起来了,我们后面的工作就很好开展,不用再重复添加每个字段对应的控件布局信息了。具体的代码是标准的HTML控件代码如下所示。

<%@ Page Language="C#"MasterPageFile="~/MasterPage.master"AutoEventWireup="true"Inherits="Pages_ItemDetail_EditItemDetail"Codebehind="EditItemDetail.aspx.cs" %>

<asp:ContentID="Content1"ContentPlaceHolderID="ContentPlaceHolder1"runat="Server">
    <h3id="h_title"align="center"runat="server">编辑备件信息表 </h3>
    <tableclass="formitem"cellspacing="1"cellpadding="0"align="Center"border="0"style="border-width: 0px; width: 98%;">
        <caption>基本信息</caption>  
        <trstyle="width: 100%;">
            <thalign="left"style="width: 20%;">备件编号 :</th>
            <tdstyle="width: 80%;"colspan="3">
                &nbsp;<asp:TextBoxID="txtItemNo"runat="server"Width="250px"MaxLength="100"></asp:TextBox>
            </td>
        </tr>
        <trstyle="width: 100%;">
            <thalign="left"style="width: 20%;">备件名称 :</th>
            <tdstyle="width: 80%;"colspan="3">
                &nbsp;<asp:TextBoxID="txtItemName"runat="server"Width="250px"MaxLength="100"></asp:TextBox>
            </td>
        </tr>
        <trstyle="width: 100%;">
            <thalign="left"style="width: 20%;">供货商 :</th>
            <tdstyle="width: 80%;"colspan="3">
                &nbsp;<asp:TextBoxID="txtManufacture"runat="server"Width="250px"MaxLength="100"></asp:TextBox>
            </td>
        </tr>
...............

当然后台代码的保存及赋值显示,肯定是少不了的了,这些也是根据我的框架代码生成的,应该来说是很有用的。

public partial classPages_ItemDetail_EditItemDetail : BasePage
{
protected override voidOnInit(EventArgs e)
{
base.OnInit(e);this.IsFunctionControl = true;this.FunctionID = "ItemDetail/Edit";
}
private voidBindData()
{
try{string id = Request.QueryString["ID"];if(!string.IsNullOrEmpty(id))
{
ItemDetailInfo info
= BLLFactory<ItemDetail>.Instance.FindByID(id);if(info != null)
{
this.txtItemNo.Text =info.ItemNo;this.txtItemName.Text =info.ItemName;this.txtManufacture.Text =info.Manufacture;this.txtMapNo.Text =info.MapNo;this.txtSpecification.Text =info.Specification;this.txtMaterial.Text =info.Material;this.txtItemBigType.Text =info.ItemBigType;this.txtItemType.Text =info.ItemType;this.txtUnit.Text =info.Unit;this.txtPrice.Text =info.Price.ToString();this.txtSource.Text =info.Source;this.txtStoragePos.Text =info.StoragePos;this.txtUsagePos.Text =info.UsagePos;this.txtNote.Text =info.Note;this.txtWareHouse.Text =info.WareHouse;this.txtDept.Text =info.Dept;
}
}
}
catch(Exception ex)
{
Helper.ShowError(
this, ex, false);return;
}
}
protected void Page_Load(objectsender, EventArgs e)
{
if(!this.IsPostBack)
{
BindData();
}
}
/// <summary> ///获取数据/// </summary> privateItemDetailInfo SetData(ItemDetailInfo info)
{
info.ItemNo
= this.txtItemNo.Text;
info.ItemName
= this.txtItemName.Text;
info.Manufacture
= this.txtManufacture.Text;
info.MapNo
= this.txtMapNo.Text;
info.Specification
= this.txtSpecification.Text;
info.Material
= this.txtMaterial.Text;
info.ItemBigType
= this.txtItemBigType.Text;
info.ItemType
= this.txtItemType.Text;
info.Unit
= this.txtUnit.Text;
info.Price
= Helper.SafeConvertDecimal(this.txtPrice.Text);
info.Source
= this.txtSource.Text;
info.StoragePos
= this.txtStoragePos.Text;
info.UsagePos
= this.txtUsagePos.Text;
info.Note
= this.txtNote.Text;
info.WareHouse
= this.txtWareHouse.Text;
info.Dept
= this.txtDept.Text;returninfo;
}
protected void imgSubmit_Click(objectsender, ImageClickEventArgs e)
{
try{string id = Request.QueryString["ID"];if(string.IsNullOrEmpty(id))
{
ItemDetailInfo info
= newItemDetailInfo();
SetData(info);

BLLFactory
<ItemDetail>.Instance.Insert(info);
}
else{
ItemDetailInfo info
= BLLFactory<ItemDetail>.Instance.FindByID(id);if(info != null)
{
SetData(info);
BLLFactory
<ItemDetail>.Instance.Update(info, id);
}
}
}
catch(Exception ex)
{
Helper.ShowError(
this, ex, false);return;
}

Response.Redirect(
"ListItemDetail.aspx");
}
protected void imgBack_Click(objectsender, ImageClickEventArgs e)
{
Response.Redirect(
"ListItemDetail.aspx");
}
}

有了这些智能的界面生成,界面开发其实是一件不再枯燥的事情,节省更多的时间,做更有意义的事情。

2、基于DevExpress的Winform界面快速生成

除了Web的界面开发,Winform的界面开发,当然也很重要,枯燥的事情一样可以交给代码生成工具Database2Sharp进行生成,不需要让人工反复的做这些无用功,或者是技术含量不太高的东西。

上面Web界面工程,我是采用SQLServer的数据库进行生成代码的,这个例子,我们介绍一下Oracle数据库的生成方式,其实这个代码生成工具,对应用什么数据库不重要,都会很好的给你生成相关的界面信息,不管你是用SQLServer还是Oracle,或者是Sqlite、Access、MySql等等,都一样可以很好的生成所需的开发代码的。

工具生成了界面布局代码,以及界面后台逻辑代码,这样的代码正是我们开发所需要的,我们看到生成后的界面代码(没有修改代码的),在VS里面的真实效果如下。

上面的布局采用了LayoutControl来进行布局控制,这是一种很好的布局控制方法,它除了使得界面更加美观外,还能非常自由调整每个控件的位置及大小。

当然,我们一般为了美观需要,会对界面进行一定的调整,由于DevExpress控件类型变化切换很方便,所以这种调整很自由高效,调整后的界面如下所示。

比起十几秒钟弄好一个界面,纯手工做这样的界面,没有十几分钟,估计弄不好的,而且容易出错,最重要的问题,就是团队开发的时候,这个统一性就很有保证,开发效率高,带来Bug也会很少。

3、基于传统的Winform界面快速生成

传统的Winform界面和DotNetBar样式的界面生成,和上面的DevExpress样式代码很类似,不过控件不同而已,由于不同的需要,用传统界面元素开发,也是一种常见的开发模式,因此我的代码生成工具也支持这种常规的界面生成,来辅助大家做好界面的工作。

除了设置以上界面样式外,其他生成操作一样。得到的界面效果如下所示。

和DevExpress控件界面代码类似,这里也使用了TableLayout的布局控件,用来较好控制布局的高度宽度,以及每行位置等,这样整体效果就很好,也容易进行界面元素的控制。

在上篇随笔《
代码生成工具之界面快速生成
》中介绍过了代码生成工具Database2Sharp是如何快速生成所需的Web界面以及各种Winform界面,其中包括生成即可运行的Web界面效果,Winform布局信息的生成。这些看似很简单的界面元素生成,其实是需要丰富的数据库元数据信息作为基础的,而且对不同的数据库处理要有所不同。本文介绍不同数据库之间元数据的处理差别,以及如何代码生成工具Database2Sharp如何兼容处理这些问题的。

1、常规的数据库表、字段名称的转义

例如,我们需要获取表的别名,对于SqlServer一般设计的时候是采用Pascal命名方式,所以表名称不需要转义,但对于Oracle表名称,基本上都是以大写来命名,而且表名称不是采用Pascal方式,而是两个词之间采用下划线“_"来分隔的,如DEPT_NAME这样的名称。

一般来说,我们为每种的数据库定义一套转义规则,来为表、字段名称增加一个别名字段,如Oracle的DEPT_NAME我们设法让它别名显示为DeptName就可以了,SQLServer的由于一般名称都是Pascal的,我们可以不用转义,数据库表、字段名称为DeptName,我们就保留它作为别名DeptName即可。

我的代码生成工具的元数据属性就是这样的模式,有一个NameElement对象,就包含Name和Alias两个属性,如下所示。

2、特别情况下的表、字段名称转义

但我们有时候反向工程的时候,可能数据库是从Oracle到SQLServer的,或者有时候考虑多数据库兼容的情况,那么可能SQLServer的表及字段的名称还是Oracle的命名规则的,如下SQLServer例子所示。

对于以上的数据库信息,如果没有转义数据库名称,那么就给生成代码造成很大的困扰,因为实体类属性名称,类名称都可能是Oracle风格的大写的标志,非常不利于阅读。

但代码生成工具已经增加了智能识别字段名称的逻辑,对于这种从Oracle过来的数据库命名规则,我们也能合理生成对应的代码,如上图的右边,它已经判断使用了Oracle的命名规则来处理别名了。

这样我们生成的代码,就是很友好的命名风格了。

3、自定义表、字段名称的别名

有时候,统一规则生成的别名不一定是我们所需要的,那么请使用代码生成工具的别名设置操作即可把某个表名、字段名设置为你想要的名称,如下操作所示。

1)表别名修改

2)字段别名修改

4、在代码工具的自定义模板中使用字段转义信息

代码生成工具Database2Sharp提供了很好的自定义模板操作交互功能,我们只需要在模板文件中书写NVelocity的模板代码就可以输出各种丰富多彩的代码的,如下面图就是自定义模板列表界面,其中左边列出一些基础的例子模板代码,大家可以参考学习,在树形目录中建立自己的模板文件和模板代码。

使用自定义模板代码的目的,就是要利用数据库的元数据信息来生成复杂而有规律的代码片段或者文件的。

我们注意到模板代码,其中利用到的数据库信息及遍历操作等。

        /// <summary>
        ///初始化/// </summary>
        /// <param name="info">实体类信息</param>
        private voidInitData(${ClassName}Info info)
{
#
foreach($ColumnInfo in${TableInfo.ColumnList.Values})#if(${ColumnInfo.AutoIncrement} == false) #if(${ColumnInfo.NetType} == "System.String" ) this.txt${ColumnInfo.Name.Alias.ToCapit()}.Text =info.${ColumnInfo.Name.Alias.ToCapit()};
#elseif(${ColumnInfo.NetType}
== "System.DateTime")this.txt${ColumnInfo.Name.Alias.ToCapit()}.Text =info.${ColumnInfo.Name.Alias.ToCapit()}.ToShortDateString();#else this.txt${ColumnInfo.Name.Alias.ToCapit()}.Text =info.${ColumnInfo.Name.Alias.ToCapit()}.ToString();
#end
#end #
#endif#end
}
/// <summary> ///获取数据/// </summary> private${ClassName}Info GetData()
{
${ClassName}Info info
= new${ClassName}Info();
#
foreach($ColumnInfo in${TableInfo.ColumnList.Values})#if(${ColumnInfo.AutoIncrement} == false) #if(${ColumnInfo.NetType} == "System.Decimal" )info.${ColumnInfo.Name.Alias.ToCapit()}= Helper.SafeConvertDecimal(this.txt${ColumnInfo.Name.Alias.ToCapit()}.Text);
#elseif(${ColumnInfo.NetType}
== "System.DateTime")
info.${ColumnInfo.Name.Alias.ToCapit()}
= Helper.SafeConvertDate(this.txt${ColumnInfo.Name.Alias.ToCapit()}.Text);#elseinfo.${ColumnInfo.Name.Alias.ToCapit()}= this.txt${ColumnInfo.Name.Alias.ToCapit()}.Text;
#end
#end #
#endif#end
}

其中的${ColumnInfo.Name.Alias.ToCapit()}为别名,而${ColumnInfo.Name.Name.ToCapit()}为数据库表字段真实名称。
以上模板,我们选择数据库表生成代码,就可以得到下面的标准赋值及获取内容的代码了。

        /// <summary>
        ///初始化/// </summary>
        /// <param name="info">实体类信息</param>
        private voidInitData(GroupInfo info)
{
this.txtEditor.Text =info.Editor.ToString();this.txtEdittime.Text =info.Edittime;this.txtName.Text =info.Name;this.txtOutGroup.Text =info.OutGroup;this.txtGroupOrder.Text =info.GroupOrder;this.txtDeptName.Text =info.DeptName;this.txtUpperDept.Text =info.UpperDept.ToString();this.txtGrade.Text =info.Grade.ToString();this.txtRemark.Text =info.Remark;
}
/// <summary> ///获取数据/// </summary> privateGroupInfo GetData()
{
GroupInfo info
= newGroupInfo();
info.Editor
= this.txtEditor.Text;
info.Edittime
= this.txtEdittime.Text;
info.Name
= this.txtName.Text;
info.OutGroup
= this.txtOutGroup.Text;
info.GroupOrder
= this.txtGroupOrder.Text;
info.DeptName
= this.txtDeptName.Text;
info.UpperDept
= this.txtUpperDept.Text;
info.Grade
= this.txtGrade.Text;
info.Remark
= this.txtRemark.Text;
}

以上就是代码生成工具Database2Sharp的数据库表及字段名称转义的智能处理以及应用,如果熟练使用NVelocity的基本语法,结合代码工具提供的数据库元数据信息,我们可以做的更多,做的更好。希望这个工具对你的开发有帮助。

在之前介绍的Winform分页控件中,都以分页控件+显示表格控件作为一个整体性的控件,不可分开,这样做的目的是可以实现更多的操作,集成更多丰富的特性,减少我们开发的工作量,这种情况虽然适用于大多数的情况,不过有时候需要更进一步对列表控件进行设置修改或者实现一些特殊的显示效果的时候,这种绑定列表控件的分页控件方式就有点捉襟见肘了。为了解决这个问题,我对分页控件进行稍微的调整,使其支持分页和列表控件分开的情况,具体实现以及效果是如何的呢,下面我们来分析了解下。

1、单独的分页信息控件

在我的开发设计过程总,实现一些关键的属性及操作后,它就可以作为一个独立的分页控件进行使用了。

运行的时候效果如下所示:

2、分页控件的使用

在实际使用的时候,我们把分页控件的dll导入到工具栏里面,然后我们在VS的控件集合里面就可以看到它的图标了。拖动到界面上就实现了分页控件的布局显示了。

由于这个纯分页没有列表控件,这个列表就需要自己放到你的界面上去,由于列表控件如DataGridView(传统控件)或者GridControl(DevExpress控件)是独立的,因此可以自由设计调整,列表的样式以及事件处理都合分页控件没有必然的联系。

使用代码如下所示:

       private void FrmCustomerPager_Load(objectsender, EventArgs e)
{
this.pager1.PageChanged += newWHC.Pager.WinControl.PageChangedEventHandler(pager1_PageChanged);this.pager1.ExportCurrent += newWHC.Pager.WinControl.ExportCurrentEventHandler(pager1_ExportCurrent);this.pager1.ExportAll += newWHC.Pager.WinControl.ExportAllEventHandler(pager1_ExportAll);

BindData();
}
void pager1_PageChanged(objectsender, EventArgs e)
{
BindData();
}
void pager1_ExportCurrent(objectsender, EventArgs e)
{
MessageUtil.ShowTips(
"导出当前页");
}
void pager1_ExportAll(objectsender, EventArgs e)
{
MessageUtil.ShowTips(
"导出所有");
}
/// <summary> ///标准的记录查询函数/// </summary> /// <param name="where"></param> /// <param name="pagerInfo"></param> /// <returns></returns> private DataTable FindToDataTable(string where, PagerInfo pagerInfo)
{
WHC.Pager.WinControl.PagerHelper helper
= new WHC.Pager.WinControl.PagerHelper("All_Customer", "*", "LastUpdated", pagerInfo.PageSize, pagerInfo.CurrenetPageIndex, true, where);string countSql = helper.GetPagingSql(WHC.Pager.WinControl.DatabaseType.Access, true);string dataSql = helper.GetPagingSql(WHC.Pager.WinControl.DatabaseType.Access, false);string value =SqlValueList(countSql);
pagerInfo.RecordCount
= Convert.ToInt32(value);//为了显示具体的信息,需要设置总记录数 DataTable dt=SqlTable(dataSql);returndt;
}
/// <summary> ///根据查询条件构造查询语句/// </summary> /// <returns></returns> private stringGetSearchSql()
{
SearchCondition condition
= newSearchCondition();
condition.AddCondition(
"Name", this.txtName.Text, SqlOperator.Like)
.AddCondition(
"Type", this.cmbType.Text, SqlOperator.Like)
.AddCondition(
"Area", this.cmbArea.Text, SqlOperator.Like);string where = condition.BuildConditionSql().Replace("Where", "");return where;
}
#endregion private voidBindData()
{
string where =GetSearchSql();this.pager1.PageSize = 30;
DataTable dt
= FindToDataTable(where, this.pager1.PagerInfo);this.gridControl1.DataSource =dt;
}
private void btnSearch_Click(objectsender, EventArgs e)
{
BindData();
}

从上面的代码,我们可以看到,使用起来还是很方便简洁的,关键代码就是下面这几行:

       private void FrmCustomerPager_Load(objectsender, EventArgs e)
{
this.pager1.PageChanged += newWHC.Pager.WinControl.PageChangedEventHandler(pager1_PageChanged);this.pager1.ExportCurrent += newWHC.Pager.WinControl.ExportCurrentEventHandler(pager1_ExportCurrent);this.pager1.ExportAll += newWHC.Pager.WinControl.ExportAllEventHandler(pager1_ExportAll);

BindData();
}
private voidBindData()
{
string where =GetSearchSql();this.pager1.PageSize = 30;
DataTable dt
= FindToDataTable(where, this.pager1.PagerInfo);this.gridControl1.DataSource =dt;
}

这样独立的纯粹的分页,给了列表展示控件最大的弹性,可以在一些复合列、复杂列表中提供很好的整合效果,弥补一些特殊分页碰到的问题。

除了DevExpress样式,还提供了传统界面样式、DotNetBar界面样式几种效果,如下所示。