VC++对话框的任意扩展

  我们在信息输入的时候,可能有很大的信息量,而这些信息又不是必须的,这时我们就需要给信息输入人员一个选择的接口。例如一个人事部门的职工信息录入系统就有这样的问题
,其中的姓名、性别、年龄、政治面目、职务、学历、部门和联系电话是必须输入的信息,而婚姻状况、毕业学校、籍贯和健康状况是可输可不输的信息且大多数情况下不需要录入,如何为信息录入人员提供一个方便的输入接口,下面我们就针对这个问题提供一个我认为比较好的方法。

  第一步:在VC编程环境下建立一个基于对话框的工程,工程名为ExpandDlg,所有的选项都取默认值。

  第二步:建立我们都对话框,其中必须要有这样两个控件,一个是PICTURE控件,一个为按钮,其ID值分别为IDC_DIVIDER和IDC_MORE。其它的控件可以任意布局,最终结果就是对话框被IDC_DIVIDER控件分成了两部分,其中下半部分可以根据你的爱好动态显示或不显示,对话框如下图:

  第三步:生成按钮IDC_MORE的消息映射函数OnMore,在ExpandDlgDlg.h中定义两个函数如下:

public:
void EnableVisibleChildren();
void ExpandDialog (int nResourceID, BOOL bExpand);

  第四步:在ExpandDlgDlg.cpp中定义函数的实现代码如下:

void CExpandDlgDlg::ExpandDialog (int nResourceID, BOOL bExpand)
{
      // 对话框被nResourceID分成上下两部分,如果bExpand的值为TRUE
      // 对话框被完整显示,否则对话框显示上半部分。
      static CRect rcLarge;
      static CRect rcSmall;
      CString sExpand;
      //开始时,对话框只显示上半部分
      if (rcLarge.IsRectNull())
      {
            CRect rcLandmark;
            CWnd* pWndLandmark = GetDlgItem (nResourceID);
            ASSERT(pWndLandmark);
            GetWindowRect (rcLarge);
            pWndLandmark->GetWindowRect (rcLandmark);
            rcSmall = rcLarge;
            rcSmall.bottom = rcLandmark.top;
      }
      if (bExpand)
      {
            //扩展对话框到最大尺寸
            SetWindowPos(NULL, 0, 0, rcLarge.Width(), rcLarge.Height(),
                  SWP_NOMOVE | SWP_NOZORDER);
            sExpand = "<< &Less";
            EnableVisibleChildren();
      }
      else
      {
            //只显示对话框的上半部分
            SetWindowPos(NULL, 0, 0, rcSmall.Width(), rcSmall.Height(),
                  SWP_NOMOVE | SWP_NOZORDER);
            sExpand = " &More >>";
            EnableVisibleChildren();
      }
      SetDlgItemText (IDC_MORE, sExpand);
}

void CExpandDlgDlg::EnableVisibleChildren()
{
      //去掉没有显示的对话框的控件的功能和快捷键。
      //得到第一个窗口
      CWnd *pWndCtl = GetWindow (GW_CHILD);
      CRect rcTest;
      CRect rcControl;
      CRect rcShow;
      //得到对话框的完整矩形框
      GetWindowRect(rcShow);
      while (pWndCtl != NULL)
      { //得到当前显示的对话框的矩形尺寸
            pWndCtl->GetWindowRect (rcControl);

            if (rcTest.IntersectRect (rcShow, rcControl))
                  pWndCtl->EnableWindow(TRUE);
            else
                  pWndCtl->EnableWindow(FALSE);
            //得到第二个矩形框
            pWndCtl = pWndCtl->GetWindow (GW_HWNDNEXT);
      }
}

void CExpandDlgDlg::OnMore()
{
      static BOOL bExpand = TRUE;
      ExpandDialog (IDC_DIVIDER, bExpand);
      bExpand = !bExpand;
}

  按照上面的步骤生成我们的可执行文件后运行,点击对话框上的〔More〕我们可以发现对话框扩展,点击〔Less〕后,我们发现对话框收缩,希望可以给你带来方便。
 

使用VC创建不规则形状窗口
 
逸仙时空

仔细查看了一下WIN32的API,发现其实创建任意形状的窗口其实也是很简单的,在VC中简单步骤如下:

  当我们注册并创建了一个窗口类以后,我们在WM_CREATE消息中做如下处理:

  (1)创建一个区域,使用CreatePolyonRgn,该函数创建一个多边形区域,(也可以使用其他方法如CreateRectRgn创建矩形区域),该函数返回一个HRGN的句柄;

  (2)调用函数SetWindowRgn,即可设置窗口的形状。

  补充说明的是,我们可以制作多个区域,然后用CombineRgn方法将多个区域合并为一个区域。这样我们就可以制作出更为丰富多采的窗口了。
 


