Wpf DataGrid数据绑定与任意列和行



对于我的一个项目,我试图从文件(例如*.csv)加载数据并在DataGrid中显示它。文件的格式没有给出。例如,列数和行数因文件而异。因此,任务是将一个通用数据表加载到程序中,并在DataGrid中查看它。

我的第一次尝试是使用DataTable并以某种方式将其绑定到DataGrid。然而,经过一段时间的搜索,我读到使用DataTable不是最好的解决方案,应该使用数据模型。不幸的是,web中的大多数示例使用具有固定数据列的数据。此外,我无法使DataGridDataTable之间的绑定正常工作。如果能够对两端的数据进行更改:DataGrid(如用户所愿)和底层数据结构(如代码所愿),那将是非常好的。

请问,谁能解释一下,为什么不能使用DataTable ?应该用什么来实现上述行为呢?

数据加载本身应该不是问题,但是如何以灵活的方式存储、处理和绑定数据表到DataGrid是一个问题。

下面的代码可以工作。然而,它似乎不响应任何更改的数据,如添加行或列,但DataTable包含正确的数据。

<DataGrid x:Name="myDataGrid" AutoGenerateColumns="True"
 x:Name="myDataGrid1" AutoGenerateColumns="True" ItemsSource="{Binding myDataTable}"  />
DataTable myDataTable  { get; private set; }
...
myDataGrid.DataContext = this;
LoadData();
myDataGrid.ItemsSource = myDataTable.DefaultView;
我非常感谢任何帮助!提前感谢!

尝试将ItemsSource设置为BindingListCollectionView。到目前为止,我的运气还不错。我使用这些扩展方法:

  public static BindingListCollectionView GetCollectionView(this DataTable table)
        {
            return new BindingListCollectionView(table.DefaultView);
        }

 public static DataTable GetDataTable(this BindingListCollectionView view)
    {
        var dataView = view.SourceCollection as DataView;
        if (dataView != null)
        {
            return dataView.Table;
        }
        return null;
    }

在您的代码中使用它,如:

myDataGrid.ItemsSource = myDataTable.GetCollectionView();

var view = myDataGrid.ItemsSource as BindingListCollectionView;
var table = view.GetDataTable();

您应该能够像往常一样使用DataTable对象,并保留网格所做的所有更改。

经过进一步的调查,我找到了一个适合我的解决方案。然而,这感觉有点奇怪,不是最好的解决方案,但我可以忍受。再次感谢您的努力,这对我帮助很大。

这个想法是刷新GridView每当一个列添加/删除通过unset/set GridViewDataContext。我使用DataTable.Columns.CollectionChanged事件来实现此行为。

现在,几乎一切都很好。剩下的唯一问题是当用户通过GridView添加新行时(单击最后一个空行并输入数据时)。然而,这只是一个小问题,因为绑定的DataTable是正确更新的,只有第二个GridView(在下面的例子中)没有正确显示。

问题仍然存在,使用DataTable是否是最佳解决方案。如果重置DataContext对大数据集造成麻烦,需要进行测试…

下面是一个关于这个问题和解决方案的完整示例:

XAML:

<Window xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         x:Class="WpfApplicationTest.MainWindow"
         Title="DataGrid and DataTable Test" Height="500" Width="800">
    <DockPanel>
        <StackPanel Orientation="Horizontal" DockPanel.Dock="Top">
            <Button Click="ButtonRefresh_Click">Refresh Grids</Button>
            <Button Click="ButtonAddRow_Click">Add Row</Button>
            <Button Click="ButtonAddCol_Click">Add Col</Button>
            <Button Click="ButtonEdit_Click">Edit Randomly</Button>
        </StackPanel>
        <Grid>
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="*"/>
                <ColumnDefinition Width="*"/>
            </Grid.ColumnDefinitions>
            <DataGrid x:Name="myDataGrid0" Grid.Column="0" AutoGenerateColumns="True" ItemsSource="{Binding}" />
            <DataGrid x:Name="myDataGrid1" Grid.Column="1" AutoGenerateColumns="True" ItemsSource="{Binding}" />
        </Grid>
    </DockPanel>
</Window>

背后的代码:

using System;
using System.Data;
using System.Windows;
namespace WpfApplicationTest
{
    public partial class MainWindow : Window
    {
        private DataTable myDataTable;
        private Random myRandom = new Random();
        public MainWindow()
        {
            InitializeComponent();
            myDataTable = new DataTable();
            // create some datatable
            myDataTable.Columns.Add("First");
            myDataTable.Columns.Add("Second");
            myDataTable.Rows.Add(myRandom.Next(100).ToString(), myRandom.Next(100).ToString());
            myDataTable.Rows.Add(myRandom.Next(100).ToString(), myRandom.Next(100).ToString());
            myDataGrid0.DataContext = myDataTable;
            myDataGrid1.DataContext = myDataTable;
            myDataTable.Columns.CollectionChanged += Columns_CollectionChanged;
        }
        private void Columns_CollectionChanged(object sender, System.ComponentModel.CollectionChangeEventArgs e)
        {
            // refresh the grid views by reseting the data context after the columns have changed
            myDataGrid0.DataContext = null;
            myDataGrid0.DataContext = myDataTable;
            myDataGrid1.DataContext = null;
            myDataGrid1.DataContext = myDataTable;
        }
        private void ButtonRefresh_Click(object sender, RoutedEventArgs e)
        {
            // refresh the grid views by reseting the data context
            myDataGrid0.DataContext = null;
            myDataGrid0.DataContext = myDataTable;
            myDataGrid1.DataContext = null;
            myDataGrid1.DataContext = myDataTable;
        }
        private void ButtonAddRow_Click(object sender, RoutedEventArgs e)
        {
            // add rows with random content
            DataRow row = myDataTable.NewRow();
            for (int i = 0; i < row.ItemArray.Length; ++i)
                row[i] = myRandom.Next(100);
            myDataTable.Rows.Add(row);
        }
        int rowcount = 0;
        private void ButtonAddCol_Click(object sender, RoutedEventArgs e)
        {
            // add columns with random content
            myDataTable.Columns.Add("New" + rowcount.ToString());
            foreach (DataRow row in myDataTable.Rows)
                row["New" + rowcount.ToString()] = myRandom.Next(100);
            ++rowcount;
        }
        private void ButtonEdit_Click(object sender, RoutedEventArgs e)
        {
            // randomly change some data table values
            myDataTable.Rows[myRandom.Next(myDataTable.Rows.Count)][myRandom.Next(myDataTable.Columns.Count)] = myRandom.Next(100);
        }
    }
}

相关内容

  • 没有找到相关文章

最新更新