实际编程中,因为我们较多地使用到Binding类的Source与Path,所以可能会有一个思维定式,那就是:有可能作为数据源的类一定要准备好一些属性,这些属性将作为Binding的Path。
如果本着这个思想去设计有可能作为数据源的类,那么会有两个问题出现:
1. 这个类的哪些属性有可能作为数据源的Path?是都需要激发NotifyPropertyChanged事件,还是用到了再添加?这很有可能让这个类迟迟不能封闭。
2. 需要用属性把一些方法包装起来,用来暴露给Binding,造成冗余和语义上的不美观。
其实,WPF类库里有一个名为ObjectDataProvider的类就是专门为了解决这个矛盾的——有了这个类,你在设计自己的类的时候就不必总想着把它设计成数据源的事儿了,该怎么抽象就怎么抽象、该怎么封装就怎么封装。
你可能会问:“如果这个类已经封闭了(不再改动)而我又需要拿它当数据源了,碰巧所需要的数据是它某个方法的返回值,没有对应属性,怎么办?”OK,这就是ObjectDataProvider的用武之地了——使用它,可以在你这个类的实例外面加上一层“包装”(或者说是加个壳儿),使它变成一个标准的Binding数据源。如果没记错的话,这应该是著名的“适配器模式”。
下面,我们用一段简单的代码来学习如何使用ObjectDataProvider。
这个例子简单到不能再简单——三个TextBox,在前两个里输入合适的数字,在第三个里会显示它们的和。按照UI与逻辑分开的原则,计算加法的功能应该由某个类来实现。
后台负责计算的类是这样:
- public class Calculator
- {
- public int Add(int arg1, int arg2)
- {
- return arg1 + arg2;
- }
- public string Add(string arg1, string arg2)
- {
- int x = 0;
- int y = 0;
- if (int.TryParse(arg1, out x) && int.TryParse(arg2, out y))
- {
- return this.Add(x, y).ToString();
- }
- else
- {
- return "Input Error!";
- }
- }
- }
大家看到了,设计这个类的时候,涉及到加法运算的逻辑时,任何一个程序员都会很自然地采用一个方法来实现,而不会为了把它做成一个Binding的数据源专门把这些方法封装进属性里——这样就破坏了面向对象的抽象。
然后,让我们看看如何使用ObjectDataProvider来包装这个类。
- <Window x:Class="WpfApplicationAdd.Window1"
- xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
- xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
- xmlns:local="clr-namespace:WpfApplicationAdd"
- xmlns:system="clr-namespace:System;assembly=mscorlib"
- Title="Add" Height="136" Width="230" Background="SteelBlue">
- <Window.Resources>
- <ObjectDataProvider x:Key="odp" ObjectType="{x:Type local:Calculator}" MethodName="Add">
- <ObjectDataProvider.MethodParameters>
- <system:String>0</system:String>
- <system:String>0</system:String>
- </ObjectDataProvider.MethodParameters>
- </ObjectDataProvider>
- </Window.Resources>
- <StackPanel>
- <TextBox x:Name="textBox1" Margin="5" Text="{Binding Source={StaticResource odp}, Path=MethodParameters[0], BindsDirectlyToSource=true, UpdateSourceTrigger=PropertyChanged}" />
- <TextBox x:Name="textBox2" Margin="5" Text="{Binding Source={StaticResource odp}, Path=MethodParameters[1], BindsDirectlyToSource=true, UpdateSourceTrigger=PropertyChanged}"/>
- <TextBox x:Name="textBox3" Margin="5" Text="{Binding Source={StaticResource odp}, Mode=OneWay}"/>
- </StackPanel>
- </Window>
运行起来之后,你就能看到这样的结果了:
网友建议:我建议在实现一个IValueConverter类,作用将string转换为double类型,那么就不需要在Calculator类中实现string为参数的重载方法(因为Calculator类的设计者并不想让string加string变成加法操作,有可能他需要concat操作)。
假设实现类为DoubleValueConverter,在资源中加入<loc:DoubleValueConverter x:Key="dc" />,然后
只需要在xaml中更改为:<TextBox x:Name="textBox1" Margin="5" Text="{Binding Source={StaticResource odp}, Path=MethodParameters[0], BindsDirectlyToSource=true, UpdateSourceTrigger=PropertyChanged,Converter={StaticResource dc}}" />
这样就更优雅些。