gpt4 book ai didi

c# - CommandParameter 上的 DataContext 与 Command 本身上的 DataContext 不同

转载 作者:行者123 更新时间:2023-11-30 17:38:35 26 4
gpt4 key购买 nike

这个问题与我之前在这里的问题有关:Predicate won't validate parameter correctly

首先是关于我的模型的一些信息:

基础 View 模型:

public abstract class BaseViewModel : INotifyPropertyChanged 
{
public event PropertyChangedEventHandler PropertyChanged;

private string _name;
public string Name
{
get
{
return _name;
}
set
{
if (Name != value)
{
_name = value;
OnPropertyChanged("Name");
}
}
}

private BaseViewModel _homePage;
public BaseViewModel HomePage
{
get
{
return _homePage;
}
set
{
if (HomePage != value)
{
_homePage = value;
OnPropertyChanged("HomePage");
}
}
}

public void OnPropertyChanged(string propertyName)
{
PropertyChangedEventHandler temp = PropertyChanged;
if (temp != null)
{
temp(this, new PropertyChangedEventArgs(propertyName));
}
}
}

问题:

<Button Content="{Binding Name}" Command="{Binding DataContext.ChangePageCommand, RelativeSource={RelativeSource AncestorType={x:Type Window}}}"
CommandParameter="{Binding HomePage}"/>

正如您在 BaseViewModel 中看到的,属性“Name”和“HomePage”在同一类中定义,因此如果 DataContext 是派生自“BaseViewModel”的 ViewModel,则它们应该是可访问的。起初我失去了理智,因为似乎没有任何效果 - 输出窗口显示无法检索“主页”的值 - 除了每次都正确绑定(bind)的名称之外。

在我几乎要放弃之后,我将我的 View 命名为“Test”并尝试沿 ElementName 重定向 CommandProperty 绑定(bind) - 它成功了:

<Button Content="{Binding Name}" Command="{Binding DataContext.ChangePageCommand, RelativeSource={RelativeSource AncestorType={x:Type Window}}}"
CommandParameter="{Binding DataContext.HomePage, ElementName=Test}"/>

但是为什么?为什么可以在不分配 ElementName 的情况下绑定(bind)的 Name 属性和需要 ElementName 的 HomePage 之间的 DataContext 不同?

更新 1:

MainViewModel : BaseViewModel

private RelayCommand _command;
public RelayCommand ChangePageCommand {
get {
return _command ?? (_command = new RelayCommand(p => ChangeViewModel((BaseViewModel)p), x => {
return x is BaseViewModel;
}));
}
}

public void ChangeViewModel(BaseViewModel viewModel) {
CurrentPageViewModel = viewModel;
}

更新 2:

<Window.Resources>
<DataTemplate DataType="{x:Type home:HomeViewModel}">
<home:Home/>
</DataTemplate>
<DataTemplate DataType="{x:Type ua:UserAdministrationViewModel}">
<ua:UserAdministration/>
</DataTemplate>
</Window.Resources>
<ContentControl Content="{Binding CurrentPageViewModel}"/>

更新 3:

现在它变得非常奇怪 - 我尝试添加一个 TextBlock 并将其直接绑定(bind)到 HomePage.Text - 它有效。

<Button Height="50" Content="{Binding Name}" Command="{Binding DataContext.ChangePageCommand, RelativeSource={RelativeSource AncestorType={x:Type Window}}}"
CommandParameter="{Binding DataContext.HomePage, ElementName=Test}"/>
<TextBlock Text="{Binding HomePage.Name}"/>

那么,为什么在绑定(bind)到 CommandParameter 时不能直接访问 HomePage,而是将 TextBlock 的 Text 属性直接绑定(bind)到 HomePage.Name 就可以了?

更新 4:

输出窗口说:

System.Windows.Data 信息:10:无法使用绑定(bind)检索值,并且不存在有效的回退值;使用默认代替。绑定(bind)表达式:路径=名称;数据项=空;目标元素是 'Button' (Name='');目标属性是“内容”(类型“对象”)

System.Windows.Data 信息:10:无法使用绑定(bind)检索值,并且不存在有效的回退值;使用默认代替。 BindingExpression:Path=DataContext.ChangePageCommand;数据项=空;目标元素是“按钮”(名称='');目标属性是“命令”(类型“ICommand”)

System.Windows.Data 信息:10:无法使用绑定(bind)检索值,并且不存在有效的回退值;使用默认代替。绑定(bind)表达式:路径=主页;数据项=空;目标元素是 'Button' (Name='');目标属性是“CommandParameter”(类型“Object”)

System.Windows.Data 信息:10:无法使用绑定(bind)检索值,并且不存在有效的回退值;使用默认代替。 BindingExpression:Path=HomePage.Name;数据项=空;目标元素是“TextBlock”(名称='');目标属性是“文本”(类型“字符串”)

但是除了 CommandParameter 之外的所有东西都成功绑定(bind)了。刷新绑定(bind)后,CommandParameter(和底层 CanExecute-Method)是否有可能不会重新验证?也许绑定(bind)一开始会失败,但除了 CommandParameter 之外的每个绑定(bind)都会得到刷新和重新验证。但是我该如何解决这个问题呢?

最佳答案

我很自豪地宣布我已经解决了这个愚蠢的问题:

我假设我用来将我的 CommandBinding 重定向到 Window 的 ViewModel 的 RelativeSource 破坏了所有后续属性(!不是控件)的 DataContext。

不工作:

<Button Grid.Row="0" 
Grid.Column="0"
Height="50"
Content="{Binding PreviousPageViewModel.Name}"
Command="{Binding DataContext.ChangePageCommand, RelativeSource={RelativeSource AncestorType=Window}}"
CommandParameter="{Binding PreviousPageViewModel}" />

工作:

<Button Grid.Row="0" 
Grid.Column="0"
Height="50"
Content="{Binding PreviousPageViewModel.Name}"
CommandParameter="{Binding PreviousPageViewModel}"
Command="{Binding DataContext.ChangePageCommand, RelativeSource={RelativeSource AncestorType=Window}}"/>

所以在这种情况下,属性的顺序很重要。

注意:以下所有控件的 DataContext 都是正确的。

关于c# - CommandParameter 上的 DataContext 与 Command 本身上的 DataContext 不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36589122/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com