正確使用 SetCapture ReleaseCapture [譯]「建议收藏」

正確使用 SetCapture ReleaseCapture [譯]「建议收藏」本文描述瞭如何正確處理自定義窗口和控件中的鼠標捕獲操作。原文鏈接: http://www.codeproject.com/Tips/127813/Using-SetCapture…correctly.aspx原作者: pasztorpisti轉載請註明出處:http://www.imoldman.com/2010/11/30/using-setcaptu…ture-corre

大家好,又见面了,我是你们的朋友全栈君。

本文描述瞭如何正確處理自定義窗口和控件中的鼠標捕獲操作。

原文鏈接: http://www.codeproject.com/Tips/127813/Using-SetCapture…correctly.aspx
原作者: pasztorpisti
轉載請註明出處:http://www.imoldman.com/2010/11/30/ using-setcaptu…ture-correctly

鼠標捕獲是Windows的一項特性。即使光標不在某個窗口或者控件(帶HWND句柄)內,它也可以將所有的鼠標消息和該指定的窗口相關聯。現假定你已經熟悉了這項特性及其相關API(包括SetCapture()函數, ReleaseCapture()函數及WM_CAPTURECHANGED消息),在這裡,我想告訴你一項開發人員經常犯的錯誤。

有些窗口和控件為了完成某項操作,有時需要捕獲鼠標。spy++程序上的窗口選擇器(即那個十字)是一個很好的例子,它允許你拖動一個十字光標到你桌面的某個窗口上,這樣你就可以選中它。

該文給出的示例程序創建了一個窗口,你可以拖動它的客戶區來移動它。只要在DefWindowProc()響應WM_NCHITTEST消息時返回HTCLIENT,就可以達到這種效果,但是這樣主循環就不工作了,就好像是你在拖拽著它的標題欄一樣。某些程序(如媒體播放器,遊戲)通常自繪整個窗口,並且以該示例代碼中的方式提供拖拽窗口的功能。這樣做,程序的主循環可以一直在運作。這樣的窗口移動功能需要捕獲鼠標,那時因為如果你按下了鼠標左鍵,然後把鼠標從客戶區猛地拉到外面去,窗口仍然要能夠接收到WM_MOUSEMOVE消息。

那麼這種“抓著客戶區移動窗口”操作的執行步驟是怎樣的呢?

  1. WM_LBUTTONDOWN,開始拖拽操作並且捕獲鼠標。
  2. WM_MOUSEMOVE,隨著光標一起移動窗口。
  3. WM_LBUTTONUP,僅僅釋放捕獲的鼠標。
  4. WM_CAPTURECHANGED,結束拖拽操作。

這篇文章的重點就是你要在WM_LBUTTONUPWM_CAPTURECHANGED消息響應中處理什麼。這裡重要的一點就是WM_LBUTTONUP僅僅釋放鼠標,只有WM_CAPTURECHANGED才會中止拖拽操作!之所以這樣,原因在於拖拽操作可以被其他操作終止掉,比如說你按了ALT+TAB組合鍵。這種情況下,你的程序根本接收不到WM_LBUTTONUP消息,但是仍然失去了鼠標捕獲,此時,窗口會接收到WM_CAPTURECHANGED消息,於是整個拖拽操作結束。

如果你把“拖拽操作結束”的相關代碼放在了WM_LBUTTONUP消息響應裡,一旦用戶按了ALT+TAB,你的程序會失去鼠標捕獲,此時另外一個窗口會成為前景窗口,你整個窗口的邏輯狀態就會亂掉,因為它還認為它是在拖動過程中。

所以結論就是,總是將操作結束的處理代碼放在WM_CAPTURECHANGED消息響應裡,並且在其他你想結束操作的地方調用ReleaseCapture(),這可以發生在任何地方,比如在WM_LBUTTONUP消息響應函數中,在WM_MOUSEMOVE消息響應函數中,等等。

