[转]VC++界面编程总结(三)

15. VC中基于 Windows 的精确定时
  
在工业生产控制系统中,有许多需要定时完成的操作,如定时显示当前时间,定时刷新屏幕上的进度条,上位 机定时向下位机发送命令和传送数据等。特别是在对控制性能要求较高的实时控制系统和数据采集系统中,就更需要精确定时操作。
  众所周知,Windows 是基于消息机制的系统,任何事件的执行都是通过发送和接收消息来完成的。 这样就带来了一些问题,如一旦计算机的CPU被某个进程占用,或系统资源紧张时,发送到消息队列 中的消息就暂时被挂起,得不到实时处理。因此,不能简单地通过Windows消息引发一个对定时要求 严格的事件。另外,由于在Windows中已经封装了计算机底层硬件的访问,所以,要想通过直接利用 访问硬件来完成精确定时,也比较困难。所以在实际应用时,应针对具体定时精度的要求,采取相适 应的定时方法。
  VC中提供了很多关于时间操作的函数,利用它们控制程序能够精确地完成定时和计时操作。本文详细介绍了 VC中基于Windows的精确定时的七种方式.
  方式一:VC中的WM_TIMER消息映射能进行简单的时间控制。首先调用函数SetTimer()设置定时 间隔,如SetTimer(0,200,NULL)即为设置200ms的时间间隔。然后在应用程序中增加定时响应函数 On Timer(),并在该函数中添加响应的处理语句,用来完成到达定时时间的操作。这种定时方法非常 简单,可以实现一定的定时功能,但其定时功能如同Sleep()函数的延时功能一样,精度非常低,最小 计时精度仅为30ms,CPU占用低,且定时器消息在多任务操作系统中的优先级很低,不能得到及时响 应,往往不能满足实时控制环境下的应用。只可以用来实现诸如位图的动态显示等对定时精度要求不高的情况。如示例工程中的Timer1。 
  方式二:VC中使用sleep()函数实现延时,它的单位是ms,如延时2秒,用sleep(2000)。精度非常 低,最小计时精度仅为30ms,用sleep函数的不利处在于延时期间不能处理其他的消息,如果时间太 长,就好象死机一样,CPU占用率非常高,只能用于要求不高的延时程序中。如示例工程中的Timer2。
  方式三:利用COleDateTime类和COleDateTimeSpan类结合WINDOWS的消息处理过程来实现秒级延时。如示例工程中的Timer3和Timer3_1。以下是实现2秒的延时代码: 
   COleDateTime   start_time = COleDateTime::GetCurrentTime();
   COleDateTimeSpan end_time= COleDateTime::GetCurrentTime()-start_time;
   while(end_time.GetTotalSeconds()< 2) //实现延时2秒
   { 
       MSG  msg;
       GetMessage(&msg,NULL,0,0);
       TranslateMessage(&msg); 
       DispatchMessage(&msg);
       
       //以上四行是实现在延时或定时期间能处理其他的消息,
       //虽然这样可以降低CPU的占有率,
       //但降低了延时或定时精度,实际应用中可以去掉。
       end_time = COleDateTime::GetCurrentTime()-start_time;
   }//这样在延时的时候我们也能够处理其他的消息。   
  方式四:在精度要求较高的情况下,VC中可以利用GetTickCount()函数,该函数的返回值是 DWORD型,表示以ms为单位的计算机启动后经历的时间间隔。精度比WM_TIMER消息映射高,在较 短的定时中其计时误差为15ms,在较长的定时中其计时误差较低,如果定时时间太长,就好象死机一样,CPU占用率非常高,只能用于要求不高的延时程序中。如示例工程中的Timer4和Timer4_1。下列代码可以实现50ms的精确定时:
    DWORD dwStart = GetTickCount();
    DWORD dwEnd  = dwStart;
    do
    {
     dwEnd = GetTickCount()-dwStart;
    }while(dwEnd <50);
为使GetTickCount()函数在延时或定时期间能处理其他的消息,可以把代码改为:
    DWORD dwStart = GetTickCount();
    DWORD dwEnd  = dwStart;
    do
    {
       MSG  msg;
       GetMessage(&msg,NULL,0,0);
       TranslateMessage(&msg); 
       DispatchMessage(&msg);
       dwEnd = GetTickCount()-dwStart;
    }while(dwEnd <50);
