如何将字符串数组自动绑定到 WPF 数据网格

本文关键字:WPF 数据 数据网 网格 绑定 字符串 数组 | 更新日期: 2023-09-27 17:57:17

我在UserControl里面有一个DataGrid。它看起来像这样:

<UserControl x:Class="ExternalDataSourceComparison.ImportedInfoGrid"
         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" 
         mc:Ignorable="d" 
         d:DesignHeight="300" d:DesignWidth="300"
         Height="Auto" Width="Auto">
<Grid>
    <Grid.RowDefinitions>
        <RowDefinition/>
    </Grid.RowDefinitions>
    <DataGrid Name="_dataGrid" Grid.Row="0">
        <DataGrid.Columns>
            <DataGridTextColumn Binding="{Binding Path[0].Length}"/>
        </DataGrid.Columns>
    </DataGrid>
</Grid>

我的MainWindow中有如下所示的UserControl

<Window x:Class="ExternalDataSourceComparison.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:local="clr-namespace:ExternalDataSourceComparison"
    Title="MainWindow" Height="350" Width="525" WindowState="Maximized">
<Grid>
    <Grid.RowDefinitions>
        <RowDefinition/>
        <RowDefinition Height="Auto"/>
    </Grid.RowDefinitions>
    <local:ImportedInfoGrid x:Name="ExternalData" Grid.Row="0"/>
    <StackPanel Height="36px" Orientation="Horizontal" Grid.Row="1">
        <Button Name="_import" Content="Import Data" Margin="5" Padding="5, 2" Click="Import_Click"/>
        <Button Name="_compare" Content="Compare" Margin="5" Padding="5, 2" Click="Compare_Click"/>
        <Button Name="_cancel" Content="Cancel" Margin="5" Padding="5, 2" Click="Cancel_Click"/>            
    </StackPanel>
</Grid>

在我使用方法fs.CSVToStringArray的窗口代码中,我打开一个 CSV 文件并将内容解析为一个string[][]外部数组表示行,内部数组是所有列,因此 string[0][3] 将是第 1 行第 4 列。

在我的代码中,我只是将ItemsSource设置为等于数组数组,如下所示:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Data;
namespace ExternalDataSourceComparison
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        FileStuff fs = new FileStuff();
        public MainWindow()
        {
            InitializeComponent();
        }
        private void Import_Click(object sender, RoutedEventArgs e)
        {
            string[][] array = fs.CSVToStringArray();
            this.ExternalData._dataGrid.ItemsSource = array;
        }
        private void Cancel_Click(object sender, RoutedEventArgs e)
        {
        }
        private void Compare_Click(object sender, RoutedEventArgs e)
        {
        }
    }
}

不过,输出根本不是我所期望的。我认为它显示了对象的属性而不是实际内容。有没有办法自动生成行和列,或者我必须编写代码来创建列并构建行?我认为使用DataGrid您可以将诸如string[][]List<string[]>之类的内容绑定到ItemsSource,它只会生成列。它目前正在自动生成列,只是没有用string[][]的内容填充它们。

如何将字符串数组自动绑定到 WPF 数据网格

DataGrid的自动生成列功能不太适合显示交错数组数据。它更适合于单维集合,其中DataGrid中的每一行表示集合中的单个项目,每列表示相应项目的不同属性。

除非ItemsSource中的项目是可以显示的模型,否则不应依赖自动生成列功能(它将为所有属性显示列,包括那些不打算显示的属性)。尝试自动DataGrid绑定到数组时的另一个问题是,DataGrid没有关于它应该如何命名列的信息。

最后,我认为您最好以某种方式定义DataGrid列,而不是让它自己生成列DataGrid。例如,从 XAML 定义它,假设数组中始终有两个"列":

<DataGrid Name="_dataGrid" Grid.Row="0" AutoGenerateColumns="False">
    <DataGrid.Columns>
        <DataGridTextColumn Header="Column 1" Binding="{Binding [0]}"/>
        <DataGridTextColumn Header="Column 2" Binding="{Binding [1]}"/>
    </DataGrid.Columns>
</DataGrid>

或者从代码中定义列,如果您有动态的列数,例如:

string[][] array = fs.CSVToStringArray();
for (int i = 0; i < array[0].Length; i++)
{
    var col = new DataGridTextColumn();
    col.Header = "Column " + i;
    col.Binding = new Binding(string.Format("[{0}]", i));
    _dataGrid.Columns.Add(col);
}
this.ExternalData._dataGrid.ItemsSource = array;