CListCtrl 列表项拖放的实现

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

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

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

private:
	BOOL m_Draging;
	int m_nDragIndex;
	int m_nDropIndex;
	CImageList *m_pDragImage;

 

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

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()
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消息,函数的代码如下

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的数据的保存,以及插入位置的微妙变化;实现代码如下

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之间的拖放。

你可能感兴趣的:(CListCtrl,拖放的实现,LVN_BEGINDRAG)