从开发的软件《备件仓库管理系统》总结的一些经验

简介:

最近做了一个备件仓库管理软件,虽然只是一个不太复杂的仓库管理业务的软件,附带产出一些相关的报表,而且有之前做过的送水管理系统、酒店管理系统等软件的基础及技术储备,不过做起来发现还是有很多细节及新的东西,在客户不断提出修改意见以及改进建议的同时,逐步吸收优化新的知识,现大概总结一些相关的开发心得,以飨读者。

首先来看看整个软件的主体界面,如下所示,软件的功能主要集备件信息管理、备件入库、备件出库、库存查询、库房管理、业务报表、权限管理、数据字典管理、备件及库存导入等功能于一体。

 
现大致就主体界面介绍一些相关的知识:

1、使用OutLook样式的工具栏,提高界面的美观及易用性,这个在我的早期文章中已经有介绍,该控件集成美观的图标,可能很好提高界面的友好性及可操作性,可以令界面增色不少,详细请查看《WinForm界面开发之“OutLookBar”工具条》 在此不再赘述,只是介绍下并略为带过即可。

 

2、使用“WeifenLuo.WinFormsUI.Docking”控件来设计多文档界面效果,可以方便多个界面进行操作,这个控件也有相关的文章进行介绍使用了,请参考《WinForm界面开发之布局控件"WeifenLuo.WinFormsUI.Docking"的使用》 和《再谈布局控件"WeifenLuo.WinFormsUI.Docking"的使用--如何控制自动停靠窗口的大小》。

 

3、 尽可能使用已经在共享软件中广泛性使用我自己的分页控件,该分页控件集成了数据分页、内容提示、数据打印、数据导出、表头中文转义等很多功能,由于集成性很好,省却很多功夫,专注客户的业务及变化即可,否则一项表头的中文转换就够呛,还不说数据的分页,由于整合性、一致性、稳定性等特点,客户使用感觉比较好。最新的版本整合了优秀的Aspose.Cell控件来进行Excel数据的导出,速度非常快,而且默认表头冻结,非常方便。

 

4、 使用Apose.Cell控件的强大功能,实现自定义模板报表的定制导出。

使用普通的二维表,虽然能满足大多数的情况,不过在一般的业务中,自定义模板的报表根据贴近实际,符合客户的要求,虽然自定义模板的报表,比普通的二维报表复杂一些,不过利用Apose.Cell控件,并在预设模板中预设变量,可以生成很复杂的报表。详细历程可以参考我介绍Apose.Cell控件的使用文章,《使用Aspose.Cell控件实现Excel高难度报表的生成(一)》、《使用Aspose.Cell控件实现Excel高难度报表的生成(二)》。利用Apose.Cell可以生成下面几类自定义模板的报表:

 

  
 其他 设计模板如下所示:

 
实际生成的报表如下所示:

 

5、利用现成独立的数据字典模块代码。由于一般复杂一点的系统,都需要有数据字典的模块,由于我在较早已经在这块已经做了一些开发,因此直接拿过来使用即可,该数据字典模块功能相对比较独立,因此是一个非常使用的模块,数据通过字典排序可以实现合理的排序,支持无限多级字典分类。

 

 实际使用的时候,也是非常方便,首先我们封装一下获取字典项目的方法如下所示:

         /// <summary>

         ///  根据字典类型获取对应的CListItem集合
        
///   </summary>
        
///   <param name="dictTypeName"></param>
        
///   <returns></returns>
         public   static  CListItem[] GetDictByDictType( string  dictTypeName)
        {
            List
< CListItem >  itemList  =   new  List < CListItem > ();
            Dictionary
< string string >  dict  =  BLLFactory < DictData > .Instance.GetDictByDictType(dictTypeName);
            
foreach  ( string  key  in  dict.Keys)
            {
                itemList.Add(
new  CListItem(key, dict[key]));
            }
            
return  itemList.ToArray();
        }

 然后再窗体初始化的时候,添加字典的初始化代码即可,如下所示:

         private   void  InitDictItem()
        {
            
this .txtManufacture.Items.Clear();
            
this .txtManufacture.Items.AddRange(DictItemUtil.GetDictByDictType( " 供货商 " ));

            
this .txtBigType.Items.Clear();
            
this .txtBigType.Items.AddRange(DictItemUtil.GetDictByDictType( " 备件属类 " ));

            
this .txtItemType.Items.Clear();
            
this .txtItemType.Items.AddRange(DictItemUtil.GetDictByDictType( " 备件类别 " ));

            
this .txtUnit.Items.Clear();
            
this .txtUnit.Items.AddRange(DictItemUtil.GetDictByDictType( " 单位 " ));

            
this .txtSource.Items.Clear();
            
this .txtSource.Items.AddRange(DictItemUtil.GetDictByDictType( " 来源 " ));

            
this .txtUsagePos.Items.Clear();
            
this .txtUsagePos.Items.AddRange(DictItemUtil.GetDictByDictType( " 使用位置 " ));
            
this .txtUsagePos.SelectedIndex  =   0 ;

            
this .txtBelongDept.Items.Clear();
            
this .txtBelongDept.Items.AddRange(DictItemUtil.GetDictByDictType( " 部门 " ));

            
this .txtBelongWareHouse.Items.Clear();
            
this .txtBelongWareHouse.Items.AddRange(Portal.gc.ManagedWareHouse.ToArray());
            
this .txtBelongWareHouse.SelectedIndex  =   0 ;
       } 

通过数据字典大类无限级分类以及对字典内容的管理,基本上可以满足绝大多数的需要。

6、使用独立又具整合性的权限管理系统,既相互独立,有相互整合,方便重用,又不需重新开发,非常方便、更提高效率。 由于权限系统精简而又能满足日常绝大多数的需要,不会复杂的难于管理,而且也是基于角色的授权访问机制(RBAC),最重要是非常适合软件的整合使用。

 


用户角色功能维护界面如下:

 

编辑角色对应的权限界面如下:

如果我们在开发的系统中要集成现有的权限系统,操作代码如下所示:

         ///   <summary>
        
///  从数据库中列出相关用户
        
///   </summary>
         private   void  InitLoginName()
        {
            User userBLL 
=   new  User();
            List
< UserInfo >  userList  =  userBLL.GetAll();
            
this .cmbzhanhao.Items.Clear();
            
foreach  (UserInfo info  in  userList)
            {
                
this .cmbzhanhao.Items.Add(info.Name);
            }
        }

登录的时候,只需要把该客户能操作的功能点放到内存列表中,如下所示:

                string  loginName  =   this .cmbzhanhao.Text.Trim();
                User userBLL 
=   new  User();
                
string  identity  =  userBLL.VerifyUser(loginName,  this .tbPass.Text, Guid.NewGuid().ToString());
                
if  ( ! string .IsNullOrEmpty(identity))
                {
                    UserInfo info 
=  userBLL.GetUserByName(loginName);

                    
#region  获取用户的功能列表

                    Function functionBLL 
=   new  Function();
                    List
< FunctionInfo >  list  =  functionBLL.GetFunctionsByUser(info.ID,  " WareMis " );
                    
if  (list  !=   null   &&  list.Count  >   0 )
                    {
                        
foreach  (FunctionInfo functionInfo  in  list)
                        {
                            
if  ( ! Portal.gc.FunctionDict.ContainsKey(functionInfo.ControlID))
                            {
                                Portal.gc.FunctionDict.Add(functionInfo.ControlID, functionInfo);
                            }
                        }
                    }

                    
#endregion
               }

 用户对某个功能点授权判断,只需要判断某功能点是否在已有集合中即可,如下所示:

         /// <summary>

         ///  根据权限屏蔽功能
        
///   </summary>
         private   void  InitAuthorizedUI()
        {
            
this .tool_Report.Enabled  =  Portal.gc.HasFunction( " Report " );
            
this .tool_Dict.Enabled =  Portal.gc.HasFunction( " Dictionary " );
            
this .tool_ItemDetail.Enabled  =  Portal.gc.HasFunction( " ItemDetail " );
            
this .tool_Purchase.Enabled  =  Portal.gc.HasFunction( " Purchase " );
            
this .tool_StockSearch.Enabled  =  Portal.gc.HasFunction( " StockSearch " );
            
this .tool_TakeOut.Enabled  =  Portal.gc.HasFunction( " TakeOut " );

            
this .menu_WareHouse.Enabled  =  Portal.gc.HasFunction( " WareHouse " );
            
this .menu_Dictionary.Enabled  =  Portal.gc.HasFunction( " Dictionary " );
            
this .menu_run_systemLog.Enabled  =  Portal.gc.HasFunction( " LoginLog " );
            
this .menu_Parameters.Enabled  =  Portal.gc.HasFunction( " Parameters " );
            
this .menu_MonthlyStatistic.Enabled  =  Portal.gc.HasFunction( " MonthlyStatistic " );
            
this .menu_AnnualStatistic.Enabled  =  Portal.gc.HasFunction( " AnnualStatistic " );
            
this .menu_ClearAll.Enabled  =  Portal.gc.HasFunction( " ClearAllData " );
            
this .menu_ImportItemDetail.Enabled  =  Portal.gc.HasFunction( " ImportItemDetail " );
        }

因此,文章到这里先小结一下,就是利用现有成熟、稳定、集成性好的控件或者模块,或者利用合适易用的控件,既能事半功倍的完成任务,又能快速响应客户的需求变化 ,还能在界面整体上给客户留下好的印象,一举三得,何乐不为? 欢迎与大家一起探讨Winform开发的点点滴滴或者相互合作交流。 

本文转自博客园伍华聪的博客,原文链接:从开发的软件《备件仓库管理系统》总结的一些经验,如需转载请自行联系原博主。



目录
相关文章
|
SQL 安全 关系型数据库
SQL错误代码1303解析与解决方案:深入理解并应对权限问题
在数据库管理和开发过程中,遇到错误代码是常见的事情,每个错误代码都代表着一种特定的问题
|
XML IDE 前端开发
IDEA忽略node_modules减少内存消耗,提升索引速度
在后端开发中,IDEA 在运行前端代码时,频繁扫描 `node_modules` 文件夹会导致高内存消耗和慢索引速度,甚至可能会导致软件卡死。为了改善这一问题,可以按照以下步骤将 `node_modules` 文件夹设为忽略:通过状态菜单右键排除该文件夹、在设置选项中将其加入忽略列表,并且手动修改项目的 `.iml` 文件以添加排除配置。这些操作可以有效提高IDE的运行性能、减少内存占用并简化项目结构,但需要注意的是,排除后将无法对该文件夹进行索引,操作文件时需谨慎。
1013 4
IDEA忽略node_modules减少内存消耗,提升索引速度
|
12月前
|
缓存 网络协议 网络安全
计算机网络基础知识
本文介绍了计算机网络中的路由协议、HTTP协议、OSI七层模型、VLAN、TCP/UDP协议等内容。路由协议部分涵盖了NAT、OSPF、RIP、IGRP和BGP的基本概念和区别。HTTP协议部分详细解释了请求和响应的结构、常用方法、状态码、重定向和转发的区别、GET和POST的区别、Cookies和Session的区别、HTTP请求访问流程以及HTTP和HTTPS的区别。OSI七层模型部分简述了各层的功能。VLAN部分介绍了VLAN的工作原理和优势。TCP/UDP部分对比了两种协议的特点,重点讲解了TCP的三次握手和四次挥手过程及其作用。
|
存储 C++ 容器
C++入门8——vector的使用
C++入门8——vector的使用
878 0
|
存储 前端开发 JavaScript
基于 GitHub Workflow和 Docker 构建 NextJS
基于 GitHub Workflow和 Docker 构建 NextJS
383 0
|
物联网 Java 数据安全/隐私保护
App Inventor 2 低功耗蓝牙(BLE) 硬件接入、数据通信及IO控制
低功耗蓝牙(BLE)以低功耗、低成本、开发简便逐渐被广泛应用,本文主要介绍一款较为通用、价格低廉的BLE设备从零开始如何利用App Inventor 2开发一款自己专属的手机蓝牙App应用。 本文主要通过一款常见的BLE硬件接入控制,介绍硬件接入App Inventor 2 的通用方法,类似的硬件接入都是大同小异的。
638 1
|
虚拟化 数据安全/隐私保护 网络安全
|
消息中间件 Kafka Go
Golang微服务框架Kratos应用Kafka消息队列
Apache Kafka 是一个分布式数据流处理平台,可以实时发布、订阅、存储和处理数据流。它旨在处理多种来源的数据流,并将它们交付给多个消费者。简而言之,它可以移动大量数据,不仅是从 A 点移到 B 点,而是能从 A 到 Z 的多个点移到任何您想要的位置,并且可以同时进行。
511 0
|
SQL
SAP ABAP——OPEN SQL(五)【GROUPING & SORT】
本文主要介绍SAP ABAP中OPEN SQL的GROUPING和SORT语句
1081 0
SAP ABAP——OPEN SQL(五)【GROUPING & SORT】
|
JavaScript
vue 渲染列表报错Avoid using non-primitive value as key, use string/number value instead. found in
vue 渲染列表报错Avoid using non-primitive value as key, use string/number value instead. found in
246 0

热门文章

最新文章