在我们开发工作流的时候,往往需要设计到具体业务表单信息的编辑,有些是采用动态编辑的,有些则是在开发过程中处理的,各有各的优点,动态编辑的则方便维护各种各样的表单,但是数据的绑定及处理则比较麻烦,而自定义开发的,则数据弹性很大,方便修改调整。本篇随笔基于表单的开发设计过程,介绍在工作流中如何新增一个业务表单,以便快速的实现审批业务的上线处理。

1、业务表单的基类继承

首先我们来了解一下业务表单的对应关系,一般创建一个业务流程处理,都需要有一个具体的创建业务表单的界面,以及一个查看处理表单的界面。

为了方便,我们尽可能减少代码编写,我们需要把大多数的逻辑处理放在基类实现,这样我们在新增一个业务表单的时候就可以减少很多代码编写及维护了。

如对于FrmAddApply类定义如下,我们定义一些抽象接口用于下面的业务表单实现

    /// <summary>
/// 创建申请单的窗体基类
/// </summary>
public partial class FrmAddApply : BaseForm
{
/// <summary>
/// 表单ID
/// </summary>
public string FormID { get; set; } /// <summary>
/// 申请单ID
/// </summary>
public string ApplyId { get; set; } public FrmAddApply()
{
InitializeComponent();
} /// <summary>
/// 显示数据的函数(子类必须实现)
/// </summary>
public virtual void DisplayData() { } /// <summary>
/// 实现控件输入检查的函数(子类必须实现)
/// </summary>
/// <returns></returns>
public virtual bool CheckInput() { return true; } /// <summary>
/// 编辑状态下的数据保存(子类必须实现)
/// </summary>
/// <returns></returns>
public virtual bool SaveUpdated() { return true; } /// <summary>
/// 新增状态下的数据保存(子类必须实现)
/// </summary>
/// <returns></returns>
public virtual bool SaveAddNew() { return true; } ...................

这样我们创建一个新的业务表单,只需要利用代码生成工具,生成所需要的各层框架代码,然后再生成Winform窗体代码,复制部分界面处理代码过来这个业务表单的子类即可。

下面是一个请假申请的业务表单设计,如下所示。

我们看到这个表单可以使用代码生成工具Database2Sharp快速生成后进行一定调整的,而这个编辑表单的界面,我们只需要使用自动生成的部分代码即可。

相关代码只需要复制上面的新增、更新、显示数据的代码即可。

对于查看申请单的基类FrmViewApply类,我们更加简单,我们需要把它的自定义界面控件加载出来即可。

下面是查看申请单的基类,封装了相关的处理逻辑。

    /// <summary>
/// 本窗体是通用的查看申请单界面基类。
/// 为减少开发相关页面的工作量,只需要创建一个新窗体,并继承本窗体,然后在子窗体Form_Load函数里面,初始化对应的申请单显示控件即可。
/// </summary>
public partial class FrmViewApply : BaseDock
{
/// <summary>
/// 申请单ID
/// </summary>
public string ApplyId { get; set; } /// <summary>
/// 申请单自定义控件
/// </summary>
public BaseUserControl ApplyControl { get; set; } /// <summary>
/// 默认构造函数
/// </summary>
public FrmViewApply()
{
InitializeComponent();
} private void FrmViewApply_Load(object sender, EventArgs e)
{
if (!this.DesignMode)
{
InitToolBar();
}
} /// <summary>
/// 初始化申请单控件
/// </summary>
protected virtual void InitApplyControl(BaseUserControl control)
{
if (control != null)
{
this.ApplyControl = control;
this.ApplyControl.Dock = DockStyle.Fill;
this.Controls.Add(control);
}
} /// <summary>
/// 打印申请单控件内容(默认调用窗体打印)
/// </summary>
protected virtual void PrintApplyControl()
{
if(this.ApplyControl != null)
{
PrintFormHelper.Print(this.ApplyControl, false);
}
} /// <summary>
/// 表单另存为
/// </summary>
protected virtual void ApplySaveAs()
{
} /// <summary>
/// 初始化工具栏的按钮和状态
/// </summary>
protected virtual void InitToolBar()
{
................//基类实现,控制什么时候该做什么审批处理,以及一些常见按钮
} .................

查看请假申请单的窗口就是继承这个FrmViewApply即可,如下所示。

    /// <summary>
/// 查看请假申请单的窗体
/// </summary>
public partial class FrmViewLeave : FrmViewApply
{
private LeaveControl control = null; public FrmViewLeave()
{
InitializeComponent();
} private void FrmViewLeave_Load(object sender, EventArgs e)
{
//初始化控件并展示在基类窗体里面
control = new LeaveControl();
control.ApplyId = this.ApplyId;
control.DisplayData(); base.InitApplyControl(control);
}
}

这个就是全部的窗体源码了,主要的内容我们看到是在LeaveControl这个用户控件类里面的了,

而这个控件主要就是上面编辑请假申请单的界面设计,并复制相关的显示数据代码即可。

相关界面代码如下所示。

    /// <summary>
/// 查看请假申请单的内容显示控件
/// </summary>
public partial class LeaveControl : BaseUserControl
{
/// <summary>
/// 申请单ID
/// </summary>
public string ApplyId { get; set; } public LeaveControl()
{
InitializeComponent(); SetReadOnly();
} /// <summary>
/// 设置整个窗体布局为只读并设置只读的背景颜色
/// </summary>
private void SetReadOnly()
{
this.layoutControl1.OptionsView.IsReadOnly = DevExpress.Utils.DefaultBoolean.True;
this.layoutControl1.Appearance.ControlReadOnly.BackColor = Color.SeaShell;
} private void LeaveControl_Load(object sender, EventArgs e)
{
this.applyInfoControl1.ApplyId = this.ApplyId;
this.applyInfoControl1.BindData();
} /// <summary>
/// 初始化数据字典
/// </summary>
private void InitDictItem()
{
//初始化代码
} /// <summary>
/// 数据显示的函数
/// </summary>
public void DisplayData()
{
InitDictItem();//数据字典加载(公用) //由于申请单一般是用申请表单入口,而非业务数据表,因此只能传入ApplyId获取信息
if (!string.IsNullOrEmpty(ApplyId))
{
....................
}

通过上面定义的对应表单的窗体基类,可以减少我们重复编码的需要,我们只需要利用最有效率的生成界面,然后复制代码后调整即可快速生成我们所需要的不同表单界面。

每个表单我们放在一个目录上,这样我们就可以很好管理它们了。

2、业务表单的动态展示处理

上面介绍了业务表单的填写、查看两个不同的窗口,我们在申请单的审批界面里面,统一显示不同的表单,以及创建不同的业务表单界面,这种动态的处理可以实现不同业务表单的创建及显示界面。

如我的审批工作中,表单的显示界面如下所示,查看具体表单后,可以动态展示不同的业务窗口界面。

另外我们在创建业务表单的时候,根据数据库的配置信息,动态展示所有可以展示的创建入口,单击相关的按钮,可以动态调用创建对应的表单界面。

创建流程业务表单的入口如下所示。

在我的审批工作界面,动态创建对应的查看表单窗体代码如下所示。

        /// <summary>
/// 分页控件编辑项操作
/// </summary>
private void winGridViewPager1_OnEditSelected(object sender, EventArgs e)
{
//获取记录ID和表单ID
string ID = this.winGridViewPager1.gridView1.GetFocusedRowCellDisplayText("ID");
string FormId = string.Concat(this.winGridViewPager1.gridView1.GetFocusedRowCellValue("FormId")); if (!string.IsNullOrEmpty(ID) && !string.IsNullOrEmpty(FormId))
{
var formInfo = BLLFactory<BLL.Form>.Instance.FindByID(FormId);
if (formInfo != null && !string.IsNullOrEmpty(formInfo.ApplyWin2))
{
try
{
//根据配置的查看窗体,动态构建查看申请单对象
FrmViewApply dlg = Assembly.GetExecutingAssembly().CreateInstance(formInfo.ApplyWin2) as FrmViewApply;
if (dlg != null)
{
dlg.ApplyId = ID;
dlg.OnDataSaved += new EventHandler(dlg_OnDataSaved); if (DialogResult.OK == dlg.ShowDialog())
{
BindData();
}
}
}
catch (Exception ex)
{
LogHelper.Error(ex);
MessageDxUtil.ShowError(ex.Message);
}
}
}
}

这个代码替代了需要手动创建不同对象的处理

    var dlg = new FrmViewAssignWork();
dlg.ApplyId = ID;
dlg.OnDataSaved += new EventHandler(dlg_OnDataSaved); if (DialogResult.OK == dlg.ShowDialog())
{
BindData();
}

同理,对于创建编辑界面,我们也可以同样的方法动态创建相关的编辑表单界面,如下代码所示。

WInform开发框架之工作流系列文章:

Winform开发框架之简易工作流设计

Winform开发框架中工作流模块的表设计分析

Winform开发框架中工作流模块的业务表单开发

Winform开发框架中工作流模块之审批会签操作

Winform开发框架中工作流模块之审批会签操作(2)

Winform开发框架中工作流模块的业务表单开发的更多相关文章

  1. 参照企业微信审批业务,在Winform开发框架中工作流模块的实现业务审批

    目前微信的企业号已经切换到企业微信里面,这个是一个APP程序,提供了很丰富的企业应用,其中包括了业务审批处理,审批业务包括请假.报销.费用.出差等很多个审批场景,在Winform开发框架中工作流模块这 ...

  2. Winform开发框架中工作流模块的表设计分析

    在较早博客随笔里面写过文章<Winform开发框架之简易工作流设计>之后,很久没有对工作流部分进行详细的介绍了,本篇继续这个主题,详细介绍其中的设计.实现及效果给大家,这个工作流在好几年前 ...

  3. Winform开发框架中工作流模块之申请单草稿处理

    在我们开发工作流模块的时候,有时候填写申请单过程中,暂时不想提交审批,那么可以暂存为草稿,以供下次继续填写或者提交处理,那么这个草稿的功能是比较实用的,否则对于一些填写内容比较多的申请单,每次要重填写 ...

  4. Winform开发框架中工作流模块之审批会签操作(2)

    前面随笔介绍了请假申请单和报销申请单两个不同的业务表单的流程处理,一个是单表信息,一个包含明细的主从表信息,后者包含了条件流程的处理,在流程审批中,一般还有一种流程处理就是会签的操作,会签处理是几个审 ...

  5. Winform开发框架中工作流模块之审批会签操作

    在前面介绍了框架中工作流的几个开发过程,本篇随笔重点介绍一下日常审批环节中的具体处理过程,从开始创建表单,以及各个审批.会签过程的流转过程,希望大家对其中流程的处理有一个大概的印象. 1.请假申请表单 ...

  6. Winform开发框架中工作流模块的动态处理

    在工作流处理表中,首先我们区分流程模板和流程实例两个部分,这个其实就是类似模板和具体文档的概念,我们一份模板可以创建很多个类似的文档,文档样式结构类似的.同理,流程模板实例为流程实例后,就是具体的一个 ...

  7. Gemini.Workflow 双子工作流入门教程五:业务表单开发

    简介: Gemini.Workflow 双子工作流,是一套功能强大,使用简单的工作流,简称双子流,目前配套集成在Aries框架中. 下面介绍本篇教程:业务表单开发. 业务表单开发 业务表单的开发,和在 ...

  8. Java工作流引擎结合可视化表单开发,10分钟完成一个业务流程发布

    回忆以前工作流引擎的应用,感觉历历在目啊!当初公司接了一个项目关于政府单位公文流转的管理系统,一开始客户跟我画了十多张业务流程图.话说这十多张业务流程图,涉及的业务范围还蛮多,像用审批授权,开通流程, ...

  9. 在Bootstrap开发框架的工作流模块中实现流程完成后更新资料状态处理

    在开发查看流程表单明细的时候,在Web界面中,我们往往通过使用@RenderPage实现页面内容模块化的隔离,减少复杂度,因此把一些常用的如审批.撤销.会签.阅办等等的流程步骤都放到了通用处理的页面V ...

随机推荐

  1. [小程序开发] 微信小程序内嵌网页web-view开发教程

    为了便于开发者灵活配置小程序,微信小程序开放了内嵌网页能力.这意味着小程序的内容不再局限于pages和large,我们可以借助内嵌网页丰富小程序的内容.下面附上详细的开发教程(含视频操作以及注意事项) ...

  2. AsciidocFX编辑器小贴士

    I. AsciidocFX支持UML生成: 要生成UML,记得要下载GRAPHVIZ,并配置GRAPHVIZ_DOT环境变量,路径是Graphviz\bin\dot.exe. II. Asciidoc ...

  3. 在C#程序中模拟发送键盘按键消息

    using System.Runtime.InteropServices; 引入键盘事件函数 [DllImport("user32.dll")]public static exte ...

  4. [转]动态管理视图和函数 (Transact-SQL)

    动态管理视图和函数返回可用于监视服务器实例的运行状况.诊断故障以及优化性能的服务器状态信息. 重要提示 动态管理视图和函数返回特定于实现的内部状态数据. 在未来的 SQL Server 版本中,它们的 ...

  5. SSH框架的多表查询(方法二)增删查改

     必须声明本文章==>http://www.cnblogs.com/zhu520/p/7773133.html  一:在前一个方法(http://www.cnblogs.com/zhu520/p ...

  6. TCP:传输控制协议

    概述: 书中采用了8章来介绍TCP,可见其重要性.TCP是一种面向连接的.可靠的字节流服务,也就是说两方要交换数据必须先建立一个连接. TCP的信息单位称为segment.TCP对字节流的内容不作任何 ...

  7. Python案例分享

    1.过桥(爬金字塔): 1 i = 1 2 while i <= 9: 3 if i < 6: 4 j = 0 5 while j < i: 6 print('*',end=' ') ...

  8. (转载)java 枚举 循环遍历以及一些简单常见的使用

    本文转载自:http://blog.csdn.net/qq_27093465/article/details/51706076 作者:李学凯 什么时候想用枚举类型: 有时候,在设计一个java mod ...

  9. QuickTime视频解析问题

    在QuickTime中可以解析出视频并播放视频,解析的格式后缀名为.mov,之后将该视频导入到Unity Project中,显示未解析到视频文件,本来应该会自动生成MovieTexture材质,但是并 ...

  10. 爬虫day 04_01(爬百度页面)

    import urllib.request import http.cookiejar from lxml import etree head = { 'Connection': 'Keep-Aliv ...