博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
企业管理软件开发之九 以数据绑定为基础的控件只读,创建时可写,必须大写,必须小写的原理与实现...
阅读量:6161 次
发布时间:2019-06-21

本文共 4634 字,大约阅读时间需要 15 分钟。

以LLBL Gen作为ORM数据访问框架,生成实体层,然后在实体层中绑定业务逻辑。有以下几个好处:

1  强类型对象,可以编译时发现错误,而不是运行时错误

2  可以借助于反射做扩展,灵活性高。

情境设定

如何让销售单中的客户编号,只能在第一次输入的时候,可以修改,保存后不能修改,如何实现?

最简单的实现方法是,在界面的OnLoad事件中,加上一行代码:

txtCustomerNo.ReadOnly=true;

但是如果一个界面中有10个字段有这种需求,则需要写10行。

很明显,发生了代码重复,为解决这一通用的问题,请参考下面我的做法。

 

首先,打开LLBL Gen,添加业务对象,设计属性:

依照我的经验,添加了五个自定义属性,它的含义如下

CapsLock 表示属性输入时,会转化为大写,比如供应商编号VENDORNO001,VENDOR002,而不允许小写字母出现

ReadOnly 表示属性为只读,不允许修改。比如采购单中的供应商名称从供应商主档中带值过来,但不允许修改

AllowEditForReadOnly 表示属性只有在创建时可修改,一旦数据保存后,不允许修改。比如输入供应商发票,输入供应商编号后,带出供应商的首选货币,当前的兑换率,保存之后,不能再修改供应商编号,以防止日记帐数据与主档数据不匹配,减少重复。

Hidden 表示属性由系统生成,不需要界面人工调整。比如日记帐中的修改日期,修改人,由系统管理。

Required 表示属性必须输入,比如采购单必须输入供应商,销售单必须输入客户编号。

这五个属性,在界面中用得相当普遍,把它放到基础框架中,可以节省大量的重复代码。

 

在LLBL Gen设计器中按F7生成代码。这些自定义的属性会添加到类型的SetupCustomPropertyHashtables方法中。

#region Custom Property Hashtable Setup///  Initializes the hashtables for the entity type and entity field custom properties. private static void SetupCustomPropertyHashtables(){            _customProperties = new Dictionary
(); _fieldsCustomProperties = new Dictionary
>(); Dictionary
fieldHashtable; fieldHashtable = new Dictionary
(); _fieldsCustomProperties.Add("Alternate", fieldHashtable); fieldHashtable = new Dictionary
(); _fieldsCustomProperties.Add("AssemblyLine", fieldHashtable); fieldHashtable = new Dictionary
(); _fieldsCustomProperties.Add("CreatedBy", fieldHashtable); fieldHashtable = new Dictionary
(); _fieldsCustomProperties.Add("CreatedDate", fieldHashtable); fieldHashtable = new Dictionary
(); fieldHashtable.Add("ReadOnly", @""); _fieldsCustomProperties.Add("Description", fieldHashtable);......}

再回到界面的控件中,对它进行数据绑定,设计效果如下图所示

如果设计界面中看不明白,可以看下面的代码,设置数据绑定成员。

this.txtCcy.AutoFind = true;this.txtCcy.CharacterCasing = System.Windows.Forms.CharacterCasing.Upper; this.txtCcy.DataBindings.Add(new System.Windows.Forms.Binding("Value", this.currencyBindingSource, "Ccy", true));this.txtCcy.Location = new System.Drawing.Point(118, 12);this.txtCcy.Lookup.FilterName = "Non Suspended";this.txtCcy.Lookup.LookupName = "CurrencyLookup";this.txtCcy.Name = "txtCcy";this.txtCcy.Required = true;this.txtCcy.Size = new System.Drawing.Size(100, 21);this.txtCcy.TabIndex = 0;
 

最重要的是第三行,对它进行数据成员绑定。Windows Form数据绑定的好处是双向绑定。如果有对控件绑定属性,修改控件的绑定属性值后,数据源即更新为新的值,反之,修改数据源,也同时会修改控件中的显示值。

