React Tree Grid の行ドラッグ

    React Tree Grid の Ignite UI for React 行ドラッグ機能は簡単に構成でき、マウスを使用して行を新しい位置にドラッグ アンド ドロップすることで、グリッド内の行を再配置するために使用されます。これはルートの IgrTreeGrid コンポーネントで初期化され、rowDraggable 入力を介して構成できます。

    React Tree Grid 行ドラッグの例

    構成

    IgrTreeGrid の行ドラッグを有効にするには、グリッドの rowDraggabletrue に設定します。これが有効になると、行ドラッグ ハンドルが各行に表示されます。このハンドルは行ドラッグを開始するために使用できます。ドラッグ ハンドルをクリックしてボタンを押しながらカーソルを動かすと、グリッドの RowDragStart イベントが発生します。クリックをリリースすると、RowDragEnd イベントが発生します。

    <IgrTreeGrid rowDraggable={true}>
    </IgrTreeGrid>
    

    ドラッグ アイコンのテンプレート化

    ドラッグ ハンドル アイコンは、グリッドの dragIndicatorIconTemplate を使用してテンプレート化できます。作成している例で、アイコンをデフォルトのもの (drag_indicator) から drag_handle に変更します。

        const dragIndicatorIconTemplate = (ctx: IgrGridEmptyTemplateContext) => {
            return (
                <>
                    <IgrIcon name="drag_handle" collection="material" />
                </>
            );
        }
    
        <IgrTreeGrid rowDraggable={true} dragIndicatorIconTemplate={dragIndicatorIconTemplate}>
        </IgrTreeGrid>
    

    新しいアイコン テンプレートの設定後、DragIcon enumDEFAULT アイコンも調整する必要があるため、ChangeIcon メソッドによって適切に変更されます。

    enum DragIcon {
        DEFAULT = "drag_handle",
    }
    

    アプリケーション デモ

    行の並べ替えデモ

    グリッドの行ドラッグ イベントを使用して、ドラッグよる行の並べ替えるが可能なグリッドを作成できます。

    <IgrTreeGrid rowDraggable={true} primaryKey="ID" onRowDragStart={webTreeGridReorderRowStartHandler} onRowDragEnd={webTreeGridReorderRowStartHandler}>
    </IgrTreeGrid>
    

    [!Note] グリッドに primaryKey が指定されていることを確認してください。ロジックが行を適切に並べ替えられるように、行には一意の識別子が必要です。

    rowDraggable が有効になり、ドロップ エリアが定義されたら、ドロップ イベントの単純なハンドラーを実装する必要があります。行をドラッグするときは、以下を確認してください:

    • 行が展開されていますか? そうであれば、行を縮小します。
    • 行はグリッド内にドロップされましたか?
    • そうであれば、ドラッグされた行が他のどの行にドロップされましたか?
    • ターゲット行が見つかれば、data 配列内のレコードの位置を入れ替えます。
    • 行は最初に選択されてましたか? そうであれば、選択済みとしてマークします。

    以下では、上記の実装を示します。

    const webTreeGridReorderRowStartHandler = (args: IgrRowDragStartEventArgs) => {
            const draggedRow = args.detail.dragData;
        if (draggedRow.expanded) {
                draggedRow.expanded = false;
            }
        }
    
    const webTreeGridReorderRowHandler = (args: IgrRowDragEndEventArgs): void => {
            const ghostElement = args.detail.dragDirective.ghostElement;
            const dragElementPos = ghostElement.getBoundingClientRect();
            const rows = Array.prototype.slice.call(document.getElementsByTagName("igx-tree-grid-row"));
        const currRowIndex = getCurrentRowIndex(rows,
            { x: dragElementPos.x, y: dragElementPos.y });
            if (currRowIndex === -1) { return; }
            const draggedRow = args.detail.dragData.data;
        const childRows = findChildRows(treeGridRef.current.data, draggedRow);
            //remove the row that was dragged and place it onto its new location
        treeGridRef.current.deleteRow(args.detail.dragData.key);
        treeGridRef.current.data.splice(currRowIndex, 0, args.detail.dragData.data);
            // reinsert the child rows
            childRows.reverse().forEach(childRow => {
            treeGridRef.current.data.splice(currRowIndex + 1, 0, childRow);
            });
        }
    
    const findChildRows = (rows: any[], parent: any): any[] => {
            const childRows: any[] = [];
            rows.forEach(row => {
                if (row.ParentID === parent.ID) {
                    childRows.push(row);
                    // Recursively find children of current row
                const grandchildren = findChildRows(rows, row);
                    childRows.push(...grandchildren);
                }
            });
            return childRows;
        }
    
    const getCurrentRowIndex = (rowList: any[], cursorPosition: any) => {
            for (const row of rowList) {
                const rowRect = row.getBoundingClientRect();
                if (cursorPosition.y > rowRect.top + window.scrollY && cursorPosition.y < rowRect.bottom + window.scrollY &&
                    cursorPosition.x > rowRect.left + window.scrollX && cursorPosition.x < rowRect.right + window.scrollX) {
                    // return the index of the targeted row
                    return parseInt(row.attributes["data-rowindex"].value);
                }
            }
            return -1;
        }
    

    これらの簡単な手順で、ドラッグ/ドロップで行を並べ替えることができるグリッドを構成しました! 次のデモで、上記コードの動作を確認できます。

    行の選択も有効で、ドラッグした行をドロップしても選択が保持されます。

    制限

    現在、rowDraggable に既知の制限はありません。

    API リファレンス

    その他のリソース

    コミュニティに参加して新しいアイデアをご提案ください。