mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-16 17:55:15 +00:00
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@5840 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
349 lines
13 KiB
Java
349 lines
13 KiB
Java
/*
|
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
* As a special exception to the terms and conditions of version 2.0 of
|
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
|
* FLOSS exception. You should have recieved a copy of the text describing
|
|
* the FLOSS exception, and it is also available here:
|
|
* http://www.alfresco.com/legal/licensing
|
|
*/
|
|
package org.alfresco.repo.template;
|
|
|
|
import java.io.Serializable;
|
|
import java.util.ArrayList;
|
|
import java.util.Date;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
|
|
import org.alfresco.model.ApplicationModel;
|
|
import org.alfresco.model.ContentModel;
|
|
import org.alfresco.model.WCMModel;
|
|
import org.alfresco.repo.avm.AVMNodeConverter;
|
|
import org.alfresco.repo.workflow.WorkflowModel;
|
|
import org.alfresco.service.ServiceRegistry;
|
|
import org.alfresco.service.cmr.avm.AVMService;
|
|
import org.alfresco.service.cmr.avmsync.AVMDifference;
|
|
import org.alfresco.service.cmr.dictionary.DictionaryService;
|
|
import org.alfresco.service.cmr.repository.ChildAssociationRef;
|
|
import org.alfresco.service.cmr.repository.NodeRef;
|
|
import org.alfresco.service.cmr.repository.NodeService;
|
|
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
|
import org.alfresco.service.cmr.workflow.WorkflowService;
|
|
import org.alfresco.service.cmr.workflow.WorkflowTask;
|
|
import org.alfresco.service.cmr.workflow.WorkflowTaskState;
|
|
import org.alfresco.service.cmr.workflow.WorkflowTransition;
|
|
import org.alfresco.service.namespace.QName;
|
|
import org.alfresco.service.namespace.QNameMap;
|
|
import org.alfresco.service.namespace.RegexQNamePattern;
|
|
|
|
/**
|
|
* Workflow and task support in FreeMarker templates.
|
|
*
|
|
* @author Kevin Roast
|
|
*/
|
|
public class Workflow extends BaseTemplateProcessorExtension
|
|
{
|
|
private static final String WCM_WF_MODEL_1_0_URI = "http://www.alfresco.org/model/wcmworkflow/1.0";
|
|
private static final QName PROP_FROM_PATH = QName.createQName(WCM_WF_MODEL_1_0_URI, "fromPath");
|
|
|
|
private ServiceRegistry services;
|
|
|
|
/**
|
|
* Sets the service registry
|
|
*
|
|
* @param services the service registry
|
|
*/
|
|
public void setServiceRegistry(ServiceRegistry services)
|
|
{
|
|
this.services = services;
|
|
}
|
|
|
|
/**
|
|
* Return a list of objects representing the assigned tasks for the current user
|
|
*
|
|
* @return list of WorkflowTaskItem bean objects {@link WorkflowTaskItem}
|
|
*/
|
|
public List<WorkflowTaskItem> getAssignedTasks()
|
|
{
|
|
// get the "in progress" tasks for the current user
|
|
List<WorkflowTask> tasks = getWorkflowService().getAssignedTasks(
|
|
this.services.getAuthenticationService().getCurrentUserName(),
|
|
WorkflowTaskState.IN_PROGRESS);
|
|
|
|
return convertTasks(tasks);
|
|
}
|
|
|
|
/**
|
|
* Return a list of objects representing the pooled tasks for the current user
|
|
*
|
|
* @return list of WorkflowTaskItem bean objects {@link WorkflowTaskItem}
|
|
*/
|
|
public List<WorkflowTaskItem> getPooledTasks()
|
|
{
|
|
// get the "pooled" tasks for the current user
|
|
List<WorkflowTask> tasks = getWorkflowService().getPooledTasks(
|
|
this.services.getAuthenticationService().getCurrentUserName());
|
|
|
|
return convertTasks(tasks);
|
|
}
|
|
|
|
/**
|
|
* Return a list of objects representing the completed tasks for the current user
|
|
*
|
|
* @return list of WorkflowTaskItem bean objects {@link WorkflowTaskItem}
|
|
*/
|
|
public List<WorkflowTaskItem> getCompletedTasks()
|
|
{
|
|
// get the "completed" tasks for the current user
|
|
List<WorkflowTask> tasks = getWorkflowService().getAssignedTasks(
|
|
this.services.getAuthenticationService().getCurrentUserName(),
|
|
WorkflowTaskState.COMPLETED);
|
|
|
|
return convertTasks(tasks);
|
|
}
|
|
|
|
/**
|
|
* Return a single object representing a task of the given taskId for the current user
|
|
*
|
|
* @return WorkflowTaskItem bean object {@link WorkflowTaskItem}
|
|
*/
|
|
public WorkflowTaskItem getTaskById(String taskId)
|
|
{
|
|
// get the task corresponding to the given taskId
|
|
WorkflowTask task = getWorkflowService().getTaskById(taskId);
|
|
|
|
return new WorkflowTaskItem(this.services, getTemplateImageResolver(), task);
|
|
}
|
|
|
|
/**
|
|
* Convert a list of WorkflowTask items into bean objects accessable from templates
|
|
*
|
|
* @param tasks List of WorkflowTask objects to convert
|
|
*
|
|
* @return List of WorkflowTaskItem bean wrapper objects
|
|
*/
|
|
private List<WorkflowTaskItem> convertTasks(List<WorkflowTask> tasks)
|
|
{
|
|
List<WorkflowTaskItem> items = new ArrayList<WorkflowTaskItem>(tasks.size());
|
|
for (WorkflowTask task : tasks)
|
|
{
|
|
items.add(new WorkflowTaskItem(this.services, getTemplateImageResolver(), task));
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
private WorkflowService getWorkflowService()
|
|
{
|
|
return this.services.getWorkflowService();
|
|
}
|
|
|
|
|
|
/**
|
|
* Simple bean wrapper around a WorkflowTask item
|
|
*/
|
|
public static class WorkflowTaskItem
|
|
{
|
|
private WorkflowTask task;
|
|
private QNameMap<String, Serializable> properties = null;
|
|
private ServiceRegistry services;
|
|
private TemplateImageResolver resolver;
|
|
|
|
public WorkflowTaskItem(ServiceRegistry services, TemplateImageResolver resolver, WorkflowTask task)
|
|
{
|
|
this.task = task;
|
|
this.services = services;
|
|
this.resolver = resolver;
|
|
}
|
|
|
|
public String getType()
|
|
{
|
|
return this.task.title;
|
|
}
|
|
|
|
public String getQnameType()
|
|
{
|
|
return this.task.definition.metadata.getName().toString();
|
|
}
|
|
|
|
public String getName()
|
|
{
|
|
return this.task.description;
|
|
}
|
|
|
|
public String getDescription()
|
|
{
|
|
return this.task.path.instance.description;
|
|
}
|
|
|
|
public String getId()
|
|
{
|
|
return this.task.id;
|
|
}
|
|
|
|
public boolean getIsCompleted()
|
|
{
|
|
return (this.task.state == WorkflowTaskState.COMPLETED);
|
|
}
|
|
|
|
public Date getStartDate()
|
|
{
|
|
return this.task.path.instance.startDate;
|
|
}
|
|
|
|
public Map<String, String>[] getTransitions()
|
|
{
|
|
Map<String, String>[] tranMaps = null;
|
|
WorkflowTransition[] transitions = this.task.definition.node.transitions;
|
|
if (transitions != null)
|
|
{
|
|
tranMaps = new HashMap[transitions.length];
|
|
for (int i=0; i<transitions.length; i++)
|
|
{
|
|
tranMaps[i] = new HashMap<String, String>(2, 1.0f);
|
|
tranMaps[i].put("label", transitions[i].title);
|
|
tranMaps[i].put("id", transitions[i].id);
|
|
}
|
|
}
|
|
return (tranMaps != null ? tranMaps : new HashMap[0]);
|
|
}
|
|
|
|
/**
|
|
* @return A TemplateNode representing the initiator (person) of the workflow
|
|
*/
|
|
public TemplateNode getInitiator()
|
|
{
|
|
return new TemplateNode(this.task.path.instance.initiator, this.services, this.resolver);
|
|
}
|
|
|
|
/**
|
|
* @return The workflow package ref
|
|
*/
|
|
public NodeRef getPackage()
|
|
{
|
|
return (NodeRef)this.task.properties.get(WorkflowModel.ASSOC_PACKAGE);
|
|
}
|
|
|
|
/**
|
|
* @return the resources from the package attached to this workflow task
|
|
*/
|
|
public List<TemplateContent> getPackageResources()
|
|
{
|
|
List<TemplateContent> resources = new ArrayList<TemplateContent>();
|
|
if (this.task.properties.get(PROP_FROM_PATH) != null)
|
|
{
|
|
AVMService avmService = this.services.getAVMService();
|
|
NodeRef workflowPackage = getPackage();
|
|
NodeRef stagingNodeRef = (NodeRef)this.services.getNodeService().getProperty(
|
|
workflowPackage, WCMModel.PROP_AVM_DIR_INDIRECTION);
|
|
String stagingAvmPath = AVMNodeConverter.ToAVMVersionPath(stagingNodeRef).getSecond();
|
|
String packageAvmPath = AVMNodeConverter.ToAVMVersionPath(workflowPackage).getSecond();
|
|
for (AVMDifference d : this.services.getAVMSyncService().compare(
|
|
-1, packageAvmPath, -1, stagingAvmPath, null))
|
|
{
|
|
if (d.getDifferenceCode() == AVMDifference.NEWER ||
|
|
d.getDifferenceCode() == AVMDifference.CONFLICT)
|
|
{
|
|
resources.add(new AVMTemplateNode(
|
|
d.getSourcePath(), d.getSourceVersion(), this.services, this.resolver));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// get existing workflow package items
|
|
NodeService nodeService = this.services.getNodeService();
|
|
DictionaryService ddService = this.services.getDictionaryService();
|
|
List<ChildAssociationRef> childRefs = nodeService.getChildAssocs(
|
|
getPackage(), ContentModel.ASSOC_CONTAINS, RegexQNamePattern.MATCH_ALL);
|
|
|
|
for (ChildAssociationRef ref: childRefs)
|
|
{
|
|
// create our Node representation from the NodeRef
|
|
NodeRef nodeRef = ref.getChildRef();
|
|
if (nodeService.exists(nodeRef))
|
|
{
|
|
// find it's type so we can see if it's a node we are interested in
|
|
QName type = nodeService.getType(nodeRef);
|
|
|
|
// make sure the type is defined in the data dictionary
|
|
if (ddService.getType(type) != null)
|
|
{
|
|
// look for content nodes or links to content
|
|
// NOTE: folders within workflow packages are ignored for now
|
|
if (ddService.isSubClass(type, ContentModel.TYPE_CONTENT) ||
|
|
ApplicationModel.TYPE_FILELINK.equals(type))
|
|
{
|
|
resources.add(new TemplateNode(nodeRef, this.services, this.resolver));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return resources;
|
|
}
|
|
|
|
/**
|
|
* @return the 'outcome' label from a completed task
|
|
*/
|
|
public String getOutcome()
|
|
{
|
|
String outcome = null;
|
|
if (task.state.equals(WorkflowTaskState.COMPLETED))
|
|
{
|
|
// add the outcome label for any completed task
|
|
String transition = (String)task.properties.get(WorkflowModel.PROP_OUTCOME);
|
|
if (transition != null)
|
|
{
|
|
WorkflowTransition[] transitions = this.task.definition.node.transitions;
|
|
for (WorkflowTransition trans : transitions)
|
|
{
|
|
if (trans.id.equals(transition))
|
|
{
|
|
outcome = trans.title;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return outcome;
|
|
}
|
|
|
|
/**
|
|
* @return A map of properties for the workflow task, includes all appropriate bpm model properties.
|
|
*/
|
|
public Map<String, Serializable> getProperties()
|
|
{
|
|
if (this.properties == null)
|
|
{
|
|
// convert properties to a QName accessable Map with TemplateNode objects as required
|
|
PropertyConverter converter = new PropertyConverter();
|
|
this.properties = new QNameMap<String, Serializable>(this.services.getNamespaceService());
|
|
for (QName qname : this.task.properties.keySet())
|
|
{
|
|
Serializable value = converter.convertProperty(
|
|
this.task.properties.get(qname), qname, this.services, this.resolver);
|
|
this.properties.put(qname.toString(), value);
|
|
}
|
|
}
|
|
return this.properties;
|
|
}
|
|
}
|
|
}
|