Enabling Drag and Drop between two Listboxes on a VBA Userform

Content

Introduction

One of the more difficult tasks in VBA programming is to enable drag and drop between two listboxes on a userform. There is an article on the Microsoft site about this that kind of gets you half-way there, but it does not work if you have the listbox setup to allow selecting more than one item. To make it easy to enable drag and drop between two listboxes on a userform I wrote a class module which you can simply insert into your VBA project. After doing that, it only takes about 8 lines of code to get it working.

This is what it looks like (tap on, or place mouse over image to see the animation):

Video showing drag and drop between two listboxes on a VBA userform

I have prepared a demo file for you!

Updates

9 March 2022: As of now, the class (see the new file in the download, file name TwoListboxDragDrop 7.xlsm) supports multi-column listboxes.

Working principle

If you want to implement Drag and Drop between controls (or within a control) on a userform, these control events are needed:

MouseMove

When the left mouse button is down while the mouse is moved, it is likely the user is starting a drag operation. We need to make sure the currently selected items of the list are remembered.

I want the code to enable drag and drop both between the two controls and within a control and I want to indicate where the drop will take place. I therefore need two things:

  1. A way to remember which list items were selected when I started the drag

    The code temporarily marks the items we're dragging by adding this bit of string in front of each selected item: "> ". This bit of string will later determine which items must be removed from the list after the drop has been done. If the user drags without a mousebutton held down, the "> " is removed from the list items to clean up.

  2. A way to indicate the drop location

    VBA does not make it easy to detect what list item is under the mouse cursor. The MouseMove and BeforeDropOrPaste events do give an X and Y coordinate, but determining which item "belongs" to a certain Y isn't easy as we do not know the Y coordinate of each item. The trick I used: temporarily add 40 list items to the listbox (to make sure the entire height of the box is used up by list items), then set the topindex of the listbox to the last one. This forces the listbox to scroll down, displaying the last item at the bottom of the listbox. The listbox then gets a new TopIndex (of the item actually displaying at the top) so we can now count how many items fit into the listbox' height. We can thus calculate the height of an individual item and use that to detect where our mouse is. You'll find the code that does this in the Property Set procedure belonging to ListBox1.

     Trick to get the height of an item in the listbox

  3. A way to scroll the listbox (up or down) if there are more items than the listbox can display in its height.

BeforeDragOver

The BeforeDragOver event is used to make sure the mouse pointer changes shape so the user has a visual indication whether or not he can drop items on that listbox. The class contains two boolean properties DragWithin1 and DragWithin2 which you can set to False if you want to disable drag withing either listbox.

BeforeDropOrPaste

This is where the magic happens, the code checks if there is anything to drop and if so, first adds the dragged items to the listbox and subsequently removes all marked items from the (other) listbox.

The code

Code needs to go in two places to enable drag and drop in your userform: In the code window behind the userform and in a separate class module called clsDragDrop you must insert into the VBAProject.

Userform

The userform has two listboxes (with their default names ListBox1 and ListBox2). All it takes to implement the drag and drop is these lines of code in the userform's code module:

Option Explicit

'Holds the instance of the drag and drop class
Private mcDragDrop As clsDragDrop

Private Sub UserForm_Initialize()
'------Start of lines just for our demo------
    Dim lCt As Long
    'Just add some items so we have something to drag
    With Me.ListBox1
        .ColumnCount = 2
        .ColumnWidths = .Width / 2.1 & "," & .Width / 2.1
        For lCt = 0 To 50
            .AddItem "Item " & lCt
            .List(.ListCount - 1, 1) = lCt
        Next
        .AddItem " "
    End With
    With Me.ListBox2
        .ColumnCount = 2
        .ColumnWidths = .Width / 2.1 & "," & .Width / 2.1
        For lCt = 60 To 100
            .AddItem "Item " & lCt
            .List(.ListCount - 1, 1) = lCt
        Next
        .AddItem " "
    End With
'------End of lines just for our demo------

    'Instantiate our class
    Set mcDragDrop = New clsDragDrop
    With mcDragDrop
        'Tell it which listboxes to work with
        Set mcDragDrop.ListBox1 = Me.ListBox1
        Set mcDragDrop.ListBox2 = Me.ListBox2
        'Enable drag and drop within both listboxes
        .DragWithin1 = True
        .DragWithin2 = True
        .DragIndicator = "> "
        .DropIndicator = "< "
    End With
End Sub

