王燕丽

博客园 首页 联系 订阅 管理

      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
posted on 2011-03-28 17:15  王燕丽  阅读(808)  评论(0)    收藏  举报