桌面上嵌入窗口(桌面日歷)原理探索(將該窗口的Owner設置成桌面的Shell 窗口,可使用SetWindowLong更改窗口的GWL_HWNDPARENT,還要使用SetWindowPos設置Z-Order)


今天在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)。

測試代碼下載: 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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM