MDI窗体关闭问题解决一例

博客围绕MDI窗体关闭问题展开。背景是导航子窗体有关闭控制,导致关闭MDI主窗体失败。先尝试设置导航窗体关闭标志不可行,后发现主窗体和子窗体共用关闭事件参数,在主窗体CLOSING事件设CANCEL=false解决问题,但有漏洞。最后给出两个处理方案及相关代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

由于工作忙,很久没有来这里了,久违了。

背景:

MDI子窗体中的一个是导航窗体,该窗体定义了CLOSING事件处理:检测窗体的关闭标志是否允许,如果不允许则取消关闭操作,这样保证了导航窗体的不被关闭而永久存在。


问题:

当关闭MDI容器窗体的时候,因为其中一个子窗体的关闭事件被取消了,所以关闭动作失败,导致要关闭MDI主窗体只能通过调用Application的Exit方法来实现真正的关闭操作。

解决:
当关闭MDI子窗体的时候,由于导航窗体有关闭控制处理,可能会取消窗体的关闭操作,因此设想:在主窗体关闭的时候,设置导航窗体的关闭标志为允许关闭,这样主窗体就不会因为子窗体的关闭被取消而取消了。但实验结果是不可行:所有子窗体的CLOSING事件都在主窗体的CLOSING事件之前触发,当在主窗体进行可关闭标志的设置的时候,对主窗体已经是无效的了。感觉很沮丧,总不能让用户执行两次关闭操作吧?使用Application.Exit来处理,也感觉不理想,便采用对导航窗体的可关闭标志进行置位后再次调用关闭函数来实现。
经过跟踪处理,发现,主窗体和所有的子窗体共用同一个CLOSING事件参数(System.ComponentModel.CancelEventArgs),因此导致只要任何一个窗体(包括主窗体和子窗体)取消的关闭事件,那么整个关闭操作就失败。基于这个原因,在主窗体的CLOSING事件里面,设置CANCEL=false轻松解决问题(因为主窗体的CLOSING事件永远都是最后一个被触发)。
以上解决方案又会带来一个漏洞:除了导航窗口外,如果有其它窗体确实需要取消关闭操作将会导致失败。对于这个问题,可以采用以下两个方案处理:
1——
在主窗体的CLOSING处理中,强制CANCEL标记复位,然后执行每个子窗体是否确实允许关闭的检测再来设置CANCEL标志。
这个方案不好的地方是主窗体必须知道子窗体是否定义了指定的函数,导致主窗体对子窗体的实现类型存在依赖

2——
基于方案1中存在的问题,考虑:各子窗体是否真的可以被关闭的标志置于Tag属性中,这样,子窗体与主窗体之间只要达成这个协议就可以很好的解决这个问题。
相关代码:
子窗体Closing事件代码:

None.gifif (CanClose)
ExpandedBlockStart.gifContractedBlock.gif
dot.gif{
InBlock.gif    
if(Tag == null)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif        Tag 
= true;
ExpandedSubBlockEnd.gif    }

InBlock.gif    
else
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif        ArrayList arrTmp  
= new ArrayList();
InBlock.gif        arrTmp.Add(
true);
InBlock.gif        arrTmp.Add(Tag);
InBlock.gif        Tag 
= arrTmp;
ExpandedSubBlockEnd.gif    }

ExpandedBlockEnd.gif}

None.gif


主窗体Closing事件处理代码:

None.gife.Cancel = false;
None.gif
foreach(Form frmTmp in this.MDIChildren)
ExpandedBlockStart.gifContractedBlock.gif
dot.gif{
InBlock.gif    
if (frmTmp.Tag == null)
InBlock.gif        
continue;
InBlock.gif    
if(frmTmp.Tag is bool)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif        
if ((bool)frmTmp.Tag))
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif            e.Cancel 
= true;
InBlock.gif            
break;
ExpandedSubBlockEnd.gif        }

InBlock.gif        frmTmp.Tag 
= null;
ExpandedSubBlockEnd.gif    }

InBlock.gif    
if(frmTmp.Tag is ArrayList)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif        ArrayList arrTmp 
= frmTmp.Tag as ArrayList;
InBlock.gif        
if(arrTmp.Count != 2 || !(arrTmp[0is bool))
InBlock.gif            
continue;
InBlock.gif        frmTmp.Tag 
= arrTmp[1];
InBlock.gif        
if((bool)arrTmp(0))
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif            e.Cancel 
= true;
InBlock.gif            
break;
ExpandedSubBlockEnd.gif        }

ExpandedSubBlockEnd.gif    }

ExpandedBlockEnd.gif}

None.gif
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值