//----------------------------------------------------------------------------- // File: flexwnd.cpp // // Desc: CFlexWnd is a generic class that encapsulates the functionalities // of a window. All other window classes are derived from CFlexWnd. // // Child classes can have different behavior by overriding the // overridable message handlers (OnXXX members). // // Copyright (C) 1999-2000 Microsoft Corporation. All Rights Reserved. //----------------------------------------------------------------------------- #include "common.hpp" #include "typeinfo.h" BOOL CFlexWnd::sm_bWndClassRegistered = FALSE; WNDCLASSEX CFlexWnd::sm_WndClass; LPCTSTR CFlexWnd::sm_tszWndClassName = _T("Microsoft.CFlexWnd.WndClassName"); HINSTANCE CFlexWnd::sm_hInstance = NULL; CFlexToolTip CFlexWnd::s_ToolTip; // Shared tooltip window object DWORD CFlexWnd::s_dwLastMouseMove; // Last GetTickCount() that we have a WM_MOUSEMOVE HWND CFlexWnd::s_hWndLastMouseMove; // Last window handle of WM_MOUSEMOVE LPARAM CFlexWnd::s_PointLastMouseMove; // Last point of WM_MOUSEMOVE HWND CFlexWnd::s_CurrPageHwnd; // For unhighlighting callouts when a click is made outside of a callout int NewID() { static int i = 0; return ++i; } CFlexWnd::CFlexWnd() : m_nID(NewID()), m_hWnd(m_privhWnd), m_privhWnd(NULL), m_hRenderInto(NULL), m_bIsDialog(FALSE), m_bRender(FALSE), m_bReadOnly(FALSE) { } CFlexWnd::~CFlexWnd() { Destroy(); } void CFlexWnd::Destroy() { if (m_hWnd != NULL) DestroyWindow(m_hWnd); assert(m_privhWnd == NULL); } BOOL CFlexWnd::IsDialog() { return HasWnd() && m_bIsDialog; } void CFlexWnd::OnRender(BOOL bInternalCall) { // if parent is flexwnd and both are in render mode, pass to parent if (!m_hWnd) return; HWND hParent = GetParent(m_hWnd); if (!hParent) return; CFlexWnd *pParent = GetFlexWnd(hParent); if (!pParent) return; if (pParent->InRenderMode() && InRenderMode()) pParent->OnRender(TRUE); } BOOL CFlexWnd::OnEraseBkgnd(HDC hDC) { if (InRenderMode()) return TRUE; /* if (IsDialog()) return FALSE;*/ return TRUE; } struct GETFLEXWNDSTRUCT { int cbSize; BOOL bFlexWnd; CFlexWnd *pFlexWnd; }; // This function takes a HWND and returns a pointer to CFlexWnd if the HWND is a window // created by the UI. CFlexWnd *CFlexWnd::GetFlexWnd(HWND hWnd) { if (hWnd == NULL) return NULL; GETFLEXWNDSTRUCT gfws; gfws.cbSize = sizeof(gfws); gfws.bFlexWnd = FALSE; gfws.pFlexWnd = NULL; SendMessage(hWnd, WM_GETFLEXWND, 0, (LPARAM)(LPVOID)(FAR GETFLEXWNDSTRUCT *)&gfws); if (gfws.bFlexWnd) return gfws.pFlexWnd; else return NULL; } // Basic window proc. It simply forward interesting messages to the appropriate handlers (OnXXX). // If child class defines this function, it should pass unhandled messages to CFlexWnd. LRESULT CFlexWnd::WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch (msg) { case WM_GETFLEXWND: { if ((LPVOID)lParam == NULL) break; GETFLEXWNDSTRUCT &gfws = *((FAR GETFLEXWNDSTRUCT *)(LPVOID)lParam); switch (gfws.cbSize) { case sizeof(GETFLEXWNDSTRUCT): gfws.bFlexWnd = TRUE; gfws.pFlexWnd = this; return 0; default: assert(0); break; } break; } case WM_CREATE: { LPCREATESTRUCT lpCreateStruct = (LPCREATESTRUCT)lParam; LRESULT lr = OnCreate(lpCreateStruct); if (lr != -1) OnInit(); return lr; } case WM_INITDIALOG: { BOOL b = OnInitDialog(); OnInit(); return b; } case WM_TIMER: OnTimer((UINT)wParam); return 0; case WM_ERASEBKGND: return OnEraseBkgnd((HDC)wParam); case WM_PAINT: { // Check the update rectangle. If we don't have it, exit immediately. if (typeid(*this) == typeid(CDeviceView) && !GetUpdateRect(m_hWnd, NULL, FALSE)) return 0; PAINTSTRUCT ps; HDC hDC = BeginPaint(hWnd, &ps); if (InRenderMode()) OnRender(TRUE); else DoOnPaint(hDC); EndPaint(hWnd, &ps); return 0; } case WM_COMMAND: { WORD wNotifyCode = HIWORD(wParam); WORD wID = LOWORD(wParam); HWND hWnd = (HWND)lParam; return OnCommand(wNotifyCode, wID, hWnd); } case WM_NOTIFY: return OnNotify(wParam, lParam); case WM_MOUSEMOVE: case WM_LBUTTONDOWN: case WM_RBUTTONDOWN: case WM_LBUTTONDBLCLK: case WM_MOUSEWHEEL: { POINT point = {int(LOWORD(lParam)), int(HIWORD(lParam))}; switch (msg) { case WM_MOUSEMOVE: OnMouseOver(point, wParam); break; case WM_LBUTTONDOWN: OnClick(point, wParam, TRUE); break; case WM_RBUTTONDOWN: OnClick(point, wParam, FALSE); break; case WM_LBUTTONDBLCLK: OnDoubleClick(point, wParam, TRUE); break; case WM_MOUSEWHEEL: { // Send wheel msg to the window beneath the cursor HWND hWnd = WindowFromPoint(point); CFlexWnd *pWnd = NULL; if (hWnd) { pWnd = GetFlexWnd(hWnd); if (pWnd) pWnd->OnWheel(point, wParam); else return DefWindowProc(hWnd, msg, wParam, lParam); } break; } } return 0; } case WM_DESTROY: OnDestroy(); m_privhWnd = NULL; return 0; } if (!m_bIsDialog) return DefWindowProc(hWnd, msg, wParam, lParam); else return 0; } //@@BEGIN_MSINTERNAL // TODO: better control id thingy //@@END_MSINTERNAL static HMENU windex = 0; BOOL CFlexWnd::EndDialog(int n) { if (!m_bIsDialog || m_hWnd == NULL) { assert(0); return FALSE; } return ::EndDialog(m_hWnd, n); } int CFlexWnd::DoModal(HWND hParent, int nTemplate, HINSTANCE hInst) { return DoModal(hParent, MAKEINTRESOURCE(nTemplate), hInst); } HWND CFlexWnd::DoModeless(HWND hParent, int nTemplate, HINSTANCE hInst) { return DoModeless(hParent, MAKEINTRESOURCE(nTemplate), hInst); } int CFlexWnd::DoModal(HWND hParent, LPCTSTR lpTemplate, HINSTANCE hInst) { if (m_hWnd != NULL) { assert(0); return -1; } if (hInst == NULL) hInst = CFlexWnd::sm_hInstance; return (int)DialogBoxParam(hInst, lpTemplate, hParent, __BaseFlexWndDialogProc, (LPARAM)(void *)this); } HWND CFlexWnd::DoModeless(HWND hParent, LPCTSTR lpTemplate, HINSTANCE hInst) { if (m_hWnd != NULL) { assert(0); return NULL; } if (hInst == NULL) hInst = CFlexWnd::sm_hInstance; return CreateDialogParam(hInst, lpTemplate, hParent, __BaseFlexWndDialogProc, (LPARAM)(void *)this); } HWND CFlexWnd::Create(HWND hParent, const RECT &rect, BOOL bVisible) { ++(*(LPBYTE*)&windex); return Create(hParent, _T("(unnamed)"), 0, WS_CHILD | WS_CLIPCHILDREN | WS_CLIPSIBLINGS | WS_EX_NOPARENTNOTIFY | (bVisible ? WS_VISIBLE : 0), rect, windex); } HWND CFlexWnd::Create(HWND hParent, LPCTSTR tszName, DWORD dwExStyle, DWORD dwStyle, const RECT &rect, HMENU hMenu) { HWND hWnd = NULL; if (m_hWnd != NULL) { assert(0); return hWnd; } if (hMenu == NULL && (dwStyle & WS_CHILD)) { ++(*(LPBYTE*)&windex); hMenu = windex; } hWnd = CreateWindowEx( dwExStyle, CFlexWnd::sm_tszWndClassName, tszName, dwStyle, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, hParent, hMenu, CFlexWnd::sm_hInstance, (void *)this); assert(m_hWnd == hWnd); return hWnd; } void CFlexWnd::SetHWND(HWND hWnd) { assert(m_hWnd == NULL && hWnd != NULL); m_privhWnd = hWnd; assert(m_hWnd == m_privhWnd); InitFlexWnd(); } void CFlexWnd::InitFlexWnd() { if (!HasWnd()) return; HWND hParent = GetParent(m_hWnd); CFlexWnd *pParent = GetFlexWnd(hParent); if (pParent && pParent->InRenderMode()) SetRenderMode(); } TCHAR sg_tszFlexWndPointerProp[] = _T("CFlexWnd *"); LRESULT CALLBACK __BaseFlexWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { CFlexWnd *pThis = (CFlexWnd *)GetProp(hWnd, sg_tszFlexWndPointerProp); if ((msg == WM_MOUSEMOVE || msg == WM_MOUSEWHEEL) && hWnd != CFlexWnd::s_ToolTip.m_hWnd) { // Filter out the message with same window handle and point. // Windows sometimes seems to send us WM_MOUSEMOVE message even though the mouse is not moved. if (CFlexWnd::s_hWndLastMouseMove != hWnd || CFlexWnd::s_PointLastMouseMove != lParam) { CFlexWnd::s_hWndLastMouseMove = hWnd; CFlexWnd::s_PointLastMouseMove = lParam; CFlexWnd::s_dwLastMouseMove = GetTickCount(); // Get timestamp CFlexWnd::s_ToolTip.SetEnable(FALSE); CFlexWnd::s_ToolTip.SetToolTipParent(NULL); } } switch (msg) { case WM_CREATE: { LPCREATESTRUCT lpcs = (LPCREATESTRUCT)lParam; if (lpcs == NULL) break; pThis = (CFlexWnd *)(void *)(lpcs->lpCreateParams); assert(sizeof(HANDLE) == sizeof(CFlexWnd *)); SetProp(hWnd, sg_tszFlexWndPointerProp, (HANDLE)pThis); if (pThis != NULL) { pThis->m_bIsDialog = FALSE; pThis->SetHWND(hWnd); } break; } } if (pThis != NULL) return pThis->WndProc(hWnd, msg, wParam, lParam); else return DefWindowProc(hWnd, msg, wParam, lParam); } INT_PTR CALLBACK __BaseFlexWndDialogProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { CFlexWnd *pThis = (CFlexWnd *)GetProp(hWnd, sg_tszFlexWndPointerProp); switch (msg) { case WM_INITDIALOG: pThis = (CFlexWnd *)(void *)lParam; assert(sizeof(HANDLE) == sizeof(CFlexWnd *)); SetProp(hWnd, sg_tszFlexWndPointerProp, (HANDLE)pThis); if (pThis != NULL) { pThis->m_bIsDialog = TRUE; pThis->SetHWND(hWnd); } break; } if (pThis != NULL) return (BOOL)pThis->WndProc(hWnd, msg, wParam, lParam); else return FALSE; } void CFlexWnd::Invalidate() { if (m_hWnd != NULL) InvalidateRect(m_hWnd, NULL, TRUE); } SIZE CFlexWnd::GetClientSize() const { RECT rect = {0, 0, 0, 0}; if (m_hWnd != NULL) ::GetClientRect(m_hWnd, &rect); SIZE size = { rect.right - rect.left, rect.bottom - rect.top}; return size; } void CFlexWnd::FillWndClass(HINSTANCE hInst) { sm_WndClass.cbSize = sizeof(WNDCLASSEX); sm_WndClass.style = CS_DBLCLKS; sm_WndClass.lpfnWndProc = __BaseFlexWndProc; sm_WndClass.cbClsExtra = 0; sm_WndClass.cbWndExtra = sizeof(CFlexWnd *); sm_WndClass.hInstance = sm_hInstance = hInst; sm_WndClass.hIcon = NULL; sm_WndClass.hCursor = NULL; sm_WndClass.hbrBackground = NULL; sm_WndClass.lpszMenuName = NULL; sm_WndClass.lpszClassName = sm_tszWndClassName; sm_WndClass.hIconSm = NULL; } void CFlexWnd::RegisterWndClass(HINSTANCE hInst) { if (hInst == NULL) { assert(0); return; } FillWndClass(hInst); RegisterClassEx(&sm_WndClass); sm_bWndClassRegistered = TRUE; } void CFlexWnd::UnregisterWndClass(HINSTANCE hInst) { if (hInst == NULL) return; UnregisterClass(sm_tszWndClassName, hInst); sm_bWndClassRegistered = FALSE; } void CFlexWnd::GetClientRect(LPRECT lprect) const { if (lprect == NULL || m_hWnd == NULL) return; ::GetClientRect(m_hWnd, lprect); } LPCTSTR CFlexWnd::GetDefaultClassName() { return CFlexWnd::sm_tszWndClassName; } void CFlexWnd::SetRenderMode(BOOL bRender) { if (bRender == m_bRender) return; m_bRender = bRender; Invalidate(); } BOOL CFlexWnd::InRenderMode() { return m_bRender; } void EnumChildWindowsZDown(HWND hParent, WNDENUMPROC proc, LPARAM lParam) { if (hParent == NULL || proc == NULL) return; HWND hWnd = GetWindow(hParent, GW_CHILD); while (hWnd != NULL) { if (!proc(hWnd, lParam)) break; hWnd = GetWindow(hWnd, GW_HWNDNEXT); } } void EnumSiblingsAbove(HWND hParent, WNDENUMPROC proc, LPARAM lParam) { if (hParent == NULL || proc == NULL) return; HWND hWnd = hParent; while (1) { hWnd = GetWindow(hWnd, GW_HWNDPREV); if (hWnd == NULL) break; if (!proc(hWnd, lParam)) break; } } static BOOL CALLBACK RenderIntoClipChild(HWND hWnd, LPARAM lParam) { CFlexWnd *pThis = (CFlexWnd *)(LPVOID)lParam; return pThis->RenderIntoClipChild(hWnd); } static BOOL CALLBACK RenderIntoRenderChild(HWND hWnd, LPARAM lParam) { CFlexWnd *pThis = (CFlexWnd *)(LPVOID)lParam; // Check if this is the immediate child. Do nothing if it's not immediate. HWND hParent = GetParent(hWnd); if (hParent != pThis->m_hWnd) return TRUE; return pThis->RenderIntoRenderChild(hWnd); } BOOL CFlexWnd::RenderIntoClipChild(HWND hChild) { if (m_hRenderInto != NULL && HasWnd() && hChild && IsWindowVisible(hChild)) { RECT rect; GetWindowRect(hChild, &rect); POINT ul = {rect.left, rect.top}, lr = {rect.right, rect.bottom}; ScreenToClient(m_hWnd, &ul); ScreenToClient(m_hWnd, &lr); ExcludeClipRect(m_hRenderInto, ul.x, ul.y, lr.x, lr.y); } return TRUE; } BOOL CFlexWnd::RenderIntoRenderChild(HWND hChild) { CFlexWnd *pChild = GetFlexWnd(hChild); if (m_hRenderInto != NULL && HasWnd() && pChild != NULL && IsWindowVisible(hChild)) { RECT rect; GetWindowRect(hChild, &rect); POINT ul = {rect.left, rect.top}; ScreenToClient(m_hWnd, &ul); pChild->RenderInto(m_hRenderInto, ul.x, ul.y); } return TRUE; } void CFlexWnd::RenderInto(HDC hDC, int x, int y) { if (hDC == NULL) return; int sdc = SaveDC(hDC); { OffsetViewportOrgEx(hDC, x, y, NULL); SIZE size = GetClientSize(); IntersectClipRect(hDC, 0, 0, size.cx, size.cy); m_hRenderInto = hDC; int sdc2 = SaveDC(hDC); { EnumChildWindows/*ZDown*/(m_hWnd, ::RenderIntoClipChild, (LPARAM)(PVOID)this); EnumSiblingsAbove(m_hWnd, ::RenderIntoClipChild, (LPARAM)(PVOID)this); DoOnPaint(hDC); } if (sdc2) RestoreDC(hDC, sdc2); EnumChildWindows/*ZDown*/(m_hWnd, ::RenderIntoRenderChild, (LPARAM)(PVOID)this); m_hRenderInto = NULL; } if (sdc) RestoreDC(hDC, sdc); } void CFlexWnd::SetCapture() { ::SetCapture(m_hWnd); } void CFlexWnd::ReleaseCapture() { ::ReleaseCapture(); } void CFlexWnd::DoOnPaint(HDC hDC) { OnPaint(hDC); }