当前位置:  开发笔记 > 编程语言 > 正文

带有绑定数据的空Datagridview单元格

如何解决《带有绑定数据的空Datagridview单元格》经验,为你挑选了0个好方法。

我有一个带有"类型"和"值"列的DataGridView.用户选择数据类型,然后输入与该类型兼容的值,某些类型(例如"Text")接受任何字符串.其他类型(例如"是/否")仅限于可能值列表.对于每一行,我将"值"列中的单元格设置为自由形式类型的文本框或列表类型的组合框.DataGridView绑定到DataTable.

如果用户输入一种类型的值,但随后将该行切换为不允许当前值的其他类型,则会出现此问题.无论我尝试什么,我都无法清除Value cell.然后当我给单元格分配一个组合框时,我得到一个DataError,因为单元格的当前值是不兼容的.如何在将单元格从文本框更改为组合框之前清除该值?

public enum DataType 
{
    Text,
    YesNo
}

public class IndexedItem 
{
    public string Name { get; set; }
    public int ID {get; set; }

    public string strID 
    {
        get { return ID.ToString(); }
    }       

    //constructors & other methods;
}

public static DataTable ParameterTable;
public List YesNoList;

在表单构造函数中(dgvInputs是DataGridView):

ParameterTable = new DataTable("ParameterTable");
ParameterTable.Columns.Add("Type", typeof(DataType));
ParameterTable.Columns.Add("Value", typeof(string));

YesNoList = new List();
YesNoList.Add(new IndexedItem("Yes", 1));
YesNoList.Add(new IndexedItem("No", 0));

var D = (DataGridViewComboBoxColumn)dgvInputs.Columns[0];
D.ValueMember = "Value";
D.DisplayMember = "Display";
D.DataSource = new DataType[] { 
        DataType.Text,
        DataType.YesNo
}.Select(x => new { Display = x.ToString(), Value = (int)x }).ToList();

BindingSource ParamSource = new BindingSource();
ParamSource.DataSource = ParameterTable;
dgvInputs.AutoGenerateColumns = false;
dgvInputs.DataSource = ParamSource;
dgvInputs.Columns[0].DataPropertyName = "Type";
dgvInputs.Columns[1].DataPropertyName = "Value";

和事件:

private void dgvInputs_CurrentCellDirtyStateChanged(object sender, EventArgs e) {
    if (dgvInputs.IsCurrentCellDirty) {
        dgvInputs.CommitEdit(DataGridViewDataErrorContexts.Commit);
    }
}

private void dgvInputs_CellValueChanged(object sender, DataGridViewCellEventArgs e) {
   if (e.RowIndex >= 0 && e.ColumnIndex == 0) {
      var cb = (DataGridViewComboBoxCell)dgvInputs[0, e.RowIndex];
      if (cb.Value != null && cb.Value != DBNull.Value) {
         DataType Type = (DataType)cb.Value;
         dgvInputs[1, e.RowIndex].Value = string.Empty;
         dgvInputs.CommitEdit(DataGridViewDataErrorContexts.Commit);
         switch (Type) {
            case DataType.YesNo:
               dgvInputs[1, e.RowIndex].Dispose();
               var newBox = new DataGridViewComboBoxCell();
               newBox.DisplayMember = "Name";
               newBox.ValueMember = "strID";
               newBox.DataSource = YesNoList;
               dgvInputs[1, e.RowIndex] = newBox;
               break;
            default:
               dgvInputs[1, e.RowIndex] = new DataGridViewTextBoxCell();
               break;
         }
      }
   }
}

如果你将它设置为"text"并输入任意内容,然后切换到"YesNo",它会给出一个错误"System.ArgumentException:DataGridViewComboBoxCell值无效.",它会在光标在单元格上方时重新出现.将其更改回文本行会导致原始值重新出现.

我假设问题是该值保存在ParameterTable中,但我不能让它将原始值的清除传播到ParameterTable.我试过nullDBNull.Value不是string.Empty,但没有人有任何区别.我添加了"CommitEdit"行,希望能让它做出改变,但这也没有任何区别.

编辑: 事实证明,问题是我在单元格更改事件中的代码:

string Default = dgvInputs[4, e.RowIndex].Value as string;
// code switching out text box and combo box above
try 
{
    dgvInputs[4, e.RowIndex].Value = Default;
} catch (Exception e2) {
    MessageBox.Show(e2.GetType().ToString());
}

如果可能的话,我的想法是保留价值,我有消息框向我展示我需要抓住的具体例外,因为我不确定.但显然这项任务不会立即引发异常.这只发生在以后,显然在其他事件中我没有处理.

事后证明,我应该在示例中包含此代码.我现在不知道我是如何忽视它的.通过遗漏关键信息,我向所有通过疯狂追逐的人道歉.我非常感谢您提供的所有帮助.

推荐阅读
coco2冰冰
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有