模态属性单实例程序主要讲解CPropertySheet和CPropertyPage类的不同方面。
1)创建程序的框架
常见一个名为ModalDemo的MFC SDI项目。再添加两个对话框,ID分别为IDD_FINDBOOK_FIND和IDD_FINDBOOK_RESULTS,并为两个对话框创建派生于CPropertyPage类的CFindPage类和CResultsPage类。添加一个ID为IDS_FIND,Caption为FindBook的字符串资源,它将被传递给CPropertySheet构造函数。最后,在IDR_MAINFRAME菜单栏添加Books,并添加子项FindBook,ID为ID_FINDBOOK。
2)单击ID_FINDBOOK事件
创建有IDD_FINDBOOK_FIND和IDD_FINDBOOK_RESULTS为属性页和IDS_FIND为属性单的组合。在这里创建CPropertySheet的派生类CFindSheet。
{
CFindSheet sheet(IDS_FIND);
CFindPage pageFind;
CResultsPage pageResults;
sheet.AddPage( & pageFind);///////////////////////////////////////////////////////////////////////////////////
sheet.AddPage( & pageResults);
sheet.DoModal();
}
标准按钮有OK(IDOK)、Cancel(IDCANCEL)、Help(IDHELP)和Apply(ID_APPLY_NOW)。这里去除掉Cancel和Apply两个按钮。
{
BOOL bResult = CPropertySheet::OnInitDialog();
int ids[] = {ID_APPLY_NOW,IDCANCEL};
for ( int i = 0 ;i < sizeof (ids) / sizeof (ids[ 0 ]);i ++ )
{
CWnd * pWnd = GetDlgItem(ids[i]);
if (pWnd)
pWnd -> ShowWindow(FALSE);
}
return bResult;
}
默认情况下,标准按钮是在属性页的下方,这里把OK按钮移到沿属性页右对齐。
{
BOOL bResult = CPropertySheet::OnInitDialog();
CWnd * pbtnOk = GetDlgItem(IDOK);
CRect rectSheet;
GetWindowRect(rectSheet);
CRect rectOkBtn;
pbtnOk -> GetWindowRect(rectOkBtn);
int iBorder = rectSheet.bottom - rectOkBtn.bottom;
rectSheet.right += rectOkBtn.Width() + iBorder;
rectSheet.bottom = rectOkBtn.top;
MoveWindow(rectSheet);/////////////////////////////////////////////////////////////////////////////
CPropertyPage * page = GetPage( 0 );//////////////////////////////////////////////////////////////////
CRect rectPage;
page -> GetWindowRect(rectPage);
int cxOk = rectOkBtn.Width();
int cyOk = rectOkBtn.Height();
rectOkBtn.top = rectPage.top;
rectOkBtn.bottom = rectOkBtn.top + cyOk;
rectOkBtn.left = rectSheet.right - (cxOk + iBorder);
rectOkBtn.right = rectOkBtn.left + cxOk;
ScreenToClient(rectOkBtn);
pbtnOk -> MoveWindow(rectOkBtn);
return bResult;
}
{
BOOL bResult = CPropertySheet::OnInitDialog();
pbtnOk -> SetWindowText(_T( " Cl&ose " ));////////////////////////////////////////////////////////////////
return bResult;
}
创建一个存储当前选项卡索引的成员变量和一个容纳当前被禁用的选项卡的成员变量
CUIntArray m_arrDisabledPages; // 被禁用的选项卡索引
void CFindSheet::DisablePage( int iFirstPage,)
{
int iPage = iFirstPage;
va_list marker;)////////////////////////////////////////////////
va_start (marker,iFirstPage);)////////////////////////////////////////////////
int nArgs = 0 ;
while (iPage !=- 1 )
{
m_arrDisabledPages.Add(iPage);
SetDisabledText(iPage);
iPage = va_arg(marker,UINT);)////////////////////////////////////////////////
}
}
void CFindSheet::SetDisabledText( int iPage)///////////////////////////////////////////////
{
CTabCtrl * pTab = GetTabControl();
TC_ITEM ti;
char szText[ 100 ];
ti.mask = TCIF_TEXT;
ti.pszText = szText;
ti.cchTextMax = 100 ;
pTab -> GetItem(iPage, & ti);
strcat(szText,DISABLED_TEXT);
pTab -> SetItem(iPage, & ti);
}
添加一个函数来验证是否禁用了选项卡
{
NMHDR * pnmh = (NMHDR * )lParam;
if (TCN_SELCHANGING == pnmh -> code)
m_iLastActivePage = GetActiveIndex();
else if (TCN_SELCHANGE == pnmh -> code)
{
int iCurrPage = GetActiveIndex();
if (IsPageDisabled(iCurrPage))
PostMessage(PSM_SETCURSEL,m_iLastActivePage);
}
return CPropertySheet::OnNotify(wParam, lParam, pResult);
}
BOOL CFindSheet::IsPageDisabled( int iPage)
{
BOOL bFoundEntry = FALSE;
int iSize = m_arrDisabledPages.GetSize();
int i = 0 ;
while (i < iSize &&! bFoundEntry)
{
if (m_arrDisabledPages.GetAt(i) == (UINT)iPage)
bFoundEntry = TRUE;
else
i ++ ;
}
return bFoundEntry;
}
{
BOOL bResult = CPropertySheet::OnInitDialog();
DisablePage( 1 , - 1 );
return bResult;
}
EnablePage函数只需要在m_arrDisabledPages数组搜索指定的属性页即可,如果找到了这个属性页,则从数组中去掉该页,并调用SetEnabledText函数去掉“-Disabled”
{
CTabCtrl * pTab = GetTabControl();
TC_ITEM ti;
char szText[ 100 ];
ti.mask = TCIF_TEXT;
ti.pszText = szText;
ti.cchTextMax = 100 ;
pTab -> GetItem(iPage, & ti);
char * pFound = strstr(szText,DISABLED_TEXT);
if (pFound)
{
* pFound = ' \0 ' ;
pTab -> SetItem(iPage, & ti);
}
}
void CFindSheet::EnablePage( int iPage)
{
BOOL bFoundEntry = FALSE;
int iSize = m_arrDisabledPages.GetSize();
int i = 0 ;
while (i < iSize &&! bFoundEntry)
{
if (m_arrDisabledPages.GetAt(i) == (UINT)iPage)
bFoundEntry = TRUE;
else
i ++ ;
}
if (bFoundEntry)
{
m_arrDisabledPages.RemoveAt(i);
SetEnabledText(iPage);
}
}
{
CFindSheet * pParentSheet = (CFindSheet * )GetParent();
ASSERT(pParentSheet -> IsKindOf(RUNTIME_CLASS(CFindSheet)));
pParentSheet -> EnablePage( 1 );
}
BOOL CFindSheet::OnInitDialog()
{
BOOL bResult = CPropertySheet::OnInitDialog();
m_fontTab.CreateFont( - 8 , 0 , 0 , 0 ,FW_BOLD, 0 , 0 , 0 , 1 , 0 , 0 , 0 , 0 ,_T( " MS Sans Serif " ));
CTabCtrl * pTab = GetTabControl();
if (pTab)
pTab -> SetFont( & m_fontTab);/////////////////////////////////////////////////////////////////////
return bResult;
}
相关函数:
sheet.AddPage(&pageFind);
MoveWindow(rectSheet);
CPropertyPage* page=GetPage(0);
pbtnOk->SetWindowText(_T("Cl&ose"));
va_list marker;
va_start (marker,iFirstPage);
iPage=va_arg(marker,UINT);
void CFindSheet::SetDisabledText(int iPage)
{
CTabCtrl* pTab=GetTabControl();
TC_ITEM ti;
char szText[100];
ti.mask=TCIF_TEXT;
ti.pszText=szText;
ti.cchTextMax=100;
pTab->GetItem(iPage,&ti);
strcat(szText,DISABLED_TEXT);
pTab->SetItem(iPage,&ti);
}
BOOL CFindSheet::OnNotify(WPARAM wParam, LPARAM lParam, LRESULT* pResult)
{
NMHDR* pnmh=(NMHDR*)lParam;
if(TCN_SELCHANGING==pnmh->code)
m_iLastActivePage=GetActiveIndex();
else if(TCN_SELCHANGE==pnmh->code)
{
int iCurrPage=GetActiveIndex();
if(IsPageDisabled(iCurrPage))
PostMessage(PSM_SETCURSEL,m_iLastActivePage);
}
return CPropertySheet::OnNotify(wParam, lParam, pResult);
}
char * pFound=strstr(szText,DISABLED_TEXT);
CFindSheet * pParentSheet=(CFindSheet*)GetParent();
ASSERT(pParentSheet->IsKindOf(RUNTIME_CLASS(CFindSheet)));
pTab->SetFont(&m_fontTab);