在Winform系统开发中,为了对系统的工具栏/菜单进行动态的控制,我们对系统的工具栏/菜单进行动态配置,这样可以把系统的功能弹性发挥到极致。通过动态工具栏/菜单的配置方式,我们可以很容易的为系统新增所需的功能,通过权限分配的方式,可以更有效的管理系统的菜单分配到不同的角色用户,也就是插件化的处理方式。

1、动态菜单的控制

我们一般的应用系统里面,由于系统是面向不同类型的用户,我们所看到的菜单会越来越多,多一点的甚至上百个,但是我们实际工作接触的菜单可能就是那么几个,那么对于这种庞大的菜单体系,寻找起来非常不便。因此对菜单的个性化配置就显得尤为重要。

但在我们开发的时候,为了方便调试和测试基础功能,有时候有需要直接在Ribbon工具栏或者菜单中固定一些功能的入口,以便快速开发某些常见功能,那么我们可以在系统中增加一个变量来控制动态展示还是采用静态工具栏/菜单的方式。

因此我们在主窗体中使用菜单/工具栏,分为了预设的静态模式(方便测试)和动态模式(实际应用)

以上图示是预设的一些基础入口,我们可以先具体测试某些功能,这样不会打断实际的开发工作,而在系统部署给客户的时候,采用动态模式构建的工具栏/菜单,用户在登录的时候,首先清空预设菜单,在加载拥有的菜单/工具栏,这样就不会相互影响。

我们在程序的主窗口,增加一个变量来控制是否动态即可,默认为false,也就是静态控件模式,开发完成后,部署的时候,把它改为True即可,如下代码。

    /// <summary>
/// 程序主界面
/// </summary>
public partial class MainForm : RibbonForm
{
/// <summary>
/// 是否标记为动态生成顶部按钮栏,从数据库读取动态菜单信息。
/// </summary>
private bool useRemoteMenu = false;

因此我们可以根据这个开关变量来处理菜单的加载处理,如下代码函数所示是处理工具栏、菜单的加载逻辑。

        /// <summary>
/// 初始化左侧功能菜单树列表
/// </summary>
private void InitToolbar()
{
//如果标记为动态生成,那么预设的菜单将清空
if (useRemoteMenu)
{
//初始化Ribbon控制类
if (ribbonHelper == null)
{
ribbonHelper = new RibbonPageHelper(this, ref this.ribbonControl);
}
ribbonHelper.ClearAllPages();//预设的菜单清空
ribbonHelper.AddPages();//动态创建界面菜单对象
} //重新加入应用程序菜单,否则访问出错
this.applicationMenu1.ItemLinks.Clear();
this.applicationMenu1.AddItems(new DevExpress.XtraBars.BarItem[] { this.menu_QuitSystem, this.menu_Relogin }); //根据权限屏蔽菜单对象
InitAuthorizedUI();
if (this.ribbonControl.Pages.Count > 0)
{
ribbonControl.SelectedPage = ribbonControl.Pages[0];
}
}

其中 RibbonPageHelper 辅助类就是用来动态构建Ribbon工具栏的,如果是静态,则默认采用设计时刻的工具按钮即可。

其中ClearAllPages就是清空按钮。

        /// <summary>
/// 清空所有按钮
/// </summary>
public void ClearAllPages()
{
if(this.control != null && this.control.Pages != null)
{
this.control.Pages.Clear();
}
}

而动态构建的工具栏按钮,就是根据用户的角色身份所控制的按钮处理的。

而构建菜单/工具栏按钮的时候,我们主要就是创建工具栏按钮的名称、图标、以及响应的事件即可,如下代码所示。

        /// <summary>
/// 动态创建Ribbon的按钮项目
/// </summary>
/// <param name="menuInfo">菜单信息</param>
/// <returns></returns>
private BarButtonItem CreateButonItem(MenuNodeInfo menuInfo)
{
//添加功能按钮(三级菜单)
var button = new BarButtonItem();
button.PaintStyle = BarItemPaintStyle.CaptionGlyph;
button.LargeGlyph = LoadIcon(menuInfo);
button.Glyph = LoadIcon(menuInfo); button.Name = menuInfo.Id;
button.Caption = menuInfo.Name;
button.Tag = menuInfo.WinformType;
button.ItemClick += (sender, e) =>
{
if (button.Tag != null && !string.IsNullOrEmpty(button.Tag.ToString()))
{
LoadPlugInForm(string.Concat(button.Tag));
}
else
{
MessageDxUtil.ShowTips(button.Caption);
}
};
return button;
}

其中 LoadPlugInForm 函数就是我们加载菜单处理的逻辑,一般就是根据配置信息动态构建出一个窗体即可。

2、动态菜单信息的解析

因此我们需要了解动态菜单的项目中,具体的配置信息是什么,才知道如何具体使用反射的方法,来构建出一个窗体的实例显示。

 

如上图所示的内容,前面部分为窗体类的全局名称,而后面是窗体所在的程序集名称,这样我们根据程序集的名称,菜单或者按钮的单击事件中加载窗体类显示即可。我们一般以逗号分开两个部分,如果是当前程序集,也可以忽略,有系统自动解析加上即可。

string[] itemArray = typeName.Split(new char[]{',',';'});

然后根据内容解析窗体类和具体的程序集路径即可。

 string type = itemArray[0].Trim();
string filePath = (itemArray.Length > 1) ? itemArray[1].Trim() : Assembly.GetExecutingAssembly().GetName().Name;//必须是相对路径

最后稍微处理下,通过具体路径构建处理即可。

//从程序集中加载窗体类,设置为多文档的模式
string dllFullPath = Path.Combine(Application.StartupPath, filePath);
var tempAssembly = Assembly.LoadFrom(dllFullPath);
if (tempAssembly != null)
{
Type objType = tempAssembly.GetType(type);
if (objType != null)
{
LoadMdiForm(this.mainForm, objType, isShowDialog);
}
}

多窗体的显示,先判断是否存在,如果不存在则创建,存在则激活即可。

if (isShowDialog)
{
tableForm.ShowDialog();
}
else
{
tableForm.MdiParent = mainDialog;
tableForm.Show();
}
tableForm.BringToFront();
tableForm.Activate();

这样就是动态工具栏、菜单的处理逻辑了。

在WInform开发中实现工具栏/菜单的动态呈现的更多相关文章

  1. 在Winform开发中使用日程控件XtraScheduler(2)--深入理解数据的存储

    在上篇随笔<在Winform开发中使用日程控件XtraScheduler>中介绍了DevExpress的XtraScheduler日程控件的各种使用知识点,对于我们来说,日程控件不陌生,如 ...

  2. Winform开发中如何将数据库字段绑定到ComboBox控件

    最近开始自己动手写一个财务分析软件,由于自己也是刚学.Net不久,所以自己写的的时候遇到了很多问题,希望通过博客把一些印象深刻的问题记录下来. Winform开发中如何将数据库字段绑定到ComboBo ...

  3. 在Winform开发中使用Grid++报表

    之前一直使用各种报表工具,如RDLC.DevExpress套件的XtraReport报表,在之前一些随笔也有介绍,最近接触锐浪的Grid++报表,做了一些测试例子和辅助类来处理报表内容,觉得还是很不错 ...

  4. 在Winform开发中使用FastReport创建报表

    FastReport.Net是一款适用于Windows Forms, ASP.NET和MVC框架的功能齐全的报表分析解决方案.可用在Microsoft Visual Studio 2005到2015, ...

  5. WinForm开发中通用附件管理控件设计开发参考

    1.引言 在WinForm开发中,文件附件的管理几乎在任何一个应用上都会存在,是一个非常通用集中的公共模块.我们日常记录会伴随着有图片.文档等附件形式来展现,如果为每个业务对象都做一个附件管理,或者每 ...

  6. 在Winform开发中,我们使用的几种下拉列表展示字典数据的方式

    在Winform开发中中,我们为了方便客户选择,往往使用系统的字典数据选择,毕竟选择总比输入来的快捷.统一,一般我们都会简单封装一下,以便方便对控件的字典值进行展示处理,本篇随笔介绍DevExpres ...

  7. 在 Web 开发中,img 标签用来呈现图片,而且一般来说,浏览器是会对这些图片进行缓存的。

