C#实现流程图设计器

最近一个偶然的机会竟然实现了一个简单的流程图设计器(虽然其功能还有很多不完善之处,但是心中还是非常高兴,满满的成就感)。

话不多说,先看一下实现的主界面效果:

左边是一个ListView(listView1),右边的画布是一个Panel(panel1)。下面将主要思路介绍如下:

  1)允许拖放,listView1和panel1设置其AllowDrop=true;

  2)非连接线类型的图形拖放处理:左边的listView1的项目被选中后,可以获取其图形类型(是路由器、是服务器还是云等),并在全局变量中记录下当前的操作对象类型,然后拖放到panel1后,panel1获得对应的图形类型,首先判断图的类型是否为非连接线,如果是则获取对应的图片,用g.DrawImage将其绘制到画布中,图片的坐标参考自当前鼠标(拖放到panel1最后松开鼠标左键时的坐标)的坐标。

  3)连接线类型的图形处理:如果是连接线,应该要有两个点来确定一条直线。当选中listView1的连接线时,会在全局变量中记录下当前的操作对象类型是连接线,当在panel1上单击时,首选判断当前的操作对象类型的全局对象是否为连接线,如果是,则记录第一次单击的点,然后等待记录单击的第二个点,当第二次单击完成后,调用绘制直线的方法在画布中进行绘制直线。

  4)当线和图形绑定后,拖放图形时,直线附属在图形的那个点会随着图形位置的变化而变化,当最后定位后,panel1会重绘网格和流程图。

  5)编辑图形信息:在panel1上双击时,程序获取双击的坐标点离所有的图形区域中最近的图形,然后计算距离,看是否满足设置的阈值,如果小于阈值,则认为是在该图形上双击,是要进行编辑操作。

下面给出绘制网格的代码:

/// <summary>
  /// 绘制网格
  /// </summary>
  private void renderGrid()
  {
   //全局变量存储最大最小值,作为绘制区域
   Graphics g = this.panel1.CreateGraphics();
   Color color = Color.DarkGray;
   Pen p = new Pen(color, 1);
   p.DashStyle = DashStyle.Dash;
   for (int x = 0; x <= this.panel1.Width; x = x + 20)
   {
    PointF p1 = new PointF(x, 0);
    PointF p2 = new PointF(x, Height);
    g.DrawLine(p, p1, p2);
   }

   for (int y = 0; y <= panel1.Height; y = y + 20)
   {
    PointF p1 = new PointF(0, y);
    PointF p2 = new PointF(Width, y);
    g.DrawLine(p, p1, p2);

   }

  }

下面给出在panel1上进行鼠标单击的处理程序:

private void panel1_MouseClick(object sender, MouseEventArgs e)
  {
   int X = e.X;
   int Y = e.Y;
   if (this.__gObjType== "")
   {
    return;
   }
   if (this.__gObjType != "Line")
   {
    AddObjectFromMouseLocation(X, Y, 0, 0, this.__gObjType);
   }
   else
   {
    //line
    if (__lineMouseClickedCount == 1)
    {
     __lineX2 = e.X;
     __lineY2 = e.Y;
     AddObjectFromMouseLocation(__lineX1, __lineY1, __lineX2, __lineY2, this.__gObjType);

     //连接线方向判断
     __lineMouseClickedCount = 0;
     __lineX1 = 0;
     __lineY1 = 0;
     __lineX2 = 0;
     __lineY2 = 0;
    }
    else if (__lineMouseClickedCount == 0)
    {
     __lineX1 = e.X;
     __lineY1 = e.Y;
     __lineMouseClickedCount = 1;
    }
    else
    {
     __lineMouseClickedCount = 0;
     __lineX1 = 0;
     __lineY1 = 0;
     __lineX2 = 0;
     __lineY2 = 0;
    }

   }
  }

下面给出重绘的程序:

private void ReDrawAll()
  {
   renderGrid();
   Graphics g = this.panel1.CreateGraphics();
   g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
   GObject CurrObj = new GObject();
   Rectangle Rct = new Rectangle();
   Pen p = new Pen(Color.Black);
   //p.Width = 2;
   p.Width = __penWidth * __zoomTimes;

   Image ObjImg;
   int xm = 0;
   int ym = 0;

   int _maxX = 0;
   int _maxY = 0;
   int _minX = 0;
   int _minY = 0;
   int _oldW = this.panel1.Width;
   int _oldH = this.panel1.Height;

   string IsLine = "";
   //Nobj==50为当前画布最大的对象个数
   for (int i = 0; i < GNetworkFlow.Nobj; i++)
   {
    CurrObj = GNetworkFlow.GObjects[i];
    //当前对象类型判断
    if (CurrObj.Type == "") IsLine = "N/D";
    if (CurrObj.Type == "Line") IsLine = "Y";
    if ((CurrObj.Type != "Line") && (CurrObj.Type != "")) IsLine = "N";
    //
    #region old panel1作为画布,将其嵌套在panel2中实现超出边界出现滚动条

    if (_maxX < CurrObj.x2)
    {
     _maxX = CurrObj.x2;
    }
    if (_maxY < CurrObj.y2)
    {
     _maxY = CurrObj.y2;
    }
    if (_minX > CurrObj.x1)
    {
     _minX = CurrObj.x1;
    }
    if (_minY > CurrObj.y1)
    {
     _minY = CurrObj.y1;
    }

    if (_oldW < _maxX - _minX)
    {
     this.panel1.Width = _maxX - _minX;
    }
    if (_oldH < _maxY - _minY)
    {
     this.panel1.Height = _maxY - _minY;
    }
    if (this.panel1.Height < this.panel2.Height)
    {
     this.panel1.Height = this.panel2.Height;
    }
    if (this.panel1.Width < this.panel2.Width)
    {
     this.panel1.Width = this.panel2.Width;
    }

    #endregion

    switch (IsLine)
    {
     case "Y":
      arrow.DrawArrow(g, p, p.Brush, CurrObj.x1, CurrObj.y1, CurrObj.x2, CurrObj.y2);
      xm = (CurrObj.x1 + CurrObj.x2) / 2;
      ym = (CurrObj.y1 + CurrObj.y2) / 2;
      AddText(xm, ym, CurrObj.Name, false);
      break;
     case "N":
      Rct.X = CurrObj.x1;
      Rct.Y = CurrObj.y1;
      Rct.Width = CurrObj.x2 - CurrObj.x1;
      Rct.Height = CurrObj.y2 - CurrObj.y1;
      if (CurrObj.Type != String.Empty)
      {
       ObjImg = FindGObjectTypeImage(CurrObj.Type);
       g.DrawImage(ObjImg, Rct);
       AddText(CurrObj.x1, CurrObj.y1, CurrObj.Name, true);
       GNetworkFlow.AdjustLinkedTo(CurrObj.Name);
      }
      break;
    }
   }

  }

下面将继续完善以下几个功能:

1)序列化:可以将图形序列化和反序列化,将序列化的信息保存到数据库,也可以从数据库加载图形;

2)图形节点必须要附加其他属性和方法,为流程记录更多的信息、例如权限配置、当前处理的人、下一步是什么节点等;

3)绘图功能的加强,绘图可以动态修改颜色,这样可以区分流程在不同节点的颜色显示;

4)布局优化算法,能否根据画布大小,自动排列图形...

现在又将界面做了美化,界面如下:

以上就是C#实现流程图设计器的全部步骤,还分享了完善设计器的技巧,希望大家喜欢。

(0)

相关推荐

  • Activiti流程图查看实例

    本文实例展示了Activiti流程图查看的实现方法,具体步骤如下所示: 1.测试用例查看图片代码如下: public void viewImage() throws Exception { // 创建仓库服务对对象 RepositoryService repositoryService = processEngine.getRepositoryService(); // 从仓库中找需要展示的文件 String deploymentId = "701"; List<String&g

  • oracle sql执行过程(流程图)

    Oracle sql执行流程图_SQL执行过程一.sql语句的执行步骤:1)语法分析,分析语句的语法是否符合规范,衡量语句中各表达式的意义.2) 语义分析,检查语句中涉及的所有数据库对象是否存在,且用户有相应的权限.3)视图转换,将涉及视图的查询语句转换为相应的对基表查询语句.4)表达式转换, 将复杂的 SQL 表达式转换为较简单的等效连接表达式.5)选择优化器,不同的优化器一般产生不同的"执行计划"  6)选择连接方式,ORACLE有三种连接方式,对多表连接ORACLE可选择适当的连

  • activiti获取流程图实例

    本文实例讲述了activiti获取流程图的方法,具体操作步骤如下: 1.发布流程代码如下: repositoryService.createDeployment() .name("expense-process.bar") .addClasspathResource("org/activiti/expenseProcess.bpmn20.xml") .addClasspathResource("org/activiti/expenseProcess.png

  • 详解js中构造流程图的核心技术JsPlumb

    项目里面用到了Web里面的拖拽流程图的技术JsPlumb,其实真不算难,不过项目里面用HTML做的一些类似flash的效果,感觉还不错,在此分享下. 一.效果图展示 1.从左边拖动元素到中间区域,然后连线 2.连线类型可以自定义:这里定义为直线.折线.曲线.实际项目中根据业务我们定义为分装线.分装支线.总装线等 3.鼠标拖动区域选中元素,并且选中元素统一拖动位置. 4.对选中的元素左对齐. 5.对选中元素居中对齐 6.右对齐 7.上对齐 8.垂直居中对齐 9.下对齐 10.根据第一个选中的元素上

  • 详解js中构造流程图的核心技术JsPlumb(2)

    前言:上篇详解js中构造流程图的核心技术JsPlumb介绍了下JsPlumb在浏览器里面画流程图的效果展示,以及简单的JsPlumb代码示例.这篇还是接着来看看各个效果的代码说明. 一.设置连线的样式和颜色效果代码示例 大概的效果如图: 这些效果看着很简单,那么,我们如何用代码去实现它呢.上章我们说过,JsPlumb的连线样式是由点的某些属性决定的,既然如此,我们就通过设置点的样式来动态改变连线的样式即可.来看代码: 首先来看看连线类型的那个select <div id="btn_line

  • photoshop制作网站流程图解

    说点题外话,这篇教程是网络上找的,为了找到他的图,我起码找了10个以上网站,这十多个网站转载别人的文章图片全是用的盗链,源头的网站把图片地址改了,导致这十多个网站都无法显示图片了,无语....盗链已经如此之严重了.还好有的站把图片上传到自己空间了,总算找齐了这些图片.教程作者已经无从考究了,如果你知道请联系我. 图片附件: [1] 104142_550699506_mNEJG.jpg (2006-11-7 11:04, 26.54 K) screen.width*0.7) {this.resiz

  • C#实现流程图设计器

    最近一个偶然的机会竟然实现了一个简单的流程图设计器(虽然其功能还有很多不完善之处,但是心中还是非常高兴,满满的成就感). 话不多说,先看一下实现的主界面效果: 左边是一个ListView(listView1),右边的画布是一个Panel(panel1).下面将主要思路介绍如下: 1)允许拖放,listView1和panel1设置其AllowDrop=true; 2)非连接线类型的图形拖放处理:左边的listView1的项目被选中后,可以获取其图形类型(是路由器.是服务器还是云等),并在全局变量中

  • Android项目开发之UI设计器

    开发人员可以用以下两种方式声明UI:一是通过.xml文件(不带预览界面)或者.axml文件(带预览界面)来描述:二是用C#代码实现.  用.axml文件描述用户界面(UI)时,设计器分为[设计]视图和[源]视图.这种方式的优点是:可以尽可能地把表现与控制行为的代码分隔开.就是说,可以直接修改XML而不需要重新编译.另外,还可以让UI结构可视化以及调试都变得更容易.  1.布局文件(.xml和.axml)  在XML中声明UI之后,必须将其以.xml扩展名或者.axml扩展名保存在Android项

  • C#利用DesignSurface如何实现简单的窗体设计器

    System.ComponentModel.Design.DesignSurface是为设计组件提供一个用户界面,通过它可以实现一个简单的窗体设计器. 在构建之前,我们需要引入System.Design.dll,否则会出现找不到DesignSurface的错误. private void Form1_Load(object sender, EventArgs e) { //引用System.Deisgn.dll DesignSurface ds = new DesignSurface(); //

  • visual studio 2019使用net core3.0创建winform无法使用窗体设计器

    微软发布正式版net core3.0后,迫不及待的想体验一下用visual studio 2019在net core3.0下创建winform程序.创建方法很简单,和以前visual studio版本步骤差不多. 创建完成之后,尴尬的事情发生了,无法使用窗体设计器,双击Form1.cs文件不行,使用快捷键shift+F7也不行,在网上找了很久,发现好多人都遇到过这种问题,目前有两种解决方案 方案1 项目中创建多目标框架,包含net framework和net core. 打开csproj文件,将

  • VisualStudio2019中为.NET Core WinForm App启用窗体设计器

    当我们在使用 Visual Studio 2019 非预览版本开发 Windows Forms App (.NET Core) 应用程序时是不能使用窗体设计器的.即使在窗体文件上右击选择"显示设计器"菜单,仍旧只能看到代码,无法打开窗体设计器. 根据微软开发者博客的描述,我们可以使用 Visual Studio 2019 预览通道,将 Visual Studio 2019 更新至 16.6 来启用设计器(参见:Updates on .NET Core Windows Forms des

  • IDEA Ui设计器JFormDesigner 永久激活插件+注册机(亲测一直在用)

    目录 一.安装插件 二.激活 三.使用 一.安装插件 1.1直接用离线安装,将安装包拖到 IDEA窗口就安装好了,需要重启IDEA生效 (推荐) 1.2 直接在idea上下JFormDesigner-7 关闭idea,重启一下, 右键 ------>新建 看有无JFormDesigner Form的选项,有就可以关闭idea了 二.激活 .找到idea插件安装目录 找到此路径,进入config文件夹 再进入plugins文件夹就可以看到JFormDesigner的安装文件夹 找到在JFormDe

  • 绘制flowable 流程图的Vue 库使用详解

    目录 引言 workflow-bpmn-modeler 注册 bpmnModeler 组件 muheflow-bpmn-modeler 引言 之前松哥发了一篇文章和小伙伴们介绍了前端的 bpmn.js 这个库,利用这个库我们可以自己将绘制流程图的功能嵌入到我们的项目中. 然而,这个库默认是给 Camunda 设计的,所以画出来的流程图导出来的 XML 文件无法直接使用,必须要做一些深度定制,才能将 XML 文件转为 Flowable 流程引擎可用的 XML 文件.这个深度定制太太太麻烦了. 所以

  • 零基础入门篇之Linux及Arm-Linux程序开发笔记

    前言:本文记录了自己从一个完全不懂Linux的人如何一步步学会Linux程序开发的过程.当然也希望本文能够达到它的目的,让那些和我一样没有任何基础的人也能快速入门Linux程序开发. 一.Arm-Linux程序开发平台简要介绍 Arm-Linux程序的开发并不像我们以前接触的Windows程序开发那样,关于平台的搭建就繁琐很多,所以在正式进入程序开发之前先对这种开发模式进行简要介绍,让一个即使没有任何Linux开发经验的程序员也能够看懂后面的内容. 1.1程序开发所需系统及开发语言 开发arm-

  • jQuery会死吗?我为什么不用vue写富文本

    一.事件背景: 我最近开源了一个个人耗时半年打造的富文本及一套适用于web后台的ui框架,在gitee上受到网友们的关注,部分网友对我采用jquery的技术栈提出了质疑.总结起来:无非是jquery已经落后,不久将死.甚至有少数网友很激进:非vue技术栈,你不应该加入我这个群,不管你做得多好.对应这个事情,首先我一般不反驳,因为我对vue,jquery有自己的应用场景度量.但是对于这种不分场景应用而盲目赶潮流的观点,我是持否定态度的. 为此我想有必要通过这篇文章,说明一下:我为什么用jquery

  • springboot更新配置Swagger3的一些小技巧

    1.引入依赖,版本3.0.0只引入一个即可 <dependency> <groupId>io.springfox</groupId> <artifactId>springfox-boot-starter</artifactId> <version>3.0.0</version> </dependency> 2. 配置类SwaggerConfig package org.fh.config; import org

随机推荐