Activiti系列——设计指定Assignee+审批

Posted 邱慕夏

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Activiti系列——设计指定Assignee+审批相关的知识,希望对你有一定的参考价值。

上面的三篇博客中,主要是说Activiti比较基本的,这个是我们项目中设计的Activiti的Designer。


主要解决的问题:


1、提交人可以是任何人。这里我们用变量。


2、审批人用Listener实现,这里我们用Listener,并保存角色,凡有这个角色的人员都可以进行审批。


3、审批有两种参数:批准和驳回。


我们看看我们的流程图:


技术分享


一、画Intershipone.bpmn


连线就不用说了,跟第二篇是一样的。我们来说之前没有说过的。


在提交申请节点上,我刚刚已经说过了,这里提交人可以是任何人,这个时候,我们用Listener其实也可以,但是是很浪费的一件事情,因此,我们这里就可以使用一个变量就可以了。


技术分享



这样我们只需要给Assignee一个map的参数就可以了。这个inputUser可以是用户账号。


技术分享


我们在节点:学术审批和老总审批中都使用taskListenerImpl.java这个TaskListener,逻辑我们就在notify里面写就好了。


二、TaskListenerImpl.java类


<span style="font-size:18px;">package com.tgb.itoo.basic.controller;

import java.util.List;
import java.util.Map;

import org.activiti.engine.ProcessEngine;
import org.activiti.engine.ProcessEngines;
import org.activiti.engine.delegate.DelegateTask;
import org.activiti.engine.delegate.TaskListener;
import org.activiti.engine.history.HistoricTaskInstance;
import org.activiti.engine.task.Task;

@SuppressWarnings("serial")
public class TaskListenerImpl implements TaskListener {

	/**用来指定任务的办理人*/
	@Override
	public void notify(DelegateTask delegateTask) {
		//指定个人任务的办理人,也可以指定组任务的办理人
		String executeInstanceId = delegateTask.getProcessInstanceId();
		
		String nextLevel="Nobody";
		
//		IntershipBeanImpl intershipbeanimpl = new IntershipBeanImpl();
//		//个人任务:通过类去查询数据库,将下一个任务的办理人查询获取,然后通过setAssignee()的方法指定任务的办理人
//		String assignee = intershipbeanimpl.findAssigneeByExecutionID("itoo_internship", executeInstanceId);	
		
		ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
		
		//根据流程实例查询任务列表[
		List<Task> listtask = processEngine.getTaskService().createTaskQuery().executionId(executeInstanceId).list();
		
		if(listtask.size() > 0 && listtask!=null){
			
			String lastassignee = listtask.get(0).getAssignee();   //取得ID
			
			String prcessName = listtask.get(0).getProcessDefinitionId();   
			
			String[] processDefinitionid =  prcessName.split(":"); 
			String processDefId = processDefinitionid[0];          //就去intershipone
					
			List<HistoricTaskInstance> list = processEngine.getHistoryService().createHistoricTaskInstanceQuery().executionId(executeInstanceId).list();
			
			if(list.size()==1){
				//如果任务的个数为1的话,则为第一个角色
				
				String sql = "select * From tb_relation where intershipid='" + processDefId + "'";
				
				Dbhelper dbCourseCurrentadd = new Dbhelper();
				dbCourseCurrentadd.setSql(sql.toString());		 
				List<Map<String,Object>> listmap =  dbCourseCurrentadd.findItemById();
				if(listmap.size() > 0 && listmap!=null){
					 for(int i=0;i<listmap.size();i++){
						 Map<String, Object> m = listmap.get(i);
						 String Next_ASSIGNEE_ROLE_CHILD = (String) m.get("childCode");     //拿到第一个childcode
						 boolean IS_FIRTHROLE = true;                                      //假设它是第一个
							 for(int j=0;j<listmap.size();j++){
								 Map<String, Object> mnext = listmap.get(j);
								 String Next_ASSIGNEE_ROLE_FATHER = (String) mnext.get("fatherCode");      //循环fathercode是否与childcode
								 if(Next_ASSIGNEE_ROLE_CHILD.equals(Next_ASSIGNEE_ROLE_FATHER)){
									 IS_FIRTHROLE = false;                                                 //一定相等则不是
									 continue;
								 }						 
							 }
						 if(IS_FIRTHROLE == true){
							  nextLevel = (String) m.get("fatherCode");
							  break;
						 }
					 }
				}
			}else{
				//如果任务的个数大于1的话,则ID为RoleID
				String sql = "select * From tb_relation where intershipid='" + processDefId + "' and childCode='" + lastassignee + "'";
				
				Dbhelper dbCourseCurrentadd = new Dbhelper();
				 dbCourseCurrentadd.setSql(sql.toString());		 
				 List<Map<String,Object>> listmap =  dbCourseCurrentadd.findItemById();
				 
				 if(listmap.size() > 0 && listmap!=null){
					 Map<String, Object> m = listmap.get(0);   	
					 nextLevel = (String) m.get("fatherCode");
				 }
				 
			}
			
		}
	
		
		System.out.println(nextLevel);	
		
		delegateTask.setAssignee(nextLevel);
	}

}
</span>

