光标闪烁问题的解决办法
在調(diào)用Windows API函數(shù)SetCursor設(shè)置光標(biāo)時(shí),可能會(huì)碰到閃爍的問題:移動(dòng)鼠標(biāo),光標(biāo)在Class Cursor(即注冊(cè)窗口類時(shí)指定的Cursor)與預(yù)設(shè)Cursor之間閃爍。
在MSDN上有關(guān)SetCursor函數(shù)的備注中強(qiáng)調(diào),如果Class Cursor非空,那么每當(dāng)鼠標(biāo)移動(dòng),系統(tǒng)都會(huì)把光標(biāo)恢復(fù)為Class Cursor。為了避免光標(biāo)閃爍這一問題,必須處理WM_SETCURSOR消息。(MSDN說明)
下面是一個(gè)例子:程序在主窗口視圖的中間位置繪制RGB條帶,當(dāng)鼠標(biāo)移動(dòng)在條帶范圍就將光標(biāo)設(shè)置成為Cross,此外根據(jù)光標(biāo)的位置,在RGB條帶上方30px處顯示所處條帶的顏色。程序運(yùn)行起來像這樣:
如果在WM_MOUSEMOVE的消息處理中判斷光標(biāo)的位置并設(shè)置光標(biāo)的話,就會(huì)碰到所說的光標(biāo)閃爍問題。WM_MOUSEMOVE的消息處理如下代碼所示:
LRESULT OnMouseMove(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM lParam, BOOL& /*bHandled*/){POINT ptCursor = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };RECT rect, rectText;get_rects(&rect, &rectText);InvalidateRect(&rectText);UpdateWindow();if (::PtInRect(&rect, ptCursor)) {::SetCursor(m_cursor);int dx = (rect.right - rect.left) / 3;LPTSTR ppsz[] = { _T("Red"), _T("Green"), _T("Blue") };int index;if (ptCursor.x - rect.left < dx)index = 0;else if (ptCursor.x - rect.left < 2 * dx)index = 1;else index = 2;WTL::CString str;str.Format(_T("Cursor on %s part"), ppsz[index]);CClientDC dc(m_hWnd);dc.DrawText(str, -1, &rectText, DT_CENTER | DT_VCENTER);}else ::SetCursor(CCursor().LoadSysCursor(IDC_ARROW));return 0;}閃爍產(chǎn)生的原因在于每次進(jìn)入OnMouseMove之前,系統(tǒng)都會(huì)先將光標(biāo)恢復(fù)成Arrow,進(jìn)入OnMouseMove之后,如果光標(biāo)處在RGB條帶范圍內(nèi)則立即被設(shè)置成Cross。
解決辦法就是將上面的判斷邏輯放在WM_SETCURSOR的消息處理中,當(dāng)然獲得光標(biāo)客戶坐標(biāo)的方式不同,代碼如下所示:
LRESULT OnSetCursor(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM lParam, BOOL& /*bHandled*/){POINT point;::GetCursorPos(&point);ScreenToClient(&point);set_cursor(point);return 0;}而代碼中的set_cursor私有方法其實(shí)就是上面的判斷邏輯,即:
// ptCursor: in client coordinatevoid set_cursor(POINT& ptCursor) throw(){RECT rect, rectText;get_rects(&rect, &rectText);InvalidateRect(&rectText);UpdateWindow();if (::PtInRect(&rect, ptCursor)) {::SetCursor(m_cursor);int dx = (rect.right - rect.left) / 3;LPTSTR ppsz[] = { _T("Red"), _T("Green"), _T("Blue") };int index;if (ptCursor.x - rect.left < dx)index = 0;else if (ptCursor.x - rect.left < 2 * dx)index = 1;else index = 2;WTL::CString str;str.Format(_T("Cursor on %s part"), ppsz[index]);CClientDC dc(m_hWnd);dc.DrawText(str, -1, &rectText, DT_CENTER | DT_VCENTER);}else ::SetCursor(CCursor().LoadSysCursor(IDC_ARROW));}這樣就解決了光標(biāo)閃爍的問題。本例的工程文件可在這里下載,它是基于WTL的SDI程序。
P.S. 另外一個(gè)問題雖然簡(jiǎn)單但仍值得一提。上面程序的截圖顯示的結(jié)果實(shí)際上并不準(zhǔn)確,Cross光標(biāo)的中心已經(jīng)處于Green條帶,但條帶上方顯示的卻是"Cursor on Red part"。造成此問題的原因是自定義光標(biāo)的熱點(diǎn)Hot Spot默認(rèn)值為(0, 0),而不是以中心+來標(biāo)識(shí)的。解決辦法是在VS資源編輯器中使用Hot Spot Tool,如下圖所示:
點(diǎn)擊Cross光標(biāo)的中心點(diǎn),熱點(diǎn)坐標(biāo)將變成你所設(shè)置的值,如下圖所示:
總結(jié)
以上是生活随笔為你收集整理的光标闪烁问题的解决办法的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 使用gdbserver远程调试
- 下一篇: Java如何判断今天本月第几周的周几?