Note that the part between "Start of lines just for our demo" and "End of lines just for our demo" is only there to add some items to the listbox for demonstration purposes, you'd remove that or replace it with your own code that populates the listboxes on your form.

The clsDragDrop class

I decided to explain the code by inserting comments. I'm not adding more explanation here so I hope this is clear enough!

Note that most event routines are very short. They just contain a call to an associated method. This is because all aforementioned events come in pairs, one for each ListBox. The code for needed for each event is identical and hence private methods are called to handle them.

Here is the entire code inside the class module clsDragDrop, fully commented:

'---------------------------------------------------------------------------------------
' File   : clsDragDrop
' Author : Jan Karel Pieterse
' (c)    : Copyright JKP Application Development Services, all rights reserved
' Date   : 11-Jan-2022
' Purpose: Enable drag and drop between two listboxes
'---------------------------------------------------------------------------------------

Option Explicit

'Used to determine list item under mouse when dropping
Private mListItemCount As Long
Private mListItemSize As Double

Private Const LEFTMOUSEBUTTON As Long = 1

Private Const COLSEP As String = "¦"  'Chr(182) separator for the columns of a selected item we're dragging

Private WithEvents mListBox1 As msforms.listBox
Private WithEvents mListBox2 As msforms.listBox

'These two determine whether or not we are allowed to drag inside the controls:
Private mDragWithin1 As Boolean
Private mDragWithin2 As Boolean

Private mXStart As Single
Private mYstart As Single

Private mDragIndicator As String
Private mDropIndicator As String

'the listbox we dragged items from
Private mSourceBox As msforms.listBox
'the listbox we dragged items to
Private mTargetBox As msforms.listBox

Private Sub Class_Initialize()
    'Set defaults
    mDragWithin1 = False
    mDragWithin2 = False
End Sub

'---------------------Events---------------------

Private Sub Class_Terminate()
    Set mListBox1 = Nothing
    Set mListBox2 = Nothing
End Sub

Private Sub mListBox1_MouseMove(ByVal Button As Integer, _
                                 ByVal Shift As Integer, _
                                 ByVal X As Single, _
                                 ByVal Y As Single)
    HandleMouseMove mListBox1, Button, Shift, X, Y
    'MsgBox "NOTE TO SELF: HANDLE MULTI-COLUMN LISTBOXES!!"
End Sub

Private Sub mListBox1_MouseDown(ByVal Button As Integer, ByVal Shift As Integer, ByVal X As Single, ByVal Y As Single)
    mXStart = X
    mYstart = Y
End Sub

'Private Sub mListBox1_MouseUp(ByVal Button As Integer, ByVal Shift As Integer, ByVal X As Single, ByVal Y As Single)
'    FixSelectionAfterDrop mListBox1
'End Sub

Private Sub mListBox2_MouseDown(ByVal Button As Integer, ByVal Shift As Integer, ByVal X As Single, ByVal Y As Single)
    mXStart = X
    mYstart = Y
End Sub
'Private Sub mListBox2_MouseUp(ByVal Button As Integer, ByVal Shift As Integer, ByVal X As Single, ByVal Y As Single)
'    FixSelectionAfterDrop mListBox2
'End Sub

Private Sub mListBox2_MouseMove(ByVal Button As Integer, _
                                 ByVal Shift As Integer, _
                                 ByVal X As Single, _
                                 ByVal Y As Single)

    HandleMouseMove mListBox2, Button, Shift, X, Y
End Sub

Private Sub mListBox1_BeforeDragOver(ByVal Cancel As msforms.ReturnBoolean, _
                                      ByVal Data As msforms.DataObject, _
                                      ByVal X As Single, _
                                      ByVal Y As Single, _
                                      ByVal DragState As msforms.fmDragState, _
                                      ByVal Effect As msforms.ReturnEffect, _
                                      ByVal Shift As Integer)

    HandleBeforeDragOver mListBox1, Cancel, Data, X, Y, DragState, Effect, Shift
End Sub

Private Sub mListBox2_BeforeDragOver(ByVal Cancel As msforms.ReturnBoolean, _
                                      ByVal Data As msforms.DataObject, _
                                      ByVal X As Single, _
                                      ByVal Y As Single, _
                                      ByVal DragState As msforms.fmDragState, _
                                      ByVal Effect As msforms.ReturnEffect, _
                                      ByVal Shift As Integer)

    HandleBeforeDragOver mListBox2, Cancel, Data, X, Y, DragState, Effect, Shift
End Sub