因为这个特性,所以可以借助于反射对它进行深度的封装,请参考下面的代码例子:

public override EntityBase2 LoadEntity(string refNo){            IItemManager manager = ClientProxyFactory.CreateProxyInstance
(); ItemEntity customer = manager.GetItem(refNo); return customer;}public override void DeleteEntity(EntityBase2 entity){ ItemEntity user = (ItemEntity)entity; IItemManager manager = ClientProxyFactory.CreateProxyInstance
(); manager.DeleteItem(user);}public override void SaveEntity(EntityBase2 entity){ ItemEntity user = (ItemEntity)entity; IItemManager manager = ClientProxyFactory.CreateProxyInstance
(); manager.SaveItem(user); }

如代码所示,界面代码完全不需要知道是什么值被用户更改过,只需要做数据验证,在页面加载时把数据绑定到界面中,页面关闭时,把用户修改过的数据写回到数据库中。

 

再回到控件设计中,给它添加自定义属性绑定代码。

public void InitLayoutFromBinding(){          InitLayoutFromBinding(false);}public void InitLayoutFromBinding(bool forceReinit){           Binding binding = null;           if (this.DataBindings["Value"] != null)           {                binding = this.DataBindings["Value"];            }            else if (this.DataBindings["Text"] != null)            {                binding = this.DataBindings["Text"];            }            InitLayoutFromBinding(binding, forceReinit);}public void InitLayoutFromBinding(Binding binding){            InitLayoutFromBinding(binding, false);}
 
 

如您所看到的代码,这一层直接通过获取BindingSource的绑定属性,把第一步中设计的五个自定义属性值写到控件中。

举例说明如下:SalesOrder的Customer No属性被添加上面所列出的五个属性值中的ReadOnly,AllowEditForNewOnly,CapsLock三个属性,绑定控件txtCustomerNo到SalesOrder.CustomerNo属性。在窗体启动时,对已经绑定过属性的成员进行一个遍历操作,读取到它的BindingMember是CustomerNo属性

再通过反射读取到

BindingMemberInfo bindingInfo = binding.BindingMemberInfo;EntityBase2 entity = ComponentCommon.GetBoundEntity(this, binding);IEntityField2 field = null;_bindingField = bindingInfo.BindingField;Dictionary
fieldsCustProps = ComponentCommon.GetFieldsCustomProperties(entity, bindingInfo.BindingField);if (fieldsCustProps != null){ if (fieldsCustProps.ContainsKey("CapsLock")) { CharacterCasing = CharacterCasing.Upper; }......

核心的骨架代码已经全盘托出,稍微加以整理即可实现这个特性,应用到您的实际项目中,减少代码重复。

转载地址:http://uhhfa.baihongyu.com/

你可能感兴趣的文章
Eclipsed的SVN插件不能识别之前工作空间的项目
查看>>
Linux 查看iptables状态-重启
查看>>
amazeui学习笔记一(开始使用2)--布局示例layouts
查看>>
c#中lock的使用(用于预约超出限额的流程)
查看>>
ODI基于源表时间戳字段获取增量数据
查看>>
并发容器之CopyOnWriteArrayList(转载)
查看>>
什么是AAC音频格式 AAC-LC 和 AAC-HE的区别是什么
查看>>
原创:goldengate从11.2升级到12.1.2
查看>>
Quartz
查看>>
正则表达式的语法规则
查看>>
C#一个关于委托和事件通俗易懂的例子
查看>>
类似于SVN的文档内容差异对比工具winmerge
查看>>
Cause: java.sql.SQLException: The user specified as a definer ('root'@'%') does not exist
查看>>
quratz线程
查看>>
execnet: rapid multi-Python deployment
查看>>
windows修改3389端口
查看>>
关于JavaScript词法
查看>>
FreeSwitch中的会议功能(4)
查看>>
MySQL中创建用户分配权限(到指定数据库或者指定数据库表中)
查看>>
AutoReleasePool 和 ARC 以及Garbage Collection
查看>>