mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
810 lines
22 KiB
810 lines
22 KiB
#include "priv.h"
|
|
#include "sccls.h"
|
|
#include "resource.h"
|
|
#include "mshtmhst.h"
|
|
#include "deskbar.h"
|
|
#include "bands.h"
|
|
#define WANT_CBANDSITE_CLASS
|
|
#include "bandsite.h"
|
|
|
|
#include <trayp.h> // TM_*
|
|
#include <desktray.h> // IDeskTray
|
|
|
|
#include "dbapp.h"
|
|
|
|
#include "mluisupp.h"
|
|
|
|
/*
|
|
this virtual app implments DeskBars that you have on the desktop.
|
|
it has the glue that combines CDeskBar with CBandSite and populates the
|
|
bands (as well as persistance and such)
|
|
|
|
-Chee
|
|
*/
|
|
|
|
#define DM_INIT 0
|
|
#define DM_PERSIST 0 // trace IPS::Load, ::Save, etc.
|
|
#define DM_MENU 0 // menu code
|
|
#define DM_DRAG 0 // drag&drop
|
|
#define DM_TRAY 0 // tray: marshal, side, etc.
|
|
|
|
#ifdef DEBUG
|
|
extern unsigned long DbStreamTell(IStream *pstm);
|
|
#else
|
|
#define DbStreamTell(pstm) ((ULONG) 0)
|
|
#endif
|
|
|
|
|
|
#define SUPERCLASS CDeskBar
|
|
|
|
/*
|
|
Instead of just 4 Deskbars on the whole desktop, we now have 4 deskbars for
|
|
each monitor, however, this brings problem whenever a monitor goes away, we
|
|
need to clean up the following datastructure.
|
|
- dli
|
|
*/
|
|
|
|
// FEATURE: (dli) maybe this should be moved into multimon.h
|
|
// however, people should not get into the habbit of depending on this.
|
|
// and it's really not used anywhere else, so, keep it here for now.
|
|
#define DSA_MONITORSGROW 1
|
|
|
|
typedef struct DeskBarsPerMonitor {
|
|
HMONITOR hMon;
|
|
IDeskBar* Deskbars[4];
|
|
} DESKBARSPERMONITOR, *LPDESKBARSPERMONITOR;
|
|
|
|
HDSA g_hdsaDeskBars = NULL;
|
|
|
|
enum ips_e {
|
|
IPS_FALSE, // reserved, must be 0 (FALSE)
|
|
IPS_LOAD,
|
|
IPS_INITNEW
|
|
};
|
|
|
|
CASSERT(IPS_FALSE == 0);
|
|
|
|
CDeskBarApp::~CDeskBarApp()
|
|
{
|
|
_LeaveSide();
|
|
|
|
if (_pbs)
|
|
_pbs->Release();
|
|
|
|
if (_pcm)
|
|
_pcm->Release();
|
|
|
|
}
|
|
|
|
LRESULT CDeskBarApp::v_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LRESULT lres = SUPERCLASS::v_WndProc(hwnd, uMsg, wParam, lParam);
|
|
|
|
if (!_hwnd) {
|
|
return lres; // destroyed by superclass
|
|
}
|
|
|
|
if (_eMode == WBM_BFLOATING) {
|
|
switch (uMsg) {
|
|
case WM_NOTIFY:
|
|
{
|
|
//
|
|
// override the hittest value to be HTCAPTION if we're docked browser based
|
|
//
|
|
NMHDR* pnm = (NMHDR*)lParam;
|
|
|
|
if (pnm->code == NM_NCHITTEST &&
|
|
pnm->hwndFrom == _hwndChild) {
|
|
//
|
|
// in the floating bug docked int he browser, we don't do
|
|
// mdi child stuff, so we make the gripper work as the caption
|
|
//
|
|
NMMOUSE* pnmm = (NMMOUSE*)pnm;
|
|
if (pnmm->dwHitInfo == RBHT_CAPTION ||
|
|
pnmm->dwHitInfo == RBHT_GRABBER)
|
|
lres = HTTRANSPARENT;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_NCHITTEST:
|
|
// all "client" areas are captions in this mode
|
|
if (lres == HTCLIENT)
|
|
lres = HTCAPTION;
|
|
break;
|
|
|
|
case WM_SETCURSOR:
|
|
DefWindowProcWrap(hwnd, uMsg, wParam, lParam);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return lres;
|
|
}
|
|
|
|
BOOL CDeskBarApp::_OnCloseBar(BOOL fConfirm)
|
|
{
|
|
// if we are closing a bar with no bands in it, don't pop up the dialog
|
|
if ((_pbs && (_pbs->EnumBands(-1,NULL)==0)) ||
|
|
(!fConfirm || ConfirmRemoveBand(_hwnd, IDS_CONFIRMCLOSEBAR, TEXT(""))) )
|
|
return SUPERCLASS::_OnCloseBar(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
// Gets the Deskbars on a specific monitor
|
|
// DBPM -- DeskBars Per Monitor
|
|
LPDESKBARSPERMONITOR GetDBPMWithMonitor(HMONITOR hMon, BOOL fCreate)
|
|
{
|
|
int ihdsa;
|
|
LPDESKBARSPERMONITOR pdbpm;
|
|
|
|
if (!g_hdsaDeskBars) {
|
|
if (fCreate)
|
|
g_hdsaDeskBars = DSA_Create(SIZEOF(DESKBARSPERMONITOR), DSA_MONITORSGROW);
|
|
}
|
|
|
|
if (!g_hdsaDeskBars)
|
|
return NULL;
|
|
|
|
// If we find the DBPM with this HMONITOR, return it.
|
|
for (ihdsa = 0; ihdsa < DSA_GetItemCount(g_hdsaDeskBars); ihdsa++) {
|
|
pdbpm = (LPDESKBARSPERMONITOR)DSA_GetItemPtr(g_hdsaDeskBars, ihdsa);
|
|
if (pdbpm->hMon == hMon)
|
|
return pdbpm;
|
|
}
|
|
|
|
if (fCreate) {
|
|
DESKBARSPERMONITOR dbpm = {0};
|
|
// This monitor is not setup, so set it, and set us the
|
|
// the ownder of _uSide
|
|
dbpm.hMon = hMon;
|
|
ihdsa = DSA_AppendItem(g_hdsaDeskBars, &dbpm);
|
|
pdbpm = (LPDESKBARSPERMONITOR)DSA_GetItemPtr(g_hdsaDeskBars, ihdsa);
|
|
return pdbpm;
|
|
}
|
|
|
|
// When all else fails, return NULL
|
|
return NULL;
|
|
}
|
|
|
|
void CDeskBarApp::_LeaveSide()
|
|
{
|
|
if (ISABE_DOCK(_uSide) && !ISWBM_FLOAT(_eMode)) {
|
|
// remove ourselves from the array list of where we were
|
|
LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(_hMon, FALSE);
|
|
if (pdbpm && (pdbpm->Deskbars[_uSide] == this)) {
|
|
ASSERT(pdbpm->hMon);
|
|
ASSERT(pdbpm->hMon == _hMon);
|
|
pdbpm->Deskbars[_uSide] = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
//***
|
|
// NOTES
|
|
// FEATURE: should we create/use IDeskTray::AppBarGetState?
|
|
UINT GetTraySide(HMONITOR * phMon)
|
|
{
|
|
LRESULT lTmp;
|
|
APPBARDATA abd;
|
|
|
|
abd.cbSize = sizeof(APPBARDATA);
|
|
abd.hWnd = GetTrayWindow();
|
|
if (phMon)
|
|
Tray_GetHMonitor(abd.hWnd, phMon);
|
|
|
|
abd.uEdge = (UINT)-1;
|
|
//lTmp = g_pdtray->AppBarGetTaskBarPos(&abd);
|
|
lTmp = SHAppBarMessage(ABM_GETTASKBARPOS, &abd);
|
|
ASSERT(lTmp);
|
|
TraceMsg(DM_TRAY, "gts: ret=ABE_%d", abd.uEdge);
|
|
return abd.uEdge;
|
|
}
|
|
|
|
//***
|
|
// ENTRY/EXIT
|
|
// fNoMerge is for the IPS::Load case
|
|
// NOTES
|
|
// warning: be careful of reentrancy! fNoMove is how we guard against it.
|
|
void CDeskBarApp::_SetModeSide(UINT eMode, UINT uSide, HMONITOR hMonNew, BOOL fNoMerge)
|
|
{
|
|
BOOL fNoMove;
|
|
|
|
// make sure we don't merge etc. on NOOP moves.
|
|
// we do such moves to force refresh (e.g. for autohide and IPS::Load);
|
|
// also happens w/ drags which end up back where they started
|
|
fNoMove = (eMode == _eMode && uSide == _uSide && hMonNew == _hMon);
|
|
|
|
if (!fNoMove)
|
|
_LeaveSide();
|
|
|
|
// warning: this may call (e.g.) AppBarRegister, which causes a
|
|
// resize, which calls back to us. careful of reentrancy!!!
|
|
// if we do reenter we end up w/ nt5:155043, where entry #1 has
|
|
// fNoMove==0, then we get a recalc, entry #2 has fNoMove==1,
|
|
// and we set our side array to us, then return back to entry
|
|
// #1 which merges into itself!
|
|
SUPERCLASS::_SetModeSide(eMode, uSide, hMonNew, fNoMerge);
|
|
|
|
if (!fNoMove) {
|
|
if (ISABE_DOCK(_uSide) && !ISWBM_FLOAT(_eMode)) {
|
|
LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(hMonNew, TRUE);
|
|
HMONITOR hMonTray = NULL;
|
|
if (pdbpm) {
|
|
if (fNoMerge) {
|
|
if (!pdbpm->Deskbars[_uSide]) {
|
|
// 1st guy on an edge owns it
|
|
// if we don't do this, when we load persisted state on logon
|
|
// we end up w/ *no* edge owner (since fNoMerge), so we don't
|
|
// merge on subsequent moves.
|
|
goto Lsetowner;
|
|
}
|
|
}
|
|
else if (pdbpm->Deskbars[_uSide]) {
|
|
// if someone already there, try merging into them
|
|
#ifdef DEBUG
|
|
// alt+drag suppresses merge
|
|
// DEBUG only since don't track >1 per side, but useful
|
|
// for testing appbars and toolbars anyway
|
|
if (!(GetKeyState(VK_MENU) < 0))
|
|
#endif
|
|
{
|
|
extern IBandSite* _GetBandSite(IDeskBar * pdb);
|
|
IBandSite *pbs;
|
|
|
|
pbs = _GetBandSite(pdbpm->Deskbars[_uSide]);
|
|
// nt5:215952: should 'never' have pbs==0 but somehow
|
|
// it does happen (during deskbar automation tests).
|
|
// call andyp or tjgreen if you hit this assert so
|
|
// we can figure out why.
|
|
if (TPTR(pbs)) {
|
|
_MergeSide(pbs); // dst=pbs, src=this
|
|
pbs->Release();
|
|
}
|
|
}
|
|
}
|
|
else if ((GetTraySide(&hMonTray) == _uSide) && (hMonTray == _hMon) && !(GetKeyState(VK_SHIFT) < 0)) {
|
|
// ditto for tray (but need to marshal/unmarshal)
|
|
#ifdef DEBUG
|
|
// alt+drag suppresses merge
|
|
// DEBUG only since don't track >1 per side, but useful
|
|
// for testing appbars and toolbars anyway
|
|
if (!(GetKeyState(VK_MENU) < 0))
|
|
#endif
|
|
{
|
|
_MergeSide((IBandSite *)1); // dst=pbs, src=this
|
|
}
|
|
}
|
|
else {
|
|
// o.w. nobody there yet, set ourselves as owner
|
|
ASSERT(pdbpm->hMon);
|
|
ASSERT(pdbpm->hMon == hMonNew);
|
|
Lsetowner:
|
|
TraceMsg(DM_TRAY, "cdba._sms: 1st side owner this=0x%x", this);
|
|
pdbpm->Deskbars[_uSide] = this;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CDeskBarApp::_UpdateCaptionTitle()
|
|
{
|
|
if (ISWBM_FLOAT(_eMode)) {
|
|
int iCount = (int)_pbs->EnumBands((UINT)-1, NULL);
|
|
if (iCount == 1) {
|
|
DWORD dwBandID;
|
|
if (SUCCEEDED(_pbs->EnumBands(0, &dwBandID))) {
|
|
WCHAR wszTitle[80];
|
|
if (SUCCEEDED(_pbs->QueryBand(dwBandID, NULL, NULL, wszTitle, ARRAYSIZE(wszTitle)))) {
|
|
USES_CONVERSION;
|
|
SetWindowText(_hwnd, W2T(wszTitle));
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
TCHAR szTitle[80];
|
|
szTitle[0] = 0;
|
|
MLLoadString(IDS_WEBBARSTITLE,szTitle,ARRAYSIZE(szTitle));
|
|
SetWindowText(_hwnd, szTitle);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void CDeskBarApp::_NotifyModeChange(DWORD dwMode)
|
|
{
|
|
SUPERCLASS::_NotifyModeChange(dwMode);
|
|
_UpdateCaptionTitle();
|
|
}
|
|
|
|
//*** GetTrayIface -- get iface from tray (w/ marshal/unmarshal)
|
|
//
|
|
HRESULT GetTrayIface(REFIID riid, void **ppvObj)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
HWND hwndTray;
|
|
IStream *pstm;
|
|
|
|
TraceMsg(DM_TRAY, "gtif: marshal!");
|
|
|
|
*ppvObj = NULL;
|
|
|
|
hwndTray = GetTrayWindow();
|
|
if (hwndTray) {
|
|
pstm = (IStream *) SendMessage(hwndTray, TM_MARSHALBS, (WPARAM)(GUID *)&riid, 0);
|
|
|
|
if (EVAL(pstm)) {
|
|
// paired w/ matching Marshal in explorer (TM_MARSHALBS)
|
|
hr = CoGetInterfaceAndReleaseStream(pstm, riid, ppvObj);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//*** _MergeSide -- merge two deskbars into one
|
|
// ENTRY/EXIT
|
|
// this [INOUT] destination deskbar (ptr:1 if tray)
|
|
// pdbSrc [INOUT] source deskbar; deleted if all bands moved successfully
|
|
// ret S_OK if all bands moved; S_FALSE if some moved; E_* o.w.
|
|
HRESULT CDeskBarApp::_MergeSide(IBandSite *pbsDst)
|
|
{
|
|
extern HRESULT _MergeBS(IDropTarget *pdtDst, IBandSite *pbsSrc);
|
|
HRESULT hr;
|
|
IDropTarget *pdtDst;
|
|
|
|
AddRef(); // make sure we don't disappear partway thru operation
|
|
|
|
if (pbsDst == (IBandSite *)1) {
|
|
// get (marshal'ed) iface from tray
|
|
hr = GetTrayIface(IID_IDropTarget, (void **)&pdtDst);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
else {
|
|
// don't merge into ourself!
|
|
ASSERT(pbsDst != _pbs);
|
|
ASSERT(!SHIsSameObject(pbsDst, SAFECAST(_pbs, IBandSite*)));
|
|
|
|
hr = pbsDst->QueryInterface(IID_IDropTarget, (void **)&pdtDst);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
ASSERT(SUCCEEDED(hr) || pdtDst == NULL);
|
|
|
|
if (pdtDst) {
|
|
hr = _MergeBS(pdtDst, _pbs);
|
|
pdtDst->Release();
|
|
}
|
|
|
|
Release();
|
|
|
|
return hr;
|
|
}
|
|
|
|
void CDeskBarApp::_CreateBandSiteMenu()
|
|
{
|
|
CoCreateInstance(CLSID_BandSiteMenu, NULL,CLSCTX_INPROC_SERVER,
|
|
IID_PPV_ARG(IContextMenu3, &_pcm));
|
|
if (_pcm)
|
|
{
|
|
IShellService* pss;
|
|
|
|
_pcm->QueryInterface(IID_IShellService, (LPVOID*)&pss);
|
|
if (pss)
|
|
{
|
|
pss->SetOwner((IBandSite*)_pbs);
|
|
pss->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
HRESULT CDeskBarApp::QueryInterface(REFIID riid, LPVOID * ppvObj)
|
|
{
|
|
if (IsEqualIID(riid, IID_IContextMenu) ||
|
|
IsEqualIID(riid, IID_IContextMenu2) ||
|
|
IsEqualIID(riid, IID_IContextMenu3))
|
|
{
|
|
if (!_pcm)
|
|
{
|
|
_CreateBandSiteMenu();
|
|
}
|
|
|
|
// only return out our pointer if we got the one we're going
|
|
// to delegate to
|
|
if (_pcm)
|
|
{
|
|
*ppvObj = SAFECAST(this, IContextMenu3*);
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
}
|
|
return SUPERCLASS::QueryInterface(riid, ppvObj);
|
|
}
|
|
|
|
HRESULT CDeskBarApp::QueryService(REFGUID guidService,
|
|
REFIID riid, void **ppvObj)
|
|
{
|
|
if (IsEqualGUID(guidService,SID_SBandSite)) {
|
|
return QueryInterface(riid, ppvObj);
|
|
}
|
|
|
|
return SUPERCLASS::QueryService(guidService, riid, ppvObj);
|
|
}
|
|
|
|
|
|
HRESULT CDeskBarApp::InvokeCommand(LPCMINVOKECOMMANDINFO pici)
|
|
{
|
|
int idCmd = -1;
|
|
|
|
if (!HIWORD(pici->lpVerb))
|
|
idCmd = LOWORD(pici->lpVerb);
|
|
|
|
if (idCmd >= _idCmdDeskBarFirst)
|
|
{
|
|
_AppBarOnCommand(idCmd - _idCmdDeskBarFirst);
|
|
return S_OK;
|
|
}
|
|
|
|
return _pcm->InvokeCommand(pici);
|
|
|
|
}
|
|
|
|
HRESULT CDeskBarApp::GetCommandString( UINT_PTR idCmd,
|
|
UINT uType,
|
|
UINT *pwReserved,
|
|
LPSTR pszName,
|
|
UINT cchMax)
|
|
{
|
|
return _pcm->GetCommandString(idCmd, uType, pwReserved, pszName, cchMax);
|
|
}
|
|
|
|
HRESULT CDeskBarApp::HandleMenuMsg(UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
return _pcm->HandleMenuMsg(uMsg, wParam, lParam);
|
|
}
|
|
|
|
HRESULT CDeskBarApp::HandleMenuMsg2(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT* plres)
|
|
{
|
|
return _pcm->HandleMenuMsg2(uMsg, wParam, lParam, plres);
|
|
}
|
|
|
|
HRESULT CDeskBarApp::QueryContextMenu(HMENU hmenu,
|
|
UINT indexMenu,
|
|
UINT idCmdFirst,
|
|
UINT idCmdLast,
|
|
UINT uFlags)
|
|
{
|
|
HRESULT hr = _pcm->QueryContextMenu(hmenu, indexMenu, idCmdFirst, idCmdLast, uFlags);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
int i = hr;
|
|
HMENU hmenuSrc;
|
|
|
|
_idCmdDeskBarFirst = i;
|
|
hmenuSrc = _GetContextMenu();
|
|
|
|
// off-by-1 and by idCmdFirst+i, i think...
|
|
i += Shell_MergeMenus(hmenu, hmenuSrc, (UINT)-1, idCmdFirst + i, idCmdLast, MM_ADDSEPARATOR) - (idCmdFirst + i);
|
|
DestroyMenu(hmenuSrc);
|
|
|
|
return ResultFromShort(i); // potentially off-by-1, but who cares...
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//***
|
|
// NOTES
|
|
// FEATURE: nuke this, fold it into CDeskBarApp_CreateInstance
|
|
HRESULT DeskBarApp_Create(IUnknown** ppunk)
|
|
{
|
|
HRESULT hres;
|
|
|
|
*ppunk = NULL;
|
|
|
|
CDeskBarApp *pdb = new CDeskBarApp();
|
|
if (!pdb)
|
|
return E_OUTOFMEMORY;
|
|
|
|
CBandSite *pcbs = new CBandSite(NULL);
|
|
if (pcbs)
|
|
{
|
|
IDeskBarClient *pdbc = SAFECAST(pcbs, IDeskBarClient*);
|
|
hres = pdb->SetClient(pdbc);
|
|
if (SUCCEEDED(hres))
|
|
{
|
|
pdb->_pbs = pcbs;
|
|
pcbs->AddRef();
|
|
*ppunk = SAFECAST(pdb, IDeskBar*);
|
|
}
|
|
|
|
pdbc->Release();
|
|
}
|
|
else
|
|
{
|
|
hres = E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (FAILED(hres))
|
|
pdb->Release();
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
STDAPI CDeskBarApp_CreateInstance(IUnknown* pUnkOuter, IUnknown** ppunk, LPCOBJECTINFO poi)
|
|
{
|
|
HRESULT hres;
|
|
IUnknown *punk;
|
|
|
|
// aggregation checking is handled in class factory
|
|
|
|
hres = DeskBarApp_Create(&punk);
|
|
if (SUCCEEDED(hres)) {
|
|
*ppunk = SAFECAST(punk, IDockingWindow*);
|
|
return S_OK;
|
|
}
|
|
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
//*** CDeskBarApp::IInputObject*::* {
|
|
//
|
|
|
|
HRESULT CDeskBarApp::TranslateAcceleratorIO(LPMSG lpMsg)
|
|
{
|
|
if (lpMsg->message == WM_SYSKEYDOWN) {
|
|
if (lpMsg->wParam == VK_F4) {
|
|
// ie4:28819: need to trap VK_F4 here, o.w. CBaseBrowser::TA
|
|
// does a last-chance (winsdk)::TA (to IDM_CLOSE) and doing a
|
|
// shutdown!
|
|
PostMessage(_hwnd, WM_CLOSE, 0, 0);
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
return SUPERCLASS::TranslateAcceleratorIO(lpMsg);
|
|
}
|
|
|
|
// }
|
|
|
|
//*** CDeskBarApp::IPersistStream*::* {
|
|
//
|
|
|
|
HRESULT CDeskBarApp::GetClassID(CLSID *pClassID)
|
|
{
|
|
*pClassID = CLSID_DeskBarApp;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CDeskBarApp::IsDirty(void)
|
|
{
|
|
return S_FALSE; // Never be dirty
|
|
}
|
|
|
|
|
|
//
|
|
// Persisted CDeskBarApp
|
|
//
|
|
#define STC_VERSION 1
|
|
|
|
struct SThisClass
|
|
{
|
|
DWORD cbSize;
|
|
DWORD cbVersion;
|
|
};
|
|
|
|
HRESULT CDeskBarApp::Load(IStream *pstm)
|
|
{
|
|
SThisClass stc;
|
|
ULONG cbRead;
|
|
HRESULT hres;
|
|
|
|
TraceMsg(DM_PERSIST, "cdba.l enter(this=%x pstm=%x) tell()=%x", this, pstm, DbStreamTell(pstm));
|
|
|
|
ASSERT(!_eInitLoaded);
|
|
_eInitLoaded = IPS_LOAD;
|
|
|
|
hres = pstm->Read(&stc, SIZEOF(stc), &cbRead);
|
|
#ifdef DEBUG
|
|
// just in case we toast ourselves (offscreen or something)...
|
|
static BOOL fNoPersist = FALSE;
|
|
if (fNoPersist)
|
|
hres = E_FAIL;
|
|
#endif
|
|
if (hres==S_OK && cbRead==SIZEOF(stc)) {
|
|
if (stc.cbSize==SIZEOF(SThisClass) && stc.cbVersion==STC_VERSION) {
|
|
_eInitLoaded = IPS_LOAD; // FEATURE: what if OLFS of bands fails?
|
|
|
|
hres = SUPERCLASS::Load(pstm);
|
|
|
|
TraceMsg(DM_INIT, "cdba::Load succeeded");
|
|
} else {
|
|
TraceMsg(DM_ERROR, "cdba::Load failed stc.cbSize==SIZEOF(SThisClass) && stc.cbVersion==SWB_VERSION");
|
|
hres = E_FAIL;
|
|
}
|
|
} else {
|
|
TraceMsg(DM_ERROR, "cdba::Load failed (hres==S_OK && cbRead==SIZEOF(_adEdge)");
|
|
hres = E_FAIL;
|
|
}
|
|
TraceMsg(DM_PERSIST, "cdba.l leave tell()=%x", DbStreamTell(pstm));
|
|
|
|
// after loading this, if we find that we're supposed to be browser docked,
|
|
// make our bandsite always have a gripper
|
|
if (_eMode == WBM_BFLOATING)
|
|
{
|
|
BANDSITEINFO bsinfo;
|
|
|
|
bsinfo.dwMask = BSIM_STYLE;
|
|
bsinfo.dwStyle = BSIS_ALWAYSGRIPPER;
|
|
|
|
_pbs->SetBandSiteInfo(&bsinfo);
|
|
}
|
|
return hres;
|
|
}
|
|
|
|
HRESULT CDeskBarApp::Save(IStream *pstm, BOOL fClearDirty)
|
|
{
|
|
HRESULT hres;
|
|
SThisClass stc;
|
|
|
|
TraceMsg(DM_PERSIST, "cdba.s enter(this=%x pstm=%x) tell()=%x", this, pstm, DbStreamTell(pstm));
|
|
stc.cbSize = SIZEOF(SThisClass);
|
|
stc.cbVersion = STC_VERSION;
|
|
|
|
hres = pstm->Write(&stc, SIZEOF(stc), NULL);
|
|
if (SUCCEEDED(hres)) {
|
|
SUPERCLASS::Save(pstm, fClearDirty);
|
|
}
|
|
|
|
TraceMsg(DM_PERSIST, "cdba.s leave tell()=%x", DbStreamTell(pstm));
|
|
return hres;
|
|
}
|
|
|
|
HRESULT CDeskBarApp::GetSizeMax(ULARGE_INTEGER *pcbSize)
|
|
{
|
|
ULARGE_INTEGER cbMax = { SIZEOF(SThisClass), 0 };
|
|
*pcbSize = cbMax;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CDeskBarApp::InitNew(void)
|
|
{
|
|
HRESULT hres;
|
|
|
|
ASSERT(!_eInitLoaded);
|
|
_eInitLoaded = IPS_INITNEW;
|
|
TraceMsg(DM_INIT, "CDeskBarApp::InitNew called");
|
|
|
|
hres = SUPERCLASS::InitNew();
|
|
if (FAILED(hres))
|
|
return hres;
|
|
|
|
// can't call _InitPos4 until set site in SetSite
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
HRESULT CDeskBarApp::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt,
|
|
VARIANTARG *pvarargIn, VARIANTARG *pvarargOut)
|
|
{
|
|
if (pguidCmdGroup == NULL) {
|
|
/*NOTHING*/
|
|
}
|
|
else if (IsEqualGUID(CGID_DeskBarClient, *pguidCmdGroup)) {
|
|
switch (nCmdID) {
|
|
case DBCID_EMPTY:
|
|
if (_pbs) {
|
|
// if we have no bands left, close
|
|
PostMessage(_hwnd, WM_CLOSE, 0, 0);
|
|
}
|
|
return S_OK;
|
|
}
|
|
}
|
|
else if (IsEqualIID(*pguidCmdGroup, CGID_DeskBand)) {
|
|
switch (nCmdID) {
|
|
case DBID_BANDINFOCHANGED:
|
|
_UpdateCaptionTitle();
|
|
return S_OK;
|
|
}
|
|
}
|
|
else if (IsEqualIID(*pguidCmdGroup, CGID_BandSite)) {
|
|
switch (nCmdID) {
|
|
case BSID_BANDADDED:
|
|
case BSID_BANDREMOVED:
|
|
_UpdateCaptionTitle();
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
return SUPERCLASS::Exec(pguidCmdGroup, nCmdID, nCmdexecopt, pvarargIn, pvarargOut);
|
|
}
|
|
|
|
HRESULT CDeskBarApp::Load(IPropertyBag *pPropBag, IErrorLog *pErrorLog)
|
|
{
|
|
HRESULT hres;
|
|
|
|
ASSERT(!_eInitLoaded);
|
|
_eInitLoaded = IPS_LOAD;
|
|
TraceMsg(DM_INIT, "CDeskBarApp::Load(bag) called");
|
|
|
|
hres = SUPERCLASS::Load(pPropBag, pErrorLog);
|
|
|
|
// after loading this, if we find that we're supposed to be browser docked,
|
|
// make our bandsite always have a gripper
|
|
if (_eMode == WBM_BFLOATING)
|
|
{
|
|
BANDSITEINFO bsinfo;
|
|
|
|
bsinfo.dwMask = BSIM_STYLE;
|
|
bsinfo.dwStyle = BSIS_ALWAYSGRIPPER;
|
|
|
|
_pbs->SetBandSiteInfo(&bsinfo);
|
|
}
|
|
return hres;
|
|
}
|
|
|
|
IBandSite * _GetBandSite(IDeskBar * pdb)
|
|
{
|
|
IBandSite* pbs = NULL;
|
|
|
|
if (pdb) {
|
|
IUnknown* punkClient;
|
|
|
|
pdb->GetClient(&punkClient);
|
|
if (punkClient) {
|
|
punkClient->QueryInterface(IID_IBandSite, (LPVOID*)&pbs);
|
|
punkClient->Release();
|
|
}
|
|
}
|
|
|
|
return pbs;
|
|
}
|
|
|
|
|
|
IBandSite* DeskBarApp_GetBandSiteOnEdge(UINT uEdge)
|
|
{
|
|
// APPCOMPAT: (dli) if no HMONITOR is passed in, use the primary monitor
|
|
// should make sure that there is always a valid HMONITOR passed in
|
|
HMONITOR hMon = GetPrimaryMonitor();
|
|
// --------------------------------------------------------------
|
|
|
|
LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(hMon, FALSE);
|
|
if (pdbpm) {
|
|
ASSERT(pdbpm->hMon);
|
|
ASSERT(pdbpm->hMon == hMon);
|
|
return _GetBandSite(pdbpm->Deskbars[uEdge]);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
IBandSite* DeskBarApp_GetBandSiteAtPoint(LPPOINT ppt)
|
|
{
|
|
HWND hwnd = WindowFromPoint(*ppt);
|
|
HMONITOR hMon = MonitorFromPoint(*ppt, MONITOR_DEFAULTTONULL);
|
|
if (hwnd && hMon) {
|
|
LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(hMon, FALSE);
|
|
if (pdbpm) {
|
|
ASSERT(pdbpm->hMon);
|
|
ASSERT(pdbpm->hMon == hMon);
|
|
int i;
|
|
for (i = 0; i < 4; i++) {
|
|
if (pdbpm->Deskbars[i]) {
|
|
HWND hwndDeskbar;
|
|
pdbpm->Deskbars[i]->GetWindow(&hwndDeskbar);
|
|
|
|
if (hwndDeskbar == hwnd) {
|
|
return _GetBandSite(pdbpm->Deskbars[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// }
|