加载中…
个人资料
  • 博客等级:
  • 博客积分:
  • 博客访问:
  • 关注人气:
  • 获赠金笔:0支
  • 赠出金笔:0支
  • 荣誉徽章:
正文 字体大小:

窗口大小变化触发的事件及异常

(2017-11-18 14:11:09)
标签:

vc

窗口大小变化

窗口大小异常

wm_size

wm_sizing

分类: VC

以下内容摘自网络,尚未测试。

窗口尺寸变化时触发三个消息:WM_SIZEWM_SIZINGWM_GETMINMAXINFO

程序员可以在对应的函数OnSizeOnSizingOnGetMinMaxInfo添加代码,获取消息,进行操作。

这三个消息响应的顺序是:WM_GETMINMAXINFO-->WM_SIZING-->WM_SIZE

因子窗口是依附于你窗口,若子窗口的风格为SS_NOTIFY(即notify属性为true),若该窗口大小变化,子窗口随之变化。

注意事项:

1、窗口大小的状态是保存在GWL_STYLE对应的窗口数据结构中, 一个窗口当前处于哪个状态是根据GWL_STYLE的值判断的,与窗口的实际大小无关, SetWindowPos, MoveWindow, SetWindowPlacement等函数不会改变窗口的最大化、最小化等状态。

2、父窗口内含多个子窗口,父窗口最大化、还原时,可能发生异常,这在下面有专门讲述。

3、设置窗口最大化,还原按钮不正常。详见后面信息。

 

一、窗口大小变化触发事件

1OnGetMinMaxInfo

这个函数在窗口初始化的时候会被调用一次,当窗口大小发生改变的时候也会被调用。利用这个函数,可以比较方便的实现窗口最大最小尺寸的控制。

参数lpMMI是一个结构体指针,其中包含了有关窗口的最大化大小和位置以及最小、最大跟踪大小的信息。

使用这个函数控制窗口最小尺寸的示例的代码如下:

 

代码如下:


void CXXXDlg::OnGetMinMaxInfo(MINMAXINFO FAR* lpMMI)
{
    lpMMI->ptMinTrackSize.x = 500;    //x
宽度
    lpMMI->ptMinTrackSize.y = 100;    //y
高度
    CDialog::OnGetMinMaxInfo(lpMMI);
}

//以上代码可以使得窗口大小变化时,最小宽度为500px,最小高度为100px

 

2OnSizing

这个函数在窗口大小发生变化时被调用。在这个函数里,也可以控制窗口的最大最小尺寸,但是没有OnGetMinMaxInfo方便。

代码如下:


void CXXXDlg::OnSizing(UINT fwSide, LPRECT pRect)
{
    if ((pRect->right - pRect->left) < 500)
    {
        //return ;    //
直接return是无效的,窗口大小还是会改变
        pRect->right = pRect->left + 500;
    }
    CDialog::OnSizing(fwSide, pRect);
}

 

 

3OnSize

这个函数会在窗口大小改变结束后被调用,通常会在这个函数里重新摆放各个控件的位置及大小。用这个函数暂时没有找到什么办法可以控制窗口的最大最小大小。

  

二、父窗大小变化,多子窗变化异常

 

1、主窗口最大化、还原时,子窗口大小异常现象:

(1) 主窗口最大化时,左边子窗口大小不变,右边子窗口填充其他区域;

(2)主窗口还原时,左边子窗口大小变大,右边子窗口填充其他区域;

(3)拉伸主窗口时,子窗口跟着变化,大小正常。

 

2、处理方法:

方法一。在OnSize消息事件中处理

//最小化 SIZE_MINIMIZED,最大化 SIZE_MAXIMIZED,还原 SIZE_RESTORED

例如:

void CMainDlg::OnSize(UINT nType, int cx, int cy) 
{
 CDialog::OnSize(nType, cx, cy);
 
 // TODO: Add your message handler code here
 //::SendMessage(g_hMain,WM_FORM_RESIZE,0,0);
 if   (nType== SIZE_MAXIMIZED)     
   
//
在这里添加最小化需要处理的事件 
         // ::SendMessage(g_hMain,WM_FORM_RESIZE,0,0);
 }
 if   (nType==SIZE_RESTORED)     
   
//
在这里添加还原时需要处理的事件 
        //  ::SendMessage(g_hMain,WM_FORM_RESIZE,0,0);
 }
}

 

方法二

OnSysCommand中处理   
  nID     SC_MAXIMIZE   
是最大化   
  nID   
   SC_MINIMIZE   是最小化

void CMainDlg::OnSysCommand(UINT nID, LPARAM lParam)

{

      if   (nID== SIZE_MAXIMIZED)     
       
           //
在这里添加最小化需要处理的事件 
            // ::SendMessage(g_hMain,WM_FORM_RESIZE,0,0);
     }
     if   (nID==SIZE_RESTORED)     
     
           //
在这里添加还原时需要处理的事件 
          //  ::SendMessage(g_hMain,WM_FORM_RESIZE,0,0);
    

}

 

方法三:

    一般情况下是在消息处理过程中进行判断,:   
  LRESULT   CALLBACK   WndProc(HWND   hWnd,   UINT   msg,   WPARAM   wParam,   LPARAM   lParam)   
  {   
          switch(msg)   
          {   
                  case   WM_DESTROY:   
                          //  
响应代码   
                  break;   
                  case   WM_GETMINMAXINFO:     //   lParam
返回一个指向MINMAXINFO结构的指针   
                          //  
响应代码   
                  break;   
                  default:   
                          return   DefWindowProc(hWnd,   msg,   wParam,   lParam);   
          }   
          return   0;   
  }

 

WM_GETMINMAXINFO结构说明: 

WM_GETMINMAXINFO

    WPARAM wParam

    LPARAM lParam;

参数:wParam:未使用。

lParam:指向MINMAXINFO结构, 包含定义最大化、最小化的指针和大小以及定义最大化和最小化跟踪大小。程序可以通过设置该结构成员重写。

 

三、最大化时,窗口还原按钮异常

 

异常现象:窗口拉伸正常,窗口最大化时还原按钮不正常。

在收到WPARAMSIZE_RESTOREDWM_SIZE消息时调用SetWindowPos将窗口大小设置为屏幕大小,窗口右上角显示的应该是还原按钮。

 

SendMessagePostMessage发送WM_SIZE消息。

 

SendMessage的基本结构如下:
  SendMessage(
  HWND hWnd,  //消息传递的目标窗口或线程的句柄。
  UINT Msg, //消息类别(这里可以是一些系统消息,也可以是自己定义,下文具体介绍,)
  WPARAM wParam, //参数1WPARAM 其实是与UINT是同种类型的

LPARAM lParam);

 

SendMessage使用方法:

LONG  xy;

xy =MAKELONG(x, y);

SendMessage(WM_SIZE, WPARAM(SIZE_MAXSHOW),(LPARAM)xy);

 

其中:x  = 宽度, y = 高度。宽度在低位,高度在高位。在发送消息时,将宽度(x),高度(y)合并成一个LONG类型。

在处理该消息中,再分解出相应的宽度和高度。宽度(x) = LOWORD(lParam),高度(y) = HIWORD(lParam)。一般不需要我们来解析,因为在OnSize()响应函数中系统已经帮我们解析出来了,即cx(宽度)cy(高度)nType = wParam

 

四、下面的资料,没有细看,感觉应有帮助,先保存一下。


MFCSendMessage的用法与相应函数的添:
 

 1::SendMessage(AfxGetMainWnd()->m_hWnd,WM_CHILDFRAMEDBCLK ,0,0);
  其中WM_CHILDFRAMEDBCLK是自定义的消息ID(IDWM_USER+1),AfxGetMainWnd()->m_hWnd是获得主窗口(这里不能使用GetParent()->m_hWnd或者GetParentFrame()->m_hWnd,因为这是获得父窗口,但父窗口不一定是主窗口,一定要注意,不然消息就会发错导致接收不到。
  

2、然后接下来定义一个消息需要映射的函数,如下:
  afx_msg LRESULT OnChlidFrameDBClick(WPARAM wParam, LPARAMlParam);
  注意格式必须是:两个参数必不可少,返回类型一定为LRESULT,网上很多文章都忽略了这两点,这也是网上文章普遍错误的地方。
  

3、添加消息函数映射
  ON_MESSAGE(WM_CHILDFRAMEDBCLK,OnChlidFrameDBClick)
  注意这里必须是ON_MESSAGE, 不能使用ON_COMMAND, 前者主要针对用户自定义消息,后者针对WM_COMMAND命令,比如菜单、工具栏等.
  

4、实现消息函数:
  我们在接收窗体里定义一个这样的事情(过程),
  LRESULTCMainFrame::OnChlidFrameDBClick(WPARAM wParam, LPARAM lParam)
      {

 CancelFullScreenWin();  //自定义函数

return 0;

 }

0

阅读 收藏 喜欢 打印举报/Report
  

新浪BLOG意见反馈留言板 欢迎批评指正

新浪简介 | About Sina | 广告服务 | 联系我们 | 招聘信息 | 网站律师 | SINA English | 产品答疑

新浪公司 版权所有