CPropertySheet(属性单) CPropertyPage(属性页)

1.添加对话框资源,为对话框创建类,继承于CPropertyPage

class CFirstPage : public CPropertyPage
// FirstPage.cpp
CFirstPage::CFirstPage()
    : CPropertyPage(CFirstPage::IDD)
{
    m_psp.dwFlags |= PSP_DEFAULT|PSP_HIDEHEADER;
}

class CSecondPage : public CPropertyPage
// SecondPage.cpp
CSecondPage::CSecondPage()
: CPropertyPage(CSecondPage::IDD)
{
    m_psp.dwFlags |= PSP_DEFAULT|PSP_USEHEADERTITLE|PSP_USEHEADERSUBTITLE;
    m_psp.pszHeaderTitle = _T("Title");
    m_psp.pszHeaderSubTitle = _T("And subtitle");
}

建立这类对话框资源模板时有几点要注意

1)保留TitleBar属性。对话框标题就是将来显示在选项卡上的标签

2)对话框风格设为Child,边框设为Thin

3)将对话框设为Disabled

设置对话框属性页属性

在编写时要注意把对话框资源以下属性设置为对应值

属性名       值

System Menu    False
Style         Child
Title Bar        False
Border        None

// 如果想要去掉“应用”和“帮助”按钮

mms.m_psh.dwFlags |= PSH_NOAPPLYNOW;

mms.m_psh.dwFlags &= ~PSH_HASHELP;

mp1.m_psp.dwFlags &= ~PSP_HASHELP; // mp1 为FirstPage变量

隐藏属性页默认按钮
propsheet.m_psh.dwFlags |= PSH_NOAPPLYNOW;
或隐藏掉Cancel取消按钮:
CWnd *pWnd = GetDlgItem( IDCANCEL );
pWnd->ShowWindow( FALSE );

2.建立新类,继承于CPropertySheet

class CMySheet: public CPropertySheet
{
    DECLARE_DYNAMIC(CMySheet)

public:
    CMySheet(UINT nIDCaption, CWnd* pParentWnd = NULL, UINT iSelectPage = 0);
    CMySheet(LPCTSTR pszCaption, CWnd* pParentWnd = NULL, UINT iSelectPage = 0);
    virtual ~CMySheet();

protected:
    DECLARE_MESSAGE_MAP()
    CFirstPage m_pgFirst;
    CSecondPage m_pgSecond;
    void Init(void);
};

// .cpp
CMySheet::CMySheet(UINT nIDCaption, CWnd* pParentWnd, UINT iSelectPage)
    :CPropertySheet(nIDCaption, pParentWnd, iSelectPage)
{
    Init();
}

CMySheet::CMySheet(LPCTSTR pszCaption, CWnd* pParentWnd, UINT iSelectPage)
    :CPropertySheet(pszCaption, pParentWnd, iSelectPage)
{
    Init();
}

// init函数
void CMySheet::Init(void)
{
    //SetWizardMode();

    AddPage(&m_pgFirst);
    AddPage(&m_pgSecond);

    m_psh.dwFlags |= PSH_WIZARD97|PSH_WATERMARK|PSH_HEADER;
    m_psh.pszbmWatermark = MAKEINTRESOURCE(IDB_WATERMARK);
    m_psh.pszbmHeader = MAKEINTRESOURCE(IDB_BANNER_ICON);
    // Step 2: Fix the problem to show images
    m_psh.hInstance = AfxGetInstanceHandle();
}

3.主对话框添加按钮,点击弹出属性页

//创建表单类对象
CMySheet pro(L"表单样例");
pro.SetWizardMode(); // 设置向导对话框模式
//显示模态表单对话框
if(ID_WIZFINISH == pro.DoModal())
{

}// do something

4.向导模式相关函数

相关函数:

  • CPropertySheet::SetWizardButtons
  • CPropertySheet::SetWizardMode
  • CPropertyPage::OnWizardBack
  • CPropertyPage::OnWizardFinish
  • CPropertyPage::OnWizardNext

激活时OnSetActive

按钮ID:

ID_APPLY_NOW
ID_WIZBACK
ID_WIZNEXT
ID_WIZFINISH

5.其他扩充(转载)

一、为了最大限度的发挥属性页的效用,首先让我们先从 CPropertySheet 继承一个新类,取名为 CMyPropSheet.
接着便可以进行下面的各种操作:
一、隐藏属性页默认按钮
隐藏掉Apply应用按钮:
propsheet.m_psh.dwFlags |= PSH_NOAPPLYNOW;
或隐藏掉Cancel取消按钮:
CWnd *pWnd = GetDlgItem( IDCANCEL );
pWnd->ShowWindow( FALSE );