虽然这样可以降低CPU的占有率,并在延时或定时期间也能处理其他的消息,但降低了延时或定时精度。
  方式五:与GetTickCount()函数类似的多媒体定时器函数DWORD timeGetTime(void),该函数定时精 度为ms级,返回从Windows启动开始经过的毫秒数。微软公司在其多媒体Windows中提供了精确定时器的底 层API持,利用多媒体定时器可以很精确地读出系统的当前时间,并且能在非常精确的时间间隔内完成一 个事件、函数或过程的调用。不同之处在于调用DWORD timeGetTime(void) 函数之前必须将 Winmm.lib 和 Mmsystem.h 添加到工程中,否则在编译时提示DWORD timeGetTime(void)函数未定义。由于使用该 函数是通过查询的方式进行定时控制的,所以,应该建立定时循环来进行定时事件的控制。如示例工程中的Timer5和Timer5_1。
  方式六:使用多媒体定时器timeSetEvent()函数,该函数定时精度为ms级。利用该函数可以实现周期性的函数调用。如示例工程中的Timer6和Timer6_1。函数的原型如下: 
    MMRESULT timeSetEvent( UINT uDelay, 
                UINT uResolution, 
                LPTIMECALLBACK lpTimeProc, 
                WORD dwUser, 
                UINT fuEvent )
  该函数设置一个定时回调事件,此事件可以是一个一次性事件或周期性事件。事件一旦被激活,便调用指定的回调函数, 成功后返回事件的标识符代码,否则返回NULL。函数的参数说明如下:
    uDelay:以毫秒指定事件的周期。
    Uresolution:以毫秒指定延时的精度,数值越小定时器事件分辨率越高。缺省值为1ms。
    LpTimeProc:指向一个回调函数。
    DwUser:存放用户提供的回调数据。
    FuEvent:指定定时器事件类型:
    TIME_ON ESHOT:uDelay毫秒后只产生一次事件
    TIME_PERIODIC :每隔uDelay毫秒周期性地产生事件。   
  具体应用时,可以通过调用timeSetEvent()函数,将需要周期性执行的任务定义在LpTimeProc回调函数 中(如:定时采样、控制等),从而完成所需处理的事件。需要注意的是,任务处理的时间不能大于周期间隔时间。另外,在定时器使用完毕后, 应及时调用timeKillEvent()将之释放。 
  方式七:对于精确度要求更高的定时操作,则应该使用QueryPerformanceFrequency()和 QueryPerformanceCounter()函数。这两个函数是VC提供的仅供Windows 95及其后续版本使用的精确时间函数,并要求计算机从硬件上支持精确定时器。如示例工程中的Timer7、Timer7_1、Timer7_2、Timer7_3。
QueryPerformanceFrequency()函数和QueryPerformanceCounter()函数的原型如下:
    BOOL QueryPerformanceFrequency(LARGE_INTEGER *lpFrequency);
    BOOL QueryPerformanceCounter(LARGE_INTEGER *lpCount);
  数据类型ARGE_INTEGER既可以是一个8字节长的整型数,也可以是两个4字节长的整型数的联合结构, 其具体用法根据编译器是否支持64位而定。该类型的定义如下:
    typedef union _LARGE_INTEGER
    {
      struct
      {
       DWORD LowPart ;// 4字节整型数
       LONG HighPart;// 4字节整型数
      };
      LONGLONG QuadPart ;// 8字节整型数
      
    }LARGE_INTEGER ;
  在进行定时之前,先调用QueryPerformanceFrequency()函数获得机器内部定时器的时钟频率, 然后在需要严格定时的事件发生之前和发生之后分别调用QueryPerformanceCounter()函数,利用两次获得的计数之差及时钟频率,计算出事件经 历的精确时间。下列代码实现1ms的精确定时:
    LARGE_INTEGER litmp; 
    LONGLONG QPart1,QPart2;
    double dfMinus, dfFreq, dfTim; 
    QueryPerformanceFrequency(&litmp);
    dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
    QueryPerformanceCounter(&litmp);
    QPart1 = litmp.QuadPart;// 获得初始值
    do
    {
     QueryPerformanceCounter(&litmp);
     QPart2 = litmp.QuadPart;//获得中止值
     dfMinus = (double)(QPart2-QPart1);
     dfTim = dfMinus / dfFreq;// 获得对应的时间值,单位为秒
    }while(dfTim<0.001);
  其定时误差不超过1微秒,精度与CPU等机器配置有关。 下面的程序用来测试函数Sleep(100)的精确持续时间:
    LARGE_INTEGER litmp; 
    LONGLONG QPart1,QPart2;
    double dfMinus, dfFreq, dfTim; 
    QueryPerformanceFrequency(&litmp);
    dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
    QueryPerformanceCounter(&litmp);
    QPart1 = litmp.QuadPart;// 获得初始值
    Sleep(100);
    QueryPerformanceCounter(&litmp);
    QPart2 = litmp.QuadPart;//获得中止值
    dfMinus = (double)(QPart2-QPart1);
    dfTim = dfMinus / dfFreq;// 获得对应的时间值,单位为秒   
  由于Sleep()函数自身的误差,上述程序每次执行的结果都会有微小误差。下列代码实现1微秒的精确定时:
    LARGE_INTEGER litmp; 
    LONGLONG QPart1,QPart2;
    double dfMinus, dfFreq, dfTim; 
    QueryPerformanceFrequency(&litmp);
    dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
    QueryPerformanceCounter(&litmp);
    QPart1 = litmp.QuadPart;// 获得初始值
    do
    {
     QueryPerformanceCounter(&litmp);
     QPart2 = litmp.QuadPart;//获得中止值
     dfMinus = (double)(QPart2-QPart1);
     dfTim = dfMinus / dfFreq;// 获得对应的时间值,单位为秒
    }while(dfTim<0.000001);
其定时误差一般不超过0.5微秒,精度与CPU等机器配置有关。
16. 为对话框中的控件增加提示的简单方法

我学VC从VC知识库中得到不少好处,相来都是一些热心朋友们的帮助,在此表示感谢!本工程可分五步:
1、建一个基于对话框的程序TipTest,在CTipTestDlg中增加成员变量:CToolTipCtrl m_tip[2],CWnd *m_pSub[2];
2、在CTipTestDlg::On InitDialog()函数中增加如下代码: 
     pSub[0] = GetDlgItem(IDC_RADIO1);   //得到单选按钮的指针
     pSub[1] = GetDlgItem(IDC_BUTTON1);

"m_tip[0].Create(pSub[0],TTS_ALWAYSTIP); //创建CToolTipCtrl
"m_tip[0].AddTool(pSub[0]);        //将CToolTipCtrl与相应的控件对应起来

"m_tip[1].Create(pSub[1],TTS_ALWAYSTIP);
"m_tip[1].AddTool(pSub[1]);

"m_tip[0].SetTipTextColor(RGB(0,0,255)); //设定文字的颜色
     m_tip[0].SetDelayTime(150);       //设定提示文字在控件上停留的时间

3、重载CTipTestDlg::PreTranslateMessage(MSG* pMsg)函数,增加如下代码: 
   if(m_tip[0].m_hWnd!=NULL)
     m_tip[0].RelayEvent(pMsg);  //如果m_tip[0]句柄不为空,就从主窗口中捕获消息,如WM_MOUSEMOVE,WM_LBUTTONDOWN等消息
   if(m_tip[1].m_hWnd!=NULL)
     m_tip[1].RelayEvent(pMsg); 
4、捕获主窗口的WM_MOUSEMOVE消息,在CTipTestDlg::On MouseMove(UINT nFlags, CPoint point)函数中增加如下代码: 
     m_tip[0].UpdateTipText("VC知识库欢迎你!",pSub[0]);    //鼠标在相应的控件上移动时显示提示文字
"m_tip[1].UpdateTipText("http://vckbase.com",pSub[1]);

17.删除文件夹

// 删除文件夹及其所有内容
void CBaseDoc::RemoveFolder(const CString &strPathName)
{
  CString path = strPathName;
  if (path.Right(1) != _T("\"))
    path += _T("\");
  path += _T("*.*");

  CFileFind ff;
  BOOL res = ff.FindFile(path);
  while (res)
  {
    res = ff.FindNextFile();
    // 是文件时直接删除
    if (!ff.IsDots() && !ff.IsDirectory())
      DeleteFile(ff.GetFilePath());
    else if (ff.IsDots())
      continue;
    else if (ff.IsDirectory())
    {
      path = ff.GetFilePath();
      // 是目录时继续递归,删除该目录下的文件
      RemoveFolder(path);
      ::RemoveDirectory(path);
    }
  }
}

18.消息映射

有对话框A,B
从A中发消息给B然后B处理。
准备工作,先定义消息,如下
#define WM_B_NOTIFY WM_USER + 300 

首先,必须将B的对话框句柄传送给A,暂时叫m_hWndB;

在A的发送消息的地方这样写:
::SendMessage( m_hWndB,WM_B_NOTIFY,TRUE,NULL );

这样A中的处理就完了,下面说B 中的
首先定义消息处理函数,如下
void B::ModiNotify( WPARAM wParam, LPARAM lParam )
{
  MessageBox("小样,我就不信,搞不定你!");
}

然后加消息隐射,如下:
BEGIN_MESSAGE_MAP(CB, CDialog)
  //{{AFX_MSG_MAP(CRPServerDlg)

  ON_MESSAGE( WM_B_NOTIFY,ModiNotify )

  //}}AFX_MSG_MAP
END_MESSAGE_MAP()

19.给从CWnd派生的窗口添加滚动条
ModifyStyle(0,WS_VSCROLL);

20. SetWindowPos
函数功能:该函数改变一个子窗口,弹出式窗口式顶层窗口的尺寸,位置和Z序。子窗口,弹出式窗口,及顶层窗口根据它们在屏幕上出现的顺序排序、顶层窗口设置的级别最高,并且被设置为Z序的第一个窗口。  
  
     函数原型:BOOL  SetWindowPos(HWN  hWnd,HWND  hWndlnsertAfter,int  X,int  Y,int  cx,int  cy,UNIT.Flags);  
  
     参数:  
  
     hWnd:窗口句柄。  
  
     hWndlnsertAfter:在z序中的位于被置位的窗口前的窗口句柄。该参数必须为一个窗口句柄,或下列值之一:  
  
     HWND_BOTTOM:将窗口置于Z序的底部。如果参数hWnd标识了一个顶层窗口,则窗口失去顶级位置,并且被置在其他窗口的底部。  
  
     HWND_DOTTOPMOST:将窗口置于所有非顶层窗口之上(即在所有顶层窗口之后)。如果窗口己经是非顶层窗口则该标志不起作用。  
  
     HWND_TOP:将窗口置于Z序的顶部。  
  
     HWND_TOPMOST:将窗口置于所有非顶层窗口之上。即使窗口未被激活窗口也将保持顶级位置。  
  
     查g看该参数的使用方法,请看说明部分。  
  
     x:以客户坐标指定窗口新位置的左边界。  
  
     Y:以客户坐标指定窗口新位置的顶边界。  
  
     cx:以像素指定窗口的新的宽度。  
  
     cy:以像素指定窗口的新的高度。  
  
     uFlags:窗口尺寸和定位的标志。该参数可以是下列值的组合:  
  
     SWP_ASNCWINDOWPOS:如果调用进程不拥有窗口,系统会向拥有窗口的线程发出需求。这就防止调用线程在其他线程处理需求的时候发生死锁。  
  
     SWP_DEFERERASE:防止产生WM_SYNCPAINT消息。  
  
     SWP_DRAWFRAME:在窗口周围画一个边框(定义在窗口类描述中)。  
  
     SWP_FRAMECHANGED:给窗口发送WM_NCCALCSIZE消息,即使窗口尺寸没有改变也会发送该消息。如果未指定这个标志,只有在改变了窗口尺寸时才发送WM_NCCALCSIZE。  
  
     SWP_HIDEWINDOW;隐藏窗口。  
  
     SWP_NOACTIVATE:不激活窗口。如果未设置标志,则窗口被激活,并被设置到其他最高级窗口或非最高级组的顶部(根据参数hWndlnsertAfter设置)。  
  
     SWP_NOCOPYBITS:清除客户区的所有内容。如果未设置该标志,客户区的有效内容被保存并且在窗口尺寸更新和重定位后拷贝回客户区。  
  
     SWP_NOMOVE:维持当前位置(忽略X和Y参数)。  
  
  SWP_NOOWNERZORDER:不改变z序中的所有者窗口的位置。  
  
  SWP_NOREDRAW:不重画改变的内容。如果设置了这个标志,则不发生任何重画动作。适用于客户区和非客户区(包括标题栏和滚动条)和任何由于窗回移动而露出的父窗口的所有部分。如果设置了这个标志,应用程序必须明确地使窗口无效并区重画窗口的任何部分和父窗口需要重画的部分。  
  
  SWP_NOREPOSITION;与SWP_NOOWNERZORDER标志相同。  
  
  SWP_NOSENDCHANGING:防止窗口接收WM_WINDOWPOSCHANGING消息。  
  
  SWP_NOSIZE:维持当前尺寸(忽略cx和Cy参数)。  
  
  SWP_NOZORDER:维持当前Z序(忽略hWndlnsertAfter参数)。  
  
  SWP_SHOWWINDOW:显示窗口。  
  
  返回值:如果函数成功,返回值为非零;如果函数失败,返回值为零。若想获得更多错误消息,请调用GetLastError函数。  
  
  备注:如果设置了SWP_SHOWWINDOW和SWP_HIDEWINDOW标志,则窗口不能被移动和改变大小。如果使用SetWindowLoog改变了窗口的某些数据,则必须调用函数SetWindowPos来作真正的改变。使用下列的组合标志:SWP_NOMOVEISWP_NOSIZEISWP_FRAMECHANGED。  
  
  有两种方法将窗口设为最顶层窗口:一种是将参数hWndlnsertAfter设置为HWND_TOPMOST并确保没有设置SWP_NOZORDER标志;另一种是设置窗口在Z序中的位置以使其在其他存在的窗口之上。当一个窗口被置为最顶层窗口时,属于它的所有窗口均为最顶层窗口,而它的所有者的z序并不改变。  
  
  如果HWND_TOPMOST和HWND_NOTOPMOST标志均未指定,即应用程序要求窗口在激活的同时改变其在Z序中的位置时,在参数hWndinsertAfter中指定的值只有在下列条件中才使用:  
  
  在hWndlnsertAfter参数中没有设定HWND_NOTOPMOST和HWND_TOPMOST标志。  
  
  由hWnd参数标识的窗口不是激活窗口。  
  
  如果未将一个非激活窗口设定到z序的顶端,应用程序不能激活该窗口。应用程序可以无任何限制地改变被激活窗口在Z序中的位置,或激活一个窗口并将其移到最高级窗口的顶部或非最高级窗口的顶部。  
  
  如果一个顶层窗口被重定位到z序的底部(HWND_BOTTOM)或在任何非最高序的窗口之后,该窗口就不再是最顶层窗口。当一个最顶层窗口被置为非最顶级,则它的所有者窗口和所属者窗口均为非最顶层窗口。  
  
  一个非最顶端窗口可以拥有一个最顶端窗口,但反之则不可以。任何属于顶层窗口的窗口(例如一个对话框)本身就被置为顶层窗口,以确保所有被属窗口都在它们的所有者之上。  
  
  如果应用程序不在前台,但应该位于前台,就应调用SetForegroundWindow函数来设置。  
  
  Windows  CE:如果这是一个可见的顶层窗口,并且未指定SWP_NOACTIVATE标志,则这个函数将激活窗口、如果这是当前的激活窗口,并且指定了SWP_NOACTIVATE或SWP_HIDEWINDOW标志,则激活另外一个可见的顶层窗口。  
  
  当在这个函数中的nFlags参数里指定了SWP_FRAMECHANGED标志时,WindowsCE重画窗口的整个非客户区,这可能会改变客户区的大小。这也是重新计算客户区的唯一途径,也是通过调用SetwindowLong函数改变窗口风格后通常使用的方法。  
  
  SetWindowPos将使WM_WINDOWPOSCHANGED消息向窗口发送,在这个消息中传递的标志与传递给函数的相同。这个函数不传递其他消息。  
  
  Windows  CE  1.0不支持在hWndlnsertAber参数中的HWND_TOPMOST和HWND_NOTOPMOST常量。  
  
   Windows  CE1.0不支持在fuFags参数中的SWP_DRAWFRAME和SWP_NOCOPYBITS标志。  
  
  速查:Windows  NT:3.1以上版本;Windows:95以上版本;Windows  CE:1.0以上版本;头文件:winuser.h库文件:eser32lib

 

21. 介绍函数过程中一种任意键退出同时能处理消息的实现方法

1. 设置定时器,用于使::GetMessage(...)函数总能快速取到消息.
2. 在函数处理中加入:

函数每执行完一步后执行下面的代码.
if (::GetMessage(&msg, 0, 0, 0))
{
  if (msg.message >= WM_KEYFIRST && msg.message <= WM_KEYLAST) return ;
  ::TranslateMessage(&msg);
  ::DispatchMessage(&msg);
}
else ::PostQuitMessage(0);

22. 如何隐藏工具栏

添加如下两个函数
隐藏:
void CMainFrame::OnHide() 
{
  if(m_wndToolBar.IsWindowVisible())
    m_wndToolBar.ModifyStyle(WS_VISIBLE,0);
  SendMessage(WM_SIZE);
}

显示:
void CMainFrame::OnShow() 
{
  if(!m_wndToolBar.IsWindowVisible())
    m_wndToolBar.ModifyStyle(0,WS_VISIBLE);
  SendMessage(WM_SIZE);
}

23. 如何动态获取工具条指针并给工具条加标题?

[问题提出]
工具条也是窗口,是窗口就有标题,如何给工具条加标题?
[程序实现]
不想动态改变工具条的标题就在CMainFrame::OnCreate()中:
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
......
m_wndToolBar.SetWindowText(_T("Standdard")); 
return 0;
}
若想动态改变工具条的标题,如下:
声明一个菜单,并响应事件,如响应:OnMyToolBar()函数
void CMainFrame::OnMyToolBar() 
{
// TODO: Add your command handler code here
CToolBar *pToolBar = (CToolBar *)AfxGetMainWnd()->GetDescendantWindow(AFX_IDW_TOOLBAR); 
pToolBar->SetWindowText (_T("Standdard"));
}
不要在TooBar悬浮时做OnMyToolBar()会出错的. 
顺便提一下如何获得状态条的指针:
CStatusBar * pStatusBar =(CStatusBar *)AfxGetMainWnd()->GetDescendantWindow(AFX_IDW_STATUS_BAR);

24. 在状态条中显示鼠标的设备坐标与逻辑坐标

显示器的设备坐标系的原点在客户区的左上角,x轴向右增长,y轴向下增长。我们要设置的逻辑坐标系的原点则在客户区的中心,x轴向右增长,y轴向上增长,如一个笛卡尔坐标系一般。

为CChildView添加一个成员函数void OnPrepareDC(CDC * pDC, CPrintInfo * pInfo = NULL);

void OnPrepareDC(CDC * pDC, CPrintInfo * pInfo){
 CRect rect;

 // 设置映射模式为LOMETRIC (0.1mm),右上为增长方向
 pDC->SetMapMode (MM_LOMETRIC);

 // 将坐标原点定在客户区的中心
 GetClientRect(rect);
 pDC->SetViewportOrg(rect.Width()/2, rect.Height()/2);
}
为CChildView响应鼠标移动消息,并在状态条中显示鼠标的坐标值。m_ptMouse数据成员是原打算做十字交叉线用的,在此使用没有实际意义。

void CChildView::OnMouseMove(UINT nFlags, CPoint point){
 CClientDC dc(this);
 CString str;
 
 OnPrepareDC(&dc);

 //要访问类CMainFrame,需要将mainfrm.h文件引入
 CMainFrame * pFrame = (CMainFrame *) AfxGetApp()->m_pMainWnd;

 //要访问CMainFrame的数据成员m_wndStatusBar,需要手工修改mainfrm.h,public这个数据成员
 CStatusBar * pStatus = (CStatusBar *) &pFrame->m_wndStatusBar;
 
 m_ptMouse = point;
 str.Format ("设备坐标 X=%i pixel, Y=%i pixel", m_ptMouse.x, m_ptMouse.y);
 pStatus->SetPaneText(1, str);
 
 dc.DPtoLP(&m_ptMouse);
 str.Format ("逻辑坐标 X=%i * 0.1mm, Y=%i * 0.1mm", m_ptMouse.x, m_ptMouse.y);
 pStatus->SetPaneText(2, str);
}

25. 如何用VC++ 动态修改应用程序菜单

 [问题提出]
  本文将介绍一些使用CMenu的方法,如查找指定菜单,在指定选项前添加菜单项.....

 [解决方法]
  使用CWnd::GetMenu( )访问主菜单,GetMenu( )返回指向CMenu对象的指针,它有一些成员函数,允许我们修改一个菜单。
  1) 如何实现找到一个菜单项:
  步骤如下:
  {
     //动态修改菜单:
     // Get the Main Menu
     CMenu* pMainMenu = AfxGetMainWnd()->GetMenu();
     CMenu* pSubMenu = NULL;
     int i;
     for (i=0; i<(int)pMainMenu->GetMenuItemCount(); i++)
     {
      pSubMenu = pMainMenu->GetSubMenu(i);
      if (pSubMenu && pSubMenu->GetMenuItemID(0) == ID_FILE_NEW)
        break;
     }
     CString s;
     s.Format("%d",i);//菜单项的位数.
     AfxMessageBox(s);
     ASSERT(pSubMenu);
  }

  2) 动态编辑菜单:
  步骤如下(可以用上例的pSubMenu,要加的菜单你自己定义.):
  1) 添加一个称为Wzd2,命令ID为IDC_NAME_NEW1的菜单命令到该菜单中,可以用:
     pSubMenu->AppendMenu(0,IDC_NAME_NEW1,"New&1");

  2) 在New1前插入New2,可以用:
     pSubMenu->InsertMenu(IDC_NAME_NEW1,MF_BYCOMMAND,IDC_NAME_NEW2, "New&2");

  3) 把New1改变成New3,可以用:
     pSubMenu->ModifyMenu(IDC_NAME_NEW1,MF_BYCOMMAND,IDC_NAME_NEW3, "New&3");

  4) 删除该菜单中第二项,可以用:
     pSubMenu->RemoveMenu(1,MF_BYPOSITION);

