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

版权声明:本文为博主原创文章,欢迎交流学习 https://blog.csdn.net/u012081284/article/details/71545505

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

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

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

因为通过鼠标拖动滚动条,实际内部用到了定时器,具体有兴趣的可以查看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就好了。

扫描二维码关注公众号,回复: 3939611 查看本文章
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;
}

另一个原因:滚动条上层可能有其他控件,把滚动条盖住了,上层控件不设置背景,界面上直接看不出来,但是会影响鼠标的事件响应。

猜你喜欢

转载自blog.csdn.net/u012081284/article/details/71545505