二、移动属性页按钮
首先,要获取按钮的句柄,然后就可以象对待窗体一样处理它们了. 下面代码先隐藏掉Apply和Help铵钮,再把OK和Cancel按移动到右侧。

    BOOL CMyPropSheet::OnInitDialog ()
    {
        BOOL bResult = CPropertySheet::OnInitDialog();
        int ids [] = {IDOK, IDCANCEL};//, ID_APPLY_NOW, IDHELP };

        // Hide Apply and Help buttons
        CWnd *pWnd = GetDlgItem (ID_APPLY_NOW);
        pWnd->ShowWindow (FALSE);
        pWnd = GetDlgItem (IDHELP);
        pWnd->ShowWindow (FALSE);

        CRect rectBtn;
        int nSpacing = 6;        // space between two buttons...
        for( int i =0; i < sizeof(ids)/sizeof(int); i++)
        {
            GetDlgItem (ids [i])->GetWindowRect (rectBtn);
            ScreenToClient (&rectBtn);

            int btnWidth = rectBtn.Width();
            rectBtn.left = rectBtn.left + (btnWidth + nSpacing)* 2;
            rectBtn.right = rectBtn.right + (btnWidth + nSpacing)* 2;

            GetDlgItem (ids [i])->MoveWindow(rectBtn);
        }
        return bResult;
    }

下面代码移动所有按钮到右侧,并且重新置属性页为合适的大小.

BOOL CMyPropSheet::OnInitDialog ()
{
BOOL bResult = CPropertySheet::OnInitDialog();
int ids[] = { IDOK, IDCANCEL, ID_APPLY_NOW };

CRect rectWnd;
CRect rectBtn;
GetWindowRect (rectWnd);

GetDlgItem (IDOK)->GetWindowRect (rectBtn);

int btnWidth = rectBtn.Width();
int btnHeight = rectBtn.Height();
int btnOffset = rectWnd.bottom - rectBtn.bottom;
int btnLeft = rectWnd.right - rectWnd.left;

rectWnd.bottom = rectBtn.top;
rectWnd.right = rectWnd.right + btnWidth + btnOffset;
MoveWindow(rectWnd);

rectBtn.left = btnLeft;
rectBtn.right = btnLeft + btnWidth;

for (int i = 0; i < sizeof (ids) / sizeof (int); i++)
{
rectBtn.top = (i + 1) * btnOffset + btnHeight * i;
rectBtn.bottom = rectBtn.top + btnHeight;
GetDlgItem (ids [i])->MoveWindow (rectBtn);
}

return bResult;
}

三、改变属性页上的标签文字
首先修改TC_ITEM结构,然后用 SetItem 来修改标签文字,如下代码:

TC_ITEM item;
item.mask = TCIF_TEXT;
item.pszText = "New Label";

//Change the label of the first tab (0 is the index of the first tab)...
GetTabControl ()->SetItem (0, &item);

四、改变属性页标签文字的字体属性
代码如下

m_NewFont.CreateFont (14, 0, 0, 0, 800, TRUE, 0, 0, 1, 0, 0, 0, 0, _T("Arial") );
GetTabControl()->SetFont (&m_NewFont);

五、在属性页标签上显示位图
可以用 CImageList 建立图像. 用 SetItem 来设置,如下代码所示:

BOOL CMyPropSheet::OnInitDialog ()
{
  BOOL bResult = CPropertySheet::OnInitDialog();
  m_imageList.Create (IDB_MYIMAGES, 13, 1, RGB(255,255,255));

  CTabCtrl *pTabCtrl = GetTabControl ();
  pTabCtrl->SetImageList (&m_imageList);

  TC_ITEM item;
  item.mask = TCIF_IMAGE;
  for (int i = 0; i < NUMBER_OF_TABS; i++)
  {
    item.iImage = i;
    pTabCtrl->SetItem (i, &item );
  }

  return bResult;
}

六、在属性页左下角显示位图
如下代码所示:

    void CMyPropSheet::OnPaint ()
    {
        CPaintDC dc(this); // device context for painting

        int nOffset = 6;
        // load IDB_BITMAP1 from our resources
        CBitmap bmp;
        if (bmp.LoadBitmap (IDB_BITMAP1))
        {
            // Get the size of the bitmap

            BITMAP bmpInfo;
            bmp.GetBitmap (&bmpInfo);
            // Create an in-memory DC compatible with the
            // display DC we‘‘re using to paint
            CDC dcMemory;
            dcMemory.CreateCompatibleDC (&dc);
            // Select the bitmap into the in-memory DC
            CBitmap* pOldBitmap = dcMemory.SelectObject (&bmp);
            // Find a bottom-left point for the bitmap in the client area
            CRect rect;
            GetClientRect (&rect);

            int nX = rect.left + nOffset;
            int nY = rect.top + (rect.Height () - bmpInfo.bmHeight) - nOffset;

            // Copy the bits from the in-memory DC into the on-
            // screen DC to actually do the painting. Use the centerpoint
            // we computed for the target offset.
            dc.BitBlt (nX, nY, bmpInfo.bmWidth, bmpInfo.bmHeight, &dcMemory,0, 0, SRCCOPY);
            dcMemory.SelectObject (pOldBitmap);
        }
        // Do not call CPropertySheet::OnPaint() for painting messages
    }

