今天在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窗口的代碼如下:
{
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:
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)。
測試代碼下載: source code
http://www.cppblog.com/weiym/archive/2012/05/03/173608.html
在Win7的Aero模式下因為桌面的窗口層次有變化, 上面的代碼會找不到窗口"SHELLDLL_DefView"窗口
簡單改了下,未完全測試:
HWND GetShellDLLForWin7()
{
HWND hWndShell = NULL;
HWND hWndDesktop = GetDesktopWindow();
if(hWndDesktop == NULL) return NULL;
HWND hWndWorkerW = NULL;
while(TRUE)
{
hWndWorkerW = FindWindowEx(hWndDesktop, hWndWorkerW, _T("WorkerW"), NULL);
if(hWndWorkerW == NULL) break;
if(::GetWindowLong(hWndWorkerW, GWL_STYLE) & WS_VISIBLE)
{
break;
}
else
{
continue;
}
}
if(hWndWorkerW != NULL)
{
hWndShell = FindWindowEx(hWndWorkerW, NULL, _T("SHELLDLL_DefView"), NULL);
}
return hWndShell;
}
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);
//Win7
if(hWndShellDLL == NULL)
{
hWndShellDLL = GetShellDLLForWin7();
}
}
if(hWndShellDLL != NULL
&& hWndShellDLL != s_hWndOldParent)
{
SetWindowLong(hWnd, GWL_HWNDPARENT, (LONG)hWndShellDLL);
s_hWndOldParent = hWndShellDLL;
return TRUE;
}
return FALSE;
}
http://www.cppblog.com/weiym/archive/2012/05/03/173608.html