我有一个带有"类型"和"值"列的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 ListYesNoList;
在表单构造函数中(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.我试过null
而DBNull.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()); }
如果可能的话,我的想法是保留价值,我有消息框向我展示我需要抓住的具体例外,因为我不确定.但显然这项任务不会立即引发异常.这只发生在以后,显然在其他事件中我没有处理.
事后证明,我应该在示例中包含此代码.我现在不知道我是如何忽视它的.通过遗漏关键信息,我向所有通过疯狂追逐的人道歉.我非常感谢您提供的所有帮助.