26. VC++中的3D按钮的编程

运行AppWizard生成一个基于对话框的test工程,在对话框中加入一个CButton控件。在CButton控件的General属性页将控件的ID改为IDC_3DTEXTBTN,Caption改为“谁与争疯”,在控件Styles属性页选中OwnerDraw,其余设置保持默认。
  用classwizard创建一个新类:C3dTextButton,基类为CButton。为C3dTextButton类添加一个protected的函数void Draw(CDC* pDC, const CRect& rect, UINT state)。如下所示编写代码:
  void C3dTextButton::Draw(CDC *pDC, const CRect &rect, UINT state)
  {
    CString text; GetWindowText(text);
    int l=text.GetLength();
    CRect rectClient=rect;
  
    //获得控件的字体
    CFont* pFont=GetFont();
  
    //确定所选字体有效高度和宽度
    LOGFONT logfont;
    pFont->GetObject(sizeof(LOGFONT),&logfont);
    if(logfont.lfHeight==0)logfont.lfHeight=20;
    logfont.lfWidth=0;//宽度设为0,宽度值由高度确定
    logfont.lfWeight=1000;
    logfont.lfEscapement=logfont.lfOrientation=0;
    CFont tryfont; VERIFY(tryfont.CreateFontIndirect(&logfont));
    CFont* pFontOld=pDC->SelectObject(&tryfont);
  
    //根据控件大小,调整字体的高度,使文本与控件协调
    CSize textSizeClient=pDC->GetTextExtent(text,l);
    if(rectClient.Width()*textSizeClient.cy>rectClient.Height()*textSizeClient.cx)
    {
      logfont.lfHeight=::MulDiv(logfont.lfHeight,rectClient.Height(),textSizeClient.cy);
    }
    else{
      logfont.lfHeight = ::MulDiv(logfont.lfHeight,rectClient.Width(),textSizeClient.cx);
    }
  
    //创建并选择协调后的字体
    CFont font; font.CreateFontIndirect(&logfont);
    pDC->SelectObject(&font);
    textSizeClient=pDC->GetTextExtent(text,l);
    //确定文本与控件边界的距离minx,miny
    int minx=rectClient.left+(rectClient.Width()-textSizeClient.cx)/2;
    int miny=rectClient.top+(rectClient.Height()-textSizeClient.cy)/2;
    int oldBkMode=pDC->SetBkMode(TRANSPARENT);
    COLORREF textcol=::GetSysColor(COLOR_BTNTEXT);
    COLORREF oldTextColor=pDC->SetTextColor(textcol);
    int cx = minx;
    int cy = miny;
    int s=(state&ODS_SELECTED)?-1:+1;
    cx+= 3; cy+= 3;
  
    //实现3D效果
    pDC->SetTextColor(::GetSysColor(COLOR_3DDKSHADOW));
    pDC->TextOut(cx-s*2,cy+s*2,text);
    pDC->TextOut(cx+s*2,cy-s*2,text);
    pDC->TextOut(cx+s*2,cy+s*2,text);
    pDC->SetTextColor(::GetSysColor(COLOR_3DHILIGHT));
    pDC->TextOut(cx+s*1,cy-s*2,text);
    pDC->TextOut(cx-s*2,cy+s*1,text);
    pDC->TextOut(cx-s*2,cy-s*2,text);
    pDC->SetTextColor(::GetSysColor(COLOR_3DSHADOW));
    pDC->TextOut(cx-s*1,cy+s*1,text);
    pDC->TextOut(cx+s*1,cy-s*1,text);
    pDC->TextOut(cx+s*1,cy+s*1,text);
    pDC->SetTextColor(::GetSysColor(COLOR_3DLIGHT));
    pDC->TextOut(cx,cy-s*1,text);
    pDC->TextOut(cx-s*1,cy,text);
    pDC->TextOut(cx-s*1,cy-s*1,text);
    pDC->SetTextColor(textcol);
  
    //输出标题
    pDC->TextOut(cx,cy,text);
  
    //恢复设备描述表
    pDC->SetTextColor(oldTextColor);
    pDC->SetBkMode(oldBkMode);
    pDC->SelectObject(pFontOld);
  }

  用classwizard重载C3dTextButton类的DrawItem函数。编写代码如下所示:
  void C3dTextButton::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
  {
    CDC* pDC=CDC::FromHandle(lpDrawItemStruct->hDC);
    ASSERT_VALID(pDC);
    CRect rectClient=lpDrawItemStruct->rcItem;
    Draw(pDC,rectClient,lpDrawItemStruct->itemState);
  }
  用classwizard为IDC_3DTEXTBTN建立一个C3dTextButton控件变量m_3dTextButton1。

把“3dTextButton.h”加入testDlg头文件。编译并测试应用程序。

27. 如何正确的得到ComBox的指针

CComboBox *mComb = (CComboBox*)GetDlgItem(IDC_DuanCB);
CComboBox *mComb = (CComboBox*)::GetDlgItem(m_hWnd,IDC_DuanCB);

28. 如何让对话框中的CEdit控件类接收对话框的消息


// 如何让对话框中的CEdit控件类接收对话框的消息

1、在对话框中增加一个ID 为IDC_EDIT1的CEdit1控件

2、通过ClassWizard 生成一个基于CEdit的新类CMyEdit,

CMyEdit m_wndEdit;

3、在对话框OnInitDialog()中,将m_wndEdit子类化,使其能够接受对话框的消息。

m_wndEdit.SubclassDlgItem (IDC_EDIT1,this);

29.利用WM_CTLCOLOR消息实现编辑控制(Edit Control)的文本与背景色的改变

首先要明白:WM_CTLCOLOR是一个由控制(Control)发送给它父窗口的通知消息(Notification message)。

实现步骤:
生成一个标准的单文档应用程序框架,假设应用程序的名称为Color。我将利用它的About对话框做示范。在About dialog中添加两个Edit control,设定其ID为IDC_EDIT1与IDC_EDIT2。

第一种方法(对应于IDC_EDIT1): 按照标准的Windows编程,由其父窗口的消息处理函数负责处理WM_CTLCOLOR消息。

1. 在CAboutDlg中添加一个数据成员:HBRUSH m_brMine;
2. 利用向导映射AboutDlg的WM_CTLCOLOR消息,产生函数:HBRUSH CAboutDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor);
pDC是AboutDlg的设备上下文,pWnd是AboutDlg中发送该消息的control指针,nCtlColor市Control的类型编码。对其进行如下修改:

HBRUSH CAboutDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor) 
{
 if ((pWnd->GetDlgCtrlID() == IDC_EDIT1) && (nCtlColor == CTLCOLOR_EDIT))
 {
   COLORREF clr = RGB(255,0,0);
   pDC->SetTextColor(clr);  //设置红色的文本
   clr = RGB(0,0,0);
   pDC->SetBkColor(clr);   //设置黑色的背景
   m_brMine = ::CreateSolidBrush(clr);
   return m_brMine; //作为约定,返回背景色对应的刷子句柄
 }
 else
 {
   HBRUSH hbr = CDialog::OnCtlColor(pDC, pWnd, nCtlColor);
   return hbr;
 }
}

第二种方法(对应于IDC_EDIT2): 
利用MFC 4.0的新特性: Message reflection。

1.利用向导添加一个新的类:CColorEdit,基类为CEdit;
2.在CColorEdit中添加一个数据成员: HBRUSH m_bkBrush;
3.利用向导映射CColorEdit的"=WM_CTLCOLOR"消息,产生函数:

HBRUSH CColorEdit::CtlColor(CDC* pDC, UINT nCtlColor); 

对其进行如下修改:

HBRUSH CColorEdit::CtlColor(CDC* pDC, UINT nCtlColor) 
{
 COLORREF clr = RGB(0,0,0);
 pDC->SetTextColor(clr);  //设置黑色的文本
 clr = RGB(255,0,0);
 pDC->SetBkColor(clr);   //设置红色的背景
 m_bkBrush = ::CreateSolidBrush(clr);
 return m_bkBrush; //作为约定,返回背景色对应的刷子句柄
}

4.利用向导为IDC_EDIT2生成一个数据成员CColorEdit m_coloredit;
5.在定义CAboutDlg的color.cpp文件中加入:#include "coloredit.h"

30. 如何防止密码被非法获取?

 [问题提出]
  这两天大家比较专注在获取Edit密码框的密码.在盗取时,我们如何防范呢?
 
 [解决方法]
  此方法针对于通过SendMessage向此窗口发送WM_GETTEXT或EM_GETLINE消息来取得密码.跟我来.
 
 [程序实现]
  方法很简单,用CWnd::DefWindowProc函数拦截得到的消息(向Edit发的).
  建立名为My的对话框工程.建立一个Edit控件ID=IDC_EDIT1.建一个新类名为CMyProtectEdit,派生于CEdit.
  在MyDlg.cpp中声明全局变量:BOOL g_bIdentity;
  BOOL g_bIdentity;

  在MyProtecEdit.cpp中:
  extern BOOL g_bIdentity;

  响应CMyProtectEdit的DefWindowProc函数:
  LRESULT CMyProtectEdit::DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam) 
  {
    // TODO: Add your specialized code here and/or call the base class
    // 对Edit的内容获取必须通过以下两个消息之一,不对其采用默认的处理:
    if(( message == WM_GETTEXT) || ( message == EM_GETLINE))
    {  //检查是否为合法
     if(!g_bIdentity)
     {  //非法获取,显示非法信息
       AfxMessageBox(_T("不能让你看我的密码,:( !"));
       return 0;
     )
     g_bIdentity = FALSE;//合法获取
    }
    
    return CEdit::DefWindowProc(message, wParam, lParam);
  }

  然后在MyDlg.cpp中
  void CMyDlg::DoDataExchange(CDataExchange* pDX)
  {
    CDialog::DoDataExchange(pDX);
    //{{AFX_DATA_MAP(CGetPasswordDlg)
    // NOTE: the ClassWizard will add DDX and DDV calls here
    if( pDX->m_bSaveAndValidate)
    {
     g_bIdentity = TRUE;
    }   
    //}}AFX_DATA_MAP
  }
  即可.找个程序(盗取)的试试.

31. 如何在编辑控件中以追加的方式添入字符?

 [问题提出]
  SetDlgItemText可以向Edit控件中输入字符,发送更新的消息也可是Edit控件显示与其关联的变量的值,但若是向已有的Edit字符后追加字符,该如何做?
 [程序实现]
  建立名为My的对话框工程,添加一个Edit和一个Button控件.Edit的ID=IDC_EDIT1,Button的ID=IDC_BUTTON1.建立和IDC_BUTTON1的响应函数:OnButton1()
  void CMyDlg::OnButton1() 
  {
    CString pText="你好";
    CEdit *m_Edit=(CEdit *)GetDlgItem(IDC_EDIT1);
    int nLen=m_Edit->GetWindowTextLength(); 
    m_Edit->SetFocus(); 
    m_Edit->SetSel(nLen, nLen); 
    m_Edit->ReplaceSel(pText); 
  }
  在Edit控件中输入字符,想追加时按IDC_BUTTON1按钮.看看效果.

32.属性页标题改名

我用CPropertySheet创建属性页,用的CPropertyPage对象只有一个,也就是每个属性页的内容一样.现在的问题是:这样每个属性页的标题都是一样的,是对话框的标题!怎样动态的改变这个标题,使每个属性页的标签的名称都不同??

CTabCtrl * pCtrl = pSheet->GetTabControl();
TCITEM tc;
tc.mask = TCIF_TEXT;
tc.pszText = "新标题";
pCtrl->SetItem(0,&tc);//0即是你要改的TAb的索引

转载于:https://www.cnblogs.com/mr-totoro/archive/2012/07/18/5785757.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值