DataGridView MouseWheel sendkeys updown问题

时间:2018-04-21 06:37:20

标签: c# performance datagridview scroll mousewheel

我创建一个子类datagridview来覆盖mousewheel事件以捕获鼠标滚动然后发送键UP或DOWN。我通过按钮单击

创建一个数据表作为mydatagridview的数据源绑定
    private void button1_Click(object sender, EventArgs e)
    {
        DataTable myDataTable = new DataTable();

        int NUM_ROWS = 150;
        int NUM_COLS_TO_CREATE = 10;

        for (int i = 0; i < NUM_COLS_TO_CREATE; i++)
        {
            myDataTable.Columns.Add("x" + i, typeof(string));
        }

        for (int i = 0; i < NUM_ROWS; i++)
        {
            var theRow = myDataTable.NewRow();

            for (int j = 0; j < NUM_COLS_TO_CREATE; j++)
            {
                theRow[j] = "whatever";
            }

            //add the row *after* populating it
            myDataTable.Rows.Add(theRow);
        }

        MyDataGridView1.DataSource = myDataTable;
    }

覆盖鼠标滚轮事件的代码
public partial class MyDataGridView : DataGridView
{
    protected override void OnMouseWheel(MouseEventArgs e)
    {
        if (e.Delta < 0)
            SendKeys.Send("{DOWN}");
        else
            SendKeys.Send("{UP}");
    }
}

如果我们使用鼠标滚轮以慢速方式滚动每个项目,它的工作正常,但如果使用鼠标滚轮滚动太快,有点数据网格视图会变得滞后。

作为第1行到第5行的示例,它会将行从1跳到3,然后从3到5跳到这样的东西,这是另一个奇怪的问题。我使用&#34; Navicat&#34;我的日常工作很多..

所以,如果我打开我的应用程序和Navicat。即使我滚动太快,鼠标滚轮滚动现在在我的应用程序上变得非常流畅。但是如果我关闭Navicat然后滚动再次变得滞后。是什么导致了这个?我很抱歉,如果我不能解释它,我想要的只是想让滚动每个项目顺利。有什么建议吗?

3 个答案:

答案 0 :(得分:0)

SendKeys方法在精确计时方面不太可靠 - 请参阅official documentation。尝试将“SendKeys”应用设置设置为“SendInput”以强制执行新行为。

但是,您最好处理 MouseWheel事件,而不是覆盖。您需要手动挂钩 - 不确定为什么它不存在于属性窗口中。鉴于您的DataGridView名为dgv

private void Form1_Load(object sender, EventArgs e)
{
    dgv.MouseWheel += Dgv_MouseWheel;
}

接下来,您考虑过FirstDisplayedScrollingRowIndex了吗?只需在事件中将其设置为适当,并设置Handled标志,如下所示:

private void dgv_MouseWheel(object sender, MouseEventArgs e)
{
    dgv.FirstDisplayedScrollingRowIndex += 3;
    var he = (HandledMouseEventArgs);
    he.Handled = true;
}

答案 1 :(得分:0)

@ pozhidar提到我应该更好地处理MouseWheel事件或覆盖它。所以我想出了解决方案,万一有人也需要它。

在Form_Load中添加

MyDataGridView1.MouseWheel += new MouseEventHandler(MyDataGridView1_MouseWheel);

然后把它放在你班级的任何地方

private void MyDataGridView1_MouseWheel(object sender, MouseEventArgs e)
{
    HandledMouseEventArgs hme = (HandledMouseEventArgs)e;
    hme.Handled = true;

    int rowIndex = MyDataGridView1.CurrentCell.RowIndex;
    int cellIndex = MyDataGridView1.CurrentCell.ColumnIndex;

    MyDataGridView1.CurrentCell = MyDataGridView1.Rows[e.Delta < 0 ? Math.Min(rowIndex + 1, MyDataGridView1.RowCount - 1) : Math.Max(rowIndex - 1, 0)].Cells[cellIndex];

}

答案 2 :(得分:0)

这是一种保留滚动视口的默认滚动行为但不更改行选择的方法。

当在虚拟网格中处理数百万行时,它的执行速度也比内置鼠标滚轮处理程序快得多:

private void Form1_Load(object sender, EventArgs e)
{
    DataGridView1.MouseWheel += DataGridView1_MouseWheel;
}

private void DataGridView1_MouseWheel(object sender, MouseEventArgs e)
{
    var hme = e as HandledMouseEventArgs;
    hme.Handled = true;

    int displayedRowIndex = DataGridView1.FirstDisplayedScrollingRowIndex;

    // each "detente" scroll appears to create a delta of 120
    // dividing delta by 120 to get the number of rows scrolled
    // taking the negative of the delta so that it can be added to the displayedRowIndex intuitively as negative is down, positive is up
    var rowDelta = -(e.Delta / 120);

    var newDisplayedRowIndex = e.Delta < 0 ? Math.Min(displayedRowIndex + rowDelta, DataGridView1.RowCount - 1) : Math.Max(displayedRowIndex + rowDelta, 0);

    DataGridView1.FirstDisplayedScrollingRowIndex = newDisplayedRowIndex;
}