《partner4java 讲述jBPM4》之第三步:图形化查看执行位置

本文介绍如何在JBPM中实现流程执行位置的图形化显示,包括打包PNG图片、获取流程定义及活动坐标等关键步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

图形化查看执行位置(或者称为:查看活动坐标显示在图中的图形位置)

完成起来只需要特殊两部分:

1、发布的时候打包进png图片,显示图形位置时取出(或不打包进发布,能够拼出位置也可)

如:

[java]  view plain copy
  1. InputStream in = new FileInputStream(  
  2.         "F:\\workspace\\jbpm\\workspace\\hello_jbpm\\src\\main\\resources\\leave.zip");  
  3. ZipInputStream zipInputStream = new ZipInputStream(in);  
  4. jpdlDeployManager.processEngine.getRepositoryService().createDeployment()  
  5.         .addResourcesFromZipInputStream(zipInputStream).deploy();  
取出:

[java]  view plain copy
  1. // 取出发布id中资源的io流  
  2. inputStream = repositoryService.getResourceAsStream(deploymentId, resourceName);  

2、取出当前活动坐标

什么意思呢?jbpm4为我们提供一个对象ActivityCoordinates,用于描述某个活动节点相对于图形中的坐标(横坐标、纵坐标、执行任务在图形中的高度和宽度)

[java]  view plain copy
  1. /** activity coordinates indicating the graphical position in the diagram. 
  2.  *  
  3.  * @see RepositoryService#getActivityCoordinates(String, String) 
  4.  * @author Tom Baeyens 
  5.  */  
  6. public interface ActivityCoordinates {  
  7.   
  8.   int getX();  
  9.   
  10.   int getY();  
  11.   
  12.   int getWidth();  
  13.   
  14.   int getHeight();  
  15.   
  16. }  
获取此对象:

[java]  view plain copy
  1. /** 
  2.  * 获取活动坐标显示在图中的图形位置 
  3.  *  
  4.  * @param processInstance 
  5.  *            流程实例 
  6.  * @return 
  7.  */  
  8. public static ActivityCoordinates getActivityCoordinates(ProcessInstance processInstance) {  
  9.     // 两个参数:1个为流程定义id;另一个为流程定义图形中具体活动的名称(或者说当前活动的名称)。  
  10.     return repositoryService.getActivityCoordinates(processInstance.getProcessDefinitionId(), processInstance  
  11.             .findActiveActivityNames().iterator().next());  
  12. }  


helloworld:

首先是我们的业务处理类,我们进行了封装,具体查看注释

