Backport workflow preview refactor
GitLab issue #532. Signed-off-by: Roberto Rosario <roberto.rosario.gonzalez@gmail.com>
This commit is contained in:
@@ -5,6 +5,7 @@
|
|||||||
links.
|
links.
|
||||||
- Use Select2 widget for the document type selection form.
|
- Use Select2 widget for the document type selection form.
|
||||||
- Backport the vertical main menu update.
|
- Backport the vertical main menu update.
|
||||||
|
- Backport workflow preview refactor. GitLab issue #532.
|
||||||
|
|
||||||
3.2.5 (2019-07-05)
|
3.2.5 (2019-07-05)
|
||||||
==================
|
==================
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ Changes
|
|||||||
The vertical menu remain open even when clicking on items and upon
|
The vertical menu remain open even when clicking on items and upon
|
||||||
a browser refresh will also restore its state to match the selected
|
a browser refresh will also restore its state to match the selected
|
||||||
view.
|
view.
|
||||||
|
- Backport workflow preview refactor. GitLab issue #532.
|
||||||
|
|
||||||
Removals
|
Removals
|
||||||
--------
|
--------
|
||||||
@@ -105,6 +106,6 @@ Backward incompatible changes
|
|||||||
Bugs fixed or issues closed
|
Bugs fixed or issues closed
|
||||||
---------------------------
|
---------------------------
|
||||||
|
|
||||||
- :gitlab-issue:`XX`
|
- :gitlab-issue:`532` Workflow preview isn't updated right after transitions are modified
|
||||||
|
|
||||||
.. _PyPI: https://pypi.python.org/pypi/mayan-edms/
|
.. _PyPI: https://pypi.python.org/pypi/mayan-edms/
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
from __future__ import absolute_import, unicode_literals
|
from __future__ import absolute_import, unicode_literals
|
||||||
|
|
||||||
|
from django.http import HttpResponse
|
||||||
from django.shortcuts import get_object_or_404
|
from django.shortcuts import get_object_or_404
|
||||||
|
from django.views.decorators.cache import cache_control, patch_cache_control
|
||||||
|
|
||||||
from rest_framework import generics
|
from rest_framework import generics
|
||||||
|
|
||||||
@@ -10,6 +12,7 @@ from mayan.apps.documents.permissions import permission_document_type_view
|
|||||||
from mayan.apps.rest_api.filters import MayanObjectPermissionsFilter
|
from mayan.apps.rest_api.filters import MayanObjectPermissionsFilter
|
||||||
from mayan.apps.rest_api.permissions import MayanPermission
|
from mayan.apps.rest_api.permissions import MayanPermission
|
||||||
|
|
||||||
|
from .literals import WORKFLOW_IMAGE_TASK_TIMEOUT
|
||||||
from .models import Workflow
|
from .models import Workflow
|
||||||
from .permissions import (
|
from .permissions import (
|
||||||
permission_workflow_create, permission_workflow_delete,
|
permission_workflow_create, permission_workflow_delete,
|
||||||
@@ -23,6 +26,10 @@ from .serializers import (
|
|||||||
WritableWorkflowTransitionSerializer
|
WritableWorkflowTransitionSerializer
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from .settings import settings_workflow_image_cache_time
|
||||||
|
from .storages import storage_workflowimagecache
|
||||||
|
from .tasks import task_generate_workflow_image
|
||||||
|
|
||||||
|
|
||||||
class APIDocumentTypeWorkflowListView(generics.ListAPIView):
|
class APIDocumentTypeWorkflowListView(generics.ListAPIView):
|
||||||
"""
|
"""
|
||||||
@@ -172,6 +179,41 @@ class APIWorkflowDocumentTypeView(generics.RetrieveDestroyAPIView):
|
|||||||
self.get_workflow().document_types.remove(instance)
|
self.get_workflow().document_types.remove(instance)
|
||||||
|
|
||||||
|
|
||||||
|
class APIWorkflowImageView(generics.RetrieveAPIView):
|
||||||
|
"""
|
||||||
|
get: Returns an image representation of the selected workflow.
|
||||||
|
"""
|
||||||
|
filter_backends = (MayanObjectPermissionsFilter,)
|
||||||
|
mayan_object_permissions = {
|
||||||
|
'GET': (permission_workflow_view,),
|
||||||
|
}
|
||||||
|
queryset = Workflow.objects.all()
|
||||||
|
|
||||||
|
def get_serializer(self, *args, **kwargs):
|
||||||
|
return None
|
||||||
|
|
||||||
|
def get_serializer_class(self):
|
||||||
|
return None
|
||||||
|
|
||||||
|
@cache_control(private=True)
|
||||||
|
def retrieve(self, request, *args, **kwargs):
|
||||||
|
task = task_generate_workflow_image.apply_async(
|
||||||
|
kwargs=dict(
|
||||||
|
document_state_id=self.get_object().pk,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
cache_filename = task.get(timeout=WORKFLOW_IMAGE_TASK_TIMEOUT)
|
||||||
|
with storage_workflowimagecache.open(cache_filename) as file_object:
|
||||||
|
response = HttpResponse(file_object.read(), content_type='image')
|
||||||
|
if '_hash' in request.GET:
|
||||||
|
patch_cache_control(
|
||||||
|
response,
|
||||||
|
max_age=settings_workflow_image_cache_time.value
|
||||||
|
)
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
class APIWorkflowListView(generics.ListCreateAPIView):
|
class APIWorkflowListView(generics.ListCreateAPIView):
|
||||||
"""
|
"""
|
||||||
get: Returns a list of all the workflows.
|
get: Returns a list of all the workflows.
|
||||||
|
|||||||
9
mayan/apps/document_states/fields.py
Normal file
9
mayan/apps/document_states/fields.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
from __future__ import absolute_import, unicode_literals
|
||||||
|
|
||||||
|
from django import forms
|
||||||
|
|
||||||
|
from .widgets import WorkflowImageWidget
|
||||||
|
|
||||||
|
|
||||||
|
class WorfklowImageField(forms.fields.Field):
|
||||||
|
widget = WorkflowImageWidget
|
||||||
@@ -12,10 +12,10 @@ from django.utils.translation import ugettext_lazy as _
|
|||||||
from mayan.apps.common.forms import DynamicModelForm
|
from mayan.apps.common.forms import DynamicModelForm
|
||||||
|
|
||||||
from .classes import WorkflowAction
|
from .classes import WorkflowAction
|
||||||
|
from .fields import WorfklowImageField
|
||||||
from .models import (
|
from .models import (
|
||||||
Workflow, WorkflowState, WorkflowStateAction, WorkflowTransition
|
Workflow, WorkflowState, WorkflowStateAction, WorkflowTransition
|
||||||
)
|
)
|
||||||
from .widgets import WorkflowImageWidget
|
|
||||||
|
|
||||||
|
|
||||||
class WorkflowActionSelectionForm(forms.Form):
|
class WorkflowActionSelectionForm(forms.Form):
|
||||||
@@ -188,9 +188,9 @@ class WorkflowInstanceTransitionForm(forms.Form):
|
|||||||
|
|
||||||
|
|
||||||
class WorkflowPreviewForm(forms.Form):
|
class WorkflowPreviewForm(forms.Form):
|
||||||
preview = forms.CharField(widget=WorkflowImageWidget())
|
workflow = WorfklowImageField()
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
instance = kwargs.pop('instance', None)
|
instance = kwargs.pop('instance', None)
|
||||||
super(WorkflowPreviewForm, self).__init__(*args, **kwargs)
|
super(WorkflowPreviewForm, self).__init__(*args, **kwargs)
|
||||||
self.fields['preview'].initial = instance
|
self.fields['workflow'].initial = instance
|
||||||
|
|||||||
@@ -9,3 +9,4 @@ WORKFLOW_ACTION_WHEN_CHOICES = (
|
|||||||
(WORKFLOW_ACTION_ON_ENTRY, _('On entry')),
|
(WORKFLOW_ACTION_ON_ENTRY, _('On entry')),
|
||||||
(WORKFLOW_ACTION_ON_EXIT, _('On exit')),
|
(WORKFLOW_ACTION_ON_EXIT, _('On exit')),
|
||||||
)
|
)
|
||||||
|
WORKFLOW_IMAGE_TASK_TIMEOUT = 60
|
||||||
|
|||||||
@@ -1,12 +1,16 @@
|
|||||||
from __future__ import absolute_import, unicode_literals
|
from __future__ import absolute_import, unicode_literals
|
||||||
|
|
||||||
|
import hashlib
|
||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
|
from furl import furl
|
||||||
from graphviz import Digraph
|
from graphviz import Digraph
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
from django.core import serializers
|
||||||
from django.core.exceptions import PermissionDenied, ValidationError
|
from django.core.exceptions import PermissionDenied, ValidationError
|
||||||
|
from django.core.files.base import ContentFile
|
||||||
from django.db import IntegrityError, models, transaction
|
from django.db import IntegrityError, models, transaction
|
||||||
from django.db.models import F, Max, Q
|
from django.db.models import F, Max, Q
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
@@ -27,6 +31,7 @@ from .literals import (
|
|||||||
)
|
)
|
||||||
from .managers import WorkflowManager
|
from .managers import WorkflowManager
|
||||||
from .permissions import permission_workflow_transition
|
from .permissions import permission_workflow_transition
|
||||||
|
from .storages import storage_workflowimagecache
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -63,9 +68,49 @@ class Workflow(models.Model):
|
|||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.label
|
return self.label
|
||||||
|
|
||||||
|
def generate_image(self):
|
||||||
|
cache_filename = '{}-{}'.format(self.id, self.get_hash())
|
||||||
|
image = self.render()
|
||||||
|
|
||||||
|
# Since open "wb+" doesn't create files, check if the file
|
||||||
|
# exists, if not then create it
|
||||||
|
if not storage_workflowimagecache.exists(cache_filename):
|
||||||
|
storage_workflowimagecache.save(
|
||||||
|
name=cache_filename, content=ContentFile(content='')
|
||||||
|
)
|
||||||
|
|
||||||
|
with storage_workflowimagecache.open(cache_filename, mode='wb+') as file_object:
|
||||||
|
file_object.write(image)
|
||||||
|
|
||||||
|
return cache_filename
|
||||||
|
|
||||||
|
def get_api_image_url(self, *args, **kwargs):
|
||||||
|
final_url = furl()
|
||||||
|
final_url.args = kwargs
|
||||||
|
final_url.path = reverse(
|
||||||
|
viewname='rest_api:workflow-image',
|
||||||
|
kwargs={'pk': self.pk}
|
||||||
|
)
|
||||||
|
final_url.args['_hash'] = self.get_hash()
|
||||||
|
|
||||||
|
return final_url.tostr()
|
||||||
|
|
||||||
def get_document_types_not_in_workflow(self):
|
def get_document_types_not_in_workflow(self):
|
||||||
return DocumentType.objects.exclude(pk__in=self.document_types.all())
|
return DocumentType.objects.exclude(pk__in=self.document_types.all())
|
||||||
|
|
||||||
|
def get_hash(self):
|
||||||
|
objects_lists = list(
|
||||||
|
Workflow.objects.filter(pk=self.pk)
|
||||||
|
) + list(
|
||||||
|
WorkflowState.objects.filter(workflow__pk=self.pk)
|
||||||
|
) + list(
|
||||||
|
WorkflowTransition.objects.filter(workflow__pk=self.pk)
|
||||||
|
)
|
||||||
|
|
||||||
|
return hashlib.sha256(
|
||||||
|
serializers.serialize('json', objects_lists)
|
||||||
|
).hexdigest()
|
||||||
|
|
||||||
def get_initial_state(self):
|
def get_initial_state(self):
|
||||||
try:
|
try:
|
||||||
return self.states.get(initial=True)
|
return self.states.get(initial=True)
|
||||||
|
|||||||
@@ -3,12 +3,21 @@ from __future__ import absolute_import, unicode_literals
|
|||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
from mayan.apps.task_manager.classes import CeleryQueue
|
from mayan.apps.task_manager.classes import CeleryQueue
|
||||||
from mayan.apps.task_manager.workers import worker_slow
|
from mayan.apps.task_manager.workers import worker_fast, worker_slow
|
||||||
|
|
||||||
queue_document_states = CeleryQueue(
|
queue_document_states = CeleryQueue(
|
||||||
name='document_states', label=_('Document states'), worker=worker_slow
|
label=_('Document states'), name='document_states', worker=worker_slow
|
||||||
)
|
)
|
||||||
|
queue_document_states_fast = CeleryQueue(
|
||||||
|
label=_('Document states fast'), name='document_states_fast',
|
||||||
|
worker=worker_fast
|
||||||
|
)
|
||||||
|
|
||||||
queue_document_states.add_task_type(
|
queue_document_states.add_task_type(
|
||||||
dotted_path='mayan.apps.document_states.tasks.task_launch_all_workflows',
|
label=_('Launch all workflows'),
|
||||||
label=_('Launch all workflows')
|
dotted_path='mayan.apps.document_states.tasks.task_launch_all_workflows'
|
||||||
|
)
|
||||||
|
queue_document_states_fast.add_task_type(
|
||||||
|
label=_('Generate workflow previews'),
|
||||||
|
dotted_path='mayan.apps.document_states.tasks.task_generate_workflow_image'
|
||||||
)
|
)
|
||||||
|
|||||||
32
mayan/apps/document_states/settings.py
Normal file
32
mayan/apps/document_states/settings.py
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
import os
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
|
from mayan.apps.smart_settings import Namespace
|
||||||
|
|
||||||
|
namespace = Namespace(label=_('Workflows'), name='document_states')
|
||||||
|
|
||||||
|
settings_workflow_image_cache_time = namespace.add_setting(
|
||||||
|
global_name='WORKFLOWS_IMAGE_CACHE_TIME', default='31556926',
|
||||||
|
help_text=_(
|
||||||
|
'Time in seconds that the browser should cache the supplied workflow '
|
||||||
|
'images. The default of 31559626 seconds corresponde to 1 year.'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
setting_workflowimagecache_storage = namespace.add_setting(
|
||||||
|
global_name='WORKFLOWS_IMAGE_CACHE_STORAGE_BACKEND',
|
||||||
|
default='django.core.files.storage.FileSystemStorage', help_text=_(
|
||||||
|
'Path to the Storage subclass to use when storing the cached '
|
||||||
|
'workflow image files.'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
setting_workflowimagecache_storage_arguments = namespace.add_setting(
|
||||||
|
global_name='WORKFLOWS_IMAGE_CACHE_STORAGE_BACKEND_ARGUMENTS',
|
||||||
|
default={'location': os.path.join(settings.MEDIA_ROOT, 'workflows')},
|
||||||
|
help_text=_(
|
||||||
|
'Arguments to pass to the WORKFLOWS_IMAGE_CACHE_STORAGE_BACKEND.'
|
||||||
|
)
|
||||||
|
)
|
||||||
12
mayan/apps/document_states/storages.py
Normal file
12
mayan/apps/document_states/storages.py
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from django.utils.module_loading import import_string
|
||||||
|
|
||||||
|
from .settings import (
|
||||||
|
setting_workflowimagecache_storage,
|
||||||
|
setting_workflowimagecache_storage_arguments
|
||||||
|
)
|
||||||
|
|
||||||
|
storage_workflowimagecache = import_string(
|
||||||
|
dotted_path=setting_workflowimagecache_storage.value
|
||||||
|
)(**setting_workflowimagecache_storage_arguments.value)
|
||||||
@@ -9,6 +9,17 @@ from mayan.celery import app
|
|||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
@app.task()
|
||||||
|
def task_generate_workflow_image(document_state_id):
|
||||||
|
Workflow = apps.get_model(
|
||||||
|
app_label='document_states', model_name='Workflow'
|
||||||
|
)
|
||||||
|
|
||||||
|
workflow = Workflow.objects.get(pk=document_state_id)
|
||||||
|
|
||||||
|
return workflow.generate_image()
|
||||||
|
|
||||||
|
|
||||||
@app.task(ignore_result=True)
|
@app.task(ignore_result=True)
|
||||||
def task_launch_all_workflows():
|
def task_launch_all_workflows():
|
||||||
Document = apps.get_model(app_label='documents', model_name='Document')
|
Document = apps.get_model(app_label='documents', model_name='Document')
|
||||||
|
|||||||
@@ -0,0 +1,2 @@
|
|||||||
|
<img class="img-responsive" name="{{ widget.name }}" {% include "django/forms/widgets/attrs.html" %} src="{{ widget.value.get_api_image_url }}" style="margin:auto;" />
|
||||||
|
|
||||||
@@ -4,9 +4,10 @@ from django.conf.urls import url
|
|||||||
|
|
||||||
from .api_views import (
|
from .api_views import (
|
||||||
APIDocumentTypeWorkflowListView, APIWorkflowDocumentTypeList,
|
APIDocumentTypeWorkflowListView, APIWorkflowDocumentTypeList,
|
||||||
APIWorkflowDocumentTypeView, APIWorkflowInstanceListView,
|
APIWorkflowDocumentTypeView, APIWorkflowImageView,
|
||||||
APIWorkflowInstanceView, APIWorkflowInstanceLogEntryListView,
|
APIWorkflowInstanceListView, APIWorkflowInstanceView,
|
||||||
APIWorkflowListView, APIWorkflowStateListView, APIWorkflowStateView,
|
APIWorkflowInstanceLogEntryListView, APIWorkflowListView,
|
||||||
|
APIWorkflowStateListView, APIWorkflowStateView,
|
||||||
APIWorkflowTransitionListView, APIWorkflowTransitionView, APIWorkflowView
|
APIWorkflowTransitionListView, APIWorkflowTransitionView, APIWorkflowView
|
||||||
)
|
)
|
||||||
from .views import (
|
from .views import (
|
||||||
@@ -22,7 +23,7 @@ from .views import (
|
|||||||
SetupWorkflowTransitionEditView,
|
SetupWorkflowTransitionEditView,
|
||||||
SetupWorkflowTransitionTriggerEventListView, ToolLaunchAllWorkflows,
|
SetupWorkflowTransitionTriggerEventListView, ToolLaunchAllWorkflows,
|
||||||
WorkflowDocumentListView, WorkflowInstanceDetailView,
|
WorkflowDocumentListView, WorkflowInstanceDetailView,
|
||||||
WorkflowImageView, WorkflowInstanceTransitionView, WorkflowListView,
|
WorkflowInstanceTransitionView, WorkflowListView,
|
||||||
WorkflowPreviewView, WorkflowStateDocumentListView, WorkflowStateListView,
|
WorkflowPreviewView, WorkflowStateDocumentListView, WorkflowStateListView,
|
||||||
)
|
)
|
||||||
from .views.workflow_views import SetupDocumentTypeWorkflowsView
|
from .views.workflow_views import SetupDocumentTypeWorkflowsView
|
||||||
@@ -167,11 +168,6 @@ urlpatterns = [
|
|||||||
view=WorkflowStateListView.as_view(),
|
view=WorkflowStateListView.as_view(),
|
||||||
name='workflow_state_list'
|
name='workflow_state_list'
|
||||||
),
|
),
|
||||||
url(
|
|
||||||
regex=r'^(?P<pk>\d+)/image/$',
|
|
||||||
view=WorkflowImageView.as_view(),
|
|
||||||
name='workflow_image'
|
|
||||||
),
|
|
||||||
url(
|
url(
|
||||||
regex=r'^(?P<pk>\d+)/preview/$',
|
regex=r'^(?P<pk>\d+)/preview/$',
|
||||||
view=WorkflowPreviewView.as_view(),
|
view=WorkflowPreviewView.as_view(),
|
||||||
@@ -204,6 +200,10 @@ api_urls = [
|
|||||||
view=APIWorkflowDocumentTypeView.as_view(),
|
view=APIWorkflowDocumentTypeView.as_view(),
|
||||||
name='workflow-document-type-detail'
|
name='workflow-document-type-detail'
|
||||||
),
|
),
|
||||||
|
url(
|
||||||
|
regex=r'^workflows/(?P<pk>\d+)/image/$',
|
||||||
|
name='workflow-image', view=APIWorkflowImageView.as_view()
|
||||||
|
),
|
||||||
url(
|
url(
|
||||||
regex=r'^workflows/(?P<pk>[0-9]+)/states/$',
|
regex=r'^workflows/(?P<pk>[0-9]+)/states/$',
|
||||||
view=APIWorkflowStateListView.as_view(), name='workflowstate-list'
|
view=APIWorkflowStateListView.as_view(), name='workflowstate-list'
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
from __future__ import absolute_import, unicode_literals
|
from __future__ import absolute_import, unicode_literals
|
||||||
|
|
||||||
from django.contrib import messages
|
from django.contrib import messages
|
||||||
from django.core.files.base import ContentFile
|
|
||||||
from django.db import transaction
|
from django.db import transaction
|
||||||
from django.http import Http404, HttpResponseRedirect
|
from django.http import Http404, HttpResponseRedirect
|
||||||
from django.shortcuts import get_object_or_404
|
from django.shortcuts import get_object_or_404
|
||||||
@@ -13,7 +12,7 @@ from mayan.apps.common.generics import (
|
|||||||
AddRemoveView, ConfirmView, FormView, SingleObjectCreateView,
|
AddRemoveView, ConfirmView, FormView, SingleObjectCreateView,
|
||||||
SingleObjectDeleteView, SingleObjectDetailView,
|
SingleObjectDeleteView, SingleObjectDetailView,
|
||||||
SingleObjectDynamicFormCreateView, SingleObjectDynamicFormEditView,
|
SingleObjectDynamicFormCreateView, SingleObjectDynamicFormEditView,
|
||||||
SingleObjectDownloadView, SingleObjectEditView, SingleObjectListView
|
SingleObjectEditView, SingleObjectListView
|
||||||
)
|
)
|
||||||
from mayan.apps.common.mixins import ExternalObjectMixin
|
from mayan.apps.common.mixins import ExternalObjectMixin
|
||||||
from mayan.apps.documents.events import event_document_type_edited
|
from mayan.apps.documents.events import event_document_type_edited
|
||||||
@@ -49,7 +48,6 @@ from ..permissions import (
|
|||||||
from ..tasks import task_launch_all_workflows
|
from ..tasks import task_launch_all_workflows
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
'WorkflowImageView', 'WorkflowPreviewView',
|
|
||||||
'SetupWorkflowListView', 'SetupWorkflowCreateView', 'SetupWorkflowEditView',
|
'SetupWorkflowListView', 'SetupWorkflowCreateView', 'SetupWorkflowEditView',
|
||||||
'SetupWorkflowDeleteView', 'SetupWorkflowDocumentTypesView',
|
'SetupWorkflowDeleteView', 'SetupWorkflowDocumentTypesView',
|
||||||
'SetupWorkflowStateActionCreateView', 'SetupWorkflowStateActionDeleteView',
|
'SetupWorkflowStateActionCreateView', 'SetupWorkflowStateActionDeleteView',
|
||||||
@@ -59,7 +57,8 @@ __all__ = (
|
|||||||
'SetupWorkflowStateListView', 'SetupWorkflowTransitionCreateView',
|
'SetupWorkflowStateListView', 'SetupWorkflowTransitionCreateView',
|
||||||
'SetupWorkflowTransitionDeleteView', 'SetupWorkflowTransitionEditView',
|
'SetupWorkflowTransitionDeleteView', 'SetupWorkflowTransitionEditView',
|
||||||
'SetupWorkflowTransitionListView',
|
'SetupWorkflowTransitionListView',
|
||||||
'SetupWorkflowTransitionTriggerEventListView', 'ToolLaunchAllWorkflows'
|
'SetupWorkflowTransitionTriggerEventListView', 'ToolLaunchAllWorkflows',
|
||||||
|
'WorkflowPreviewView'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -750,26 +749,15 @@ class ToolLaunchAllWorkflows(ConfirmView):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class WorkflowImageView(SingleObjectDownloadView):
|
|
||||||
attachment = False
|
|
||||||
model = Workflow
|
|
||||||
object_permission = permission_workflow_view
|
|
||||||
|
|
||||||
def get_file(self):
|
|
||||||
workflow = self.get_object()
|
|
||||||
return ContentFile(workflow.render(), name=workflow.label)
|
|
||||||
|
|
||||||
def get_mimetype(self):
|
|
||||||
return 'image'
|
|
||||||
|
|
||||||
|
|
||||||
class WorkflowPreviewView(SingleObjectDetailView):
|
class WorkflowPreviewView(SingleObjectDetailView):
|
||||||
form_class = WorkflowPreviewForm
|
form_class = WorkflowPreviewForm
|
||||||
model = Workflow
|
model = Workflow
|
||||||
object_permission = permission_workflow_view
|
object_permission = permission_workflow_view
|
||||||
|
pk_url_kwarg = 'pk'
|
||||||
|
|
||||||
def get_extra_context(self):
|
def get_extra_context(self):
|
||||||
return {
|
return {
|
||||||
'hide_labels': True,
|
'hide_labels': True,
|
||||||
|
'object': self.get_object(),
|
||||||
'title': _('Preview of: %s') % self.get_object()
|
'title': _('Preview of: %s') % self.get_object()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
from django import forms
|
from django import forms
|
||||||
from django.urls import reverse
|
from django.utils.html import format_html_join
|
||||||
from django.utils.html import format_html_join, mark_safe
|
|
||||||
|
|
||||||
|
|
||||||
def widget_transition_events(transition):
|
def widget_transition_events(transition):
|
||||||
@@ -15,23 +14,10 @@ def widget_transition_events(transition):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def widget_workflow_diagram(workflow):
|
|
||||||
return mark_safe(
|
|
||||||
'<img class="img-responsive" src="{}" style="margin:auto;">'.format(
|
|
||||||
reverse(
|
|
||||||
viewname='document_states:workflow_image', kwargs={
|
|
||||||
'pk': workflow.pk
|
|
||||||
}
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class WorkflowImageWidget(forms.widgets.Widget):
|
class WorkflowImageWidget(forms.widgets.Widget):
|
||||||
def render(self, name, value, attrs=None):
|
template_name = 'document_states/forms/widgets/workflow_image.html'
|
||||||
if value:
|
|
||||||
output = []
|
def format_value(self, value):
|
||||||
output.append(widget_workflow_diagram(value))
|
if value == '' or value is None:
|
||||||
return mark_safe(''.join(output))
|
return None
|
||||||
else:
|
return value
|
||||||
return ''
|
|
||||||
|
|||||||
Reference in New Issue
Block a user