您的位置:首页 > 产品设计 > UI/UE

duilib 滚动条不能拖动 问题处理

2017-05-10 16:51 225 查看
遇到过很多次群里朋友问:为什么滚动条不能拖动,点击两端的按钮可以滚动,通过鼠标滚轮也可以滚动,就是鼠标拖动时拖不动?

这是个提问次数较高的问题。

下面的内容只针对可能的原因中的一个,也是最可能的原因。

因为通过鼠标拖动滚动条,实际内部用到了定时器,具体有兴趣的可以查看duilib的源码。我们的程序在处理时常常也会用到WM_TIMER消息,通过拦截WM_TIMER消息来做一些定时器处理,不能拖动的原因很有可能就是我们拦截了定时器消息之后,不管触发这个消息的定时器ID是多少,我们统统都不再继续传递给duilib内部处理了。这样就造成了无法拖动。

下面我们来简单看下代码流程:

LRESULT CMainWnd::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
LRESULT lRes = 0;
BOOL bHandled = TRUE;

switch( uMsg )
{
case WM_CREATE:				lRes = OnCreate(uMsg, wParam, lParam, bHandled);		break;
case WM_NCHITTEST:			lRes = OnNcHitTest(uMsg, wParam, lParam, bHandled);		break;
case WM_SYSCOMMAND:			lRes = OnSysCommand(uMsg, wParam, lParam, bHandled);	break;
case WM_TIMER:				lRes = OnTimer(uMsg, wParam, lParam, bHandled);			break;
case WM_KEYDOWN:
{
if (wParam == VK_ESCAPE)
{
Close();
}
else
bHandled = FALSE;
}
default:
bHandled = FALSE;
}
if( bHandled ) return lRes;
if( m_PM.MessageHandler(uMsg, wParam, lParam, lRes) ) return lRes;
return CWindowWnd::HandleMessage(uMsg, wParam, lParam);
}
上面是一个常见的HandleMessage的过程(其他的消息拦截函数类似),在这里面我们拦截了WM_TIMER,这样的话,如果不去设置bHandled的值,那他因为无法走default的处理而自然变为TRUE,这样在switch的下面,可以看到:if(bHandled) return lRes;当bHandled为true时,本次处理就返回了,并没有继续给m_PM.MessageHandler和CWindowWnd::HandleMessage去继续处理此消息的机会,相当于WM_TIMER彻底拦截了,那么其他在你这个HandleMessage之后才有机会处理消息的地方,都因为无法收到此消息而再无机会处理了。

此时该怎么办?

很简单,针对定时器ID,根据需求拦截就行了。在OnTimer里面,凡是自己用到的定时器ID,又不想让它在其他地方也可能被处理,此时在将bHandled置为TRUE,其他时候都置为FALSE就好了。

LRESULT CMainWnd::OnTimer(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
{
if (wParam == TIMER_ID_MYUSE)
{
//do
OutputDebugString(_T("自己用的定时器"));
bHandled = TRUE;
}
else
bHandled = FALSE;
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: