欢迎光临
我们一直在努力

创建有个性的对话框之MFC篇(二)-ASP教程,系统相关

建站超值云服务器,限时71元/月

下面我们针对每个控件设置特殊的颜色,区分控件可以通过控件的id,修改控件背景也很简单,直接返回相应的画刷就可以了,下面就是颜色设置的完整代码:

 

hbrush ccustdlgdlg::onctlcolor(cdc* pdc, cwnd* pwnd, uint nctlcolor)
{
 hbrush hbr = cdialog::onctlcolor(pdc, pwnd, nctlcolor);
 tchar szclassname[64];

 ::getclassname(pwnd->getsafehwnd(),szclassname,64);
 if(lstrcmpi(szclassname,_t(“edit”)) == 0) //是edit 控件
 {
  dword dwstyle = pwnd->getstyle();
  if((dwstyle & es_multiline)  == es_multiline) //多行edit控件
  {
   pdc->settextcolor(m_clrtext);
   return hbr;
  }
  else
  {
   pdc->settextcolor(m_clrtext);
   pdc->setbkmode(transparent);

   return (hbrush)m_brbkgnd;
  }
 }
 else //不是编辑控件
 {
  if(pwnd->getdlgctrlid() == idc_stc_redtext)
  {
   pdc->settextcolor(rgb(255,0,0));
   pdc->setbkmode(transparent);
   return (hbrush)m_brbkgnd;
  }
  else if(pwnd->getdlgctrlid() == idc_stc_bluetext)
  {
   pdc->settextcolor(rgb(0,0,255));
   pdc->setbkmode(transparent);
   return (hbrush)m_brbkgnd;
  }
  else if(pwnd->getdlgctrlid() == idc_stc_bluetextwhiteback)
  {
   pdc->settextcolor(rgb(0,0,255));
   pdc->setbkmode(transparent);
   return (hbrush)m_brcontrolbkgnd1;
  }
  else if(pwnd->getdlgctrlid() == idc_chk_green)
  {
   pdc->settextcolor(rgb(0,255,0));
   pdc->setbkmode(transparent);
   return (hbrush)m_brbkgnd;
  }
  else if(pwnd->getdlgctrlid() == idc_rad_blue)
  {
   pdc->settextcolor(rgb(0,0,255));
   pdc->setbkmode(transparent);
   return (hbrush)m_brbkgnd;
  }
  else if(pwnd->getdlgctrlid() == idc_chk_green2)
  {
   pdc->settextcolor(rgb(0,255,0));
   pdc->setbkmode(transparent);
   return (hbrush)m_brcontrolbkgnd2;
  }
  else if(pwnd->getdlgctrlid() == idc_radio2)
  {
   pdc->settextcolor(rgb(0,0,255));
   pdc->setbkmode(transparent);
   return (hbrush)m_brcontrolbkgnd2;
  }
  else
  {
   pdc->settextcolor(m_clrtext);
   pdc->setbkmode(transparent);
   return (hbrush)m_brbkgnd;
  }
 }
}

 

现在看看效果:

图.5 修改onctlcolor之后的效果

上面的代码是根据控件id来设置颜色,还可以根据控件的类型统一设置某种控件的颜色,这就要用到nctlcolor参数,nctlcolor参数用来指明发送这个通知消息的控件的类型,nctlcolor可以是以下取值:

ctlcolor_btn
ctlcolor_dlg
ctlcolor_edit
ctlcolor_listbox
ctlcolor_msgbox
ctlcolor_scrollbar
ctlcolor_static

第三步:使用位图作对话框的背景

    使用位图作为对话框的背景也很简单,就是在onerasebkgnd中用位图填充客户区,只是在onctlcolor中需要注意返回空画刷代替原来的画刷,返回空画刷是为了阻止控件绘制自己的背景,从而破坏位图背景的完整性,但是有时候返回空画刷会对其他控件产生不良影响,所以我们只处理了ctlcolor_btn和ctlcolor_static两种类型的消息:

 

hbrush cbmpbkgnddlg::onctlcolor(cdc* pdc, cwnd* pwnd, uint nctlcolor)
{
 hbrush hbr = cdialog::onctlcolor(pdc, pwnd, nctlcolor);
 
 if(nctlcolor == ctlcolor_btn || nctlcolor == ctlcolor_static)
 {
  pdc->settextcolor(rgb(0,0,255));
  pdc->setbkmode(transparent);
  return (hbrush)m_hollowbrush;
 }
 
 pdc->settextcolor(rgb(0,0,255));
 pdc->setbkmode(transparent);
 return hbr;
}

 

下面是使用位图背景和空画刷的效果:

图.6 使用位图背景的效果

第四步:单独处理按钮控件

    现在看来按钮控件还是影响整体效果,wm_ctlcolorbtn好像对于push button类型的按钮控件没有效果,不过push button也是支持自画的,在使用自画按钮之前,我们先来看看控件自画的原理。windows的控件都有默认的外观,但是许多控件有支持“自画”,也就是让用户定制控件的外观,当给一个控件指定自画的样式之后,控件在重画自己的时候向父窗口发送wm_measureitem和wm_drawitem消息,父窗口响应这两个消息,定位控件的大小并绘制控件,从而使控件有定制的外观。但是每个控件的自画都由父窗口完成加重了父窗口的负担,也不利于代码重用,所以,mfc对这些消息进行了反射处理,就是将消息发还位控件,由控件响应消息,自己绘制,这样将自画代码封装在控件类中,提高了代码的重用性。很多mfc的控件类都自己处理这两个消息,派生类可以重载measureitem和drawitem自己画控件的外观,cbutton就是这样的控件类。
    现在就来做一个自画的按钮类,首先从cbutton派生一个类,我们命名为csmbutton,然后重载drawitem和presubclasswindow,重载presubclasswindow的原因是在csmbutton子类化按钮控件之前先给按钮添加bs_ownerdraw样式,否则按钮就不会向父窗口发送wm_drawitem消息,mfc的消息反射就不会发生,我们的drawitem就不会被调用,嗯,后果很严重。当然也可以让csmbutton的使用者自己给按钮添加bs_ownerdraw样式,但是会让人觉得没水平,嗯,后果也很严重。接下来添加对wm_capturechanged、wm_mousemove、wm_setcursor和wm_killfocus四个消息的响应函数,对这四个消息的响应是为了给按钮增加更多的功能,比如使按钮看起来象工具栏的按钮,改变鼠标的形状等等。
    关于csmbutton类的使用就像cbutton一样,为按钮添加变量就行了,演示代码中包含了这个类的源代码以及用法,这里不在赘述。csmbutton类的功能很简单,但是完成了一个自画按钮的框架,大家可以修改代码实现自己的风格,网上也有很多这样的类,功能更强大,比如stbutton等。现在看看csmbutton的效果:

图.7 使用自画按钮后的效果

第五步:使用picture box控件

    想要在对话框上显示位图,可以使用很复杂的控件或cximage之类的库,也可以很简单地使用picture box。picture box默认的样式使frame,需要手工改成bitmap,如下图所示:

图.8 使用位图

vc6的集成环境不支持24位位图的浏览和编辑,但是并不影响使用,本例使用的位图都是24位的,为的是省去调色板的处理,本人比较懒。使用如下代码就可以更改picture box中的位图:

 

m_hcat1 = (hbitmap)::loadimage(afxgetresourcehandle(),makeintresource(idb_bitmap1),image_bitmap,0,0,lr_createdibsection);
getdlgitem(idc_stc_picture)->sendmessage(stm_setimage,image_bitmap, (lparam)m_hcat1);

 

装载位图还可以这样:

 

 

m_hcat1 = (hbitmap)::loadimage(afxgetresourcehandle(),(lpctstr)idb_bitmap1,image_bitmap,0,0,lr_createdibsection);

 

这是最终的效果:

图.9 对话框的最终效果

下载演示代码(http://blog.csdn.net/images/blog_csdn_net/orbit/custdlg2.zip)
(如果代码不能下载,请在此留言或给我发邮件所要代码)

赞(0)
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com 特别注意:本站所有转载文章言论不代表本站观点! 本站所提供的图片等素材,版权归原作者所有,如需使用,请与原作者联系。未经允许不得转载:IDC资讯中心 » 创建有个性的对话框之MFC篇(二)-ASP教程,系统相关
分享到: 更多 (0)