DataGridView

DataGridViewは、DataGridを拡張したクラスです。Differences Between the Windows Forms DataGridView and DataGrid Controls | Microsoft Learn

クラス階層

  • System.Object
    • System.MarshalByRefObject
      • System.ComponentModel.Component
        • System.Windows.Forms.Control
          • System.Windows.Forms.DataGridView

コンストラクタ

DataGridView dataGridView = new DataGridView();

DataGridViewColumn column1 = new DataGridViewTextBoxColumn() { HeaderText = "A" };
DataGridViewColumn column2 = new DataGridViewTextBoxColumn() { HeaderText = "B" };
dataGridView.Columns.AddRange(column1, column2);

dataGridView.Rows.Add("A1", "B1");
dataGridView.Rows.Add("A2", "B2");

プロパティ

プロパティ 内容
object DataSource データソース
string DataMember データソースに複数のテーブルまたはリストが含まれるときに、対象とするそれの名前。単一のテーブルまたはリストならば、指定する必要はない
DataGridViewColumnCollection Columns すべての列 (DataGridViewColumn) を含むコレクション。ただし行ヘッダーを含む列は除く
DataGridViewRowCollection Rows すべての行 (DataGridViewRow) を含むコレクション。ただし列ヘッダーを含む行は除く
DataGridViewCell Item[Int32, Int32] 指定の列の、指定の行にあるセル
DataGridViewHeaderCell TopLeftHeaderCell 左上隅にあるヘッダーセル。MultiSelectがtrueのとき、これをクリックするとSelectAll()が呼ばれる OnTopLeftHeaderMouseDown - DataGridViewMethods.cs
int ColumnCount 表示されている列の数。これに設定すると、その数になるように列が削除または追加される

これから取得するのはColumns.Countと等しい ColumnCount - DataGridView.cs

ユーザーに表示されている列の数は、DisplayedColumnCount()で得られる

int RowCount 表示されている行の数。これに設定すると、その数になるように行が削除または追加される

これから取得するのはRows.Countと等しい RowCount - DataGridView.cs

ユーザーに表示されている行の数は、DisplayedRowCount()で得られる

bool IsCurrentCellDirty trueならば、現在のセルがコミット (確定) されていないデータを含む
bool IsCurrentRowDirty trueならば、現在の行がコミット (確定) されていないデータを含む
bool IsCurrentCellInEditMode trueならば、現在のセルが編集中
Control EditingControl セルが編集モードのとき、そのセルでホストされているコントロール。編集モードでなければnull
Panel EditingPanel EditingControlを格納しているパネル。EditingControlとは異なり、編集モードとは無関係につねに利用できる
int HorizontalScrollingOffset 水平にスクロールされているピクセル数
int VerticalScrollingOffset 垂直にスクロールされているピクセル数
Rectangle DisplayRectangle コントロールの表示領域を表す四角形
プロパティ 内容
DataGridViewCell FirstDisplayedCell 表示されている最初のセル。通常は左上隅のセルだが、RTL言語では右上となる。またこれに設定すると、このセルが表示されるようにスクロールする
int FirstDisplayedScrollingColumnIndex 表示されている最初ののインデックス
int FirstDisplayedScrollingRowIndex 表示されている最初ののインデックス。表示される前か表示する行がなければ-1が返されるが、-1を設定するとArgumentOutOfRangeExceptionとなる
DataGridViewCell CurrentCell 現在アクティブなセル。アクティブなセルがなければnull
Point CurrentCellAddress 現在アクティブなセルの行と列のインデックス。アクティブなセルがないならば、そのインデックスは(-1,-1)となる CurrentCellAddress - DataGridView.cs
DataGridViewRow CurrentRow 現在アクティブなセルを格納している行。アクティブなセルがなければnull。これを変更するには、アクティブにしたい行にあるセルにCurrentCellを設定する
DataGridViewSelectedCellCollection SelectedCells ユーザーによって選択されているセルのコレクション
DataGridViewSelectedColumnCollection SelectedColumns ユーザーによって選択されている列のコレクション (これが機能するのは、SelectionModeがFullColumnSelectまたはColumnHeaderSelectのときだけ)
DataGridViewSelectedRowCollection SelectedRows ユーザーによって選択されている行のコレクション (これが機能するのは、SelectionModeがFullRowSelectまたはRowHeaderSelectのときだけ) (コレクション内の項目の順は、選択された順と一致する保証はない Remarks - DataGridViewSelectedRowCollection Class (System.Windows.Forms) | Microsoft Learn)

逆にプログラムで行を選択するには、DataGridViewRow.Selectedをtrueに設定する

DataGridViewColumn SortedColumn 並べ替えの列。読取専用。並べ替えられていないならnull。設定するにはSort()を呼ぶ。並べ替えを独自に実装しているならば、この値は意味をなさない
SortOrder SortOrder 並べ替えの順。読取専用。設定するにはSort()を呼ぶ。並べ替えを独自に実装しているならば、この値は意味をなさない
プロパティ 内容 既定値
bool AutoGenerateColumns trueならば、DataSourceまたはDataMemberが設定されているときに、列が自動的に作成される。ただしColumnMappingがMappingType.Hiddenのように、非表示にされている列は作成されない true
DataGridViewAutoSizeColumnsMode AutoSizeColumnsMode 列の幅を決定する方法 ※1

行ヘッダーの幅はRowHeadersWidthSizeModeで指定する。Fillとしたとき、表示領域が小さいことによりMinimumWidthの幅を下回るときには、水平スクロールが表示される

None
DataGridViewAutoSizeRowsMode AutoSizeRowsMode 行の高さを幅を決定する方法 ※1 None

※1 パフォーマンスを考慮するならば自動サイズを用いない、または表示されているヘッダーやセルに基づいてサイズ調整させます。自動サイズ変更の効率的な使用 - DataGridView コントロールの拡大縮小に関するベスト プラクティス - Windows Forms .NET Framework | Microsoft Learn

