桌面上嵌入窗口(桌面日曆)原理探索

轉自:http://www.cppblog.com/weiym/archive/2012/05/03/173608.html
今天在QQ羣裏有人問怎樣實現將自己的窗口嵌入桌面,讓它和桌面融爲一體,就像很多桌面日曆軟件那樣。

我當時想到的就是建立一個Child Window,將他的父窗口設置成桌面Shell窗口就可以了。但是後來想想覺得不對,因爲很多桌面日曆窗口都有半透明和陰影效果,明顯是用Layered Window實現的,而大家知道Layered Window一定要用Pop Up Window才能實現的。

那麼如何用Pop up Window實現這種效果呢? 這裏關鍵的一點就是要將該窗口的Owner設置成桌面的Shell 窗口。

很多以爲Pop Up Window的Owner窗口只能在Create時關聯, 建立後沒法動態修改,實際上微軟是有接口讓我們改的,只是他們不建議我們動態改,因爲這樣會影響窗口的層次關係,尤其是對於Modal Dialog。

我們將窗口Owner改成桌面Shell窗口的代碼如下:
BOOL CheckParent(HWND hWnd)
{
static HWND s_hWndOldParent = NULL;

HWND hWndProgram = NULL;
HWND hWndShellDLL = NULL;
hWndProgram = FindWindow(_T(“Progman”), _T(“Program Manager”));
if(hWndProgram != NULL)
{
hWndShellDLL = FindWindowEx(hWndProgram, NULL, _T(“SHELLDLL_DefView”), NULL);
}

if(hWndShellDLL != NULL
&& hWndShellDLL != s_hWndOldParent)
{
SetWindowLong(hWnd, GWL_HWNDPARENT, (LONG)hWndShellDLL);
s_hWndOldParent = hWndShellDLL;
return TRUE;
}

return FALSE;
}

另外還有一個問題是一般Pop up窗口在Show出來時會顯示在最上面,而我們是要讓它顯示在最下面, 所以要設置下Z-Order:
hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);

CheckParent(hWnd);

SetWindowPos(hWnd, HWND_BOTTOM, 0, 0, 0, 0,
SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_SHOWWINDOW);

這樣上面的代碼就實現了的窗口永遠在桌面上,即使你點“顯示桌面”或是WIN+D,也不受影響。

另外,如果你要讓你的窗口在激活時也不會跑到其他窗口上面,只要創建時設置WS_EX_NOACTIVATE屬性就可以了。

還有個問題是桌面Shell有可能重啓,比如我們Kill掉Explorer.exe進程,所以我們最好一開始就啓一個定時器,然後不停調用CheckParent(HWND hWnd)。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章