2020-09-30 16:53:55 +02:00

1672 lines
43 KiB
C++

/////////////////////////////////////////////////////////////////////////////
// Copyright (C) 1993-1998 Microsoft Corporation. All Rights Reserved.
//
// MODULE: navpane.cpp
//
// PURPOSE:
//
#include "pch.hxx"
#include "navpane.h"
#include "treeview.h"
#include "baui.h"
#include "browser.h"
#include "menuutil.h"
#include "inpobj.h"
/////////////////////////////////////////////////////////////////////////////
// Local Stuff
//
const TCHAR c_szNavPaneClass[] = _T("Outlook Express Navigation Pane");
const TCHAR c_szPaneFrameClass[] = _T("Outlook Express Pane Frame");
// Sizing consts
const int c_cxBorder = 1;
const int c_cyBorder = 1;
const int c_cxTextBorder = 4;
const int c_cyTextBorder = 2;
const int c_cyClose = 3;
const int c_cySplit = 4;
const int c_cxSplit = 3;
#define ID_PANE_CLOSE 2000
#define ID_PANE_PIN 2001
#define ID_PANE_TITLE 2002
#define IDT_PANETIMER 100
#define ELAPSE_MOUSEOVERCHECK 250
/////////////////////////////////////////////////////////////////////////////
// CNavPane Implementation
//
CNavPane::CNavPane()
{
m_cRef = 1;
m_fShow = FALSE;
m_fTreeVisible = FALSE;
m_fContactsVisible = FALSE;
m_hwnd = 0;
m_hwndParent = 0;
m_hwndTree = 0;
m_hwndContacts = 0;
m_pSite = NULL;
m_pTreeView = NULL;
m_pContacts = NULL;
m_pContactsFrame = NULL;
m_pContactsTarget = NULL;
m_cxWidth = 200;
m_fResizing = FALSE;
m_fSplitting = FALSE;
m_cySplitPct = 50;
ZeroMemory(&m_rcSplit, sizeof(RECT));
ZeroMemory(&m_rcSizeBorder, sizeof(RECT));
m_cyTitleBar = 32;
}
CNavPane::~CNavPane()
{
SafeRelease(m_pContactsFrame);
}
HRESULT CNavPane::Initialize(CTreeView *pTreeView)
{
// We've got to have this
if (!pTreeView)
return (E_INVALIDARG);
// Keep it
m_pTreeView = pTreeView;
m_pTreeView->AddRef();
// Load some settings
m_cxWidth = DwGetOption(OPT_NAVPANEWIDTH);
if (m_cxWidth < 0)
m_cxWidth = 200;
m_cySplitPct = DwGetOption(OPT_NAVPANESPLIT);
// Do some parameter checking
if (m_cySplitPct > 100 || m_cySplitPct < 2)
m_cySplitPct = 66;
return (S_OK);
}
//
// FUNCTION: CNavPane::QueryInterface()
//
// PURPOSE: Allows caller to retrieve the various interfaces supported by
// this class.
//
HRESULT CNavPane::QueryInterface(REFIID riid, LPVOID *ppvObj)
{
TraceCall("CNavPane::QueryInterface");
*ppvObj = NULL;
if (IsEqualIID(riid, IID_IUnknown))
*ppvObj = (LPVOID) (IDockingWindow *) this;
else if (IsEqualIID(riid, IID_IDockingWindow))
*ppvObj = (LPVOID) (IDockingWindow *) this;
else if (IsEqualIID(riid, IID_IObjectWithSite))
*ppvObj = (LPVOID) (IObjectWithSite *) this;
else if (IsEqualIID(riid, IID_IOleCommandTarget))
*ppvObj = (LPVOID) (IOleCommandTarget *) this;
else if (IsEqualIID(riid, IID_IInputObjectSite))
*ppvObj = (LPVOID) (IInputObjectSite *) this;
else if (IsEqualIID(riid, IID_IInputObject))
*ppvObj = (LPVOID) (IInputObject *) this;
if (*ppvObj)
{
AddRef();
return (S_OK);
}
return (E_NOINTERFACE);
}
//
// FUNCTION: CNavPane::AddRef()
//
// PURPOSE: Adds a reference count to this object.
//
ULONG CNavPane::AddRef(void)
{
TraceCall("CNavPane::AddRef");
return ((ULONG) InterlockedIncrement((LONG *) &m_cRef));
}
//
// FUNCTION: CNavPane::Release()
//
// PURPOSE: Releases a reference on this object.
//
ULONG CNavPane::Release(void)
{
TraceCall("CNavPane::Release");
if (0 == InterlockedDecrement((LONG *) &m_cRef))
{
delete this;
return 0;
}
return (m_cRef);
}
//
// FUNCTION: CNavPane::GetWindow()
//
// PURPOSE: Returns the handle of our outer window
//
// PARAMETERS:
// [out] pHwnd - return value
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::GetWindow(HWND *pHwnd)
{
TraceCall("CNavPane::GetWindow");
if (!pHwnd)
return (E_INVALIDARG);
if (IsWindow(m_hwnd))
{
*pHwnd = m_hwnd;
return (S_OK);
}
return (E_FAIL);
}
//
// FUNCTION: CNavPane::ContextSensitiveHelp()
//
// PURPOSE: Does anyone _ever_ implement this?
//
HRESULT CNavPane::ContextSensitiveHelp(BOOL fEnterMode)
{
TraceCall("CNavPane::ContextSensitiveHelp");
return (E_NOTIMPL);
}
//
// FUNCTION: CNavPane::ShowDW()
//
// PURPOSE: Show's or hides the Nav pane. If the pane has not yet been
// created it does that too.
//
// PARAMETERS:
// [in] fShow - TRUE to show, FALSE to hide
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::ShowDW(BOOL fShow)
{
HRESULT hr;
WNDCLASSEX wc;
TraceCall("CNavPane::ShowDW");
// Nothing works without a site pointer
if (!m_pSite)
return (E_UNEXPECTED);
// Check to see if we've been created yet
if (!m_hwnd)
{
// Register the window class if necessary
wc.cbSize = sizeof(WNDCLASSEX);
if (!GetClassInfoEx(g_hInst, c_szNavPaneClass, &wc))
{
wc.style = 0;
wc.lpfnWndProc = _WndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = g_hInst;
wc.hCursor = LoadCursor(0, IDC_SIZEWE);
wc.hbrBackground = (HBRUSH) (COLOR_3DFACE + 1);
wc.lpszMenuName = NULL;
wc.lpszClassName = c_szNavPaneClass;
wc.hIcon = NULL;
wc.hIconSm = NULL;
RegisterClassEx(&wc);
}
// Get the parent window before we create ours
if (FAILED(m_pSite->GetWindow(&m_hwndParent)))
{
AssertSz(FALSE, "CNavPane::ShowDW() - Failed to get a parent window handle.");
}
// Create the window
m_hwnd = CreateWindowEx(WS_EX_CONTROLPARENT, c_szNavPaneClass, NULL,
WS_CHILD | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
0, 0, 10, 10, m_hwndParent, (HMENU) 0, g_hInst, this);
if (!m_hwnd)
{
AssertSz(FALSE, "CNavPane::ShowDW() - Failed to create main window.");
return (E_OUTOFMEMORY);
}
// Create any children
if (FAILED(hr = _CreateChildWindows()))
{
AssertSz(FALSE, "CNavPane::ShowDW() - Failed to create child windows.");
DestroyWindow(m_hwnd);
return (hr);
}
}
// Show or hide the window appropriately
m_fShow = (fShow && (m_fTreeVisible || m_fContactsVisible));
ResizeBorderDW(0, 0, FALSE);
ShowWindow(m_hwnd, fShow ? SW_SHOW : SW_HIDE);
return (S_OK);
}
//
// FUNCTION: CNavPane::ResizeBorderDW()
//
// PURPOSE: Called when it's time for us to re-request space from our
// parent.
//
// PARAMETERS:
// [in] prcBorder - a RECT containing the outer rectangle the object can request space in
// [in] punkSite - pointer to the site that changed
// [in] fReserved - unused.
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::ResizeBorderDW(LPCRECT prcBorder, IUnknown *punkSite, BOOL fReserved)
{
const DWORD c_cxResizeBorder = 3;
HRESULT hr = S_OK;
RECT rcRequest = { 0 };
RECT rcBorder;
TraceCall("CNavPane::ResizeBorderDW");
// If we don't have a site pointer, this ain't gonna work
if (!m_pSite)
return (E_UNEXPECTED);
// If we visible, then calculate our border requirements. If we're not
// visible, the our requirements are zero and we can use the default
// values in rcRequest.
Assert(IsWindow(m_hwnd));
// If the caller didn't provide us with a rect, get one ourselves
if (!prcBorder)
{
m_pSite->GetBorderDW((IDockingWindow *) this, &rcBorder);
prcBorder = &rcBorder;
}
// The space we need is the min of either what we want to be or the
// width of the parent minus some
if (m_fShow)
{
rcRequest.left = min(prcBorder->right - prcBorder->left - 32, m_cxWidth);
}
// Ask for the space we need
if (SUCCEEDED(m_pSite->RequestBorderSpaceDW((IDockingWindow *) this, &rcRequest)))
{
// Tell the site how be we're going to be
if (SUCCEEDED(m_pSite->SetBorderSpaceDW((IDockingWindow *) this, &rcRequest)))
{
// Now once that's all done, resize ourselves if we're visible
if (m_fShow)
{
SetWindowPos(m_hwnd, 0, prcBorder->left, prcBorder->top, rcRequest.left,
prcBorder->bottom - prcBorder->top, SWP_NOZORDER | SWP_NOACTIVATE);
}
}
}
return (S_OK);
}
//
// FUNCTION: CNavPane::CloseDW()
//
// PURPOSE: Called when the parent want's to destroy this window
//
// PARAMETERS:
// [in] dwReserved - unused
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::CloseDW(DWORD dwReserved)
{
TraceCall("CNavPane::CloseDW");
// Save our settings
SetDwOption(OPT_NAVPANEWIDTH, m_cxWidth, NULL, 0);
SetDwOption(OPT_NAVPANESPLIT, m_cySplitPct, NULL, 0);
if (m_pTreeView)
m_pTreeView->DeInit();
if (m_hwnd)
{
DestroyWindow(m_hwnd);
m_hwnd = NULL;
}
// Destroy our children here
SafeRelease(m_pTreeView);
SafeRelease(m_pContactsTarget);
SafeRelease(m_pContacts);
return (S_OK);
}
//
// FUNCTION: CNavPane::GetSite()
//
// PURPOSE: Called to request an interface to our site
//
// PARAMETERS:
// [in] riid - Requested interface
// [out] ppvSite - Returned interface if available
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::GetSite(REFIID riid, LPVOID *ppvSite)
{
HRESULT hr;
TraceCall("CNavPane::GetSite");
if (m_pSite)
{
// Ask our site for the requested interface
hr = m_pSite->QueryInterface(riid, ppvSite);
return (hr);
}
return (E_FAIL);
}
//
// FUNCTION: CNavPane::SetSite()
//
// PURPOSE: Called to tell us who our site will be.
//
// PARAMETERS:
// [in] pUnkSite - Pointer to the new site
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::SetSite(IUnknown *pUnkSite)
{
HRESULT hr = S_OK;
TraceCall("CNavPane::SetSite");
// If we already have a site, release it
if (m_pSite)
{
m_pSite->Release();
m_pSite = 0;
}
// If we were given a new site, keep it
if (pUnkSite)
{
hr = pUnkSite->QueryInterface(IID_IDockingWindowSite, (LPVOID *) &m_pSite);
return (hr);
}
return (hr);
}
//
// FUNCTION: CNavPane::_WndProc()
//
// PURPOSE: External callback.
//
LRESULT CALLBACK CNavPane::_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
CNavPane *pThis;
if (uMsg == WM_NCCREATE)
{
pThis = (CNavPane *) ((LPCREATESTRUCT) lParam)->lpCreateParams;
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LPARAM) pThis);
}
else
pThis = (CNavPane *) GetWindowLongPtr(hwnd, GWLP_USERDATA);
if (pThis)
return (pThis->_NavWndProc(hwnd, uMsg, wParam, lParam));
return (FALSE);
}
//
// FUNCTION: CNavPane::_NavWndProc()
//
// PURPOSE: Left as an exercise for the reader
//
LRESULT CALLBACK CNavPane::_NavWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
HANDLE_MSG(hwnd, WM_SETCURSOR, _OnSetCursor);
HANDLE_MSG(hwnd, WM_SIZE, _OnSize);
HANDLE_MSG(hwnd, WM_MOUSEMOVE, _OnMouseMove);
HANDLE_MSG(hwnd, WM_LBUTTONDOWN, _OnLButtonDown);
HANDLE_MSG(hwnd, WM_LBUTTONUP, _OnLButtonUp);
case WM_SYSCOLORCHANGE:
case WM_WININICHANGE:
{
// Forward these to all our children
if (IsWindow(m_hwndTree))
SendMessage(m_hwndTree, uMsg, wParam, lParam);
if (IsWindow(m_hwndContacts))
SendMessage(m_hwndContacts, uMsg, wParam, lParam);
// Update any of our own sizes
m_cyTitleBar =(UINT) SendMessage(m_hwndTree, WM_GET_TITLE_BAR_HEIGHT, 0, 0);
return (0);
}
}
return (DefWindowProc(hwnd, uMsg, wParam, lParam));
}
//
// FUNCTION: CNavPane::_OnSize()
//
// PURPOSE: When our window get's resized, we need to resize our child
// windows too.
//
void CNavPane::_OnSize(HWND hwnd, UINT state, int cx, int cy)
{
RECT rc;
DWORD cyTree;
DWORD cySplit = c_cySplit;
TraceCall("CNavPane::_OnSize");
// If only the tree is visible
if (m_fTreeVisible && !m_fContactsVisible)
cyTree = cy;
else if (m_fTreeVisible && m_fContactsVisible)
cyTree = (cy * m_cySplitPct) / 100;
else if (!m_fTreeVisible && m_fContactsVisible)
{
cyTree = 0;
cySplit = 0;
}
// Resize the TreeView to fit inside our window
if (m_hwndTree)
SetWindowPos(m_hwndTree, 0, 0, 0, cx - c_cxSplit, cyTree, SWP_NOZORDER | SWP_NOACTIVATE);
if (m_hwndContacts)
SetWindowPos(m_hwndContacts, 0, 0, cyTree + cySplit, cx - 3, cy - cyTree - cySplit, SWP_NOZORDER | SWP_NOACTIVATE);
// Figure out where a few things are, starting with the split bar
SetRect(&rc, c_cxBorder, cyTree, cx - c_cxSplit - c_cxBorder, cyTree + cySplit);
m_rcSplit = rc;
// Figure out where the right side is
SetRect(&rc, cx - c_cxSplit, 0, cx, cy);
m_rcSizeBorder = rc;
}
//
// FUNCTION: CNavPane::_OnLButtonDown()
//
// PURPOSE: When the user clicks down and we get this notification, it
// must be because they want to resize.
//
void CNavPane::_OnLButtonDown(HWND hwnd, BOOL fDoubleClick, int x, int y, UINT keyFlags)
{
TraceCall("CNavPane::_OnLButtonDown");
if (!m_fResizing)
{
SetCapture(hwnd);
m_fResizing = TRUE;
POINT pt = {x, y};
if (PtInRect(&m_rcSplit, pt))
{
m_fSplitting = TRUE;
}
}
}
//
// FUNCTION: CNavPane::_OnMouseMove()
//
// PURPOSE: If we're resizing, update our position etc.
//
void CNavPane::_OnMouseMove(HWND hwnd, int x, int y, UINT keyFlags)
{
POINT pt = {x, y};
RECT rcClient;
TraceCall("CNavPane::_OnMouseMove");
if (m_fResizing)
{
if (m_fSplitting)
{
GetClientRect(m_hwnd, &rcClient);
m_cySplitPct = (int)(((float) pt.y / (float) rcClient.bottom) * 100);
// Make sure we have the min's and max's right
int cy = (rcClient.bottom * m_cySplitPct) / 100;
if (cy < m_cyTitleBar)
{
m_cySplitPct = (int)(((float) m_cyTitleBar / (float) rcClient.bottom) * 100);
}
else if (rcClient.bottom - cy < m_cyTitleBar)
{
m_cySplitPct = (int)(((float) (rcClient.bottom - m_cyTitleBar) / (float) rcClient.bottom) * 100);
}
_OnSize(hwnd, 0, rcClient.right, rcClient.bottom);
}
else
{
if (pt.x > 32)
{
GetClientRect(m_hwndParent, &rcClient);
m_cxWidth = max(0, min(pt.x, rcClient.right - 32));
ResizeBorderDW(0, 0, FALSE);
}
}
}
}
//
// FUNCTION: CNavPane::_OnLButtonUp()
//
// PURPOSE: If the user was resizing, then they're done now and we can
// clean up.
//
void CNavPane::_OnLButtonUp(HWND hwnd, int x, int y, UINT keyFlags)
{
TraceCall("CNavPane::_OnLButtonUp");
if (m_fResizing)
{
ReleaseCapture();
m_fResizing = FALSE;
m_fSplitting = FALSE;
}
}
//
// FUNCTION: CNavPane::_OnSetCursor()
//
// PURPOSE: Do some jimmying with the cursor
//
BOOL CNavPane::_OnSetCursor(HWND hwnd, HWND hwndCursor, UINT codeHitTest, UINT msg)
{
POINT pt;
TraceCall("_OnSetCursor");
// Get the cursor position
GetCursorPos(&pt);
ScreenToClient(m_hwnd, &pt);
// If the cursor is within the split bar, update the cursor
if (PtInRect(&m_rcSplit, pt))
{
SetCursor(LoadCursor(NULL, IDC_SIZENS));
return (TRUE);
}
if (PtInRect(&m_rcSizeBorder, pt))
{
SetCursor(LoadCursor(NULL, IDC_SIZEWE));
return (TRUE);
}
return (FALSE);
}
//
// FUNCTION: CNavPane::_OnNCHitTest()
//
// PURPOSE: We monkey around with the non client area to get the correct
// cursors
//
// PARAMETERS:
// [in] hwnd - Window handle the mouse is in
// [in] x, y - Position of the mouse in screen coordinates
//
// RETURN VALUE:
// Our personal opinion of where the mouse is.
//
UINT CNavPane::_OnNCHitTest(HWND hwnd, int x, int y)
{
POINT pt = {x, y};
// If the cursor is in the split bar
if (PtInRect(&m_rcSplit, pt))
return (HTTOP);
if (PtInRect(&m_rcSizeBorder, pt))
return (HTRIGHT);
return (HTCLIENT);
}
//
// FUNCTION: CNavPane::_CreateChildWindows()
//
// PURPOSE: Creates the child windows that will be displayed.
//
// RETURN VALUE:
// HRESULT
//
HRESULT CNavPane::_CreateChildWindows(void)
{
IOleWindow *pWindow = NULL;
IInputObject *pInputObj = NULL;
HRESULT hr;
TraceCall("CNavPane::_CreateChildWindows");
// The treeview is always created by the browser. All we have to do
// is tell it to create it's UI.
m_hwndTree = m_pTreeView->Create(m_hwnd, (IInputObjectSite *) this, TRUE);
Assert(m_hwndTree);
// If the tree is supposed to be visible, show it
if (DwGetOption(OPT_SHOWTREE))
{
ShowWindow(m_hwndTree, SW_SHOW);
m_fTreeVisible = TRUE;
m_cyTitleBar = (UINT) SendMessage(m_hwndTree, WM_GET_TITLE_BAR_HEIGHT, 0, 0);
}
// If we're showing contacts, create it
if (DwGetOption(OPT_SHOWCONTACTS) && (!(g_dwAthenaMode & MODE_OUTLOOKNEWS)))
{
ShowContacts(TRUE);
}
return (S_OK);
}
//
// FUNCTION: CNavPane::ShowFolderList()
//
// PURPOSE: Shows and hides the folder list doodad
//
// PARAMETERS:
// BOOL fShow
//
BOOL CNavPane::ShowFolderList(BOOL fShow)
{
TraceCall("CNavPane::ShowFolderList");
// The folder list _always_ exists. We just toggle the state
ShowWindow(m_hwndTree, fShow ? SW_SHOW : SW_HIDE);
m_fTreeVisible = fShow;
_UpdateVisibleState();
RECT rc;
GetClientRect(m_hwnd, &rc);
_OnSize(m_hwnd, 0, rc.right, rc.bottom);
return (TRUE);
}
//
// FUNCTION: CNavPane::ShowContacts()
//
// PURPOSE:
//
// PARAMETERS:
// BOOL fShow
//
// RETURN VALUE:
// BOOL
//
BOOL CNavPane::ShowContacts(BOOL fShow)
{
CMsgrAb *pMsgrAb;
HWND hwnd;
IAthenaBrowser *pBrowser;
HRESULT hr;
RECT rc = {0};
if (!m_pContacts)
{
hr = CreateMsgrAbCtrl(&m_pContacts);
if (SUCCEEDED(hr))
{
// Initialize the control
m_pContactsFrame = new CPaneFrame();
if (!m_pContactsFrame)
return (0);
m_hwndContacts = m_pContactsFrame->Initialize(m_hwnd, this, idsABBandTitle, IDR_BA_TITLE_POPUP);
pMsgrAb = (CMsgrAb *) m_pContacts;
hwnd = pMsgrAb->CreateControlWindow(m_hwndContacts, rc);
if (hwnd)
{
if (SUCCEEDED(m_pSite->QueryInterface(IID_IAthenaBrowser, (LPVOID *) &pBrowser)))
{
m_pContactsFrame->SetChild(hwnd, DISPID_MSGVIEW_CONTACTS, pBrowser, pMsgrAb, pMsgrAb);
pBrowser->Release();
}
}
// Get the command target
m_pContacts->QueryInterface(IID_IOleCommandTarget, (LPVOID *) &m_pContactsTarget);
}
}
SetWindowPos(m_hwndContacts, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
ShowWindow(m_hwndContacts, fShow ? SW_SHOW : SW_HIDE);
m_fContactsVisible = fShow;
_UpdateVisibleState();
GetClientRect(m_hwnd, &rc);
_OnSize(m_hwnd, 0, rc.right, rc.bottom);
return (TRUE);
}
//
// FUNCTION: CNavPane::_UpdateVisibleState()
//
// PURPOSE: Checks to see if we need to show our hide ourselves
//
void CNavPane::_UpdateVisibleState(void)
{
// If this leaves us with nothing visible, then we hide ourselves
if (!m_fTreeVisible && !m_fContactsVisible)
{
ShowWindow(m_hwnd, SW_HIDE);
m_fShow = FALSE;
ResizeBorderDW(0, 0, 0);
}
else if (m_fShow == FALSE && (m_fTreeVisible || m_fContactsVisible))
{
// Show ourselves
m_fShow = TRUE;
ShowWindow(m_hwnd, SW_SHOW);
ResizeBorderDW(0, 0, 0);
}
}
HRESULT CNavPane::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds[],
OLECMDTEXT *pCmdText)
{
if (m_pContactsTarget)
{
for (UINT i = 0; i < cCmds; i++)
{
if (prgCmds[i].cmdf == 0 && prgCmds[i].cmdID == ID_CONTACTS_MNEMONIC)
{
prgCmds->cmdf = OLECMDF_SUPPORTED | OLECMDF_ENABLED;
}
}
}
if (m_pContactsTarget)
return (m_pContactsTarget->QueryStatus(pguidCmdGroup, cCmds, prgCmds, pCmdText));
return (S_OK);
}
HRESULT CNavPane::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdExecOpt,
VARIANTARG *pvaIn, VARIANTARG *pvaOut)
{
if (m_pContactsTarget && nCmdID == ID_CONTACTS_MNEMONIC)
{
m_pContactsFrame->ShowMenu();
return (S_OK);
}
if (m_pContactsTarget)
return (m_pContactsTarget->Exec(pguidCmdGroup, nCmdID, nCmdExecOpt, pvaIn, pvaOut));
return (OLECMDERR_E_NOTSUPPORTED);
}
BOOL CNavPane::IsContactsFocus(void)
{
IInputObject *pInputObject = 0;
HRESULT hr = S_FALSE;
if (m_pContacts)
{
if (SUCCEEDED(m_pContacts->QueryInterface(IID_IInputObject, (LPVOID *) &pInputObject)))
{
hr = pInputObject->HasFocusIO();
pInputObject->Release();
return (S_OK == hr);
}
}
return (S_OK == hr);
}
HRESULT CNavPane::OnFocusChangeIS(IUnknown *punkSrc, BOOL fSetFocus)
{
// Simply call through to our host
UnkOnFocusChangeIS(m_pSite, (IInputObject*) this, fSetFocus);
return (S_OK);
}
HRESULT CNavPane::UIActivateIO(BOOL fActivate, LPMSG lpMsg)
{
if (fActivate)
{
UnkOnFocusChangeIS(m_pSite, (IInputObject *) this, TRUE);
SetFocus(m_hwnd);
}
return (S_OK);
}
HRESULT CNavPane::HasFocusIO(void)
{
if (m_hwnd == 0)
return (S_FALSE);
HWND hwndFocus = GetFocus();
return (hwndFocus == m_hwnd || IsChild(m_hwnd, hwndFocus)) ? S_OK : S_FALSE;
}
HRESULT CNavPane::TranslateAcceleratorIO(LPMSG pMsg)
{
if (m_pTreeView && (m_pTreeView->HasFocusIO() == S_OK))
return m_pTreeView->TranslateAcceleratorIO(pMsg);
if (m_pContacts && (UnkHasFocusIO(m_pContacts) == S_OK))
return UnkTranslateAcceleratorIO(m_pContacts, pMsg);
return (S_FALSE);
}
/////////////////////////////////////////////////////////////////////////////
// CPaneFrame
//
CPaneFrame::CPaneFrame()
{
m_cRef = 1;
m_hwnd = 0;
m_hwndChild = 0;
m_hwndParent = 0;
m_szTitle[0] = 0;
m_hFont = 0;
m_hbr3DFace = 0;
m_cyTitleBar = 0;
m_fHighlightIndicator = FALSE;
m_fHighlightPressed = FALSE;
ZeroMemory(&m_rcTitleButton, sizeof(RECT));
m_hwndClose = 0;
m_cButtons = 1;
m_pBrowser = NULL;
m_dwDispId = 0;
m_pTarget = 0;
m_idMenu = 0;
m_fPin = FALSE;
}
CPaneFrame::~CPaneFrame()
{
if (m_hFont != 0)
DeleteObject(m_hFont);
if (m_hbr3DFace != 0)
DeleteObject(m_hbr3DFace);
}
//
// FUNCTION: CPaneFrame::Initialize()
//
// PURPOSE: Initializes the frame by telling the pane what it's title
// should be.
//
// PARAMETERS:
// [in] hwndParent
// [in] idsTitle
//
// RETURN VALUE:
// HWND
//
HWND CPaneFrame::Initialize(HWND hwndParent, IInputObjectSite *pSite, int idsTitle, int idMenu)
{
WNDCLASSEX wc;
TraceCall("CPaneFrame::Initialize");
// This should be NULL
Assert(NULL == m_hwnd);
// Save this for later
m_hwndParent = hwndParent;
m_idMenu = idMenu;
m_pSite = pSite;
// Load the title
AthLoadString(idsTitle, m_szTitle, ARRAYSIZE(m_szTitle));
// Register the window class if necessary
wc.cbSize = sizeof(WNDCLASSEX);
if (!GetClassInfoEx(g_hInst, c_szPaneFrameClass, &wc))
{
wc.style = 0;
wc.lpfnWndProc = _WndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = g_hInst;
wc.hCursor = LoadCursor(0, IDC_ARROW);
wc.hbrBackground = (HBRUSH) (COLOR_3DFACE + 1);
wc.lpszMenuName = NULL;
wc.lpszClassName = c_szPaneFrameClass;
wc.hIcon = NULL;
wc.hIconSm = NULL;
RegisterClassEx(&wc);
}
// Create the window
m_hwnd = CreateWindowEx(WS_EX_CONTROLPARENT, c_szPaneFrameClass, m_szTitle,
WS_CHILD | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
0, 0, 0, 0, hwndParent, 0, g_hInst, this);
if (!m_hwnd)
{
AssertSz(m_hwnd, "CPaneFrame::Initialize() - Failed to create a frame");
return (0);
}
return (m_hwnd);
}
//
// FUNCTION: CPaneFrame::SetChild()
//
// PURPOSE: Allows the owner to tell us what the child window handle is.
//
BOOL CPaneFrame::SetChild(HWND hwndChild, DWORD dwDispId, IAthenaBrowser *pBrowser,
IObjectWithSite *pObject, IOleCommandTarget *pTarget)
{
TraceCall("CPaneFrame::SetChild");
if (IsWindow(hwndChild))
{
m_hwndChild = hwndChild;
if (pBrowser)
{
m_pBrowser = pBrowser;
m_dwDispId = dwDispId;
}
if (pObject)
{
pObject->SetSite((IInputObjectSite *) this);
}
if (pTarget)
{
m_pTarget = pTarget;
}
return (TRUE);
}
return (FALSE);
}
void CPaneFrame::ShowMenu(void)
{
if (m_idMenu)
{
_OnLButtonDown(m_hwnd, 0, m_rcTitleButton.left, m_rcTitleButton.top, 0);
}
}
//
// FUNCTION: CPaneFrame::QueryInterface()
//
// PURPOSE: Allows caller to retrieve the various interfaces supported by
// this class.
//
HRESULT CPaneFrame::QueryInterface(REFIID riid, LPVOID *ppvObj)
{
TraceCall("CPaneFrame::QueryInterface");
*ppvObj = NULL;
if (IsEqualIID(riid, IID_IUnknown))
*ppvObj = (LPVOID) (IInputObjectSite *) this;
else if (IsEqualIID(riid, IID_IInputObjectSite))
*ppvObj = (LPVOID) (IInputObjectSite *) this;
if (*ppvObj)
{
AddRef();
return (S_OK);
}
return (E_NOINTERFACE);
}
//
// FUNCTION: CPaneFrame::AddRef()
//
// PURPOSE: Adds a reference count to this object.
//
ULONG CPaneFrame::AddRef(void)
{
TraceCall("CPaneFrame::AddRef");
return ((ULONG) InterlockedIncrement((LONG *) &m_cRef));
}
//
// FUNCTION: CPaneFrame::Release()
//
// PURPOSE: Releases a reference on this object.
//
ULONG CPaneFrame::Release(void)
{
TraceCall("CPaneFrame::Release");
if (0 == InterlockedDecrement((LONG *) &m_cRef))
{
delete this;
return 0;
}
return (m_cRef);
}
HRESULT CPaneFrame::OnFocusChangeIS(IUnknown *punkSrc, BOOL fSetFocus)
{
// Simply call through to our host
UnkOnFocusChangeIS(m_pSite, (IInputObject*) this, fSetFocus);
return (S_OK);
}
//
// FUNCTION: CPaneFrame::_WndProc()
//
// PURPOSE: External callback.
//
LRESULT CALLBACK CPaneFrame::_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
CPaneFrame *pThis;
if (uMsg == WM_NCCREATE)
{
pThis = (CPaneFrame *) ((LPCREATESTRUCT) lParam)->lpCreateParams;
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LPARAM) pThis);
}
else
pThis = (CPaneFrame *) GetWindowLongPtr(hwnd, GWLP_USERDATA);
if (pThis)
return (pThis->_FrameWndProc(hwnd, uMsg, wParam, lParam));
return (FALSE);
}
//
// FUNCTION: CPaneFrame::_FrameWndProc()
//
// PURPOSE: Left as an exercise for the reader
//
LRESULT CALLBACK CPaneFrame::_FrameWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
HANDLE_MSG(hwnd, WM_CREATE, _OnCreate);
HANDLE_MSG(hwnd, WM_SIZE, _OnSize);
HANDLE_MSG(hwnd, WM_PAINT, _OnPaint);
HANDLE_MSG(hwnd, WM_COMMAND, _OnCommand);
HANDLE_MSG(hwnd, WM_MOUSEMOVE, _OnMouseMove);
HANDLE_MSG(hwnd, WM_LBUTTONDOWN, _OnLButtonDown);
HANDLE_MSG(hwnd, WM_TIMER, _OnTimer);
case WM_TOGGLE_CLOSE_PIN:
_OnToggleClosePin(hwnd, (BOOL) lParam);
return (0);
case WM_GET_TITLE_BAR_HEIGHT:
return (m_cyTitleBar + (c_cyBorder * 2) + 1);
case WM_SYSCOLORCHANGE:
case WM_WININICHANGE:
{
// Forward these to all our children
if (IsWindow(m_hwndChild))
SendMessage(m_hwndChild, uMsg, wParam, lParam);
_UpdateDrawingInfo();
break;
}
case WM_SETFOCUS:
{
if (m_hwndChild && ((HWND)wParam) != m_hwndChild)
SetFocus(m_hwndChild);
break;
}
}
return (DefWindowProc(hwnd, uMsg, wParam, lParam));
}
//
// FUNCTION: CPaneFrame::_OnCreate()
//
// PURPOSE: Loads some info that will be handy later
//
BOOL CPaneFrame::_OnCreate(HWND hwnd, LPCREATESTRUCT lpCreateStruct)
{
TraceCall("CPaneFrame::_OnCreate");
m_hwnd = hwnd;
_UpdateDrawingInfo();
_CreateCloseToolbar();
return (TRUE);
}
//
// FUNCTION: CPaneFrame::_OnSize()
//
// PURPOSE: Resizes our child to fit in the right place
//
void CPaneFrame::_OnSize(HWND hwnd, UINT state, int cx, int cy)
{
TraceCall("CPaneFrame::_OnSize");
m_rcChild.left = c_cyBorder;
m_rcChild.top = m_cyTitleBar;
m_rcChild.right = cx - (2 * c_cyBorder);
m_rcChild.bottom = cy - m_cyTitleBar - c_cyBorder;
if (m_hwndChild)
SetWindowPos(m_hwndChild, 0, m_rcChild.left, m_rcChild.top, m_rcChild.right,
m_rcChild.bottom, SWP_NOZORDER | SWP_NOACTIVATE);
POINT pt = {cx, cy};
_PositionToolbar(&pt);
// Invalidate the title area
RECT rc = m_rcChild;
rc.top = 0;
rc.bottom = m_rcChild.top;
InvalidateRect(m_hwnd, &rc, FALSE);
rc.left = 0;
rc.right = c_cyBorder;
rc.bottom = cy;
InvalidateRect(m_hwnd, &rc, FALSE);
rc.left = cx - c_cyBorder;
rc.right = cx;
InvalidateRect(m_hwnd, &rc, FALSE);
}
//
// FUNCTION: CPaneFrame::_OnPaint()
//
// PURPOSE: Called when it's time to paint our borders and title area.
//
void CPaneFrame::_OnPaint(HWND hwnd)
{
HDC hdc;
PAINTSTRUCT ps;
RECT rc;
RECT rcClient;
POINT pt[3];
HBRUSH hBrush,
hBrushOld;
HPEN hPen,
hPenOld;
// Get our window size
GetClientRect(m_hwnd, &rcClient);
rc = rcClient;
// Start painting
hdc = BeginPaint(hwnd, &ps);
// Draw a simple edge around or window
DrawEdge(hdc, &rc, BDR_SUNKENOUTER, BF_TOPRIGHT | BF_BOTTOMLEFT);
// Now draw a raised edge around our title bar area
InflateRect(&rc, -1, -1);
rc.bottom = m_cyTitleBar;
DrawEdge(hdc, &rc, BDR_RAISEDINNER, BF_TOPRIGHT | BF_BOTTOMLEFT);
// Paint the background
InflateRect(&rc, -c_cxBorder, -c_cyBorder);
FillRect(hdc, &rc, m_hbr3DFace);
// Now draw some groovy text
SelectFont(hdc, m_hFont);
SetBkColor(hdc, GetSysColor(COLOR_3DFACE));
SetTextColor(hdc, GetSysColor(COLOR_BTNTEXT));
// Draw the text
InflateRect(&rc, -c_cxTextBorder, -c_cyTextBorder);
if (!m_fPin)
{
DrawText(hdc, m_szTitle, -1, &rc, DT_CALCRECT | DT_VCENTER | DT_LEFT);
DrawText(hdc, m_szTitle, -1, &rc, DT_VCENTER | DT_LEFT);
}
else
{
TCHAR sz[CCHMAX_STRINGRES];
AthLoadString(idsPushPinInfo, sz, ARRAYSIZE(sz));
IDrawText(hdc, sz, &rc, DT_VCENTER | DT_END_ELLIPSIS | DT_LEFT,
rc.bottom - rc.top);
DrawText(hdc, sz, -1, &rc, DT_CALCRECT | DT_VCENTER | DT_END_ELLIPSIS | DT_LEFT);
}
// Drop-down indicator
if (m_idMenu)
{
COLORREF crFG = GetSysColor(COLOR_WINDOWTEXT);
pt[0].x = rc.right + 6;
pt[0].y = (m_cyTitleBar - 6) / 2 + 2;
pt[1].x = pt[0].x + 6;
pt[1].y = pt[0].y;
pt[2].x = pt[0].x + 3;
pt[2].y = pt[0].y + 3;
hPen = CreatePen(PS_SOLID, 1, crFG);
hBrush = CreateSolidBrush(crFG);
hPenOld = SelectPen(hdc, hPen);
hBrushOld = SelectBrush(hdc, hBrush);
Polygon(hdc, pt, 3);
SelectPen(hdc, hPenOld);
SelectBrush(hdc, hBrushOld);
DeleteObject(hPen);
DeleteObject(hBrush);
if (m_fHighlightIndicator)
{
rc = m_rcTitleButton;
DrawEdge(hdc, &rc, m_fHighlightPressed ? BDR_SUNKENOUTER : BDR_RAISEDINNER,
BF_TOPRIGHT | BF_BOTTOMLEFT);
}
}
EndPaint(hwnd, &ps);
}
//
// FUNCTION: _OnCommand()
//
// PURPOSE: We get the occasional command now and again
//
void CPaneFrame::_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)
{
switch (id)
{
case ID_PANE_CLOSE:
{
if (m_pBrowser)
m_pBrowser->SetViewLayout(m_dwDispId, LAYOUT_POS_NA, FALSE, 0, 0);
return;
}
case ID_PANE_PIN:
{
SendMessage(m_hwndChild, WMR_CLICKOUTSIDE, CLK_OUT_DEACTIVATE, 0);
if (m_pBrowser)
m_pBrowser->SetViewLayout(m_dwDispId, LAYOUT_POS_NA, TRUE, 0, 0);
return;
}
}
return;
}
//
// FUNCTION: CPaneFrame::_OnToggleClosePin()
//
// PURPOSE: Sent to the frame when we should change the close button
// to a pin button.
//
// PARAMETERS:
// [in] fPin - TRUE to turn the Pin on, FALSE to turn it off.
//
void CPaneFrame::_OnToggleClosePin(HWND hwnd, BOOL fPin)
{
TraceCall("CPaneFrame::_OnToggleClosePin");
if (fPin)
{
static const TBBUTTON tb[] =
{
{ 2, ID_PANE_PIN, TBSTATE_ENABLED, TBSTYLE_BUTTON, {0, 0}, 0, 0}
};
SendMessage(m_hwndClose, TB_DELETEBUTTON, 0, 0);
SendMessage(m_hwndClose, TB_ADDBUTTONS, ARRAYSIZE(tb), (LPARAM) tb);
SendMessage(m_hwndClose, TB_SETHOTITEM, (WPARAM) -1, 0);
m_fPin = TRUE;
}
else
{
static const TBBUTTON tb[] =
{
{ 1, ID_PANE_CLOSE, TBSTATE_ENABLED, TBSTYLE_BUTTON, {0, 0}, 0, 0}
};
SendMessage(m_hwndClose, TB_DELETEBUTTON, 0, 0);
SendMessage(m_hwndClose, TB_ADDBUTTONS, ARRAYSIZE(tb), (LPARAM) tb);
SendMessage(m_hwndClose, TB_SETHOTITEM, (WPARAM) -1, 0);
m_fPin = FALSE;
}
}
//
// FUNCTION: CPaneFrame::_UpdateDrawingInfo()
//
// PURPOSE: When we get created or when the user changes their settings,
// we need to reload our fonts, colors, and sizes.
//
void CPaneFrame::_UpdateDrawingInfo(void)
{
LOGFONT lf;
TEXTMETRIC tm;
HDC hdc;
TraceCall("CPaneFrame::_UpdateDrawingInfo");
if (m_hFont)
DeleteObject(m_hFont);
// Figure out which font to use
SystemParametersInfo(SPI_GETICONTITLELOGFONT, sizeof(LOGFONT), &lf, FALSE);
// Create the font
m_hFont = CreateFontIndirect(&lf);
// Get the metrics of this font
hdc = GetDC(m_hwnd);
SelectFont(hdc, m_hFont);
GetTextMetrics(hdc, &tm);
// Calculate the height
m_cyTitleBar = tm.tmHeight + (2 * c_cyBorder) + (2 * c_cyTextBorder);
RECT rc = {2 * c_cxBorder, 2 * c_cyBorder, 0, m_cyTitleBar - c_cyBorder};
SIZE s;
GetTextExtentPoint32(hdc, m_szTitle, lstrlen(m_szTitle), &s);
m_rcTitleButton = rc;
m_rcTitleButton.right = 14 + (2 * c_cxTextBorder) + s.cx + (2 * c_cxBorder);
ReleaseDC(m_hwnd, hdc);
// Get the brush we need
if (m_hbr3DFace)
DeleteObject(m_hbr3DFace);
m_hbr3DFace = CreateSolidBrush(GetSysColor(COLOR_3DFACE));
}
//
// FUNCTION: CPaneFrame::_CreateCloseToolbar()
//
// PURPOSE: Creates the toolbar that has our close button
//
void CPaneFrame::_CreateCloseToolbar()
{
CHAR szTitle[255];
TraceCall("CPaneFrame::_CreateCloseToolbar");
AthLoadString(idsHideFolders, szTitle, ARRAYSIZE(szTitle));
m_hwndClose = CreateWindowEx(0, TOOLBARCLASSNAME, szTitle,
WS_VISIBLE | WS_CHILD | TBSTYLE_FLAT | TBSTYLE_CUSTOMERASE |
WS_CLIPCHILDREN | WS_CLIPSIBLINGS | CCS_NODIVIDER | CCS_NOMOVEY |
CCS_NOPARENTALIGN | CCS_NORESIZE,
0, c_cyClose, 30, 15, m_hwnd, 0, g_hInst, NULL);
if (m_hwndClose)
{
static const TBBUTTON tb[] =
{
{ 1, ID_PANE_CLOSE, TBSTATE_ENABLED, TBSTYLE_BUTTON, {0, 0}, 0, 0}
};
SendMessage(m_hwndClose, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0);
SendMessage(m_hwndClose, TB_SETBITMAPSIZE, 0, (LPARAM) MAKELONG(11, 9));
TBADDBITMAP tbab = { g_hLocRes, idbClosePin };
SendMessage(m_hwndClose, TB_ADDBITMAP, 4, (LPARAM) &tbab);
SendMessage(m_hwndClose, TB_ADDBUTTONS, ARRAYSIZE(tb), (LPARAM) tb);
SendMessage(m_hwndClose, TB_SETINDENT, 0, 0);
_SizeCloseToolbar();
}
}
//
// FUNCTION: CPaneFrame::_SizeCloseToolbar()
//
// PURPOSE: Set's the size of the toolbar appropriately.
//
void CPaneFrame::_SizeCloseToolbar(void)
{
TraceCall("CPaneFrame::_SizeCloseToolbar");
RECT rc;
LONG lButtonSize;
GetWindowRect(m_hwndClose, &rc);
lButtonSize = (LONG) SendMessage(m_hwndClose, TB_GETBUTTONSIZE, 0, 0L);
SetWindowPos(m_hwndClose, NULL, 0, 0, LOWORD(lButtonSize) * m_cButtons,
rc.bottom - rc.top, SWP_NOMOVE | SWP_NOACTIVATE);
_PositionToolbar(NULL);
}
//
// FUNCTION: CPaneFrame::_PositionToolbar()
//
// PURPOSE: Does the work of correctly positioning the close button
// toolbar.
//
// PARAMETERS:
// LPPOINT ppt
//
void CPaneFrame::_PositionToolbar(LPPOINT ppt)
{
TraceCall("CPaneFrame::_PositionToolbar");
if (m_hwndClose)
{
RECT rc;
GetClientRect(m_hwnd, &rc);
if (ppt)
{
rc.left = 0;
rc.right = ppt->x;
}
RECT rcTB;
GetWindowRect(m_hwndClose, &rcTB);
rc.left = rc.right - (rcTB.right - rcTB.left) - 3;
DWORD top = max((int) ((m_cyTitleBar - (rcTB.bottom - rcTB.top)) / 2) + 1, 0);
SetWindowPos(m_hwndClose, HWND_TOP, rc.left, top, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
}
}
void CPaneFrame::_OnLButtonDown(HWND hwnd, BOOL fDoubleClick, int x, int y, UINT keyFlags)
{
POINT pt = {x, y};
UINT id;
if (m_idMenu && PtInRect(&m_rcTitleButton, pt))
{
m_fHighlightPressed = TRUE;
InvalidateRect(m_hwnd, &m_rcTitleButton, TRUE);
UpdateWindow(m_hwnd);
HMENU hMenu = LoadPopupMenu(m_idMenu);
MenuUtil_EnablePopupMenu(hMenu, m_pTarget);
if (m_idMenu == IDR_BA_TITLE_POPUP && ((g_dwHideMessenger == BL_HIDE) || (g_dwHideMessenger == BL_DISABLE)))
{
DeleteMenu(hMenu, ID_NEW_ONLINE_CONTACT, MF_BYCOMMAND);
DeleteMenu(hMenu, ID_SET_ONLINE_CONTACT, MF_BYCOMMAND);
DeleteMenu(hMenu, SEP_MESSENGER, MF_BYCOMMAND);
DeleteMenu(hMenu, ID_SORT_BY_NAME, MF_BYCOMMAND);
DeleteMenu(hMenu, ID_SORT_BY_STATUS, MF_BYCOMMAND);
}
pt.x = m_rcTitleButton.left;
pt.y = m_rcTitleButton.bottom;
ClientToScreen(m_hwnd, &pt);
id = TrackPopupMenuEx(hMenu, TPM_RETURNCMD | TPM_LEFTALIGN | TPM_LEFTBUTTON | TPM_RIGHTBUTTON,
pt.x, pt.y, m_hwnd, NULL);
if (id)
{
m_pTarget->Exec(NULL, id, OLECMDEXECOPT_DODEFAULT, NULL, NULL);
}
m_fHighlightPressed = m_fHighlightIndicator = FALSE;
KillTimer(m_hwnd, IDT_PANETIMER);
InvalidateRect(m_hwnd, &m_rcTitleButton, TRUE);
UpdateWindow(m_hwnd);
if(hMenu)
{
//Bug #101329 - (erici) Destroy leaked MENU.
BOOL bMenuDestroyed = DestroyMenu(hMenu);
Assert(bMenuDestroyed);
}
}
}
void CPaneFrame::_OnMouseMove(HWND hwnd, int x, int y, UINT keyFlags)
{
POINT pt = {x, y};
if (m_idMenu && (m_fHighlightIndicator != PtInRect(&m_rcTitleButton, pt)))
{
m_fHighlightIndicator = !m_fHighlightIndicator;
InvalidateRect(m_hwnd, &m_rcTitleButton, TRUE);
if (m_fHighlightIndicator)
SetTimer(m_hwnd, IDT_PANETIMER, ELAPSE_MOUSEOVERCHECK, NULL);
else
KillTimer(m_hwnd, IDT_PANETIMER);
}
}
void CPaneFrame::_OnTimer(HWND hwnd, UINT id)
{
RECT rcClient;
POINT pt;
DWORD dw;
dw = GetMessagePos();
pt.x = LOWORD(dw);
pt.y = HIWORD(dw);
ScreenToClient(m_hwnd, &pt);
if (id == IDT_PANETIMER)
{
GetClientRect(m_hwnd, &rcClient);
// No need to handle mouse in client area, OnMouseMove will catch this. We
// only need to catch the mouse moving out of the client area.
if (!PtInRect(&rcClient, pt) && !m_fHighlightPressed)
{
KillTimer(m_hwnd, IDT_PANETIMER);
m_fHighlightIndicator = FALSE;
InvalidateRect(m_hwnd, &m_rcTitleButton, TRUE);
}
}
}