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

转载 2008年07月18日 15:04:00
这个模型的大概构成是这样的:
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();
            }
        }
    }
}

2006/12/25 附记:今天早上发现了.net Framework 2.0新增加的
一个类ToolStripManager,使用此类可以简化上述代码。

相关文章推荐

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

在MDI窗体定义        public interface ICipherInputForm        {            ToolStrip CipherToolStrip     ...

Delphi7高级应用开发随书源码

  • 2003年04月30日 00:00
  • 676KB
  • 下载

Linux/POSIX Socket编程 笔记1及反思

昨天深夜醒来,一时难以入睡,从书架上找来买了许久的《Internetworking with TCP/IP Volumn 3》,翻看起来。虽然目前只看了前6章,但感觉的确对Socket理解不够。可以说...
  • Yasker
  • Yasker
  • 2006年04月12日 18:45
  • 4143

MailBee.NET Objects接收电子邮件(POP3)教程一:接收简单的电子邮件

MailBee.NET Objects是一款为创建、发送、接收以及处理电子邮件而设计的健壮、功能丰富的.NET控件。几行代码便可为应用程序添加E-Mail支持,简单高效。具备“必需”以及独特的功能,这...

WPF实现类似MDI子窗体的功能

在我的上一篇博客中 http://blog.csdn.net/soft2buy/article/details/5479946,提到了用UserControl做为WPF的“MDI子窗体”。今天偶然发现...
  • soft2buy
  • soft2buy
  • 2013年11月12日 17:06
  • 10448

C# Winform实现MDI程序子窗体与主窗体菜单合并

这里只讲在MDI中,子菜单与父菜单合并的简单方法。   首先在MDI的主窗体中: 1.  在主窗体构造函数中,添加如下代码,注意要在InitializeComponent调用之后,如:   ...
  • zjg528
  • zjg528
  • 2013年04月30日 19:57
  • 642

使用QT实现MDI风格的主窗体

转载至:http://blog.csdn.net/henreash/article/details/7618414 QT提供了MDIArea控件可以很方便的实现标准的MDI窗体,但用起来并不方便.感...
  • AUdiQt
  • AUdiQt
  • 2012年07月22日 17:39
  • 732

使用QT实现MDI风格的主窗体

QT提供了MDIArea控件可以很方便的实现标准的MDI窗体,但用起来并不方便.感觉像360浏览器一样通过页签来切换子窗体的方式比较好.点击菜单项或工具栏上的按钮创建新的页签,并生成一个子窗体.点击页...
  • henreash
  • henreash
  • 2012年05月31日 09:21
  • 13039

使用Qt实现MDI风格的主窗体

文章来源:http://hi.baidu.com/wuyunju/item/3d20164c99a276f6dc0f6c52 QT提供了MDIArea控件可以很方便的实现标准的MDI窗体,但用起来并不...

WPF MDI子窗体实现

  • 2014年04月18日 14:05
  • 303KB
  • 下载
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:MDI窗体的工具栏合并(ToolStrip Merging)的功能实现
举报原因:
原因补充:

(最多只允许输入30个字)