您的位置:首页 > 其它

利用Binding扩展Silverlight控件(一)

2011-03-28 17:15 218 查看
WPF由于其局限性,目前并不普及,但是其控件模型设计也许是有史以来最优秀的。Silverlight很好的继承了这一特性,各种模版尤其是ControlTemplate的引入让我们很轻易的可以开发出五花八门的控件,但是也正由于它的灵活强大,在基础类库中的控件并没有提供足够多的属性和事件来让我们使用,在很多时候只需要改变一个微小的特性,不得不复制大量ControlTemplate的xaml修改其他的一两处代码,然后在继承类中也做一点小的修改。

SL有控件有大量和WinForm对应控件相似的属性,但是和WinForm不一样,在SL中我们找到了名为XXX属性之后,很难发现有与之关联的XXXChanged事件,而这是通过最小的修改量改变现有控件外观和行为的基础。还好SL强大的DependencyProperty属性和Binding类为这一实现提供了可行方法,本文的第一部分我们先实现一个方案,使任何一个DependencyProperty特性的属性都可以有Changed的事件通知供我们使用,然后利用这一点来实现2个实例:带有文字说明的DataPager控件和更多控制的ComboBox。

让一个DataPager带有一个常见的类似”总计xxx记录共xx页“的文字说明思路很简单,在OnApplyTemplate中利用GetTemplateChild来取得主要的呈现容器对象(SL4中这是一个水平排列的StackPanel),然后在StackPanel的Children中添加一个TextBlock,接下来要做的就是在DataPager的Source属性在发生变化的时候更新TextBlock的文本,很自然,我们想到了SourceChanged事件,很不幸的是它并不存在,甚至在派生类中也不能找到OnSourceChanged方法来重写,在派生类中重写一个Source来覆盖父类的Source属性显然不是一个好办法,是的我们仅仅需要的是一个SourceChanged事件而已,而Binding一个DependencyProperty属性正好可以帮我们达到这一个目的,因为我们绑定一个DependencyProperty之后,被绑定属性值的变化都可以正确的反应到我们的目标对象。

整个流程已经很清晰了:把Source属性绑定到一个对象,当Source发生变化时,由于Binding的存在,绑定引擎会把这变化通知到目标对象。就在这个通知过程中,我们利用Binding的Converter来截获这个新的Source值就可以了。利用这个思路,我们构造一个类来实现这个方案,简易代码如下:

publicpartialclassDependencyPropertyWatcher { TextBlocktb=newTextBlock(); publicvoidAttach(objectsource,stringproperty,Action<object>onChange)
{
tb.SetBinding(TextBlock.TextProperty,newBinding()
{
Path=newPropertyPath(property),
Source=source,
Converter=newSpyer(onChange)
});
}
}
partialclassDependencyPropertyWatcher
{
classSpyer:IValueConverter
{
Action<object>_OnChange;
publicSpyer(ActiononChange)
{
_OnChange=onChange;
}

publicobjectConvert(objectvalue,TypetargetType,objectparameter,System.Globalization.CultureInfoculture)
{
if(_OnChange!=null)
{
_OnChange(value);
}
returnnull;
}

publicobjectConvertBack(objectvalue,TypetargetType,objectparameter,System.Globalization.CultureInfoculture)
{
thrownewNotImplementedException();
}
}
} 这样,我们可以给任何一个DependencyProperty挂接一个Changed的回调方法。值得注意的是,这个类必须被实例化并被引用,不然tb被GC回收挂接就无效了。s
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: