您的位置:首页 > 其它

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

2013-01-17 09:01 691 查看
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;

}

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: