具有可变数量的按钮/形状的 WPF DataGrid 列

标签 wpf data-binding mvvm datagrid

我有一个要求,在一个 DataGrid 列上,我需要并排显示 x 个可点击按钮(水平堆叠)。要显示的实际数字或按钮取决于该列中的绑定(bind)值。

下图显示了这一点。左边的网格是我现在的网格,右边的网格是我正在寻找的。

Current and desired DataGrid

Grid 绑定(bind)到 ViewModel,如:

<DataGrid ItemsSource="{Binding employees}" AutoGenerateColumns="False"  >
        <DataGrid.Columns>
            <DataGridTextColumn Binding="{Binding Path=id}" Header="Id" />
            <DataGridTextColumn Binding="{Binding Path=numberofplatforms}" Header="No" Width="50" IsReadOnly="True" />
        </DataGrid.Columns>
    </DataGrid>

而 ViewModel 是:
Public Class ViewModel

    Public Property employees As ObservableCollection(Of employee)
        Get
            Return _employees
        End Get
        Set(ByVal value As ObservableCollection(Of employee))
            _employees = value
        End Set
    End Property
    Private _employees As New ObservableCollection(Of employee)

End Class

员工是:
Public Class employee
    Public Property id As String
    Public Property numberofplatforms As Integer
    Public Property selectedplatform As Integer
End Class

除了显示按钮之外,按钮本身必须像单选按钮一样,即在任何 DataGrid 行上,只有一个按钮被“按下”(图像中的蓝色背景按钮),其他按钮未被按下(灰色背景)。按钮(或形状)必须是可点击的,以便可以更改选择。

根据 selectedplatform 属性,哪个按钮被“按下”由 ViewModel 确定。此示例中的该属性对于 ABC 是 1,对于 DEF 是 1,对于 GHI 是 2。如果 numberofplatforms 属性为零,则不显示任何按钮 (JKL)。

我该如何设置这个机制?

最佳答案

部分代码是用 C# 编写的,我希望这不是问题。 wpf代码:

<Window.Resources>
    <ResourceDictionary>
        <local:PositiveIntegersConverter x:Key="PositiveIntegersConverter" />
        <local:EmployeePlatformOptionConverter x:Key="EmployeePlatformOptionConverter"/>
    </ResourceDictionary>
</Window.Resources>

<DataGrid ItemsSource="{Binding employees}" AutoGenerateColumns="False" x:Name="DataGrid1">
    <DataGrid.Columns>
        <DataGridTextColumn Binding="{Binding Path=id}" Header="Id" />
        <DataGridTemplateColumn Header="No" IsReadOnly="True">
            <DataGridTemplateColumn.CellTemplate>
                <DataTemplate>
                        <ListBox ItemsSource="{Binding numberofplatforms, Converter={StaticResource PositiveIntegersConverter}}" SelectedItem="{Binding selectedplatform}"
                                    x:Name="ListBox1">
                            <ListBox.ItemTemplate>
                                <DataTemplate>
                                    <Button Content="{Binding}" Command="{Binding Source={x:Reference DataGrid1}, Path=DataContext.SelectOptionCommand}">
                                        <Button.CommandParameter>
                                            <MultiBinding Converter="{StaticResource EmployeePlatformOptionConverter}">
                                                <Binding ElementName="ListBox1" Path="DataContext"/>
                                                <Binding Path="."/>
                                            </MultiBinding>
                                        </Button.CommandParameter>
                                        <Button.Style>
                                            <Style TargetType="Button">
                                                <Setter Property="Background" Value="Gray" />
                                                <Setter Property="Foreground" Value="White" />
                                                <Style.Triggers>
                                                    <DataTrigger Binding="{Binding RelativeSource={RelativeSource FindAncestor, AncestorType=ListBoxItem}, Path=IsSelected}"
                                                                    Value="True">
                                                        <Setter Property="Background" Value="Blue" />
                                                    </DataTrigger>
                                                </Style.Triggers>
                                            </Style>
                                        </Button.Style>
                                    </Button>
                                </DataTemplate>
                            </ListBox.ItemTemplate>
                            <ListBox.ItemsPanel>
                                <ItemsPanelTemplate>
                                    <StackPanel Orientation="Horizontal" />
                                </ItemsPanelTemplate>
                            </ListBox.ItemsPanel>
                        </ListBox>
                </DataTemplate>
            </DataGridTemplateColumn.CellTemplate>
        </DataGridTemplateColumn>
    </DataGrid.Columns>
</DataGrid>

我在这里使用了两个转换器:

PositiveIntegersConverter,它为给定数量的平台返回正整数 - 这些是员工可用的平台选项

EmployeePlatformOptionConverter,它将我们要传递给 SelectOptionCommand 的两个参数:员工和选定的平台选项转换为一个 EmployeePlatformOption 类型的对象。
public class PositiveIntegersConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        var toInteger = System.Convert.ToInt32(value);
        return toInteger > 0 ? Enumerable.Range(1, toInteger) : null;
    }

    public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
    {
        throw new NotImplementedException();
    }
}

public class EmployeePlatformOptionConverter
    : IMultiValueConverter
{
    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        return new EmployeePlatformOption
            {
                Employee = values[0] as Employee,
                Platform = (int)values[1]
            };
    }

    public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
    {
        throw new NotImplementedException();
    }
}

使用封装对象:
public class EmployeePlatformOption
{
    public Employee Employee { get; set; }
    public int Platform { get; set; }
}

您将选择传递给 ViewModel 类中的 SelectOptionCommand:
public ICommand SelectOptionCommand { get; set; }
private void SelectOptionExecute(EmployeePlatformOption employeePlatformOption)
{
    if (employeePlatformOption != null && employeePlatformOption.Employee != null &&
        employeePlatformOption.Platform > 0)
    {
        employeePlatformOption.Employee.selectedplatform = employeePlatformOption.Platform;
    }
}

Employee 应该实现 INotifyPropertyChange 接口(interface),以便在屏幕上可以看到 selectedplatform 更新。

关于具有可变数量的按钮/形状的 WPF DataGrid 列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17913591/

相关文章:

c# - 取消选择 RowDetailsTemplate 后调整数据网格高度

c# - 将 TreeView 与 ObservableCollection 绑定(bind)

c# - 上下文菜单绑定(bind)永远不会被执行

silverlight - 使用 WCF RIA 和 MVVM 模式验证 UI 上的数据

c# - 如何从 C# 项目中调用另一个进程?

c# - 使用RenderTargetBitmap保存部分显示图像

c# - 在鼠标悬停时更改矩形背景

asp.net - DataGrid 列换行和宽度问题

c++ - C/C++ 将二进制数据映射到结构成员

c# - 为什么不调用 RelayCommand?