七、在属性页右下角显示3D文字Logo
代码如下:

void CMyPropSheet::OnPaint ()
{
        /////////////////////////////////////////////////////////////////
    //在TAB按钮旁边显示3D文字提示,jingzhou xu
    Cstring m_LogoName = “属性页”;
    //  if(m_LogoName == "")
    //      return;

    GetWindowRect(rect);

    ScreenToClient(rect);

    LOGFONT logFont;

    ZeroMemory((void*)&logFont,sizeof(logFont));
    strcpy(logFont.lfFaceName,"宋体");
    logFont.lfHeight = -12;
    logFont.lfWeight = 400;
    logFont.lfCharSet = GB2312_CHARSET;
    logFont.lfOutPrecision = 3;
    logFont.lfClipPrecision = 2;
    logFont.lfQuality = 1;
    logFont.lfPitchAndFamily = 2;
    m_font.CreateFontIndirect(&logFont);
    SetFont(&m_font);

    CFont   *pOldFont = pDC->SelectObject(&m_font);

    rect.left += 6;
    rect.right -= 6;
    rect.bottom -= 1;
    rect.top = rect.bottom - ITEMBUTTON_HEIGHT + 1;

    CFont m_LogoFont;
    CString sLogoString;
    m_LogoFont.CreateFont(rect.Height()*4/5, 0, 0, 0, FW_BOLD, 1, FALSE, FALSE,
        DEFAULT_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY,
        FIXED_PITCH | FF_ROMAN, "楷体_GB2312");
    sLogoString = m_LogoName;

    RECT m_rDataBox;
    CopyRect(&m_rDataBox,&rect);

    TEXTMETRIC tm;
    pDC->GetTextMetrics(&tm);
    CFont* oldFont = pDC->SelectObject(&m_LogoFont);
    CSize sz = pDC->GetTextExtent(sLogoString, sLogoString.GetLength());
    //用GetTextExtent来计算字体logo大小,依靠于设备环境,使用logo位于右下角
    m_rDataBox.left = m_rDataBox.right  - sz.cx - tm.tmAveCharWidth/2;
    m_rDataBox.top  = m_rDataBox.bottom - sz.cy - tm.tmHeight/5;
    pDC->SetBkMode(TRANSPARENT);
    //用3D字体显示,先黑后白,最后再用默认色
    COLORREF oldColor = pDC->SetTextColor(GetSysColor(COLOR_3DDKSHADOW));
    pDC->DrawText(sLogoString, sLogoString.GetLength(), &m_rDataBox, DT_VCENTER | DT_SINGLELINE | DT_CENTER);
    m_rDataBox.left -= tm.tmAveCharWidth;
    pDC->SetTextColor(GetSysColor(COLOR_3DHILIGHT));
    pDC->DrawText(sLogoString, sLogoString.GetLength(), &m_rDataBox, DT_VCENTER | DT_SINGLELINE | DT_CENTER);
    m_rDataBox.left += 3*tm.tmAveCharWidth/5;
    pDC->SetTextColor(RGB(0,0,255));
    pDC->DrawText(sLogoString, sLogoString.GetLength(), &m_rDataBox, DT_VCENTER | DT_SINGLELINE | DT_CENTER);

    //释放资源

    pDC->SelectObject(oldFont);
    pDC->SetTextColor(oldColor);
    m_LogoFont.DeleteObject();
    //////////////////////////////////////////////////////////////////
}        

八、在属性页中动态加入其它控件
下面演示如何在左下角加入一Edit控件:
MyPropSheet.h中:

public:
CEdit m_edit;
MyPropSheet.cpp中:
BOOL CMyPropSheet::OnInitDialog ()
{
  BOOL bResult = CPropertySheet::OnInitDialog ();
  CRect rect;
  int nHeight = 24;
  int nWidth = 120;
  int nOffset = 6;
  GetClientRect (&rect);
  // Find a bottom-left point for the edit control in the client area
  int nX = rect.left + nOffset;
  int nY = rect.top + (rect.Height() - nHeight) - nOffset;

  // finally create the edit control
  m_Edit.CreateEx (WS_EX_CLIENTEDGE, _T("EDIT"), NULL,
  WS_CHILD | WS_VISIBLE | WS_TABSTOP | WS_BORDER,
  nX, nY, nWidth, nHeight, m_hWnd, 0, 0 );

  return bResult;
}   

