{lang: ‘ru’}
.NET Framework позволяет легко обнаруживать перетаскиваемые объекты в/из приложения Windows Forms. Для этого можно использовать одно или несколько из доступных событий drag and drop. В обработчиках этих событий можно проверить, является ли объект файлом.
Для включения отслеживания событий drag and drops, нужно установить свойство AllowDrop в true и использовать одно или несколько следующих событий:
-
DragEnter: Возникает при завершении операции перетаскивания
-
DragOver: Происходит, когда элемент перетаскивается с помощью мыши в клиентскую область этого элемента.
-
DragDrop: Возникает, когда объект перетаскивается за пределы элемента управления.
-
DragLeave: Возникает, когда объект перетаскивается на элемент управления.
-
GiveFeedback: Происходит, когда элемент перетаскивается с помощью мыши. Система запрашивает у элемента управления обеспечения обратной совместимости с этим эффектом.
-
QueryContinueDrag: Происходит при перетаскивании элемента. Система опрашивает, можно ли продолжать операцию перетаскивания мышью.
Чтобы обработать перетаскивание одного или нескольких файлов внутрь элемента управления, необходимо обработать два события: DragEnter и DragDrop. В DragEnter мы проверяем, что перетаскиваемый элемент имеет тип DataFormats.FileDrop. Если это так, операция перетаскивания правда операция перетаскивания разрешается. Событие DragDrop получает список файлов для помещения в элемент с помощью метода GetData и добавляет их в массив строк. Каждый элемент массива будет содержать полный путь к каждому файлу, помещенному в элемент управления.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
private void filesListBox_DragEnter(object sender, DragEventArgs e) { if (e.Data.GetDataPresent(DataFormats.FileDrop, false) == true) { e.Effect = DragDropEffects.All; } } private void filesListBox_DragDrop(object sender, DragEventArgs e) { string[] files = (string[])e.Data.GetData(DataFormats.FileDrop); foreach (string file in files) { filesListBox.Items.Add(file); } } |
Статья написана по мотивам оригинальной англоязычной версии. Исходный код с работающим примером так же доступен по ссылке из неё.
Полезная статья? Их будет больше, если вы поддержите меня!
One of the important features of Graphical User Interfaces (GUI) is drag and drop.
Using a mouse you can drag and drop a file from one location to another or you may drag a file and drop it onto an application to launch it.
However, not many developers take the effort to implement drag and drop functionality in their applications. While it does take considerable effort to implement, the support for drag and drop in your application will greatly increase its usefulness. In this article, I will show you how to implement drag and drop functionality in your Windows Forms application.
Drag and Drop Event Handlers
To understand how you can drag objects from one control onto another control, you need to become acquainted with a couple of event handlers. Consider the example of Figure 1 where you drag an image displayed in a PictureBox control to another PictureBox control.
On the control to be dragged (left PictureBox control):
The MouseDown/MouseMove event is probably a good starting point to load the data that is going to be dragged. In this case, you will copy the image stored in the left PictureBox control.
The QueryContinueDrag event allows you to know the outcome of the drag operation, i.e., whether the user eventually drops the item. If the dragging is a move operation and the user successfully executes it, you may need to remove the image on the left PictureBox control.
You can call the DoDragDrop() method either in the MouseDown or the MouseMove event.
The control constantly fires the GiveFeedBack event during the drag operation. You can handle this event if you wish to modify the appearance of the mouse pointer.
On the control to be dropped upon (right PictureBox control):
When the mouse enters the control to be dropped upon, the control fires the DragEnter event. This is usually the event that you need to service to change the mouse pointer to reflect the action it is performing (such as copy, move, etc). You can also modify the appearance of the control so that it is obvious to the user that the control is a drop target.
When the mouse hovers over the control to be dropped upon, the control fires the DragOver event. The control fires this event continuously as long as the mouse is over the target control. You can either use this event or the DragEnter event to change the appearance of the mouse pointer.
When the mouse leaves the control to be dropped upon, the control fires the DragLeave event. You usually service this event if you need to modify the appearance of the target control to reflect that it is a drop target.
When the mouse drops over the control to be dropped upon, the control fires the DragDrop event. In this event you can modify the target control so that it accepts the dropped item. In the above example, you will load the right PictureBox control with the image that the user drags from the left PictureBox control.
Note that not all controls support the entire set of events just described. For example, the RichTextBox control does not support the DragOver event. In this case, you will have to use other supported events to detect drag and drop, such as the DragEnter event.
Drag and Drop Text
I’ll now create a Windows Forms application so that you can examine how you can implement drag and drop. Using Visual Studio 2005, create a new Windows Forms application and name it DragAndDrop. For this section, I’ll show you how to drag and drop some text into a TextBox control.
For most controls, you can set the AllowDrop property in the Properties window during design time. However, some controls (such as the PictureBox control) do not expose this property in the Properties window. As such, you need to dynamically set them in code (note that IntelliSense will not show the AllowDrop property).
On the default Form1, populate it with a TextBox control and set its properties as follows (Figure 2):
Multiline-True
BorderStyle-Fixed3D
Switch to the code behind of Form1 and declare the following constant:
Public Class Form1
Const CtrlMask As Byte = 8
In the Form1_Load event, set its AllowDrop property to True:
Private Sub Form1_Load( _
ByVal sender As System.Object, _
ByVal e As System.EventArgs) _
Handles MyBase.Load
'---set the control to allow
' drop---
TextBox1.AllowDrop = True
End Sub
The next event to service is the DragEnter event. As mentioned earlier, the control fires this event when you drag something into the TextBox control. Here, you will determine if the operation is copy or move, and then set the mouse pointer accordingly. To check for this special keystroke, you use the KeyState property together with a Control Mask (defined as a byte with a value of 8). In addition, you will also change the border style of the TextBox control to FixedSingle so that it serves as a visual cue to the user (Listing 1).
You can now press F5 to test the application. Figure 3 shows the mouse pointer when a user performs a move (top of the figure) and a copy (bottom of the figure) operation on the TextBox control. To perform a copy operation, simply hold the CTRL key when dragging your mouse.
Drag and Drop Images
In the previous section you saw how to drag and drop text into a TextBox control. In this section, you will learn how to drag and drop images using a PictureBox control.
Using the same project, drag and drop a PictureBox control onto the default Form1 (Figure 5).
In the Form1_Load event, set the PictureBox control with the properties as shown below (Listing 3).
Implementing Drop
As usual, service the DragEnter event so that you can change the mouse pointer appropriately when the mouse hovers over the PictureBox control. The only difference this time is that you check if the data to be dropped onto the PictureBox control is of type image (Listing 4).
For the DragDrop event, you will first verify that the dropped object is of type image and then proceed to set the PictureBox control to display the dropped image (Listing 5).
Lastly, for the DragLeave event, change the border style of the PictureBox back to its original:
Private Sub PictureBox1_DragLeave( _
ByVal sender As Object, _
ByVal e As System.EventArgs) _
Handles PictureBox1.DragLeave
'---set the borderstyle back to
' its original---
PictureBox1.BorderStyle = _
BorderStyle.FixedSingle
End Sub
Press F5 to test the application. You can drag an image from Microsoft Word and drop it onto the PictureBox control, which will then display the image. Interestingly, dragging and dropping an image from WordPad does not cause the PictureBox control to display the image. I will explain this in a short while.
Implementing Drag
Now I’ll write some code so that you can drag the image displayed in the PictureBox control onto somewhere else, such as another control or Microsoft Word (or WordPad).
First you’ll handle the MouseDown event handler. The control fires this event when the user clicks the image in the PictureBox control:
Private Sub PictureBox1_MouseDown( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
MouseEventArgs) _
Handles PictureBox1.MouseDown
If PictureBox1.Image IsNot _
Nothing Then
PictureBox1.DoDragDrop( _
PictureBox1.Image, _
DragDropEffects.Move Or _
DragDropEffects.Copy)
End If
End Sub
Here, you use the DoDragDrop() method of the PictureBox control to copy the image displayed in the PictureBox control. The second parameter of this method indicates the type of drag operations that can occur (in this case it is either a move or copy operation).
That’s it! You can now drag and drop the image displayed in the PictureBox control to another control. Again, ironically you can drag the image onto WordPad, but not into Microsoft Word.
Let me recap what happened:
You can drag a picture from Microsoft Word and drop it onto the PictureBox control. But you cannot drag and drop from WordPad.
When you try to drag the image from the PictureBox control onto Word, it does not accept the picture. But you can drag and drop the same image onto WordPad.
To understand why an image can be dropped from Word but not from WordPad, you should modify your application so that you can observe in detail the types of data that are being passed to the PictureBox control.
In the DragEnter event, add the line in bold. The GetFormats() method returns the type of data that is passed into the event (Listing 6).
You should set a breakpoint after the line just added so that you can observe the type of data being passed into the event. Press F5 to debug the application and drop an image from Word onto the PictureBox control. Figure 6 shows the data type of the image dragged and dropped from Word.
Here you can observe that one of the formats is bitmap, and hence you can safely convert the data into a bitmap image and display it in the PictureBox control. In contrast, if you drag and drop an image from WordPad, its corresponding type for the image is shown in Figure 7.
This time round, the control does not represent the image data in the bitmap format. Instead, the control passes it in the Rich Text Format (RTF).
To ensure that the PictureBox control can accept images dragged and dropped from WordPad, you need to modify the DragEnter event (as shown in bold) (Listing 7).
In addition, you also need to modify the DragDrop event so that you can write your custom code to deal with the particular data format (Listing 8).
In my example above, I simply write the data of the image (as RTF) to the console window (Figure 8). To display the image in the PictureBox control, you will have to write your own code to extract the image data.
Move versus Copy
The previous example shows how you can copy or move an image from the PictureBox control. In a typical move operation, after copying the image to another control/location, you need to remove the original image. So how do you know when a drop operation has completed? The answer lies in the QueryContinueDrag event.
Let’s add another PictureBox control to Form1 so that Form1 now looks like Figure 9. I will demonstrate how you can move an image (by drag and drop) from PictureBox1 to PictureBox2.
Configure PictureBox2 in the Form1_Load event as follows (Listing 9).
Code PictureBox2 to allow for dropping (Listing 10).
You should remove the image in PictureBox1 after the user has moved it to PictureBox2. To do this, you need to service the QueryContinueDrag event of PictureBox1. The control fires this event when you drag and drop an image from PictureBox1. Here you determine if the operation is a move, and if it is, you will delete the image from PictureBox1 (Listing 11).
You can now test to see if it works. Press F5 in Visual Studio 2005 and drag and drop an image onto PictureBox1. Then, drag the image in PictureBox1 and drop it onto PictureBox2. You can notice that the image in PictureBox1 is now gone. In contrast, note that if you instead perform a copy operation (by holding down the CTRL key); the control merely copies the image and does not remove it.
Note that using the QueryContinueDrag event to remove an item that a user has moved is not a foolproof method. If you abort the move operation (such as dropping the image outside PictureBox2), the control will still remove the picture. In other words, using the QueryContinueDrag event, you have no idea where the user moved (or dropped) the item (or if the user aborted the operation).
Drag and Drop Files
A very common operation performed in a Windows application is dragging and dropping files onto applications. For example, you drag and drop files onto folders or perhaps you drag music files onto Windows Media player so that you can play them immediately.
Let me show you how to modify your application so that users can drag an image file from Windows Explorer and drop it onto the PictureBox control. First, modify the DragEnter event (Listing 12).
The data type for a file that is dragged from the Windows Explorer is FileDrop, and hence you check for this data type in the DragEnter event handler.
Next, modify the DragDrop event so that you can manually open up the image file and display its content in the PictureBox control (Listing 13).
That’s it! You can now drag an image file from Windows Explorer and drop it onto the PictureBox control and view its content. Note that in this particular implementation, users can drop one or more files onto the PictureBox control. What I have done is to display each photo one by one with a two-second interval.
Implementing Drag and Drop for Custom Objects
Most Windows Forms controls support the set (or subset) of events that I described in this article for drag and drop operations. However, what happens if the control you want to enable for drag and drop does not support the list of events that I have just described? A good example is the Windows Media Player ActiveX control. You might want to embed the Windows Media Player control in a Windows application so that users can simply drag and drop media files onto it to play. The Windows Media Player ActiveX control by itself does not support events like DragEnter and DragDrop, and hence there is no easy way to implement drag and drop.
A workaround is to wrap the ActiveX control using a User control. First, add a new User Control item (right-click the project name in Solution Explorer and select Add > New Item… > select User Control) to the existing project. Name the file as MediaPlayer.vb.
Right-click the Toolbox and select Choose Items…. In the Choose Toolbox Items dialog box, click the COM Components tab and check the Windows Media Player object (Figure 10). Click OK to add the Windows Media Player control onto the Toolbox.
Drag the Windows Media Player control from the Toolbox and drop it onto the MediaPlayer.vb design surface (Figure 11).
In the code behind of MediaPlayer.vb, code the following:
Public Class MediaPlayer
Private _URL As String
Public Property URL() As String
Get
Return _URL
End Get
Set(ByVal value As String)
_URL = value
AxWindowsMediaPlayer1. _
URL = _URL
End Set
End Property
End Class
Essentially you expose the URL property to let the user of this control set the URL of the media file to play.
Right-click the project name in Solution Explorer and select Build. The MediaPlayer control should now appear in the toolbox (Figure 12).
Drag and drop the MediaPlayer user control onto Form1 (Figure 13).
Switch to the code behind of Form1 and handle the DragEnter event of the MediaPlayer user control (Listing 14).
Finally, handle its DragDrop event so that you can play the media file dropped by the user (Listing 15).
Note that since the user can drop multiple files onto the control, you will only load the first file using the MediaPlayer control. Figure 14 shows the MediaPlayer control hosted in a Windows Form playing the file dropped onto it.
Dragging and Dropping Custom Objects
So far in this article I’ve shown you how to use the various data types as specified in the DataFormats class: Bitmap, CommaSeparatedValue, Dib, Dif, EnhancedMetafile, FileDrop, Html, Locale, MetafilePict, OemText, Palette, PenData, Riff, Rtf, Serializable, StringFormat, SymbolicLink, Text, Tiff, UnicodeText, and WaveAudio. What happens if you want to drag and drop data of a specific type? For example, you might want to drag an item in a ListView control. In this case, the DragEnter event will look something like this:
If (e.Data.GetDataPresent _
("System.Windows.Forms. _
ListViewItem()")) Then
'---determine if this is a copy
'---or move---
If (e.KeyState And CtrlMask) =
CtrlMask Then
e.Effect =
DragDropEffects.Copy
Else
e.Effect =
DragDropEffects.Move
End If
The MouseDown event handler will now look like this:
Control.DoDragDrop(New _
DataObject("System.Windows.Forms.
ListViewItem()", _
Items), DragDropEffects.Move Or _
DragDropEffects.Copy)
Summary
It is not really difficult to implement drag and drop functionality in your Windows application. All you need is to understand the type of data you want to support and make the necessary provisions for dealing with that particular data type.
Listing 1: Servicing the DragEnter event for the TextBox control
Private Sub TextBox1_DragEnter( _
ByVal sender As Object, _
ByVal e As _
System.Windows.Forms.DragEventArgs) _
Handles TextBox1.DragEnter
'---if the data to be dropped is a text format---
If (e.Data.GetDataPresent(DataFormats.Text)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
TextBox1.BorderStyle = BorderStyle.FixedSingle
End If
End Sub
Listing 2: Servicing the DragDrop event for the TextBox control
Private Sub TextBox1_DragDrop( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms.DragEventArgs) _
Handles TextBox1.DragDrop
'---if the data to be dropped is a text format---
If (e.Data.GetDataPresent(DataFormats.Text)) Then
'---set the control to display
' the text being dropped---
TextBox1.Text = e.Data.GetData( _
DataFormats.Text)
End If
'---set the borderstyle back to its original---
TextBox1.BorderStyle = BorderStyle.Fixed3D
End Sub
Listing 3: Setting the PictureBox control properties
Private Sub Form1_Load( _
ByVal sender As System.Object, _
ByVal e As System.EventArgs) _
Handles MyBase.Load
'---set the control to allow drop---
TextBox1.AllowDrop = True
'---set the control to allow drop---
With PictureBox1
.AllowDrop = True
.BorderStyle = BorderStyle.FixedSingle
.SizeMode = PictureBoxSizeMode.StretchImage
End With
Listing 4: Servicing the DragEnter event for the PictureBox control
Private Sub PictureBox1_DragEnter( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox1.DragEnter
'---if the data to be dropped is an image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
PictureBox1.BorderStyle = BorderStyle.Fixed3D
End If
End Sub
Listing 5: Servicing the DragDrop event for the PictureBox control
Private Sub PictureBox1_DragDrop( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox1.DragDrop
'---if the data to be dropped is a bitmap format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---set the control to display
' the text being dropped---
PictureBox1.Image = e.Data.GetData( _
DataFormats.Bitmap)
End If
'---set the border style back to its original---
PictureBox1.BorderStyle = BorderStyle.FixedSingle
End Sub
Listing 6: The GetFormats() method returns the type of data that is passed into the event
Private Sub PictureBox1_DragEnter( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms.DragEventArgs) _
Handles PictureBox1.DragEnter
Dim formats As String() = e.Data.GetFormats
'---if the data to be dropped is an image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
...
Listing 7: Modifying the DragEnter event
Private Sub PictureBox1_DragEnter( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox1.DragEnter
Dim formats As String() = e.Data.GetFormats
'---if the data to be dropped is an image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
PictureBox1.BorderStyle = BorderStyle.Fixed3D
ElseIf (e.Data.GetDataPresent(DataFormats.Rtf)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
PictureBox1.BorderStyle = BorderStyle.Fixed3D
End If
End Sub
Listing 8: Modifying the DragDrop event
Private Sub PictureBox1_DragDrop( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox1.DragDrop
'---if the data to be dropped is a image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---set the control to display
' the text being dropped---
PictureBox1.Image = e.Data.GetData( _
DataFormats.Bitmap)
ElseIf (e.Data.GetDataPresent(DataFormats.Rtf)) Then
'---display the rich text---
Console.WriteLine( _
e.Data.GetData(DataFormats.Rtf))
End If
'---set the borderstyle back to its original---
PictureBox1.BorderStyle = BorderStyle.FixedSingle
End Sub
Listing 9: Setting the properties for the second PictureBox control
Private Sub Form1_Load( _
ByVal sender As System.Object, _
ByVal e As System.EventArgs) _
Handles MyBase.Load
'---set the control to allow drop---
TextBox1.AllowDrop = True
'---set the control to allow drop---
With PictureBox1
.AllowDrop = True
.BorderStyle = BorderStyle.FixedSingle
.SizeMode = PictureBoxSizeMode.StretchImage
End With
'---set the control to allow drop---
With PictureBox2
.AllowDrop = True
.BorderStyle = BorderStyle.FixedSingle
.SizeMode = PictureBoxSizeMode.StretchImage
End With
Listing 10: Servicing the DragDrop, DragEnter, and DragLeave events for the second PictureBox control
Private Sub PictureBox2_DragDrop( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox2.DragDrop
'---if the data to be dropped is a bitmap format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---set the control to display
' the bitmap being dropped---
PictureBox2.Image = e.Data.GetData( _
DataFormats.Bitmap)
End If
'---set the borderstyle back to its original---
PictureBox2.BorderStyle = BorderStyle.FixedSingle
End Sub
Private Sub PictureBox2_DragEnter( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox2.DragEnter
'---if the data to be dropped is an image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
PictureBox2.BorderStyle = BorderStyle.Fixed3D
End If
End Sub
Private Sub PictureBox2_DragLeave( _
ByVal sender As Object, _
ByVal e As System.EventArgs) _
Handles PictureBox2.DragLeave
'---set the borderstyle back to its original---
PictureBox2.BorderStyle = BorderStyle.FixedSingle
End Sub
Listing 11: Servicing the QueryContinueDrag event for the PictureBox control
Private Sub PictureBox1_QueryContinueDrag( _
ByVal sender As Object, _
ByVal e As _
System.Windows.Forms.QueryContinueDragEventArgs) _
Handles PictureBox1.QueryContinueDrag
If e.Action = DragAction.Drop Then
If (e.KeyState And CtrlMask) <> CtrlMask Then
'---a move operation---
PictureBox1.Image = Nothing
End If
End If
End Sub
Listing 12: Modifying the DragEnter event
Private Sub PictureBox1_DragEnter( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles PictureBox1.DragEnter
Dim formats As String() = e.Data.GetFormats
'---if the data to be dropped is an image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
PictureBox1.BorderStyle = BorderStyle.Fixed3D
ElseIf (e.Data.GetDataPresent( _
DataFormats.FileDrop)) Then
'---if this is a file drop---
e.Effect = DragDropEffects.All
ElseIf (e.Data.GetDataPresent( _
DataFormats.Rtf)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
'---change the border style of the control---
PictureBox1.BorderStyle = BorderStyle.Fixed3D
End If
End Sub
Listing 13: Modifying the DragDrop event
Private Sub PictureBox1_DragDrop( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms.DragEventArgs) _
Handles PictureBox1.DragDrop
'---if the data to be dropped is a image format---
If (e.Data.GetDataPresent(DataFormats.Bitmap)) Then
'---set the control to
' display the bitmap being dropped---
PictureBox1.Image = e.Data.GetData( _
DataFormats.Bitmap)
ElseIf (e.Data.GetDataPresent( _
DataFormats.FileDrop)) Then
Dim files() As String
files = e.Data.GetData( _
DataFormats.FileDrop)
For Each file As String In files
file = UCase(file)
If file.EndsWith(".GIF") Or _
file.EndsWith(".JPG") Or _
file.EndsWith(".BMP") Then
PictureBox1.Image = New Bitmap(file)
End If
System.Threading.Thread.Sleep(2000)
Application.DoEvents()
Next
ElseIf (e.Data.GetDataPresent(DataFormats.Rtf)) Then
'---display the rich text---
Console.WriteLine(e.Data.GetData( _
DataFormats.Rtf))
End If
'---set the borderstyle back to its original---
PictureBox1.BorderStyle = _
BorderStyle.FixedSingle
End Sub
Listing 14: Servicing the DragEnter event for the MediaPlayer control
Private Sub MediaPlayer1_DragEnter( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms.DragEventArgs) _
Handles MediaPlayer1.DragEnter
'---if the data to be dropped is
' an filedrop format---
If (e.Data.GetDataPresent( _
DataFormats.FileDrop)) Then
'---determine if this is a copy or move---
If (e.KeyState And CtrlMask) = CtrlMask Then
e.Effect = DragDropEffects.Copy
Else
e.Effect = DragDropEffects.Move
End If
End If
End Sub
Listing 15: Servicing the DragDrop event for the MediaPlayer control
Private Sub MediaPlayer1_DragDrop( _
ByVal sender As Object, _
ByVal e As System.Windows.Forms. _
DragEventArgs) _
Handles MediaPlayer1.DragDrop
If (e.Data.GetDataPresent( _
DataFormats.FileDrop)) Then
Dim files() As String
'---get all the file names---
files = e.Data.GetData( _
DataFormats.FileDrop)
If files.Length > 0 Then
'---load only the first file---
files(0) = UCase(files(0))
If files(0).EndsWith(".WMV") Then
'---get the media player to play the
' first file---
MediaPlayer1.URL = files(0)
End If
End If
End If
End Sub
Dialog to open a file
Create a project of type “Windows Forms Application”:
On form add a textbox and a button like this:
Click twice on button “Open” to generate the click event:
Add the follow code:
- using (OpenFileDialog dialog = new OpenFileDialog())
- {
- if (dialog.ShowDialog() == System.Windows.Forms.DialogResult.OK)
- {
- textBox1.Text = dialog.FileName;
- }
- }
Run your application.
Drag file from explorer to your Application
We need to set the property “allowdrop” to true on our textbox:
Now we need to implement the event DragOver on the textbox component:
At event created we need to add the follow code:
- private void textBox1_DragOver(object sender, DragEventArgs e)
- {
- if (e.Data.GetDataPresent(DataFormats.FileDrop))
- e.Effect = DragDropEffects.Link;
- else
- e.Effect = DragDropEffects.None;
- }
The method “.Data.GetDataPresent(DataFormats.FileDrop)” check if is a File droping. In case true, we set the effect to “Link”.
Now we need to create event DragDrop on textbox:
At event created put this:
- private void textBox1_DragDrop(object sender, DragEventArgs e)
- {
- string[] files = e.Data.GetData(DataFormats.FileDrop) as string[];
- if (files != null && files.Any())
- textBox1.Text = files.First();
- }
Run your application from generated “.exe”, from debug doesn’t work.
How to support drag-and-drop in Windows Forms.
Last week I was writing a Windows Forms application that needed to process data
that had been dragged onto it.
This sent me down a (small) rabbit hole. Supporting drag and drop was not quite
as straightforward as I initially thought!
Here’s how to do it, broken down into a few manageable steps.
1. Allow drag-and-drop on the control
Set the AllowDrop
property to true
on the control which data can be dragged
onto. If you don’t do this, then the events detailed below won’t fire.
This may be the whole form, or perhaps a label saying «drop here».
2. Handle the «enter» event
The DragEnter
event fires when data is dragged over the control in question.
That is, the mouse enters the control’s bounds while the left button is pressed
and the user is dragging something.
This event uses DragEventArgs
, which has the property AllowedEffect
. This is
a flags enum containing all the different «effects» (Link, Copy,
Move)
allowed by the source of the drag. For example, if you drag a file from Windows
File Explorer, you’re allowed to link or copy but not move.
In the event handler, you need to set the value of DragEventArgs.Effect
to one
of the allowed values. This does two things:
- Changes the cursor to provide a visual cue to the user that drag-and-drop is
possible (the default is a «no entry» cursor to indicate that drag-and-drop is
not supported). The exact cursor used depends on which value you choose. - Sets up the
DragDrop
event to fire if/when the user releases the left mouse
button (see below).
If you want to make a bigger visual change than just the cursor (e.g. change the
appearance of the control), then you should also do that in the DragEnter
event handler.
3. Handle the «drop» event
The DragDrop
event fires when the user releases the mouse to drop data onto
the control.
The data that has been dragged can be found in DragEventArgs.Data
. This is an
IDataObject
, which contains the data in a number of «formats». The exact
formats used will depend on where the data was dragged from, so when developing
there’s a bit of trial and error to see which formats are appropriate for you to
use.
Some helpful methods on IDataObject
:
GetFormats()
: this returns astring[]
, each item of which is the name of a
format the data can be provided in.GetData(string format)
: this returns the data in the specified format.
Because this method returnsobject
, you’ll have to cast it before using it.
Ifformat
is not recognised, it returnsnull
.GetDataPresent(string format)
: this returns abool
indicating whether the
data is present in the specified format. This method could be used in the
DragEnter
event handler to determine whether to enable drag-and-drop at all.
As an example, one of the formats when dragging data from Windows File Explorer
was "FileName"
. If you call GetData("FileName")
then you get a string[]
containing the full paths of the files being dragged.
Once you have the data you want you can then do whatever you like. For example,
my application reads in the file, transforms it, and puts the result in a text
box.
This event also uses DragEventArgs
, and it uses the same instance as the one
provided in DragEnter
, meaning you can inspect the value you chose for
Effect
if you need to.
4. Handle the «leave» event (maybe)
The DragLeave
event fires when the mouse leave the bounds of the control and
the user is still dragging the item.
Handling this event is only necessary if you need to tidy up from the
DragEnter
event; perhaps you need to reset some visuals to indicate that
releasing the mouse won’t complete the drag-and-drop operation.
Example
The following example shows you how to accept files dragged from File Explorer.
It captures a file dragged onto dragDropLabel
and displays the file contents
in outputTextBox
.
using System.Linq;
using System.Windows.Forms;
private void dragDropLabel_DragEnter(object sender, DragEventArgs e)
{
if (e.AllowedEffect.HasFlag(DragDropEffects.Copy) &&
e.Data.GetData("FileName") is string[] fileNames &&
fileNames.Any())
{
e.Effect = DragDropEffects.Copy;
}
}
private void dragDropLabel_DragDrop(object sender, DragEventArgs e)
{
var fileNames = (string[])e.Data.GetData("FileName");
var fileName = fileNames[0];
this.outputTextBox.Text = File.ReadAllText(fileName);
}
Appendix: exploring the data
While developing, the following snippet was useful for discovering what formats
were supported by DragEventArgs.Data
and exactly how the data looked.
foreach (var format in e.Data.GetFormats())
{
var data = e.Data.GetData(format);
string text;
if (data is MemoryStream ms)
{
using (var sr = new StreamReader(ms))
{
text = sr.ReadToEnd();
}
}
else
{
text = data?.ToString();
}
System.Diagnostics.Debug.WriteLine($"{format}:{text}");
}
Search code, repositories, users, issues, pull requests…
Provide feedback
Saved searches
Use saved searches to filter your results more quickly
Sign up