Allow passing a widget class to SourceColumn. This makes using lambdas to render model column unnecesary and are mostly removed too. Signed-off-by: Roberto Rosario <Roberto.Rosario@mayan-edms.com>
910 lines
29 KiB
Python
910 lines
29 KiB
Python
from __future__ import absolute_import, unicode_literals
|
|
|
|
from django.contrib import messages
|
|
from django.db import transaction
|
|
from django.http import Http404, HttpResponseRedirect
|
|
from django.shortcuts import get_object_or_404
|
|
from django.template import RequestContext
|
|
from django.urls import reverse, reverse_lazy
|
|
from django.utils.translation import ugettext_lazy as _
|
|
|
|
from mayan.apps.acls.models import AccessControlList
|
|
from mayan.apps.common.views import (
|
|
AssignRemoveView, ConfirmView, FormView, SingleObjectCreateView,
|
|
SingleObjectDeleteView, SingleObjectDetailView,
|
|
SingleObjectDynamicFormCreateView, SingleObjectDynamicFormEditView,
|
|
SingleObjectEditView, SingleObjectListView
|
|
)
|
|
from mayan.apps.documents.models import Document
|
|
from mayan.apps.documents.views import DocumentListView
|
|
from mayan.apps.events.classes import EventType
|
|
from mayan.apps.events.models import StoredEventType
|
|
|
|
from .classes import WorkflowAction
|
|
from .forms import (
|
|
WorkflowActionSelectionForm, WorkflowForm, WorkflowInstanceTransitionForm,
|
|
WorkflowPreviewForm, WorkflowStateActionDynamicForm, WorkflowStateForm,
|
|
WorkflowTransitionForm, WorkflowTransitionTriggerEventRelationshipFormSet
|
|
)
|
|
from .icons import (
|
|
icon_workflow_list, icon_workflow_state, icon_workflow_state_action,
|
|
icon_workflow_transition
|
|
)
|
|
from .links import (
|
|
link_setup_workflow_create, link_setup_workflow_state_action_selection,
|
|
link_setup_workflow_state_create, link_setup_workflow_transition_create
|
|
)
|
|
from .models import (
|
|
Workflow, WorkflowInstance, WorkflowRuntimeProxy, WorkflowState,
|
|
WorkflowStateAction, WorkflowStateRuntimeProxy, WorkflowTransition
|
|
)
|
|
from .permissions import (
|
|
permission_workflow_create, permission_workflow_delete,
|
|
permission_workflow_edit, permission_workflow_tools,
|
|
permission_workflow_view
|
|
)
|
|
from .tasks import task_launch_all_workflows
|
|
|
|
|
|
class DocumentWorkflowInstanceListView(SingleObjectListView):
|
|
def dispatch(self, request, *args, **kwargs):
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_view, user=request.user,
|
|
obj=self.get_document()
|
|
)
|
|
|
|
return super(
|
|
DocumentWorkflowInstanceListView, self
|
|
).dispatch(request, *args, **kwargs)
|
|
|
|
def get_document(self):
|
|
return get_object_or_404(klass=Document, pk=self.kwargs['pk'])
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'no_results_icon': icon_workflow_list,
|
|
'no_results_text': _(
|
|
'Assign workflows to the document type of this document '
|
|
'to have this document execute those workflows. '
|
|
),
|
|
'no_results_title': _(
|
|
'There are no workflow for this document'
|
|
),
|
|
'object': self.get_document(),
|
|
'title': _(
|
|
'Workflows for document: %s'
|
|
) % self.get_document(),
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return self.get_document().workflows.all()
|
|
|
|
|
|
class WorkflowInstanceDetailView(SingleObjectListView):
|
|
def dispatch(self, request, *args, **kwargs):
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_view, user=request.user,
|
|
obj=self.get_workflow_instance().document
|
|
)
|
|
|
|
return super(
|
|
WorkflowInstanceDetailView, self
|
|
).dispatch(request, *args, **kwargs)
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'navigation_object_list': ('object', 'workflow_instance'),
|
|
'no_results_text': _(
|
|
'This view will show the states changed as a workflow '
|
|
'instance is transitioned.'
|
|
),
|
|
'no_results_title': _(
|
|
'There are no details for this workflow instance'
|
|
),
|
|
'object': self.get_workflow_instance().document,
|
|
'title': _('Detail of workflow: %(workflow)s') % {
|
|
'workflow': self.get_workflow_instance()
|
|
},
|
|
'workflow_instance': self.get_workflow_instance(),
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow_instance().log_entries.order_by('-datetime')
|
|
|
|
def get_workflow_instance(self):
|
|
return get_object_or_404(klass=WorkflowInstance, pk=self.kwargs['pk'])
|
|
|
|
|
|
class WorkflowInstanceTransitionView(FormView):
|
|
form_class = WorkflowInstanceTransitionForm
|
|
template_name = 'appearance/generic_form.html'
|
|
|
|
def form_valid(self, form):
|
|
self.get_workflow_instance().do_transition(
|
|
comment=form.cleaned_data['comment'],
|
|
transition=form.cleaned_data['transition'], user=self.request.user
|
|
)
|
|
messages.success(
|
|
self.request, _(
|
|
'Document "%s" transitioned successfully'
|
|
) % self.get_workflow_instance().document
|
|
)
|
|
return HttpResponseRedirect(self.get_success_url())
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': ('object', 'workflow_instance'),
|
|
'object': self.get_workflow_instance().document,
|
|
'submit_label': _('Submit'),
|
|
'title': _(
|
|
'Do transition for workflow: %s'
|
|
) % self.get_workflow_instance(),
|
|
'workflow_instance': self.get_workflow_instance(),
|
|
}
|
|
|
|
def get_form_extra_kwargs(self):
|
|
return {
|
|
'user': self.request.user,
|
|
'workflow_instance': self.get_workflow_instance()
|
|
}
|
|
|
|
def get_success_url(self):
|
|
return self.get_workflow_instance().get_absolute_url()
|
|
|
|
def get_workflow_instance(self):
|
|
return get_object_or_404(klass=WorkflowInstance, pk=self.kwargs['pk'])
|
|
|
|
|
|
# Setup
|
|
|
|
class SetupWorkflowListView(SingleObjectListView):
|
|
model = Workflow
|
|
object_permission = permission_workflow_view
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'no_results_icon': icon_workflow_list,
|
|
'no_results_main_link': link_setup_workflow_create.resolve(
|
|
context=RequestContext(request=self.request)
|
|
),
|
|
'no_results_text': _(
|
|
'Workflows store a series of states and keep track of the '
|
|
'current state of a document. Transitions are used to change the '
|
|
'current state to a new one.'
|
|
),
|
|
'no_results_title': _(
|
|
'No workflows have been defined'
|
|
),
|
|
'title': _('Workflows'),
|
|
}
|
|
|
|
|
|
class SetupWorkflowCreateView(SingleObjectCreateView):
|
|
form_class = WorkflowForm
|
|
model = Workflow
|
|
post_action_redirect = reverse_lazy('document_states:setup_workflow_list')
|
|
view_permission = permission_workflow_create
|
|
|
|
|
|
class SetupWorkflowEditView(SingleObjectEditView):
|
|
form_class = WorkflowForm
|
|
model = Workflow
|
|
object_permission = permission_workflow_edit
|
|
post_action_redirect = reverse_lazy('document_states:setup_workflow_list')
|
|
|
|
|
|
class SetupWorkflowDeleteView(SingleObjectDeleteView):
|
|
model = Workflow
|
|
object_permission = permission_workflow_delete
|
|
post_action_redirect = reverse_lazy('document_states:setup_workflow_list')
|
|
|
|
|
|
class SetupWorkflowDocumentTypesView(AssignRemoveView):
|
|
decode_content_type = True
|
|
left_list_title = _('Available document types')
|
|
object_permission = permission_workflow_edit
|
|
right_list_title = _('Document types assigned this workflow')
|
|
|
|
def add(self, item):
|
|
self.get_object().document_types.add(item)
|
|
# TODO: add task launching this workflow for all the document types
|
|
# of item
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'object': self.get_object(),
|
|
'subtitle': _(
|
|
'Removing a document type from a workflow will also '
|
|
'remove all running instances of that workflow for '
|
|
'documents of the document type just removed.'
|
|
),
|
|
'title': _(
|
|
'Document types assigned the workflow: %s'
|
|
) % self.get_object(),
|
|
}
|
|
|
|
def get_object(self):
|
|
return get_object_or_404(klass=Workflow, pk=self.kwargs['pk'])
|
|
|
|
def left_list(self):
|
|
return AssignRemoveView.generate_choices(
|
|
self.get_object().get_document_types_not_in_workflow()
|
|
)
|
|
|
|
def right_list(self):
|
|
return AssignRemoveView.generate_choices(
|
|
self.get_object().document_types.all()
|
|
)
|
|
|
|
def remove(self, item):
|
|
# When removing a document type to workflow association
|
|
# also remove all running workflows in documents of that type.
|
|
with transaction.atomic():
|
|
self.get_object().document_types.remove(item)
|
|
self.get_object().instances.filter(document__document_type=item).delete()
|
|
|
|
|
|
# Workflow state actions
|
|
|
|
|
|
class SetupWorkflowStateActionCreateView(SingleObjectDynamicFormCreateView):
|
|
form_class = WorkflowStateActionDynamicForm
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_class(self):
|
|
try:
|
|
return WorkflowAction.get(name=self.kwargs['class_path'])
|
|
except KeyError:
|
|
raise Http404(
|
|
'{} class not found'.format(self.kwargs['class_path'])
|
|
)
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': ('object', 'workflow'),
|
|
'object': self.get_object(),
|
|
'title': _(
|
|
'Create a "%s" workflow action'
|
|
) % self.get_class().label,
|
|
'workflow': self.get_object().workflow
|
|
}
|
|
|
|
def get_form_extra_kwargs(self):
|
|
return {
|
|
'request': self.request,
|
|
'action_path': self.kwargs['class_path']
|
|
}
|
|
|
|
def get_form_schema(self):
|
|
return self.get_class()().get_form_schema(request=self.request)
|
|
|
|
def get_instance_extra_data(self):
|
|
return {
|
|
'action_path': self.kwargs['class_path'],
|
|
'state': self.get_object()
|
|
}
|
|
|
|
def get_object(self):
|
|
return get_object_or_404(klass=WorkflowState, pk=self.kwargs['pk'])
|
|
|
|
def get_post_action_redirect(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_state_action_list',
|
|
args=(self.get_object().pk,)
|
|
)
|
|
|
|
|
|
class SetupWorkflowStateActionDeleteView(SingleObjectDeleteView):
|
|
model = WorkflowStateAction
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': (
|
|
'object', 'workflow_state', 'workflow'
|
|
),
|
|
'object': self.get_object(),
|
|
'title': _('Delete workflow state action: %s') % self.get_object(),
|
|
'workflow': self.get_object().state.workflow,
|
|
'workflow_state': self.get_object().state,
|
|
}
|
|
|
|
def get_post_action_redirect(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_state_action_list',
|
|
args=(self.get_object().state.pk,)
|
|
)
|
|
|
|
|
|
class SetupWorkflowStateActionEditView(SingleObjectDynamicFormEditView):
|
|
form_class = WorkflowStateActionDynamicForm
|
|
model = WorkflowStateAction
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': (
|
|
'object', 'workflow_state', 'workflow'
|
|
),
|
|
'object': self.get_object(),
|
|
'title': _('Edit workflow state action: %s') % self.get_object(),
|
|
'workflow': self.get_object().state.workflow,
|
|
'workflow_state': self.get_object().state,
|
|
}
|
|
|
|
def get_form_extra_kwargs(self):
|
|
return {
|
|
'request': self.request,
|
|
'action_path': self.get_object().action_path,
|
|
}
|
|
|
|
def get_form_schema(self):
|
|
return self.get_object().get_class_instance().get_form_schema(
|
|
request=self.request
|
|
)
|
|
|
|
def get_post_action_redirect(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_state_action_list',
|
|
args=(self.get_object().state.pk,)
|
|
)
|
|
|
|
|
|
class SetupWorkflowStateActionListView(SingleObjectListView):
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'navigation_object_list': ('object', 'workflow'),
|
|
'no_results_icon': icon_workflow_state_action,
|
|
'no_results_main_link': link_setup_workflow_state_action_selection.resolve(
|
|
context=RequestContext(
|
|
request=self.request, dict_={
|
|
'object': self.get_workflow_state()
|
|
}
|
|
)
|
|
),
|
|
'no_results_text': _(
|
|
'Workflow state actions are macros that get executed when '
|
|
'documents enters or leaves the state in which they reside.'
|
|
),
|
|
'no_results_title': _(
|
|
'There are no actions for this workflow state'
|
|
),
|
|
'object': self.get_workflow_state(),
|
|
'title': _(
|
|
'Actions for workflow state: %s'
|
|
) % self.get_workflow_state(),
|
|
'workflow': self.get_workflow_state().workflow,
|
|
}
|
|
|
|
def get_form_schema(self):
|
|
return {'fields': self.get_class().fields}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow_state().actions.all()
|
|
|
|
def get_workflow_state(self):
|
|
return get_object_or_404(klass=WorkflowState, pk=self.kwargs['pk'])
|
|
|
|
|
|
class SetupWorkflowStateActionSelectionView(FormView):
|
|
form_class = WorkflowActionSelectionForm
|
|
view_permission = permission_workflow_edit
|
|
|
|
def form_valid(self, form):
|
|
klass = form.cleaned_data['klass']
|
|
return HttpResponseRedirect(
|
|
reverse(
|
|
'document_states:setup_workflow_state_action_create',
|
|
args=(self.get_object().pk, klass,),
|
|
)
|
|
)
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': (
|
|
'object', 'workflow'
|
|
),
|
|
'object': self.get_object(),
|
|
'title': _('New workflow state action selection'),
|
|
'workflow': self.get_object().workflow,
|
|
}
|
|
|
|
def get_object(self):
|
|
return get_object_or_404(klass=WorkflowState, pk=self.kwargs['pk'])
|
|
|
|
|
|
# Workflow states
|
|
|
|
|
|
class SetupWorkflowStateCreateView(SingleObjectCreateView):
|
|
form_class = WorkflowStateForm
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'object': self.get_workflow(),
|
|
'title': _(
|
|
'Create states for workflow: %s'
|
|
) % self.get_workflow()
|
|
}
|
|
|
|
def get_instance_extra_data(self):
|
|
return {'workflow': self.get_workflow()}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow().states.all()
|
|
|
|
def get_success_url(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_state_list', args=(self.kwargs['pk'],)
|
|
)
|
|
|
|
def get_workflow(self):
|
|
workflow = get_object_or_404(klass=Workflow, pk=self.kwargs['pk'])
|
|
AccessControlList.objects.check_access(
|
|
permissions=(permission_workflow_edit,), obj=workflow,
|
|
user=self.request.user
|
|
)
|
|
return workflow
|
|
|
|
|
|
class SetupWorkflowStateDeleteView(SingleObjectDeleteView):
|
|
model = WorkflowState
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': ('object', 'workflow_instance'),
|
|
'object': self.get_object(),
|
|
'workflow_instance': self.get_object().workflow,
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow().states.all()
|
|
|
|
def get_success_url(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_state_list',
|
|
args=(self.get_object().workflow.pk,)
|
|
)
|
|
|
|
def get_workflow(self):
|
|
workflow = get_object_or_404(klass=Workflow, pk=self.kwargs['pk'])
|
|
AccessControlList.objects.check_access(
|
|
permissions=(permission_workflow_edit,), obj=workflow,
|
|
user=self.request.user
|
|
)
|
|
return workflow
|
|
|
|
|
|
class SetupWorkflowStateEditView(SingleObjectEditView):
|
|
form_class = WorkflowStateForm
|
|
model = WorkflowState
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': ('object', 'workflow_instance'),
|
|
'object': self.get_object(),
|
|
'workflow_instance': self.get_object().workflow,
|
|
}
|
|
|
|
def get_success_url(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_state_list',
|
|
args=(self.get_object().workflow.pk,)
|
|
)
|
|
|
|
|
|
class SetupWorkflowStateListView(SingleObjectListView):
|
|
object_permission = permission_workflow_view
|
|
|
|
def dispatch(self, request, *args, **kwargs):
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_view, user=request.user,
|
|
obj=self.get_workflow()
|
|
)
|
|
|
|
return super(
|
|
SetupWorkflowStateListView, self
|
|
).dispatch(request, *args, **kwargs)
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'no_results_icon': icon_workflow_state,
|
|
'no_results_main_link': link_setup_workflow_state_create.resolve(
|
|
context=RequestContext(
|
|
self.request, {'object': self.get_workflow()}
|
|
)
|
|
),
|
|
'no_results_text': _(
|
|
'Create states and link them using transitions.'
|
|
),
|
|
'no_results_title': _(
|
|
'This workflow doesn\'t have any states'
|
|
),
|
|
'object': self.get_workflow(),
|
|
'title': _('States of workflow: %s') % self.get_workflow()
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow().states.all()
|
|
|
|
def get_workflow(self):
|
|
return get_object_or_404(klass=Workflow, pk=self.kwargs['pk'])
|
|
|
|
|
|
# Transitions
|
|
|
|
|
|
class SetupWorkflowTransitionCreateView(SingleObjectCreateView):
|
|
form_class = WorkflowTransitionForm
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'object': self.get_workflow(),
|
|
'title': _(
|
|
'Create transitions for workflow: %s'
|
|
) % self.get_workflow()
|
|
}
|
|
|
|
def get_form_kwargs(self):
|
|
kwargs = super(
|
|
SetupWorkflowTransitionCreateView, self
|
|
).get_form_kwargs()
|
|
kwargs['workflow'] = self.get_workflow()
|
|
return kwargs
|
|
|
|
def get_instance_extra_data(self):
|
|
return {'workflow': self.get_workflow()}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow().transitions.all()
|
|
|
|
def get_success_url(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_transition_list',
|
|
args=(self.kwargs['pk'],)
|
|
)
|
|
|
|
def get_workflow(self):
|
|
workflow = get_object_or_404(klass=Workflow, pk=self.kwargs['pk'])
|
|
AccessControlList.objects.check_access(
|
|
permissions=(permission_workflow_edit,), obj=workflow,
|
|
user=self.request.user
|
|
)
|
|
return workflow
|
|
|
|
|
|
class SetupWorkflowTransitionDeleteView(SingleObjectDeleteView):
|
|
model = WorkflowTransition
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'object': self.get_object(),
|
|
'navigation_object_list': ('object', 'workflow_instance'),
|
|
'workflow_instance': self.get_object().workflow,
|
|
}
|
|
|
|
def get_success_url(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_transition_list',
|
|
args=(self.get_object().workflow.pk,)
|
|
)
|
|
|
|
|
|
class SetupWorkflowTransitionEditView(SingleObjectEditView):
|
|
form_class = WorkflowTransitionForm
|
|
model = WorkflowTransition
|
|
object_permission = permission_workflow_edit
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'navigation_object_list': ('object', 'workflow_instance'),
|
|
'object': self.get_object(),
|
|
'workflow_instance': self.get_object().workflow,
|
|
}
|
|
|
|
def get_form_kwargs(self):
|
|
kwargs = super(
|
|
SetupWorkflowTransitionEditView, self
|
|
).get_form_kwargs()
|
|
kwargs['workflow'] = self.get_object().workflow
|
|
return kwargs
|
|
|
|
def get_success_url(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_transition_list',
|
|
args=(self.get_object().workflow.pk,)
|
|
)
|
|
|
|
|
|
class SetupWorkflowTransitionListView(SingleObjectListView):
|
|
object_permission = permission_workflow_view
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'no_results_icon': icon_workflow_transition,
|
|
'no_results_main_link': link_setup_workflow_transition_create.resolve(
|
|
context=RequestContext(
|
|
self.request, {'object': self.get_workflow()}
|
|
)
|
|
),
|
|
'no_results_text': _(
|
|
'Create a transition and use it to move a workflow from '
|
|
' one state to another.'
|
|
),
|
|
'no_results_title': _(
|
|
'This workflow doesn\'t have any transitions'
|
|
),
|
|
'object': self.get_workflow(),
|
|
'title': _(
|
|
'Transitions of workflow: %s'
|
|
) % self.get_workflow()
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return self.get_workflow().transitions.all()
|
|
|
|
def get_workflow(self):
|
|
return get_object_or_404(klass=Workflow, pk=self.kwargs['pk'])
|
|
|
|
|
|
# Other
|
|
|
|
|
|
class WorkflowListView(SingleObjectListView):
|
|
object_permission = permission_workflow_view
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_object': True,
|
|
'no_results_icon': icon_workflow_list,
|
|
'no_results_main_link': link_setup_workflow_create.resolve(
|
|
context=RequestContext(
|
|
self.request, {}
|
|
)
|
|
),
|
|
'no_results_text': _(
|
|
'Create some workflows and associated them with a document '
|
|
'type. Active workflows will be shown here and the documents '
|
|
'for which they are executing.'
|
|
),
|
|
'no_results_title': _('There are no workflows'),
|
|
'title': _('Workflows'),
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return WorkflowRuntimeProxy.objects.all()
|
|
|
|
|
|
class WorkflowDocumentListView(DocumentListView):
|
|
def dispatch(self, request, *args, **kwargs):
|
|
self.workflow = get_object_or_404(
|
|
klass=WorkflowRuntimeProxy, pk=self.kwargs['pk']
|
|
)
|
|
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_view, user=request.user,
|
|
obj=self.workflow
|
|
)
|
|
|
|
return super(
|
|
WorkflowDocumentListView, self
|
|
).dispatch(request, *args, **kwargs)
|
|
|
|
def get_document_queryset(self):
|
|
return Document.objects.filter(workflows__workflow=self.workflow)
|
|
|
|
def get_extra_context(self):
|
|
context = super(WorkflowDocumentListView, self).get_extra_context()
|
|
context.update(
|
|
{
|
|
'no_results_text': _(
|
|
'Associate a workflow with some document types and '
|
|
'documents of those types will be listed in this view.'
|
|
),
|
|
'no_results_title': _(
|
|
'There are no documents executing this workflow'
|
|
),
|
|
'object': self.workflow,
|
|
'title': _('Documents with the workflow: %s') % self.workflow
|
|
}
|
|
)
|
|
return context
|
|
|
|
|
|
class WorkflowStateDocumentListView(DocumentListView):
|
|
def get_document_queryset(self):
|
|
return self.get_workflow_state().get_documents()
|
|
|
|
def get_extra_context(self):
|
|
workflow_state = self.get_workflow_state()
|
|
context = super(WorkflowStateDocumentListView, self).get_extra_context()
|
|
context.update(
|
|
{
|
|
'object': workflow_state,
|
|
'navigation_object_list': ('object', 'workflow'),
|
|
'no_results_title': _(
|
|
'There are documents in this workflow state'
|
|
),
|
|
'title': _(
|
|
'Documents in the workflow "%s", state "%s"'
|
|
) % (
|
|
workflow_state.workflow, workflow_state
|
|
),
|
|
'workflow': WorkflowRuntimeProxy.objects.get(
|
|
pk=workflow_state.workflow.pk
|
|
),
|
|
}
|
|
)
|
|
return context
|
|
|
|
def get_workflow_state(self):
|
|
workflow_state = get_object_or_404(
|
|
klass=WorkflowStateRuntimeProxy, pk=self.kwargs['pk']
|
|
)
|
|
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_view, user=self.request.user,
|
|
obj=workflow_state.workflow
|
|
)
|
|
|
|
return workflow_state
|
|
|
|
|
|
class WorkflowStateListView(SingleObjectListView):
|
|
def dispatch(self, request, *args, **kwargs):
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_view, user=request.user,
|
|
obj=self.get_workflow()
|
|
)
|
|
|
|
return super(
|
|
WorkflowStateListView, self
|
|
).dispatch(request, *args, **kwargs)
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_columns': True,
|
|
'hide_link': True,
|
|
'no_results_main_link': link_setup_workflow_state_create.resolve(
|
|
context=RequestContext(
|
|
self.request, {'object': self.get_workflow()}
|
|
)
|
|
),
|
|
'no_results_text': _(
|
|
'Create states and link them using transitions.'
|
|
),
|
|
'no_results_title': _(
|
|
'This workflow doesn\'t have any state'
|
|
),
|
|
'object': self.get_workflow(),
|
|
'title': _('States of workflow: %s') % self.get_workflow()
|
|
}
|
|
|
|
def get_object_list(self):
|
|
return WorkflowStateRuntimeProxy.objects.filter(
|
|
workflow=self.get_workflow()
|
|
)
|
|
|
|
def get_workflow(self):
|
|
return get_object_or_404(klass=WorkflowRuntimeProxy, pk=self.kwargs['pk'])
|
|
|
|
|
|
class SetupWorkflowTransitionTriggerEventListView(FormView):
|
|
form_class = WorkflowTransitionTriggerEventRelationshipFormSet
|
|
submodel = StoredEventType
|
|
|
|
def dispatch(self, *args, **kwargs):
|
|
AccessControlList.objects.check_access(
|
|
permissions=permission_workflow_edit,
|
|
user=self.request.user, obj=self.get_object().workflow
|
|
)
|
|
|
|
EventType.refresh()
|
|
return super(
|
|
SetupWorkflowTransitionTriggerEventListView, self
|
|
).dispatch(*args, **kwargs)
|
|
|
|
def form_valid(self, form):
|
|
try:
|
|
for instance in form:
|
|
instance.save()
|
|
except Exception as exception:
|
|
messages.error(
|
|
self.request,
|
|
_(
|
|
'Error updating workflow transition trigger events; %s'
|
|
) % exception
|
|
)
|
|
else:
|
|
messages.success(
|
|
self.request, _(
|
|
'Workflow transition trigger events updated successfully'
|
|
)
|
|
)
|
|
|
|
return super(
|
|
SetupWorkflowTransitionTriggerEventListView, self
|
|
).form_valid(form=form)
|
|
|
|
def get_object(self):
|
|
return get_object_or_404(klass=WorkflowTransition, pk=self.kwargs['pk'])
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'form_display_mode_table': True,
|
|
'navigation_object_list': ('object', 'workflow'),
|
|
'object': self.get_object(),
|
|
'subtitle': _(
|
|
'Triggers are events that cause this transition to execute '
|
|
'automatically.'
|
|
),
|
|
'title': _(
|
|
'Workflow transition trigger events for: %s'
|
|
) % self.get_object(),
|
|
'workflow': self.get_object().workflow,
|
|
}
|
|
|
|
def get_initial(self):
|
|
obj = self.get_object()
|
|
initial = []
|
|
|
|
# Return the queryset by name from the sorted list of the class
|
|
event_type_ids = [event_type.id for event_type in EventType.all()]
|
|
event_type_queryset = StoredEventType.objects.filter(
|
|
name__in=event_type_ids
|
|
)
|
|
|
|
for event_type in event_type_queryset:
|
|
initial.append({
|
|
'transition': obj,
|
|
'event_type': event_type,
|
|
})
|
|
return initial
|
|
|
|
def get_post_action_redirect(self):
|
|
return reverse(
|
|
'document_states:setup_workflow_transition_list',
|
|
args=(self.get_object().workflow.pk,)
|
|
)
|
|
|
|
|
|
class ToolLaunchAllWorkflows(ConfirmView):
|
|
extra_context = {
|
|
'title': _('Launch all workflows?'),
|
|
'subtitle': _(
|
|
'This will launch all workflows created after documents have '
|
|
'already been uploaded.'
|
|
)
|
|
}
|
|
view_permission = permission_workflow_tools
|
|
|
|
def view_action(self):
|
|
task_launch_all_workflows.apply_async()
|
|
messages.success(
|
|
self.request, _('Workflow launch queued successfully.')
|
|
)
|
|
|
|
|
|
class WorkflowPreviewView(SingleObjectDetailView):
|
|
form_class = WorkflowPreviewForm
|
|
model = Workflow
|
|
object_permission = permission_workflow_view
|
|
|
|
def get_extra_context(self):
|
|
return {
|
|
'hide_labels': True,
|
|
'object': self.get_object(),
|
|
'title': _('Preview of: %s') % self.get_object()
|
|
}
|