DataGrid:在单元格验证错误上,其他行单元格是不可编辑/只读

时间:2010-11-04 01:06:46

标签: wpf datagrid

在我的wpf datagrid中,我使用IDataErrorInfo实现了验证。当单元格中出现错误时,其他行中的单元格将变为ReadOnly。对我来说这是有道理的,但是企业希望能够在不修复错误的情况下更改其他行单元格,即在某些情况下让用户弄得一团糟,而且开发人员的生活很糟糕。

我尝试将HasCellValidationError重置为false,但它没有修复它。我非常感谢有关此问题的任何反馈/建议。

BindingFlags bf = BindingFlags.FlattenHierarchy | BindingFlags.NonPublic | BindingFlags.Instance;
PropertyInfo inf = myDataGrid.GetType().GetProperty("HasCellValidationError", bf);

if (inf != null)
{
    inf.SetValue(myDataGrid, false, null);
}

2 个答案:

答案 0 :(得分:13)

通过覆盖datagrid的OnCanExecuteBeginEdit方法找到解决方案。 请参阅下面的代码,目前测试人员没有投诉。

/// <summary>
/// This class overrides the OnCanExecuteBeginEdit method of the standard grid
/// </summary>
public partial class DataGrid : System.Windows.Controls.DataGrid
{

    /// <summary>
    /// This method overrides the 
    /// if (canExecute && HasRowValidationError) condition of the base method to allow
    /// ----entering edit mode when there is a pending validation error
    /// ---editing of other rows
    /// </summary>
    /// <param name="e"></param>
    protected override void OnCanExecuteBeginEdit(System.Windows.Input.CanExecuteRoutedEventArgs e)
    {

        bool hasCellValidationError = false;
        bool hasRowValidationError = false;
        BindingFlags bindingFlags = BindingFlags.FlattenHierarchy | BindingFlags.NonPublic | BindingFlags.Instance;
        //Current cell
        PropertyInfo cellErrorInfo = this.GetType().BaseType.GetProperty("HasCellValidationError", bindingFlags);
        //Grid level
        PropertyInfo rowErrorInfo = this.GetType().BaseType.GetProperty("HasRowValidationError", bindingFlags);

        if (cellErrorInfo != null) hasCellValidationError = (bool)cellErrorInfo.GetValue(this, null);
        if (rowErrorInfo != null) hasRowValidationError = (bool)rowErrorInfo.GetValue(this, null);

        base.OnCanExecuteBeginEdit(e);
        if (!e.CanExecute && !hasCellValidationError && hasRowValidationError )
        {
            e.CanExecute = true;
            e.Handled = true;
        }
    }

    #region baseOnCanExecuteBeginEdit
    //protected virtual void OnCanExecuteBeginEdit(CanExecuteRoutedEventArgs e)
    //{
    //    bool canExecute = !IsReadOnly && (CurrentCellContainer != null) && !IsEditingCurrentCell && !IsCurrentCellReadOnly && !HasCellValidationError;

    //    if (canExecute && HasRowValidationError)
    //    {
    //        DataGridCell cellContainer = GetEventCellOrCurrentCell(e);
    //        if (cellContainer != null)
    //        {
    //            object rowItem = cellContainer.RowDataItem;

    //            // When there is a validation error, only allow editing on that row
    //            canExecute = IsAddingOrEditingRowItem(rowItem);
    //        }
    //        else
    //        {
    //            // Don't allow entering edit mode when there is a pending validation error
    //            canExecute = false;
    //        }
    //    }

    //    e.CanExecute = canExecute;
    //    e.Handled = true;
    //}
    #endregion baseOnCanExecuteBeginEdit
}

答案 1 :(得分:1)

Gazi's答案可以使表格保持可编辑状态。但是,尽管使用此实现,用户可以继续编辑其他字段,但是这些其他更改的值将不会写入模型,并且无法进行进一步的验证。 当进一步的用户更改将导致其他验证错误时,此问题变得很明显。这些不会被提出,因为不会进行源更新和验证。

要克服此问题,需要在OnExecutedCommitEdit内部完成类似您的实现。参见the answer to this multiple DataGrid row validation question