这几个月一直忙APP的项目,没来得及更新项目,想想该抽出时间整理一下开发思路,跟大家分享,同时也希望得到宝贵的建议。
先说一下我们的权限管理的的设计思路,首先一个企业信息化管理系统一定会用到权限管理, 那么一个动态的菜单在企业信息化管理系统占有一定的分量。
下面介绍我的一些思路。
由于原声的winform界面美观性不够, 系统采用dotnetbar第三方控件来辅助编程。
首先我们看红色方框部分为我们的动态模块功能。这样我们获得到了以下几个信息, 我们需要记录模块功能, 并且需要父子集关系, 那么对应的我们就需要在数据库中建立一张表来维护我们的动态菜单模块。我们举例命名为BaseModuleTabel, 那么会有一下的内容,Id ParentId Name,并且由于需要跟不同的用户有对应的关系, 我们还要建立用户以及菜单的关系表,来维护用户与菜单的权限关系。
这里就不介绍表的设计。我们先来看看在dotnetbar中如何呈现动态菜单逻辑。
我们先从数据库中获取到所有菜单模块。并通过循环便利动态的加载模块按钮。 并添加对应的模块按钮事件,
/// <summary>
/// 加载菜单
/// </summary>
private void LoadMenu()
{
var parentEntity = _baseModuleBll.Repository().FindEntity("ParentId", ""); var result = _baseModuleBll.GetModuleList().Select(p => new BaseTreeViewMenuModel()
{
Id = p.ModuleId,
ParentId = p.ParentId,
Name = p.FullName,
Class = p.Location,
Namespace = StringHelper.SubFirstChart(p.Location, '.'),
IsForm = p.Target == "iframe"
}).ToList();
RibbonTabItem ribbonTabItemFrist = null;
var rabbonTabItemEntities = result.Where(p => p.ParentId == parentEntity.ModuleId);
foreach (var ribbonTabItemEntity in rabbonTabItemEntities)
{
var ribbonPanel = new RibbonPanel
{
Text = ribbonTabItemEntity.Name,
Dock = DockStyle.Fill
};
var ribbonTabItem = new RibbonTabItem
{
Text = ribbonTabItemEntity.Name,
Panel = ribbonPanel
};
if (ribbonTabItemFrist == null)
ribbonTabItemFrist = ribbonTabItem;
this.ribbonControlMenu.Controls.Add(ribbonPanel);
this.ribbonControlMenu.Items.Add(ribbonTabItem); var ribbonBar = new RibbonBar { Text = ribbonTabItemEntity.Name };
var buttonItemEntities = result.Where(p => p.ParentId == ribbonTabItemEntity.Id);
foreach (var buttonItem in buttonItemEntities.Select(buttonItemEntitiy =>
new ButtonItem("ButtonItem" + buttonItemEntitiy.Name)
{
Text = buttonItemEntitiy.Name,
Tag = buttonItemEntitiy
}))
{
buttonItem.Click += ButtonItem_Click;
ribbonBar.Items.Add(buttonItem);
ribbonPanel.Controls.Add(ribbonBar);
}
}
this.ribbonControlMenu.SelectedRibbonTabItem = ribbonTabItemFrist; } #endregion
当然我们的框架采用了反射机制来实现组件式开发。通常在一个项目里,为了降低耦合度,我一边将模块力度最小话,并将按一定的规则分类, 处理成摸个模块,在我们的系统中并不影响彼此开发的同时, 每个成员都可以独立的去完成自己的小任务, 说句不好听的, 就算是能力很差的新手,开发出来的耦合度极高的代码, 通过这样一种方式,他的影响范围也紧紧在与他的模块内, 并不影响其他模块操作。 仅作为个人的一种方式,当然通过我们的代码生成器,会生成一定的代码规范,这虽然能避免一些, 但起不到决定性的作用。 毕竟每个人的能力也是大不相同的。作为PM我们需要做的就是降低项目整体风险,按照规定的项目周期,制定完善的项目计划,系统模块划分力度越大,项目把控的越高。
下面打开点击模块按钮来,通过反射来实现对各个模块的初始化以及加载。代码效率目前开无太大障碍。当然如果一个模块dll你要弄个几百兆,那我也只能无语了。你的力度相对较小,会跟你反射的时间挂钩的。
/// <summary>
/// 打开模块菜单
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void ButtonItem_Click(object sender, EventArgs e)
{
var isOpened = false;
var buttonItem = (ButtonItem)sender;
var baseTreeViewMenuModel = (BaseTreeViewMenuModel)buttonItem.Tag;
if (baseTreeViewMenuModel == null) return;
if (!baseTreeViewMenuModel.IsForm) return;
if (string.IsNullOrWhiteSpace(baseTreeViewMenuModel.Namespace)) return;
//遍历现有的Tab页面,如果存在,那么设置为选中即可
foreach (var tabitem in
superTabControlContent.Tabs.Cast<SuperTabItem>()
.Where(tabitem => tabitem.Name == "superTabItem" + baseTreeViewMenuModel.Name))
{
superTabControlContent.SelectedTab = tabitem;
isOpened = true;
break;
}
if (isOpened) return;
var dll = Application.StartupPath + "\\" + baseTreeViewMenuModel.Namespace + ".dll";
if (!File.Exists(dll)) return;
var assembly = Assembly.LoadFrom(dll);
var type = assembly.GetType(baseTreeViewMenuModel.Class);
if (type == null) return;
var form = (FormBase)Activator.CreateInstance(type);
form.ModuleId = baseTreeViewMenuModel.Id;
form.FormBorderStyle = FormBorderStyle.None;
form.TopLevel = false;
form.Visible = true;
form.Dock = DockStyle.Fill;
var superTabItem = superTabControlContent.CreateTab(baseTreeViewMenuModel.Name);
superTabItem.Text = baseTreeViewMenuModel.Name;
superTabItem.Name = "superTabItem" + baseTreeViewMenuModel.Name;
superTabItem.AttachedControl.Controls.Add(form);
superTabControlContent.SelectedTab = superTabItem;
} #endregion
通过以上的代码我们就可以实现如下图所示的动态菜单功能 。
分享是美德,要提倡!
Winform快速开发平台系列:
1.winform快速开发平台 -> 让有限的资源创造无限的价值!
3.winform快速开发平台 -> 绑定ComboBox数据控件