VC编程实现IE风格的界面
 
刘 涛··yesky

  使用过IE浏览器的朋友都知道IE界面上的扁平工具条、地址栏,扁平工具栏上的按钮正常状态下为扁平态,按钮上的图像为灰色,当鼠标放在按钮上时,按钮突起(这种状态称为手柄),并且其上的图像变得鲜艳醒目,一些按钮上还有汉字说明或标有小黑三角的下拉按钮,单击时显示下拉菜单,这些技术是怎么实现的呢,本文针对这些问题介绍了如何利用VC编程来实现它们。

  IE风格的实现主要在主框架类的CMainFrame::OnCreate()实现,它的主要思想如下:首先定义一个CReBar对象,用以作工具条、地址栏的容器,然后分别声明图像列表对象img用于存储工具栏上按钮的热点图像和正常状态下显示的图像。为了显示扁平工具栏,需要用CreateEx()函数创建CToolBar对象m_wndToolBar,用ModifyStyle()函数将工具栏的风格设为扁平类型,你不能用CToolBar::Create() 或 CToolBar:: SetBarStyle()设置这种新风格。CToolBar 类不支持TBSTYLE_FLAT。要解决这个问题,必须绕过CToolBar类,使用CWnd::ModifyStyle()。工具栏对象调用SetButtonInfo()设置按钮的风格为TBSTYLE_DROPDOWN,就可以将工具栏按钮设置为附带有下拉按钮。至于按钮带有中文提示,用工具栏的SetButtonText()就可以轻松实现了。下面是实现IE风格界面的部分代码和注释:

int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
 CReBar m_wndReBar;//声明CReBar对象
 CImageList img;//声明图像列表对象
 CString str;
 if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
  return -1;
 if (!m_wndReBar.Create(this))//创建CReBar对象
 {
  TRACE0("Failed to create rebar/n");
  return -1; // fail to create
 }
 if (!m_wndToolBar.CreateEx(this))//创建工具条对象
 {
  TRACE0("Failed to create toolbar/n");
  return -1; // fail to create
 }
 // set up toolbar properties
 m_wndToolBar.GetToolBarCtrl().SetButtonWidth(50, 150);
 file://设置工具条上按钮的最大、最小尺寸
 m_wndToolBar.GetToolBarCtrl().SetExtendedStyle(TBSTYLE_EX_DRAWDDARROWS);
 file://工具条可以带有下拉按钮
 img.Create(IDB_HOTTOOLBAR, 22, 0, RGB(255, 0, 255));
 file://向图像列表装载热点图像资源,IDB_HOTTOOLBAR为热点图像资源ID
 m_wndToolBar.GetToolBarCtrl().SetHotImageList(&img);//工具条装载热点图像
 img.Detach();
 img.Create(IDB_COLDTOOLBAR, 22, 0, RGB(255, 0, 255));
 file://图象列表装载正常状态的图像资源,IDB_COLDTOOLBAR为图像资源ID
 m_wndToolBar.GetToolBarCtrl().SetImageList(&img);//将图像装入工具条
 img.Detach();
 m_wndToolBar.ModifyStyle(0, TBSTYLE_FLAT | TBSTYLE_TRANSPARENT);
 file://工具条为扁平风格
 m_wndToolBar.SetButtons(NULL, 9);//工具条上有9个按钮
 // set up each toolbar button
 file://以下分别对九个按钮分别设置风格和按钮汉语提示
 m_wndToolBar.SetButtonInfo(0, ID_BUTTON0, TBSTYLE_BUTTON, 0);
 str.LoadString(IDS_ BUTTON0);
 m_wndToolBar.SetButtonText(0, str);
 m_wndToolBar.SetButtonInfo(1, ID_BUTTON1, TBSTYLE_BUTTON, 1);
 str.LoadString(IDS_ BUTTON1);
 m_wndToolBar.SetButtonText(1, str);
 m_wndToolBar.SetButtonInfo(2, ID_BUTTON2, TBSTYLE_BUTTON, 2);
 str.LoadString(IDS_ BUTTON2);
 m_wndToolBar.SetButtonText(2, str);
 m_wndToolBar.SetButtonInfo(3, ID_BUTTON3, TBSTYLE_BUTTON, 3);
 str.LoadString(IDS_ BUTTON3);
 m_wndToolBar.SetButtonText(3, str);
 m_wndToolBar.SetButtonInfo(4, ID_BUTTON4, TBSTYLE_BUTTON, 4);
 str.LoadString(IDS_ BUTTON4);
 m_wndToolBar.SetButtonText(4, str);
 m_wndToolBar.SetButtonInfo(5, ID_BUTTON5, TBSTYLE_BUTTON, 5);
 str.LoadString(IDS_ BUTTON5);
 m_wndToolBar.SetButtonText(5, str);
 m_wndToolBar.SetButtonInfo(6, ID_BUTTON6, TBSTYLE_BUTTON | TBSTYLE_DROPDOWN, 6);
 str.LoadString(IDS_ BUTTON6);
 m_wndToolBar.SetButtonText(6, str);
 m_wndToolBar.SetButtonInfo(7, ID_BUTTON7, TBSTYLE_BUTTON, 7);
 str.LoadString(IDS_ BUTTON7);
 m_wndToolBar.SetButtonText(7, str);
 m_wndToolBar.SetButtonInfo(8,ID_BUTTON8, TBSTYLE_BUTTON | TBSTYLE_DROPDOWN, 8);
 str.LoadString(IDS_ BUTTON8);
 m_wndToolBar.SetButtonText(8, str);
 file://重新调整按钮的尺寸
 CRect rectToolBar;
 m_wndToolBar.GetItemRect(0, &rectToolBar);//得到工具条第一个按钮的尺寸
 m_wndToolBar.SetSizes(rectToolBar.Size(), CSize(30,20));
 file://第一个参数为按钮尺寸,第二个参数为图像尺寸
 file://创建一个组合框作为地址栏
 if (!m_wndAddress.Create(CBS_DROPDOWN | WS_CHILD, CRect(0, 0, 200, 120), this, AFX_IDW_TOOLBAR + 1))
 {
  TRACE0("Failed to create combobox/n");
  return -1; // fail to create
 }
 file://加入工具栏、地址栏
 m_wndReBar.AddBar(&m_wndToolBar);
 str.LoadString(IDS_ADDRESS);
 m_wndReBar.AddBar(&m_wndAddress, str, NULL, RBBS_FIXEDBMP | RBBS_BREAK);
