我正在与WinForms DataGridView结合使用数据绑定。我将一个包含DataSet的DataView分配给DataGridView.DataSource,它到目前为止运行良好。实现自定义DataGridViewCell 时会出现此问题。我的目标是提供一个ComboBoxCell,用于选择始终完全交互的Enum值,并且不需要用户明确地进入编辑模式。
这是绑定设置:
就功能而言,我的自定义单元格的行为完全符合预期。但是,它不会导致触发DataTable.RowChanged事件,除非整个DataGridView失去焦点......但所有其他非自定义单元格都会失去焦点。我仍然得到一个CellValueChanged事件,并且DataSet具有新值..但是既没有DataTable.RowChanged也没有DataGridView.DataBindingComplete,并且该行不会像通常那样自动失效。
我显然做错了什么。我可能错过了一个通知程序事件或实现了一些错误,但经过两天的搜索,踩踏和反汇编.Net代码我仍然完全卡住了。
以下是类定义中最重要的部分(不是完整的源代码):
public class DataGridViewEnumCell : DataGridViewCell, IDataGridViewEditingCell
{
private Type enumType = null;
private Enum enumValue = default(Enum);
private bool enumValueChanged = false;
public virtual object EditingCellFormattedValue
{
get { return this.GetEditingCellFormattedValue(DataGridViewDataErrorContexts.Formatting); }
set { this.enumValue = (Enum)Utility.SafeCast(value, this.enumType); }
}
public virtual bool EditingCellValueChanged
{
get { return this.enumValueChanged; }
set { this.enumValueChanged = value; }
}
public override Type EditType
{
get { return null; }
}
public override Type FormattedValueType
{
get { return this.enumType; }
}
public override Type ValueType
{
get
{
if (this.OwningColumn != null && this.OwningColumn.ValueType != null)
{
return this.OwningColumn.ValueType;
}
else
{
return this.enumType;
}
}
set
{
base.ValueType = value;
}
}
// The kind of Enum that is edited in this cell.
public Type EnumValueType
{
get { return this.enumType; }
set { this.enumType = value; }
}
public virtual object GetEditingCellFormattedValue(DataGridViewDataErrorContexts context)
{
if (context.HasFlag(DataGridViewDataErrorContexts.ClipboardContent))
{
return Convert.ToString(this.enumValue);
}
else
{
return this.enumValue ?? this.enumType.GetDefaultValue();
}
}
public override object ParseFormattedValue(object formattedValue, DataGridViewCellStyle cellStyle, TypeConverter formattedValueTypeConverter, TypeConverter valueTypeConverter)
{
// Cast the Enum value to the original cell value type
object cellVal;
Utility.SafeCast(formattedValue, this.ValueType, out cellVal);
return cellVal;
}
protected override object GetFormattedValue(object value, int rowIndex, ref DataGridViewCellStyle cellStyle, TypeConverter valueTypeConverter, TypeConverter formattedValueTypeConverter, DataGridViewDataErrorContexts context)
{
if (this.DataGridView == null || value == null)
{
return this.enumType.GetDefaultValue();
}
// Cast the cell value to the appropriate Enum value type
object enumVal;
Utility.SafeCast(value, this.enumType, out enumVal);
// Let the base implementation apply additional formatting
return base.GetFormattedValue(enumVal, rowIndex, ref cellStyle, valueTypeConverter, formattedValueTypeConverter, context);
}
private Enum GetCurrentValue()
{
object unknownVal = (this.enumValueChanged ? this.enumValue : this.Value);
object enumVal;
Utility.SafeCast(unknownVal, this.enumType, out enumVal);
return (Enum)enumVal;
}
public virtual void PrepareEditingCellForEdit(bool selectAll)
{
this.enumValue = this.GetCurrentValue();
}
protected override void OnClick(DataGridViewCellEventArgs e)
{
base.OnClick(e);
if (this.DataGridView.CurrentCell == this && (DateTime.Now - this.mouseClosed).TotalMilliseconds > 200)
{
// Due to some reason I don't understand sometimes EditMode is already active.
// Don't do it twice in these cases.
if (!this.IsInEditMode)
{
// Begin editing
this.DataGridView.BeginEdit(true);
}
this.ShowDropDown();
}
}
public void HideDropDown()
{
// ... snip ...
// Revert value to original state, if not accepted explicitly
// It will also run into this code after the new selection
// has been accepted (see below)
if (this.DataGridView != null)
{
this.enumValue = this.GetCurrentValue();
this.enumValueChanged = false;
this.DataGridView.EndEdit();
}
}
// Called when a value has been selected. All calue changes run through this method!
private void dropdown_AcceptSelection(object sender, EventArgs e)
{
Enum selectedEnum = (Enum)this.dropdown.SelectedItem;
if (!this.enumValue.Equals(selectedEnum))
{
this.enumValue = selectedEnum;
this.enumValueChanged = true;
this.DataGridView.NotifyCurrentCellDirty(true);
this.DataGridView.CommitEdit(DataGridViewDataErrorContexts.Commit);
}
}
}
同样,当DataGridView失去焦点或编辑DataGridView中的任何其他单元格时,DataSource 正在正确触发事件,但在编辑自定义单元格后,我无法以某种方式更新它。
我怎样才能做到这一点?
答案 0 :(得分:1)
我终于能够解决这个问题了。
事实证明,整体问题与我的自定义IDataGridViewEditingCell没有任何关系。没有收到RowChanged事件只是因为 DataGridView行未经验证,直到离开当前选定的行 - 我没有注意到,因为我的测试中只有一行,所以我不得不专注于不同的控制来实现这一目标。
在取消选择/散焦之前,不验证当前行似乎是预期的并且DataGridView中的正常行为。这不是我想要的,所以我派生了我自己的DataGridView,并做了以下事情:
protected override void OnCellEndEdit(DataGridViewCellEventArgs e)
{
base.OnCellEndEdit(e);
// Force validation after each cell edit, making sure that
// all row changes are validated in the DataSource immediately.
this.OnValidating(new System.ComponentModel.CancelEventArgs());
}
到目前为止,它似乎完美无缺,但我可能只是幸运。任何经验丰富的DataGridView开发人员的批准都将受到高度赞赏,所以...随时发表评论!