プロパティ 内容 既定値
bool AllowUserToAddRows trueならば、ユーザーに新しい行の追加オプションが表示される true
bool AllowUserToDeleteRows trueならば、ユーザーは行を削除できる true
bool AllowUserToOrderColumns trueならば、ユーザーは列の順を変更できる false
bool AllowUserToResizeColumns trueならば、ユーザーは列のサイズを変更できる。行ヘッダーは、RowHeadersWidthSizeModeで指定する true
bool AllowUserToResizeRows trueならば、ユーザーは行のサイズを変更できる。列ヘッダーは、ColumnHeadersHeightSizeModeで指定する true
int RowHeadersWidth 行ヘッダーの幅。4以上、32768以下 RowHeadersWidth - DataGridView.cs 43
int ColumnHeadersHeight 列ヘッダーの高さ。4以上、32768以下 23
DataGridViewRowHeadersWidthSizeMode RowHeadersWidthSizeMode 行ヘッダーの幅の調整方法 EnableResizing
DataGridViewColumnHeadersHeightSizeMode ColumnHeadersHeightSizeMode 列ヘッダーの高さの調整方法 EnableResizing
bool ReadOnly trueならば、ユーザーはセルを編集できない false
bool MultiSelect trueならば、ユーザーはセル、行や列を同時に複数選択できる true
DataGridViewSelectionMode SelectionMode セルの選択方法  
DataGridViewEditMode EditMode セルの編集を開始する方法 EditOnKeystrokeOrF2
DataGridViewRow RowTemplate コントロール内のすべての行のテンプレートを表す行。これは新規に作成される行に適用されるため、行を追加する前に設定しておく。またこれを変更しても、既存の行には影響しない (列のテンプレートはDataGridViewColumn.CellTemplateで指定する)  
DataGridViewClipboardCopyMode ClipboardCopyMode ユーザーがセルのテキストの値をクリップボードへコピー可能か、そして行や列ヘッダーのテキストをそれに含めるかどうか  
bool ShowCellToolTips trueならば、セルの上でマウス ポインターが停止したときにツールチップを表示する true
bool ShowCellErrors trueならば、セルエラーを表示する true
bool ShowEditingIcon trueならば、セルの編集中に行ヘッダーに編集グリフ (editing glyph) を表示する (グリフを表示する領域が不足するならば、グリフは表示されない) true
bool ShowRowErrors trueならば、DataGridViewRow.ErrorTextが空文字列ではないときに行ヘッダーにエラーグリフ (error glyphs) を表示する (グリフを表示する領域が不足するならば、グリフは表示されない) true
bool ColumnHeadersVisible trueならば、列ヘッダーを表示する true
bool RowHeadersVisible trueならば、行ヘッダーを格納している列を表示する true
bool StandardTab trueならば、Tabキーでタブ オーダーの次のコントロールにフォーカスが移動する。さもなくば次のセルのフォーカスが移動する false
bool VirtualMode trueならば、独自のデータ管理操作を提供している。仮想モード false
       
プロパティ - DataGridView クラス (System.Windows.Forms) | Microsoft Learn
スタイル関連
区分 プロパティ 内容
既定値 DataGridViewCellStyle DefaultCellStyle 他のスタイルが設定されていないときに適用される、既定のセル スタイル
DataGridViewCellStyle RowsDefaultCellStyle 行のセル
DataGridViewCellStyle AlternatingRowsDefaultCellStyle 奇数行に適用
DataGridViewCellStyle RowHeadersDefaultCellStyle 行ヘッダーのセルに適用 ※1
DataGridViewCellStyle ColumnHeadersDefaultCellStyle 列ヘッダーに適用。左上隅のヘッダーセル (TopLeftHeaderCell) も含む ※1
境界線 (組み込みのスタイル) Color GridColor セルのグリッド線 (grid lines) の色
BorderStyle BorderStyle 境界線のスタイル (border style)。FixedSingle、Fixed3Dなど
DataGridViewCellBorderStyle CellBorderStyle Single、SingleVertical、SingleHorizontal、Raised、Sunkenなど
DataGridViewHeaderBorderStyle ColumnHeadersBorderStyle Single、Raised、Sunkenなど
DataGridViewHeaderBorderStyle RowHeadersBorderStyle  
境界線 DataGridViewAdvancedBorderStyle AdvancedCellBorderStyle  
DataGridViewAdvancedBorderStyle AdvancedColumnHeadersBorderStyle  
DataGridViewAdvancedBorderStyle AdvancedRowHeadersBorderStyle  
DataGridViewAdvancedBorderStyle AdjustedTopLeftHeaderBorderStyle  
  bool EnableHeadersVisualStyles trueならば、視覚スタイルが有効なときに、ヘッダーにそれを適用する。既定はtrue
※1 EnableHeadersVisualStylesをfalseとしないと、視覚スタイルで設定されているスタイルは適用されない

列ごとの既定のスタイルは、DataGridViewColumn.DefaultCellStyleで指定します。

DataSource

以下のインターフェイスを実装するオブジェクトを、データソースに指定できます。Remarks - DataGridView.DataSource Property (System.Windows.Forms) | Microsoft Learn

DataSourceを変更すると、CurrentCellはnullになります。DataSource - DataGridView.cs

DataSourceにしているオブジェクトを更新してもそれがDataGridViewに反映されないならば、BindingList<T>やBindingSourceを介して設定します。c# - Refresh DataGridView when updating data source - Stack Overflow

DataGridViewが親コントロールに含まれていなかったり、データソースとバインドする (結びつける) 列が存在しないときは何も表示されず、RowCountは0となります。

IndexOutOfRangeException

DataSourceへ設定後にそれへアクセスしたときに、CurrencyManager.get_Item(Int32 index)から「インデックス n に値がありません。(Index n does not have a value)」としてIndexOutOfRangeExceptionが投げられるときには、CurrencyManager.Refresh()でデータを更新します。c# - Datagridview error System.IndexOutOfRangeException: Index 0 does not have a value - Stack Overflow

CurrencyManager currencyManager = (CurrencyManager)dataGridView.BindingContext[dataGridView.DataSource];
currencyManager.Refresh();
CurrencyManager.Refresh メソッド (System.Windows.Forms) | Microsoft Learn

Columns

すべての列を表します。ただし行ヘッダーを含む列は除きます。DataGridViewColumnCollection クラス (System.Windows.Forms) | Microsoft Learn

個々の列はItem[]プロパティから取得できます。文字列で指定したときはDataGridViewColumn.Nameが対象となり、該当する列がなければnullが返されます。一方で数値で指定したときに範囲外ならば、ArgumentOutOfRangeException例外が投げられます。

CurrentCell

現在アクティブなセル (DataGridViewCell) を表します。これに設定するとこのセルが表示されるようにスクロールされ、フォーカスされます。また同時に選択された状態となるため、それが不要ならばSelectedをfalseとするか、SetCurrentCellAddressCore()で現在のセルを設定するようにします。

これにnullを設定することでフォーカスを表す四角形を一時的に除去できますが、コントロールがフォーカスを得たときこれがnullならば、これにFirstDisplayedCellの値が自動的に設定されます。

SelectionMode

セルの選択方法を設定できます。

DataGridViewSelectionMode 列挙型
列挙子 数値 選択方法
CellSelect 0 つねにセルごと
FullRowSelect 1 つねに行全体 ※2
FullColumnSelect ※1 2 つねに列全体
RowHeaderSelect 3 行のヘッダーセルがクリックされたならば行全体、セルがクリックされたならばセルごと [既定値]
ColumnHeaderSelect ※1 4 列のヘッダーセルがクリックされたならば列全体、セルがクリックされたならばセルごと
※1 SortModeがAutomaticのDataGridViewColumnが含まれていると、列全体を選択する設定にはできない
※2 .NET Framework 4.7.2以降、列ヘッダーの色で現在の列が識別できるようになった .NET 4.7.2 の Windows フォーム コントロールでのアクセシビリティの向上 - .NET Framework 4.7.1 から 4.7.2 への移行に関する変更の再ターゲット - .NET Framework | Microsoft Learn

