ERP系统的单据具备标准的功能,这里的单据可翻译为Bill,Document,Entry,具备相似的工具条操作界面。通过设计可复用的基类,子类只需要继承基类窗体即可完成单据功能的程序设计。先看标准的销售合同单据界面:

本篇通过销售合同单据功能,依次讲解编程要点,供参考。

1 新增 Insert

窗体有二种状态,一种是编辑状态,别一种是数据浏览状态,区别在于编辑状态的窗体数据被修改(dirty),在窗体关闭时需要保存数据。点击工具条的新增(Insert)按钮,窗体进入编辑状态。新增状态需要对窗体所编辑的单据设置默认值。一般我们在实体映射文件中设置默认值,参考下面的例子代码:

public partial class SalesContractEntity
{protected override void OnInitialized(){base.OnInitialized();// Assign default value for new entityif (Fields.State == EntityState.New){#region DefaultValue                // __LLBLGENPRO_USER_CODE_REGION_START DefaultValuethis.Fields[(int) SalesContractFieldIndex.Closed].CurrentValue = false;// __LLBLGENPRO_USER_CODE_REGION_END#endregion}}
}
 
 

也可以考虑在窗体中做默认值设定。当遇到这样一种场景,两个功能对应同一个实体类型,则需要在界面中根据需要初始化值,参考下面的程序片段。

protected override EntityBase2 Add()
{base.Add();this._inventoryMovement = new InventoryMovementEntity();this._inventoryMovement.TranType = GetStringValue(InventoryTransactionType.Movement);return this._inventoryMovement;
}
 
 

2 保存 Save

窗体基类检测到界面中的控件值被修改过,窗体状态变为编辑状态,点击保存按钮执行保存方法。保存方法的主要内容是将数据源控件(BindingSource)所绑定的控件值更新到它映射的实体中,再调用窗体保存方法保存实体。

protected override EntityBase2 Save(EntityBase2 entityToSave, EntityCollection entitiesToDelete)
{SalesContractEntity SalesContractEntity = (SalesContractEntity)entityToSave;this._salesContractEntity = this._salesContractEntityManager.SaveSalesContract( SalesContractEntity, entitiesToDelete, SeriesCode);return this._salesContractEntity;
}
 

entityToSave是数据源控件绑定的实体类型,在保存完成后,这个值再次绑定到数据源控件中。

3  删除 Delete

窗体只有在浏览状态时,才可以点击删除按钮,删除按钮的内容是获取窗体数据源控件绑定的实体,调用窗体删除方法。

protected override void Delete(EntityBase2 entityToDelete)
{base.Delete(entityToDelete);SalesContractEntity SalesContractEntity = (SalesContractEntity)entityToDelete;this._salesContractEntityManager.DeleteSalesContract(SalesContractEntity);
}

对实体的任何操作,都会跑实体验证类型,比如在保存时,需要验证主键值是否已经保存过,参考下面的代码。

public override void ValidateEntityBeforeSave(IEntityCore involvedEntity)
{base.ValidateEntityBeforeSave(involvedEntity);SalesContractEntity salesContract = (SalesContractEntity)involvedEntity;
    if (string.IsNullOrEmpty(salesContract.ContractNo))throw new EntityValidationException("Contract No. is required");if (string.IsNullOrEmpty(salesContract.CustomerNo))throw new EntityValidationException("Customer No. is required");if (salesContract.IsNew){ISalesContractManager salesContractManager = CreateProxyInstance<ISalesContractManager>();if (salesContractManager.IsSalesContractExist(salesContract.ContractNo))throw new RecordDuplicatedException(salesContract.ContractNo, "Cotract No. is already used");}
}
 
 

4 复制 Clone

窗体支持两种复制方法,复制当前加载的值,复制其它对象的值。对象值复制完成后,需用重置新的对象的主键值,让它为空或是为默认值,供用户修改。复制其它对象的值需要弹出对象选择窗体。这两种复制都需要注意复制完后,重置对象初始化默认值。因为复制时采用的是深拷贝,没有跑对象初始化值。

protected override object Clone(Dictionary<string, string> refNo){base.Clone(refNo);string receiptRefNo;refNo.TryGetValue("ContractNo", out receiptRefNo);if (string.IsNullOrEmpty(receiptRefNo)){using (ILookupForm lookup = GetLookupForm("SalesContractLookup")){if (!AllowViewAllTransaction)lookup.PredicateBucket = new RelationPredicateBucket(SalesContractFields.CreatedBy == Shared.CurrentUser.UserId);lookup.SetCurrentValue(CurrentRefNo);if (lookup.ShowDialog() != DialogResult.OK) 
                   return null;receiptRefNo = lookup.GetFirstSelectionValue();}}if (!string.IsNullOrEmpty(receiptRefNo)){this._salesContractEntity = this._salesContractEntityManager.CloneSalesContract(receiptRefNo);return this._salesContractEntity;}return null;
}
 

注意到这些方法全部是以override重写的方式出现,被基类调用。每个方法都运行在后台线程控件BackgroundWorker线程中,所以都不能操作界面控件。

5  记录浏览 Record Navigator

主要用于工具条的前四个按钮,分别对应第一笔数据,前一笔数据,下一笔数据,最后一笔数据。工具条浏览需要设定窗体的NavigateBindingSource属性,传入一个空白的BindingSource控件或是一个装载页面所有数据的BindingSource控件,工具条浏览方法重写参考下面的程序片段。

protected override void InitNavigator(InitNavigatorArgs args)
{base.InitNavigator(args);args.SortExpression.Add(SalesContractFields.ContractNo | SortOperator.Ascending);args.PredicateBucket.PredicateExpression.Add(SalesContractFields.Closed == false);
}

6 记录过帐 Record Post

主要用于业务单据过帐逻辑,基类的方法为空方法,不同的业务单据有不同的逻辑定义,没有可复用的代码。

protected override void Post(EntityBase2 entityToPost)
{base.Post(entityToPost);SalesContractEntity resignEntity = entityToPost as SalesContractEntity;_salesContractEntityManager.PostSalesContract(resignEntity);
}

这里的过帐可以理解为确认,批核,不可更改的意思。在有些系统中叫送审,审核。

7  打印 Print

一般在设计视图绑定当前窗体对应的水晶报表文件以及要传入的参数。也可以通过重写打印方法传入传数值。

protected override void Print(ref Dictionary<string, SelectionFormula> selectionFormulas, ref List<FormulaField> formulaFields, ref List<ParameterField> parameterFields)
{base.Print(ref selectionFormulas, ref formulaFields, ref parameterFields);
}
 

重写方法常用于动态指定报表文件,或是动态的参数值。不推荐在代码中这样写,这样做导致每次都需要重新编译和分发程序,推荐在水晶报表中做公式,在分发时只需要拷贝文件即可。

转载于:https://www.cnblogs.com/JamesLi2015/p/4738267.html

解析大型.NET ERP系统 单据标准(新增,修改,删除,复制,打印)功能程序设计...相关推荐

