如何以编程方式从数据网格视图中的一个单元格移动到另一个单元格?

时间:2012-08-30 22:28:09

标签: c# windows datagridview keydown datagridviewtextboxcell

我只需要允许一个字符输入到可编辑的datagridview单元格中(每隔一列,奇数编号的单元格都是可编辑的);如果用户在其中一个单元格中添加第二个字符,则光标应该向下移动到下一个单元格并将第二个值放在那里(再次按下该键再次向下移动,依此类推)。如果在网格的底部(第12行),它应该移动到第0行并且还向右移动两列。

我试过这样做:

private void dataGridViewPlatypus_KeyDown(object sender, KeyEventArgs e) {
    var currentCell = dataGridViewPlatypus.CurrentCell;
    int currentCol = currentCell.ColumnIndex;
    int currentRow = currentCell.RowIndex;
    if (currentCell.Value.ToString().Length > 0) {
        if (currentRow < 11) {
            dataGridViewPlatypus.CurrentCell.RowIndex = currentRow+1;
        } else if (currentRow == 11) {
            currentCell.RowIndex = 0;
            currentCell.ColumnIndex = currentCell.ColumnIndex + 2;
            dataGridViewPlatypus.CurrentCell = currentCell;
        }
    }
}

...但是我得到的错误信息是RowIndex和ColumnIndex无法分配,因为它们是只读的。

那我怎么能做到这一点?

警告:我知道如果当前位于最后一个可编辑列的底部,我还必须添加逻辑以移至第1列。

更新

从tergiver的回答中,这是我到目前为止所得到的,但我不知道如何前进到下一个单元格。

protected override bool ProcessCmdKey(ref Message msg, Keys keyData) {
    if (this.ActiveControl == dataGridViewPlatypus)
    {
        var currentCell = dataGridViewPlatypus.CurrentCell;
        if (currentCell.Value.ToString().Length == 1) 
        {
            ;//Now what?
        }
    }
    return base.ProcessCmdKey(ref msg, keyData);
}

更新2

谢谢大家;这就是我正在使用它来使它非常有用(我仍然希望能够让用户只需按住键,并在后续单元格中连续输入该值):

private void dataGridViewPlatypus_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e) {
    int columnIndex = (((DataGridView)(sender)).CurrentCell.ColumnIndex);
    if (columnIndex % 2 == 1) {
        e.Control.KeyDown -= TextboxNumeric_KeyDown;
        e.Control.KeyDown += TextboxNumeric_KeyDown;
        e.Control.KeyUp -= TextboxNumeric_KeyUp;
        e.Control.KeyUp += TextboxNumeric_KeyUp;
    }
}

private void TextboxNumeric_KeyDown(object sender, KeyEventArgs e) {
    var tb = sender as TextBox;
    if (tb != null) {
        tb.MaxLength = 1;
    }
}

// TODO: Now need to find a way to be able to just press down once
private void TextboxNumeric_KeyUp(object sender, KeyEventArgs e) {
    var tb = sender as TextBox;
    if (tb != null && tb.TextLength >= 1) {
        if (dataGridViewPlatypus.CurrentCell.RowIndex != dataGridViewPlatypus.Rows.Count - 1) {
            dataGridViewPlatypus.CurrentCell = dataGridViewPlatypus[
                dataGridViewPlatypus.CurrentCell.ColumnIndex,
                dataGridViewPlatypus.CurrentCell.RowIndex + 1];
        } else { // on last row
            this.dataGridViewPlatypus.CurrentCell = this.dataGridViewPlatypus.CurrentCell.ColumnIndex !=    dataGridViewPlatypus.Columns.Count - 1 ? this.dataGridViewPlatypus[this.dataGridViewPlatypus.CurrentCell.ColumnIndex    + 2, 0] : this.dataGridViewPlatypus[1, 0];
        }
    }
}

4 个答案:

答案 0 :(得分:3)

CurrentCell的{​​{1}}属性有一个setter,允许您传入一个新单元格。

解决此问题的一种方法是处理网格的DataGridView事件,并将EditingControlShowing处理程序附加到编辑控件,如下所示:

KeyPress

然后在按键处理程序中你有:

