自定义DataGridViewCell不会触发DataSource更改事件

时间:2014-02-20 14:41:28

标签: c# .net data-binding datagridview datatable

我正在与WinForms DataGridView结合使用数据绑定。我将一个包含DataSet的DataView分配给DataGridView.DataSource,它到目前为止运行良好。实现自定义DataGridViewCell 时会出现此问题。我的目标是提供一个ComboBoxCell,用于选择始终完全交互的Enum值,并且不需要用户明确地进入编辑模式。

How it looks like in action

这是绑定设置:

  • DataSet S只包含一个DataTable,T
  • DataView V包装表
  • DataGridView.DataSource设置为V
  • 应用程序的某些部分订阅了T.RowChanged事件。这是关键部分。

就功能而言,我的自定义单元格的行为完全符合预期。但是,它不会导致触发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 正在正确触发事件,但在编辑自定义单元格后,我无法以某种方式更新它。

我怎样才能做到这一点?

1 个答案:

答案 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开发人员的批准都将受到高度赞赏,所以...随时发表评论!