如何使用dotnetbar控件来构造多文档界面
在前段时间一篇随笔《利用优秀的.NET界面控件,打造新潮的界面效果》中介绍过Dotnetbar的界面效果,虽然引发不少关于该控件效果的争议,不过话说回来,使用该控件也不失为一个界面的解决方案,本文继续探寻该控件的使用,在QQ搜通天企业版软件中使用该控件做了一次完整的改造,碰到并解决了一些问题,本文主要总结介绍如何利用Dotnetbar控件来实现多文档界面的效果。首先我们先来看看软件的主体界面效果,如下图所示。
本界面主要利用Bar控件来实现多文档的界面效果,每个子窗体皆为一个用户控件,当然也包括容纳各种窗体的容器CtrlMdiBar类,也是一个用户控件,在主界面Ribbon控件的RibbonClientPanel区域放置容器,然后每次打开窗体,就动态创建或者激活一个窗体页,这样就实现了多文档界面的效果了。
bar = new CtrlMdiBar();
bar.DockTabClosed = new DockTabClosedDelegate(OnDockItemClosed);
SetDetailPanel(bar);
bar.DockTabClosed = new DockTabClosedDelegate(OnDockItemClosed);
SetDetailPanel(bar);
private CtrlMdiBar bar;
private Dictionary<string, DockContainerItem> MdiDict = new Dictionary<string, DockContainerItem>();
public void SetDetailPanel(UserControl uc)
{
if (uc == null)
{
throw new ArgumentNullException("uc", @"用户控件uc不能为空");
}
uc.Dock = DockStyle.Fill;
ribbonDetailPanel.Controls.Clear();
ribbonDetailPanel.Controls.Add(uc);
}
public void SetMdiForm(UserControl uc, string itemText)
{
DockContainerItem item = null;
string type = itemText;//uc.GetType().Name;
if (MdiDict.ContainsKey(type))
{
item = MdiDict[type];
}
else
{
PanelDockContainer panel = new PanelDockContainer();
item = new DockContainerItem();
item.Control = panel;
item.Text = itemText;
uc.Dock = DockStyle.Fill;
panel.Controls.Add(uc);
MdiDict.Add(type, item);
bar.bar1.Items.Add(item);
}
bar.bar1.SelectedDockContainerItem = item;
this.Refresh();
}
private void OnDockItemClosed(DockContainerItem item)
{
string type = item.Text;
if (MdiDict.ContainsKey(type))
{
MdiDict.Remove(type);
}
}
private Dictionary<string, DockContainerItem> MdiDict = new Dictionary<string, DockContainerItem>();
public void SetDetailPanel(UserControl uc)
{
if (uc == null)
{
throw new ArgumentNullException("uc", @"用户控件uc不能为空");
}
uc.Dock = DockStyle.Fill;
ribbonDetailPanel.Controls.Clear();
ribbonDetailPanel.Controls.Add(uc);
}
public void SetMdiForm(UserControl uc, string itemText)
{
DockContainerItem item = null;
string type = itemText;//uc.GetType().Name;
if (MdiDict.ContainsKey(type))
{
item = MdiDict[type];
}
else
{
PanelDockContainer panel = new PanelDockContainer();
item = new DockContainerItem();
item.Control = panel;
item.Text = itemText;
uc.Dock = DockStyle.Fill;
panel.Controls.Add(uc);
MdiDict.Add(type, item);
bar.bar1.Items.Add(item);
}
bar.bar1.SelectedDockContainerItem = item;
this.Refresh();
}
private void OnDockItemClosed(DockContainerItem item)
{
string type = item.Text;
if (MdiDict.ContainsKey(type))
{
MdiDict.Remove(type);
}
}
然后每次按钮打开一个窗体页的时候,只需要简单的调用函数即可,如下面几个窗体页的打开操作一样
private void btnMyQunUser_Click(object sender, EventArgs e)
{
SetMdiForm(new CtrlGroupUser(), MDIForm.查询个人群成员.ToString());
}
private void btnMyQQUser_Click(object sender, EventArgs e)
{
SetMdiForm(new CtrlQQContact(), MDIForm.查询个人QQ好友.ToString());
}
private void btnXiaoyou_Click(object sender, EventArgs e)
{
SetMdiForm(new CtrlXiaoyou(), MDIForm.查询QQ校友.ToString());
}
SetMdiForm(new CtrlGroupUser(), MDIForm.查询个人群成员.ToString());
}
private void btnMyQQUser_Click(object sender, EventArgs e)
{
SetMdiForm(new CtrlQQContact(), MDIForm.查询个人QQ好友.ToString());
}
private void btnXiaoyou_Click(object sender, EventArgs e)
{
SetMdiForm(new CtrlXiaoyou(), MDIForm.查询QQ校友.ToString());
}
其中CtrlMDIBar容器,主要是一个用户控件放置一个Dotnetbar的Bar控件,然后在该控件里面调用委托处理控件关闭的事件,主要代码如下所示。
public delegate void DockTabClosedDelegate(DockContainerItem item);
public partial class CtrlMdiBar : UserControl
{
public DockTabClosedDelegate DockTabClosed;
public CtrlMdiBar()
{
InitializeComponent();
}
private void bar1_DockTabClosed(object sender, DevComponents.DotNetBar.DockTabClosingEventArgs e)
{
//MessageExUtil.ShowTips("DockTabClosed");
if (DockTabClosed != null)
{
DockTabClosed(e.DockContainerItem);
}
}
...............
public partial class CtrlMdiBar : UserControl
{
public DockTabClosedDelegate DockTabClosed;
public CtrlMdiBar()
{
InitializeComponent();
}
private void bar1_DockTabClosed(object sender, DevComponents.DotNetBar.DockTabClosingEventArgs e)
{
//MessageExUtil.ShowTips("DockTabClosed");
if (DockTabClosed != null)
{
DockTabClosed(e.DockContainerItem);
}
}
...............
这样,我们后面如果要增加一个窗体页放到容器里面,只需要再定义一个用户控件,并设计好界面等处理方式即可,动态增加一个界面窗体页将非常简单。
专注于代码生成工具、.Net/.NetCore 框架架构及软件开发,以及各种Vue.js的前端技术应用。著有Winform开发框架/混合式开发框架、微信开发框架、Bootstrap开发框架、ABP开发框架、SqlSugar开发框架等框架产品。
转载请注明出处:撰写人:伍华聪 http://www.iqidi.com
转载请注明出处:撰写人:伍华聪 http://www.iqidi.com