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;
}
}