ListView、UserControl、DependencyProperty、ObservableCollection

本文关键字:ObservableCollection DependencyProperty UserControl ListView | 更新日期: 2023-09-27 18:31:06

我的绑定不起作用。我搜索了错误,但我不明白如何在我的情况下解决它。

System.Windows.Data 错误: 1 : 无法创建默认转换器以在类型"MyApplication.MyUserControl"和"MyApplication.Person"之间执行"单向"转换。请考虑使用绑定的转换器属性。绑定表达式:路径=;DataItem='MyUserControl' (Name='');目标元素是"我的用户控件"(名称=");目标属性为"人员信息"(类型"人员")

System.Windows.Data 错误: 5 : BindingExpression 生成的值对目标属性无效。值='MyApplication.MyUserControl' BindingExpression:Path=;DataItem='MyUserControl' (Name='');目标元素是"我的用户控件"(名称=");目标属性为"人员信息"(类型"人员")

基本上,它是一个绑定到类 Person 的可观察集合的列表视图。

MainWindow.xaml.cs

public partial class MainWindow : Window
{
    public ObservableCollection<Person> PersonCollection { set; get; }
    public MainWindow()
    {
        PersonCollection = new ObservableCollection<Person>();
        InitializeComponent();
        PersonCollection.Add(new Person() { Name = "Bob", Age = 20 });
    }
}

MainWindow.xaml

<Window DataContext="{Binding RelativeSource={RelativeSource Self}}" xmlns:self="clr-namespace:MyApplication" x:Class="MyApplication.MainWindow" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<ListView ItemsSource="{Binding PersonCollection}">
    <ListView.ItemTemplate>
        <DataTemplate>
            <StackPanel>
                <self:MyUserControl PersonInfo="{Binding}" />
            </StackPanel>
        </DataTemplate>
    </ListView.ItemTemplate>
</ListView>
</Window>

MyUserControl.xaml.cs

public partial class MyUserControl : UserControl
{
    public static readonly DependencyProperty PersonProperty = DependencyProperty.Register("PersonInfo", typeof(Person), typeof(MyUserControl));
    public Person PersonInfo
    {
        get { return (Person)GetValue(PersonProperty); }
        set { SetValue(PersonProperty, value); }
    }
    public MyUserControl()
    {
        InitializeComponent();
    }
}

MyUserControl.xaml

<UserControl DataContext="{Binding RelativeSource={RelativeSource Self}}" x:Class="MyApplication.MyUserControl" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006">
<TextBlock Text="{Binding PersonInfo.Name}" />
</UserControl>

人.cs

public class Person : INotifyPropertyChanged
{
    public int Age { set; get; }
    public string Name { set; get; }
    public event PropertyChangedEventHandler PropertyChanged;
    private void NotifyPropertyChanged(String info)
    {
        if (PropertyChanged != null)
            PropertyChanged(this, new PropertyChangedEventArgs(info));
    }
}

ListView、UserControl、DependencyProperty、ObservableCollection

我不太明白你为什么要让它变得那么复杂。您可以在没有 PersonInfo 属性且无需修改其 DataContext 的情况下轻松绑定用户控件。

<UserControl x:Class="MyApplication.MyUserControl" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"> 
    <TextBlock Text="{Binding Name}" /> 
</UserControl>

然后将用户控件放在没有显式绑定的数据模板中。然后,其 DataContext 将已经包含一个 Person 对象。

<DataTemplate>      
    <StackPanel>      
        <self:MyUserControl />      
    </StackPanel>      
</DataTemplate>      

虽然你解决了你的问题,但你的整个绑定代码对我来说似乎是错误的,所以我提出了这个替代方案:

为所有绑定源对象提供一个基类 - ObservableObject.cs

public abstract class ObservableObject : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;
    protected virtual void OnPropertyChanged(PropertyChangedEventArgs e)
    {
        var handler = this.PropertyChanged;
        if (handler != null)
            handler(this, e);
    }
    protected void SetValue<T>(ref T field, T value, string propertyName)
    {
        if (!EqualityComparer<T>.Default.Equals(field, value))
        {
            field = value;
            this.OnPropertyChanged(new PropertyChangedEventArgs(propertyName));
        }
    }
}

有一个主窗口的视图模型 - 主窗口模型.cs

public class MainWindowModel : ObservableObject
{
    private readonly ObservableCollection<Person> personCollection = new ObservableCollection<Person>()
    {
        new Person() { Name = "Bob", Age = 20 }
    };
    public ObservableCollection<Person> PersonCollection
    {
        get { return this.personCollection; }
    }
}

MainWindow.xaml.cs现在基本上是空的。

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
    }
}

MainWindow.xaml 将 DataContext 设置为新的 MainWindowModel 实例。

<Window x:Class="MyApplication.MainWindow" 
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:self="clr-namespace:MyApplication">
    <Window.DataContext>
        <self:MainWindowModel/>
    </Window.DataContext>
    <ListView ItemsSource="{Binding PersonCollection}">
        <ListView.ItemTemplate>
            <DataTemplate>
                <StackPanel>
                    <self:MyUserControl/>
                </StackPanel>
            </DataTemplate>
        </ListView.ItemTemplate>
    </ListView>
</Window>

MyUserControl.xaml.cs 也基本上是空的(只包含自动生成的代码)。

public partial class MyUserControl : UserControl
{
    public MyUserControl()
    {
        InitializeComponent();
    }
}

MyUserControl.xaml

<UserControl x:Class="MyApplication.MyUserControl" 
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <TextBlock Text="{Binding Name}"/>
</UserControl>

人.cs

public class Person : ObservableObject
{
    private int age;
    private string name;
    public int Age
    {
        get { return this.age; }
        set { this.SetValue(ref this.age, value, "Age"); }
    }
    public string Name
    {
        get { return this.name; }
        set { this.SetValue(ref this.name, value, "Name"); }
    }
}

将用户控件 XAML 更改为

<UserControl x:Class="MyApplication.MyUserControl"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006">
    <TextBlock DataContext="{Binding RelativeSource={RelativeSource AncestorType=UserControl}}"
               Text="{Binding PersonInfo.Name}" />
</UserControl>

以下是对 DataContext 问题的良好解释。