VerticalScrollingOffset

このVerticalScrollingOffsetはHorizontalScrollingOffsetとは異なり、値を設定することはできません。DataGridView.VerticalScrollingOffset プロパティ (System.Windows.Forms) | Microsoft Learn

しかしinternalなプロパティであるVerticalOffsetに設定すれば、スクロールさせることは可能です。

Type type = typeof(DataGridView);
PropertyInfo verticalOffset = type.GetProperty("VerticalOffset", BindingFlags.Instance | BindingFlags.NonPublic);

verticalOffset.SetValue(dataGridView, newValue);
How do I programmatically scroll a winforms datagridview control? - Stack Overflow

ただし、そもそも垂直方向にはピクセル単位でスクロールしないため、FirstDisplayedScrollingRowIndexを用いて行単位でスクロールさせるのが簡単です。

またマウスの中ボタンによる水平スクロールやオートスクロールに対応するには、独自に実装する必要があります。

FirstDisplayedScrollingRowIndex

これに設定するときに行を表示する領域が不足していると、「行を表示する場所がありません。(No room is available to display rows.)」としてInvalidOperationExceptionが投げられます。FirstDisplayedScrollingRowIndex - DataGridView.cs

設定値が0未満またはRowCount以上ならば、「指定された引数は、有効な値の範囲内にありません。」としてArgumentOutOfRangeExceptionが投げられます。

DisplayRectangle

コントロールの表示領域を表す四角形であり、ClientRectangleからスクロールバーの領域を除いた領域に一致します。DisplayRectangle - DataGridView.cs

表示領域全体ではなく、個別の列や行の領域は

  • GetCellDisplayRectangle(Int32, Int32, Boolean)
  • GetColumnDisplayRectangle(Int32, Boolean)
  • GetRowDisplayRectangle(Int32, Boolean)

で得られます。また条件を満たす列や行の累積した幅や高さは

  • DataGridViewRowCollection.GetRowsHeight(DataGridViewElementStates)
  • DataGridViewColumnCollection.GetColumnsWidth(DataGridViewElementStates)

で得られます。

ClipboardCopyMode

ユーザーがセルのテキストをクリップボードへコピー可能か、そしてそのとき行や列のヘッダーを含めるかどうかを設定できます。コピーする内容の変更や、クリップボードからのコピーなどを実現するには、独自に実装する必要があります。

DataGridViewClipboardCopyMode 列挙型
列挙子 数値  
Disable 0 クリップボードへのコピーは無効
EnableWithAutoHeaderText 1 選択されているセルのテキストの値はクリップボードへコピーできる。ヘッダーテキストは、SelectionModeがRowHeaderSelectまたはColumnHeaderSelectであり、ヘッダーが選択されているとき含まれる [既定値]
EnableWithoutHeaderText 2 選択されているセルのテキストの値はクリップボードへコピーできる。ヘッダーテキストは、含まれない
EnableAlwaysIncludeHeaderText 3 選択されているセルのテキストの値はクリップボードへコピーできる。ヘッダーテキストは、つねに含まれる
DataGridViewClipboardCopyMode Enum (System.Windows.Forms) | Microsoft Learn ※1 行全体がコピーされる設定となっていても、 DataGridViewRow.HeaderCell.Valueに文字列が設定されていなければ、ヘッダーの文字列はコピーされない。

RowHeadersVisible

現在の行 (CurrentRow) には、それを示すグリフが表示されます。これを表示しない方法は提供されていないため、RowHeadersVisibleをfalseとして行ヘッダーを非表示にするか、オーナー描画でグリフを描画しないようにします。winforms - .NET DataGridView: Remove "current row" black triangle - Stack Overflow

メソッド

メソッド 機能
Sort(DataGridViewColumn, ListSortDirection) 内容を並べ替えられる
ProcessDataGridViewKey(KeyEventArgs) DataGridView内の移動に使用されるキーの処理を変更できる
ProcessDialogKey(Keys) 編集モードでのキー入力や、修飾キーの処理を変更できる
HitTest(Int32, Int32) 座標に対する列や行のインデックスの情報を得られる
ISupportInitialize.BeginInit() 初期化が開始されていることを、オブジェクトに通知できる
DisplayedColumnCount(Boolean) ユーザーに表示されている、列の数を得られる
DisplayedRowCount(Boolean) ユーザーに表示されている、行の数を得られる DisplayedRowCount - DataGridViewMethods.cs
GetCellCount(DataGridViewElementStates) 条件に合う、セルの数を得られる
BeginEdit(Boolean) 現在のセルを、編集モードにできる。現在のセルがReadOnlyまたはEditTypeがnullならば、falseが返される
CancelEdit() 現在のセルの編集モードをキャンセルして、変更を破棄できる
UpdateCellValue(Int32, Int32) 指定位置のセルを新しい値で更新できる。自動サイズモード (automatic sizing modes) が適用される。セルの再描画が必要なだけならば、InvalidateCell()を呼ぶ
InvalidateCell(Int32, Int32) 指定位置のセルを無効化して、強制的に再描画させられる (forcing it to be repainted)

内部ではセルの描画位置を取得してControl.Invalidate()を呼ぶだけのため、描画は強制されない InvalidateCell - DataGridViewMethods.cs

InvalidateColumn(Int32) 指定位置の列を無効化して、強制的に 再描画させられる
InvalidateRow(Int32) 指定位置の行を無効化して、強制的に 再描画させられる
GetCellDisplayRectangle(Int32, Int32, Boolean) 指定のセルの表示領域を得られる。インデックスに-1を指定して、行ヘッダーや列ヘッダーのセルを対象とできる
GetColumnDisplayRectangle(Int32, Boolean) 指定の列の表示領域を得られる。指定できる列インデックスは0以上。返される領域のX座標は行ヘッダーの左端が基準であり、HorizontalScrollingOffsetを考慮しない表示上の位置
GetRowDisplayRectangle(Int32, Boolean) 指定の行の表示領域を得られる。指定できる行インデックスは0以上。第2引数にtrueを渡すと表示されている領域のみ、さもなくば行全体が返される。一部も表示されていなければRectangle.Emptyが返される GetRowDisplayRectangle - DataGridViewMethods.cs
SelectAll() すべてのセルを選択できる
ClearSelection() 選択されているセルの、選択を解除できる
   
メソッド - DataGridView クラス (System.Windows.Forms) | Microsoft Learn

EndEdit()

現在のセルの編集操作をコミットし、終了させられます。DataGridView.EndEdit メソッド (System.Windows.Forms) | Microsoft Learn

DataGridViewDataErrorContextsを省略すると、既定のエラー コンテキストとしてParsing | Commitが適用されます。EndEdit() - DataGridView.EndEdit Method (System.Windows.Forms) | Microsoft Learn

EditModeがEditOnEnterならば、CommitEdit()を呼び出すことと同じです。EndEdit - DataGridViewMethods.cs

Sort()

列を指定して、昇順または降順に並べ替えられます。

