mirror of
https://github.com/reactos/reactos.git
synced 2024-11-06 22:52:46 +00:00
527f2f9057
* Create a branch for some evul shell experiments. svn path=/branches/shell-experiments/; revision=61927
1071 lines
33 KiB
C
1071 lines
33 KiB
C
/*
|
|
* ReactOS Explorer
|
|
*
|
|
* Copyright 2006 - 2007 Thomas Weidenmueller <w3seek@reactos.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
#include <shdeprecated.h>
|
|
|
|
#include "undoc.h"
|
|
|
|
/*****************************************************************************
|
|
** ITrayBandSite ************************************************************
|
|
*****************************************************************************/
|
|
|
|
static const ITrayBandSiteVtbl ITrayBandSiteImpl_Vtbl;
|
|
static const IBandSiteVtbl IBandSiteImpl_Vtbl;
|
|
|
|
typedef struct
|
|
{
|
|
const ITrayBandSiteVtbl *lpVtbl;
|
|
const IBandSiteVtbl *lpBandSiteVtbl;
|
|
LONG Ref;
|
|
|
|
ITrayWindow *Tray;
|
|
|
|
IUnknown *punkInner;
|
|
IBandSite *BandSite;
|
|
ITaskBand *TaskBand;
|
|
IWinEventHandler *WindowEventHandler;
|
|
IContextMenu *ContextMenu;
|
|
|
|
HWND hWndRebar;
|
|
|
|
union
|
|
{
|
|
DWORD dwFlags;
|
|
struct
|
|
{
|
|
DWORD Locked : 1;
|
|
};
|
|
};
|
|
} ITrayBandSiteImpl;
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_Update(IN OUT ITrayBandSiteImpl *This);
|
|
|
|
static IUnknown *
|
|
IUnknown_from_ITrayBandSiteImpl(ITrayBandSiteImpl *This)
|
|
{
|
|
return (IUnknown *)&This->lpVtbl;
|
|
}
|
|
|
|
IMPL_CASTS(ITrayBandSite, ITrayBandSite, lpVtbl)
|
|
IMPL_CASTS(IBandSite, ITrayBandSite, lpBandSiteVtbl)
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_AddRef(IN OUT ITrayBandSite *iface)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
|
|
return InterlockedIncrement(&This->Ref);
|
|
}
|
|
|
|
static VOID
|
|
ITrayBandSiteImpl_Free(IN OUT ITrayBandSiteImpl *This)
|
|
{
|
|
if (This->BandSite != NULL)
|
|
{
|
|
IBandSite_Release(This->BandSite);
|
|
This->BandSite = NULL;
|
|
}
|
|
|
|
if (This->WindowEventHandler != NULL)
|
|
{
|
|
IWinEventHandler_Release(This->WindowEventHandler);
|
|
This->WindowEventHandler = NULL;
|
|
}
|
|
|
|
if (This->ContextMenu != NULL)
|
|
{
|
|
IContextMenu_Release(This->ContextMenu);
|
|
This->ContextMenu = NULL;
|
|
}
|
|
|
|
if (This->punkInner != NULL)
|
|
{
|
|
IUnknown_Release(This->punkInner);
|
|
This->punkInner = NULL;
|
|
}
|
|
|
|
HeapFree(hProcessHeap,
|
|
0,
|
|
This);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_Release(IN OUT ITrayBandSite *iface)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
ULONG Ret;
|
|
|
|
Ret = InterlockedDecrement(&This->Ref);
|
|
|
|
if (Ret == 0)
|
|
ITrayBandSiteImpl_Free(This);
|
|
|
|
return Ret;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_QueryInterface(IN OUT ITrayBandSite *iface,
|
|
IN REFIID riid,
|
|
OUT LPVOID *ppvObj)
|
|
{
|
|
ITrayBandSiteImpl *This;
|
|
|
|
if (ppvObj == NULL)
|
|
return E_POINTER;
|
|
|
|
This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
|
|
if (IsEqualIID(riid,
|
|
&IID_IUnknown) ||
|
|
IsEqualIID(riid,
|
|
&IID_IBandSiteStreamCallback))
|
|
{
|
|
/* NOTE: IID_IBandSiteStreamCallback is queried by the shell, we
|
|
implement this interface directly */
|
|
*ppvObj = IUnknown_from_ITrayBandSiteImpl(This);
|
|
}
|
|
else if (IsEqualIID(riid,
|
|
&IID_IBandSite))
|
|
{
|
|
*ppvObj = IBandSite_from_ITrayBandSiteImpl(This);
|
|
}
|
|
else if (IsEqualIID(riid,
|
|
&IID_IWinEventHandler))
|
|
{
|
|
DbgPrint("ITaskBandSite: IWinEventHandler queried!\n");
|
|
*ppvObj = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
else if (This->punkInner != NULL)
|
|
{
|
|
return IUnknown_QueryInterface(This->punkInner,
|
|
riid,
|
|
ppvObj);
|
|
}
|
|
else
|
|
{
|
|
*ppvObj = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
ITrayBandSiteImpl_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_OnLoad(IN OUT ITrayBandSite *iface,
|
|
IN OUT IStream *pStm,
|
|
IN REFIID riid,
|
|
OUT PVOID *pvObj)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
LARGE_INTEGER liPosZero;
|
|
ULARGE_INTEGER liCurrent;
|
|
CLSID clsid;
|
|
ULONG ulRead;
|
|
HRESULT hRet;
|
|
|
|
/* NOTE: Callback routine called by the shell while loading the task band
|
|
stream. We use it to intercept the default behavior when the task
|
|
band is loaded from the stream.
|
|
|
|
NOTE: riid always points to IID_IUnknown! This is because the shell hasn't
|
|
read anything from the stream and therefore doesn't know what CLSID
|
|
it's dealing with. We'll have to find it out ourselves by reading
|
|
the GUID from the stream. */
|
|
|
|
/* Read the current position of the stream, we'll have to reset it everytime
|
|
we read a CLSID that's not the task band... */
|
|
ZeroMemory(&liPosZero,
|
|
sizeof(liPosZero));
|
|
hRet = IStream_Seek(pStm,
|
|
liPosZero,
|
|
STREAM_SEEK_CUR,
|
|
&liCurrent);
|
|
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* Now let's read the CLSID from the stream and see if it's our task band */
|
|
#if defined(IStream_Read)
|
|
hRet = IStream_Read(pStm,
|
|
&clsid,
|
|
sizeof(clsid),
|
|
&ulRead);
|
|
#else
|
|
ulRead = sizeof(clsid);
|
|
hRet = IStream_Read(pStm,
|
|
&clsid,
|
|
sizeof(clsid));
|
|
#endif
|
|
if (SUCCEEDED(hRet) && ulRead == sizeof(clsid))
|
|
{
|
|
if (IsEqualGUID(&clsid,
|
|
&CLSID_ITaskBand))
|
|
{
|
|
ASSERT(This->TaskBand != NULL);
|
|
/* We're trying to load the task band! Let's create it... */
|
|
|
|
hRet = ITaskBand_QueryInterface(This->TaskBand,
|
|
riid,
|
|
pvObj);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* Load the stream */
|
|
DbgPrint("IBandSiteStreamCallback::OnLoad intercepted the task band CLSID!\n");
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Reset the position and let the shell do all the work for us */
|
|
hRet = IStream_Seek(pStm,
|
|
*(LARGE_INTEGER*)&liCurrent,
|
|
STREAM_SEEK_SET,
|
|
NULL);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* Let the shell handle everything else for us :) */
|
|
hRet = OleLoadFromStream(pStm,
|
|
riid,
|
|
pvObj);
|
|
}
|
|
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
DbgPrint("IBandSiteStreamCallback::OnLoad(0x%p, 0x%p, 0x%p) returns 0x%x\n", pStm, riid, pvObj, hRet);
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_OnSave(IN OUT ITrayBandSite *iface,
|
|
IN OUT IUnknown *pUnk,
|
|
IN OUT IStream *pStm)
|
|
{
|
|
/* NOTE: Callback routine called by the shell while saving the task band
|
|
stream. We use it to intercept the default behavior when the task
|
|
band is saved to the stream */
|
|
/* FIXME: Implement */
|
|
DbgPrint("IBandSiteStreamCallback::OnSave(0x%p, 0x%p) returns E_NOTIMPL\n", pUnk, pStm);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT
|
|
IsSameObject(IN IUnknown *punk1,
|
|
IN IUnknown *punk2)
|
|
{
|
|
HRESULT hRet;
|
|
|
|
hRet = IUnknown_QueryInterface(punk1,
|
|
&IID_IUnknown,
|
|
(PVOID*)&punk1);
|
|
if (!SUCCEEDED(hRet))
|
|
return hRet;
|
|
|
|
hRet = IUnknown_QueryInterface(punk2,
|
|
&IID_IUnknown,
|
|
(PVOID*)&punk2);
|
|
IUnknown_Release(punk1);
|
|
|
|
if (!SUCCEEDED(hRet))
|
|
return hRet;
|
|
|
|
IUnknown_Release(punk2);
|
|
|
|
/* We're dealing with the same object if the IUnknown pointers are equal */
|
|
return (punk1 == punk2) ? S_OK : S_FALSE;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_IsTaskBand(IN OUT ITrayBandSite *iface,
|
|
IN IUnknown *punk)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
return IsSameObject((IUnknown *)This->BandSite,
|
|
punk);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_ProcessMessage(IN OUT ITrayBandSite *iface,
|
|
IN HWND hWnd,
|
|
IN UINT uMsg,
|
|
IN WPARAM wParam,
|
|
IN LPARAM lParam,
|
|
OUT LRESULT *plResult)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
HRESULT hRet;
|
|
|
|
ASSERT(This->hWndRebar != NULL);
|
|
|
|
/* Custom task band behavior */
|
|
switch (uMsg)
|
|
{
|
|
case WM_NOTIFY:
|
|
{
|
|
const NMHDR *nmh = (const NMHDR *)lParam;
|
|
|
|
if (nmh->hwndFrom == This->hWndRebar)
|
|
{
|
|
switch (nmh->code)
|
|
{
|
|
case NM_NCHITTEST:
|
|
{
|
|
LPNMMOUSE nmm = (LPNMMOUSE)lParam;
|
|
|
|
if (nmm->dwHitInfo == RBHT_CLIENT || nmm->dwHitInfo == RBHT_NOWHERE ||
|
|
nmm->dwItemSpec == (DWORD_PTR)-1)
|
|
{
|
|
/* Make the rebar control appear transparent so the user
|
|
can drag the tray window */
|
|
*plResult = HTTRANSPARENT;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
case RBN_MINMAX:
|
|
/* Deny if an Administrator disabled this "feature" */
|
|
*plResult = (SHRestricted(REST_NOMOVINGBAND) != 0);
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
//DbgPrint("ITrayBandSite::ProcessMessage: WM_NOTIFY for 0x%p, From: 0x%p, Code: NM_FIRST-%u...\n", hWnd, nmh->hwndFrom, NM_FIRST - nmh->code);
|
|
break;
|
|
}
|
|
};
|
|
|
|
/* Forward to the shell's IWinEventHandler interface to get the default
|
|
shell behavior! */
|
|
if (This->WindowEventHandler != NULL)
|
|
{
|
|
/*DbgPrint("Calling IWinEventHandler::ProcessMessage(0x%p, 0x%x, 0x%p, 0x%p, 0x%p) This->hWndRebar=0x%p\n", hWnd, uMsg, wParam, lParam, plResult, This->hWndRebar);*/
|
|
hRet = IWinEventHandler_OnWinEvent(This->WindowEventHandler,
|
|
hWnd,
|
|
uMsg,
|
|
wParam,
|
|
lParam,
|
|
plResult);
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
if (uMsg == WM_NOTIFY)
|
|
{
|
|
const NMHDR *nmh = (const NMHDR *)lParam;
|
|
DbgPrint("ITrayBandSite->IWinEventHandler::ProcessMessage: WM_NOTIFY for 0x%p, From: 0x%p, Code: NM_FIRST-%u returned 0x%x\n", hWnd, nmh->hwndFrom, NM_FIRST - nmh->code, hRet);
|
|
}
|
|
else
|
|
{
|
|
DbgPrint("ITrayBandSite->IWinEventHandler::ProcessMessage(0x%p,0x%x,0x%p,0x%p,0x%p->0x%p) returned: 0x%x\n", hWnd, uMsg, wParam, lParam, plResult, *plResult, hRet);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
hRet = E_FAIL;
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_AddContextMenus(IN OUT ITrayBandSite *iface,
|
|
IN HMENU hmenu,
|
|
IN UINT indexMenu,
|
|
IN UINT idCmdFirst,
|
|
IN UINT idCmdLast,
|
|
IN UINT uFlags,
|
|
OUT IContextMenu **ppcm)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
IShellService *pSs;
|
|
HRESULT hRet;
|
|
|
|
if (This->ContextMenu == NULL)
|
|
{
|
|
/* Cache the context menu so we don't need to CoCreateInstance all the time... */
|
|
hRet = CoCreateInstance(&CLSID_IShellBandSiteMenu,
|
|
NULL,
|
|
CLSCTX_INPROC_SERVER,
|
|
&IID_IShellService,
|
|
(PVOID*)&pSs);
|
|
DbgPrint("CoCreateInstance(CLSID_IShellBandSiteMenu) for IShellService returned: 0x%x\n", hRet);
|
|
if (!SUCCEEDED(hRet))
|
|
return hRet;
|
|
|
|
hRet = IShellService_SetOwner(pSs,
|
|
IUnknown_from_ITrayBandSiteImpl(This));
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
IShellService_Release(pSs);
|
|
return hRet;
|
|
}
|
|
|
|
hRet = IShellService_QueryInterface(pSs,
|
|
&IID_IContextMenu,
|
|
(PVOID*)&This->ContextMenu);
|
|
|
|
IShellService_Release(pSs);
|
|
|
|
if (!SUCCEEDED(hRet))
|
|
return hRet;
|
|
}
|
|
|
|
if (ppcm != NULL)
|
|
{
|
|
IContextMenu_AddRef(This->ContextMenu);
|
|
*ppcm = This->ContextMenu;
|
|
}
|
|
|
|
/* Add the menu items */
|
|
return IContextMenu_QueryContextMenu(This->ContextMenu,
|
|
hmenu,
|
|
indexMenu,
|
|
idCmdFirst,
|
|
idCmdLast,
|
|
uFlags);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_Lock(IN OUT ITrayBandSite *iface,
|
|
IN BOOL bLock)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_ITrayBandSite(iface);
|
|
BOOL bPrevLocked = This->Locked;
|
|
BANDSITEINFO bsi;
|
|
HRESULT hRet;
|
|
|
|
ASSERT(This->BandSite != NULL);
|
|
|
|
if (bPrevLocked != bLock)
|
|
{
|
|
This->Locked = bLock;
|
|
|
|
bsi.dwMask = BSIM_STYLE;
|
|
bsi.dwStyle = (This->Locked ? BSIS_LOCKED | BSIS_NOGRIPPER : BSIS_AUTOGRIPPER);
|
|
|
|
hRet = IBandSite_SetBandSiteInfo(This->BandSite,
|
|
&bsi);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = ITrayBandSiteImpl_Update(This);
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
static const ITrayBandSiteVtbl ITrayBandSiteImpl_Vtbl =
|
|
{
|
|
/*** IUnknown methods ***/
|
|
ITrayBandSiteImpl_QueryInterface,
|
|
ITrayBandSiteImpl_AddRef,
|
|
ITrayBandSiteImpl_Release,
|
|
/*** IBandSiteStreamCallback methods ***/
|
|
ITrayBandSiteImpl_OnLoad,
|
|
ITrayBandSiteImpl_OnSave,
|
|
/*** ITrayBandSite methods ***/
|
|
ITrayBandSiteImpl_IsTaskBand,
|
|
ITrayBandSiteImpl_ProcessMessage,
|
|
ITrayBandSiteImpl_AddContextMenus,
|
|
ITrayBandSiteImpl_Lock
|
|
};
|
|
|
|
/*******************************************************************/
|
|
|
|
METHOD_IUNKNOWN_INHERITED_ADDREF(IBandSite, ITrayBandSite)
|
|
METHOD_IUNKNOWN_INHERITED_RELEASE(IBandSite, ITrayBandSite)
|
|
METHOD_IUNKNOWN_INHERITED_QUERYINTERFACE(IBandSite, ITrayBandSite)
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_AddBand(IN OUT IBandSite *iface,
|
|
IN IUnknown *punk)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
IOleCommandTarget *pOct;
|
|
HRESULT hRet;
|
|
|
|
hRet = IUnknown_QueryInterface(punk,
|
|
&IID_IOleCommandTarget,
|
|
(PVOID*)&pOct);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* Send the DBID_DELAYINIT command to initialize the band to be added */
|
|
/* FIXME: Should be delayed */
|
|
IOleCommandTarget_Exec(pOct,
|
|
&IID_IDeskBand,
|
|
DBID_DELAYINIT,
|
|
0,
|
|
NULL,
|
|
NULL);
|
|
|
|
IOleCommandTarget_Release(pOct);
|
|
}
|
|
|
|
return IBandSite_AddBand(This->BandSite,
|
|
punk);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_EnumBands(IN OUT IBandSite *iface,
|
|
IN UINT uBand,
|
|
OUT DWORD *pdwBandID)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
return IBandSite_EnumBands(This->BandSite,
|
|
uBand,
|
|
pdwBandID);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_QueryBand(IN OUT IBandSite *iface,
|
|
IN DWORD dwBandID,
|
|
OUT IDeskBand **ppstb,
|
|
OUT DWORD *pdwState,
|
|
OUT LPWSTR pszName,
|
|
IN int cchName)
|
|
{
|
|
HRESULT hRet;
|
|
IDeskBand *pstb = NULL;
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
|
|
hRet = IBandSite_QueryBand(This->BandSite,
|
|
dwBandID,
|
|
&pstb,
|
|
pdwState,
|
|
pszName,
|
|
cchName);
|
|
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = IsSameObject((IUnknown *)pstb,
|
|
(IUnknown *)This->TaskBand);
|
|
if (hRet == S_OK)
|
|
{
|
|
/* Add the BSSF_UNDELETEABLE flag to pdwState because the task bar band shouldn't be deletable */
|
|
if (pdwState != NULL)
|
|
*pdwState |= BSSF_UNDELETEABLE;
|
|
}
|
|
else if (!SUCCEEDED(hRet))
|
|
{
|
|
IDeskBand_Release(pstb);
|
|
pstb = NULL;
|
|
}
|
|
|
|
if (ppstb != NULL)
|
|
*ppstb = pstb;
|
|
}
|
|
else if (ppstb != NULL)
|
|
*ppstb = NULL;
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_SetBandState(IN OUT IBandSite *iface,
|
|
IN DWORD dwBandID,
|
|
IN DWORD dwMask,
|
|
IN DWORD dwState)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
return IBandSite_SetBandState(This->BandSite,
|
|
dwBandID,
|
|
dwMask,
|
|
dwState);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_RemoveBand(IN OUT IBandSite *iface,
|
|
IN DWORD dwBandID)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
return IBandSite_RemoveBand(This->BandSite,
|
|
dwBandID);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_GetBandObject(IN OUT IBandSite *iface,
|
|
IN DWORD dwBandID,
|
|
IN REFIID riid,
|
|
OUT VOID **ppv)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
return IBandSite_GetBandObject(This->BandSite,
|
|
dwBandID,
|
|
riid,
|
|
ppv);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_SetBandSiteInfo(IN OUT IBandSite *iface,
|
|
IN const BANDSITEINFO *pbsinfo)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
return IBandSite_SetBandSiteInfo(This->BandSite,
|
|
pbsinfo);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayBandSiteImpl_GetBandSiteInfo(IN OUT IBandSite *iface,
|
|
IN OUT BANDSITEINFO *pbsinfo)
|
|
{
|
|
ITrayBandSiteImpl *This = ITrayBandSiteImpl_from_IBandSite(iface);
|
|
return IBandSite_GetBandSiteInfo(This->BandSite,
|
|
pbsinfo);
|
|
}
|
|
|
|
static const IBandSiteVtbl IBandSiteImpl_Vtbl =
|
|
{
|
|
/*** IUnknown methods ***/
|
|
METHOD_IUNKNOWN_INHERITED_QUERYINTERFACE_NAME(IBandSite, ITrayBandSite),
|
|
METHOD_IUNKNOWN_INHERITED_ADDREF_NAME(IBandSite, ITrayBandSite),
|
|
METHOD_IUNKNOWN_INHERITED_RELEASE_NAME(IBandSite, ITrayBandSite),
|
|
/*** IBandSite methods ***/
|
|
ITrayBandSiteImpl_AddBand,
|
|
ITrayBandSiteImpl_EnumBands,
|
|
ITrayBandSiteImpl_QueryBand,
|
|
ITrayBandSiteImpl_SetBandState,
|
|
ITrayBandSiteImpl_RemoveBand,
|
|
ITrayBandSiteImpl_GetBandObject,
|
|
ITrayBandSiteImpl_SetBandSiteInfo,
|
|
ITrayBandSiteImpl_GetBandSiteInfo,
|
|
};
|
|
|
|
static BOOL
|
|
ITrayBandSiteImpl_HasTaskBand(IN OUT ITrayBandSiteImpl *This)
|
|
{
|
|
ASSERT(This->TaskBand != NULL);
|
|
|
|
return SUCCEEDED(ITaskBand_GetRebarBandID(This->TaskBand,
|
|
NULL));
|
|
}
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_AddTaskBand(IN OUT ITrayBandSiteImpl *This)
|
|
{
|
|
#if 0
|
|
/* FIXME: This is the code for the simple taskbar */
|
|
IObjectWithSite *pOws;
|
|
HRESULT hRet;
|
|
|
|
hRet = ITaskBand_QueryInterface(This->TaskBand,
|
|
&IID_IObjectWithSite,
|
|
(PVOID*)&pOws);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = IObjectWithSite_SetSite(pOws,
|
|
(IUnknown *)This->TaskBand);
|
|
|
|
IObjectWithSite_Release(pOws);
|
|
}
|
|
|
|
return hRet;
|
|
#else
|
|
if (!ITrayBandSiteImpl_HasTaskBand(This))
|
|
{
|
|
return IBandSite_AddBand(This->BandSite,
|
|
(IUnknown *)This->TaskBand);
|
|
}
|
|
|
|
return S_OK;
|
|
#endif
|
|
}
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_Update(IN OUT ITrayBandSiteImpl *This)
|
|
{
|
|
IOleCommandTarget *pOct;
|
|
HRESULT hRet;
|
|
|
|
hRet = IUnknown_QueryInterface(This->punkInner,
|
|
&IID_IOleCommandTarget,
|
|
(PVOID*)&pOct);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* Send the DBID_BANDINFOCHANGED command to update the band site */
|
|
hRet = IOleCommandTarget_Exec(pOct,
|
|
&IID_IDeskBand,
|
|
DBID_BANDINFOCHANGED,
|
|
0,
|
|
NULL,
|
|
NULL);
|
|
|
|
IOleCommandTarget_Release(pOct);
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static VOID
|
|
ITrayBandSiteImpl_BroadcastOleCommandExec(IN OUT ITrayBandSiteImpl *This,
|
|
const GUID *pguidCmdGroup,
|
|
DWORD nCmdID,
|
|
DWORD nCmdExecOpt,
|
|
VARIANTARG *pvaIn,
|
|
VARIANTARG *pvaOut)
|
|
{
|
|
IOleCommandTarget *pOct;
|
|
DWORD dwBandID;
|
|
UINT uBand = 0;
|
|
|
|
/* Enumerate all bands */
|
|
while (SUCCEEDED(IBandSite_EnumBands(This->BandSite,
|
|
uBand,
|
|
&dwBandID)))
|
|
{
|
|
if (SUCCEEDED(IBandSite_GetBandObject(This->BandSite,
|
|
dwBandID,
|
|
&IID_IOleCommandTarget,
|
|
(PVOID*)&pOct)))
|
|
{
|
|
/* Execute the command */
|
|
IOleCommandTarget_Exec(pOct,
|
|
pguidCmdGroup,
|
|
nCmdID,
|
|
nCmdExecOpt,
|
|
pvaIn,
|
|
pvaOut);
|
|
|
|
IOleCommandTarget_Release(pOct);
|
|
}
|
|
|
|
uBand++;
|
|
}
|
|
}
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_FinishInit(IN OUT ITrayBandSiteImpl *This)
|
|
{
|
|
/* Broadcast the DBID_FINISHINIT command */
|
|
ITrayBandSiteImpl_BroadcastOleCommandExec(This,
|
|
&IID_IDeskBand,
|
|
DBID_FINISHINIT,
|
|
0,
|
|
NULL,
|
|
NULL);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_Show(IN OUT ITrayBandSiteImpl *This,
|
|
IN BOOL bShow)
|
|
{
|
|
IDeskBarClient *pDbc;
|
|
HRESULT hRet;
|
|
|
|
hRet = IBandSite_QueryInterface(This->BandSite,
|
|
&IID_IDeskBarClient,
|
|
(PVOID*)&pDbc);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = IDeskBarClient_UIActivateDBC(pDbc,
|
|
bShow ? DBC_SHOW : DBC_HIDE);
|
|
IDeskBarClient_Release(pDbc);
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_LoadFromStream(IN OUT ITrayBandSiteImpl *This,
|
|
IN OUT IStream *pStm)
|
|
{
|
|
IPersistStream *pPStm;
|
|
HRESULT hRet;
|
|
|
|
ASSERT(This->BandSite != NULL);
|
|
|
|
/* We implement the undocumented COM interface IBandSiteStreamCallback
|
|
that the shell will query so that we can intercept and custom-load
|
|
the task band when it finds the task band's CLSID (which is internal).
|
|
This way we can prevent the shell from attempting to CoCreateInstance
|
|
the (internal) task band, resulting in a failure... */
|
|
hRet = IBandSite_QueryInterface(This->BandSite,
|
|
&IID_IPersistStream,
|
|
(PVOID*)&pPStm);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = IPersistStream_Load(pPStm,
|
|
pStm);
|
|
DbgPrint("IPersistStream_Load() returned 0x%x\n", hRet);
|
|
IPersistStream_Release(pPStm);
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static IStream *
|
|
GetUserBandsStream(IN DWORD grfMode)
|
|
{
|
|
HKEY hkStreams;
|
|
IStream *Stream = NULL;
|
|
|
|
if (RegCreateKey(hkExplorer,
|
|
TEXT("Streams"),
|
|
&hkStreams) == ERROR_SUCCESS)
|
|
{
|
|
Stream = SHOpenRegStream(hkStreams,
|
|
TEXT("Desktop"),
|
|
TEXT("TaskbarWinXP"),
|
|
grfMode);
|
|
|
|
RegCloseKey(hkStreams);
|
|
}
|
|
|
|
return Stream;
|
|
}
|
|
|
|
static IStream *
|
|
GetDefaultBandsStream(IN DWORD grfMode)
|
|
{
|
|
HKEY hkStreams;
|
|
IStream *Stream = NULL;
|
|
|
|
if (RegCreateKey(HKEY_LOCAL_MACHINE,
|
|
TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Streams"),
|
|
&hkStreams) == ERROR_SUCCESS)
|
|
{
|
|
Stream = SHOpenRegStream(hkStreams,
|
|
TEXT("Desktop"),
|
|
TEXT("Default Taskbar"),
|
|
grfMode);
|
|
|
|
RegCloseKey(hkStreams);
|
|
}
|
|
|
|
return Stream;
|
|
}
|
|
|
|
static HRESULT
|
|
ITrayBandSiteImpl_Load(IN OUT ITrayBandSiteImpl *This)
|
|
{
|
|
IStream *pStm;
|
|
HRESULT hRet;
|
|
|
|
/* Try to load the user's settings */
|
|
pStm = GetUserBandsStream(STGM_READ);
|
|
if (pStm != NULL)
|
|
{
|
|
hRet = ITrayBandSiteImpl_LoadFromStream(This,
|
|
pStm);
|
|
|
|
DbgPrint("Loaded user bands settings: 0x%x\n", hRet);
|
|
IStream_Release(pStm);
|
|
}
|
|
else
|
|
hRet = E_FAIL;
|
|
|
|
/* If the user's settings couldn't be loaded, try with
|
|
default settings (ie. when the user logs in for the
|
|
first time! */
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
pStm = GetDefaultBandsStream(STGM_READ);
|
|
if (pStm != NULL)
|
|
{
|
|
hRet = ITrayBandSiteImpl_LoadFromStream(This,
|
|
pStm);
|
|
|
|
DbgPrint("Loaded default user bands settings: 0x%x\n", hRet);
|
|
IStream_Release(pStm);
|
|
}
|
|
else
|
|
hRet = E_FAIL;
|
|
}
|
|
|
|
return hRet;
|
|
}
|
|
|
|
static ITrayBandSiteImpl *
|
|
ITrayBandSiteImpl_Construct(IN OUT ITrayWindow *Tray,
|
|
OUT HWND *phWndRebar,
|
|
OUT HWND *phwndTaskSwitch)
|
|
{
|
|
ITrayBandSiteImpl *This;
|
|
IDeskBarClient *pDbc;
|
|
IDeskBand *pDb;
|
|
IOleWindow *pOw;
|
|
HRESULT hRet;
|
|
|
|
*phWndRebar = NULL;
|
|
*phwndTaskSwitch = NULL;
|
|
|
|
This = HeapAlloc(hProcessHeap,
|
|
HEAP_ZERO_MEMORY,
|
|
sizeof(*This));
|
|
if (This == NULL)
|
|
return NULL;
|
|
|
|
This->lpVtbl = &ITrayBandSiteImpl_Vtbl;
|
|
This->lpBandSiteVtbl = &IBandSiteImpl_Vtbl;
|
|
This->Ref = 1;
|
|
This->Tray = Tray;
|
|
|
|
/* Create a RebarBandSite provided by the shell */
|
|
hRet = CoCreateInstance(&CLSID_RebarBandSite,
|
|
(LPUNKNOWN)IBandSite_from_ITrayBandSiteImpl(This),
|
|
CLSCTX_INPROC_SERVER,
|
|
&IID_IUnknown,
|
|
(LPVOID*)&This->punkInner);
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
ITrayBandSiteImpl_Free(This);
|
|
return NULL;
|
|
}
|
|
|
|
hRet = IUnknown_QueryInterface(This->punkInner,
|
|
&IID_IBandSite,
|
|
(PVOID*)&This->BandSite);
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
ITrayBandSiteImpl_Free(This);
|
|
return NULL;
|
|
}
|
|
|
|
hRet = IUnknown_QueryInterface(This->punkInner,
|
|
&IID_IWinEventHandler,
|
|
(PVOID*)&This->WindowEventHandler);
|
|
if (!SUCCEEDED(hRet))
|
|
{
|
|
ITrayBandSiteImpl_Free(This);
|
|
return NULL;
|
|
}
|
|
|
|
This->TaskBand = CreateTaskBand(Tray);
|
|
if (This->TaskBand != NULL)
|
|
{
|
|
/* Add the task band to the site */
|
|
hRet = IBandSite_QueryInterface(This->BandSite,
|
|
&IID_IDeskBarClient,
|
|
(PVOID*)&pDbc);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = ITaskBand_QueryInterface(This->TaskBand,
|
|
&IID_IOleWindow,
|
|
(PVOID*)&pOw);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* We cause IDeskBarClient to create the rebar control by passing the new
|
|
task band to it. The band reports the tray window handle as window handle
|
|
so that IDeskBarClient knows the parent window of the Rebar control that
|
|
it wants to create. */
|
|
hRet = IDeskBarClient_SetDeskBarSite(pDbc,
|
|
(IUnknown *)pOw);
|
|
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* The Rebar control is now created, we can query the window handle */
|
|
hRet = IDeskBarClient_GetWindow(pDbc,
|
|
&This->hWndRebar);
|
|
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
/* We need to manually remove the RBS_BANDBORDERS style! */
|
|
SetWindowStyle(This->hWndRebar,
|
|
RBS_BANDBORDERS,
|
|
0);
|
|
}
|
|
}
|
|
|
|
IOleWindow_Release(pOw);
|
|
}
|
|
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
DWORD dwMode = 0;
|
|
|
|
/* Set the Desk Bar mode to the current one */
|
|
|
|
/* FIXME: We need to set the mode (and update) whenever the user docks
|
|
the tray window to another monitor edge! */
|
|
|
|
if (!ITrayWindow_IsHorizontal(This->Tray))
|
|
dwMode = DBIF_VIEWMODE_VERTICAL;
|
|
|
|
hRet = IDeskBarClient_SetModeDBC(pDbc,
|
|
dwMode);
|
|
}
|
|
|
|
IDeskBarClient_Release(pDbc);
|
|
}
|
|
|
|
/* Load the saved state of the task band site */
|
|
/* FIXME: We should delay loading shell extensions, also see DBID_DELAYINIT */
|
|
ITrayBandSiteImpl_Load(This);
|
|
|
|
/* Add the task bar band if it hasn't been added already */
|
|
hRet = ITrayBandSiteImpl_AddTaskBand(This);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = ITaskBand_QueryInterface(This->TaskBand,
|
|
&IID_IDeskBand,
|
|
(PVOID*)&pDb);
|
|
if (SUCCEEDED(hRet))
|
|
{
|
|
hRet = IDeskBand_GetWindow(pDb,
|
|
phwndTaskSwitch);
|
|
if (!SUCCEEDED(hRet))
|
|
*phwndTaskSwitch = NULL;
|
|
|
|
IDeskBand_Release(pDb);
|
|
}
|
|
}
|
|
|
|
/* Should we send this after showing it? */
|
|
ITrayBandSiteImpl_Update(This);
|
|
|
|
/* FIXME: When should we send this? Does anyone care anyway? */
|
|
ITrayBandSiteImpl_FinishInit(This);
|
|
|
|
/* Activate the band site */
|
|
ITrayBandSiteImpl_Show(This,
|
|
TRUE);
|
|
}
|
|
|
|
*phWndRebar = This->hWndRebar;
|
|
|
|
return This;
|
|
}
|
|
|
|
/*******************************************************************/
|
|
|
|
ITrayBandSite *
|
|
CreateTrayBandSite(IN OUT ITrayWindow *Tray,
|
|
OUT HWND *phWndRebar,
|
|
OUT HWND *phWndTaskSwitch)
|
|
{
|
|
ITrayBandSiteImpl *This;
|
|
|
|
This = ITrayBandSiteImpl_Construct(Tray,
|
|
phWndRebar,
|
|
phWndTaskSwitch);
|
|
if (This != NULL)
|
|
{
|
|
return ITrayBandSite_from_ITrayBandSiteImpl(This);
|
|
}
|
|
|
|
return NULL;
|
|
}
|