这里的审批人,我们保存的是角色,这样只要有这个角色的人,都可以看到这个Task,这样,就完美的解决了,一个任务多个人可以进行审批的问题。


注:


这里我要说一个问题,这个问题,当时我用了很长的时间类解决这个问题,因为我的前台用的是Spring MVC,Spring MVC 调用EJB,因此,按照架构来讲,我这里的所有的业务逻辑以及对数据库的操作都应该在EJB的Service和DAO中,进行的,但是,我将用EJB没有办法在TaskListener中进行获取,都不行,查了一些资料,都没有解决,之后,我就直接对数据库的操作和业务逻辑都放在了DbHelper中,然后,直接放在了Web层,各位大神们,如果能够解释是否EJB和TaskListener能不能同时使用这个问题,一定要给我留言奥。


三、IntershipApplicationController.java类


<span style="font-size:18px;">package com.tgb.itoo.basic.controller;

import java.io.InputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.activiti.engine.ProcessEngine;
import org.activiti.engine.ProcessEngines;
import org.activiti.engine.repository.Deployment;
import org.activiti.engine.runtime.ProcessInstance;
import org.activiti.engine.task.Task;
import org.junit.Test;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

import com.tgb.itoo.basic.service.IntershipBean;

@Controller
public class IntershipApplicationController {


	
	private IntershipBean intershipBean;

	public IntershipBean getIntershipBean() {
		return intershipBean;
	}

	public void setIntershipBean(IntershipBean intershipBean) {
		this.intershipBean = intershipBean;
	}
	
	@RequestMapping("/startIntership")
    public String startIntership(HttpServletRequest request,
            HttpServletResponse response){

		return "/DictionaryList";

	}
	

	// 启动流程实例,也就是人们申请实习
	
	public void deploymentProcessDefinition_inputStream(HttpServletRequest request,
				HttpServletResponse response) {
		System.out.println("qmx");
		
		//根据流程实例查询任务列表
		processEngine.getTaskService().createTaskQuery().orderByTaskAssignee().taskAssignee("0");
	
	}
	
	
	
	ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
	
	/**部署流程定义(从inputStream)*/
	@Test
	@RequestMapping("/deploymentProcessDefinition_inputStream")
	public void deploymentProcessDefinition_inputStream(){
		InputStream inputStreamBpmn = this.getClass().getResourceAsStream("/programs/Intershipone.bpmn");
		InputStream inputStreamPng = this.getClass().getResourceAsStream("/programs/Intershipone.png");
		Deployment deployment = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service
						.createDeployment()//创建一个部署对象
						.name("邱慕夏任务")//添加部署的名称
						.addInputStream("/programs/Intershipone.bpmn", inputStreamBpmn)//
						.addInputStream("/programs/Intershipone.png", inputStreamPng)//
						.deploy();//完成部署
		System.out.println("部署ID:"+deployment.getId());//
		System.out.println("部署名称:"+deployment.getName());//
	}
	
