c# - DataGridView 列不是从自定义数据源生成的

标签 c# winforms data-binding datagridview

我写了一个实现了 IBindingListView 的类界面,BindableList<T> : IBindingListView .

_abilityList = new BindableList<Ability>();
dataGridViewAbility.AutoGenerateColumns = true;
dataGridViewAbility.DataSource = _abilityList;

出于某种原因 DataGridView不生成列,所以当我将数据添加到我的 BindableList<T>DataGridView抛出一个错误,指出我需要先添加列。

奇怪的是,如果我将一个对象添加到我的 BindableList<T>然后生成的列只是找到。我还为 BindingList<T> 关闭了我的类(class),不向其中添加任何内容,列生成得很好。

我不知道为什么 DataGridView当它为空时,不会从我的通用绑定(bind)列表生成列。框架在程序构建时知道泛型的类型。

更新:

能力等级

class Ability
{
    public Int32 ID { get; set; }
    public Int32 Generation { get; set; }
    public String Name { get; set; }
    public String Effect { get; set; }
}

我不明白为什么人们难以理解这一点,但我制作了一个新的 Winform 应用程序,但仍然存在同样的问题。

private BindableList<Ability> _abilityList;

    public Form1()
    {
        InitializeComponent();

        _abilityList = new BindableList<Ability>();

        Button b = new Button();
        b.Location = new Point(0, 0);
        b.Click += new EventHandler(buttonGetData_Click);
        this.Controls.Add(b);

        DataGridView _grid = new DataGridView();
        _grid.Location = new Point(100, 0);
        _grid.AutoGenerateColumns = true;
        _grid.DataSource = _abilityList;
        this.Controls.Add(_grid);
    }

    private void buttonGetData_Click(object sender, EventArgs e)
    {
        _abilityList.Add(new Ability());
    }

OnListChanged 事件

protected virtual void OnListChanged(ListChangedEventArgs e)
{
    if (ListChanged != null)
    {
        ListChanged(this, e);
    }
}

当行_abilityList.Add运行,这会触发 ListChanged(this, e); .这运行并且 DataGridView 抛出错误,没有列添加到 View 。如果我将对象添加到 BindableList<T>首先在将其分配为 DataSource 之前列将生成。

最佳答案

简单地分配 DataGridView.DataSource 属性不会立即生成列。根据我的经验,如果您立即需要 DataGridViewColumns,则有 4 行关键代码强制创建列:

        DataGridView dgv = new DataGridView() { Dock = DockStyle.Fill };
        BindingList<Ability> list = new BindingList<Ability>();
        dgv.AutoGenerateColumns = true;
        dgv.DataSource = list;

        int colCount = dgv.Columns.Count; // 0
        //--------
        // force columns to be created
        using (var g = dgv.CreateGraphics()) {}
        using (var f = new Form()) {
            f.Controls.Add(dgv);
            f.Controls.Remove(dgv);
        }
        //--------
        int colCount2 = dgv.Columns.Count; // 4

编辑:替代方法要快得多,尤其是当 DataGridViewColumns 上的属性(例如 AutoSizeModeSortMode) 需要设置。

public static void Refresh(DataGridView view, DataTable table) {
    bool origAuto = view.AutoGenerateColumns;
    view.AutoGenerateColumns = true;
    view.DataSource = table;
    view.BindingContext = new BindingContext();
    view.AutoGenerateColumns = origAuto;
}

关于c# - DataGridView 列不是从自定义数据源生成的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32528465/

相关文章:

c# - ASP.NET CORE Web 应用程序抛出 SQLiteException : unable to open database file

c# - 将 B2B 外部用户添加到 Azure AD,而不发送邀请电子邮件 C#

.net - 从鼠标位置获取 DataGrid 单元格

c# - 每次都会出现权限屏幕

c# - 将可为 null 的 DateTime 绑定(bind)到 MaskedTextBox

c# - "CS8700: Multiple analyzer config files cannot be in the same directory"但只有一个 StyleCop 文件

c# - 使用 Winform 更新 WPF 文本框用户控件中的文本

.net - 我应该将什么类型的对象绑定(bind)到 n 层应用程序中的 WPF 表单?

c# - 如何将 DataBinder.Eval 应用于超链接的 Text 属性

c# - 将 ByteArray 转换为 String 以在 TextBox c# 中使用