模態屬性單實例程序主要講解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);