Cast Binding Path,以便在Design-Time识别ViewModel属性

AAA*_*ddd 10 c# xamarin.forms

好吧,这更令人烦恼而不是问题.没有错误

<ContentPage
   ...
   x:Name="This"
   //hack to have typed xaml at design-time
   BindingContext="{Binding Source={x:Static viewModels:ViewModelLocator.ChooseTargetLocationVm}}"
Run Code Online (Sandbox Code Playgroud)

子视图

<views:ProductStandardView
    ...
    BindingContext="{Binding Product}">
    <Grid.Triggers>
        <DataTrigger
            Binding="{Binding Path=BindingContext.IsVacate, Source={x:Reference This}}"
            TargetType="Grid"
            Value="true">
            <Setter Property="BackgroundColor" Value="{StaticResource WarningColor}" />
        </DataTrigger>
    </Grid.Triggers>
Run Code Online (Sandbox Code Playgroud)

绑定BindingContext 参考This,我得到一个XAML"警告"

无法在"对象"类型的数据上下文中解析属性"IsVacate"

Binding="{Binding Path=BindingContext.IsVacate, Source={x:Reference This}}"
Run Code Online (Sandbox Code Playgroud)

显然,BindingContext是一个对象并且是无类型的.但是上面的代码编译和工作

我想要做的就是施放它,首先是因为我有OCD,但主要是因为它很容易在IDE页面通道栏上发现真正的问题

以下似乎合乎逻辑但不起作用

Binding="{Binding Path=BindingContext.(viewModels:ChooseTargetLocationVm.IsVacate), 
                  Source={x:Reference This}}"
Run Code Online (Sandbox Code Playgroud)

在输出中我得到

[0:]绑定:'(viewModels:ChooseTargetLocationVm'找不到属性' Inhouse.Mobile.Standard.ViewModels.ChooseTargetLocationVm',目标属性:' Inhouse.Mobile.Standard.Views.ProductStandardView.Bound'

我理解错误,但我还会怎么样?


只是为了愚蠢,显然以下不会编译

Binding="{Binding Path=((viewModels:ChooseTargetLocationVm)BindingContext).IsVacate, Source={x:Reference This}}"
Run Code Online (Sandbox Code Playgroud)

那么有没有办法将BindingContext转换为ViewModel,以便在设计时键入任何SubProperty引用?

更新

这对于内部DataTemplate或在这种情况下控件有其自身这是相关的,BindingContext这就是我需要使用Source={x:Reference This}目标页面的原因.

注意:<ContentPage.BindingContext>对我来说不起作用,因为我使用的是棱镜和统一,并且在初始测试中似乎没有很好地使用默认构造函数,尽管我可能会更多地使用它

Sha*_*raj 5

您可以扩展ContentPage以创建通用类型-支持视图模型的类型参数-进而可以在Binding标记扩展中使用。

虽然它可能不会给您提供智能支持,但绝对应该为您删除警告。

例如:

/// <summary>
/// Create a base page with generic support
/// </summary>
public class ContentPage<T> : ContentPage
{
    /// <summary>
    /// This property basically type-casts the BindingContext to expected view-model type
    /// </summary>
    /// <value>The view model.</value>
    public T ViewModel { get { return (BindingContext != null) ? (T)BindingContext : default(T); } }

    /// <summary>
    /// Ensure ViewModel property change is raised when BindingContext changes
    /// </summary>
    protected override void OnBindingContextChanged()
    {
        base.OnBindingContextChanged();

        OnPropertyChanged(nameof(ViewModel));
    }
}
Run Code Online (Sandbox Code Playgroud)

样品用法

<?xml version="1.0" encoding="utf-8"?>
<l:ContentPage 
    ...
    xmlns:l="clr-namespace:SampleApp" 
    x:TypeArguments="l:ThisPageViewModel"
    x:Name="This"
    x:Class="SampleApp.SampleAppPage">

    ...                            
         <Label Text="{Binding ViewModel.PropA, Source={x:Reference This}}" />
    ...
</l:ContentPage>
Run Code Online (Sandbox Code Playgroud)

代码隐藏

public partial class SampleAppPage : ContentPage<ThisPageViewModel>
{
    public SampleAppPage()
    {
        InitializeComponent();

        BindingContext = new ThisPageViewModel();
    }
}
Run Code Online (Sandbox Code Playgroud)

查看模型

/// <summary>
/// Just a sample viewmodel with properties
/// </summary>
public class ThisPageViewModel
{
    public string PropA { get; } = "PropA";
    public string PropB { get; } = "PropB";
    public string PropC { get; } = "PropC";

    public string[] Items { get; } = new[] { "1", "2", "3" };
}
Run Code Online (Sandbox Code Playgroud)