	/**启动流程实例*/
	@Test
	@RequestMapping("/startProcessInstance")
	public void startProcessInstance(){
		//流程定义的key
		String processDefinitionKey = "intershiptone";
		/**启动流程实例的同时,设置流程变量,使用流程变量用来指定任务的办理人,对应task.pbmn文件中#{userID}*/
		Map<String, Object> variables = new HashMap<String, Object>();
		variables.put("inputUser", "168BGELgzyDkbcVRWVkfBB");  //放入邱慕夏的ID
		ProcessInstance pi = processEngine.getRuntimeService()//与正在执行的流程实例和执行对象相关的Service
						.startProcessInstanceByKey(processDefinitionKey,variables);//使用流程定义的key启动流程实例,key对应helloworld.bpmn文件中id的属性值,使用key值启动,默认是按照最新版本的流程定义启动
		System.out.println("流程实例ID:"+pi.getId());//流程实例ID    101
		System.out.println("流程定义ID:"+pi.getProcessDefinitionId());//流程定义ID   helloworld:1:4
	}
	
	
	
	/**查询当前人的个人任务*/
	@Test
	@RequestMapping("/findMyPersonalTask")
	public void findMyPersonalTask(){
		String assignee = "168BGELgzyDkbcVRWVkfBB";
		List<Task> list = processEngine.getTaskService()//与正在执行的任务管理相关的Service
						.createTaskQuery()//创建任务查询对象
						/**查询条件(where部分)*/
						.taskAssignee(assignee)//指定个人任务查询,指定办理人
						/**排序*/
						.orderByTaskCreateTime().asc()//使用创建时间的升序排列
						/**返回结果集*/
						.list();//返回列表
		if(list!=null && list.size()>0){
			for(Task task:list){
				System.out.println("任务ID:"+task.getId());
				System.out.println("任务名称:"+task.getName());
				System.out.println("任务的创建时间:"+task.getCreateTime());
				System.out.println("任务的办理人:"+task.getAssignee());
				System.out.println("流程实例ID:"+task.getProcessInstanceId());
				System.out.println("执行对象ID:"+task.getExecutionId());
				System.out.println("流程定义ID:"+task.getProcessDefinitionId());
				System.out.println("########################################################");
			}
		}
	}
	
	
	/**完成我的任务*/
	@Test
	@RequestMapping("/completeMyPersonalTaskSuquest")
	public void completeMyPersonalTaskSuquest(){
		//任务ID
		String taskId = "9618";
		String IsAgree = "批准";
		String IsNotAgree = "驳回";
		String IsStudent = "";
		
		//完成任务的同时,设置流程变量,使用流程变量用来指定完成任务后,下一个连线,对应sequenceFlow.bpmn文件中${message=='不重要'}
		Map<String, Object> variables = new HashMap<String, Object>();
		variables.put("outcome", IsAgree);
		processEngine.getTaskService()//与正在执行的任务管理相关的Service
					.complete(taskId,variables);
		System.out.println("完成任务:任务ID:"+taskId);
	
		
	}	
	
}
</span>

上面的方法很简单,大家看看代码很容易明白,不多说了。


欢迎大家给我提问题,咱们一起交流,一起学习。



































以上是关于Activiti系列——设计指定Assignee+审批的主要内容,如果未能解决你的问题,请参考以下文章

activiti 7 + springboot2 Activiti流程实例动态设置assignee执行人

activiti 会签多实例任务,设置为候选组或个人任务的总结

activiti入门学习二

Activiti动态设置办理人扩展

Activiti7工作流引擎进阶

Activiti动态设置办理人扩展