private void dataGridView1_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e)
{                
    if ((int)(((System.Windows.Forms.DataGridView)(sender)).CurrentCell.ColumnIndex) == 1)
    {
        e.Control.KeyPress += TextboxNumeric_KeyPress;
    }
}

上面的逻辑当然不适用于你的情况,但传递新的CurrentCell的原则(在从网格中检索所需的单元格之后)就是这样。

答案 1 :(得分:1)

DGV上的KeyDown将无法工作,因为DataGridViewTextBoxColumn使用就地显示的TextBox控件,并且可以及时移动到位以进行编辑。

由于所有文本列只有一个就地TextBox,您可以订阅其KeyDown事件,但是获取对该控件的引用可能存在鸡和蛋问题。

最好是使用Form的ProcessCmdKey覆盖并在那里执行此逻辑。当按键发生时,检查DGV是否为ActiveControl,检查当前单元格是否为文本单元格,检查单元格是否已包含字符,然后在允许处理密钥之前更改当前单元格。

已更新 - 选择

using System;
using System.Drawing;
using System.Linq;
using System.Runtime.InteropServices;
using System.Windows.Forms;

class Item
{
    public string A { get; set; }
    public string B { get; set; }
    public string C { get; set; }
    public string D { get; set; }
}

class Form1 : Form
{
    [STAThread]
    static void Main()
    {
        Application.EnableVisualStyles();
        Application.SetCompatibleTextRenderingDefault(false);
        Application.Run(new Form1());
    }

    DataGridView dataGridViewPlatypus;

    public Form1()
    {
        ClientSize = new Size(480, 260);
        Controls.Add(dataGridViewPlatypus = new DataGridView
        {
            Dock = DockStyle.Fill,
            DataSource = Enumerable.Range(1, 10).Select(i => new Item { A = "", B = "", C = "", D = "" }).ToList(),
        });
    }

    [DllImport("User32.dll")]
    extern static int PostMessage(IntPtr hWnd, int msg, IntPtr wParam, IntPtr lParam);

    protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
    {
        if (msg.Msg == 256) // WM_KEYDOWN
        {
            if (this.ActiveControl == dataGridViewPlatypus.EditingControl)
            {
                var currentCell = dataGridViewPlatypus.CurrentCell;
                if (currentCell.OwningColumn is DataGridViewTextBoxColumn && dataGridViewPlatypus.EditingControl.Text.Length > 0)
                {
                    int rowIndex = currentCell.RowIndex;
                    int columnIndex = currentCell.ColumnIndex;

                    if (++columnIndex >= dataGridViewPlatypus.Columns.Count)
                    {
                        columnIndex = 0;
                        if (++rowIndex >= dataGridViewPlatypus.Rows.Count)
                            rowIndex = 0;
                    }

                    dataGridViewPlatypus.CurrentCell = dataGridViewPlatypus[columnIndex, rowIndex];
                    PostMessage(dataGridViewPlatypus.Handle, msg.Msg, msg.WParam, msg.LParam);
                    return true; // Don't process this message, we re-sent it to the DGV
                }
            }
        }
        return base.ProcessCmdKey(ref msg, keyData);
    }
}

答案 2 :(得分:1)

单元格具有您可以设置的Selected属性。只需按列和行索引访问单元格即可。

我相信你可以做到

dgView.rows[0].cells[0].selected = true

将在(0,0)或第一行,第一列相交处给出单元格。 或者,您可以抓住这样的行:

我认为这是班级 - &gt; DataGridViewRow row = dgView.rows[0]

然后

row[0].cells[0].Selected = true

       Column 1  Column 2
Row 1 [this guy][        ]
Row 2 [        ][        ]

编辑:

要完成下一个单元格,只需执行以下操作:

sameRow.cells[currentCell.ColumnIndex+1].Selected = true;

我可能错过了那里的一些大写,但你明白了。

答案 3 :(得分:1)

如果所选单元格在第一列中,我有一个过程会炸弹。因此,在该按钮的代码中,这是其中的第一个代码:(顺便说一句,我在网格中使用了单元格选择)

if (dgvGrid.CurrentCell.ColumnIndex == 0) // first column
    dgvGrid.Rows[dgvGrid.CurrentCell.RowIndex].Cells[1].Selected = true;

这有效地“跳至”下一列,然后我的其余过程开始工作。