4

我为这个错误搜索了很多很多相同的问题,但它并没有解决我的问题。我正进入(状态

操作无效,因为它会导致对 SetCurrentCellAddressCore 函数的可重入调用。

场景是我 使用datagridview它来转换它,然后我再次将 ComboBoxColumn 更改为 TextboxColumn。这些代码适用于所有人。但是在确切的行中得到所述错误会引发此异常,但其他行不会显示错误。如果我忽略此错误并继续,则 e.RowIndex = 2 单元格值变为空白,其他行值有效。TextboxColumnCellBeginEditComboBoxColumnCellValidatee.RowIndex = 2

这是代码CellBeginEdit

 if (e.ColumnIndex == 2 && e.RowIndex >= 0)
  {
    try
     {
      string s = Convert.ToString(_dgvCoarseAggegateTest[e.ColumnIndex, e.RowIndex].Value);
      string s1 = Convert.ToString(_dgvCoarseAggegateTest[e.ColumnIndex, 0].Value);
      DataGridViewComboBoxCell c = new DataGridViewComboBoxCell();

      string _SizeName = _cGetParrent._mGetParentCellValue(ref _dgvCoarseAggegateTest, e.RowIndex, 1);                  
      _mFillSieveSizeGridCombo(_mGetMetalSizeID(_SizeName), ref c); // Here My Combo Will GetValues from SQL and it Returning Value
      _dgvCoarseAggegateTest[e.ColumnIndex, e.RowIndex] = c; // Heres the error When e.RowIndex == 2 and if e.RowIndex != 2 then no error
      _dgvCoarseAggegateTest[e.ColumnIndex, e.RowIndex].Value = s;
      _dgvCoarseAggegateTest[e.ColumnIndex, 0].Value = s1;
     }
     catch (Exception ex)
      {
        MessageBox.Show(ex.Message);
      }
    }

如何解决这个问题。

更新: 没有行用户将添加新行并选择值,基本的事情是我想从数据库中显示组合和填充值,填充值取决于条件,所以每次新值都会出现,

样本数据

testTable
1      A
2      B
3      C
4      D
5      E
6      F
7      G
8      H
9      I

在第 1 列中,我添加了一个值从 1 到 9 的组合,在_mFillSieveSizeGridCombo我将 id 传递给 sql server 2008 并使用Combo.Item.Add(x)方法填充组合。

4

3 回答 3

2

内部有一个标志,SetCurrentCellAddressCore()可防止任何重入调用破坏DataGridView. 通常,使用 flag = true 引发事件并在事件结束时重置。

要解决此问题,您可以简单地在事件内部添加一个包装器BeginInvoke(),以使您的进程在事件之后以异步方式运行。

编辑

该问题可以在EditOnEnter模式下重现,并且事件外的单元格设置器BeginInvoke导致无限循环

private bool _suppressCellBeginEdit = false;
private void dgv_CellBeginEdit(object sender, DataGridViewCellCancelEventArgs e)
{
    var dgv = sender as DataGridView;
    if (_suppressCellBeginEdit)
        return;

    if (e.ColumnIndex == 2 && e.RowIndex >= 0)
    {
        string s = Convert.ToString(dgv[e.ColumnIndex, e.RowIndex].Value);
        string s1 = Convert.ToString(dgv[e.ColumnIndex, 0].Value);
        DataGridViewComboBoxCell c = new DataGridViewComboBoxCell();

        c.Items.Add(string.Format("x{0}:y{1} {2}", e.RowIndex, e.ColumnIndex, 0));
        c.Items.Add(string.Format("x{0}:y{1} {2}", e.RowIndex, e.ColumnIndex, 1));
        c.Items.Add(string.Format("x{0}:y{1} {2}", e.RowIndex, e.ColumnIndex, 2));

        // special handling
        if (e.RowIndex == e.ColumnIndex)
        {
            this.BeginInvoke(new Action(() =>
            {
                _suppressCellBeginEdit = true;
                this.Invoke(new Action(() => 
                    {
                        c.Value = s;
                        dgv[e.ColumnIndex, e.RowIndex] = c;
                        dgv[e.ColumnIndex, 0].Value = s1;
                    }));
                _suppressCellBeginEdit = false;
            }));
        }
        else
        {
            c.Value = s;
            dgv[e.ColumnIndex, e.RowIndex] = c;
            dgv[e.ColumnIndex, 0].Value = s1;
        }
    }
}
于 2015-03-16T09:12:21.767 回答
2

从您在实现此功能时遇到的麻烦中可以看出,DataGridView 对您尝试拉地板垫感到非常不满。它明确禁止在关键时刻更改单元格对象。虽然它正在处理一个事件本身就是这样一个关键时刻。事件的一个通用问题,称为re-entrancy。@Eric 的方法遇到的问题表明,这确实是一个难以解决的问题。

所以你不想做的是修改单元格类型或引用。盯着球看,你真正想做的是修改下拉列表的内容。那不是问题。返回设计器并将列的 ColumnType 属性更改为 DataGridViewComboBoxColumn。并使用 CellBeginEdit 事件动态更改组合框项目集合。一个简单的例子:

    private void _dgvCoarseAggegateTest_CellBeginEdit(object sender, DataGridViewCellCancelEventArgs e) {
        var dgv = (DataGridView)sender;
        if (e.ColumnIndex == 2) {
            var cell = (DataGridViewComboBoxCell)dgv.Rows[e.RowIndex].Cells[e.ColumnIndex];
            cell.Items.Clear();
            // Run your dbase query here to fill cell.Items
            //...
            // We'll just fake it here for demo purposes:
            cell.Items.Add(e.RowIndex.ToString());
            cell.Items.Add((e.RowIndex+1).ToString());
            cell.Items.Add((e.RowIndex+2).ToString());
        }
    }
于 2015-03-16T12:52:17.773 回答
1

这是一种解决方法:在CellBeginEdit事件中首先检查是否ColumnTypeDataGridViewComboBoxCell. 如果不是我们cancel事件,请调用一个更改列类型的函数,然后再次调用该事件:

void switchCellType(object sender, DataGridViewCellCancelEventArgs e)
{
    DataGridViewComboBoxCell c = new DataGridViewComboBoxCell();
    // prepare the cell:
    //..
    // fill the drop down items..
    c.Items.Add("1");  // use
    c.Items.Add("2");  // your 
    c.Items.Add("3");  // code here!
    DGV[e.ColumnIndex, e.RowIndex] = c;  // change the cell
    DGV_CellBeginEdit(sender, e);        // restart the edit with the original parms  
}


private void DGV_CellBeginEdit(object sender, DataGridViewCellCancelEventArgs e)
{

    DataGridViewCell cell = DGV[e.ColumnIndex, e.RowIndex];
    if (!(cell is DataGridViewComboBoxCell))
    {
        e.Cancel = true;
        switchCellType(sender, e);
        return;
    }
    //..

现在您的代码可以继续进行,显然无需更改单元格。可能您还想传入要设置的文本值..

请注意,您必须确保CellEndEdit事件不会在时间之前恢复更改!也许是一面旗帜,也许Tag会有所帮助。如果你愿意,我可以看看你的CellEndEdit代码,如果有的话..

于 2015-03-16T14:41:57.170 回答