[java]  view plain copy
  1. package com.partner4java.service;  
  2.   
  3. import java.io.BufferedOutputStream;  
  4. import java.io.File;  
  5. import java.io.FileOutputStream;  
  6. import java.io.IOException;  
  7. import java.io.InputStream;  
  8. import java.net.URL;  
  9. import java.util.concurrent.ConcurrentHashMap;  
  10. import java.util.concurrent.ConcurrentMap;  
  11.   
  12. import org.jbpm.api.Configuration;  
  13. import org.jbpm.api.ExecutionService;  
  14. import org.jbpm.api.ProcessDefinition;  
  15. import org.jbpm.api.ProcessEngine;  
  16. import org.jbpm.api.ProcessInstance;  
  17. import org.jbpm.api.RepositoryService;  
  18. import org.jbpm.api.TaskService;  
  19. import org.jbpm.api.model.ActivityCoordinates;  
  20. import org.jbpm.api.task.Task;  
  21.   
  22. /** 
  23.  * 图片展示执行位置服务类<br/> 
  24.  * 提供了两类方法:获取当前流程定义的图形和获取当前活动坐标。<br/> 
  25.  * 两类方法都分别有两份:一份是根据taskId获取,另一份是根据processInstanceId获取 
  26.  *  
  27.  * @author partner4java 
  28.  *  
  29.  */  
  30. public class PicViewService {  
  31.     public static ProcessEngine processEngine = Configuration.getProcessEngine();  
  32.     public static RepositoryService repositoryService = processEngine.getRepositoryService();  
  33.     public static ExecutionService executionService = processEngine.getExecutionService();  
  34.     public static TaskService taskService = processEngine.getTaskService();  
  35.     /** 保存已经查询过的流程定义id对应的图片地址 */  
  36.     private static ConcurrentMap<String, String> picViewNames = new ConcurrentHashMap<String, String>();  
  37.   
  38.     /** 
  39.      * 获取图片URL地址<br/> 
  40.      * 打包的png图片名字需与jpdl的name属性名字一致 
  41.      *  
  42.      * @param taskId 
  43.      *            任务id 
  44.      * @return 如:"jbpm/pic/leave_1.png",相对于web根目录的位置 
  45.      */  
  46.     public static String getPicViewByTaskId(String taskId) {  
  47.         // 根据taskId获取任务  
  48.         Task task = taskService.getTask(taskId);  
  49.   
  50.         // 根据任务获取执行id  
  51.         String executionId = task.getExecutionId();  
  52.   
  53.         return getPicView(executionId);  
  54.     }  
  55.   
  56.     /** 
  57.      * 获取图片URL地址<br/> 
  58.      * 打包的png图片名字需与jpdl的name属性名字一致 
  59.      *  
  60.      * @param processInstanceId 
  61.      *            流程实例id 
  62.      * @return 如:"jbpm/pic/leave_1.png",相对于web根目录的位置 
  63.      */  
  64.     public static String getPicView(String processInstanceId) {  
  65.         // 先根据流程实例id拿到流程实例  
  66.         ProcessInstance processInstance = executionService.findProcessInstanceById(processInstanceId);  
  67.   
  68.         // 然后根据流程实例拿到流程定义id  
  69.         String processDefinitionId = processInstance.getProcessDefinitionId();  
  70.   
  71.         String fileName = doGetPivView(processDefinitionId);  
  72.         return fileName;  
  73.     }  
  74.   
  75.     /** 
  76.      * 获取图片URL地址<br/> 
  77.      * 打包的png图片名字需与jpdl的name属性名字一致 
  78.      *  
  79.      * @param processDefinitionId 
  80.      *            流程定义id 
  81.      * @return 如:"jbpm/pic/leave_1.png",相对于web根目录的位置 
  82.      */  
  83.     private static String doGetPivView(String processDefinitionId) {  
  84.         // 先判断一下有没有取过,如果取过直接返回缓存地址  
  85.         String cachePicName = picViewNames.get(processDefinitionId);  
  86.         if (cachePicName != null) {  
  87.             return cachePicName;  
  88.         }  
  89.         // 根据流程实例id拿到流程实例  
  90.         ProcessDefinition processDefinition = repositoryService.createProcessDefinitionQuery()  
  91.                 .processDefinitionId(processDefinitionId).uniqueResult();  
  92.   
  93.         // 根据流程实例拿到发布id(和processDefinitionId的区别一般就是没有key和横杠标识)  
  94.         String deploymentId = processDefinition.getDeploymentId();  
  95.   
  96.         // 默认和name属性的名字相同  
  97.         String resourceName = processDefinition.getName() + ".png";  
  98.   
  99.         InputStream inputStream = null;  
  100.         FileOutputStream fileOutputStream = null;  
  101.         BufferedOutputStream out = null;  
  102.         String fileName = getFileName(processDefinitionId);  
  103.         try {  
  104.             // 取出发布id中资源的io流  
  105.             inputStream = repositoryService.getResourceAsStream(deploymentId, resourceName);  
  106.   
  107.             // 创建一个文件写入(这里我们做了cache),当然你也可以输出到out中response.getOutputStream().write(b,  
  108.             // 0, len);  
  109.             File file = createFile(fileName);  
  110.             fileOutputStream = new FileOutputStream(file);  
  111.             out = new BufferedOutputStream(fileOutputStream);  
  112.             byte[] b = new byte[1024];  
  113.             int len = -1;  
  114.             while ((len = inputStream.read(b, 01024)) > -1) {  
  115.                 out.write(b, 0, len);  
  116.             }  
  117.         } catch (Exception e) {  
  118.             e.printStackTrace();  
  119.         } finally {  
  120.             try {  
  121.                 inputStream.close();  
  122.             } catch (IOException e) {  
  123.                 e.printStackTrace();  
  124.             } finally {  
  125.                 try {  
  126.                     out.close();  
  127.                 } catch (IOException e) {  
  128.                     e.printStackTrace();  
  129.                 }  
  130.             }  
  131.         }  
  132.         fileName = "jbpm/pic/" + processDefinitionId.replace("-""_") + ".png";  
  133.         picViewNames.put(processDefinitionId, fileName);  
  134.         return fileName;  
  135.     }  
  136.   
  137.     /** 
  138.      * 创建文件目录 
  139.      *  
  140.      * @param fileName 
  141.      * @return 
  142.      * @throws IOException 
  143.      */  
  144.     private static File createFile(String fileName) throws IOException {  
  145.         File file = new File(fileName);  
  146.         File parentFile = file.getParentFile();  
  147.         if (parentFile != null && !parentFile.exists()) {  
  148.             parentFile.mkdirs();  
  149.         }  
  150.         return file;  
  151.     }  
  152.   
  153.     /** 
  154.      * 获取文件的磁盘路径 
  155.      *  
  156.      * @param processDefinitionId 
  157.      * @return 
  158.      */  
  159.     public static String getFileName(String processDefinitionId) {  
  160.         URL url = PicViewService.class.getClassLoader().getResource("");  
  161.         if (url != null) {  
  162.             String path = url.getPath();  
  163.             path = path.replace("/WEB-INF/classes""");  
  164.             return path + "jbpm/pic/" + processDefinitionId.replace("-""_") + ".png";  
  165.         }  
  166.         return null;  
  167.     }  
  168.   
  169.     /** 
  170.      * 获取活动坐标显示在图中的图形位置 
  171.      *  
  172.      * @param taskId 
  173.      *            任务id 
  174.      * @return 
  175.      */  
  176.     public static ActivityCoordinates getActivityCoordinatesByTaskId(String taskId) {  
  177.         // 根据taskId获取任务  
  178.         Task task = taskService.getTask(taskId);  
  179.   
  180.         // 根据任务获取执行id  
  181.         String executionId = task.getExecutionId();  
  182.         return getActivityCoordinates(executionId);  
  183.     }  
  184.   
  185.     /** 
  186.      * 获取活动坐标显示在图中的图形位置 
  187.      *  
  188.      * @param processInstanceId 
  189.      *            流程实例id 
  190.      * @return 
  191.      */  
  192.     public static ActivityCoordinates getActivityCoordinates(String processInstanceId) {  
  193.         ProcessInstance processInstance = executionService.findProcessInstanceById(processInstanceId);  
  194.         return getActivityCoordinates(processInstance);  
  195.     }  
  196.   
  197.     /** 
  198.      * 获取活动坐标显示在图中的图形位置 
  199.      *  
  200.      * @param processInstance 
  201.      *            流程实例 
  202.      * @return 
  203.      */  
  204.     public static ActivityCoordinates getActivityCoordinates(ProcessInstance processInstance) {  
  205.         // 两个参数:1个为流程定义id;另一个为流程定义图形中具体活动的名称(或者说当前活动的名称)。  
  206.         return repositoryService.getActivityCoordinates(processInstance.getProcessDefinitionId(), processInstance  
  207.                 .findActiveActivityNames().iterator().next());  
  208.     }  
  209.   
  210. }  

