在MDi窗体嵌入子窗体后不显示菜单栏
背景:
由于之前做的一个程序的功能全部都是放在一个界面上的,有一个功能能够在数据库查询数据,并返回到界面上,数据量比较小的时候还好,但是数据量多了,导致它阻塞的其他线程,经过一系列讨论之后,决定将一个界面换成一个主界面加多个子界面。
实施:
多个子界面迁移完成之后,使用下面的方法将其放置的主界面中(先将主窗体的IsMdiContainer设置为true,不然会报错)
c#
Form frm = new SssForm();
foreach (Form childForm in MdiChildren)
{
if (childForm != CccForm)
{
childForm.Close();
}
}
this.CccForm.Visible = false;
frm.MdiParent = this;
frm.Show();
frm.WindowState = FormWindowState.Maximized;
然后在主窗体和子窗体的属性那里把ControlBox属性设置为false,然后测试发现主窗体一开始是没有菜单栏的,但是打开子窗体之后就会在主窗体的右上角显示出菜单栏
查阅了一些网上的办法去改变其他属性值,但是测试之后发现没有用,有说用pannel的,但是我的这个程序需要在Mdi中实现,就没有去实践了,后面就去问gpt4了,最后用gpt4给的方法实现了
这个方法是重写WndProc,在获取到子窗体需要重新计算大小时,直接告诉系统我们只需要计算工作区,而不需要把菜单栏加入进来,这里微软的官方文档里面也有写到(https://learn.microsoft.com/zh-cn/windows/win32/winmsg/wm-nccalcsize)
以下是代码实现:
c#
protected override void WndProc(ref Message m)
{
const int WM_NCCALCSIZE = 0x0083;
const int WM_NCHITTEST = 0x0084;
switch (m.Msg)
{
case WM_NCCALCSIZE:
// 当窗体的大小需要重新计算时,系统会发送WM_NCCALCSIZE消息
// 这里可以修改消息的处理,以改变窗体非客户区的大小
// 如果wParam是TRUE(非零),则指示客户区大小需要重新计算
// 通过简单地返回0,我们可以告诉Windows我们已处理消息,不需要默认的非客户区
// 这实际上会移除所有的非客户区,包括边框和标题栏
if (m.WParam.ToInt32() != 0)
{
// 返回0表示我们处理了这个消息,不再需要默认的处理
// 这将去除非客户区,包括标题栏和边框
m.Result = IntPtr.Zero;
return;
}
break;
case WM_NCHITTEST:
// 可以在这里处理鼠标事件,例如检测鼠标是否在我们自定义的标题栏区域内
// 这对于添加拖动行为等自定义交互是有用的
base.WndProc(ref m);
if ((int)m.Result == 0x01) // HTCLIENT
{
// 可以修改m.Result来改变鼠标的行为,例如使其支
// m.Result = (IntPtr)2; // HTCAPTION 表示标题栏,允许拖动窗体
}
return;
}
base.WndProc(ref m);
}
然后根据我想要的功能给它简单的优化了一些
c#
protected override void WndProc(ref Message m)
{
if (m.Msg == 0x0083 && m.WParam.ToInt32() != 0)
{
m.Result = IntPtr.Zero;
return;
}
base.WndProc(ref m);
}
效果图: