Liferay7 BPM门户开发之8: Activiti实用问题集合

1、如何实现审核的上级获取(任务逐级审批)

这个是必备功能,通过Spring的注入+Activiti表达式可以很容易解决。

可参考:

http://blog.csdn.net/sunxing007/article/details/8491552

http://linhongyu.blog.51cto.com/6373370/1656596

2、如何设置流程发起人,并动态在其他任务中使用

通过变量,在启动的时候设置。

activiti里设置流程发起人的功能很绕。

activiti 指定发起人,并作为流程变量在流程中的其他任务中使用,

 在流程文件中定义开始事件
 <startEvent id="start"   activiti:initiator="initiator" />

 initiator 作为一个流程变量在其他任务节点中使用
 <userTask id="theTask" name="My Task">
     <humanPerformer>
     <resourceAssignmentExpression>
     <formalExpression>${initiator}</formalExpression>
     </resourceAssignmentExpression>
     </humanPerformer>
    </userTask>

发起流程时指定发起人:

identityService.setAuthenticatedUserId("startuser");

runtimeService.startProcessInstanceByKey("simpleProcess");

3、spring帮助TaskListener自动注入依赖

1. 如果想让spring对ExecutionListener实现依赖注入。就要让spring管理对应的bean,再从spring中获取这个bean。关键就是不能自己new一个class,如果你自己new了一个class,spring怎么知道这个class什么时候需要注入?

2. 确认spring管理好bean之后,在actviiti中,使用expression表达式引用这个bean。

不能用这种方式,这种方式就是直接new一个class,创建出的实例不是spring托管的,无法自动注入依赖:

<activiti:executionlistener< span=""><activiti:executionListener class="org.activiti.engine.test.bpmn.event.IntermediateNoneEventTest$MyExecutionListener" event="start" />

要使用expression从spring中获取托管的bean,expression部分写bean的id:

<activiti:executionlistener< span=""><activiti:executionListener delegateExpression="${myExecutionListener}" event="end" />

 4、流程图中文乱码

解决方法:设置ProcessEngineConfiguration中的属性,注意版本必须是5.13或以上版本,老版本不支持labelFontName

<property name="activityFontName" value="宋体"/>
<property name="labelFontName" value="宋体"/>

参考https://github.com/xuhuisheng/activiti-demo/blob/master/src/main/resources/spring/applicationContext-activiti.xml

衍生问题:设置了字体之后,图片依然乱码,有两种可能:

  • 没有重新部署流程,所以没有重新使用新配置生成图片。
  • 没有使用cmd的方式包装生成图片的方法,导致配置没有启用

通过封装cmd的方式调用生成图片的方法:

https://github.com/xuhuisheng/activiti-demo/blob/master/src/main/java/com/mossle/bpm/cmd/ProcessDefinitionDiagramCmd.java

如果不想用优雅的封装cmd的方法,只能手工初始化Context了。

Context.setProcessEngineConfiguration(processEngine.getProcessEngineConfiguration());

5、在流程运行中指定下一个处理人

前一个节点设置流程变量,completeTask(taskId, map)这样传递进一个变量,叫step3Assignee="nextUserName"

下一个节点设置activiti:assigne="${step3Assignee}",就实现了前一个节点指定后一个节点的负责人

手动设置任务办理人

<usertask id="hrAudit" name="人事审批" activiti:assignee="${hrUserId}"></usertask>

动态指定只要在任务完成的时候传递activiti:assignee属性中的变量即可。


Map<String, Object> variables = new HashMap<String, Object>();
variables.put("hrUserId", hrUserId);
taskService.complete(taskId, variables);

6、activiti如何与现有系统集成

要走的路任重道远:

1.集成用户账号

2.集成组织机构

3.集成权限

4.集成事务

5.集成业务数据

6.集成表单体系

7.集成服务

8.组件服务:人员、组、上级、组织架构通用选择器 

7、自定义业务流程关联

定义表单的方式在每个Task标签中定义extensionElementsactiviti:formProperty即可,到达这个节点的时候可以通过API读取表单元素。

Activiti官方的例子使用的就是在流程定义中设置每一个节点显示什么样的表单哪些字段需要显示、哪些字段只读、哪些字段必填。

但是这种方式仅仅适用于比较简单的流程,对于稍微复杂或者页面需要业务逻辑的判断的情况就不适用了。

业务和流程关联可以用

startProcessInstanceByKey

其中businessKey就是业务ID,例如要申请请假,那么先填写登记信息,然后(保存+启动流程),因为请假是单独设计的数据表,所以保存后得到实体ID就可以把它传给processInstanceBusinessKey方法启动流程。当需要根据businessKey查询流程的时候就可以通过API查询。

建议数据库冗余设计:在业务表设计的时候添加一列:PROCESS_INSTANCE_ID varchar2(64),在流程启动之后把流程ID更新到业务表中,这样不管从业务还是流程都可以查询到对方!

特别说明: 此方法启动时自动选择最新版本的流程定义。

startProcessInstanceById

javadoc对其说明:

startProcessInstanceById(String processDefinitionId, String businessKey, Map<string,object> variables) 
          Starts a new process instance in the exactly specified version of the process definition with the given id.

processDefinitionId:这个参数的值可以通过repositoryService.createProcessDefinitionQuery()方法查询,对应数据库:ACT_RE_PROCDEF;每次部署一次流程定义就会添加一条数据,同名的版本号累加。

8、查询我发起的流程

首先要通过setAuthenticatedUserid("lingo")在发起流程之前设置流程发起人。

然后可以通过processEngine.getHistoryService().createHistoricProcessInstanceQuery().starterUserid("lingo").list()

原文地址:https://www.cnblogs.com/starcrm/p/5965031.html