C# 实现改造 GooFlow 流程图插件与数据库应用的结合

news2025/1/12 1:35:00

 

目录

关于 GooFlow

功能需求

范例运行环境

设计数据表

流程项目表

流程项目节点明细表

流程项目节点审批人表

 人员信息表

示例代码

流程图主功能

设置审批人信息

运行结果演示

总结


关于 GooFlow

GooFlow 一个基于 Jquery/FontAwesome 的流程图/架构图画图插件,本文介绍的是基于JS的一个版本,能够适用于大部分支持H5的浏览器。通过创建画布,我们可以在其上通过工具栏绘制想要绘制的流程图/框/线等,如下图所示:

图中包括操作工具栏(顶部)、绘图工具栏(左侧)和绘制画布区域,左上角显示的是当前流程的名称。 具体的操作我们不再详述,本文将介绍通过改造后的 GooFlow 简化版本,结合数据库满足实际的开发需求。

功能需求

我们需要实现一个审批流程,在关键流程节点可能会添加审批人信息,通过GoolFlow绘制并显示,则更加友好和直观, 基本需要实现如下功能:

1、改造 GooFlow JS 程序,简化绘制工具栏。

2、设计相关数据库数据表保存流程图设计相关信息、节点明细信息等。

3、改造 GooFlow 操作工具栏,满足实际应用。

范例运行环境

操作系统: Windows Server 2019 DataCenter

GooFlow 版本:GooFlow V1 JS 版 

数据库:Microsoft SQL Server 2016

.net版本: .netFramework4.7.1 或以上

开发工具:VS2019  C#

设计数据表

流程项目表

cc_flow 表记录项目信息,主要说明见下表:

序号字段名类型说明
1ciduniqueidentifier记录唯一标识
2flowNamenvarchar(50)流程项目名称
3nodeCountint节点总个数
4desriptnvarchar(50)项目描述
5flowJSONnvarchar(MAX)流程全部 JSON 数据
6sys_insusernvarchar(100)创建信息用户名
7sys_instimedatetime创建时间
8sys_updusernvarchar(100)最后修改信息用户名
9sys_updtimedatetime修改时间

 创建脚本如下:

CREATE TABLE [dbo].[cc_flow](
	[cid] [uniqueidentifier] ROWGUIDCOL  NOT NULL,
	[flowName] [nvarchar](50) NOT NULL,
	[nodeCount] [int] NULL,
	[desript] [nvarchar](100) NULL,
	[flowJSON] [nvarchar](max) NULL,
	[sys_insuser] [nvarchar](100) NULL,
	[sys_instime] [datetime] NULL,
	[sys_upduser] [nvarchar](100) NULL,
	[sys_updtime] [datetime] NULL,
 CONSTRAINT [PK_cc_flow] PRIMARY KEY CLUSTERED 
(
	[cid] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY],
 CONSTRAINT [IX_cc_flow] UNIQUE NONCLUSTERED 
(
	[flowName] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]
GO

ALTER TABLE [dbo].[cc_flow] ADD  CONSTRAINT [DF_cc_flow_cid]  DEFAULT (newid()) FOR [cid]
GO


流程项目节点明细表

cc_flowNodes 表记录了项目绘制节点的所有信息明细情况,主要说明见下表:

序号字段名类型说明
1ciduniqueidentifier记录唯一标识
2nodeIdnvarchar(50)节点ID
3nodeNamenvarchar(50)节点流程名称
4desriptnvarchar(100)节点流程描述
5parentNodeIdnvarchar(50)父节点ID
6projectCiduniqueidentifier流程项目ID,所属的项目唯一标识
7sortidint节点排序
8sys_insusernvarchar(100)创建信息用户名
9sys_instimedatetime创建时间
10sys_updusernvarchar(100)最后修改信息用户名
11sys_updtimedatetime修改时间

 创建脚本如下:

CREATE TABLE [dbo].[cc_flowNodes](
	[cid] [uniqueidentifier] NOT NULL,
	[nodeId] [nvarchar](50) NOT NULL,
	[nodeName] [nvarchar](50) NOT NULL,
	[desript] [nvarchar](100) NULL,
	[parentNodeId] [nvarchar](50) NULL,
	[projectCid] [uniqueidentifier] NOT NULL,
	[sortid] [int] NULL,
	[sys_insuser] [nvarchar](100) NULL,
	[sys_instime] [datetime] NULL,
	[sys_upduser] [nvarchar](100) NULL,
	[sys_updtime] [datetime] NULL
) ON [PRIMARY]
GO

ALTER TABLE [dbo].[cc_flowNodes] ADD  CONSTRAINT [DF_cc_flowNodes_cid]  DEFAULT (newid()) FOR [cid]
GO


流程项目节点审批人表

cc_flowNodeAccounts 表记录了项目关键节点的参与审批人情况,主要说明见下表:

序号字段名类型说明
1ciduniqueidentifier记录唯一标识
2nodeIdnvarchar(50)节点ID
3AccountCiduniqueidentifier所属的人员信息ID
4desriptnvarchar(100)描述,可以记录诸如姓名相关的一些信息
5rolenvarchar(50)审批人权限(如决策、抄送)
6projectCiduniqueidentifier流程项目ID,所属的项目唯一标识
7sys_insusernvarchar(100)创建信息用户名
8sys_instimedatetime创建时间
9sys_updusernvarchar(100)最后修改信息用户名
10sys_updtimedatetime修改时间

 创建脚本如下:


CREATE TABLE [dbo].[cc_flowNodeAccounts](
	[cid] [uniqueidentifier] NOT NULL,
	[projectCid] [uniqueidentifier] NOT NULL,
	[AccountCid] [uniqueidentifier] NOT NULL,
	[descript] [nvarchar](100) NULL,
	[nodeId] [nvarchar](50) NOT NULL,
	[role] [nvarchar](50) NULL,
	[sys_insuser] [nvarchar](100) NULL,
	[sys_instime] [datetime] NULL,
	[sys_upduser] [nvarchar](100) NULL,
	[sys_updtime] [datetime] NULL
) ON [PRIMARY]
GO

ALTER TABLE [dbo].[cc_flowNodeAccounts] ADD  CONSTRAINT [DF_cc_flowNodeBuss_cid]  DEFAULT (newid()) FOR [cid]
GO

 人员信息表

accounts 表审批人的基本个人情况,主要说明见下表:

序号字段名类型说明
1ciduniqueidentifier记录唯一标识
2namenvarchar(50)姓名
3nickNamenvarchar(50)昵称

 创建脚本如下:


CREATE TABLE [dbo].[accounts](
	[cid] [uniqueidentifier] ROWGUIDCOL  NOT NULL,
	[name] [nvarchar](50) NULL,
	[nickname] [nvarchar](500) NULL,
 CONSTRAINT [PK_wxmp_accounts] PRIMARY KEY CLUSTERED 
(
	[cid] ASC
)
)
GO

ALTER TABLE [dbo].[accounts] ADD  CONSTRAINT [DF_wxmp_accounts_cid]  DEFAULT (newid()) FOR [cid]
GO

示例代码

流程图主功能

本示例代码包含后端、前端及 JS 代码,代码如下:

<%@ Page Language="C#" AutoEventWireup="true" MaintainScrollPositionOnPostback="true"  ValidateRequest="FALSE" %>
<%@ Import Namespace="System.IO" %>
<%@ Import Namespace="System.Data" %>
<%@ Import Namespace="System.Data.SqlClient" %>

<script language="C#" runat="server">
    public CosysJaneCommonAPI.CODAL dal = new CosysJaneCommonAPI.CODAL();
    public DateTime dt = DateTime.Now;
    void Page_load(Object sender, EventArgs e)
    {
        dal.RunAt = Form;
        if (Page.IsPostBack == true) return;
        dal.ConnKeyString = "JaneConnectionCloud";
        ArrayList paras = new ArrayList();
        paras.Clear();
        paras.Add(new SqlParameter("a", "2"));
        dal.simpledatalist("select cid,flowName from cc_flow order by flowName ", paras, "cid", "flowName",flowlist, true, "", "选择打开已创建流程");

     }

    protected void saveflow_Click(object sender, EventArgs e)
    {
        dal.ConnKeyString = "JaneConnectionCloud";

        ArrayList paras = new ArrayList();
        paras.Clear();
        paras.Add(new SqlParameter("flowname", x_flowname.Text));
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        dal.ExecDbScripts("insert into cc_flow (flowName) values(@flowname) ", paras);
        if (dal.RowsCount > 0)
        {
            tip.Text = "创建成功!";
            flowpanel.Visible = true;
        }else
        {
            if (dal.ErrorMessage.IndexOf("重复键") != -1)
            {
                dal.RowsCount = 0;
                dal.ErrorMessage = "";
                paras.Clear();
                paras.Add(new SqlParameter("flowname", x_flowname.Text));
                object rv = dal.GetDataSet("select * from  cc_flow where flowname=@flowname ", paras);
                if (dal.RowsCount > 0)
                {
                    DataSet ds = rv as DataSet;
                    DataTable dt = ds.Tables[0];
                    x_cid.Text = dt.Rows[0]["cid"].ToString();
                    x_flowJSON.Text = dt.Rows[0]["flowJSON"].ToString();
                    
                    tip.Text = "打开成功!";
                    flowpanel.Visible = true;
                }
                else
                {
                    tip.Text = dal.ErrorMessage;
                    flowpanel.Visible = false;
                }
            }
            else
            {
                tip.Text = dal.ErrorMessage;
                flowpanel.Visible = false;
            }
        }
    }

    protected void updflow_Click(object sender, EventArgs e)
    {
        dal.ConnKeyString = "JaneConnectionCloud";

        ArrayList paras = new ArrayList();

        Newtonsoft.Json.Linq.JObject jsonObj = Newtonsoft.Json.Linq.JObject.Parse(x_flowJSON.Text);
        string nodelist = "";
        foreach (Newtonsoft.Json.Linq.JProperty jt in  jsonObj["nodes"])
        {
            nodelist+=jt.Name+",";
        }
        string cid = x_cid.Text;

        dal.ConnKeyString = "JaneConnectionCloud";

        paras.Clear();
        paras.Add(new SqlParameter("cid", cid));
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        dal.ExecDbScripts("delete cc_flowNodes where projectCid=@cid ", paras);

        string[] la = nodelist.Split(',');
        int added = 0;
        for (int i = 0; i < la.GetLength(0); i++)
        {
            if (la[i] != "")
            {
                paras.Clear();
                paras.Add(new SqlParameter("projectCid", cid));
                paras.Add(new SqlParameter("parentNodeId", (i == 0 ? cid : la[i - 1])));
                paras.Add(new SqlParameter("nodeId", la[i]));
                paras.Add(new SqlParameter("nodeName", jsonObj["nodes"][la[i]]["name"].ToString()));
                paras.Add(new SqlParameter("sortid", i + 1));
                dal.RowsCount = 0;
                dal.ErrorMessage = "";
                dal.ExecDbScripts("insert into cc_flowNodes(projectCid,parentNodeId,nodeId,nodeName,sortid) values(@projectCid,@parentNodeId,@nodeId,@nodeName,@sortid) ", paras);
                if (dal.RowsCount > 0)
                {
                    added++;
                }
            }
        }

        
        
        paras.Clear();
        paras.Add(new SqlParameter("cid", x_cid.Text));
        paras.Add(new SqlParameter("flowName",x_flowname.Text));
        paras.Add(new SqlParameter("flowJSON", x_flowJSON.Text));
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        dal.ExecDbScripts("update cc_flow set flowName=@flowname,flowJSON=@flowJSON where cid=@cid ", paras);
        if (dal.RowsCount > 0)
        {
            tip.Text = "更新成功!";
        }
        else
        {
            tip.Text = dal.ErrorMessage;
        }

    }

    protected void flowlist_SelectedIndexChanged(object sender, EventArgs e)
    {
        if (flowlist.SelectedValue == "")
        {
            return;
        }
        
        dal.ConnKeyString = "JaneConnectionCloud";

        ArrayList paras = new ArrayList();
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        paras.Clear();
        paras.Add(new SqlParameter("cid",flowlist.SelectedValue));
        object rv = dal.GetDataSet("select * from  cc_flow where cid=@cid ", paras);
        if (dal.RowsCount > 0)
        {
            DataSet ds = rv as DataSet;
            DataTable dt = ds.Tables[0];
            x_cid.Text = dt.Rows[0]["cid"].ToString();
            x_flowname.Text = dt.Rows[0]["flowName"].ToString();
            x_flowJSON.Text = dt.Rows[0]["flowJSON"].ToString();

            tip.Text = "打开成功!";
            flowpanel.Visible = true;
        }
        else
        {
            tip.Text = dal.ErrorMessage;
            flowpanel.Visible = false;
        }

    }
</script>        

<!DOCTYPE html>
<html lang="cn" >
<head>
  <meta charset="UTF-8">
  <title></title>
<link rel="stylesheet" href="dist/GooFlow.min.css?v=10" />
<link rel="stylesheet" href="fontawesome/css/font-awesome.min.css" />
</head>

<script src="dist/jquery.min.js"></script>
<script src="dist/GooFlow.js?v=7"></script>
<body>
  <form runat="server">
  <div style=" display:flex">
  <asp:TextBox ID="x_flowname" placeholder="输入项目名称" runat="server"></asp:TextBox>
  <asp:Button ID="saveflow" Text="创建项目" runat="server" onclick="saveflow_Click"></asp:Button>
  <asp:Button ID="updflow" Text="更新项目" style=" display:none" runat="server" onclick="updflow_Click" ></asp:Button>
  <asp:TextBox ID="x_cid" placeholder="cid" style="display:none" ReadOnly="true" runat="server"></asp:TextBox>
  <asp:DropDownList ID="flowlist" runat="server" AutoPostBack="true"
          onselectedindexchanged="flowlist_SelectedIndexChanged"></asp:DropDownList>
  </div>
  <div style=" margin:10px; display:flex">
  <asp:Label ID="tip" Text="这是一些提示信息..." runat="server"></asp:Label>
  </div>
      <iframe id="tframe" width="500" height="400" style=" display:none;z-index:99;position:absolute;" runat="server"></iframe>
    <div id="flowpanel" visible="false" runat="server">
</div>
    <div>
  <asp:TextBox ID="x_flowJSON" Rows="10" style="display:none"  width="100%" TextMode="MultiLine" runat="server"></asp:TextBox>
    </div>
  </form>
</body>
</html>
<script language="javascript" type="text/javascript">
        var tframe=$('#tframe');
        var curnode=null;
        var flowpanel=null;
        var sortBy = function (filed, rev, primer) {
            rev = (rev) ? -1 : 1;
            return function (a, b) {
                a = a[filed];
                b = b[filed];
                if (primer && typeof (primer) != 'undefined') {
                    a = primer(a);
                    b = primer(b);
                }else{
			        if(!isNaN(a) && !isNaN(a)){
				        a = Number(a);
				        b = Number(b);
			        }
		        }
                if (a < b) { return rev * -1; }
                if (a > b) { return rev * 1; }
                return 1;
            }
        };


        $(function(){
            var options={initLabelText:"未命名",toolBtns:["task","chat","state"],toolBtnRemarks:{cursor:"选择指针",direct:"连线",task:"任务结点",chat:"决策结点",state:"状态结点",group:"组织区域框"},headBtns:["save","undo","redo","new","expand"],headBtnRemarks:['保存','撤销','重做','清空画布','结点设置']};
            flowpanel = $('#flowpanel').createGooFlow(options);
            var flowpanelData = {
                areas: {
                    flowpanel_area_17: {
                        color: "yellow",
                        height: 300,
                        left: 40,
                        name: "部门一",
                        top: 30,
                        width: 150,
                    },
                    flowpanel_area_18: {
                        color: "blue",
                        height: 300,
                        left: 200,
                        name: "部门二",
                        top: 30,
                        width: 150,
                    },
                    flowpanel_area_19: {
                        color: "green",
                        height: 300,
                        left: 360,
                        name: "部门三",
                        top: 30,
                        width: 150,
                    },
                    flowpanel_area_20: {
                        color: "red",
                        height: 300,
                        left: 520,
                        name: "部门四",
                        top: 30,
                        width: 150,
                    },
                },
                lines: {
                    flowpanel_line_11: {
                        from: "flowpanel_node_1",
                        marked: false,
                        name: "",
                        to: "flowpanel_node_2",
                        type: "sl",
                    },
                },
                nodes: {
                    flowpanel_node_1: {
                        height: 24,
                        left: 70,
                        name: "节点流程1",
                        top: 66,
                        type: "chat",
                        width: 86,
                        sortid:1,
                        id:"flowpanel_node_1",
                    },
                    flowpanel_node_2: {
                        height: 24,
                        left: 240,
                        name: "节点流程2",
                        top: 132,
                        type: "state",
                        width: 86,
                        sortid:2,
                        id:"flowpanel_node_2",
                    },
                }
            };
            flowpanel.loadData(flowpanelData);

            if($('#x_flowJSON').val()!=""){
                flowpanel.clearData();
                flowpanel.loadData($.parseJSON($('#x_flowJSON').val()));
                alert("加载流程图成功");
            }
            flowpanel.setTitle($('#x_flowname').val());
            $('#flowpanel').css('backgroundColor','silver');
            $('#flowpanel').css('borderColor','silver');
           
            var obj = flowpanel.exportData();

            flowpanel.onExpandClick=function(){
                if(curnode==null) return;
                 var x=(curnode.offset().left+parseInt(curnode.css('width'),10))+'px';
                 var y=curnode.offset().top+'px';
                 tframe.css('left',x);
                 tframe.css('top',y);
                 tframe.attr('src','test_server.aspx?cid='+$('#x_cid').val()+'&nid='+curnode.selector.substring(1));
                 tframe.css('display',(tframe.css('display')=='none'?'':'none'));
                 
            }

            flowpanel.onBtnSaveClick=function(){
                  $('#x_flowJSON').val(JSON.stringify(this.exportData()));
                  $('#updflow').click();
            }
            

            flowpanel.onItemFocus=function(id,type){
                 flowpanel.focusItem(id,false);
                 curnode=$("#"+id);
                 tframe.css('display','none');
            }
       });

</script>
设置审批人信息

页面名称为test_server.aspx,在流程图主功能的JS代码部分有体现, 本示例代码包含后端、前端及 JS 代码,代码如下:

<%@ Page Language="C#" AutoEventWireup="true" enableEventValidation="false" MaintainScrollPositionOnPostback="true"  ValidateRequest="FALSE" %>
<%@ Import Namespace="System.IO" %>
<%@ Import Namespace="System.Data" %>
<%@ Import Namespace="System.Data.SqlClient" %>

<script language="C#" runat="server">
    public CosysJaneCommonAPI.CODAL dal = new CosysJaneCommonAPI.CODAL();
    public DateTime dt = DateTime.Now;
    void Page_load(Object sender, EventArgs e)
    {
        dal.RunAt = Form;
        if (Page.IsPostBack == true) return;
        ArrayList paras = new ArrayList();
        dal.simpledatalist("select cid,nickname,name from accounts ", null, "cid", "name", sprlist, false, "", "");
        dal.simpledatalist("select cid,'抄送' name from accounts ", null, "cid", "name", spr2list, false, "", "");

        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        string cid = Request.QueryString["cid"];
        string nodeid = Request.QueryString["nid"];
        paras.Clear();
        paras.Add(new SqlParameter("cid", cid));
        paras.Add(new SqlParameter("nodeid", nodeid));
        object rv = dal.GetDataSet("select AccountCid,nodeId,role from cc_flowNodeAccounts where projectCid=@cid and nodeId=@nodeid", paras);
        if (dal.RowsCount > 0)
        {
            DataTable dt = (rv as DataSet).Tables[0];
            for (int i = 0; i < dt.Rows.Count; i++)
            {
                string accountid = dt.Rows[i]["AccountCid"].ToString();
                string role = dt.Rows[i]["role"].ToString();
                ListItem li = sprlist.Items.FindByValue(accountid);
                if (li!=null)
                {
                    li.Selected = true;
                    if(role=="抄送"){
                        spr2list.Items.FindByValue(accountid).Selected = true;
                    }
                }
            }
        }

        
     }

    protected void saveflow_Click(object sender, EventArgs e)
    {
        string cid = Request.QueryString["cid"];
        string nodeid = Request.QueryString["nid"];

        ArrayList paras = new ArrayList();
        paras.Clear();
        paras.Add(new SqlParameter("cid", cid));
        paras.Add(new SqlParameter("nodeid", nodeid));
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        dal.ExecDbScripts("delete cc_flowNodeAccounts where projectCid=@cid and nodeId=@nodeid", paras);
        for (int i = 0; i < sprlist.Items.Count; i++)
        {
            ListItem li = sprlist.Items[i];
            
            if (li.Selected==true)
            {
                paras.Clear();
                paras.Add(new SqlParameter("projectCid", cid));
                paras.Add(new SqlParameter("AccountCid", li.Value));
                paras.Add(new SqlParameter("nodeid", nodeid));
                paras.Add(new SqlParameter("role", (spr2list.Items[i].Selected==true?"抄送":"决策")  ));
                paras.Add(new SqlParameter("descript", li.Text));
                dal.RowsCount = 0;
                dal.ErrorMessage = "";
                dal.ExecDbScripts("insert into cc_flowNodeAccounts(projectCid,AccountCid,nodeId,role,descript) values(@projectCid,@AccountCid,@nodeid,@role,@descript) ", paras);
                Response.Write(dal.ErrorMessage);
            }
        }
        Alert(Form, "保存数据信息完成!");

    }

    protected void updflow_Click(object sender, EventArgs e)
    {

        ArrayList paras = new ArrayList();
        paras.Clear();
        paras.Add(new SqlParameter("cid", x_cid.Text));

    }

    protected void tooltab_SelectedIndexChanged(object sender, EventArgs e)
    {
        spr.Style["display"] = "none";
        sortnode.Style["display"] = "none";

        if (tooltab.SelectedValue == "0")
        {
            spr.Style["display"] = "";
        }else if (tooltab.SelectedValue == "1")
        {
            sortnode.Style["display"] = "";
        }
    }
    void SortRow(Object sender, EventArgs e)
    {
        Newtonsoft.Json.Linq.JObject jsonObj = Newtonsoft.Json.Linq.JObject.Parse(x_flowJSON.Text);

       
        string cid = Request.QueryString["cid"];


        ArrayList paras = new ArrayList();
        paras.Clear();
        paras.Add(new SqlParameter("cid", cid));
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        dal.ExecDbScripts("delete cc_flowNodes where projectCid=@cid ", paras);

        string[] la = sortstate1.Text.Split(',');
        int added = 0;
        for (int i = 0; i < la.GetLength(0); i++)
        {
            if (la[i] != "")
            {
                paras.Clear();
                paras.Add(new SqlParameter("projectCid", cid));
                paras.Add(new SqlParameter("parentNodeId",(i==0?cid:la[i-1])) );
                paras.Add(new SqlParameter("nodeId", la[i]));
                paras.Add(new SqlParameter("nodeName", jsonObj["nodes"][la[i]]["name"].ToString()));
                paras.Add(new SqlParameter("sortid", i+1));
                dal.RowsCount = 0;
                dal.ErrorMessage = "";
                dal.ExecDbScripts("insert into cc_flowNodes(projectCid,parentNodeId,nodeId,nodeName,sortid) values(@projectCid,@parentNodeId,@nodeId,@nodeName,@sortid) ", paras);
                if (dal.RowsCount > 0)
                {
                    added++;
                }
            }
        }
        paras.Clear();
        paras.Add(new SqlParameter("cid", cid));
        paras.Add(new SqlParameter("flowJSON", x_flowJSON.Text));
        paras.Add(new SqlParameter("nodeCount", added));
        dal.RowsCount = 0;
        dal.ErrorMessage = "";
        dal.ExecDbScripts("update cc_flow set flowJSON=@flowJSON,nodeCount=@nodeCount where cid=@cid ", paras);
        if (dal.RowsCount > 0)
        {
        }
        else
        {
            Alert(Form, "保存流程数据信息失败!");
            return;
        }
        Alert(Form,"保存排序信息完毕!");
    }
    public void Alert(Control updatePanel, string msg)
    {

        msg = msg.Replace("\r\n", "").Replace("'", "\\'");
        ScriptManager.RegisterClientScriptBlock(updatePanel, this.GetType(), "", "alert('" + msg + "')", true);
    }
</script>        

<!DOCTYPE html>
<html lang="cn" >
<head>
  <meta charset="UTF-8">
  <title></title>
</head>

<body style=" background-color:Gray">
  <form runat="server">
  <div style=" display:flex">
  <asp:RadioButtonList ID="tooltab" AutoPostBack="true"   Font-Names="微软雅黑" 
          RepeatDirection="Horizontal" runat="server" 
          onselectedindexchanged="tooltab_SelectedIndexChanged" >
          <asp:ListItem Value="0" Text="审批人" Selected="True"></asp:ListItem>
          <asp:ListItem Value="1" Text="节点排序" ></asp:ListItem>

  </asp:RadioButtonList>
  </div>
  <asp:Panel ID="spr" runat="server" Visible="true" BackColor="Silver"  >
  <div style=" display:flex">
      <asp:CheckBoxList ID="sprlist" runat="server" Height="100px" ForeColor="White" Width="134px">
      </asp:CheckBoxList>
      <asp:CheckBoxList ID="spr2list" runat="server" Height="100px" BackColor="Gray" ForeColor="Silver" Width="134px">
      </asp:CheckBoxList>

  <asp:Button ID="updflow"  Text="更新项目" style=" display:none" runat="server" onclick="updflow_Click" ></asp:Button>
  <asp:TextBox ID="x_cid" style=" display:none" ReadOnly="true" runat="server"></asp:TextBox>
  </div>
  <div style=" display:flex">
      <asp:Button ID="saveflow" style=" display:none1" Text="保存审批人" runat="server" onclick="saveflow_Click"></asp:Button>
  </div>
  </asp:Panel>

  <asp:Panel ID="sortnode" runat="server" Visible="true" style="display:none" BackColor="Silver">
  <table border="0" width="100%" cellspacing="0" cellpadding="0" id="table1">
		<tr>
			<td valign="top">
			<div align="center">
			<table border="0" width="94%" cellspacing="0" cellpadding="0" id="table2">
				<tr>
					<td align="left" valign="top">
					<asp:listbox id="funclist" ondblclick="SetValues(this)" 
                            style="border: 1px outset #6699FF; background-color: #FFFFFF;" Runat="server"
                            Height="328px" Width="405px" Font-Names="微软雅黑"/></td>
					<td>
					<p align="center">
					<input id="moveUp" onclick="moveSelected(document.getElementById('funclist'), false)" type=button style="font-family:微软雅黑;" value="上移"/></p>
					<p align="center">
					<input id="moveDown" onclick="moveSelected(document.getElementById('funclist'), true)" type=button style="font-family:微软雅黑;" value="下移"/></p>
					<p  align="center"><input id="moveDown0" onclick="sortfunc()" type="button" value="保存"  style="font-family:微软雅黑;"/></p></td>
				</tr>
			</table>
			</div>
			</td>
		</tr>
	</table>
  </asp:Panel>
<asp:TextBox ID="sortstate"  Text="" style="display:none" Runat=server/>
<asp:TextBox ID="sortstate1"  Text="" style="display:none" Runat="server"/>
<asp:Button ID="sortbtn" Text="sort" OnClick="SortRow" style="display:none" Runat="server"/>
  <asp:TextBox ID="x_flowJSON" Rows="10" style="display:none"  width="100%" TextMode="MultiLine" runat="server"></asp:TextBox>

  </form>
</body>
</html>
<script language="javascript" type="text/javascript">
    form = document.forms[0];
    function sortfunc() {
        if (!confirm('您确认保存排序结果吗?')) {
            return;
        }
        form.sortstate1.value = "";
        for (var i = 0; i < form.funclist.length; i++) {
            var _id = form.funclist.options[i].value;
            flowpanel.$nodeData[_id].sortid = i;
            form.sortstate1.value += _id + ",";
        }
        //  return;
        form.sortstate.value = "1";
        var obj = flowpanel.exportData();
        form.x_flowJSON.value = JSON.stringify(obj);
        form.sortbtn.click();
    }
    function AddListBoxOption(obj, _text, _value) {
        newOption = document.createElement("OPTION");
        newOption.text = _text;
        newOption.value = _value;
        obj.options.add(newOption);
    }
    function moveSelected(select, down) {

        if (select.selectedIndex != -1) {
            if (down) {
                if (select.selectedIndex != select.options.length - 1)
                    var i = select.selectedIndex + 1;
                else
                    return;
            }
            else {
                if (select.selectedIndex != 0)
                    var i = select.selectedIndex - 1;
                else
                    return;
            }
            var swapOption = new Object();
            swapOption.text = select.options[select.selectedIndex].text;
            swapOption.value = select.options[select.selectedIndex].value;
            swapOption.selected = select.options[select.selectedIndex].selected;
            //    swapOption.defaultSelected = select.options[select.selectedIndex].defaultSelected;
            for (var property in swapOption)
                select.options[select.selectedIndex][property] = select.options[i][property];
            for (var property in swapOption)
                select.options[i][property] = swapOption[property];
        }
    }
    function jsonSort(array, field, reverse) {
        //数组长度小于2 或 没有指定排序字段 或 不是json格式数据
        if (array.length < 2 || !field || typeof array[0] !== "object") return array;
        //数字类型排序
        if (typeof array[0][field] === "number") {
            array.sort(function (x, y) { return x[field] - y[field] });
        }
        //字符串类型排序
        if (typeof array[0][field] === "string") {
            array.sort(function (x, y) { return x[field].localeCompare(y[field]) });
        }
        //倒序
        if (reverse) {
            array.reverse();
        }
        return array;
    }
    var flowpanel = (window.parent.flowpanel);
    window.onload = function () {
        var obj = flowpanel.exportData();
        form.x_flowJSON.value = JSON.stringify(obj);

        var nodelist = document.getElementById('funclist');
        nodelist.length = 0;
        var sortid = 0;
        var obj2 = new Array();

        for (var i in obj.nodes) {
            flowpanel.$nodeData[i].id = i;
            obj2.push(flowpanel.$nodeData[i]);
        }
        var obj3 = jsonSort(obj2, 'sortid', false);
        for (var j = 0; j < obj3.length;j++ ) {
            flowpanel.$nodeData[obj3[j].id].sortid = sortid;
            AddListBoxOption(nodelist, obj3[j].name, obj3[j].id);
            sortid++;
        }
    }

</script>
运行结果演示

代码正常运行后如下图所示:

点击某一节点,点击控制栏最后的设置图标,会提供选择审批人的操作界面,如下图所示:

总结

关于 GooFlow 的引用,请下载我的资源:

https://download.csdn.net/download/michaelline/89601233

CosysJaneCommonAPI.CODAL 类的 dal.simpledatalist 方法需要在实际中自行改造,可参考我的文章:

《C# Web控件与数据感应之 ListControl 类》

CosysJaneCommonAPI.CODAL 类的 dal.GetDataSet 方法需要在实际中自行改造,可参考我的文章:

CosysJaneCommonAPI.CODAL 类的 dal.ExecDbScripts 方法需要在实际中自行改造,可参考我的文章:

《C#利用IDbCommand实现通用数据库脚本执行程序》

代码这里仅供大家参考,欢迎大家评论指教!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1980484.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

探索Python异步编程的秘境:Eventlet的魔力与魅力

文章目录 **探索Python异步编程的秘境&#xff1a;Eventlet的魔力与魅力**第一部分&#xff1a;背景与引言第二部分&#xff1a;Eventlet是什么&#xff1f;第三部分&#xff1a;如何安装Eventlet&#xff1f;第四部分&#xff1a;Eventlet的基本使用第五部分&#xff1a;Event…

如何用ai来完成数据库分析(2)

一样的前言 因一些课程设计要写长篇分析报告&#xff0c;这里借用ai做一篇指导教程&#xff0c;分上下两篇。这篇也会教如何让ai给你你想要的答案&#xff0c;众所周知&#xff0c;现在的ai并不智能&#xff0c;不针对各类厂家&#xff0c;但是放出来的确实表象如此。 但其实…

CMIP6数据处理技术教程

原文链接&#xff1a;CMIP6数据处理技术教程https://mp.weixin.qq.com/s?__bizMzUzNTczMDMxMg&mid2247611539&idx4&sn3eb0490b72ef694dcd41f6de0b623388&chksmfa827574cdf5fc62372bfcf188ea1ae6e41350ff8469c2940ed097244b205e0dd73f4c8832fa&token616696…

绝对干货!分享把Fildder录制的脚本导出成Jmeter脚本的方法(实测在Fiddler5中有效)

相信做过接口测试的小伙伴都面临过这样的窘境——在没有接口测试文档的情况下&#xff0c;进行接口的压力测试&#xff01;怎么破&#xff1f; 通常的做法都是利用抓包工具&#xff08;以fiddler为例&#xff09;进行抓包&#xff0c;然后把抓包请求手写到压力测试工具中&…

解决Solidworks 2024运行的Windows资源极低,执行此命令可能会导致SOLIDWORKS 失败。

Solidworks Resource Monitor 可供使用的系统内存很低。请关闭一些应用程序以释放资源。即使内存十分充足&#xff0c;也容易出现这些提示&#xff0c;而我个人感觉2024版相比以前版本更容易出现这个问题&#xff0c;这显然是Solidworks本身的Bug。网上普遍流传的解决方法&…

再讲Langchain 提示词模板(PromptTemplate)

在LangChain 0.2中,提示词模板(Prompt Template)是一个非常重要的概念和功能。它允许我们创建结构化和可重用的提示,以便更有效地与语言模型进行交互。 提示词模板的主要作用包括: 结构化提示: 允许我们定义一个固定的提示结构,包含静态文本和动态变量。 参数化: 可以在模板中…

汽车测试-ADAS测试设备RT3000介绍及数据处理

目录 一、分析目的 二、数据采集仪器 三、数据文件 1.数据样本名词解释 2.数据应用 3.简单举例 汽车数据分析示例 1. 行驶轨迹与路径优化 2. 速度与加速度分析 3. 燃油效率与能耗分析 4. 驾驶行为与安全分析 5. 车辆维护与故障诊断 评估导航与定位的准确性。 一、…

测试工程师职业道路管理方向有哪些

目录 01测试组长 测试组长的职责及掌握技能&#xff1a; 测试组长需要掌握的技能&#xff1a; 02测试经理 测试经理具备的职责&#xff1a; 测试经理具备的技能&#xff1a; 03测试总监 测试总监具备的职责&#xff1a; 测试总监具备的技能&#xff1a; 测试工程师管理…

蚓链数字化生态系统:助力企业上下游资源的强大引擎

在当今数字化浪潮汹涌的时代&#xff0c;集团型企业面临着日益复杂的市场环境和激烈的竞争挑战。如何有效地拉通产业上下游资源&#xff0c;实现协同发展&#xff0c;成为了众多企业亟待解决的关键问题。而蚓链数字化生态系统的出现&#xff0c;犹如一道曙光&#xff0c;为集团…

前端工程化12-Git版本控制工具详解

2.1、Git知识概述 理解版本控制工具&#xff0c;聊下他的历史&#xff0c;之后会讲两种版本控制工具的区别&#xff08;集中式|分布式&#xff09;、他的基本环境搭建安装&#xff0c;如何初始化本地仓库&#xff0c;如何往仓库里提交一些东西&#xff0c;文件的话会有那些变化…

文本相似度 HanPL汉语言处理

文章目录 前言需求简介实操开始1. 添加pom.xml依赖2. 文本相似度工具类3. 案例验证4. 验证结果 总结 前言 请各大网友尊重本人原创知识分享&#xff0c;谨记本人博客&#xff1a;南国以南i、 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 需求 当我…

【redis 第五篇章】持久化之AOF和RDB

一、概述 Redis 是内存数据库&#xff0c;如果不能将内存中的数据保存到磁盘中&#xff0c;那么一旦服务器进程退出&#xff0c;数据库中数据会消失&#xff0c;所以 Redis 提供了持久化的功能, Redis 分为两种持久化方式&#xff1a;RDB 和 AOF&#xff0c;有以下几个特点&am…

荟萃科技:国外问卷调查有没有实时更新的题库?

有的&#xff0c;口子查和渠道查都是。 口子查的题目都是国外的公司发放在网络上&#xff0c;都是实时发布&#xff0c;所以我们需要去国外的各大社交平台做题。 这些题目不是集中的&#xff0c;而是散布在网站里面&#xff0c;需要我们去找&#xff0c;都是老外上班实时发放…

数据结构(其二)--线性表(其三)

目录 11.特殊矩阵的压缩存储 &#xff08;1&#xff09;.一维数组的储存结构 &#xff08;2&#xff09;.二维数组的存储结构 &#xff08;3&#xff09;.普通矩阵的存储 &#xff08;4&#xff09;.特殊矩阵的压缩存储 i.对称矩阵 ii.三角矩阵 iii.三对角矩阵 iiii.稀疏矩…

社科经管类:7本期刊被标记为“On Hold”状态!

本周投稿推荐 SCI&EI • 4区“水刊”&#xff0c;纯正刊&#xff08;来稿即录&#xff09; • CCF-B类&#xff0c;IEEE一区-Top&#xff08;3天初审&#xff09; EI • 各领域沾边均可&#xff08;2天录用&#xff09; 知网&#xff08;CNKI&#xff09;、谷歌学术 …

Github 2024-08-05 开源项目周报 Top15

根据Github Trendings的统计,本周(2024-08-05统计)共有15个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Go项目4JavaScript项目3Python项目3Java项目2TypeScript项目2C项目1Shell项目1Dockerfile项目1非开发语言项目1免费编程学习平台:freeCodeCamp…

2.2 (2) 调度算法

文章目录 调度算法的评价指标CPU利用率系统吞吐量周转时间等待时间响应时间总结 典型的调度算法先来先服务&#xff08;FCFS&#xff09;短作业优先&#xff08;SJF&#xff09;高响应比优先&#xff08;HRRN&#xff09;时间片轮转调度算法优先级调度算法多级反馈列队调度算法…

动态规划:打家劫舍系列

目录 1. 打家劫舍1(线性数组)(LeetCode198) 解法1&#xff1a;动态规划&#xff08;二维dp数组&#xff09; 解法2&#xff1a;动态规划&#xff08;一维dp数组&#xff09; 解法3&#xff1a;动态规划&#xff08;一维dp数组优化&#xff09; 2. 打家劫舍2(环形数组)…

信号处理——自相关和互相关分析

1.概括 在信号处理中&#xff0c;自相关和互相关是相关分析非常重要的概念&#xff0c;它们能分析一个信号或两个信号在时间维度的相似性&#xff0c;在振动测试分析、雷达测距和声发射探伤得到了广泛的应用。自相关分析的研究对象为一个信号&#xff0c;互相关分析的研究对象…

spring原理(第八天)

aop的实现原理 AOP 底层实现方式之一是代理&#xff0c;由代理结合通知和目标&#xff0c;提供增强功能 除此以外&#xff0c;aspectj 提供了两种另外的 AOP 底层实现&#xff1a; 第一种是通过 ajc 编译器在编译 class 类文件时&#xff0c;就把通知的增强功能&#xff0c;织…