Private Sub mListBox1_BeforeDropOrPaste(ByVal Cancel As msforms.ReturnBoolean, _
                                         ByVal Action As msforms.fmAction, _
                                         ByVal Data As msforms.DataObject, _
                                         ByVal X As Single, _
                                         ByVal Y As Single, _
                                         ByVal Effect As msforms.ReturnEffect, _
                                         ByVal Shift As Integer)

    HandleBeforeDropOrPaste mListBox1, Cancel, Action, Data, X, Y, Effect, Shift
End Sub

Private Sub mListBox2_BeforeDropOrPaste(ByVal Cancel As msforms.ReturnBoolean, _
                                         ByVal Action As msforms.fmAction, _
                                         ByVal Data As msforms.DataObject, _
                                         ByVal X As Single, _
                                         ByVal Y As Single, _
                                         ByVal Effect As msforms.ReturnEffect, _
                                         ByVal Shift As Integer)

    HandleBeforeDropOrPaste mListBox2, Cancel, Action, Data, X, Y, Effect, Shift
End Sub

'---------------------Methods---------------------

Private Sub HandleMouseMove(ByRef localListbox As Object, _
                            ByVal Button As Integer, _
                            ByVal Shift As Integer, _
                            ByVal X As Single, _
                            ByVal Y As Single)
    Dim objData As DataObject
    Dim lEffect As Long
    Dim lCt As Long
    Dim lclicked As Long
    Dim sSelected As String
    Dim listCol As Long
    If Button = LEFTMOUSEBUTTON Then
        With localListbox
            'Detect on which item we last clicked
            'This is because if you first click 3 items and then initiate your drag by
            'clicking on the middle one, VBA unchecks the one you clicked on
            lclicked = .TopIndex + Int(Y / mListItemSize)
            If lclicked >= .ListCount Then lclicked = .ListCount
            If lclicked < 0 Then lclicked = 0
            For lCt = 0 To .ListCount - 1
                If (.Selected(lCt) Or lCt = lclicked) And Len(Trim(.List(lCt))) > 0 Then
                    .Selected(lCt) = False
                    'Build a list of the items we're about to drag
                    sSelected = sSelected & "|" & lCt & COLSEP & .List(lCt)
                    For listCol = 1 To .ColumnCount - 1
                        sSelected = sSelected & COLSEP & .List(lCt, listCol)
                    Next
                    'We add a small marker in front of each dragged item
                    'This provides a visual clue to the user which items he is dragging.
                    'In addition, it makes it easier to later recognize the ones that need to be removed from the listbox
                    'after the drop has been done
                    If Not .List(lCt) Like "[" & DragIndicator & "]*" Then
                        .List(lCt) = DragIndicator & .List(lCt)
                    End If
'                ElseIf lCt = lclicked Then
'                    'The one where we clicked might be deselected, but we always want to include that one with the drag
'                    sSelected = sSelected & "|" & lCt & ITEMSEP & .List(lCt)
'
'                    For listCol = 1 To .ColumnCount - 1
'                        sSelected = sSelected & COLSEP & .List(lCt, listCol)
'                    Next
'                    If Not .List(lCt) Like "[" & dragindicator & "]*" Then
'                        .List(lCt) = DragIndicator & .List(lCt)
'                    End If
                End If
            Next
            'If any items were selected, add them to the clipboard
            If Len(sSelected) > 0 Then
                Set mSourceBox = localListbox
                Set objData = New DataObject
                objData.SetText sSelected
                'Now initiate the actual drag
                lEffect = objData.StartDrag
            End If
        End With
    ElseIf Button = 0 And Not mTargetBox Is Nothing Then
        'clean up our temporarily added '> ' markings
        If mTargetBox Is localListbox Then FixSelectionAfterDrop mTargetBox
        With mTargetBox
            For lCt = .ListCount - 1 To 0 Step -1
                .Selected(lCt) = False
                If Left(.List(lCt), Len(DragIndicator)) = DragIndicator Or Left(.List(lCt), Len(DropIndicator)) = DropIndicator Then
                    .List(lCt) = Replace(.List(lCt), DragIndicator, "")
                    .List(lCt) = Replace(.List(lCt), DropIndicator, "")
                    .Selected(lCt) = True
                End If
            Next
        End With
        Set mTargetBox = Nothing
    ElseIf Button = 0 Then
        'clean up our temporarily added '> ' markings, needed when dragging within listbox is not allowed
        With localListbox
            For lCt = .ListCount - 1 To 0 Step -1
                If Left(.List(lCt), Len(DragIndicator)) = DragIndicator Or Left(.List(lCt), Len(DropIndicator)) = DropIndicator Then
                    .List(lCt) = Replace(.List(lCt), DragIndicator, "")
                    .List(lCt) = Replace(.List(lCt), DropIndicator, "")
                    .Selected(lCt) = True
                End If
            Next
        End With
    End If
