当表现控件触发DataBinding事件,分页控件就可以获取C# DataSource属性。遗憾的是,微软没有提供所 有数据绑定类实现的接口,诸如IdataSourceProvider之类,而且并非所有从Control或WebControl类继承的控件都有一个 C# DataSource属性,因此向上定型成Control类没有意义,唯一可行的办法是通过Reflection API直接操作DataSoruce属性。在讨论事件句柄方法之前,应该指出的是,为了注册事件句柄,首先必须获得一个表现控件的引用。分页控件显露了一 个简单的字符串属性BindToControl:
- public string BindToControl
- ...{
- get
- ...{
- if (_bindcontrol == null)
- throw new NullReferenceException
("在使用分页控件之前,请先通过设置BindToControl属性绑定到一个控件。");- return _bindcontrol;}
- set...{_bindcontrol=value;}
- }
这个方法非常重要,所以最好能够抛出一个含义更明确的信息,而不是抛出标准的NullReferenceException异常。在分页控件的 OnInit方法中,我们解析了对表现控件的引用。本例应当用OnInit事件句柄(而不是构造函数)来确保JIT编译的aspx页面已经设置了 BindToControl。
- protected override void OnInit(EventArgs e)
- ...{
- _boundcontrol = Parent.FindControl(BindToControl);
- BoundControl.DataBinding += new EventHandler(BoundControl_DataBound);
- base.OnInit(E);
- ...
- }
搜索表现控件的操作通过搜索分页控件的Parent控件完成,在这里,Parent就是页面本身。按照这种方式使用Parent比较危险,举例来说,如果分页控件嵌入到了另一个控件之中,例如嵌入到了Table控件之中,则Parent引用实际上将是一个对Table控件的引用。由于 FindControl方法只搜索当前的控件集合,除非表现控件就在该集合之中,否则不可能搜索到。一种比较安全的方法是递归地搜索各个控件集合,直至找 到目标控件为止。
找到BoundControl之后,我们将分页控件注册成为DataBinding事件的监听器。由于分页控件要操作数 据源,所以该事件句柄应当是调用链中的最后一个,这一点很重要。不过,只要表现控件在OnInit事件句柄中注册DataBinding的事件句柄(默认 行为),分页控件操作数据源时就不会出现问题。
DataBound事件句柄负责获取表现控件的C# DataSource属性。
- private void BoundControl_DataBound(object sender,System.EventArgs e)
- ...{
- if (HasParentControlCalledDataBinding) return;
- Type type = sender.GetType();
- _datasource = type.GetProperty("DataSource");
- if (_datasource == null)
- throw new NotSupportedException("分页控件要求表现控件必需包含一个DataSource。");
- object data = _datasource.GetGetMethod().Invoke(sender,null);
- _builder = Adapters[data.GetType()];
- if (_builder == null)
- throw new NullReferenceException
("没有安装适当的适配器来处理下面的数据源类型:"+data.GetType());- _builder.Source = data;
- ApplyDataSensitivityRules();
- BindParent();
- RaiseEvent(DataUpdate,this);
- }
【编辑推荐】