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

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

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开发框架中工作流模块的业务表单开发

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容