編譯運行我給出的代碼,在拖拽主窗口客戶區的過程中,使用ALT+TAB按鍵將一個大些的窗口提到前面,這樣示例程序的主窗口就會全部被蓋住。之後釋放鼠標按鍵並且切回到示例程序,將鼠標在示例窗口上移動,一切正常。然後將WM_LBUTTONUPWM_CAPTURECHANGED消息處理代碼註釋掉,並且去掉有問題的代碼的註釋,再按照步驟試試ALT+TAB!這次,在ALT+TAB切換窗口、釋放鼠標按鍵並且使用ALT+TAB再切回我們的窗口後,將鼠標在示例窗口上移動,並且嘗試很快的速度移動光標,此時你會發現窗口的行為很瘋狂,除非你在窗口上單擊一下給它發個WM_LBUTTONUP消息,它才能回歸正常。

#include <windows.h>

HINSTANCE g_hInstance = (HINSTANCE)GetModuleHandle(NULL);
HWND g_hMainWnd = NULL;
bool g_MovingMainWnd = false;
POINT g_OrigCursorPos;
POINT g_OrigWndPos;
LRESULT CALLBACK MainWndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
	switch (uMsg)
	{
	case WM_LBUTTONDOWN:
		// here you can add extra check and decide whether to start
		// the window move or not
		if (GetCursorPos(&g_OrigCursorPos))
		{
			RECT rt;
			GetWindowRect(hWnd, &rt);
			g_OrigWndPos.x = rt.left;
			g_OrigWndPos.y = rt.top;
			g_MovingMainWnd = true;
			SetCapture(hWnd);
		}
		return 0;
	// THE RIGHT WAY OF DOING IT:
	//*
	case WM_LBUTTONUP:
		ReleaseCapture();
		return 0;
	case WM_CAPTURECHANGED:
		g_MovingMainWnd = (HWND)lParam == hWnd;
		return 0;
	/**/
	// THE WRONG WAY OF DOING IT:
	/*
	case WM_LBUTTONUP:
		g_MovingMainWnd = false;
		ReleaseCapture();
		return 0;
	// buggy programs usually do not handle WM_CAPTURECHANGED at all
	case WM_CAPTURECHANGED:
		break;
	/**/
	case WM_MOUSEMOVE:
		if (g_MovingMainWnd)
		{
			POINT pt;
			if (GetCursorPos(&pt))
			{
				int wnd_x = g_OrigWndPos.x +
				  (pt.x - g_OrigCursorPos.x);
				int wnd_y = g_OrigWndPos.y +
				  (pt.y - g_OrigCursorPos.y);
				SetWindowPos(hWnd, NULL, wnd_x,
				  wnd_y, 0, 0, SWP_NOACTIVATE|
				  SWP_NOOWNERZORDER|SWP_NOZORDER|
				  SWP_NOSIZE);
			}
		}
		return 0;
	case WM_DESTROY:
		PostQuitMessage(0);
		return 0;
	}
	return DefWindowProc(hWnd, uMsg, wParam, lParam);
}
bool CreateMainWnd()
{
	static const char CLASS_NAME[] = "MainWndClass";
	WNDCLASS wc;
	wc.cbClsExtra = 0;
	wc.cbWndExtra = 0;
	wc.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
	wc.hCursor = LoadCursor(NULL, IDC_ARROW);
	wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
	wc.hInstance = g_hInstance;
	wc.lpfnWndProc = &MainWndProc;
	wc.lpszClassName = CLASS_NAME;
	wc.lpszMenuName = NULL;
	wc.style = CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS;
	if (!RegisterClass(&wc))
		return false;
	g_hMainWnd = CreateWindowEx(
		0,
		CLASS_NAME,
		"Main Window",
		WS_OVERLAPPED | WS_SYSMENU | WS_MINIMIZEBOX | WS_MAXIMIZEBOX,
		CW_USEDEFAULT, CW_USEDEFAULT, 300, 200,
		NULL,
		NULL,
		g_hInstance,
		NULL
		);
	return true;
}
int main()
{
	if (!CreateMainWnd())
		return -1;
	ShowWindow(g_hMainWnd, SW_SHOW);
	UpdateWindow(g_hMainWnd);
	MSG msg;
	while (GetMessage(&msg, NULL, 0, 0))
	{
		TranslateMessage(&msg);
		DispatchMessage(&msg);
	}
	return (int)msg.wParam;
}

ps:這篇文章是昨天看到的,當時感覺寫的挺好的(其實主要是因為短
害羞),所以想拿來翻譯下,一來鍛煉下自己的英語水平,二來充實下自己的blog,跑去CPOL看了下,沒有提到翻譯相關的內容,然後給作者留了言,問他能不能翻譯下,人家說可以,不過你還是問下CodeProject的工作人員吧,然後屁顛屁顛的跑去問,結果被告知他們沒權利允許我翻譯拿來貼自己的blog上,因為文章的版權是原作者的,我想這好辦了,原作者都同意了,跟人家說下然後開始翻譯吧,結果在作者的About頁面愣是沒找到他的Email。於是在沒有得到最終許可的情況下,我翻譯了這篇文章,想來應該不會違反License的,如果需要轉載記得保留開始三個鏈接!

就這樣吧,謝謝原作者pasztorpisti和CodeProjcet平台微笑

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/141917.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • latex大的中括号_文献引用中括号怎么标注

    latex大的中括号_文献引用中括号怎么标注括号是数学中最常用的符号之一。括号不仅能使我们的公式更加美观,还能使我们的表达更为清晰、丰富。latex里面的括号和我们常见的括号是一样的,主要是小括号、中括号(或者叫方括号)和花括号。这里我们看到花括号如果直接打出来的话是不显示任何东西的,这里我们需要加一个转义符,也就是反斜杠。如果只用直接的括号字符,只能打出固定大小的括号,比如这样打出的括号就比较小。那么如何打出更大一点的括号呢?在latex…

    2022年10月10日
  • java中高级面试题总结(全面)_java高级面试题大汇总

    java中高级面试题总结(全面)_java高级面试题大汇总参考了网上的面试题,整理了一份面试题的资料。String,StringBuffer,StringBuilder的区别是什么?String为什么是不可变的?String是字符串常量,后两者是字符串变量。其中,StringBuffer是线程安全的,而StringBuilder是非线程安全的,线程安全会带来额外的开销,所以StringBuilde

  • 来谈谈MySQL事务及事务引发的问题

    点击上方“全栈程序员社区”,星标公众号 重磅干货,第一时间送达 作者:王啸tr1912 blog.csdn.net/tr1912/article/details/81988459 …

  • 强化学习(Q-Learning,Sarsa)

    强化学习(Q-Learning,Sarsa)ReinforcementLearning监督学习–>非监督学习–>强化学习。监督学习:拥有“标签”可监督算法不断调整模型,得到输入与输出的映射函数。非监督学习:无“标签”,通过分析数据本身进行建模,发掘底层信息和隐藏结构。但是1.标签需要花大量的代价进行收集,在有些情况如子任务的组合数特别巨大寻找监督项是不切实际的。2.如何更好的理解数据,学习到具体的映射而不仅仅是数据的底…

  • APP应用平台有哪些?

    APP应用平台有哪些?1、小米应用商店小米开放平台网站:https://account.xiaomi.com注册帐号教程地址:http://dev.xiaomi.com/doc/?p=90应用提交流程:http://dev.xiaomi.com/doc/?p=1292、360手机助手360开放平台地址:http://i.360.cn/注册及应用提交流程教程地址:http:…

  • xp的终极优化

    xp的终极优化总体设想:让WinXP更苗条、性感、速度更快,使用更便捷。为了达到这个目的,我们主要从四个方面入手:1、减少磁盘空间占用2、终止不常用的系统服务3、安全问题4、另外一些技巧首先问一下,你是不是很想激活XP,不。。。准确的说你是不是想在ms的站上能够升级。如果答案是肯定的话,那我们就先来探讨一下安装的问题,目前流行的V4、V5、V6版本我还是比较推荐的,尤其是V5和V6这两个。安装的过程中有个序

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号