303 lines
7.3 KiB
C++
303 lines
7.3 KiB
C++
|
#include "bands.h"
|
||
|
|
||
|
#define DM_PERSIST 0 // trace IPS::Load, ::Save, etc.
|
||
|
#define DM_MENU 0 // menu code
|
||
|
#define DM_FOCUS 0 // focus
|
||
|
#define DM_FOCUS2 0 // like DM_FOCUS, but verbose
|
||
|
|
||
|
//=================================================================
|
||
|
// Implementation of CToolBand
|
||
|
//=================================================================
|
||
|
|
||
|
ULONG CToolBand::AddRef()
|
||
|
{
|
||
|
_cRef++;
|
||
|
return _cRef;
|
||
|
}
|
||
|
|
||
|
ULONG CToolBand::Release()
|
||
|
{
|
||
|
ASSERT(_cRef > 0);
|
||
|
_cRef--;
|
||
|
|
||
|
if (_cRef > 0)
|
||
|
return _cRef;
|
||
|
|
||
|
delete this;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::QueryInterface(REFIID riid, void **ppvObj)
|
||
|
{
|
||
|
static const QITAB qit[] =
|
||
|
{
|
||
|
QITABENT(CToolBand, IDeskBand), // IID_IDeskBand
|
||
|
QITABENTMULTI(CToolBand, IOleWindow, IDeskBand), // IID_IOleWindod
|
||
|
QITABENTMULTI(CToolBand, IDockingWindow, IDeskBand), // IID_IDockingWindow
|
||
|
QITABENT(CToolBand, IInputObject), // IID_IInputObject
|
||
|
QITABENT(CToolBand, IOleCommandTarget), // IID_IOleCommandTarget
|
||
|
QITABENT(CToolBand, IServiceProvider), // IID_IServiceProvider
|
||
|
QITABENT(CToolBand, IPersistStream), // IID_IPersistStream
|
||
|
QITABENTMULTI(CToolBand, IPersist, IPersistStream), // IID_IPersist
|
||
|
QITABENT(CToolBand, IObjectWithSite), // IID_IObjectWithSite
|
||
|
{ 0 },
|
||
|
};
|
||
|
|
||
|
return QISearch(this, qit, riid, ppvObj);
|
||
|
}
|
||
|
|
||
|
// *** IOleCommandTarget methods ***
|
||
|
|
||
|
HRESULT CToolBand::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD rgCmds[], OLECMDTEXT *pcmdtext)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANTARG *pvarargIn, VARIANTARG *pvarargOut)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
// *** IServiceProvider methods ***
|
||
|
|
||
|
HRESULT CToolBand::QueryService(REFGUID guidService,
|
||
|
REFIID riid, void **ppvObj)
|
||
|
{
|
||
|
return IUnknown_QueryService(_punkSite, guidService, riid, ppvObj);
|
||
|
}
|
||
|
|
||
|
// *** IOleWindow methods ***
|
||
|
|
||
|
HRESULT CToolBand::GetWindow(HWND * lphwnd)
|
||
|
{
|
||
|
*lphwnd = _hwnd;
|
||
|
|
||
|
if (*lphwnd)
|
||
|
return S_OK;
|
||
|
|
||
|
return E_FAIL;
|
||
|
}
|
||
|
|
||
|
// *** IInputObject methods ***
|
||
|
|
||
|
HRESULT CToolBand::TranslateAcceleratorIO(LPMSG lpMsg)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::HasFocusIO()
|
||
|
{
|
||
|
HRESULT hres;
|
||
|
HWND hwndFocus = GetFocus();
|
||
|
|
||
|
hres = SHIsChildOrSelf(_hwnd, hwndFocus);
|
||
|
ASSERT(hwndFocus != NULL || hres == S_FALSE);
|
||
|
ASSERT(_hwnd != NULL || hres == S_FALSE);
|
||
|
|
||
|
return hres;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::UIActivateIO(BOOL fActivate, LPMSG lpMsg)
|
||
|
{
|
||
|
ASSERT(NULL == lpMsg || IS_VALID_WRITE_PTR(lpMsg, MSG));
|
||
|
|
||
|
TraceMsg(DM_FOCUS, "ctb.uiaio(fActivate=%d) _fCanFocus=%d _hwnd=%x GF()=%x", fActivate, _fCanFocus, _hwnd, GetFocus());
|
||
|
|
||
|
if (!_fCanFocus) {
|
||
|
TraceMsg(DM_FOCUS, "ctb.uiaio: !_fCanFocus ret S_FALSE");
|
||
|
return S_FALSE;
|
||
|
}
|
||
|
|
||
|
if (fActivate)
|
||
|
{
|
||
|
IUnknown_OnFocusChangeIS(_punkSite, SAFECAST(this, IInputObject*), TRUE);
|
||
|
SetFocus(_hwnd);
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::ResizeBorderDW(LPCRECT prcBorder,
|
||
|
IUnknown* punkToolbarSite,
|
||
|
BOOL fReserved)
|
||
|
{
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT CToolBand::ShowDW(BOOL fShow)
|
||
|
{
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::SetSite(IUnknown *punkSite)
|
||
|
{
|
||
|
if (punkSite != _punkSite)
|
||
|
{
|
||
|
IUnknown_Set(&_punkSite, punkSite);
|
||
|
IUnknown_GetWindow(_punkSite, &_hwndParent);
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::_BandInfoChanged()
|
||
|
{
|
||
|
VARIANTARG v = {0};
|
||
|
VARIANTARG* pv = NULL;
|
||
|
if (_dwBandID != (DWORD)-1)
|
||
|
{
|
||
|
v.vt = VT_I4;
|
||
|
v.lVal = _dwBandID;
|
||
|
pv = &v;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// if this fires, fix your band's GetBandInfo to set _dwBandID.
|
||
|
// o.w. it's a *big* perf loss since we refresh *all* bands rather
|
||
|
// than just yours.
|
||
|
// do *not* remove this ASSERT, bad perf *is* a bug.
|
||
|
ASSERT(_dwBandID != (DWORD)-1);
|
||
|
}
|
||
|
return IUnknown_Exec(_punkSite, &CGID_DeskBand, DBID_BANDINFOCHANGED, 0, pv, NULL);
|
||
|
}
|
||
|
|
||
|
// *** IPersistStream methods ***
|
||
|
|
||
|
HRESULT CToolBand::IsDirty(void)
|
||
|
{
|
||
|
return S_FALSE; // never be dirty
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::GetSizeMax(ULARGE_INTEGER *pcbSize)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
CToolBand::CToolBand() : _cRef(1)
|
||
|
{
|
||
|
_dwBandID = (DWORD)-1;
|
||
|
DllAddRef();
|
||
|
}
|
||
|
|
||
|
CToolBand::~CToolBand()
|
||
|
{
|
||
|
ASSERT(_hwnd == NULL); // CloseDW was called
|
||
|
ASSERT(_punkSite == NULL); // SetSite(NULL) was called
|
||
|
|
||
|
DllRelease();
|
||
|
}
|
||
|
|
||
|
HRESULT CToolBand::CloseDW(DWORD dw)
|
||
|
{
|
||
|
if (_hwnd)
|
||
|
{
|
||
|
DestroyWindow(_hwnd);
|
||
|
_hwnd = NULL;
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
//=================================================================
|
||
|
// Implementation of CToolbarBand
|
||
|
//=================================================================
|
||
|
// Class for bands whose _hwnd is a toolbar control. Implements
|
||
|
// functionality generic to all such bands (e.g. hottracking
|
||
|
// behavior).
|
||
|
//=================================================================
|
||
|
|
||
|
HRESULT CToolbarBand::_PushChevron(BOOL bLast)
|
||
|
{
|
||
|
if (_dwBandID == (DWORD)-1)
|
||
|
return E_UNEXPECTED;
|
||
|
|
||
|
VARIANTARG v;
|
||
|
v.vt = VT_I4;
|
||
|
v.lVal = bLast ? DBPC_SELECTLAST : DBPC_SELECTFIRST;
|
||
|
|
||
|
return IUnknown_Exec(_punkSite, &CGID_DeskBand, DBID_PUSHCHEVRON, _dwBandID, &v, NULL);
|
||
|
}
|
||
|
|
||
|
LRESULT CToolbarBand::_OnHotItemChange(LPNMTBHOTITEM pnmtb)
|
||
|
{
|
||
|
LRESULT lres = 0;
|
||
|
|
||
|
if (!(pnmtb->dwFlags & (HICF_LEAVING | HICF_MOUSE)))
|
||
|
{
|
||
|
// check to see if new hot button is clipped. if it is,
|
||
|
// then we pop down the chevron menu.
|
||
|
RECT rc;
|
||
|
GetClientRect(_hwnd, &rc);
|
||
|
|
||
|
int iButton = (int)SendMessage(_hwnd, TB_COMMANDTOINDEX, pnmtb->idNew, 0);
|
||
|
DWORD dwEdge = SHIsButtonObscured(_hwnd, &rc, iButton);
|
||
|
if (dwEdge)
|
||
|
{
|
||
|
//
|
||
|
// Only pop down the menu if the button is obscured
|
||
|
// along the axis of the toolbar
|
||
|
//
|
||
|
BOOL fVertical = (ToolBar_GetStyle(_hwnd) & CCS_VERT);
|
||
|
|
||
|
if ((fVertical && (dwEdge & (EDGE_TOP | EDGE_BOTTOM)))
|
||
|
|| (!fVertical && (dwEdge & (EDGE_LEFT | EDGE_RIGHT))))
|
||
|
{
|
||
|
// clear hot item
|
||
|
SendMessage(_hwnd, TB_SETHOTITEM, -1, 0);
|
||
|
|
||
|
// figure out whether to highlight first or last button in dd menu
|
||
|
int cButtons = (int)SendMessage(_hwnd, TB_BUTTONCOUNT, 0, 0);
|
||
|
BOOL bLast = (iButton == cButtons - 1);
|
||
|
_PushChevron(bLast);
|
||
|
lres = 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return lres;
|
||
|
}
|
||
|
|
||
|
LRESULT CToolbarBand::_OnNotify(LPNMHDR pnmh)
|
||
|
{
|
||
|
LRESULT lres = 0;
|
||
|
|
||
|
switch (pnmh->code)
|
||
|
{
|
||
|
case TBN_HOTITEMCHANGE:
|
||
|
lres = _OnHotItemChange((LPNMTBHOTITEM)pnmh);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return lres;
|
||
|
}
|
||
|
|
||
|
// *** IWinEventHandler methods ***
|
||
|
|
||
|
HRESULT CToolbarBand::OnWinEvent(HWND hwnd, UINT dwMsg, WPARAM wParam, LPARAM lParam, LRESULT* plres)
|
||
|
{
|
||
|
HRESULT hres = S_OK;
|
||
|
|
||
|
switch (dwMsg)
|
||
|
{
|
||
|
case WM_NOTIFY:
|
||
|
*plres = _OnNotify((LPNMHDR)lParam);
|
||
|
break;
|
||
|
|
||
|
case WM_WININICHANGE:
|
||
|
InvalidateRect(_hwnd, NULL, TRUE);
|
||
|
_BandInfoChanged();
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return hres;
|
||
|
}
|
||
|
|
||
|
HRESULT CToolbarBand::IsWindowOwner(HWND hwnd)
|
||
|
{
|
||
|
if (hwnd == _hwnd)
|
||
|
return S_OK;
|
||
|
else
|
||
|
return S_FALSE;
|
||
|
}
|