public virtual void Sort (
    System.Windows.Forms.DataGridViewColumn dataGridViewColumn,
    System.ComponentModel.ListSortDirection direction
    );
Sort(DataGridViewColumn, ListSortDirection) - DataGridView.Sort メソッド (System.Windows.Forms) | Microsoft Learn

このメソッドは存在している行を並べ替えるものであり、これの呼び出し後に追加した行には適用されません。

個々のDataGridViewCell.Valueに異なる型のインスタンスを設定している場合、既定の方法で並べ替えるとComparer.Compare()でArgumentException例外が発生することがあります。

並べ替えに対応しているDataSourceが設定されているならば、その機能を使用しても並べ替えられます。そのようなときにこのメソッドを用いるには制約があります。

  • DataSourceが並べ替えに非対応だと「並び替えをサポートしていない IBindingList にバインドされた DataGridView コントロールを並べ替えることはできません。」としてInvalidOperationExceptionが投げられる。
  • バインドしていない列では並べ替えられない。並べ替えようとすると「データバインドされた DataGridView コントロールは、データバインドされた列でのみ並べ替えることができます。」としてArgumentExceptionが投げられる。このような場合には並べ替え用の列を追加して、それをバインドする。そしてその列に、対象の列を表示したい順に値を設定する。Programmatic Sorting - Column Sort Modes in the Windows Forms DataGridView Control | Microsoft Learn
  • 比較子を指定するオーバーロードSort(IComparer)は使用できない。呼び出すとInvalidOperationExceptionで失敗する。

DataGridViewColumn.SortModeは既定でAutomaticとなっており、ユーザーは任意の列ヘッダーをクリックすることで並べ替えできます。

並べ替えの解除

DataSourceでバインドしているならば、それの並べ替えを無効にすることで解除できます。.NET / WinForms - Clear a Sort on a DataGridView - Stack Overflow

DataTableにバインドしている場合
DataTable table = (DataTable)dataGridView.DataSource;
DataView view = table.DefaultView;
view.Sort = "";
BindingSourceにバインドしている場合
BindingSource source = (BindingSource)dataGridView.DataSource;
source.RemoveSort(); // source.Sort = null; としても同じ

複数の列で並べ替える

Sort()では1つの列しか指定できないため、複数の列の値を考慮して並べ替えるにはSort(IComparer)で並べ替え方法を指定するか、DataSourceにバインドしているならばそれが備える並べ替えの機能を用います。c# - how to sort a datagridview by 2 columns - Stack Overflow

選択を維持して並べ替える

Sort()を呼ぶか既定の方法で並べ替えられると、現在アクティブな行と列のインデックスが維持され、アクティブなセルと選択は失われます。これを選択を維持するようにするにはSortModeをProgrammaticとして既定の並べ替えを無効にして、並べ替え前にその情報を保存しておき、並べ替え後に選択します。

次の例ではDataSourceが設定され、SelectionModeがFullRowSelectであると仮定しています。

private void dataGridView_ColumnHeaderMouseClick(object sender, DataGridViewCellMouseEventArgs e)
{
    // 現在のセルを保持
    int prevColumnIndex = dataGridView.CurrentCell.ColumnIndex;
    object prevItem = dataGridView.CurrentCell.OwningRow.DataBoundItem;

    // 選択を保持
    HashSet<object> selectedItems = new HashSet<object>(dataGridView.SelectedRows.Count);
    foreach (DataGridViewRow row in dataGridView.SelectedRows)
    {
        selectedItems.Add(row.DataBoundItem);
    }


    ListSortDirection direction;
    if (dataGridView.SortedColumn != null && dataGridView.SortedColumn.Index == e.ColumnIndex
    && dataGridView.SortOrder == SortOrder.Ascending)
    {
        direction = ListSortDirection.Descending;
    }
    else
    {
        direction = ListSortDirection.Ascending;
    }

    // 並べ替える
    dataGridView.Sort(dataGridView.Columns[e.ColumnIndex], direction);

    foreach (DataGridViewRow row in dataGridView.Rows)
    {
        // 現在のセルを復元
        if (object.ReferenceEquals(prevItem, row.DataBoundItem))
        {
            dataGridView.CurrentCell = row.Cells[prevColumnIndex];
        }

        // 選択を復元
        if (selectedItems.Remove(row.DataBoundItem))
        {
            row.Selected = true;
        }
    }
}

ProcessDataGridViewKey()

移動に使用されるキーを独自に処理することで、既定の処理を変更できます。

protected override bool ProcessDataGridViewKey(KeyEventArgs e)
{
    switch(e.KeyData)
    {
        case Keys.Tab: // Tabキー
            return base.ProcessDownKey(e.KeyData); // ↓キーの処理を実行する
        default:
            return base.ProcessDataGridViewKey(e);
    }
}

このメソッドは、キー入力によって現在のセルが移動するときに呼び出されます。たとえば編集モードでキーが押されたとき、テキスト間でキャレットが移動するだけならば呼ばれませんが、テキストの末尾で押されると次のセルへの移動となるため、そのタイミングで呼ばれます。

キーを処理するメソッドとしてProcessDownKey()やProcessEndKey()などが用意されていますが、これらのメソッドが共通して取るKeys列挙型はShiftやCtrlといった修飾キーの状態を調べることに使用されるだけで、それ以外のキーの指定は考慮されません。ProcessDownKey - DataGridViewMethods.cs

ProcessZeroKey()のような特定のキーを対象としたメソッドは、Ctrl+0のような既定のショートカットキーを処理するために用意されています。ProcessZeroKey - datagridview.processzero

ショートカットキー

キー入力に対する既定の処理は、DataGridView コントロールの既定のキーボードおよびマウス処理 - Windows Forms .NET Framework | Microsoft Learnで一覧できます。

ProcessDialogKey()

Shift+Enterなど一部のキーは編集モードであるか否かによってProcessDataGridViewKey()やProcessDialogKey()で処理されます。これをそのモードにかかわらず処理するにはProcessCmdKey()を用います。

HitTest()

座標に対する列や行のインデックスの情報を得られます。

public System.Windows.Forms.DataGridView.HitTestInfo HitTest (
    int x,
    int y
    );
DataGridView.HitTest(Int32, Int32) メソッド (System.Windows.Forms) | Microsoft Learn
private void dataGridView_MouseDown(object sender, MouseEventArgs e)
{
    DataGridView.HitTestInfo hitTestInfo = dataGridView.HitTest(e.X, e.Y);
    if (hitTestInfo.Type == DataGridViewHitTestType.Cell)
    {
        Debug.Write("Row:{0} Column:{1}",
            hitTestInfo.RowIndex,
            hitTestInfo.ColumnIndex);
    }
}

DataGridView内の種類を表すHitTestInfo.Typeプロパティからは基本的な情報しか得られませんが、より詳細な情報はinternalな型であるDataGridViewHitTestTypeInternalから得られます。c# - How to detect if a MouseDown is over Row Resize Area - Stack Overflow

