Winform应用系统如何快速开发 winform下拉列表控件

在Winform开发中 。我们往往除了常规的单表信息录入外 。有时候设计到多个主从表的数据显示、编辑等界面 。单表的信息一般就是控件和对象实体一一对应 。然后调用API保存即可 。主从表就需要另外特殊处理 。本随笔介绍如何快速实现主从表编辑界面的处理 。结合GridControl控件的GridView控件对象 。实现数据在列表中的实时编辑 。非常方便 。
一、主从表的界面设计及展示主从表一般涉及两个以上的表 。一个是主表 。其他的是从表的 。在实际情况下 。一般包含两个表较多 。我们这里以两个表的主从表关系进行分析处理 。
例如我们建立两个报销申请单表关系如下所示 。

Winform应用系统如何快速开发 winform下拉列表控件

文章插图

Winform应用系统如何快速开发 winform下拉列表控件

文章插图
对于报销的主从表信息 。我们可以在列表中进行展示 。如下界面所示 。分为两部分:一部分是主表信息 。一部分是从表信息 。单击主表信息后 。显示对应从表的列表信息 。
Winform应用系统如何快速开发 winform下拉列表控件

文章插图

Winform应用系统如何快速开发 winform下拉列表控件

文章插图
那么我们新增一条主表记录的时候 。那么可以弹出一个新的界面进行数据的维护处理 。方便我们录入主从表的信息 。界面如下所示 。
Winform应用系统如何快速开发 winform下拉列表控件

文章插图

Winform应用系统如何快速开发 winform下拉列表控件

文章插图
上面界面包括了主表信息 。以及从表的信息(在GridView中实时录入)两部分 。这样填写后统一进行提交处理 。
二、主从表编辑界面的处理这里主要介绍一下主从表的编辑界面处理 。也就是上面这个界面的实现处理 。
Winform应用系统如何快速开发 winform下拉列表控件

文章插图

Winform应用系统如何快速开发 winform下拉列表控件