相关链接:

http://blog.csdn.net/rackyye/article/details/2147172

// 深入浅出CPropertySheet

http://www.vckbase.com/index.php/wv/237.html

时间: 2024-10-10 08:37:41

CPropertySheet(属性单) CPropertyPage(属性页)的相关文章

[ jquery 表单UI选择器和表单元素属性选择器 ] 表单UI选择器和表单元素属性选择器

表单UI选择器和表单元素属性选择器: 实例: <!DOCTYPE html> <html lang='zh-cn'> <head> <title>Insert you title</title> <meta http-equiv='description' content='this is my page'> <meta http-equiv='keywords' content='keyword1,keyword2,keywo

表单的属性和方法, 获取表单和表单的元素, 验证表单

表单的属性和方法 一. 表单字段的属性(id/name/value/form),这里用value属性来举例 上面的form属性代表获取表单字段的父级表单对象 1. 属性的获取         console.log(document.myform.username.value); 2. 属性的设置            document.myform.username.value="123"; 3. 获取表单字段的父级表单对象 console.log(document.myform.u

HTML5表单新增属性

------------------siwuxie095 HTML5 表单新增元素与属性 1.form 属性 在 HTML4 中,表单内的从属元素必须书写在表单内部,而 在 HTML5 中,可以把它们书写在页面上任何地方,然后为 该元素指定一个 form 属性,属性值为该表单的 id,这样就 可以声明该元素从属于指定表单了 2.formaction 属性 在 HTML4 中,一个表单内的所有元素只能通过表单的 action 属性被统一提交到另一个页面,而在 HTML5 中,可以为所有 的提交按钮

HTML5之表单新增属性

之前已经接触过的新增属性:autocomplete.autofocus.list.multiple.placeholder.required.min.max.step from属性:将表单外的内容与表单进行关联 <form action="" id="kc" > <input type="text" name="dd" form="kc" >(不再form表单中) novalidat

Html5之高级-2 HTML5表单属性(属性介绍、属性详解)

一.属性介绍 属性介绍 - 有一些输入类型要求使用特定的属性才能显示效果,如前面提到过min,max,step. 其他输入类型需要使用一些属性来改进其他性能,或者决定验证过程的重要性.HTML5 标准中再原来的基础上增加了一些新的属性. - Placeholder 属性 - Nultiple 属性 - Autofocus 属性 - Form 属性 二.属性详解 Placeholder 属性 - Placeholder 属性通常用于search输入类型,也可以用在文本域.它表示一个简单提示.单词或

HTML5(二)——特殊符号、新增属性、表单重写属性、

一.HTML5 特殊符号 &nbsp :空格 &gt:大于号 > &It :小于号 < &quot :引号" &copy:"版权符号 @ 二.HTML 新增属性 1.contentEditable 规定是否可编辑元素的内容属性值:true -----可以编辑元素的内容false -----无法编辑元素的内容inherit -----继承父元素的contenteditable属性当为空字符串时,效果和true一致.当一个元素的conte

input表单type属性值

<!-- input表单type属性值: --> <!-- type = "email" 限制用户输入必须为Email类型 --> email<input type="email"> <!-- type="url" 限制用户输入必须为URL类型 --> url<input type="url"> <!-- type="date" 限制用户输入

Activity之taskAffinity属性、allowTaskReparenting属性和Android退出整个应用解决方案

allowTaskReparenting属性 属性: android:allowTaskReparenting 除了launchMode可以用来调配Task,的另一属性taskAffinity,也是常常被使用.taskAffinity,是一种物以类聚的思想,它倾向于将taskAffinity属性相同的Activity,扔进同一个Task中.不过,它的约束力,较之launchMode而言,弱了许多.只有当中的allowTaskReparen ting设置为true,抑或是调用方将Intent的fl

实例成员:计算属性 监听属性以及vue的项目开发

6)字符串补充 双引号: "前缀" + 变量 + "后缀" 单引号: '前缀' + 变量 + '后缀' 反引号: `前缀$(变量)后缀` ps:在反引号中可以用$()来包裹变量,实现字符串的拼接 7)实例成员:计算属性 监听属性 计算属性: ? 1)计算属性其实就是vue中的方法属性,方法名可以作为属性来使用,属性值就是方法的返回值 ? 2) 在computed中声明的方法属性,不能在data中重复声明,比data中声明的属性要多出些逻辑的的地方 ? 3)方法属性自