End Sub

Private Sub HandleBeforeDragOver(localListbox As Object, _
                                 ByRef Cancel As msforms.ReturnBoolean, _
                                 ByRef Data As msforms.DataObject, _
                                 ByRef X As Single, _
                                 ByRef Y As Single, _
                                 ByRef DragState As msforms.fmDragState, _
                                 ByRef Effect As msforms.ReturnEffect, _
                                 ByRef Shift As Integer)
    Dim lTo As Long
    Dim lCt As Long

    Static prevTime As Double

    If Y = 0 Then Exit Sub    'Kludge to work around bug, sometimes Y is zero even when it shouldn't

    Cancel = True

    'This changes the mouse cursor to indicate whether drop is allowed
    If mSourceBox Is localListbox Then
        If localListbox.Name = Me.ListBox1.Name Then
            'We're in ListBox1, decide on dropeffect
            If DragWithin1 Then
                Effect = fmDropEffectMove
            Else
                Effect = fmDropEffectNone
            End If
        Else
            'We're in ListBox2, decide on dropeffect
            If DragWithin2 Then
                Effect = fmDropEffectMove
            Else
                Effect = fmDropEffectNone
            End If
        End If
    Else
        Effect = fmDropEffectMove
    End If
    If Effect = fmDropEffectNone Then Exit Sub
    With localListbox
        'Scroll listbox if we're close to the top and there are hidden items higher up
        If Y < CSng(mListItemSize / 2) Then
            If Timer - prevTime > 0.3 Then
                prevTime = Timer
                If .TopIndex > 0 Then
                    .TopIndex = .TopIndex - 1
                End If
            End If
        End If

        'Scroll listbox if we're close to the bottom and there are hidden items further down the list
        If Y > (.Height - mListItemSize * 0.7) Then
            If Timer - prevTime > 0.3 Then
                prevTime = Timer
                On Error Resume Next
                .TopIndex = .TopIndex + 1
                On Error GoTo 0
            End If
        End If
        'Indicate where we'll be pasting
        lTo = .TopIndex + Int(Y / mListItemSize)
        If lTo >= .ListCount Then lTo = .ListCount
'        If Len(Trim(.List(.ListCount - 1))) = 0 Then
'            lTo = .ListCount - 1
'        End If
        If lTo < 0 Then lTo = 0
        For lCt = 0 To .ListCount - 1
            If lCt = lTo Then
                .Selected(lCt) = True
            Else
                .Selected(lCt) = False
            End If
        Next
    End With
End Sub


Private Sub HandleBeforeDropOrPaste(localListbox As Object, _
                                    ByRef Cancel As msforms.ReturnBoolean, _
                                    ByRef Action As msforms.fmAction, _
                                    ByRef Data As msforms.DataObject, _
                                    ByRef X As Single, _
                                    ByRef Y As Single, _
                                    ByRef Effect As msforms.ReturnEffect, _
                                    ByRef Shift As Integer)
    Dim lTo As Long
    Dim lCt As Long
    Dim sSelected As String
    Dim vSel As Variant
    Dim vCols As Variant
    Dim listCol As Long

    If Abs(X - mXStart) < 2 And Abs(Y - mYstart) < 2 Then Exit Sub

    sSelected = Data.GetText
    vSel = Split(sSelected, "|")
    Cancel = True
    If Not mSourceBox Is localListbox Then
        'We arrive here when we drag from ListBox2 to ListBox1
        'Determine where we let go of our mouse, insert the items above this list item
        With localListbox
            lTo = .TopIndex + Int(Y / mListItemSize)
            If lTo >= .ListCount Then
                lTo = .ListCount
                If Len(Trim(.List(.ListCount - 1))) = 0 Then
                    lTo = .ListCount - 1
                End If
            End If
            If lTo < 0 Then lTo = 0
        End With
        'Add the dragged items at the indicated location
        AddItemsToBox localListbox, vSel, lTo
        Set mTargetBox = localListbox
        'Now delete the ones we dragged from the list
        With mSourceBox
            For lCt = .ListCount - 1 To 0 Step -1
                .Selected(lCt) = False
                If Len(Trim(.List(lCt))) > 0 And .List(lCt) Like "[" & DragIndicator & "]*" Then
                    .RemoveItem lCt
                End If
            Next
        End With
    ElseIf (DragWithin1 And mSourceBox Is mListBox1) Or (DragWithin2 And mSourceBox Is mListBox2) Then
        With localListbox
            'Now we're dragging within the same ListBox
            'Determine where we let go of our mouse, insert the items above this list item
            lTo = .TopIndex + Int(Y / mListItemSize)
            If lTo >= .ListCount Then
                lTo = .ListCount
                If Len(Trim(.List(.ListCount - 1))) = 0 Then
                    lTo = .ListCount - 1
                End If
            End If
            If lTo < 0 Then lTo = 0
            'Add the dragged items at the indicated location
            AddItemsToBox localListbox, vSel, lTo
            Set mTargetBox = localListbox
            'Now delete the ones we dragged from the list
            For lCt = .ListCount - 1 To 0 Step -1
                .Selected(lCt) = False
                If Len(Trim(.List(lCt))) > 0 And .List(lCt) Like "[" & DragIndicator & "]*" Then
                    .RemoveItem lCt
                End If
            Next
        End With
    End If
    Set mSourceBox = Nothing
