CListCtrl实现拖拽 效果

    技术2025-03-24  13

    方法1:

    void ClistOx::OnLvnBegindrag(NMHDR *pNMHDR, LRESULT *pResult){LPNMLISTVIEW pNMLV = reinterpret_cast<LPNMLISTVIEW>(pNMHDR);// TODO: 在此添加控件通知处理程序代码CPoint ptItem, //ptAction, //事件发生的位置 ptImage;//移动位图的位置NM_LISTVIEW *pnmListView = (NM_LISTVIEW *)pNMHDR;ASSERT(!m_bDragging);m_bDragging = TRUE;//获得选种的项的索引m_iItemDrag = pnmListView-> iItem; //获得事件发生的位置ptAction = pnmListView-> ptAction;//获得项的位置GetItemPosition(m_iItemDrag, &ptItem); // ptItem is relative to (0,0) and not the view origin//获得视图原点位置GetOrigin(&m_ptOrigin);ASSERT(m_pimageListDrag == NULL);m_pimageListDrag = CreateDragImage(m_iItemDrag, &ptImage);IMAGEINFO imageInfo;m_pimageListDrag->GetImageInfo(0,&imageInfo);m_ptHotSpot = ptAction - ptItem + m_ptOrigin; // calculate hotspot for the cursorm_pimageListDrag-> DragShowNolock(TRUE); // lock updates and show drag imagem_pimageListDrag->Add(((HICON)LoadImage(NULL,_T("test.ico"),IMAGE_ICON,92,68,LR_LOADFROMFILE|LR_LOADTRANSPARENT)));//m_pimageListDrag-> SetDragCursorImage(2, m_ptHotSpot); // define the hot spot for the new cursor imagem_pimageListDrag-> BeginDrag(0, m_ptHotSpot);//ptAction -= m_sizeDelta;m_pimageListDrag-> DragEnter(this,ptAction);//m_pimageListDrag-> DragMove(ptAction); // move image to overlap original iconSetCapture();CRect rect;GetClientRect(&rect);ClientToScreen(&rect);ClipCursor(rect);//*pResult = 0;}void ClistOx::OnMouseMove(UINT nFlags, CPoint point){int iItem;if (m_bDragging){ASSERT(m_pimageListDrag != NULL);m_pimageListDrag-> DragMove(point); // move the image  if ((iItem = HitTest(point )) != -1){m_iItemDrop = iItem;m_pimageListDrag-> SetDragCursorImage(1, CPoint(4,4)); //m_pimageListDrag-> DragLeave(this); // unlock the window and hide drag image// if (lStyle == LVS_REPORT || lStyle == LVS_LIST)// {// lvitem.iItem = iItem;// lvitem.iSubItem = 0;// lvitem.mask = LVIF_STATE;// lvitem.stateMask = LVIS_DROPHILITED; // highlight the drop target// SetItem(&lvitem);//试图修改多种选中问题// SetItemState(iItem, 0, LVIS_SELECTED|LVIS_FOCUSED);// }//point -= m_sizeDelta;//m_pimageListDrag-> DragEnter(this, point); // lock updates and show drag image}else m_pimageListDrag-> SetDragCursorImage(-1, CPoint(0,0)); }CListCtrl::OnMouseMove(nFlags, point); }void ClistOx::OnLButtonUp(UINT nFlags, CPoint point){//当是Icon和smallIcon时用FindItem()方法,其余遍厉视图列表,确定要替换的项CListCtrl::OnLButtonUp(nFlags, point);if (m_bDragging) // end of the drag operation{ASSERT(m_pimageListDrag != NULL);m_pimageListDrag-> DragLeave(this);m_pimageListDrag-> EndDrag();delete m_pimageListDrag;m_pimageListDrag = NULL;//获得这两项的位置if(m_iItemDrop==m_iItemDrag){SetItemPosition(m_iItemDrag,CPoint(point + m_ptOrigin - m_ptHotSpot));}else{CPoint pointDrag,pointDrop;GetItemPosition(m_iItemDrop,&pointDrop);GetItemPosition(m_iItemDrag,&pointDrag);SetItemPosition(m_iItemDrag,pointDrop);SetItemPosition(m_iItemDrop,pointDrag);}m_bDragging = FALSE;Invalidate();::ReleaseCapture();ClipCursor(NULL);}//CListCtrl::OnLButtonUp(nFlags, point);}

    方法2:

    条目拖动#define MAX_DRAG_SIZE 128#define MAX_DRAG_SIZE_2 (MAX_DRAG_SIZE/2)添加数据成员 CImageList* m_pDragImage = NULL;//拖动时的图片int m_nDragDrop, m_nDrag[MAX_DRAG_SIZE];//用于记录被拖动条目的index以及拖动到的位置注:本文中仅以拖动1条为例,重在介绍方法,如果需要可作相应的扩展响应消息LVN_BEGINDRAG, WM_MOUSEMOVE WM_LBUTTONUPON_NOTIFY_REFLECT(LVN_BEGINDRAG, OnBegindrag)void CMyList::OnBegindrag(NMHDR *pNMHDR, LRESULT *pResult){   LPNMHEADER phdr = reinterpret_cast(pNMHDR);   // TODO: 在此添加控件通知处理程序代码   NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;   *pResult = 0;   CPoint ptAction( pNMListView->ptAction );   for ( int i = 0; i < MAX_DRAG_SIZE; i++ ) {   m_nDrag[i] = -1;}   m_pDragImage = CreateDragImage( ptAction );//绘制图像   if ( m_pDragImage == NULL ) return;   m_nDragDrop = -1;   UpdateWindow();   CRect rcClient;   GetClientRect( &rcClient );   ClientToScreen( &rcClient );   ClipCursor( &rcClient );   SetCapture();   SetFocus();   UpdateWindow();   m_pDragImage->DragEnter( this, ptAction );   *pResult = 0;}CImageList* CMyList::CreateDragImage(const CPoint& ptMouse){CRect rcClient, rcOne, rcAll( 32000, 32000, -32000, -32000 );int nIndex;if ( GetSelectedCount() == 0 ) return NULL;SetFocus();GetClientRect( &rcClient );int nCount = 0;for ( nIndex = -1 ; ( nIndex = GetNextItem( nIndex, LVNI_SELECTED ) ) >= 0 && MAX_DRAG_SIZE >= nCount; ){   m_nDrag[ nCount++ ] = nIndex;   GetItemRect( nIndex, rcOne, LVIR_BOUNDS );   if ( rcOne.IntersectRect( &rcClient, &rcOne ) )   {    rcAll.left   = min( rcAll.left, rcOne.left );    rcAll.top   = min( rcAll.top, rcOne.top );    rcAll.right   = max( rcAll.right, rcOne.right );    rcAll.bottom = max( rcAll.bottom, rcOne.bottom );   }   SetItemState( nIndex, 0, LVIS_FOCUSED );}BOOL bClipped = rcAll.Height() > MAX_DRAG_SIZE;if ( bClipped ){   rcAll.left   = max( rcAll.left, ptMouse.x - MAX_DRAG_SIZE_2 );   rcAll.right   = max( rcAll.right, ptMouse.x + MAX_DRAG_SIZE_2 );   rcAll.top   = max( rcAll.top, ptMouse.y - MAX_DRAG_SIZE_2 );   rcAll.bottom = max( rcAll.bottom, ptMouse.y + MAX_DRAG_SIZE_2 );}CClientDC dcClient( this );CBitmap bmAll, bmDrag;CDC dcAll, dcDrag;if ( ! dcAll.CreateCompatibleDC( &dcClient ) )   return NULL;if ( ! bmAll.CreateCompatibleBitmap( &dcClient, rcClient.Width(), rcClient.Height() ) )   return NULL;if ( ! dcDrag.CreateCompatibleDC( &dcClient ) )   return NULL;if ( ! bmDrag.CreateCompatibleBitmap( &dcClient, rcAll.Width(), rcAll.Height() ) )   return NULL;CBitmap *pOldAll = dcAll.SelectObject( &bmAll );COLORREF crDrag = RGB( 250, 255, 250 );dcAll.FillSolidRect( &rcClient, crDrag );COLORREF crBack = GetBkColor();SetBkColor( crDrag );SendMessage( WM_PAINT, (WPARAM)dcAll.GetSafeHdc() );SetBkColor( crBack );CBitmap *pOldDrag = dcDrag.SelectObject( &bmDrag );dcDrag.FillSolidRect( 0, 0, rcAll.Width(), rcAll.Height(), crDrag );CRgn pRgn;if ( bClipped ){   CPoint ptMiddle( ptMouse.x - rcAll.left, ptMouse.y - rcAll.top );   pRgn.CreateEllipticRgn( ptMiddle.x - MAX_DRAG_SIZE_2, ptMiddle.y - MAX_DRAG_SIZE_2,    ptMiddle.x + MAX_DRAG_SIZE_2, ptMiddle.y + MAX_DRAG_SIZE_2 );   dcDrag.SelectClipRgn( &pRgn );}for ( nIndex = -1 ; ( nIndex = GetNextItem( nIndex, LVNI_SELECTED ) ) >= 0 ; ){   GetItemRect( nIndex, rcOne, LVIR_BOUNDS );   if ( rcOne.IntersectRect( &rcAll, &rcOne ) )   {    dcDrag.BitBlt( rcOne.left - rcAll.left, rcOne.top - rcAll.top,     rcOne.Width(), rcOne.Height(), &dcAll, rcOne.left, rcOne.top, SRCCOPY );   }}dcDrag.SelectObject( pOldDrag );dcAll.SelectObject( pOldAll );dcDrag.DeleteDC();bmAll.DeleteObject();dcAll.DeleteDC();CImageList* pAll = new CImageList();pAll->Create( rcAll.Width(), rcAll.Height(), ILC_COLOR16|ILC_MASK, 1, 1 );pAll->Add( &bmDrag, crDrag );bmDrag.DeleteObject();pAll->BeginDrag( 0, ptMouse - rcAll.TopLeft() );return pAll;}void CMyList::OnMouseMove(UINT nFlags, CPoint point)//移动的视觉效果{// TODO: 在此添加消息处理程序代码和/或调用默认值int nHit = HitTest( point );if ( m_pDragImage != NULL ){   m_pDragImage->DragMove( point );   if ( nHit != m_nDragDrop )   {    CImageList:ragShowNolock( FALSE );    if ( m_nDragDrop >= 0 ) SetItemState( m_nDragDrop, 0, LVIS_DROPHILITED );    m_nDragDrop = nHit;    if ( m_nDragDrop >= 0 ) SetItemState( m_nDragDrop, LVIS_DROPHILITED, LVIS_DROPHILITED );    UpdateWindow();    CImageList:ragShowNolock( TRUE );   }}CListCtrl::OnMouseMove(nFlags, point);}void CMyList::OnLButtonUp(UINT nFlags, CPoint point){// TODO: 在此添加消息处理程序代码和/或调用默认值if ( m_pDragImage == NULL ){   CListCtrl::OnLButtonUp( nFlags, point );   return;} ClipCursor( NULL );ReleaseCapture();m_pDragImage->DragLeave( this );m_pDragImage->EndDrag();delete m_pDragImage;m_pDragImage = NULL;if ( m_nDragDrop >= 0 ){   SetItemState( m_nDragDrop, 0, LVIS_DROPHILITED );   /*由于本类中只是加了拖动的效果和记录拖动的相关信息,当拖动结束以后并没有实际效果。故需要传递消息 在其父窗口中进行拖动的实际处理*/   NM_LISTVIEW pNotify;   pNotify.hdr.hwndFrom = GetSafeHwnd();   pNotify.hdr.idFrom   = GetDlgCtrlID();   pNotify.hdr.code   = HDN_ENDDRAG;   pNotify.iItem    = m_nDragDrop;//拖动到的index   pNotify.lParam    = (LPARAM)m_nDrag;//被拖动的Item   GetOwner()->SendMessage( WM_NOTIFY, pNotify.hdr.idFrom, (LPARAM)&pNotify );}}在其listbox的父窗口中进行以下处理ON_NOTIFY(HDN_ENDDRAG, IDC_LIST, OnEnddragList)//IDC_LIST为list控件的IDvoid CDlgXX::OnEnddragList(NMHDR *pNMHDR, LRESULT *pResult){NM_LISTVIEW* pNotify = (NM_LISTVIEW *)pNMHDR;int* pBegin = (int*)pNotify->lParam;int nEnd = pNotify->iItem;m_bChange = TRUE;CString strTmp;BOOL bAhead = pBegin[0] > nEnd ? TRUE : FALSE;for ( int i = 0; i < MAX_DRAG_SIZE && pBegin[i] >= 0; i++ ){   if ( pBegin[i] == nEnd )   {    nEnd++;    continue;   }   strTmp = m_ctrlList.GetItemText( pBegin[i], 0 );   m_ctrlList.DeleteItem( pBegin[i] );     /*向上托第一条插在nEnd之前,向下托第一条插在nEnd之后,所拖动的条目保持原来的顺序*/   if ( bAhead )   {    if ( pBegin[i] < nEnd )     nEnd--;    for ( int j = i + 1; j < MAX_DRAG_SIZE && pBegin[j] >= 0; j++)    {     if ( pBegin[j] <= nEnd )      pBegin[j]--;    }    m_ctrlList.InsertItem( nEnd, strTmp );    nEnd++;   }   else   {    if ( pBegin[i] >= nEnd )     nEnd++;    for ( int j = i + 1; j < MAX_DRAG_SIZE && pBegin[j] >= 0; j++)     if ( pBegin[j] <= nEnd )      pBegin[j]--;    m_ctrlList.InsertItem( nEnd, strTmp );    }}    *pResult = 0;}

     

    最新回复(0)