CListCtrl 列表项拖放的实现

最近在做一个好友列表,通过查找资料自己现在了CListCtrl的(内部)拖放,这里写下来和大家分享分享,同时也寻求更好更多的实现方法,如果大家有好的方法,或者其他的方法也可以分享出来;

      拖放的实现总的来说可以分为三步:第一步、开始拖放,做拖放数据的初始化和记录;第二步、实现拖放的移动;第三步、删除原有数据,插入现有数据;

      准备:实现(CListCtrl内部)拖放需要一个记录拖放动作的变量、一个记录被拖放item位置的变量、一个拖放到目标item位置的变量、一个保存拖放中显示图标的变量;定义代码如下:

  1. private
  2.     BOOL m_Draging; 
  3.     int m_nDragIndex; 
  4.     int m_nDropIndex; 
  5.     CImageList *m_pDragImage; 
private:
	BOOL m_Draging;
	int m_nDragIndex;
	int m_nDropIndex;
	CImageList *m_pDragImage;

    第一步:定义LVN_BEGINDRAG消息,响应开始拖放消息;实现代码如下

  1. BEGIN_MESSAGE_MAP(CDragListDlg, CDialog) 
  2.     //{{AFX_MSG_MAP(CDragListDlg)    
  3.     ON_WM_PAINT() 
  4.     ON_WM_QUERYDRAGICON() 
  5.     ON_WM_SYSCOMMAND() 
  6.     ON_NOTIFY(LVN_BEGINDRAG, IDC_LIST1, OnBegindrag) 
  7.     ON_WM_MOUSEMOVE() 
  8.     ON_WM_LBUTTONUP() 
  9.     //}}AFX_MSG_MAP 
  10. END_MESSAGE_MAP() 
BEGIN_MESSAGE_MAP(CDragListDlg, CDialog)
	//{{AFX_MSG_MAP(CDragListDlg)	
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	ON_WM_SYSCOMMAND()
	ON_NOTIFY(LVN_BEGINDRAG, IDC_LIST1, OnBegindrag)
	ON_WM_MOUSEMOVE()
	ON_WM_LBUTTONUP()
	//}}AFX_MSG_MAP
END_MESSAGE_MAP()
  1. void CDragListDlg::OnBegindrag(NMHDR* pNMHDR, LRESULT* pResult) 
  2.     NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR; 
  3.  
  4.         //保存被拖动的Item        
  5.     m_nDragIndex = pNMListView->iItem; 
  6.  
  7.     POINT pt; 
  8.     pt.x = 8; 
  9.     pt.y = 8; 
  10.  
  11.         //创建拖动显示的图标并让图标开始拖动显示 
  12.     m_pDragImage = m_list.CreateDragImage(m_nDragIndex, &pt); 
  13.     if(m_pDragImage == NULL) 
  14.     { 
  15.         m_nDragIndex = -1; 
  16.         return
  17.     } 
  18.  
  19.     CBitmap bitmap; 
  20.     bitmap.LoadBitmap(IDB_BMP_DRAGS); 
  21.     m_pDragImage->Replace(0, &bitmap, &bitmap); 
  22.     m_pDragImage->BeginDrag(0, CPoint(8, 8)); 
  23.     m_pDragImage->DragEnter(GetDesktopWindow(), pNMListView->ptAction); 
  24.  
  25.     m_Draging = true
  26.     m_nDropIndex = -1; 
  27.  
  28.          
  29.     SetCapture(); 
  30.  
  31.     *pResult = 0; 
void CDragListDlg::OnBegindrag(NMHDR* pNMHDR, LRESULT* pResult)
{
	NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;

        //保存被拖动的Item       
	m_nDragIndex = pNMListView->iItem;

	POINT pt;
	pt.x = 8;
	pt.y = 8;

        //创建拖动显示的图标并让图标开始拖动显示
	m_pDragImage = m_list.CreateDragImage(m_nDragIndex, &pt);
	if(m_pDragImage == NULL)
	{
		m_nDragIndex = -1;
		return ;
	}

	CBitmap bitmap;
	bitmap.LoadBitmap(IDB_BMP_DRAGS);
	m_pDragImage->Replace(0, &bitmap, &bitmap);
	m_pDragImage->BeginDrag(0, CPoint(8, 8));
	m_pDragImage->DragEnter(GetDesktopWindow(), pNMListView->ptAction);

	m_Draging = true;
	m_nDropIndex = -1;

        
	SetCapture();

	*pResult = 0;
}


在上面函数中我们需要记录被拖放的item,和创建拖放图标,开始让图标拖动显示

       第二步:对于拖放操作的移动我们需要在WM_MOUSEMOVE响应函数中实现,因此我们需要响应WM_MOUSEMOVE消息,函数的代码如下

  1. void CDragListDlg::OnMouseMove(UINT nFlags, CPoint point)  
  2.     if (m_Draging) 
  3.     { 
  4.         CPoint pt(point); 
  5.         ClientToScreen(&pt); 
  6.         m_pDragImage->DragMove(pt); 
  7.         m_pDragImage->DragShowNolock(false); 
  8.  
  9.         CWnd *pWnd = WindowFromPoint(pt); 
  10.         if (pWnd->IsKindOf(RUNTIME_CLASS(CListCtrl))) 
  11.         { 
  12.             SetCursor(LoadCursor(NULL, IDC_ARROW)); 
  13.             CListCtrl* pList = (CListCtrl*)pWnd; 
  14.  
  15.             ScreenToClient(&pt); 
  16.             m_nDropIndex = pList->HitTest(pt); 
  17.             pList->RedrawItems(m_nDropIndex, m_nDropIndex); 
  18.             pList->UpdateWindow(); 
  19.         } 
  20.         else 
  21.         { 
  22.             SetCursor(LoadCursor(NULL, IDC_NO)); 
  23.         } 
  24.     } 
  25.      
  26.     m_pDragImage->DragShowNolock(true); 
  27.  
  28.     CDialog::OnMouseMove(nFlags, point); 
void CDragListDlg::OnMouseMove(UINT nFlags, CPoint point) 
{
	if (m_Draging)
	{
		CPoint pt(point);
		ClientToScreen(&pt);
		m_pDragImage->DragMove(pt);
		m_pDragImage->DragShowNolock(false);

		CWnd *pWnd = WindowFromPoint(pt);
		if (pWnd->IsKindOf(RUNTIME_CLASS(CListCtrl)))
		{
			SetCursor(LoadCursor(NULL, IDC_ARROW));
			CListCtrl* pList = (CListCtrl*)pWnd;

			ScreenToClient(&pt);
			m_nDropIndex = pList->HitTest(pt);
			pList->RedrawItems(m_nDropIndex, m_nDropIndex);
			pList->UpdateWindow();
		}
		else
		{
			SetCursor(LoadCursor(NULL, IDC_NO));
		}
	}
	
	m_pDragImage->DragShowNolock(true);

	CDialog::OnMouseMove(nFlags, point);
}

  其实拖动操作移动的实现图标显示和鼠标图标的显示,当鼠标移出当前List我们现有显示为不可用的图标,并在移动的过程中保存m_nDropIndex

    第三步:这一步是最后的一步,我们需要结束拖动图标的显示,删除被拖动的item,并插入到新的位置;这里要注意的是被删除item的数据的保存,以及插入位置的微妙变化;实现代码如下

  1. void CDragListDlg::OnLButtonUp(UINT nFlags, CPoint point)  
  2.     if (m_Draging) 
  3.     { 
  4.         ReleaseCapture(); 
  5.         m_Draging = false
  6.  
  7.         if (m_nDropIndex == -1) m_nDropIndex = m_list.GetItemCount() - 1; 
  8.  
  9.         m_pDragImage->DragLeave(GetDesktopWindow()); 
  10.         m_pDragImage->EndDrag(); 
  11.         delete m_pDragImage; 
  12.  
  13.         CPoint pt(point); 
  14.         ClientToScreen(&pt); 
  15.         CWnd *pWnd = WindowFromPoint(pt); 
  16.  
  17.         if (pWnd == &m_list) 
  18.         { 
  19.             char strCol[2][256]; 
  20.  
  21.             m_list.GetItemText(m_nDragIndex, 0, strCol[0], 255);//得到该行的第二列 
  22.             m_list.GetItemText(m_nDragIndex, 1, strCol[1], 255);//得到该行的第二列 
  23.  
  24.             m_list.DeleteItem(m_nDragIndex); 
  25.             if(m_nDragIndex < m_nDropIndex) m_nDropIndex--; 
  26.  
  27.             m_list.InsertItem(m_nDropIndex, strCol[0]); 
  28.             m_list.SetItemText(m_nDropIndex, 1, strCol[2]); 
  29.  
  30.             m_list.SetItemState (m_nDropIndex, LVIS_SELECTED, LVIS_SELECTED); 
  31.         } 
  32.     } 
  33.      
  34.     CDialog::OnLButtonUp(nFlags, point); 
void CDragListDlg::OnLButtonUp(UINT nFlags, CPoint point) 
{
	if (m_Draging)
	{
		ReleaseCapture();
		m_Draging = false;

		if (m_nDropIndex == -1) m_nDropIndex = m_list.GetItemCount() - 1;

		m_pDragImage->DragLeave(GetDesktopWindow());
		m_pDragImage->EndDrag();
		delete m_pDragImage;

		CPoint pt(point);
		ClientToScreen(&pt);
		CWnd *pWnd = WindowFromPoint(pt);

		if (pWnd == &m_list)
		{
			char strCol[2][256];

			m_list.GetItemText(m_nDragIndex, 0, strCol[0], 255);//得到该行的第二列
			m_list.GetItemText(m_nDragIndex, 1, strCol[1], 255);//得到该行的第二列

			m_list.DeleteItem(m_nDragIndex);
			if(m_nDragIndex < m_nDropIndex) m_nDropIndex--;

			m_list.InsertItem(m_nDropIndex, strCol[0]);
			m_list.SetItemText(m_nDropIndex, 1, strCol[2]);

			m_list.SetItemState (m_nDropIndex, LVIS_SELECTED, LVIS_SELECTED);
		}
	}
	
	CDialog::OnLButtonUp(nFlags, point);
}


以上是我个人用来实现CListCtrl内部拖放的实现方法,其中可能有很多地方做的不是很好(例如保存被拖动item的数据),这里主要是方法,这样的原理我们也可以实现不同CListCtrl之间的拖放,或者CListCtrl和CTreeCtrl之间的拖放。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值