Source code for aristotle_mdr.templatetags.aristotle_tags

# -*- coding: utf-8 -*-
"""
Tags and filters available in aristotle templates
=================================================

A number of convenience tags are available for performing common actions in custom
templates.

Include the aristotle template tags in every template that uses them, like so::

    {% load aristotle_tags %}

Available tags and filters
--------------------------
"""
from django import template
from django.template.loader import get_template
from django.urls import reverse
from django.utils.translation import ugettext_lazy as _
from django.utils.html import mark_safe

from aristotle_mdr import perms
import aristotle_mdr.models as MDR
from aristotle_mdr.utils import (
    fetch_metadata_apps,
    fetch_aristotle_downloaders
)
from aristotle_mdr.models import STATES

register = template.Library()


@register.filter
def can_alter_comment(user, comment):
    try:
        return perms.user_can_alter_comment(user, comment)
    except:
        return False


@register.filter
def can_alter_post(user, post):
    try:
        return perms.user_can_alter_post(user, post)
    except:
        return False


@register.filter()
def can_manage_workgroup(workgroup, user):
    try:
        return perms.user_can_manage_workgroup(user, workgroup)
    except:  # pragma: no cover
        return None


@register.filter
def is_in(item, iterable):
    return item in iterable


[docs]@register.filter def in_workgroup(user, workgroup): """ A filter that acts as a wrapper around ``aristotle_mdr.perms.user_in_workgroup``. Returns true if the user has permission to administer the workgroup, otherwise it returns False. If calling ``user_in_workgroup`` throws an exception it safely returns False. For example:: {% if request.user|in_workgroup:workgroup %} {{ something }} {% endif %} """ try: return perms.user_in_workgroup(user, workgroup) except: return False
[docs]@register.filter def can_add_status(item, user): """ A filter that acts as a wrapper around ``aristotle_mdr.perms.user_can_add_status``. Returns true if the user has permission to change status the item, otherwise it returns False. If calling ``user_can_add_status`` throws an exception it safely returns False. For example:: {% if myItem|can_add_status:request.user %} {{ item }} {% endif %} """ # try: return perms.user_can_add_status(user, item)
# except: # pragma: no cover -- passing a bad item or user is the template authors fault # return None
[docs]@register.filter def can_edit(item, user): """ A filter that acts as a wrapper around ``aristotle_mdr.perms.user_can_edit``. Returns true if the user has permission to edit the item, otherwise it returns False. If calling ``user_can_edit`` throws an exception it safely returns False. For example:: {% if myItem|can_edit:request.user %} {{ item }} {% endif %} """ # return perms.user_can_edit(user, item) try: return perms.user_can_edit(user, item) except: # pragma: no cover -- passing a bad item or user is the template authors fault return None
[docs]@register.filter def can_view(item, user): """ A filter that acts as a wrapper around ``aristotle_mdr.perms.user_can_view``. Returns true if the user has permission to view the item, otherwise it returns False. If calling ``user_can_view`` throws an exception it safely returns False. For example:: {% if myItem|can_view:request.user %} {{ item }} {% endif %} """ try: return perms.user_can_view(user, item) except: # pragma: no cover -- passing a bad item or user is the template authors fault return None
[docs]@register.filter def can_supersede(item, user): """ A filter that acts as a wrapper around ``aristotle_mdr.perms.user_can_supersede``. Returns true if the user has permission to supersede the item, otherwise it returns False. If calling ``user_can_supersede`` throws an exception it safely returns False. For example:: {% if myItem|can_supersede:request.user %} {{ item }} {% endif %} """ return perms.user_can_supersede(user, item)
[docs]@register.filter def can_view_iter(qs, user): """ A filter that is a simple wrapper that applies the ``aristotle_mdr.models.ConceptManager.visible(user)`` for use in templates. Filtering on a Django ``Queryset`` and passing in the current user as the argument returns a list (not a ``Queryset`` at this stage) of only the items from the ``Queryset`` the user can view. If calling ``can_view_iter`` throws an exception it safely returns an empty list. For example:: {% for item in myItems|can_view_iter:request.user %} {{ item }} {% endfor %} """ if qs is not None: return qs.visible(user)
[docs]@register.filter def user_can_view_statuses_revisions(user, ra): """ A filter that is a simple wrapper that applies the ``aristotle_mdr.perms.user_can_view_statuses_revisions`` Returns true if the user has permission to view the statuses reversion history, otherwise it returns False. If calling ``user_can_view`` throws an exception it safely returns False. If calling ``user_can_view_statuses_revisions`` throws an exception it safely returns False. For example:: {% if request.user|user_can_view_statuses_revisions:ra %} {{ item }} {% endif %} """ return perms.user_can_view_statuses_revisions(user, ra)
[docs]@register.filter def visible_supersedes_items(item, user): """ Fetch older items for a newer item """ objects = type(item).objects.prefetch_related( 'superseded_by_items_relation_set__older_item', 'superseded_by_items_relation_set__registration_authority', ).visible(user).filter( superseded_by_items_relation_set__newer_item_id=item.pk, # superseded_by_items_relation_set__proposed=False ).distinct() sup_rels = [ { "pk": obj.pk, "older_item": obj, "rels": [ { # "newer_item": sup.newer_item, "message": sup.message, "date_effective": sup.date_effective, "registration_authority": sup.registration_authority, "proposed": sup.proposed } for sup in obj.superseded_by_items_relation_set.all() if sup.newer_item_id == item.pk # and sup.proposed is False ] } for obj in objects ] sup_rels.sort(key=lambda x: x['pk']) return sup_rels
[docs]@register.filter def visible_superseded_by_items(item, user): """ Fetch newer items for an older item """ # Get all the objects that have superseded items objects = item.__class__.objects.prefetch_related( 'superseded_items_relation_set__newer_item', 'superseded_items_relation_set__registration_authority', ).visible(user).filter( superseded_items_relation_set__older_item_id=item.pk, ).distinct() sup_rels = [ { "pk": obj.pk, "newer_item": obj, "rels": [ { "message": sup.message, "date_effective": sup.date_effective, "registration_authority": sup.registration_authority, "proposed": sup.proposed } for sup in obj.superseded_items_relation_set.all() if sup.older_item_id == item.pk ] } for obj in objects ] sup_rels.sort(key=lambda x: x['pk']) return sup_rels
[docs]@register.filter def public_standards(regAuth, itemType="aristotle_mdr._concept"): """ This is a filter that accepts a registration Authority and an item type and returns a list of tuples that contain all *public* items with a status of "Standard" or "Preferred Standard" *in that Registration Authority only*, as well as a the status object for that Authority. The item type should consist of the name of the app the item is from and the name of the item itself separated by a period (``.``). This requires the django ``django.contrib.contenttypes`` app is installed. If calling ``public_standards`` throws an exception or the item type requested is not found it safely returns an empty list. For example:: {% for item, status in registrationAuthority|public_standards:'aristotle_mdr.DataElement' %} {{ item }} - made standard on {{ status.registrationDate }}. {% endfor %} """ try: from django.contrib.contenttypes.models import ContentType app_label, model_name = itemType.lower().split('.', 1)[0:2] standard_states = [MDR.STATES.standard, MDR.STATES.preferred] return [ (i, i.statuses.filter(registrationAuthority=regAuth, state__in=standard_states).first()) for i in ContentType.objects.filter(app_label__in=fetch_metadata_apps()).get(app_label=app_label, model=model_name).model_class().objects.filter( statuses__registrationAuthority=regAuth, statuses__state__in=standard_states).public() ] except: return []
@register.simple_tag def paginator_get(request, pageNumber, pop=''): # http://stackoverflow.com/questions/2047622/how-to-paginate-django-with-other-get-variables dict_ = request.GET.copy() for p in pop.split(','): dict_.pop(p, None) dict_['page'] = pageNumber return dict_.urlencode() @register.simple_tag def ifeq(a, b, val, default=""): return val if a == b else default
[docs]@register.simple_tag def ternary(condition, a, b): """ A simple ternary tag - it beats verbose if/else tags in templates for simple strings If the ``condition`` is 'truthy' return ``a`` otherwise return ``b``. For example:: <a class="{% ternary item.is_public 'public' 'private' %}">{{item.name}}</a> """ if condition: return a else: return b
@register.filter def paginator_range(page, mode): page_range = list(page.paginator.page_range) if mode == "start": if page.number <= 5: # show 4,5,6 if page is 4, 5,6,7 if page is 5... return page_range[:max(5, page.number + 2)] else: return page_range[:3] if mode == "middle": if 5 < page.number < page.paginator.num_pages - 5: return page_range[page.number - 3:page.number + 2] if mode == "end": if page.number > page.paginator.num_pages - 5: return page_range[-5:] else: return page_range[-1:]
[docs]@register.filter def state_to_text(state): # @register.simple_tag """ This tag takes the integer value of a state for a registration status and converts it to its text equivilent. """ return MDR.STATES[int(state)]
@register.filter def unique_recent(recent): seen = [] out = [] for item in recent: if item._model and item.object and item.object.id not in seen: seen.append(item.object.id) out.append(item) return out
[docs]@register.simple_tag def zws(string): # Adds a zerowidth space before an em-dash """ ``zws`` or "zero width space" is used to insert a soft break near em-dashed. Since em-dashs are commonly used in Data Element Concept names, this helps them wrap in the right places. For example:: <h1>{% zws item.name %}</h1> """ from aristotle_mdr.utils.utils import strip_tags return mark_safe(strip_tags(string.replace("—", "&shy;—")))
[docs]@register.simple_tag def adminEdit(item): """ A tag for easily generating the link to an admin page for editing an item. For example:: <a href="{% adminEdit item %}">Advanced editor for {{item.name}}</a> """ app_name = item._meta.app_label return reverse("admin:%s_%s_change" % (app_name, item._meta.model_name), args=[item.id])
[docs]@register.inclusion_tag('aristotle_mdr/helpers/downloadMenu.html') def downloadMenu(item): """ Returns the complete download menu for a partcular item. It accepts the id of the item to make a download menu for, and the id must be of an item that can be downloaded, otherwise the links will show, but not work. For example:: {% downloadMenu item %} """ downloadOpts = fetch_aristotle_downloaders() from aristotle_mdr.utils import get_download_template_path_for_item downloadsForItem = [] app_label = item._meta.app_label model_name = item._meta.model_name for d in downloadOpts: template_type = d.template_type item_register = d.metadata_register if type(item_register) is not str: if item_register.get(app_label, []) == '__all__': downloadsForItem.append(d) elif model_name in item_register.get(app_label, []): downloadsForItem.append(d) else: if item_register == '__all__': downloadsForItem.append(d) elif item_register == '__template__': try: get_template(get_download_template_path_for_item(item, template_type)) downloadsForItem.append(d) except template.TemplateDoesNotExist: pass # This is ok. dlOptionsForItem = [] for dl_class in downloadsForItem: dlOptionsForItem.append(dl_class.get_class_info()) return {'item': item, 'download_options': dlOptionsForItem, }
@register.simple_tag def bootstrap_modal(_id, size=None): size_class = "" if size == 'lg': size_class = "modal-lg" elif size == 'sm': size_class = "modal-sm" modal = '<div id="%s" class="modal fade" data-backdrop="static"><div class="modal-dialog %s"><div class="modal-content"></div></div></div>' return mark_safe(modal % (_id, size_class))
[docs]@register.simple_tag def doc(item, field=None): """Gets the appropriate help text or docstring for a model or field. Accepts 1 or 2 string arguments: If 1, returns the docstring for the given model in the specified app. If 2, returns the help_text for the field on the given model in the specified app. """ from aristotle_mdr.utils.doc_parse import parse_rst, parse_docstring ct = item if field is None: model_name = ct._meta.model_name title, body, metadata = parse_docstring(ct.__doc__) if title: title = parse_rst(title, 'model', _('model:') + model_name) if body: body = parse_rst(body, 'model', _('model:') + model_name) return title else: if ct._meta.get_field(field).help_text: return _(ct._meta.get_field(field).help_text) else: # return _("No help text for the field '%(field)s' found on the model '%(model)s' in the app '%(app)s'") % {'app':app_label,'model':model_name,'field':field} return _("No help text for the field '%(field)s' found for the model '%(model)s'") % { 'model': item.get_verbose_name(), 'field': field}
@register.filter def can_use_action(user, bulk_action, *args): from aristotle_mdr.views.bulk_actions import get_bulk_actions bulk_action = get_bulk_actions().get(bulk_action) return bulk_action['can_use'](user) @register.filter def template_path(item, _type): from aristotle_mdr.utils import get_download_template_path_for_item _type, subpath = _type.split(',') return get_download_template_path_for_item(item, _type, subpath) @register.filter def visibility_text(item): visibility = _("hidden") if item._is_locked: visibility = _("locked") if item._is_public: visibility = _("public") return visibility @register.filter def is_active_module(module_name): from aristotle_mdr.utils.utils import is_active_module return is_active_module(module_name) @register.filter def is_active_extension(extension_name): from aristotle_mdr.utils.utils import is_active_extension return is_active_extension(extension_name) @register.filter def get_dataelements_from_m2m(ded, field_name): throughmodel = getattr(ded, field_name).through throughs = throughmodel.objects.filter(data_element_derivation=ded).select_related('data_element') de_list = [] for through_item in throughs: de_list.append(through_item.data_element) return de_list @register.filter def distinct_workgroups_count(user): return user.workgroupmembership_set.all().count() @register.filter def distinct_members_count(workgroup): return workgroup.members.all().count() @register.filter def get_item(dictionary, key): return dictionary.get(key) @register.simple_tag(takes_context=True) def has_group_perm(context, group, permission): request = context['request'] if not request.user.is_authenticated: return False return group.user_has_permission(request.user, permission) @register.filter def publish_item_url(item): return reverse('aristotle_publishing:publish_item', args=[item._meta.model_name, item.id]) @register.filter def publish_registry_item_url(item): return reverse('aristotle_publishing:publish_registry_item', args=[item._meta.model_name, item.id]) @register.filter def can_edit_label(label, user): return perms.user_can_edit_issue_label(user, label) @register.inclusion_tag('aristotle_mdr_help/helpers/field_icon.html', takes_context=True) def field_help_icon(context, item_or_model, field_name): klass = item_or_model.__class__ return { 'app': klass._meta.app_label, 'model_name': klass._meta.model_name, 'field_name': field_name, 'model_class': klass, } @register.inclusion_tag('aristotle_mdr/helpers/styled_difference.html', takes_context=True) def render_difference(context, difference): raw = False if 'raw' in context: raw = context['raw'] return {'difference': difference, 'raw': raw} @register.inclusion_tag('aristotle_mdr/helpers/difference_url.html') def transform_difference(difference): return {'difference': difference} @register.filter() def get_field(content_type, field_name): return content_type.model_class()._meta.get_field(field_name)
[docs]@register.simple_tag def get_status_from_dict(dictionary, current_status, key, with_icon=True): """ Get the Status of a particular item from a dictionary mapping. :param dictionary: dictionary mapping that must contain key-value pairs where the key must correspond to the concept_id, and the value must correspond to the state id. :param current_status: string that represents the numerical form of the status object that belongs to the Data Element. :param key: string that represents the concept id to be looked up. :param with_icon: boolean value to add a Fontawesome icon. :return: HTML with the name of the corresponding status state. """ state_value = dictionary.get(key, None) if state_value: if with_icon: if current_status == str(state_value): element = '<em><spam style="display:inline-block">[%s]&nbsp;<span class="text-success"><i class="fa fa-check"></i></span></span></em>' else: element = '<em><span class="text-danger">[%s]&nbsp;<i class="fa fa-times"></i></span></em>' else: element = '<em>[%s]&nbsp;<span class="text-success"></span></em>' return mark_safe(element % (STATES[state_value])) else: return ""
[docs]@register.filter def append_asterisk_if_required(field): """ Add an asterisk symbol to the required fields of a form. Usage: {{ field | append_asterisk_if_required }} Thanks to Moses Koledoye: https://stackoverflow.com/questions/37389855/django-label-tag-required-asterisk """ if field.field.required: return field.label + ': *' else: return field.label + ':'