本文由作者鄒啟文授權(quán)網(wǎng)易云社區(qū)發(fā)布。
在郵箱大師PC版中,我們需要實(shí)現(xiàn)一個(gè)功能:賬號(hào)和郵件夾拖拽排序。
準(zhǔn)備?
封裝win32 API。我們使用到的API有,?ImageList_Create、ImageList_Destroy、?ImageList_Add、ImageList_AddMasked、ImageList_Remove、?ImageList_BeginDrag、ImageList_DragEnter、ImageList_EndDrag、ImageList_DragLeave、?ImageList_DragMove、ImageList_SetDragCursorImage、ImageList_DragShowNoLock。?
For more information,see?Using?Image?ListsDrag開始?
攔截鼠標(biāo)的LButtonDown消息,判斷point所在的控件是否符合拖拽要求,如果符合,記住狀態(tài)(這里使用bool will_drag_和MailFolderNode* drag_from_標(biāo)識(shí)可以拖拽,使用bool is_draging_標(biāo)識(shí)正在拖拽中,使用MailFolderNode* drag_to_標(biāo)識(shí)接受控件)Drag進(jìn)行中?
攔截鼠標(biāo)的MouseMove消息,判斷will_drag_,如果是,那么需要做如下操作:?
->ImageList_Create創(chuàng)建ImageList對(duì)象?
->ImageList_AddMasked加入拖拽時(shí)將要顯示的圖像?
->ImageList_BeginDrag即將開始拖拽,并設(shè)置鼠標(biāo)在拖拽圖像中的位置?
->ImageList_DragEnter進(jìn)入拖拽,并設(shè)置拖拽圖像的位置?
->修改will_drag_為false,is_draging_為true,保證上面幾步只做一次?
->判斷point是否在拖拽接受范圍內(nèi),以及point對(duì)應(yīng)控件是否接受拖拽(及時(shí)更新drag_to_)?
->如果是,那么執(zhí)行ImageList_DragMove、ImageList_DragShowNoLock(TRUE)、SetCursor(LoadCursor(NULL, IDC_ARROW));?
->如果不是,那么執(zhí)行ImageList_DragShowNoLock(FALSE)、SetCursor(LoadCursor(NULL, IDC_NO));Drag結(jié)束?
攔截鼠標(biāo)的LButtonUp消息,如果is_draging_,那么執(zhí)行ImageList_EndDrag、ImageList_DragLeave、ImageList_Destroy,然后根據(jù)drag_from_和drag_to_處理本次拖拽操作。-
問題?
Ⅰ、如何使拖拽圖像背景透明??
使用CreateCompatibleBitmap創(chuàng)建位圖,調(diào)用FillRect將位圖背景刷成白色RGB(255,255,255);?
在ImageList_Create時(shí)指定ILC_COLOR32 | ILC_MASK;?
調(diào)用ImageList_AddMasked(bitmap, RGB(255,255,255));?
至此,拖拽圖像中的白色會(huì)與mask'中和'?
Ⅱ、在拖拽時(shí)出現(xiàn)窗口繪制被'破壞',并且殘留痕跡??
這是由于在調(diào)用ImageList_DragEnter時(shí)鎖定了窗口導(dǎo)致,我們使用NULL代替HWND即可解決此問題。同時(shí),由于鎖定導(dǎo)致了我們?cè)谕献r(shí)對(duì)窗口的繪制無法生效的問題亦可解決。另,記得在拖拽結(jié)束時(shí)ImageList_DragLeave(NULL);?
Ⅲ、ImageList_BeginDrag和ImageList_DragEnter中的位置是何含義??
在ImageList_BeginDrag中,此處位置為鼠標(biāo)圖標(biāo)相對(duì)于拖拽圖像的位置。?
在mageList_DragEnter中,此處位置為鼠標(biāo)相對(duì)于HWND的位置,如果HWND為NULL,那么便是相對(duì)于屏幕的位置。?
Ⅳ、ImageList_DragShowNoLock含義是什么??
參數(shù)BOOL表示是否顯示拖拽圖像。當(dāng)拖拽move在可接受控件時(shí),顯示拖拽鼠標(biāo),更改鼠標(biāo)樣式(或使用ImageList_SetDragCursorImage設(shè)置自定義鼠標(biāo)樣式,注意此時(shí)需隱藏ShowCursor(FALSE));否則,不顯示拖拽圖像,并顯示IDC_NO不可操作樣式。?
Ⅴ、will_drag_的標(biāo)識(shí)是否多余??
為何要等到MouseMove再去真正開始drag?因?yàn)檎5膯螕舨僮鳎覀儾幌M吹酵献D像,因此設(shè)置此標(biāo)記。實(shí)際應(yīng)用中發(fā)現(xiàn),有時(shí)候單擊時(shí)發(fā)生'抖動(dòng)',此時(shí)也出現(xiàn)拖拽圖像,因此我們加入了延時(shí),在按下100ms后才將will_drag_設(shè)置成true,可在一定程度上減小抖動(dòng)出現(xiàn)拖拽圖像的情況。
免費(fèi)領(lǐng)取驗(yàn)證碼、內(nèi)容安全、短信發(fā)送、直播點(diǎn)播體驗(yàn)包及云服務(wù)器等套餐
更多網(wǎng)易技術(shù)、產(chǎn)品、運(yùn)營(yíng)經(jīng)驗(yàn)分享請(qǐng)?jiān)L問網(wǎng)易云社區(qū)。
相關(guān)文章:
【推薦】?Docker小白使用筆記
【推薦】?多大規(guī)模的數(shù)據(jù)才值得用大數(shù)據(jù)的方式來處理?
【推薦】?線上日志集中化可視化管理:ELK