企业管理软件开发之九 以数据绑定为基础的控件只读,创建时可写,必须大写,必须小写的原理与实现

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

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

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

情境设定

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

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

txtCustomerNo.ReadOnly=true;

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

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

 

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

image

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

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

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

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

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

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

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

 

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

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

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

image

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

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<IItemManager>();
            ItemEntity customer = manager.GetItem(refNo);
            return customer;
}

public override void DeleteEntity(EntityBase2 entity)
{
            ItemEntity user = (ItemEntity)entity;
            IItemManager manager = ClientProxyFactory.CreateProxyInstance<IItemManager>();
            manager.DeleteItem(user);
}

public override void SaveEntity(EntityBase2 entity)
{
            ItemEntity user = (ItemEntity)entity;
            IItemManager manager = ClientProxyFactory.CreateProxyInstance<IItemManager>();
            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<string, string> fieldsCustProps = ComponentCommon.GetFieldsCustomProperties(entity, bindingInfo.BindingField);
if (fieldsCustProps != null)
{
     if (fieldsCustProps.ContainsKey("CapsLock"))
      {
          CharacterCasing = CharacterCasing.Upper;
      }
......

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

posted @ 2013-07-11 09:18  信息化建设  阅读(1844)  评论(2编辑  收藏  举报