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值就可以了。利用这个思路,我们构造一个类来实现这个方案,简易代码如下:
public partial class DependencyPropertyWatcher
{
TextBlock tb = new TextBlock();
public void Attach(object source, string property, Action<object> onChange)
{
tb.SetBinding(TextBlock.TextProperty, new Binding()
{
Path = new PropertyPath(property),
Source = source,
Converter = new Spyer(onChange)
});
}
}
partial class DependencyPropertyWatcher
{
class Spyer : IValueConverter
{
Action<object> _OnChange;
public Spyer(ActiononChange)
{
_OnChange = onChange;
}
public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
if (_OnChange != null)
{
_OnChange(value);
}
return null;
}
public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotImplementedException();
}
}
}
这样,我们可以给任何一个DependencyProperty挂接一个Changed的回调方法。值得注意的是,这个类必须被实例化并被引用,不然tb被GC回收挂接就无效了。s
浙公网安备 33010602011771号