internal enum DataGridViewHitTestTypeInternal
{
    None,                      // 0
    Cell,                      // 1
    ColumnHeader,              // 2
    RowHeader,                 // 3
    ColumnResizeLeft,          // 4
    ColumnResizeRight,         // 5
    RowResizeTop,              // 6
    RowResizeBottom,           // 7
    FirstColumnHeaderLeft,     // 8
    TopLeftHeader,             // 9
    TopLeftHeaderResizeLeft,   // 10
    TopLeftHeaderResizeRight,  // 11
    TopLeftHeaderResizeTop,    // 12
    TopLeftHeaderResizeBottom, // 13
    ColumnHeadersResizeBottom, // 14
    ColumnHeadersResizeTop,    // 15
    RowHeadersResizeRight,     // 16
    RowHeadersResizeLeft,      // 17
    ColumnHeaderLeft,          // 18
    ColumnHeaderRight          // 19
}

private void dataGridView_MouseDown(object sender, MouseEventArgs e)
{
    HitTestInfo hitTestInfo = dataGridView.HitTest(e.X, e.Y);

    System.Reflection.FieldInfo fieldInfo = typeof(HitTestInfo).GetField("typeInternal", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
    DataGridViewHitTestTypeInternal hitTestType = (DataGridViewHitTestTypeInternal)fieldInfo.GetValue(hitTestInfo);
}
DataGridViewHitTestTypeInternal - DataGridView.cs HitTestInfo - DataGridViewHitTestInfo.cs

SetCurrentCellAddressCore()

アクティブなセルを設定できます。CurrentCellへの設定とは異なり、このセルが表示されるようにスクロールしたり、他の選択が解除されたりしません。

protected virtual bool SetCurrentCellAddressCore (
    int columnIndex,
    int rowIndex,
    bool setAnchorCellAddress, // Shiftキーで複数選択するためのアンカー セル (anchor cell) とするならば、true
    bool validateCurrentCell,  // 以前のCurrentCellの値の検証に失敗したならば変更を取り消すならば、true
    bool throughMouseClick     // マウス クリックの結果として設定するならば、true
    );
DataGridView.SetCurrentCellAddressCore メソッド (System.Windows.Forms) | Microsoft Learn

CurrentCellを設定するときは、内部ではこのメソッドによりアクティブなセルが設定されます。

ClearSelection(value.ColumnIndex, value.RowIndex, true /*selectExceptionElement*/);
if (!SetCurrentCellAddressCore(value.ColumnIndex, value.RowIndex, true, false, false))
{
    throw new InvalidOperationException(SR.GetString(SR.DataGridView_CellChangeCannotBeCommittedOrAborted));
}
CurrentCell - DataGridView.cs

設定に成功するとtrueが返されます。

SetSelectedCellCore()

セルの選択状態を設定できます。

protected virtual void SetSelectedCellCore (
    int columnIndex,
    int rowIndex,
    bool selected // 選択するならtrue、解除するならfalse
    );
DataGridView.SetSelectedCellCore(Int32, Int32, Boolean) メソッド (System.Windows.Forms) | Microsoft Learn

選択するときには、SelectionModeやMultiSelectは考慮されません。 Remarks - DataGridView.SetSelectedCellCore(Int32, Int32, Boolean) Method (System.Windows.Forms) | Microsoft Learn SetSelectedCellCore - DataGridViewMethods.cs

DataGridViewCell.Selectedプロパティに設定するときは、内部でこのメソッドが呼ばれています。Selected - DataGridViewCell.cs

セルではなく列を対象とするならばSetSelectedColumnCore()、行ならばSetSelectedRowCore()を用います。

選択したときに他の選択を解除したいならば、第3引数をtrueとしてClearSelection(Int32, Int32, Boolean)を呼びます。SelectionModeがFullRowSelectやFullColumnSelectのときは列や行の指定は意味をなしませんが、範囲外の値を指定するとArgumentOutOfRangeExceptionが投げられます。

GetCellCount()

条件に合うセルの数を得られます。

public int GetCellCount (System.Windows.Forms.DataGridViewElementStates includeFilter);
DataGridView.GetCellCount(DataGridViewElementStates) メソッド (System.Windows.Forms) | Microsoft Learn

条件のincludeFilterには、DataGridViewElementStatesの値を指定できます。

セルではなく行や列の数は、

で得られます。

選択されているセルの数

選択されているセルの数を効率的に取得するには、このメソッドを用います。選択したセル、行、列のコレクションの効率的な使用 - DataGridView コントロールの拡大縮小に関するベスト プラクティス - Windows Forms .NET Framework | Microsoft Learn

つまりdataGridView.SelectedCells.Countではなく、dataGridView.GetCellCount(DataGridViewElementStates.Selected)とします。しかしこれはセルの数を得るためだけにSelectedCellsを介すのが非効率なだけであり、選択されているセルの情報も必要ならばSelectedCellsをローカルで保持しておき、それのCountプロパティから数を得るようにします。 SelectedCells - DataGridView.cs Count - DataGridViewSelectedCellCollection.cs GetCellCount - DataGridViewMethods.cs

なおすべてのセルが選択されているかどうかは、AreAllCellsSelected()で確認できます。

表示されている行や列の数

スクリーンに表示されている行の数はdataGridView.Rows.GetRowCount(DataGridViewElementStates.Displayed)としても得られますが、それ専用のdataGridView.DisplayedRowCount(true)の方が効率的です。同様に列はDisplayedColumnCount()で得られます。

InvalidateCell()

指定のセルが範囲外だと「指定された引数は、有効な値の範囲内にありません。(Specified argument out of the range of valid values.)」として、ArgumentOutOfRangeExceptionが投げられます。

ISupportInitialize.BeginInit()

初期化が開始されていることを、オブジェクトに通知できます。これは次のように実装されています。

void ISupportInitialize.BeginInit()
{
    if (this.dataGridViewState2[DATAGRIDVIEWSTATE2_initializing])
    {
        throw new InvalidOperationException(SR.GetString(SR.DataGridViewBeginInit));
    }

    this.dataGridViewState2[DATAGRIDVIEWSTATE2_initializing] = true;
}
ISupportInitialize.BeginInit() - DataGridView.cs

ISupportInitialize.BeginInit メソッド (System.ComponentModel) | Microsoft Learn

幅や高さの自動調整

対象 メソッド 機能 対応する自動調整のプロパティ
高さ AutoResizeColumnHeadersHeight() 列ヘッダーの高さを調整できる ColumnHeadersHeightSizeMode
AutoResizeColumn(int) 指定の列の幅を調整できる AutoSizeColumnsMode
AutoResizeColumns() すべての列の幅を調整できる

ただしAutoSizeModeがDataGridViewAutoSizeColumnMode.Noneに設定されている列は調整されない

AutoResizeRowHeadersWidth(DataGridViewRowHeadersWidthSizeMode) 行ヘッダーの幅を調整できる RowHeadersWidthSizeMode
高さ AutoResizeRow(int) 指定の行の高さを調整できる AutoSizeRowsMode
AutoResizeRows() すべての行の高さを調整できる

サイズ変更モードを省略すると、DataGridViewAutoSizeRowsMode.AllCellsが適用される AutoResizeRows - DataGridViewMethods.cs

AutoResizeColumnHeadersHeight()以外は、調整方法をそれぞれ異なる列挙型で指定します。

高さを調整するときWrapModeがTrueとなっていると、折り返されたテキストがすべて表示されるように高さが大きくなることがあります。

イベント

セルの内容
イベント 発生タイミング
EventHandler DataSourceChanged DataSourceプロパティの値が変更されたとき
DataGridViewDataErrorEventHandler DataError データの解析や検証で例外が投げられたとき
DataGridViewCellValidatingEventHandler CellValidating セルがフォーカスを失い、検証できるようになったとき
DataGridViewCellEventHandler CellValidated セルの検証が終わった後
DataGridViewCellCancelEventHandler RowValidating 行が検証されているとき
DataGridViewCellEventHandler RowValidated 行の検証が終わった後
DataGridViewCellEventHandler CellValueChanged セルの値が変更され、フォーカスが失われたとき。同じ値が設定された場合には発生しない
DataGridViewCellParsingEventHandler CellParsing セルの値が変更され、編集モードを離れたとき。同じ値が設定され、実際には変更されていない場合にも発生する
DataGridViewCellCancelEventHandler CellBeginEdit 選択されているセルの編集モードが開始するとき。これが発生した時点ではIsCurrentCellInEditModeはfalseであり、まだ編集中ではない
DataGridViewCellEventHandler CellEndEdit 選択されているセルの編集モードが停止されたとき。ESCにより編集がキャンセルされた場合にも発生する
DataGridViewRowEventHandler DefaultValuesNeeded ユーザーが新しい行を追加したとき
DataGridViewRowsAddedEventHandler RowsAdded 新しい行が追加された後
DataGridViewRowsRemovedEventHandler RowsRemoved 1つまたは複数の行が削除されたとき
フォーカス
イベント 発生タイミング
DataGridViewCellEventHandler RowEnter 行が入力フォーカスを得て、CurrentRowが更新される前

新しい行の情報は、DataGridViewCellEventArgs.RowIndexから得られる

DataGridViewCellEventHandler CellEnter 現在のセルが変更されたとき、またはコントロールがフォーカスを得たとき
DataGridViewCellMouseEventHandler CellMouseMove マウスポインタがセルの上を移動したとき。MouseMoveと異なり、セルが表示されていない部分では発生しない
選択
イベント 発生タイミング
EventHandler SelectionChanged 現在の選択が変更されるとき。選択されているセルは、SelectedCellsプロパティから得られる

CurrentCellの値を変更するとき、このイベントはCurrentCellChangedより前に発生し、その時点ではCurrentCellはまだ変更されていない。しかしユーザーによって選択が変更されたときは、CurrentCellは変更されている Remarks - DataGridView.SelectionChanged Event (System.Windows.Forms) | Microsoft Learn

SelectionModeがFullRowSelectのとき、行内の他のセルを方向キーやクリックで選択したときはこのイベントは発生しないが、Tabで移動すると選択が変更されていないにもかかわらず発生する

EventHandler CurrentCellChanged CurrentCellプロパティが変更されるとき
DataGridViewCellEventHandler CellClick セルの任意の部分がクリックされたとき
DataGridViewCellMouseEventHandler CellMouseClick マウスでセルの任意の部分が、ユーザーによってクリックされたとき

このイベントのハンドラが受け取るDataGridViewCellMouseEventArgsはMouseEventArgsを継承するため、クリックされた位置やボタンの情報を得られる。ただしクリックの位置はセル相対のため、DataGridView内の相対位置はdataGridView.PointToClient(Control.MousePosition)で得る

このイベントはOnMouseClick()でHitTest()が呼ばれた結果、セルの部分がクリックされたと判定されたときに発生する OnMouseClick - DataGridViewMethods.cs

DataGridViewCellMouseEventHandler CellMouseDown マウス ボタンでセルの境界内が、ユーザーによって押されたとき

境界内とはヘッダーを含むセルが表示されている範囲。セルはマウス ボタンが押されたときに選択されるため、それを変更するならばこのイベントに応じる

DataGridViewCellEventHandler CellContentClick セルのコンテンツがクリックされたとき。このときコンテンツとは、DataGridViewTextBoxCellのテキスト部分などが該当する。またそれにはDataGridViewHeaderCellも含まれるため、行や列のヘッダーセルも対象となる

DataGridViewButtonCellやDataGridViewCheckBoxCellにフォーカスがあるときに、Spaceキーが押されたときにも発生する

DataGridViewCellMouseEventHandler ColumnHeaderMouseClick 列ヘッダーがクリックされたとき。左上隅のセルでは発生しない
DataGridViewCellMouseEventHandler RowHeaderMouseClick 行ヘッダーがクリックされたとき。左上隅のセルでは発生しない
幅と高さ
イベント 発生タイミング
DataGridViewColumnEventHandler ColumnWidthChanged 列のWidthプロパティの値が変更されたとき。AutoSizeColumnsModeがNone以外に設定されていると、DataSourceやDataGridViewColumn.FillWeightが変更されるたびに発生する

Widthの既定値は100であり、この値に設定しても変更されていないため、このイベントは発生しない

DataGridViewRowEventHandler RowHeightChanged 行のHeightプロパティの値が変更されたとき
DataGridViewRowHeightInfoNeededEventHandler RowHeightInfoNeeded 行の高さについての情報が要求されたとき
DataGridViewRowHeightInfoPushedEventHandler RowHeightInfoPushed ユーザーが行の高さを変更したとき 行の高さが変更されたとき
EventHandler ColumnHeadersHeightChanged ColumnHeadersHeightプロパティの値が変更されたとき
EventHandler RowHeadersWidthChanged RowHeadersWidthプロパティの値が変更されたとき
境界線
イベント 発生タイミング
DataGridViewColumnEventHandler ColumnDividerWidthChanged DividerWidthプロパティの値が変更されたとき
DataGridViewRowEventHandler RowDividerHeightChanged DividerHeightプロパティの値が変更されたとき
DataGridViewColumnDividerDoubleClickEventHandler ColumnDividerDoubleClick 列の間の境界線がダブルクリックされたとき。既定では、その列に適用されているInheritedAutoSizeModeに従い列の幅が調整される OnColumnDividerDoubleClick - DataGridViewMethods.cs

幅を明示的に調整するならばAutoResizeColumn()を呼び出して、e.Handledをtrueとする 注釈 - DataGridView.ColumnDividerDoubleClick イベント (System.Windows.Forms) | Microsoft Learn

行ヘッダーとの境界がダブルクリックされたときはe.ColumnIndexが-1となるため、AutoResizeRowHeadersWidth()を呼び出す。

DataGridViewRowDividerDoubleClickEventHandler RowDividerDoubleClick 行の間の境界線がダブルクリックされたとき
コンテキストメニュー
イベント 発生タイミング
DataGridViewCellContextMenuStripNeededEventHandler CellContextMenuStripNeeded セルのショートカットメニューが必要なとき。これのe.ContextMenuStripにnullを指定するか何も指定しなければ、ショートカットメニューの表示を抑制できる

このイベントはDataSourceが設定されているか、VirtualModeがtrueでなければ発生しない

DataGridViewRowContextMenuStripNeededEventHandler RowContextMenuStripNeeded 行のショートカットメニューが必要なとき

このイベントはDataSourceが設定されているか、VirtualModeがtrueでなければ発生しない

表示
イベント 発生タイミング
DataGridViewCellFormattingEventHandler CellFormatting セルの内容が、表示用に書式設定されなければならないとき
DataGridViewCellErrorTextNeededEventHandler CellErrorTextNeeded セルに関連付けられたエラー条件を表すテキストが必要なとき

このイベントはDataSourceが設定されているか、VirtualModeがtrueでなければ発生しない

DataGridViewRowErrorTextNeededEventHandler RowErrorTextNeeded 行のエラー テキストが必要なとき
DataGridViewCellToolTipTextNeededEventHandler CellToolTipTextNeeded セルのツールチップが必要なとき
DataGridViewRowPrePaintEventHandler RowPrePaint DataGridViewRowが描画される前

セルがクリックされたとき、MouseDownとMouseUpのタイミングで2回発生する。これはRowPostPaintも同様

DataGridViewRowPostPaintEventHandler RowPostPaint DataGridViewRowが描画された後
DataGridViewCellPaintingEventHandler CellPainting セルが描画される必要があるとき
     
並べ替え
イベント 発生タイミング
EventHandler Sorted 並べ替え操作が完了したとき
DataGridViewSortCompareEventHandler SortCompare 並べ替え操作を実行するのに2つのセルの値を比較するとき
イベント - DataGridView Class (System.Windows.Forms) | Microsoft Learn

DataError

データの解析や検証で例外が投げられたときに発生し、これを捕捉しなければ[DataGridView の既定のエラー ダイアログ]が表示されます。 DataGridView.DataError イベント (System.Windows.Forms) | Microsoft Learn Walkthrough: Handling errors that occur during data entry in DataGridView control - Windows Forms | Microsoft Learn

エラーの原因は、ハンドラに渡されるDataGridViewDataErrorEventArgs.Contextで得られます。

CellValidating

DataGridView.CellValidating イベント (System.Windows.Forms) | Microsoft Learn

ハンドラの引数で渡されるDataGridViewCellValidatingEventArgsはCancelEventArgsを継承しており、検証に失敗したときはそれのCancelプロパティをtrueとすることで、確定を取り消し編集モードに戻せます。ただしこのイベントはこのコントロールが配置されたFormが閉じられるときにもセルのフォーカスが失われることで発生し、そこでCancelをtrueとするとForm.FormClosingのFormClosingEventArgs.Cancelもtrueとなり、フォームが閉じられなくなります。この問題に対処するにはフォームのAutoValidateをAutoValidate.Disableにして、フォームのフォーカスが失われたときに検証が発生しないようにします。

またはフォームが閉じられるときにCausesValidationをfalseにして、検証を無効にします。

protected override void WndProc(ref Message m)
{
    const int WM_CLOSE = 0x0010;
    if (m.Msg == WM_CLOSE)
    {
        dataGridView.CausesValidation = false;
    }

    base.WndProc(ref m);
}
c# - DataGridView CellValidated Event triggers when closing form - Stack Overflow

RowHeightInfoPushed

このイベントは「Note that this event occurs only when the user changes the height of a row.」としてユーザーが変更したときのみ発生するとされていますが、Remarks - DataGridView.RowHeightInfoPushed Event (System.Windows.Forms) | Microsoft Learn

実際にはAutoResizeRowInternal()でAutoSizeRowsModeがNoneのときにも発生するため、これを呼び出すAutoResizeRows()などからも発生します。

CellValueChanged

変更されたセルは、ハンドラに渡されるDataGridViewCellEventArgsから、

DataGridViewCell cell = dataGridView[e.RowIndex, e.ColumnIndex];

として取得できます。そのとき行が存在していないとe.RowIndexは-1となり、HeaderCellの値が変更されたときはe.ColumnIndexは-1となります。

データソースの変更によりセルの値が変更されたときは、このイベントは発生しません。

CellParsing

ハンドラを抜けるときに、

  • ConvertEventArgs.Valueが、null
  • ConvertEventArgs.Valueの型が、不正確
  • DataGridViewCellParsingEventArgs.ParsingAppliedが、false

ならば、既定の型変換を用いてParseFormattedValue()で解析されます。

CellFormatting

ハンドラで引数のDataGridViewCellFormattingEventArgs.Valueに値を設定したならば、同じく引数のFormattingAppliedをtrueとしてこれ以上の書式設定が不要であることを伝えます。それをfalseのままとすると、引数のCellStyleにより書式が適用されます。

FormattingAppliedをtrueとしたときにDesiredTypeと異なる型が設定されていると、「セルのフォーマットされた値に間違った型が指定されています。(Formatted value of the cell has a wrong type.)」としてFormatExceptionが投げられます。そのときDataErrorイベントにハンドラを登録していないと、既定のエラーダイアログで通知されます。

RowPrePaint

描画に関するイベントは、次の順で呼ばれます。

  1. RowPrePaint (列ヘッダーでは呼ばれない)
  2. CellPainting
  3. RowPostPaint (列ヘッダーでは呼ばれない)
  4. Paint

複数の列がある場合には、CellPaintingはその数だけくり返し呼ばれます。複数の行がある場合には、RowPrePaint、CellPainting、RowPostPaintの順でくり返し呼ばれます。そしてすべての行のイベントの後に、Paintが呼ばれます。

ハンドラの引数のDataGridViewRowPrePaintEventArgs.PaintPartsで描画するセルの部分を指定すると、その値がCellPaintingに渡されます。

private void dataGridView_RowPrePaint(object sender, DataGridViewRowPrePaintEventArgs e)
{
    e.PaintParts &= ~DataGridViewPaintParts.ContentForeground;
}

RowPostPaint

描画時にはHorizontalScrollingOffsetを考慮しないと、水平スクロールされたときに描画位置がずれます。

private void dataGridView_RowPostPaint(object sender, DataGridViewRowPostPaintEventArgs e)
{
    Rectangle rowBounds = e.RowBounds;
    rowBounds.X = dataGridView.RowHeadersWidth; // 行ヘッダーの領域を除外する
    rowBounds.Width = dataGridView.Columns.GetColumnsWidth(DataGridViewElementStates.Visible); // 表示可能な領域の幅を設定する

    RectangleF clip = e.Graphics.ClipBounds; // 現在のクリッピング領域を保持する

    // 描画する範囲のクリッピング領域を制限する
    e.Graphics.SetClip(rowBounds);

    Rectangle textArea = rowBounds;
    textArea.X -= dataGridView.HorizontalScrollingOffset; // 水平方向にスクロールされている大きさ分、描画領域を移動する

    String text = e.RowIndex.ToString();
    DataGridViewCellStyle style = e.InheritedRowStyle;

    // クリッピング領域が適用されるようにPreserveGraphicsClippingを指定する
    TextFormatFlags flags = TextFormatFlags.Left | TextFormatFlags.PreserveGraphicsClipping;
    TextRenderer.DrawText(e.Graphics, text, style.Font, textArea, style.ForeColor, flags);

    // クリッピング領域を元に戻す
    e.Graphics.SetClip(clip);
}

CellPainting

セルの描画にオーナー描画が必要なときに呼ばれます。セルに値が設定されておらず、描画の必要がないときは呼ばれません。

private void dataGridView_CellPainting(object sender, DataGridViewCellPaintingEventArgs e)
{
    if (e.ColumnIndex == -1 || e.RowIndex == -1)
    {
        // ヘッダーセルの描画
    }
    else
    {
        // 前景以外を描画する
        e.Paint(e.CellBounds, e.PaintParts & ~DataGridViewPaintParts.ContentForeground);

        if (e.Value != null)
        {
            // セルのテキストを赤で描画する
            string text = (string)e.Value;
            e.Graphics.DrawString(text, e.CellStyle.Font, Brushes.Red, e.CellBounds);
        }

        e.Handled = true;
    }
}
DataGridViewCellPaintingEventArgsのメソッド
メソッド 機能
Paint(Rectangle, DataGridViewPaintParts) 指定部分を描画できる
PaintContent(Rectangle) ContentBackground | ContentForeground | ErrorIcon をDataGridViewPaintPartsの引数にして、Paint()を呼び出すのに等しい PaintContent - DataGridViewCellPaintingEventArgs.cs
PaintBackground(Rectangle, Boolean) Background | Border、選択されているならばSelectionBackgroundも追加して、Paint()を呼び出すのに等しい PaintBackground - DataGridViewCellPaintingEventArgs.cs
   

DataGridViewPaintParts列挙型

列挙子 内容
None 0 何も描画しない
Background 1 セルの背景を描画する
Border 2 セルの境界線を描画する
ContentBackground 4 セルの内容の背景を描画する (現在の行や並べ替えを示すグリフが含まれる)
ContentForeground 8 セルの内容の前景を描画する
ErrorIcon 16 セルのエラー アイコンを描画する
Focus 32 セルの周囲にフォーカスを示す四角形を描画する
SelectionBackground 64 セルが選択されたときに、セルの背景を描画する
All 127 セルのすべての部分を描画する
フィールド - DataGridViewPaintParts 列挙型 (System.Windows.Forms) | Microsoft Learn

CellMouseEnter

マウス ポインタがセルに入ったときに発生します。DataGridView.CellMouseEnter イベント (System.Windows.Forms) | Microsoft Learn

ヘッダーセル

ヘッダーセルにマウス ポインターが入ると、既定でそれがハイライトされます。

  クリックしたときの既定の動作 ハイライトを無効にする方法
行ヘッダー クリックした行が選択される SelectionModeをFullRowSelectとRowHeaderSelect以外にして、行が選択されないようにする
列ヘッダー クリックした列で並べ替えられる DataGridViewColumn.SortModeをNotSortableとして、並べ替えを無効にする
左上隅のヘッダー すべてのセルが選択される CellPaintingイベントで、セルの背景を描画しない

左上隅のヘッダーをクリックすると既定ではすべてのセルが選択されますが、MultiSelectをfalseとしてセルを複数選択できないようにすると、クリックしても何も起きません。このときクリックに応答しないことを示すには、CellPaintingイベントでセルの背景をつねに同色で描画するようにして、セルにマウス ポインターが入ってもハイライトしないようにします。

private void dataGridView_CellPainting(object sender, DataGridViewCellPaintingEventArgs e)
{
    if (e.ColumnIndex == -1 && e.RowIndex == -1)
    {
        using (Brush brush = new SolidBrush(dataGridView.ColumnHeadersDefaultCellStyle.BackColor))
        {
            e.Graphics.FillRectangle(brush, e.ClipBounds);
        }

        e.Paint(e.ClipBounds, e.PaintParts & ~DataGridViewPaintParts.Background);
        e.Handled = true;
    }
}

RowsAdded

追加された行を個別に特定するには、RowIndexを基点としてRowCountの数だけループさせます。c# - DataGridView row added event - Stack Overflow

private void dataGridView_RowsAdded(object sender, DataGridViewRowsAddedEventArgs e)
{
    for (int i = 0; i < e.RowCount; i++)
    {
        DataGridViewRow row = dataGridView.Rows[e.RowIndex + i];
    }
}

EditingControlShowing

セルの編集用のコントロールが表示されるときに発生します。このイベントを処理することで、セルが編集モードに入るときに編集用のコントロール (EditingControl) を操作できます。DataGridViewでセルの編集に使われているテキストボックスを取得する - .NET Tips (VB.NET,C#...)

DataGridViewは編集と編集の間にセルの型が変更されなければ、編集用のコントロールを再利用します。そのためハンドラでControlを操作するときは、イベントのたびにControlに必要な設定をします。Remarks - DataGridView.EditingControlShowing Event (System.Windows.Forms) | Microsoft Learn

DPIの相違による問題

Per-Monitor V2 によってディスプレイによってDPIが異なるとき、TextBoxにAutoCompleteを設定しそのコントロールが他のセルで再利用されると、「ウィンドウのハンドルを作成中にエラーが発生しました。」としてWin32Exceptionが投げられます。 c# - Winforms application throws Win32Exception if marked as DPI aware and used with multiple monitors - Stack Overflow

private void dataGridView_EditingControlShowing(object sender, DataGridViewEditingControlShowingEventArgs e)
{
    TextBox textBox = e.Control as TextBox;
    if (textBox.AutoCompleteMode == AutoCompleteMode.None)
    {
        textBox.AutoCompleteMode = AutoCompleteMode.Suggest;
        textBox.AutoCompleteSource = AutoCompleteSource.CustomSource;
        textBox.AutoCompleteCustomSource.AddRange(new[] { "a1", "a2", "a3" });
    }

    if (dataGridView.DeviceDpi != textBox.DeviceDpi)
    {
        // このようにDPIが異なる状態でtextBoxが表示されるとき、そのCreateHandle()でWin32Exceptionが投げられる
    }
}

この問題はコントロールが再利用されないように、編集が終了したときにそれを破棄することで解決できます。c# - DataGridViewComboBoxColumn - value lags for a split second - Stack Overflow

private void dataGridView_CellEndEdit(object sender, DataGridViewCellEventArgs e)
{
    if (dataGridView.DeviceDpi != dataGridView.EditingPanel.DeviceDpi)
    {
        FieldInfo latestEditingControl = typeof(DataGridView).GetField("latestEditingControl", BindingFlags.NonPublic | BindingFlags.Instance);

        ((Control)latestEditingControl.GetValue(dataGridView)).Dispose();
        latestEditingControl.SetValue(dataGridView, null);
    }
}
Microsoft Learnから検索