5

我有一个带有几个 DataGridViewComboBoxColumns 的 DataGridView。DataGridView 上有一个 CellEnter 事件处理程序,用于单击下拉组合框。

该列绑定到 KeyValuePairs 列表,ValueMember 为“Key”,DisplayMember 为“Value”。

当我单击组合框列时,它工作正常。但是,如果单元格处于“下拉”状态并且我单击另一个组合框(同一列,不同行),它会正确取消选择旧单元格,选择并下拉新单元格,但是顶部的选定值会更改为从旧单元格中获取值一瞬间,然后再更改回正确的单元格。

例如,假设列表是 A、B、C。在第 1 行中,选择了 A,在第 2 行中,选择了 B。我单击 row1 中的单元格,一切正常。然后,当这个单元格被下拉时,我单击 row2 中的单元格。它正确下降,但顶部的选定值变为 A,然后立即切换回 B(正确的值)。

如果我在单击第二个组合框单元格之前单击其他列中的单元格,则不会发生这种情况。

有没有办法防止这种情况发生?

重现问题的示例代码(事件处理程序连接到明显的事件):

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace PDGV
{
    public partial class Form1 : Form
    {
        List<KeyValuePair<string, string>> bindingList = new List<KeyValuePair<string, string>>();

        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            dataGridView1.Rows.Add(10);
            bindingList.Add(new KeyValuePair<string,string>("aaa", "111"));
            bindingList.Add(new KeyValuePair<string,string>("bbb", "222"));
            bindingList.Add(new KeyValuePair<string,string>("ccc", "333"));
            bindingList.Add(new KeyValuePair<string,string>("ddd", "444"));
            bindingList.Add(new KeyValuePair<string,string>("eee", "555"));
            BindComboList(2, bindingList);

        }

        private void BindComboList(int columnIndex, object list)
        {
            var column = dataGridView1.Columns[columnIndex] as DataGridViewComboBoxColumn;
            if (column != null)
            {
                column.DataSource = new BindingSource(list, null);
                column.DisplayMember = "Value";
                column.ValueMember = "Key";
            }
        }

        private void dataGridView1_CellEnter(object sender, DataGridViewCellEventArgs e)
        {
            if (e.RowIndex == -1)
                return;

            dataGridView1.BeginEdit(true);
            var control = dataGridView1.EditingControl as DataGridViewComboBoxEditingControl;
            if (control != null)
                control.DroppedDown = true;
        }
    }
}
4

3 回答 3

4

理由:

DataGridViewEndEdit 方法的深处,它存储了以前使用的 EditingControl 的副本。this.latestEditingControl = this.editingControl; 然后,当您开始编辑另一个单元格时,它就会启动BeginEditInternal。在此期间,它会检查 latestEditingControl 是否不为空,并且编辑类型是否与最后一个单元格相同,如果是,它会重用该控件,这就是您看到 flash 的原因,它确实是同一个控件。

解决方案:

将 latestEditingControl 设置为 null,此控件无法通过普通属性/方法访问,必须使用反射。注意:这会导致完全不同的烦恼,现在您要离开的单元格闪烁白色:),但它确实解决了您寻求帮助的特定问题。

    private void dataGridView1_CellEndEdit(object sender, DataGridViewCellEventArgs e)
    {
        if (e.RowIndex == -1)
            return;

        //this.latestEditingControl
        Type t = dataGridView1.GetType();
        FieldInfo viewSetter = t.GetField("latestEditingControl", BindingFlags.Default | BindingFlags.NonPublic | BindingFlags.Instance);
        viewSetter.SetValue(dataGridView1, null);
    }
于 2012-07-17T23:01:49.880 回答
2

我没有 100% 重现所描述的问题(我从未见过滞后),但尝试取消订阅该CellEnter事件并尝试将您的代码更改为此(来自如何让 DataGridView 组合框一键显示其下拉列表? )

public Form1() {
  InitializeComponent();

  dataGridView1.EditMode = DataGridViewEditMode.EditOnEnter;
  dataGridView1.EditingControlShowing += dataGridView1_EditingControlShowing;
}

void dataGridView1_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e) {
  if (e.Control is ComboBox) {
    SendKeys.Send("{F4}");
  }
}

当用户输入单元格时,它应该会弹出打开组合框。


或从这个解决方案单击打开下拉列表(在数据网格视图中)项目

void dataGridView1_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e) {
  if (e.Control is ComboBox) {
    ComboBox ctl = e.Control as ComboBox;
    ctl.Enter -= new EventHandler(ctl_Enter);
    ctl.Enter += new EventHandler(ctl_Enter);
  }
}

void ctl_Enter(object sender, EventArgs e) {
  (sender as ComboBox).DroppedDown = true;
}
于 2012-07-16T18:58:45.530 回答
1

我假设它与 Windows 7 的 3d 样式组合框(也许 vista 也是,我还没有检查)

我从来没有见过这种情况如果你

comboboxcolumn.FlatStyle = System.Windows.Forms.FlatStyle.Flat;

不过,您可能不喜欢它的外观。你可以隐藏丑陋的扁平向下箭头

comboboxcolumn.DisplayStyle = DataGridViewComboBoxDisplayStyle.Nothing;

就像另一个人说的那样,每列只有一个组合框控件(“编辑控件”),并且相同的组合框显示在该列的当前活动单元格中。当单元格未被选中时,由列类的绘制方法来绘制一个假的组合框供用户查看。因此,当您将一个值推送到一个不可见的组合框然后在不同的位置重新绘制它时,可能会发生某种特定的 3d 弹出样式的内部组合框绘制问题,谁知道呢。

尝试切换到“经典”Windows 主题。组合框应该表现自己。如果您可以强制组合框在 Aero 中以经典主题下的方式呈现它,那将会很有趣。我不确定它是否可能,但你可能想调查一下。

但无论如何,dgv 与其他组合框样式没有这个问题的事实让我认为它只是新的组合框样式不适合 datagridview。

于 2012-07-18T07:22:55.793 回答