#Generate JSF forms with the BPMN model API
This quickstart demonstrates how to use the BPMN model API to generate JSF forms based on the underlying process XML.
We aim to generate two generic JSF forms for a start event and a user task. The forms should be used in two different process of similar type.
Note: Please note that currently the BPMN model API is only available throug GitHub. Which means
that you have to checkout the git repository of the camunda-bpmn-model project. After this run
mvn clean install
inside the repository to install the maven artifact locally. Without the
artifact you are not able to compile this quickstart.
The support process example demonstrates a simple decision process where a user decides how to handle a support ticket. The process starts with the selection of a support ticket. After this a user decides how to handle the selected support ticket.
The feature process example also demonstrates a similar decision process where a user decides if a feature request is accepted or rejected. The process starts with the creation of a feature request. After that a user decides if the feature request is accepted or not.
The next step is to analyse the two example processes to figure out which information can be used in the process forms.
The start event in both processes has a name which describes the task of the start event. So it could be used to display as a request to the user. The name of following user task clarifies what the next step is after the start event, which could be used as a button label to submit the start form.
So a generic start event form for booth processes could be (see start-form.xhtml):
<div class="row quickstart-form">
<h:form class="form-horizontal">
<legend>#{startEventController.getStartEventName(processDefinitionKey)}</legend>
<div class="control-group">
<label class="control-label" for="inputName">Title:</label>
<div class="controls">
<h:inputText disabled="false" id="inputName" value="#{processVariables['ticket-title']}" />
</div>
</div>
<div class="control-group">
<div class="controls">
<h:commandButton id="submit_button" type="submit" value="#{startEventController.getUserTaskName(processDefinitionKey)}"
action="#{camundaTaskForm.completeProcessInstanceForm()}" styleClass="btn btn-primary" />
</div>
</div>
</h:form>
</div>
The method startEventController.getStartEventName
returns the name of the start event and startEventController.getUserTaskName
the name of the following user task. How this methods gather the required informations is explained in the next section.
The user task in both processes is followed by an exclusive gateway. The exclusive gateway name is the question which the user has to answer. While the outgoing sequence flows are labeled with the possible actions to take. So a generic approach would be to prompt the user the gateway question and the sequence flow labels as buttons. By a click on a button the user decides which sequence flow to choose.
So a generic user task form for booth processes could be (see user-form.xhtml):
<div class="row quickstart-form">
<h:form>
<fieldset class="quickstarts-buttons">
<legend>#{userTaskController.question} <#{processVariables['ticket-title']}></legend>
<ui:repeat value="#{userTaskController.buttons}" var="context">
<h:commandButton action="#{userTaskController.completeTask(context['variableName'], context['variableValue'])}" value="#{context['conditionName']}" styleClass="btn btn-large btn-primary" />
</ui:repeat>
</fieldset>
</h:form>
</div>
The method userTaskController.question
returns the name of the following exclusive gateway and userTaskController.buttons
is a list
of the corresponding sequence flow conditions. The task is completed by the userTaskController.completeTask
method which handles
the logic to choose the correct execution path which is explained in the next section.
To gather the necessary information for the generic forms the camunda BPMN model API is used.
To get the start event and user task name we need the XML representation of the current process. At the start process
form we only have the processDefinitionKey. But with the repositoryService
we can get the processId
and the corresponding
modelInstance
. With this modelInstance
we can use the BPMN model API to get the start event of the process.
private StartEvent getStartEvent(BpmnModelInstance modelInstance) {
ModelElementType startEventType = modelInstance.getModel().getType(StartEvent.class);
return (StartEvent) modelInstance.getModelElementsByType(startEventType).iterator().next();
}
The start event can we use to return its name
protected String getStartEventName(BpmnModelInstance modelInstance) {
StartEvent startEvent = getStartEvent(modelInstance);
return stripLineBreaks(startEvent.getName());
}
and to find the following user task and return its name.
public String getUserTaskName(String processDefinitionKey) {
BpmnModelInstance modelInstance = getModelInstance(processDefinitionKey);
return getUserTaskName(modelInstance);
}
To get the information for the user task form we need the following exclusive gateway and the outgoing sequence flows.
With the help of the respositoryService
we can get again the current modelInstance
. Furthermore we
can use the taskForm
to get the id of the current task, which we use to find the task and the following gateway
with the BPMN model API.
private ExclusiveGateway getExclusiveGateway(String taskId, BpmnModelInstance modelInstance) {
UserTask userTask = (UserTask) modelInstance.getModelElementById(taskId);
return (ExclusiveGateway) userTask.getSucceedingNodes().singleResult();
}
We now can return the name of the gateway
protected String getGatewayName(String taskId, BpmnModelInstance modelInstance) {
ExclusiveGateway gateway = getExclusiveGateway(taskId, modelInstance);
return stripLineBreaks(gateway.getName());
}
and analyze the outgoing sequence flows.
public List<Map<String, String>> getButtons() {
String taskId = getTaskId();
BpmnModelInstance modelInstance = getModelInstance();
return getButtons(taskId, modelInstance);
}
Every button needs a label, which is the name of the sequence flow. Also we parse the expression condition
of the sequence flow to know which variable has to be set to which value so that the correct execution path
is chosen by the process. For example a condition could be #{do == 'close'}
where the variable do
has to
be set to "close"
. Or #{action == 'reject'}
where the variable action
should be set to "reject"
.
protected static Pattern EXPRESSION_PATTERN = Pattern.compile("[\\$#]\\{\\s*(\\w+)\\s*==\\s*'([^']+)'\\s*}");
private Map<String, String> getConditionValues(SequenceFlow sequenceFlow) {
Map<String, String> values = new HashMap<String, String>();
values.put("conditionName", stripLineBreaks(sequenceFlow.getName()));
String condition = sequenceFlow.getConditionExpression().getTextContent();
Matcher matcher = EXPRESSION_PATTERN.matcher(condition);
if (matcher.matches()) {
values.put("variableName", stripLineBreaks(matcher.group(1)));
values.put("variableValue", stripLineBreaks(matcher.group(2)));
}
return values;
}
Finally to complete the task the correct variable has to be set.
public void completeTask(String variableName, String variableValue) throws IOException {
businessProcess.setVariable(variableName, variableValue);
taskForm.completeTask();
}
This are the generated forms for booth processes.
Start event form:
User task form:
Start event form:
User task form: