VC禁止或允许拖拽改变窗口尺寸

1 使用MFC:PreCreateWindow中添加:
  固定大小:cs.style=WS_OVERLAPPED   |   WS_SYSMENU   |WS_MINIMIZEBOX   ;
  可改变大小,把如上改为:
cs.style=WS_OVERLAPPED   |   WS_SYSMENU   |WS_MINIMIZEBOX   |   WS_THICKFRAME;  这样的窗口MaxBox为disabled..
2 WIN32 API方式:
  调用CreateWindowEx函数前,设置WNDCLASS参数,同上面;

其实方法很多的,这里列出几个比较常见的方法来,这里是以SDI的框架窗口为例子来试验的。

1.创建一个最大化的窗口,并且不能改变其大小

那么首先,

app的InitInstance中     在原来pMainFrame->ShowWindow(m_nCmdShow);                     的前面加上         DWORD   dwStyle   =   GetWindowLongm_pMainWnd->m_hWnd,   GWL_STYLE);    // dwStyle   &=   ~(WS_SIZEBOX);     dwStyle   &=   ~(WS_MAXIMIZEBOX);     dwStyle   &=   ~(WS_MINIMIZEBOX);     SetWindowLong(m_pMainWnd->m_hWnd,   GWL_STYLE,   dwStyle);  

m_pMainWnd->ShowWindow(SW_SHOWMAXIMIZED);         然后把m_pMainWnd->ShowWindow(m_nCmdShow);可以删了 

这里我把dwStyle   &=   ~(WS_SIZEBOX);注释调了,因为把这个属性去掉的话,会产生麻烦,就是我这个窗口最大话显示的时候,会把任务栏也遮了,靠,实在是另人郁闷但是如果不把这个属性去掉的话,那么通过拖拉,还是可以改变这个窗口的大小的,那怎么办呢,真伤脑筋阿,换个思路吧

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

用class wizard看了以下,竟然没有找到WM_NCHITTEST这个消息,郁闷,只能手动添加消息映射了

在BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd)下添上 ON_WM_NCHITTEST()

在框架类的头文件下声明 afx_msg UINT OnNcHitTest(CPoint point);

实现

UINT CMainFrame::OnNcHitTest(CPoint point) {      if(CWnd::OnNcHitTest(point) == HTRIGHT || CWnd::OnNcHitTest(point) == HTLEFT || CWnd::OnNcHitTest(point) == HTTOP || CWnd::OnNcHitTest(point) == HTBOTTOM)        return HTCLIENT;

      return CWnd::OnNcHitTest(point); } 这样好了吗,没这只是限制了四条边,虽然不能拖拉四条边了,但是四个角阿呢,郁闷,真麻烦

再加上 HTTOPLEFT HTTOPRIGHT HTBOTTOMLEFT   HTBOTTOMRIGHT

这样4边+4角,靠,看你怎么玩

圆满了吗??没,还缺一点点 ,缺什么双击窗口最上方也就是caption区域时窗口会变小,而且没办法回复怎么办??凉拌………………

有办法的啦

添加ON_WM_NCLBUTTONDBLCLK消息         void   CMainFrame::OnNcLButtonDblClk(UINT   nFlags,   CPoint   point)     {             if(nFlags   !=   HTCAPTION)                 CFrameWnd::OnNcLButtonDblClk(nFlags,   point);     }           手动添加一下     WM_NCLBUTTONDBLCLK     这个消息的处理         记得BEGIN_MESSAGE_MAP     那边要加     ON_WM_NCLBUTTONDBLCLK

这也是手动添加了消息映射,处理以下,ok,搞定,手工

2.创建一个不可改变大小的窗口,哦,yeah,这个简单了

app的InitInstance中     在原来pMainFrame->ShowWindow(m_nCmdShow);                     的前面加上         DWORD   dwStyle   =   GetWindowLongm_pMainWnd->m_hWnd,   GWL_STYLE);     dwStyle   &=   ~(WS_SIZEBOX);       SetWindowLong(m_pMainWnd->m_hWnd,   GWL_STYLE,   dwStyle);  

m_pMainWnd->ShowWindow(SW_SHOW);  

如果还想把最大,最小话窗口去掉,再加上这个 dwStyle   &=   ~(WS_MAXIMIZEBOX);     dwStyle   &=   ~(WS_MINIMIZEBOX); 把  (WS_SIZEBOX)去掉以后,我们就不能改变窗口大小了,

当然,也可以用WM_NCHITTEST的方法了

3.限制窗口的大小范围

响应WM_GETMAXMININFO  的消息

处理之 void CMainFrame::OnGetMinMaxInfo(MINMAXINFO FAR* lpMMI) {  // TODO: Add your message handler code here and/or call default    lpMMI->ptMinTrackSize.x   = 100 ;      lpMMI->ptMinTrackSize.y   = 100  ;      lpMMI->ptMaxTrackSize.x   = 200  ;       lpMMI->ptMaxTrackSize.y   =  200;  

CFrameWnd::OnGetMinMaxInfo(lpMMI); }

这样的话,窗口就被我框死啦,嘿嘿长和宽的范围都是100-200

当然 MINMAXINFO这个结构体内容也是非常丰富的,可以做的事很多,具体可以察msdn了。

————————————————————————————————————————————————

还有一篇参考MSDN,讲当窗口的大小改变后,如何控制窗口的控件变化:

http://blog.csdn.net/starlee/archive/2006/04/17/666222.aspx

_____________________________________________________________________

LONG   style   =   ::GetWindowLong(this-> m_hWnd,GWL_STYLE);

style   &=   ~(WS_DLGFRAME   |   WS_THICKFRAME);
    SetWindowLong(this-> m_hWnd,GWL_STYLE,   style);
    this-> ShowWindow(SW_SHOWMAXIMIZED);
    CRect   rect;
    this-> GetWindowRect(&rect);
    ::SetWindowPos(this-> m_hWnd,HWND_NOTOPMOST,rect.left-1,   rect.top-1,   rect.right-rect.left   +   3,   rect.bottom-rect.top   +   3,   SWP_FRAMECHANGED);

style   &=   ~(WS_DLGFRAME   |   WS_THICKFRAME);
是可以达到要求。但用了后就出现了新问题
我是想最大化时无法拖动边框,但还原后可以拖动,
我在还原响应里用了
GetWindowLong(..style);
style   |=   WS_DLGFRAME   |   WS_THICKFRAM;
SetWindowLong(..style);
发现style无法改回来,边框还是跟最大化一样没了

style   |=   WS_DLGFRAME   |   WS_THICKFRAME;
    SetWindowLong(this-> m_hWnd,   GWL_STYLE,   style);
    this-> ShowWindow(SW_NORMAL);

试过了,重设的Style没起作用,晕了
这消息响应是放在CMainFrame下的一个dialogBar里,难道跟这个有关吗?
可是为什么
style   &=   ~(WS_DLGFRAME   |   WS_THICKFRAME);起作用了
而   style   |=   WS_DLGFRAME   |   WS_THICKFRAME;不起作用呢?

代码贴出来给看看:


void   CShapeDialogBar::OnShapeButtonCommand(CString   sShapeName)
{
                  if(sShapeName==_T( "MAXMIZE "))
{
CWnd*   pParent=GetParent();//取得CMainFrame指针

LONG   nStyle=GetWindowLong(pParent-> GetSafeHwnd(),GWL_STYLE);
if(!m_bMWndMaxed)   //BOOL   m_bWndMaxed,指示窗口状态
{
pParent-> GetWindowRect(&m_rcOriMainWnd);
CRect   rcWorkArea;
SystemParametersInfo(SPI_GETWORKAREA,0,&rcWorkArea,0);
pParent-> MoveWindow(&rcWorkArea);

nStyle   &=   ~(   WS_DLGFRAME   |   WS_THICKFRAME   );
}
else
{
pParent-> MoveWindow(&m_rcOriMainWnd);
pParent-> CenterWindow();

nStyle   |=   WS_DLGFRAME   |   WS_THICKFRAME   ;
}
SetWindowLong(pParent-> GetSafeHwnd(),GWL_STYLE,nStyle);

pParent-> ShowWindow(SW_NORMAL);

m_bMWndMaxed=!m_bMWndMaxed;

}
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值