file://定义REBARBANDINFO对象,对工具条和地址栏设置理想尺寸
 REBARBANDINFO rbbi;
 rbbi.cbSize = sizeof(rbbi);
 rbbi.fMask = RBBIM_CHILDSIZE | RBBIM_IDEALSIZE | RBBIM_SIZE;
 rbbi.cxMinChild = rectToolBar.Width();
 rbbi.cyMinChild = rectToolBar.Height();
 rbbi.cx = rbbi.cxIdeal = rectToolBar.Width() * 9;
 m_wndReBar.GetReBarCtrl().SetBandInfo(0, &rbbi);//设置工具栏尺寸
 rbbi.cxMinChild = 0;
 CRect rectAddress;
 rbbi.fMask = RBBIM_CHILDSIZE | RBBIM_IDEALSIZE;
 m_wndAddress.GetEditCtrl()->GetWindowRect(&rectAddress);
 rbbi.cyMinChild = rectAddress.Height() + 10;
 rbbi.cxIdeal = 200;
 m_wndReBar.GetReBarCtrl().SetBandInfo(2, &rbbi);//设置地址栏尺寸
 m_wndToolBar.SetBarStyle(m_wndToolBar.GetBarStyle() |
 CBRS_TOOLTIPS | CBRS_FLYBY | CBRS_SIZE_FIXED);
 if (!m_wndStatusBar.Create(this) ||
  !m_wndStatusBar.SetIndicators(indicators,
  sizeof(indicators)/sizeof(UINT)))
  {
   TRACE0("Failed to create status bar/n");
   return -1; // fail to create
  }
  return 0;
 }

  以上代码在Windows2000和Visual C++环境下编译通过,程序运行正常,有兴趣的朋友可以动手亲自实验一下。
 


VC限制窗口大小又一法
 