End Sub

Private Sub AddItemsToBox(box2Add2 As msforms.listBox, vSel As Variant, lTo As Long)
    Dim vCols As Variant
    Dim listCol As Long
    Dim ct As Long
    
    With box2Add2
        For ct = UBound(vSel) To 1 Step -1
            vCols = Split(vSel(ct), COLSEP) 'vcols contains: ListIndex¦col 0¦col 1¦col 2¦......
            'Ignoring vCols(0) as that is the ListIndex, so we first add vCols(1) as a new item: "col 0"
            .AddItem DropIndicator & vCols(1), lTo
            'Now we add the remaining columns from vCols(2): "col 1", "col 2" and up as additional columns to this item of the listbox
            For listCol = 2 To UBound(vCols)
                .List(lTo, listCol - 1) = vCols(listCol)
            Next
        Next
    End With
End Sub
Private Sub FixSelectionAfterDrop(localListbox As Object)
    Dim lCt As Long
    Dim lbSource As msforms.listBox
    If localListbox Is mListBox1 Then
        Set lbSource = mListBox2
    Else
        Set lbSource = mListBox1
    End If
    With lbSource
        For lCt = 0 To .ListCount - 1
            .Selected(lCt) = False
        Next
    End With
End Sub
'---------------------Property let/set and gets---------------------

Public Property Get ListBox1() As msforms.listBox
    Set ListBox1 = mListBox1
End Property

Public Property Set ListBox1(ByVal oNewValue As msforms.listBox)
    Dim lCt As Long
    Set mListBox1 = oNewValue
    'The code below is used to temporarily fill the listbox with 40 items.
    'This is used to be able to determine which item of the list is under the mouse pointer when we drop
    With mListBox1
        For lCt = 1 To 40
            .AddItem lCt
        Next
        .TopIndex = .ListCount - 1
        mListItemCount = .ListCount - .TopIndex
        mListItemSize = .Height / mListItemCount
        For lCt = 1 To 40
            .RemoveItem .ListCount - 1
        Next
    End With
End Property

Public Property Get ListBox2() As msforms.listBox
    Set ListBox2 = mListBox2
End Property

Public Property Set ListBox2(ByVal oNewValue As msforms.listBox)
    Set mListBox2 = oNewValue
End Property

Public Property Get DragWithin1() As Boolean
    DragWithin1 = mDragWithin1
End Property

Public Property Let DragWithin1(ByVal bNewValue As Boolean)
    mDragWithin1 = bNewValue
End Property

Public Property Get DragWithin2() As Boolean
    DragWithin2 = mDragWithin2
End Property

Public Property Let DragWithin2(ByVal bNewValue As Boolean)
    mDragWithin2 = bNewValue
End Property

Public Property Get DragIndicator() As String
    DragIndicator = mDragIndicator
End Property

Public Property Let DragIndicator(ByVal sNewValue As String)
    mDragIndicator = sNewValue
End Property

Public Property Get DropIndicator() As String
    DropIndicator = mDropIndicator
End Property

Public Property Let DropIndicator(ByVal sNewValue As String)
    mDropIndicator = sNewValue
End Property

Comments

Loading comments...