5

我只需要允许一个字符输入到可编辑的 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

4 回答 4

3

CurrentCell属性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;
    }
}

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

private void TextboxNumeric_KeyPress(object sender, KeyPressEventArgs e)
{
    TextBox tb = sender as TextBox;
     if (tb.TextLength >= 5)
     {
         dataGridView1.CurrentCell = dataGridView1[dataGridView1.CurrentCell.ColumnIndex + 1, dataGridView1.CurrentCell.RowIndex];
     }
}

上面的逻辑当然不适合您的情况,但传入新 CurrentCell 的原则(在从网格中检索所需的单元格之后)仍然存在。

于 2012-08-31T12:29:14.793 回答
1

DGV 上的 KeyDown 不起作用,因为 DataGridViewTextBoxColumn 使用就地 TextBox 控件,它使其可见并及时移动到适当位置以进行编辑。

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

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

更新 - 拿 2

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);
    }
}
于 2012-08-30T22:34:56.327 回答
1

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

我相信你能做到

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

这将为您提供 (0,0) 或第一行第一列相交的单元格。或者,您可以像这样抓取行:

我认为这就是类-->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;

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

于 2012-08-30T22:35:10.207 回答
1

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

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

这有效地“选项卡”到下一列,然后我的其余过程开始工作。

于 2018-07-19T22:11:19.053 回答