逸仙时空

  一般说见到的方法,,都是截获WM_GETMAXMININFO消息。

  俺有另一经验可实现之。

  由于一般窗口大小的改变,都是用户拖动窗口边框而造成的。所以,我们可以截获主窗口消息WM_NCHITTEST在其响应函数中判断CWnd::OnNcHitTest()的返回值是否为HTRIGHT,HTLEFT,HTTOP,HTBOTTOM四个值之一,如果是,说明用户此时已点击了四个边框之一,此时我们应该返回HTCLIENT.那么,鼠标的形状就不会变成水平或垂直的双向箭头,用户就不可能依靠拖动边框来改变窗口大小了。

  另外,还应补上一个小漏洞,就是还要把系统菜单中的SC_SIZE去掉。
 

主程序之前的版权窗口
CPCW

WINAPI WinMain(HINSTANCE, HINSTANCE, LPSTR, int)
{
      DWORD lTime;
      try
      {
            Application->Initialize();
            AboutBox=new TAboutBox(AboutBox);
            AboutBox->BorderStyle=bsNone;
            AboutBox->OKButton->Visible=false;
            AboutBox->Height=185;
            AboutBox->Show();
            AboutBox->Update();
            lTime=GetTickCount();
            Application->CreateForm(__classid(TMainForm), &MainForm);
            while((GetTickCount()-lTime) / 1000 < 3);
            AboutBox->Hide();
            AboutBox->Free();
            Application->Run();
      }
      catch (Exception &exception)
      {
            Application->ShowException(&exception);
      }
      return 0;
}

VISUAL C++6.0在MDI主框架窗口中添加位图
 
刘 涛  yesky

  笔者在开发项目时想在MDI程序中添加彩色位图以美化界面,也实验了几种方法,但都有一些小问题,经多方查找资料,终于圆满的实现了这种功能,现把我的实现方法介绍给大家。

  首先要清楚对于一个MDI应用程序的主框架窗口来说包含一个特殊的子窗口称为MDICLIENT窗口,应用程序的主框架类中有一个成员变量m_hWndMDIClient 指的就是MDICLIENT窗口。MDICLIENT窗口负责管理主框架窗口的客户区,对MDI客户窗口编程有一定的难度。原因是MDIFrameWnd的客户区完全被MDICLIENT窗口覆盖掉了。这样,MDI主窗口类MDIFrameWnd的背景色和光标都不起作用。同时,微软并不支持将MDICLIENT窗口作为子类,MDICLIENT窗口只能使用标准的背景色和光标。所以,对MDI客户窗口编程不能象对普通窗口那样简单地重载WM_PAINT的消息处理函数。我们可以在主框架窗口截获关于MDICLIENT窗口的重画消息,然后加入自己设计的代码。我用PreTranslateMessage(MSG* pMsg) 截获MDI客户窗口WM_PAINT消息,在这个函数中向主框架窗口发送WM_PAINT消息,在该消息的处理函数中实现彩色位图的显示。我的具体实现如下:1、向程序添加256色彩色位图资源,命名为IDB_BITMAP1;2、用ClassWizard向主框架类添加函数CMainFrame::PreTranslateMessage(MSG* pMsg);3、用ClassWizard向主框架类添加函数CMainFrame::OnPaint();现给出两个函数的实现:

BOOL CMainFrame::PreTranslateMessage(MSG* pMsg)
{
 // TODO: Add your specialized code here and/or call the base class
 if(pMsg->hwnd==m_hWndMDIClient && pMsg->message==WM_PAINT)
  PostMessage(WM_PAINT);
  return CMDIFrameWnd::PreTranslateMessage(pMsg);
 }

 void CMainFrame::OnPaint()
 {
  CDC dc, memdc;
  dc.m_hDC=::GetDC(this->m_hWndMDIClient);
  CRect rect;
  CBitmap bitmap;
  BITMAP szbitmap;
  bitmap.LoadBitmap(IDB_BITMAP1);
  bitmap.GetObject(sizeof(BITMAP),&szbitmap);
  CSize size(szbitmap.bmWidth,szbitmap.bmHeight);
  memdc.CreateCompatibleDC(&dc);
  CBitmap *oldbitmap=memdc.SelectObject(&bitmap);
  GetClientRect(&rect);
  StretchBlt(dc.m_hDC,0,0,rect.Width(),rect.Height(),
       memdc.m_hDC,0,0,size.cx,size.cy,SRCCOPY);
  memdc.SelectObject(oldbitmap);
  memdc.DeleteDC();
  dc.DeleteDC();
  CMDIFrameWnd::OnPaint();
 }

  按上述步骤就可以实现在MDI程序中显示彩色位图了,我举的例子用的是256色位图,你也可以实现真彩色位图的显示,具体方法我就不多说了,有兴趣的朋友可以试一试

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值