如何将命令绑定到 ListView WPF 中的复选框?
How to bind Command to Check Box in ListView WPF?
更新
我编辑了下面的代码以符合建议,现在它可以正常工作了。
我见过几个与此类似的堆栈溢出问题,但我无法将它们全部放在一起。我有以下 xaml 代码。
<UserControl x:Class="AuditEfficiencyMVVM.View.AuditTestsMain"
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"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:AuditEfficiencyMVVM.View"
xmlns:viewmodel="clr-namespace:AuditEfficiencyMVVM.ViewModel"
mc:Ignorable="d"
d:DesignHeight="500" d:DesignWidth="1000">
<UserControl.DataContext>
<viewmodel:AuditTests/>
</UserControl.DataContext>
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition/>
<ColumnDefinition/>
</Grid.ColumnDefinitions>
<Grid.RowDefinitions>
<RowDefinition Height="50"/>
<RowDefinition/>
<RowDefinition Height="50"/>
</Grid.RowDefinitions>
<ListView Grid.Row="1" Grid.Column="0" Margin="10" ItemsSource="{Binding Path=Tests}">
<ListView.View>
<GridView>
<GridViewColumn>
<GridViewColumn.CellTemplate>
<DataTemplate>
<CheckBox Name="TestSelected" IsChecked="{Binding Path=Selected, Mode=TwoWay}" Command="{Binding Path=TestSelected, RelativeSource={RelativeSource AncestorType=ListView}}"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn Header="Test Type" DisplayMemberBinding="{Binding Type, Mode=OneWay}"/>
<GridViewColumn Header="Progress">
<GridViewColumn.CellTemplate>
<DataTemplate>
<ProgressBar Name="TestProgress" Width="50" Height="20" Value="{Binding Progress, Mode=OneWay}"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn Header="Status" DisplayMemberBinding="{Binding Status, Mode=OneWay}"/>
</GridView>
</ListView.View>
<ListView.GroupStyle>
<GroupStyle>
<GroupStyle.ContainerStyle>
<Style TargetType="{x:Type GroupItem}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<Expander IsExpanded="True">
<Expander.Header>
<TextBlock FontWeight="Bold" FontSize="14" Text="{Binding Name}"/>
</Expander.Header>
<ItemsPresenter/>
</Expander>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</GroupStyle.ContainerStyle>
</GroupStyle>
</ListView.GroupStyle>
</ListView>
<ListView Grid.Row="1" Grid.Column="1" Margin="10" ItemsSource="{Binding Path=Files}">
<ListView.View>
<GridView>
<GridViewColumn Header="File Type" DisplayMemberBinding="{Binding Type, Mode=OneWay}"/>
<GridViewColumn Header="File Location" Width="250">
<GridViewColumn.CellTemplate>
<DataTemplate>
<TextBox Text="{Binding Path=Location, Mode=TwoWay}" Width="225"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn>
<GridViewColumn.CellTemplate>
<DataTemplate>
<Button Width="30" Height="20">...</Button>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
</GridView>
</ListView.View>
</ListView>
<Button Grid.Row="2" Grid.Column="1" HorizontalAlignment="Right" Margin="10" Width="50" Height="30">Run</Button>
</Grid>
</UserControl>
下面是我的代码
public class AuditTests : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
private RelayCommand _testSelected;
private void AddTest()
{
MessageBox.Show("Success");
}
public RelayCommand TestSelected
{
get
{
return _testSelected;
}
private set
{
if (_testSelected != value)
{
_testSelected = value;
RaisePropertyChanged("TestSelected");
}
}
}
private void RaisePropertyChanged(string property)
{
if (PropertyChanged != null)
{
PropertyChanged(this, new PropertyChangedEventArgs(property));
}
}
public AuditTests()
{
TestSelected = new RelayCommand(AddTest);
}
public ObservableCollection<Model.File> Files
{
get;
set;
}
public ObservableCollection<Model.Test> Tests
{
get;
set;
}
public void LoadFiles()
{
ObservableCollection<Model.File> files = new ObservableCollection<Model.File>();
foreach (Model.Test test in Tests)
{
foreach (Enums.FileType type in test.ExpectedSources)
{
Boolean containsType = false;
foreach (Model.File file in files)
{
if (file.Type == type)
{
containsType = true;
break;
}
}
if (!containsType)
{
files.Add(new Model.File { Type = type, Location = "", Tests = new List<Enums.TestType> { test.Type } });
}
else
{
files.Where(t => t.Type == type).First().Tests.Add(test.Type);
}
}
}
Files = files;
}
public void LoadTests()
{
ObservableCollection<Model.Test> tests = new ObservableCollection<Model.Test>();
foreach (var prop in Enum.GetValues(typeof(Enums.TestType)).Cast<Enums.TestType>().ToList())
{
tests.Add(new Model.Test { Category = prop.GetCategory(), Type = prop, Progress = 0, Selected = true, Status = Enums.TestStatus.NotStarted, ExpectedSources = prop.GetExpectedFiles() });
}
Tests = tests;
}
}
}
根据我的阅读,这似乎应该有效,但是当我 check/uncheck 复选框时,消息框未激活。为了使 check/uncheck 命令起作用,我在这里缺少什么?
TestSelected
命令是一个 属性 AuditTests
对象。 CheckBox 的 DataContext 是 Model.Test 对象。他们在不同的层次上。您可以使用 RelativeSource
参数将命令绑定到 ListView DataContext 中的 属性:
Command="{Binding Path=DataContext.TestSelected,
RelativeSource={RelativeSource AncestorType=ListView}}"
首先,我会让 AuditTests
实现 INotifyPropertyChanged
,并在其所有 属性 setter 的值发生变化时引发 PropertyChanged
。那里有很多关于这样做的文档。
其次,您必须为用户提供它的一个副本。我不知道您是否为 UserControl 提供了 MainWindow 可能想要绑定的任何属性;如果是这样的话,这是一个更复杂的问题。但最简单的是:
public AuditTestsMain()
{
InitializeComponent();
// Now its parent view can't bind this guy's properties to properties of the
// parent's view's viewmodel, because we've broken DataContext inheritance.
DataContext = new ViewModels.AuditTests();
}
然后看ASh的回答如何绑定命令。那是 DataContext 的另一件事:列表视图项是 Model.Test
个实例。因此在项目的 DataTemplate
中,绑定默认绑定到 Model.Test
的属性,因为那是 DataContext
。但是命令不是 Model.Test
的 属性;这是 AuditTests
中的 属性。 AuditTests
是 UserControl 的 DataContext,因此它也是 ListView 的 DataContext。控件继承父级的 DataContext 除非有什么干扰——比如我添加到你的 AuditTestsMain
构造函数的行,或者像 ListView 创建子项的方式,这些子项将 listview 的项作为它们的 DataContext。
更新
我编辑了下面的代码以符合建议,现在它可以正常工作了。
我见过几个与此类似的堆栈溢出问题,但我无法将它们全部放在一起。我有以下 xaml 代码。
<UserControl x:Class="AuditEfficiencyMVVM.View.AuditTestsMain"
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"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:AuditEfficiencyMVVM.View"
xmlns:viewmodel="clr-namespace:AuditEfficiencyMVVM.ViewModel"
mc:Ignorable="d"
d:DesignHeight="500" d:DesignWidth="1000">
<UserControl.DataContext>
<viewmodel:AuditTests/>
</UserControl.DataContext>
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition/>
<ColumnDefinition/>
</Grid.ColumnDefinitions>
<Grid.RowDefinitions>
<RowDefinition Height="50"/>
<RowDefinition/>
<RowDefinition Height="50"/>
</Grid.RowDefinitions>
<ListView Grid.Row="1" Grid.Column="0" Margin="10" ItemsSource="{Binding Path=Tests}">
<ListView.View>
<GridView>
<GridViewColumn>
<GridViewColumn.CellTemplate>
<DataTemplate>
<CheckBox Name="TestSelected" IsChecked="{Binding Path=Selected, Mode=TwoWay}" Command="{Binding Path=TestSelected, RelativeSource={RelativeSource AncestorType=ListView}}"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn Header="Test Type" DisplayMemberBinding="{Binding Type, Mode=OneWay}"/>
<GridViewColumn Header="Progress">
<GridViewColumn.CellTemplate>
<DataTemplate>
<ProgressBar Name="TestProgress" Width="50" Height="20" Value="{Binding Progress, Mode=OneWay}"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn Header="Status" DisplayMemberBinding="{Binding Status, Mode=OneWay}"/>
</GridView>
</ListView.View>
<ListView.GroupStyle>
<GroupStyle>
<GroupStyle.ContainerStyle>
<Style TargetType="{x:Type GroupItem}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<Expander IsExpanded="True">
<Expander.Header>
<TextBlock FontWeight="Bold" FontSize="14" Text="{Binding Name}"/>
</Expander.Header>
<ItemsPresenter/>
</Expander>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</GroupStyle.ContainerStyle>
</GroupStyle>
</ListView.GroupStyle>
</ListView>
<ListView Grid.Row="1" Grid.Column="1" Margin="10" ItemsSource="{Binding Path=Files}">
<ListView.View>
<GridView>
<GridViewColumn Header="File Type" DisplayMemberBinding="{Binding Type, Mode=OneWay}"/>
<GridViewColumn Header="File Location" Width="250">
<GridViewColumn.CellTemplate>
<DataTemplate>
<TextBox Text="{Binding Path=Location, Mode=TwoWay}" Width="225"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn>
<GridViewColumn.CellTemplate>
<DataTemplate>
<Button Width="30" Height="20">...</Button>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
</GridView>
</ListView.View>
</ListView>
<Button Grid.Row="2" Grid.Column="1" HorizontalAlignment="Right" Margin="10" Width="50" Height="30">Run</Button>
</Grid>
</UserControl>
下面是我的代码
public class AuditTests : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
private RelayCommand _testSelected;
private void AddTest()
{
MessageBox.Show("Success");
}
public RelayCommand TestSelected
{
get
{
return _testSelected;
}
private set
{
if (_testSelected != value)
{
_testSelected = value;
RaisePropertyChanged("TestSelected");
}
}
}
private void RaisePropertyChanged(string property)
{
if (PropertyChanged != null)
{
PropertyChanged(this, new PropertyChangedEventArgs(property));
}
}
public AuditTests()
{
TestSelected = new RelayCommand(AddTest);
}
public ObservableCollection<Model.File> Files
{
get;
set;
}
public ObservableCollection<Model.Test> Tests
{
get;
set;
}
public void LoadFiles()
{
ObservableCollection<Model.File> files = new ObservableCollection<Model.File>();
foreach (Model.Test test in Tests)
{
foreach (Enums.FileType type in test.ExpectedSources)
{
Boolean containsType = false;
foreach (Model.File file in files)
{
if (file.Type == type)
{
containsType = true;
break;
}
}
if (!containsType)
{
files.Add(new Model.File { Type = type, Location = "", Tests = new List<Enums.TestType> { test.Type } });
}
else
{
files.Where(t => t.Type == type).First().Tests.Add(test.Type);
}
}
}
Files = files;
}
public void LoadTests()
{
ObservableCollection<Model.Test> tests = new ObservableCollection<Model.Test>();
foreach (var prop in Enum.GetValues(typeof(Enums.TestType)).Cast<Enums.TestType>().ToList())
{
tests.Add(new Model.Test { Category = prop.GetCategory(), Type = prop, Progress = 0, Selected = true, Status = Enums.TestStatus.NotStarted, ExpectedSources = prop.GetExpectedFiles() });
}
Tests = tests;
}
}
}
根据我的阅读,这似乎应该有效,但是当我 check/uncheck 复选框时,消息框未激活。为了使 check/uncheck 命令起作用,我在这里缺少什么?
TestSelected
命令是一个 属性 AuditTests
对象。 CheckBox 的 DataContext 是 Model.Test 对象。他们在不同的层次上。您可以使用 RelativeSource
参数将命令绑定到 ListView DataContext 中的 属性:
Command="{Binding Path=DataContext.TestSelected,
RelativeSource={RelativeSource AncestorType=ListView}}"
首先,我会让 AuditTests
实现 INotifyPropertyChanged
,并在其所有 属性 setter 的值发生变化时引发 PropertyChanged
。那里有很多关于这样做的文档。
其次,您必须为用户提供它的一个副本。我不知道您是否为 UserControl 提供了 MainWindow 可能想要绑定的任何属性;如果是这样的话,这是一个更复杂的问题。但最简单的是:
public AuditTestsMain()
{
InitializeComponent();
// Now its parent view can't bind this guy's properties to properties of the
// parent's view's viewmodel, because we've broken DataContext inheritance.
DataContext = new ViewModels.AuditTests();
}
然后看ASh的回答如何绑定命令。那是 DataContext 的另一件事:列表视图项是 Model.Test
个实例。因此在项目的 DataTemplate
中,绑定默认绑定到 Model.Test
的属性,因为那是 DataContext
。但是命令不是 Model.Test
的 属性;这是 AuditTests
中的 属性。 AuditTests
是 UserControl 的 DataContext,因此它也是 ListView 的 DataContext。控件继承父级的 DataContext 除非有什么干扰——比如我添加到你的 AuditTestsMain
构造函数的行,或者像 ListView 创建子项的方式,这些子项将 listview 的项作为它们的 DataContext。