如何在 DataGridView 的一列中的单元格之间切换

How to tab between cells in one column of a DataGridView

今天我运行入了如下要求的一个DataGridView:

假设您有一个包含多个列的 DataGridView,这些列只有信息性(不可编辑)值。 您可能只想在可编辑列之间切换。 在我的要求中,我有一个 DataGridView,其中包含 3 个不可编辑的文本列和一个可编辑的 DataGridViewComboBoxColumn。 只需要在 DataGridViewComboBoxCells 之间切换。

我想出了以下解决方案,我想分享给大家。

/// <summary>
/// Overrided method to tab only in a DataGridComboBox column.
/// </summary>
/// <param name="msg">A Message, passed by reference, that represents the window message to process.</param>
/// <param name="keyData">One of the Keys values that represents the key to process.</param>
/// <returns>True if the character was processed by the control. Otherwise, False.</returns>
protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
{
    bool TabIsPressed = ((keyData == Keys.Tab) || (keyData == (Keys.Shift | Keys.Tab))); // Checks if Tab or Shift/Tab is pressed.
    Control CurrentControl = this.ActiveControl; // Gets the currently focused control.
    bool IsDataGridViewControl = (CurrentControl.GetType() == typeof(DataGridView)); // Checks if the currently focused control is a DataGridView control.
    bool IsComboBoxEditingControl = (CurrentControl.GetType() == typeof(DataGridViewComboBoxEditingControl)); // Checks if the currently focused control is a DataGridViewComboBoxEditingControl control.
    bool DataGridViewIsValid = (IsDataGridViewControl && dataGridViewDMSSettings.Rows.Count > -1); // Checks if a DataGridView control is focused is not empty.
    if
    (
        TabIsPressed // If a Tab key is pressed.
        && (
            DataGridViewIsValid // And a DataGridView is focused and valid.
            || IsComboBoxEditingControl // Or a DataGridViewComboBoxEditingControl is focused.
        )
    )
    {
        int CurrentRowIndex = dataGridViewDMSSettings.CurrentCell.RowIndex; // Gets the current rows index.
        int LastRowIndex = dataGridViewDMSSettings.RowCount - 1; // Gets the last rows index.
        if (IsComboBoxEditingControl) // If a DataGridViewComboBoxEditingControl is focused.
        {
            ((DataGridViewComboBoxEditingControl)CurrentControl).EditingControlDataGridView.EndEdit(); // Ends the editing mode for the control.
        }
        int NextRowIndex = CurrentRowIndex; // Creates a variable for the next rows index.
        switch (keyData) // Switches through the pressed keys.
        {
            case (Keys.Tab): // If the Tab key is pressed.
                if (IsComboBoxEditingControl) // If a DataGridViewComboBoxEditingControl is focused.
                {
                    NextRowIndex++; // Sets the next row index to the next row.
                    if (NextRowIndex > LastRowIndex) // If next row index is greater than the last row index.
                    {
                        NextRowIndex = 0; // Focuses the first row. (Alternatively call "base.ProcessCmdKey(ref msg, keyData)" to exit the DataGridView)
                    }
                }
                break;
            case (Keys.Shift | Keys.Tab): // If Shift and Tab key is pressed.
                NextRowIndex--; // Sets the next row index to the previous row.
                if (NextRowIndex < 0) // If previous row index is smaller than the first row index.
                {
                    NextRowIndex = LastRowIndex; // Focuses the last row. (Alternatively call "base.ProcessCmdKey(ref msg, keyData)" to exit the DataGridView)
                }
                break;
        }
        dataGridViewDMSSettings.CurrentCell = dataGridViewDMSSettings.Rows[NextRowIndex].Cells["DMSIndex"]; // Sets the focus on the next DataGridViewComboBox.
        this.Refresh(); // Rerenders the current form.
        return true; // Returns true to the caller.
    }
    else // If another key (Not a Tab key) is pressed or the currently focused control is not a DataGridView or DataGridViewComboBoxEditing control.
    {
        return base.ProcessCmdKey(ref msg, keyData); // Performs the standard ProcessCmdKey method.
    }
}

这个解决方案唯一的小问题是退出的 DataGridComboBoxCell 视觉呈现有点慢(你可以看到它变白了一秒钟)我已经最小化了影响:

this.Refresh();

但它并没有消失。 P.S。启用表单双缓冲区并没有改变这个效果。

也许有人可以改进我的方法来解决最后一个问题。