(转)MDI窗体的工具栏合并(ToolStrip Merging)的功能实现 .

原文地址:http://blog.csdn.net/asciil/article/details/2672120

这个模型的大概构成是这样的:
1个MDI主窗体,1~n个MDI子窗体;
主窗体上一般会有1个主菜单栏,1~n个主工具栏;
子窗体设计时上可能有0~1个主菜单栏,0~1个主工具栏,运行时子窗体的菜单栏和工具栏都显示在主窗体上,与主窗体的菜单栏和工具栏合并。(实际上也许会有多个工具栏,但先简化为一个工具栏的模型,实现了一个工具栏的合并后,要实现多个工具栏合并也并非难事)。
对于菜单栏的合并,无论是传统的Windows API编程或者Window Forms,都已经完美地解决了,不再赘述。
工具栏合并的过程最好在子窗体的激活和关闭处理处理,一个容易想到的方法是写在子窗体的事件代码中,要么需在每一个子窗体的代码文件写上类似处理的代码,要么让所有子窗体继承自一个写好相关事件处理代码的窗体。前者显然不可取,后者在单根继承的时代,剥夺了子窗体从其它窗体继承的可能,也不是一个好办法。同时还有一个弊端是子窗体要访问主窗体的内容,耦合度太高。
最佳方案也自然而然地出来了,即在主窗体的MdiChildActivate事件中处理(不能望文生义以为仅仅是子窗体的激活会触发此事件,事实上子窗体的关闭也会触发此事件)。这个时候应该把主窗体从上一个子窗体合并来的工具栏内容清除掉(如果有的话),并将新激活的子窗体上的待合并的工具栏内容合并到主窗体的工具栏上(如果新激活的子窗体上无待合并的工具栏内容或者MdiChildActivate是关闭最后一个子窗体触发的时候则无此步)。为简化代码,假设子窗体的工具栏总是合并到主窗体工具栏的最后位置。
这时候,主窗体必须知道子窗体的工具栏的构成,完全解耦是不太可能的。一种办法是将子窗体的工具栏控件的作用范围设置为public, 我采取了另外一个办法,就是定义了一个接口,让需合并工具栏的子窗口实现这个接口即可。
接口代码如下:

using  System;
using  System.Collections.Generic;
using  System.Text;
using  System.Windows.Forms;

namespace  Osmanthus.WinForms
{
    
interface  IToolStripMergableForm
    {

        ToolStrip MergedToolStrip
        {
            
get ;
        }
    }
}

让子窗体实现这个接口,所要写的代码也简单,只有一句,即:

public  ToolStrip MergedToolStrip
{
    
get  {  return   this .toolStrip1; }
}

剩下的内容似乎理想当然地是写主窗体的MdiChildActivate事件处理代码,不过我想增加重用性,就把它写成了一个组件,以后只要把该组件拖放在MDI主窗体,并设置此组件MainToolStrip属性为主窗体的工具栏(其实也可以写成自动获取默认属性值,即主窗体上的最后一个ToolStrip控件),不需对再对主窗体写一行代码。这个部件的代码内容如下(其中为实现组件对Form的存取参照了Chris Sells的大作,我不知道是否有其它的更好办法):

using  System;
using  System.Collections.Generic;
using  System.Text;
using  System.Windows.Forms;
using  System.ComponentModel;
using  System.ComponentModel.Design;

namespace  Osmanthus.WinForms
{
    
class  SdiToolStripMerger : Component, ISupportInitialize
    {
        
private  Form hostingForm;
        
private  ToolStrip mainToolStrip;
        
private  IToolStripMergableForm currentMdiChild  =   null ;

        
private  List < ToolStripItem >  toolStripItemList  =   new  List < ToolStripItem > ();

        
public  ToolStrip MainToolStrip
        {
            
get  {  return  mainToolStrip; }
            
set  { mainToolStrip  =  value; }
        }

        [BrowsableAttribute(
false )]
        
public  Form HostingForm
        {
            
get
            {
                
if  ((hostingForm  ==   null &&   this .DesignMode)
                {
                    IDesignerHost designer 
=   this .GetService( typeof (IDesignerHost))  as  IDesignerHost;
                    
if  (designer  !=   null )
                    {
                        hostingForm 
=  designer.RootComponent  as  Form;
                    }
                }
                
return  hostingForm;
            }
            
set
            {
                hostingForm 
=  value; //这里需要改进一下, 应该一旦设置了host form,就不能再修改.
            }
        }

        
#region  ISupportInitialize Members

        
public   void  BeginInit()
        {
        }

        
public   void  EndInit()
        {
            
if  (( ! DesignMode)  &&  (hostingForm  !=   null ))
            {
                hostingForm.MdiChildActivate 
+=   new  EventHandler(UpdateToolStrip);
            };
        }
        
#endregion

        
void  UpdateToolStrip( object  sender, EventArgs e)
        {
             //清除从上个激活子窗体合并来的工具栏内容
            if  (currentMdiChild  !=   null )
            {
                
foreach  (ToolStripItem toolItem  in  toolStripItemList)
                {
                    currentMdiChild.MergedToolStrip.Items.Add(toolItem);
                }
                toolStripItemList.Clear();
            }

            IToolStripMergableForm form 
=  (IToolStripMergableForm)hostingForm.ActiveMdiChild;
            
//如当前激活的子窗体实现了 IToolStripMergableForm接口,则合并其工具栏内容
            if  (form  !=   null )
            {

                currentMdiChild 
=  form;
                
foreach  (ToolStripItem toolItem  in  currentMdiChild.MergedToolStrip.Items)
                {
                    toolStripItemList.Add(toolItem);
                }
                
foreach  (ToolStripItem toolItem  in  toolStripItemList)
                {
                    mainToolStrip.Items.Add(toolItem);
                }
            }
            
else
            {
                currentMdiChild 
=   null ;
                toolStripItemList.Clear();
            }
        }
    }
}

转载于:https://www.cnblogs.com/liangyuhuidespace/archive/2013/03/12/2955492.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值