我有一个datagrid,它绑定到一个collectionviewsource,它绑定到一个observablecollection。在遵循指南时,我将其设置为:
我的人类:
public class Persons : ObservableCollection<Person>
{
//...
}
xaml数据绑定:
<Window.Resources>
<local:Persons x:Key="_Persons"/>
<CollectionViewSource x:Key="cvsPersons" Source="{StaticResource _Persons}" />
</Window.Resources>
数据网格绑定:
<DataGrid x:Name="myDataGrid" ItemsSource="{Binding Source={StaticResource cvsPersons}}"/>
背后的代码:
_Persons = (Persons)this.Resources["_Persons"];
_persons = //some method to fill perons;
cvsPersons = (CollectionViewSource)this.Resources["cvsPersons"];
cvsPersons.Source = _Persons;
以上作品。 我的问题是,为什么我需要在代码中使用cvsPersons.Source = _Persons设置collectionviewsource.source;?我认为我的第一个片段中的xaml完成了这项工作:
_cvsPersons.Source = _Persons;
如果我需要所有这些代码,那么xaml数据绑定代码似乎没什么好处,我也可以在代码中做所有事情。根据我的(可能很少)理解,后面代码中唯一需要的代码是引用xaml设置的实例,即:
_Persons = (Persons)this.Resources["_Persons"];
_persons = //some method to fill perons;
cvsPersons = (CollectionViewSource)this.Resources["cvsPersons"];
如果我没有_cvsPersons.Source = _Persons;然后我的数据网格不会被填充。我现在的xaml并没有完成这项工作。我想我的问题更多与概念有关..
答案 0 :(得分:4)
为避免代码隐藏,您应该使用MVVM模式MVVM Model View ViewModel 。一个可能的解决方案可能是&#34; Person&#34; (作为模型)像这样:
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
}
您可以使用ObservableCollection为人员实现一个ViewModel初始化属性。
public class ViewModel
{
public ObservableCollection<Person> Persons { get; set; }
public ViewModel()
{
Persons = new ObservableCollection<Person>();
}
}
你的MainWindow.cs现在必须初始化ViewModel:
public partial class MainWindow : Window
{
public ViewModel ViewModel;
public MainWindow()
{
ViewModel = new ViewModel();
ViewModel.Persons.Add(new Person
{
Age = 29,
Name = "Mustermann"
});
ViewModel.Persons.Add(new Person
{
Age = 35,
Name = "Meyer"
});
this.DataContext = ViewModel;
InitializeComponent();
}
将DataContext设置为ViewModel对象非常重要。我添加了一个Button来添加Person的方法。
private void AddPersonOnClick(object sender, RoutedEventArgs e)
{
ViewModel.Persons.Add(new Person
{
Age = 55,
Name = "Sand"
});
}
现在,您可以在XAML中实例化CollectionViewSource并将其绑定到ViewModel中的Persons ObservableCollection属性。
<Window x:Class="DataGridStackoverflow.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow" Height="350" Width="525">
<Window.Resources>
<CollectionViewSource x:Key="PersonsCollectionViewSource" Source="{Binding Persons}" />
</Window.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="*"/>
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<DataGrid Grid.Row="0" ItemsSource="{Binding Source={StaticResource PersonsCollectionViewSource}}" />
<Button x:Name="AddPerson" Grid.Row="1" Click="AddPersonOnClick" HorizontalAlignment="Left">Add Person</Button>
</Grid>
最后,您必须在将ItemsSource发布到CollectionViewSource时设置它,它就像魅力一样。
修改强>
我尝试了你的解决方案并且它也在工作。 MainWindow.xaml:
<Window.Resources>
<dataGridStackoverflow:Persons x:Key="Persons" />
<CollectionViewSource x:Key="PersonsCollectionViewSource" Source="{StaticResource Persons}" />
</Window.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="*"/>
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<DataGrid Grid.Row="0" ItemsSource="{Binding Source={StaticResource PersonsCollectionViewSource}}" />
<Button x:Name="AddPerson" Grid.Row="1" Click="AddPersonOnClick" HorizontalAlignment="Left">Add Person</Button>
</Grid>
在 InitializeComponent()之后初始化您的Persons Collection非常重要。 MainWindow.cs
InitializeComponent();
Persons persons = (Persons)this.FindResource("Persons");
persons.Add(new Person
{
Age = 23,
Name = "Dude"
});
此解决方案无需代码隐藏构造即可设置ItemsSource。