Activiti6.0.0 跟踪流程执行情况用红色框在流程图上标识路线跟节点

升级到Activiti6.0.0  之后,发现pvm 包整个被删掉了。。。。这样一来就导致之前的跟踪流失效了。代码连编译都通过不了。

因为pvm包没了,所以就不能再使用ActivityImpl 等相关类了。只能改成用org.activiti.bpmn.model包下的FlowNode类来替代。好在他们差不多,所以代码改动也不大。下面是完整代码:
    /**
 * @Author 葛明
 * @Note 读取流程资源
 * @Date 2017-1-3 15:11
 * @param processDefinitionId 流程定义ID
 * @param resourceName        资源名称
 */
@RequestMapping(value = "/read-resource")
public void readResource(String processDefinitionId, String resourceName,String pProcessInstanceId, HttpServletResponse response)
        throws Exception {
    // 设置页面不缓存
    response.setHeader("Pragma", "No-cache");
    response.setHeader("Cache-Control", "no-cache");
    response.setDateHeader("Expires", 0);

    ProcessDefinitionQuery pdq = repositoryService.createProcessDefinitionQuery();
    ProcessDefinition pd = pdq.processDefinitionId(processDefinitionId).singleResult();

    if(resourceName.endsWith(".png") && StringUtils.isEmpty(pProcessInstanceId) == false)
    {
        getActivitiProccessImage(pProcessInstanceId,response);
        //ProcessDiagramGenerator.generateDiagram(pde, "png", getRuntimeService().getActiveActivityIds(processInstanceId));
    }
    else
    {
        // 通过接口读取
        InputStream resourceAsStream = repositoryService.getResourceAsStream(pd.getDeploymentId(), resourceName);

        // 输出资源内容到相应对象
        byte[] b = new byte[1024];
        int len = -1;
        while ((len = resourceAsStream.read(b, 0, 1024)) != -1) {
            response.getOutputStream().write(b, 0, len);
        }
    }
}


/**
 * 获取流程图像,已执行节点和流程线高亮显示
 */
public void getActivitiProccessImage(String pProcessInstanceId, HttpServletResponse response)
{
    //logger.info("[开始]-获取流程图图像");
    try {
        //  获取历史流程实例
        HistoricProcessInstance historicProcessInstance = historyService.createHistoricProcessInstanceQuery()
                .processInstanceId(pProcessInstanceId).singleResult();

        if (historicProcessInstance == null) {
            //throw new BusinessException("获取流程实例ID[" + pProcessInstanceId + "]对应的历史流程实例失败!");
        }
        else
        {
            // 获取流程定义
            ProcessDefinitionEntity processDefinition = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService)
                    .getDeployedProcessDefinition(historicProcessInstance.getProcessDefinitionId());

            // 获取流程历史中已执行节点,并按照节点在流程中执行先后顺序排序
            List<HistoricActivityInstance> historicActivityInstanceList = historyService.createHistoricActivityInstanceQuery()
                    .processInstanceId(pProcessInstanceId).orderByHistoricActivityInstanceId().asc().list();

            // 已执行的节点ID集合
            List<String> executedActivityIdList = new ArrayList<String>();
            int index = 1;
            //logger.info("获取已经执行的节点ID");
            for (HistoricActivityInstance activityInstance : historicActivityInstanceList) {
                executedActivityIdList.add(activityInstance.getActivityId());

                //logger.info("第[" + index + "]个已执行节点=" + activityInstance.getActivityId() + " : " +activityInstance.getActivityName());
                index++;
            }

            BpmnModel bpmnModel = repositoryService.getBpmnModel(historicProcessInstance.getProcessDefinitionId());

            // 已执行的线集合
            List<String> flowIds = new ArrayList<String>();
            // 获取流程走过的线 (getHighLightedFlows是下面的方法)
            flowIds = getHighLightedFlows(bpmnModel,processDefinition, historicActivityInstanceList);



            // 获取流程图图像字符流
            ProcessDiagramGenerator pec = processEngine.getProcessEngineConfiguration().getProcessDiagramGenerator();
            //配置字体
            InputStream imageStream = pec.generateDiagram(bpmnModel, "png", executedActivityIdList, flowIds,"宋体","微软雅黑","黑体",null,2.0);

            response.setContentType("image/png");
            OutputStream os = response.getOutputStream();
            int bytesRead = 0;
            byte[] buffer = new byte[8192];
            while ((bytesRead = imageStream.read(buffer, 0, 8192)) != -1) {
                os.write(buffer, 0, bytesRead);
            }
            os.close();
            imageStream.close();
        }
        //logger.info("[完成]-获取流程图图像");
    } catch (Exception e) {
        System.out.println(e.getMessage());
        //logger.error("【异常】-获取流程图失败!" + e.getMessage());
        //throw new BusinessException("获取流程图失败!" + e.getMessage());
    }
}

