Taher
Taher

Reputation: 593

Drag and Drop item from DataGridView to a panel

I have searched for a solution for drag and drop in DataGridView but all I found was about reordering the items.

But what I have is a DataGridView bound to a SQL server database using LINQ to Entities and reordering is not applicable, all I want was to drag an item from the DataGridView and drop it on a panel in the same form. How can I do that ?

The code that I found that allows reordering:

    private Rectangle dragBoxFromMouseDown;
    private int rowIndexFromMouseDown;
    private int rowIndexOfItemUnderMouseToDrop;

    private void dgvResult_MouseMove(object sender, MouseEventArgs e)
    {
        if ((e.Button & MouseButtons.Left) == MouseButtons.Left)
        {
            // If the mouse moves outside the rectangle, start the drag.
            if (dragBoxFromMouseDown != Rectangle.Empty &&
                !dragBoxFromMouseDown.Contains(e.X, e.Y))
            {

                // Proceed with the drag and drop, passing in the list item.                    
                DragDropEffects dropEffect = dgvResult.DoDragDrop(
                dgvResult.Rows[rowIndexFromMouseDown],
                DragDropEffects.Move);
            }
        }
    }

    private void dgvResult_MouseDown(object sender, MouseEventArgs e)
    {
        // Get the index of the item the mouse is below.
        rowIndexFromMouseDown = dgvResult.HitTest(e.X, e.Y).RowIndex;
        if (rowIndexFromMouseDown != -1)
        {
            // Remember the point where the mouse down occurred. 
            // The DragSize indicates the size that the mouse can move 
            // before a drag event should be started.                
            Size dragSize = SystemInformation.DragSize;

            // Create a rectangle using the DragSize, with the mouse position being
            // at the center of the rectangle.
            dragBoxFromMouseDown = new Rectangle(new Point(e.X - (dragSize.Width / 2),
                                                           e.Y - (dragSize.Height / 2)),
                                dragSize);
        }
        else
            // Reset the rectangle if the mouse is not over an item in the ListBox.
            dragBoxFromMouseDown = Rectangle.Empty;
    }

    private void dgvResult_DragOver(object sender, DragEventArgs e)
    {
        e.Effect = DragDropEffects.Move;
    }

    private void dgvResult_DragDrop(object sender, DragEventArgs e)
    {
        // The mouse locations are relative to the screen, so they must be 
        // converted to client coordinates.
        Point clientPoint = dgvResult.PointToClient(new Point(e.X, e.Y));

        // Get the row index of the item the mouse is below. 
        rowIndexOfItemUnderMouseToDrop =
            dgvResult.HitTest(clientPoint.X, clientPoint.Y).RowIndex;

        // If the drag operation was a move then remove and insert the row.
        if (e.Effect == DragDropEffects.Move)
        {
            DataGridViewRow rowToMove = e.Data.GetData(
                typeof(DataGridViewRow)) as DataGridViewRow;
            dgvResult.Rows.RemoveAt(rowIndexFromMouseDown);
            dgvResult.Rows.Insert(rowIndexOfItemUnderMouseToDrop, rowToMove);

        }
    }

And any attempt for drag and drop inside the DataGridView throws an exception:

Rows cannot be programmatically removed unless the DataGridView is data-bound to an IBindingList that supports change notification and allows deletion.

Upvotes: 1

Views: 1581

Answers (1)

TaW
TaW

Reputation: 54453

You didn't explain what you want to do with the Row in the Panel, so I made up something..

The first thing you do is do set the Panel's AllowDrop = true. Then you script the DragEnter and the DragDrop events:

private void panel1_DragEnter(object sender, DragEventArgs e)
{
    // we indicate that we allow dropping
    e.Effect = DragDropEffects.Copy;
}

private void panel1_DragDrop(object sender, DragEventArgs e)
{
    // the mousemove has packed a row as data, so we unpack it:
    DataGridViewRow row = e.Data.GetData( typeof(DataGridViewRow)) as DataGridViewRow;
    // not sure what to do with it - store it in the tag
    panel1.Tag = row;
    // let the paint show something
    panel1.Invalidate();
}

private void panel1_Paint(object sender, PaintEventArgs e)
{
    // if we hava data in the tag we cast them to DataGridViewRow..
    // ..and display the content of the 2nd cell
    if (panel1.Tag != null)
        e.Graphics.DrawString( ((DataGridViewRow)panel1.Tag).Cells[1].Value.ToString(), 
                                this.Font, Brushes.DarkRed, 8, 8);
}

Note that you can delete the dgvResult_DragDrop & dgvResult_DragOver events, as we don't want to drop on the DGV.

Of course the paint action is pretty much random..

Upvotes: 2

Related Questions