Pau*_*air 8 c# datagridview winforms
我有一个带有"类型"和"值"列的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<IndexedItem> YesNoList;
Run Code Online (Sandbox Code Playgroud)
在表单构造函数中(dgvInputs是DataGridView):
ParameterTable = new DataTable("ParameterTable");
ParameterTable.Columns.Add("Type", typeof(DataType));
ParameterTable.Columns.Add("Value", typeof(string));
YesNoList = new List<IndexedItem>();
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";
Run Code Online (Sandbox Code Playgroud)
和事件:
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;
}
}
}
}
Run Code Online (Sandbox Code Playgroud)
如果你将它设置为"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());
}
Run Code Online (Sandbox Code Playgroud)
如果可能的话,我的想法是保留价值,我有消息框向我展示我需要抓住的具体例外,因为我不确定.但显然这项任务不会立即引发异常.这只发生在以后,显然在其他事件中我没有处理.
事后证明,我应该在示例中包含此代码.我现在不知道我是如何忽视它的.通过遗漏关键信息,我向所有通过疯狂追逐的人道歉.我非常感谢您提供的所有帮助.
小智 1
您的问题不在于清除该值,而在于 YesNoList。
网格组合框尝试查找当前记录的值,并且 YesNoList 中既没有空值也没有空值。
如果您尝试添加新记录并首先设置数据类型而不设置值,您甚至会收到错误。
您可以通过向 YesNoList 添加一个空项目或在切换数据类型时为现有记录设置默认值来解决此问题。