public List<String> getHighLightedFlows(BpmnModel bpmnModel,ProcessDefinitionEntity processDefinitionEntity,List<HistoricActivityInstance> historicActivityInstances)
{
    SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); //24小时制
    List<String> highFlows = new ArrayList<String>();// 用以保存高亮的线flowId

    for (int i = 0; i < historicActivityInstances.size() - 1; i++)
    {
        // 对历史流程节点进行遍历
        // 得到节点定义的详细信息
        FlowNode activityImpl = (FlowNode)bpmnModel.getMainProcess().getFlowElement(historicActivityInstances.get(i).getActivityId());


        List<FlowNode> sameStartTimeNodes = new ArrayList<FlowNode>();// 用以保存后续开始时间相同的节点
        FlowNode sameActivityImpl1 = null;

        HistoricActivityInstance activityImpl_ = historicActivityInstances.get(i);// 第一个节点
        HistoricActivityInstance activityImp2_ ;

        for(int k = i + 1 ; k <= historicActivityInstances.size() - 1; k++)
        {
            activityImp2_ = historicActivityInstances.get(k);// 后续第1个节点

            if ( activityImpl_.getActivityType().equals("userTask") && activityImp2_.getActivityType().equals("userTask") &&
                    df.format(activityImpl_.getStartTime()).equals(df.format(activityImp2_.getStartTime()))   ) //都是usertask,且主节点与后续节点的开始时间相同,说明不是真实的后继节点
            {

            }
            else
            {
                sameActivityImpl1 = (FlowNode)bpmnModel.getMainProcess().getFlowElement(historicActivityInstances.get(k).getActivityId());//找到紧跟在后面的一个节点
                break;
            }

        }
        sameStartTimeNodes.add(sameActivityImpl1); // 将后面第一个节点放在时间相同节点的集合里
        for (int j = i + 1; j < historicActivityInstances.size() - 1; j++)
        {
            HistoricActivityInstance activityImpl1 = historicActivityInstances.get(j);// 后续第一个节点
            HistoricActivityInstance activityImpl2 = historicActivityInstances.get(j + 1);// 后续第二个节点

            if (df.format(activityImpl1.getStartTime()).equals(df.format(activityImpl2.getStartTime()))  )
            {// 如果第一个节点和第二个节点开始时间相同保存
                FlowNode sameActivityImpl2 = (FlowNode)bpmnModel.getMainProcess().getFlowElement(activityImpl2.getActivityId());
                sameStartTimeNodes.add(sameActivityImpl2);
            }
            else
            {// 有不相同跳出循环
                break;
            }
        }
        List<SequenceFlow> pvmTransitions = activityImpl.getOutgoingFlows() ; // 取出节点的所有出去的线

        for (SequenceFlow pvmTransition : pvmTransitions)
        {// 对所有的线进行遍历
            FlowNode pvmActivityImpl = (FlowNode)bpmnModel.getMainProcess().getFlowElement( pvmTransition.getTargetRef());// 如果取出的线的目标节点存在时间相同的节点里,保存该线的id,进行高亮显示
            if (sameStartTimeNodes.contains(pvmActivityImpl)) {
                highFlows.add(pvmTransition.getId());
            }
        }

    }
    return highFlows;

}
  • 12
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 12
    评论
### 回答1: 在 Activiti 中,结束节点流程图中的一种特殊节点,它表示流程的终止。如果想要在结束节点添加执行监听器,可以使用以下方法: 1. 在结束节点的 XML 配置文件中添加执行监听器配置: ``` <endEvent id="endevent1" name="End"> <extensionElements> <activiti:executionListener event="end" class="com.sample.MyExecutionListener"/> </extensionElements> </endEvent> ``` 2. 在 Java 代码中编写执行监听器类,实现 Activiti 的 ExecutionListener 接口,并覆盖 notifiy 方法,在该方法中编写监听器的具体逻辑。 如果在添加执行监听器后流程没有任何启动信息,可能是因为流程图的配置有误,导致流程无法正常启动。建议检查流程图的配置,确保所有节点和连线都配置正确,并检查是否有遗漏的配置。 ### 回答2: 在Activiti工作流引擎中,可以通过为结束节点添加执行监听器来实现在流程结束时执行一些特定操作。然而,在添加执行监听器后,可能会观察到流程没有启动信息的情况。 首先,我们需要理解结束节点的含义。结束节点是一个流程实例完成的地点,当流程实例到达结束节点时,即表示流程正常结束。因此,完成流程的信息已经被记录下来,不再需要显示启动的相关信息。 其次,监听器的作用是在特定的事件发生时执行一段代码逻辑。如果在结束节点添加了执行监听器,监听器的代码逻辑将在流程实例完成时执行。因此,在结束节点执行监听器之前,流程的启动信息已经被记录下来,执行监听器后不再需要显示启动的相关信息。 最后,需要注意的是,添加执行监听器并不会影响流程的正常执行,只是在流程结束时执行一些额外的操作。如果确实需要在流程启动时显示相关信息,可以在开始节点上添加执行监听器,在监听器的代码逻辑中记录启动相关信息。 总结起来,当在Activiti的结束节点添加执行监听器后,流程没有任何启动信息是正常现象,因为完成流程的信息已经被记录下来,不再需要显示启动的相关信息。如果需要显示启动信息,可以在开始节点添加执行监听器来实现。 ### 回答3: 在Activiti中,流程的结束节点是指流程的最后一个节点,一旦流程到达结束节点,表示该流程已经完成。通常情况下,结束节点不需要添加执行监听器,因为流程已经结束,不再需要执行任何操作。 然而,如果在结束节点上添加了执行监听器,那么在流程到达结束节点时将会执行监听器中定义的逻辑。根据您的描述,当您在结束节点添加了执行监听器后,流程没有任何启动信息。有以下几种可能的原因: 1. 监听器逻辑未正确配置:请检查您在结束节点上添加的执行监听器是否正确配置了监听器的类和方法。确保方法中的逻辑是否正确执行了。 2. 监听器逻辑存在异常:如果监听器中的逻辑存在异常,可能会导致流程执行监听器时出错,从而无法获得启动信息。请检查监听器中的逻辑是否正确,并尝试捕获并处理异常。 3. 流程配置错误:除了监听器的问题外,还要确保流程的其他配置没有问题。请检查整个流程的配置,确保正确配置了所有的节点和流转条件。 综上所述,如果您在结束节点上添加了执行监听器,但流程没有任何启动信息,可能是由于监听器配置错误或者流程配置错误所导致的。检查并修复这些问题将有助于解决该问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

编程序的艺术家

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值