FIX: Border of Modeless CPropertySheet Is Not 3D in Windows NT
ID: Q147206
|
The information in this article applies to:
-
The Microsoft Foundation Classes (MFC), included with:
-
Microsoft Visual C++, 32-bit Editions, version 4.0
SYMPTOMS
If you create a modeless CPropertySheet under Windows NT, its outer border
will not be a 3D border. A modal CPropertySheet will have a 3D border. Both
modal and modeless CPropertySheets have a 3D border under Windows 95.
CAUSE
MFC sets up a callback function for modeless CPropertySheets called
AfxPropSheetCallback(). This function checks for PSCB_PRECREATE, which is a
message that is received before a property sheet is created. In this
handler, MFC takes out the DS_MODALFRAME style.
Ctl3d32.dll is used by Windows NT to paint 3D borders for controls and
windows. For a dialog box to have a 3D border, CTL3D requires a style of
DS_MODALFRAME.
RESOLUTION
- Derive a class from CPropertySheet (CMySheet) and override its
Create(). Comment the call to CWnd::Create() that the ClassWizard
puts in.
- Copy code from CPropertySheet::Create() in DLGPROP.CPP (as shown
below) into CMySheet::Create(), and change the parameters it takes
accordingly.
- Change this line:
m_psh.dwFlags |= (PSH_MODELESS|PSH_USECALLBACK);
to:
m_psh.dwFlags |= PSH_MODELESS;
The callback function isn't set so that PSCB_PRECREATE is not handled,
and the DS_MODALFRAME style is not removed.
- Comment out this line:
m_psh.pfnCallback = AfxPropSheetCallback;
- Include <afxpriv.h> in MySheet.h.
- In MySheet.h, change the MySheet.Create to take the corresponding
parameters:
BOOL Create(CWnd* pParentWnd = NULL, DWORD dwStyle = WS_SYSMENU |
WS_POPUP | WS_CAPTION | DS_MODALFRAME | WS_VISIBLE,
DWORD dwExStyle = WS_EX_DLGMODALFRAME );
STATUS
Microsoft has confirmed this to be a bug in the Microsoft products listed
at the beginning of this article. This problem was corrected in Visual C++,
32-bit Edition, version 4.1.
MORE INFORMATION
Sample Code
// in MySheet.h
#include <afxpriv.h>
BOOL Create(CWnd* pParentWnd = NULL, DWORD dwStyle = WS_SYSMENU |
WS_POPUP | WS_CAPTION | DS_MODALFRAME | WS_VISIBLE,
DWORD dwExStyle = WS_EX_DLGMODALFRAME );
// In MySheet.cpp
// CMySheet is derived from CPropertySheet
BOOL CMySheet::Create(CWnd* pParentWnd, DWORD dwStyle, DWORD dwExStyle)
{
_AFX_THREAD_STATE* pState = AfxGetThreadState();
pState->m_dwPropStyle = dwStyle;
pState->m_dwPropExStyle = dwExStyle;
ASSERT_VALID(this);
ASSERT(m_hWnd == NULL);
// WS_SYSMENU must not be set if a property sheet is created as a child
if (dwStyle & WS_CHILD)
dwStyle &= ~WS_SYSMENU;
// finish building PROPSHEETHEADER structure
BuildPropPageArray();
m_bModeless = TRUE;
// original line specifies a callback function
// m_psh.dwFlags |= (PSH_MODELESS|PSH_USECALLBACK);
// new line does not specify a callback function
m_psh.dwFlags |= PSH_MODELESS;
// m_psh.pfnCallback = AfxPropSheetCallback;
m_psh.hwndParent = pParentWnd->GetSafeHwnd();
// hook the window creation process
AfxHookWindowCreate(this);
HWND hWnd = (HWND)PropertySheet(&m_psh);
// clean up on failure, otherwise return TRUE
if (!AfxUnhookWindowCreate())
PostNcDestroy(); // cleanup if Create fails
if (hWnd == NULL || hWnd == (HWND)-1)
return FALSE;
ASSERT(hWnd == m_hWnd);
return TRUE;
}
/* Compile options needed: default
*/
Additional query words:
kbVC400bug 4.00 border CPropertySheet modeless Property Sheet vcfixlist410
Keywords : kbMFC KbUIDesign kbVC
Version : 4.00
Platform : NT WINDOWS
Issue type :