    在 Web 开发中,img 标签用来呈现图片,而且一般来说,浏览器是会对这些图片进行缓存的. 比如访问百度,我们可以发现,图片.脚本这种都是从缓存(内存缓存/磁盘缓存)中加载的,而不是再去访问一次百度 ...

  8. WinForm开发中针对TreeView控件改变当前选择节点的字体与颜色

    本文转载:http://www.cnblogs.com/umplatform/archive/2012/08/29/2660240.html 在B/S开发中,对TreeView控件要改变当前选中节点的 ...

  9. Winform开发中的困境及解决方案

    在我们开发各种应用的时候,都会碰到很多不同的问题,这些问题涉及架构.模块组合.界面处理.共同部分抽象等方面,我们这里以Winform开发为例,从系统模块化.界面组件选择.业务模块场景划分.界面基类和辅 ...

  10. 在Winform开发中,使用Async-Awati异步任务处理代替BackgroundWorker

    在Winform开发中有时候我们为了不影响主UI线程的处理,以前我们使用后台线程BackgroundWorker来处理一些任务操作,不过随着异步处理提供的便利性,我们可以使用Async-Awati异步 ...

随机推荐

  1. Python从0到1丨详解图像锐化的Sobel、Laplacian算子

    本文分享自华为云社区<[Python从零到壹] 五十八.图像增强及运算篇之图像锐化Sobel.Laplacian算子实现边缘检测>,作者: eastmount . 一.Sobel算子 So ...

  2. 探索计算机的I/O控制方式:了解DMA控制器的作用与优势

    I/O控制方式 在前面我们已经了解到,每个设备都配备了一个设备控制器.当CPU向设备控制器发送命令并将其存储在寄存器中时,设备控制器会执行相应的操作.然而,尽管设备控制器会更新状态寄存器的状态,但是如 ...

  3. 小知识:vi 查找如何不区分大小写

    在使用vi查找数据库的truncate记录日志时,发现对应语句夹杂了大小写,不够规范: 而vi默认查找是区分大小写的,如何不区分大小写查找指定内容呢? 有两种方式: (1)在查找指令后面额外加上\c标 ...

  4. stm32开发笔记

    STM32F103C8T6单片机简介 标准库与HAL库区别 寄存器 寄存器众多,需要经常翻阅芯片手册,费时费力: 更大灵活性,可以随心所欲达到自己的目的: 深入理解单片机的运行原理,知其然更知其所以然 ...

  5. 其它——DevOps简介

    文章目录 DevOps简介 DevOps的概念 历史变革 好处是什么? 为什么DevOps会兴起? 实现DevOps需要什么? DevOps的采用现状 DevOps简介 DevOps 是一个完整的面向 ...

  6. PKCS#11:密码设备与应用程序的密码学接口

    密码学在信息安全中扮演着至关重要的角色.为了保护敏感信息.数字身份和网络通信的安全性,密码设备(如硬件安全模块HSM)与应用程序之间的安全通信和互操作性变得至关重要.PKCS#11(Public-Ke ...

  7. 手撕Vuex-Vuex实现原理分析

    本章节主要围绕着手撕 Vuex,那么在手撕之前,先来回顾一下 Vuex 的基本使用. 创建一个 Vuex 项目,我这里采用 vue-cli 创建一个项目,然后安装 Vuex. vue create v ...

  8. P1144 最短路计数 题解

    Problem 考察算法:拓扑排序 + \(DP\) + \(Dijkstra\). 题目简述 给出一个无向无权图,问从顶点 \(1\) 开始,到其他每个点的最短路有几条. 思路 先求出 \(1\) ...

  9. k8s集群证书过期,重新生成证书

    Kubernetes集群证书过期后,使用kubeadm重新颁发证书 默认情况下部署kubernetes集群的证书一年内便过期,如果不及时升级证书导致证书过期,Kubernetes控制节点便会不可用,所 ...

  10. 解决 IAR中 Warning[Pa082] 的警告问题

    这个警告不属于严重问题 在 IAR (for STM8)的编译中,经常有如下的警告: Warning[Pa082]: undefined behavior: the order of volatile ...