有了服务类,然后是我们的servlet:

[java]  view plain copy
  1. package com.partner4java.web;  
  2.   
  3. import java.io.IOException;  
  4.   
  5. import javax.servlet.ServletException;  
  6. import javax.servlet.http.HttpServlet;  
  7. import javax.servlet.http.HttpServletRequest;  
  8. import javax.servlet.http.HttpServletResponse;  
  9.   
  10. import org.jbpm.api.model.ActivityCoordinates;  
  11.   
  12. import com.partner4java.service.PicViewService;  
  13.   
  14. /** 
  15.  * 以图形的形式展示工作流执行的位置<br/> 
  16.  * 如:http://localhost:8080/jbpm_pic/view/pic?taskid=20008或http://localhost:8080/jbpm_pic/view/pic?piid=leave.20001 
  17.  * @author partner4java 
  18.  *  
  19.  */  
  20. public class PicViewServlet extends HttpServlet {  
  21.     private static final long serialVersionUID = -1307016588425325378L;  
  22.   
  23.     @Override  
  24.     protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {  
  25.         String fileName = null;  
  26.         ActivityCoordinates activityCoordinates = null;  
  27.         String processInstanceId = req.getParameter("piid");  
  28.         // 先根据流程实例id取  
  29.         if (processInstanceId != null && !"".equals(processInstanceId)) {  
  30.             fileName = PicViewService.getPicView(processInstanceId);  
  31.             activityCoordinates = PicViewService.getActivityCoordinates(processInstanceId);  
  32.         }  
  33.         // 再根据任务id  
  34.         else {  
  35.             String taskId = req.getParameter("taskid");  
  36.             if (taskId != null && !"".equals(taskId)) {  
  37.                 fileName = PicViewService.getPicViewByTaskId(taskId);  
  38.                 activityCoordinates = PicViewService.getActivityCoordinatesByTaskId(taskId);  
  39.             }  
  40.         }  
  41.   
  42.         req.setAttribute("fileName", fileName);  
  43.         req.setAttribute("activityCoordinates", activityCoordinates);  
  44.         req.getRequestDispatcher("/WEB-INF/view.jsp").forward(req, resp);  
  45.     }  
  46.   
  47. }  

最后是我们的jsp:

[html]  view plain copy
  1. <body>  
  2.     <img src="${fileName }"  
  3.         style="position:absolute;left:0px;top:0px;">  
  4.     <div  
  5.         style="position:absolute;border:1px solid red;left:${activityCoordinates.x}px;top:${activityCoordinates.y}px;width:${activityCoordinates.width}px;height:${activityCoordinates.height}px;"></div>  
  6.   
  7. </body>  
效果如图:(红框标出了执行位置)


代码下载地址:http://download.youkuaiyun.com/detail/partner4java/5013331(你也可以按照servlet注释格式的地址访问,debug进代码看执行流程和涉及到的方法)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值