最近在做一个好友列表,通过查找资料自己现在了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之间的拖放。