绑定的静态验证

Fyo*_*kin 11 wpf binding

或者" 你如何确保你的所有绑定都保持正确? "
(这有点长,但请耐心等待,我试着让它尽可能短)

请考虑以下示例:

    <TextBox Name="tb" />
    <TextBlock Text="{Binding Text.TheProp, ElementName=tb}" />
Run Code Online (Sandbox Code Playgroud)

在编译时完全知道绑定是不正确的(即解析器知道元素的类型tb,因此,它知道它的Text属性的类型,因此,它知道它TheProp不存在).
但是,此代码将编译并运行(尽管在调试输出中有绑定错误消息).

在某些情况下,这种行为可能非常方便:无论我的数据类型是什么,只要它具有适当命名的属性,我就可以了.因此,我们得到了一种"声明性的鸭子打字".

然而,鸭子打字并不总是一件好事.
具体来说,在使用MVVM模式时,我(大多数时候)知道所有ViewModel对象的确切类型.另一方面,模型随着时间的推移变得越来越复杂,这使我对未来的重构感到担忧:如果我决定重命名某些属性,或者上帝禁止将它们放在一个单独的聚合对象中,该怎么办?那么我的所有绑定会发生什么?我是否必须手动耙所有XAML文件?即使没有重构 - 如果我只是打错了怎么办?

在XAML的其他地方已经解决了类似的问题.例如,如果您输入了不正确的属性名称Style/Setter/@Property,则会出现编译时错误.
TemplateBinding还提供此类验证.哪个非常方便.

所以,理想情况下,我希望看到这样的事情:

ProductViewModel.cs:

    public class ProductViewModel
    {
        public Name { get; set; }
        public Price { get; set; }
    }
Run Code Online (Sandbox Code Playgroud)

ProductView.XAML:

    <UserControl x:Class="Shopping.View.ProductView"
                 x:DataContextType="vm:ProductViewModel"
                 xmlns:vm="clr-namespace:Shopping.ViewModel"
                 ... >
        <TextBox Text="{Binding Name}" />  <!-- OK -->
        <TextBox Text="{Binding Price}" /> <!-- OK -->
        <TextBox Text="{Binding ABC}" />   <!-- Compile time error: there is no property ABC in ProductViewModel -->
    </UserControl>
Run Code Online (Sandbox Code Playgroud)

ShoppingCart.XAML:

    <UserControl x:Class="Shopping.View.ShoppingCartView"
                 x:DataContextType="vm:ShoppingCartViewModel"
                 xmlns:vm="clr-namespace:Shopping.ViewModel"
                 ... >
        <ItemsControl ItemsSource="{Binding Products}"
                      ItemType="vm:ProductViewModel" >  <!-- Static check happens here 
                                                             ShoppingCartViewModel.Products must 
                                                             implement IEnumerable<ProductViewModel> -->
            <ItemsControl.ItemTemplate>
                <DataTemplate DataType="vm:ProductViewModel">
                    <view:ProductView /> <!-- DataContext is known to be of correct type
                                              because of DataTemplate.DataType property -->
                </DataTemplate>
            </ItemsControl.ItemTemplate>
        </ItemsControl>
    </UserControl>
Run Code Online (Sandbox Code Playgroud)

但让我们回到现实.实际上,所有梦想都不会在不久的将来发生.

但是,我确信我不是第一个遇到这个问题的人.
所以,最后,问题是:你如何确保你的绑定是正确的?而且他们保持这种状态?

Sam*_*ack 8

您的Xaml静态分析如何作为构建后步骤执行?

作为.Net 4的一部分,Microsoft发布了一个新的System.Xaml库,以提供独立于WPF的强大的Xaml解析和序列化支持.他们现在开始在它上面构建各种有趣的东西,其中一些可能会帮助你.

例如,在XamlToolkit中,您将找到XamlDOM,使您可以轻松地对Xaml文件进行静态分析.进一步说,XAMLFxCop规则.

最引人关注的是Rob Relyea的BindingFinder,其明确的目标是在Xaml中检查Bindings的类型.这要求您在Xaml中具有类型提示,例如DataTemplate中的DataType属性,或者视图上的新d:DataContext属性(Blend用于提供设计时数据).然后它使用XamlDOM检查一切是否匹配.

更新: Resharper 6现在提供数据绑定的智能感知,并在您的属性路径错误时提供警告.