学习资料:《Activiti实战》

第六章 任务表单(二)外置表单

6.3 外置表单

考虑到动态表单的缺点(见上节),外置表单使用的更多。

外置表单的特点:

 页面的原样显示
字段值的自动填充

6.3.1 流程定义

(1)form文件

leave-start.form作为示例展示(字段要和后面代码中variables变量的key互相对应):

 <div class="control-group">
<label class="control-label" for="startDate">开始时间:</label>
<div class="controls">
<input type="text" id="startDate" name="startDate" class="datepicker" data-date-format="yyyy-mm-dd" required />
</div>
</div>
<div class="control-group">
<label class="control-label" for="endDate">结束时间:</label>
<div class="controls">
<input type="text" id="endDate" name="endDate" class="datepicker" data-date-format="yyyy-mm-dd" required />
</div>
</div>
<div class="control-group">
<label class="control-label" for="reason">请假原因:</label>
<div class="controls">
<textarea id="reason" name="reason" required></textarea>
</div>
</div>

(2)流程文件

这里只显示部分xml内容,其他的一些见上节动态表单。这里的xml文件只是为了展示外置表单的使用方法。

基本使用方式就是:activiti:formkey="chapter6/leave-formkey/approve.form" 

form的值支持动态设置:activiti:formkey="${fooFormName}.form"

 <process id="leave-formkey" name="请假流程-外置表单">
<startEvent id="startevent1" name="Start"
activiti:initiator="applyUserId"
activiti:formkey="chapter6/leave-formkey/leave-start.form">
</startEvent>
<userTask id="deptLeaderVerify" name="部门经理审批"
activiti:candidateGroups="deptLeader"
activiti:formkey="chapter6/leave-formkey/approve.form">
</userTask>
<userTask id="hrVerify" name="人事审批"
activiti:candidateGroups="hr"
activiti:formkey="chapter6/leave-formkey/approve.form">
</userTask>
<userTask id="reportBack" name="销假"
activiti:assignee="${applyUserId}"
activiti:formkey="chapter6/leave-formkey/report-back.form">
</userTask>
<userTask id="modifyApply" name="调整申请内容"
activiti:assignee="${applyUserId}"
activiti:formkey="chapter6/leave-formkey/modify-apply.form">
</userTask>
<endEvent id="endevent1" name="End"
</endEvent>
</process>

(3)单元测试

部署表单流程时需要把bpmn文件和form文件同时打包部署。这样部署了同名的form文件时多个流程定义,或相同流程不同版本之间,都不会有冲突。

 public class LeaveFormKeyTest extends AbstractTest {

     @Test
@Deployment(resources = {"chapter6/leave-formkey/leave-formkey.bpmn",
"chapter6/leave-formkey/leave-start.form",
"chapter6/leave-formkey/approve-deptLeader.form",
"chapter6/leave-formkey/approve-hr.form",
"chapter6/leave-formkey/report-back.form",
"chapter6/leave-formkey/modify-apply.form"}) public void allPass() throws Exception {
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Map<String, String> variables = new HashMap<String, String>();
Calendar ca = Calendar.getInstance();
String startDate = sdf.format(ca.getTime());
ca.add(Calendar.DAY_OF_MONTH, 2); // 当前日期加2天
String endDate = sdf.format(ca.getTime()); // 启动流程
variables.put("startDate", startDate);
variables.put("endDate", endDate);
variables.put("reason", "公休"); ProcessDefinition processDefinition = repositoryService.createProcessDefinitionQuery().singleResult(); // 读取启动表单
Object renderedStartForm = formService.getRenderedStartForm(processDefinition.getId());
assertNotNull(renderedStartForm); // 启动流程
// 设置当前用户
String currentUserId = "henryyan";
identityService.setAuthenticatedUserId(currentUserId);
ProcessInstance processInstance = formService.submitStartFormData(processDefinition.getId(), variables);
assertNotNull(processInstance); // 部门领导审批通过
Task deptLeaderTask = taskService.createTaskQuery().taskCandidateGroup("deptLeader").singleResult();
assertNotNull(formService.getRenderedTaskForm(deptLeaderTask.getId()));
variables = new HashMap<String, String>();
variables.put("deptLeaderApproved", "true");
formService.submitTaskFormData(deptLeaderTask.getId(), variables); // 人事审批通过
Task hrTask = taskService.createTaskQuery().taskCandidateGroup("hr").singleResult();
assertNotNull(formService.getRenderedTaskForm(hrTask.getId()));// 读取任务表单
variables = new HashMap<String, String>();
variables.put("hrApproved", "true");
formService.submitTaskFormData(hrTask.getId(), variables);

// 销假(根据申请人的用户ID读取)
Task reportBackTask = taskService.createTaskQuery().taskAssignee(currentUserId).singleResult();
assertNotNull(formService.getRenderedTaskForm(reportBackTask.getId()));
variables = new HashMap<String, String>();
variables.put("reportBackDate", sdf.format(ca.getTime()));
formService.submitTaskFormData(reportBackTask.getId(), variables); // 验证流程是否已经结束
HistoricProcessInstance historicProcessInstance = historyService.createHistoricProcessInstanceQuery().finished().singleResult();
assertNotNull(historicProcessInstance); // 读取历史变量
Map<String, Object> historyVariables = packageVariables(processInstance); // 验证执行结果
assertEquals("ok", historyVariables.get("result")); }
...
}

6.3.2 自定义表单引擎

activiti既可以可以支持B/S结构的应用,也可以支持C/S结构的应用。getRenderd***Form()返回的内容是经过activiti的默认Form引擎处理过的,返回的值可以让B/S结构的应用直接使用,但是却不能直接支持C/S结构的应用。所以如果要生成C/S程序需要的java控件,需要事先自定义的form引擎

这一块暂时用不到,略过。

6.3.3 读取流程启动表单

Activiti Explorer支持动态表单,却不支持外置表单。所以需要为Activiti Explorer增加外置表单支持。

2017.2.28 activiti实战--第六章--任务表单(二)外置表单的更多相关文章

  1. 2017.2.28 activiti实战--第六章--任务表单(一)动态表单

    学习资料:<Activiti实战> 第六章 任务表单(一)动态表单 内容概览:本章要完成一个OA(协同办公系统)的请假流程的设计,从实用的角度,讲解如何将activiti与业务紧密相连. ...

  2. 2017.2.28 activiti实战--第七章--Spring容器集成应用实例(五)普通表单

    学习资料:<Activiti实战> 第七章  Spring容器集成应用实例(五)普通表单 第六章中介绍了动态表单.外置表单.这里讲解第三种表单:普通表单. 普通表单的特点: 把表单内容写在 ...

  3. 2017.2.22 activiti实战--第六章--任务表单

    学习资料:<Activiti实战> 第六章 任务表单 本章将一步步完成一个协同办公系统(OA)的请假流程的设计,讲解如何将Activiti和实际业务联系起来. 首先讲解动态表单与外置表单的 ...

  4. 2017.2.28 activiti实战--第五章--用户与组及部署管理(三)部署流程及资源读取

    学习资料:<Activiti实战> 第五章 用户与组及部署管理(三)部署流程及资源读取 内容概览:如何利用API读取已经部署的资源,比如读取流程定义的XML文件,或流程对应的图片文件. 以 ...

  5. 2017.2.28 activiti实战--第五章--用户与组及部署管理(二)部署流程资源

    学习资料:<Activiti实战> 第五章 用户与组及部署管理(二)部署流程资源 内容概览:讲解流程资源的读取与部署. 5.2 部署流程资源 5.2.1 流程资源 流程资源常用的有以下几种 ...

  6. 2017.2.21 activiti实战--第十三章--流量数据查询与跟踪(一)查询接口介绍及运行时数据查询

    学习资料:<Activiti实战> 第十三章 流量数据查询与跟踪 本章讲解运行时与历史数据的查询方法.主要包含三种:标准查询,Native查询,CustomSql查询. 13.1 Quer ...

  7. 2017.2.21 activiti实战--第七章--Activiti与spring集成(一)配置文件

    学习资料:<Activiti实战> 第七章 Activiti与容器集成 本章讲解activiti-spring可以做的事情,如何与现有系统集成,包含bean的注入.统一事务管理等. 7.1 ...

  8. 2017.2.20 activiti实战--第五章--用户与组及部署管理(一)用户与组

    学习资料:<Activiti实战> 第五章 用户与组及部署管理(一)用户与组 内容概览:讲解activiti中内置的一套用户.组的关系,以及如何通过API添加.删除.查询. 5.1 用户与 ...

  9. 2017.2.20 activiti实战--第二章--搭建Activiti开发环境及简单示例(二)简单示例

    学习资料:<Activiti实战> 第一章 搭建Activiti开发环境及简单示例 2.5 简单流程图及其执行过程 (1)leave.bpmn 后缀名必须是bpmn.安装了activiti ...

随机推荐

  1. lcd1602如何自定义汉字(verilog)

    今天一鼓作气,再研究了一下如何用LCD1602自定义汉字 1.用字模软件获取汉字所对应的数据(因为嫌麻烦所以直接用了网上一个帖子里有关“电”的数据,如下:04,1f,15,1f,15,15,1f,04 ...

  2. mysql物理备份恢复 xtrabackup 初试

    听闻xtrabackup开源且强大 2018-03-06 11:54:41 在官网下载安装了最新的2.4.9版本 网上文章都用的innobackupex,但是最新版已经抛弃了,自己看看手册<Pe ...

  3. iOS中常见的自定义宏

    //字符串是否为空 #define kStringIsEmpty(str) ([str isKindOfClass:[NSNull class]] || str == nil || [str leng ...

  4. libcmt.lib和msvcrt.lib冲突,原因和解决方法

    libcmt.lib和msvcrt.lib冲突,原因和解决方法 https://blog.csdn.net/longlijun/article/details/7331093 libcmt.lib是w ...

  5. token防止重复提交

    Token,可以翻译成标记!最大的特点就是随机性,不可预测,一般黑客或软件无法猜测出来. Token一般用在两个地方: 1: 防止表单重复提交 2: anti csrf攻击(Cross-site re ...

  6. 关于ofbiz13.07和ofbiz14.12部署的问题解决方法

    必备软件 Jdk7 文件名:java_ee_sdk-6u4-jdk7-windows.exe eclipse MySQL Installer 5.6 for Windows MySql Connect ...

  7. 球形空间产生器sphere(bzoj 1013)

    Description 有一个球形空间产生器能够在n维空间中产生一个坚硬的球体.现在,你被困在了这个n维球体中,你只知道球面上n+1个点的坐标,你需要以最快的速度确定这个n维球体的球心坐标,以便于摧毁 ...

  8. pat 甲级 1066. Root of AVL Tree (25)

    1066. Root of AVL Tree (25) 时间限制 100 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue An A ...

  9. codechef AUG17 T4 Palindromic Game

    Palindromic Game Problem Code: PALINGAM There are two players A, B playing a game. Player A has a st ...

  10. 【Educational Codeforces Round 53 (Rated for Div. 2)】

    A:https://www.cnblogs.com/myx12345/p/9853775.html B:https://www.cnblogs.com/myx12345/p/9853779.html ...