  1. 解析大型.NET ERP系统:十三种界面设计模式

    成熟的ERP系统的界面应该都是从模板中拷贝出来的,各类功能的界面有规律可遵循.软件界面设计模式化或是艺术性的创作,我认可前者,模式化的界面客户容易举一反三,降低学习门槛.除了一些小部分的功能界面设计特 ...

  2. 解析大型.NET ERP系统架构设计 Framework+ Application 设计模式

    我对大型系统的理解,从数量上面来讲,源代码超过百万行以上,系统有超过300个以上的功能,从质量上来讲系统应该具备良好的可扩展性和可维护性,系统中的功能紧密关联.除去业务上的复杂性,如何设计这样的一个协 ...

  3. 解析大型.NET ERP系统 20条数据库设计规范

    数据库设计规范是个技术含量相对低的话题,只需要对标准和规范的坚持即可做到.当系统越来越庞大,严格控制数据库的设计人员,并且有一份规范书供执行参考.在程序框架中,也有一份强制性的约定,当不遵守规范时报错 ...

  4. 解析大型.NET ERP系统 设计异常处理模块

    异常处理模块是大型系统必备的一个组件,精心设计的异常处理模块可提高系统的健壮性.下面从我理解的角度,谈谈异常处理的方方面面.我的设计仅仅限定于Windows Forms,供参考. 1 定义异常类型 . ...

  5. 解析大型.NET ERP系统 高质量.NET代码设计模式

    1 缓存 Cache 系统中大量的用到缓存设计模式,对系统登入之后不变的数据进行缓存,不从数据库中直接读取.耗费一些内存,相比从SQL Server中再次读取数据要划算得多.缓存的基本设计模式参考下面 ...

  6. 解析大型.NET ERP系统 多国语言实现

    实现多国语言有许多种实现方案,无外乎是一种字符串替换技术,将界面控件的文本标签替换成相应语言的文字..NET Windows Forms实现多国语言的方法有以下几种: 1 .NET的方案,使用资源文件 ...

  7. 解析大型.NET ERP系统 设计通用Microsoft Excel导入功能

    做企业管理软件很难避免与Microsoft Excel打交道,常常是软件做好了,客户要求说再做一个Excel导入功能.导入Excel数据的功能的难度不大,从Excel列数据栏位的取值,验证值,再导入到 ...

  8. TIPTOP ERP系统用户可视化界面修改账号密码

    TIPTOP ERP系统用户可视化界面修改账号密码 客制一支作业p_pw,用于登录用户自行更改密码 --------------------华--------------------丽-------- ...

  9. delphi百度人脸搜索(支持人脸搜索、人脸库新增修改删除查询)

    下载链接:https://download.csdn.net/download/liushenglin123/12550077 [delphi百度人脸搜索]支持人脸搜索.人脸库新增修改删除查询

  10. sharepoint 2016 学习系列篇(14)-自定义列表应用篇-(3)列表数据的新增,修改,删除操作

    前面添加完列表的字段之后,接下来介绍列表数据的新增,修改,删除操作,不需要开发,就能实现数据的新增,修改,删除等操作. 重新返回到列表的http://192.168.1.73:7003/Lists/U ...

最新文章

  1. oracle 外部表及解决ora-29400,ora-29913错误
  2. 【Android 应用开发】 Android 相关代码规范 更新中 ...
  3. 软考之运筹学计算-车床问题
  4. (chap1 网络基础知识)一些概念
  5. oracle ebs技术开发,Oracle EBS应用架构技术方案.pdf
  6. (二)利用Java WebService调用天气预报实践
  7. Intent 隐示意图
  8. 以太坊源码学习(一)
  9. git从远程仓库gitLab上拉取指定分支到本地仓库
  10. python1e2_Python-1 数据类型
  11. 服务器利用docker搭建CTFd平台、webug靶场,并映射到公网IP
  12. 【持续更新】MARL 算法汇总
  13. python gzip_Python之gzip模块的使用
  14. iPhone添加节假日日历地址
  15. 如何按州跟踪缺席选票
  16. 制作简单刮刮乐View
  17. MATLAB点云处理(十四):圆柱体拟合(RANSAC | MSAC)
  18. matlab整数规划--简单入门
  19. solidworks怎么画螺纹
  20. 有奖答题小程序知识问答pk答题app源码

热门文章

  1. 雷军:程序员如何成功创业?
  2. 春招冷淡,跳槽无望?
  3. Android开发的前景到底怎么样?
  4. 【精选】Java高频面试题278道附答案,通关中大型互联网企业工程师必备
  5. CTF加密题型解析:RSA算法的CTF解法之一
  6. 不用static,巧用对象.方法调用java中的函数
  7. shell 字符串包含关系
  8. 关于Ajax的一揽子工程(2)
  9. iOS 接入微信 支付宝 参数设置
  10. 《JavaScript高级程序设计(第3版)》阅读总结记录第一章之JavaScript简介