文章插图
其中初始化GridView的代码如下所示 。
///<summary>///初始化明细表的GridView数据显示///</summary>privatevoidInitDetailGrid(){//初始清空列this.gridView1.Columns.Clear();//设置部分列隐藏this.gridView1.CreateColumn("ID","编号").Visible=false;this.gridView1.CreateColumn("Header_ID","主表编号").Visible=false;this.gridView1.CreateColumn("apply_ID","申请单编号").Visible=false;//添加下拉列表列 。并绑定数据源this.gridView1.CreateColumn("FeeType","费用类型",100).CreateComboBox().BindDictItems("费用类型");//创建日期列并指定格式varOccurTime=this.gridView1.CreateColumn("OccurTime","发生时间",120).CreateDateEdit();OccurTime.EditMask="yyyy-MM-ddHH:mm";OccurTime.DisplayFormat.FormatString="yyyy-MM-ddHH:mm";//创建数值列this.gridView1.CreateColumn("FeeAmount","费用金额").CreateSpinEdit();//创建备注列this.gridView1.CreateColumn("FeeDescription","费用说明",200).CreateMemoEdit();//初始化GridView 。可以新增列this.gridView1.InitGridView(GridType.NewItem,false,EditorShowMode.MouseDownFocused,"");//转义列内容显示this.gridView1.CustomColumnDisplayText+=newCustomColumnDisplayTextEventHandler(gridView1_CustomColumnDisplayText);//处理单元格的样式this.gridView1.RowCellStyle+=newRowCellStyleEventHandler(gridView1_RowCellStyle);//不允许头部排序this.gridView1.OptionsCustomization.AllowSort=false;//绘制序号this.gridView1.CustomDrawRowIndicator+=(s,e)=>{if(e.Info.IsRowIndicator&&e.RowHandle>=0){e.Info.DisplayText=(e.RowHandle+1).ToString();}};//对输入单元格进行非空校验this.gridView1.ValidateRow+=delegate(objectsender,ValidateRowEventArgse){varresult=gridControl1.ValidateRowNull(e,newstring[]{"FeeType"});};//新增行的内容初始化this.gridView1.InitNewRow+=(s,e)=>{gridView1.SetRowCellValue(e.RowHandle,"ID",Guid.NewGuid().ToString());gridView1.SetRowCellValue(e.RowHandle,"Header_ID",tempInfo.ID);gridView1.SetRowCellValue(e.RowHandle,"Apply_ID",tempInfo.Apply_ID);gridView1.SetRowCellValue(e.RowHandle,"OccurTime",DateTime.Now);};}voidgridView1_RowCellStyle(objectsender,DevExpress.XtraGrid.Views.Grid.RowCellStyleEventArgse){GridViewgridView=this.gridView1;if(e.Column.FieldName=="FeeAmount"){e.Appearance.BackColor=Color.Green;e.Appearance.BackColor2=Color.LightCyan;}}voidgridView1_CustomColumnDisplayText(objectsender,DevExpress.XtraGrid.Views.Base.CustomColumnDisplayTextEventArgse){stringcolumnName=e.Column.FieldName;if(e.Column.ColumnType==typeof(DateTime)){if(e.Value!=null){if(e.Value=http://www.shenzhoubaby.com/=DBNull.Value||Convert.ToDateTime(e.Value)<=Convert.ToDateTime("1900-1-1")){e.DisplayText="";}else{e.DisplayText=Convert.ToDateTime(e.Value).ToString("yyyy-MM-ddHH:mm");//yyyy-MM-dd}}}}
上面代码都有详细的备注 。主要就是我们根据数据库表的关系 。创建对应显示的字段即可 。其中有需要隐藏的那么就不要显示(方便获取对应的值)
//设置部分列隐藏this.gridView1.CreateColumn("ID","编号").Visible=false;this.gridView1.CreateColumn("Header_ID","主表编号").Visible=false;this.gridView1.CreateColumn("Apply_ID","申请单编号").Visible=false;
如果需要绑定下拉列表类似的字段 。那么创建对应的数据类型 。然后调用绑定函数绑定即可 。如下面代码
【Winform应用系统如何快速开发 winform下拉列表控件】//添加下拉列表列 。并绑定数据源this.gridView1.CreateColumn("FeeType","费用类型",100).CreateComboBox().BindDictItems("费用类型");
如果是一些特殊的输入需要设置格式显示或者掩码 。那么如下所示
//创建日期列并指定格式varOccurTime=this.gridView1.CreateColumn("OccurTime","发生时间",120).CreateDateEdit();OccurTime.EditMask="yyyy-MM-ddHH:mm";OccurTime.DisplayFormat.FormatString="yyyy-MM-ddHH:mm";
另外有一个值得注意的就是我们新增一行从表记录的时候 。需要记录一些主表的属性 。这样的话 。我们就是在行初始化的时候 。赋值给从表的隐藏列即可 。
//新增行的内容初始化this.gridView1.InitNewRow+=(s,e)=>{gridView1.SetRowCellValue(e.RowHandle,"ID",Guid.NewGuid().ToString());gridView1.SetRowCellValue(e.RowHandle,"Header_ID",tempInfo.ID);gridView1.SetRowCellValue(e.RowHandle,"Apply_ID",tempInfo.Apply_ID);gridView1.SetRowCellValue(e.RowHandle,"OccurTime",DateTime.Now);};
在界面中如果我们需要显示主表的信息 。那么就根据条件获取对应的主表记录对象 。然后显示给界面控件即可 。
///<summary>///显示常规的对象内容///</summary>///<paramname="info"></param>privatevoidDisplayInfo(ReimbursementInfoinfo){tempInfo=info;//重新给临时对象赋值 。使之指向存在的记录对象txtCategory.Text=info.Category;txtReason.Text=info.Reason;txtTotalAmount.Value=http://www.shenzhoubaby.com/info.TotalAmount;txtNote.Text=info.Note;}
而保存的时候 。我们把界面内容重新赋值给对应的主表对象 。
///<summary>///编辑或者保存状态下取值函数///</summary>///<paramname="info"></param>privatevoidSetInfo(ReimbursementInfoinfo){info.Category=txtCategory.Text;info.Reason=txtReason.Text;info.TotalAmount=txtTotalAmount.Value;info.Note=txtNote.Text;info.ApplyDate=DateTime.Now;info.ApplyDept=base.LoginUserInfo.DeptId;info.CurrentLoginUserId=base.LoginUserInfo.ID;}
而我们需要获取GridView明细输入的时候 。就通过一个函数遍历获取GridView的行记录 。转换为相应的对象即可 。如下所示 。
///<summary>///获取明细列表///</summary>///<returns></returns>privateList<ReimbursementDetailInfo>GetDetailList(){varlist=newList<ReimbursementDetailInfo>();for(inti=0;i<this.gridView1.RowCount;i++){vardetailInfo=gridView1.GetRow(i)asReimbursementDetailInfo;if(detailInfo!=null){list.Add(detailInfo);}}returnlist;}
这样处理完这些信息后 。我们就可以在主表保存的时候 。同时保存明细表信息即可 。
///<summary>///新增状态下的数据保存///</summary>///<returns></returns>publicoverrideboolSaveAddNew(){ReimbursementInfoinfo=tempInfo;//必须使用存在的局部变量 。因为部分信息可能被附件使用SetInfo(info);info.Creator=base.LoginUserInfo.ID;info.CreateTime=DateTime.Now;try{#region新增数据boolsucceed=BLLFactory<Reimbursement>.Instance.Insert(info);if(succeed){//可添加其他关联操作varlist=GetDetailList();foreach(vardetailInfoinlist){BLLFactory<ReimbursementDetail>.Instance.InsertUpdate(detailInfo,detailInfo.ID);}returntrue;}#endregion}catch(Exceptionex){LogTextHelper.Error(ex);MessageDxUtil.ShowError(ex.Message);}returnfalse;}
其中代码
BLLFactory<ReimbursementDetail>.Instance.InsertUpdate(detailInfo,detailInfo.ID);
可以对新增记录保存 。也可以对存在的记录进行更新 。
通过上面的介绍 。我们可以看到不同的主从表其实逻辑还是很通用的 。我们可以把它们的逻辑抽取出来 。通过代码生成工具进行快速生成即可 。