Skip to main content

Micro-library to easily write custom Django template tags

Project description

https://img.shields.io/travis/edoburu/django-tag-parser/master.svg?branch=master https://img.shields.io/pypi/v/django-tag-parser.svg https://img.shields.io/pypi/l/django-tag-parser.svg https://img.shields.io/codecov/c/github/edoburu/django-tag-parser/master.svg

django-tag-parser

A micro-library to easily write custom Django template tags.

Features:

  • Functions to parse tags, especially: “args”, “kwargs”, and “as varname” syntax.

  • Real OOP classes to write custom inclusion tags.

Functions:

  • parse_token_kwargs: split a token into the tag name, args and kwargs.

  • parse_as_var: extract the “as varname” from a token.

Base classes (in tag_parser.basetags):

  • BaseNode: A template Node object which features some basic parsing abilities.

  • BaseInclusionNode: a Node that has inclusion_tag like behaviour, but allows to override the template_name dynamically.

  • BaseAssignmentNode: a Node that returns the value in the context, using the as var syntax.

  • BaseAssignmentOrOutputNode: a Node that either displays the value, or inserts it in the context.

  • BaseAssignmentOrInclusionNode: a class that allows a {% get_items template="..." %} and {% get_items as var %} syntax.

The base classes allows to implement @register.simple_tag, @register.inclusion_tag and @register.assignment_tag like functionalities, while still leaving room to extend the parsing, rendering or syntax validation. For example, not all arguments need to be seen as template variables, filters or literal keywords.

As of v3.0, the @template_tag decorator is no longer needed. Use @register.tag("name") directly on the class names.

Installation

First install the module, preferably in a virtual environment. It can be installed from PyPI:

pip install django-tag-parser

Examples

At the top of your template tags library, always include the standard Django register variable and our template_tag decorator:

from django.template import Library
from tag_parser import template_tag

register = Library()

Arguments and keyword arguments

To parse a syntax like:

{% my_tag "arg1" keyword1="bar" keyword2="foo" %}

use:

from django.template import Library
from tag_parser.basetags import BaseNode

register = Library()


@register.tag('my_tag')
class MyTagNode(BaseNode):
    max_args = 1
    allowed_kwargs = ('keyword1', 'keyword2',)

    def render_tag(self, context, *tag_args, **tag_kwargs):
        return "Tag Output"

Inclusion tags

To create an inclusion tag with overwritable template_name:

{% my_include_tag "foo" template="custom/example.html" %}

use:

from django.template import Library
from tag_parser.basetags import BaseInclusionNode

register = Library()

@register.tag("my_include_tag")
class MyIncludeTag(BaseInclusionNode):
    template_name = "mytags/default.html"
    max_args = 1

    def get_context_data(self, parent_context, *tag_args, **tag_kwargs):
        (foo,) = *tag_args
        return {
            'foo': foo
        }

The get_template_name() method can be overwritten too to support dynamic resolving of template names. By default it checks the template tag_kwarg, and template_name attribute. Note the template nodes are cached afterwards, it’s not possible to return random templates at each call.

Assignment tags

To create assignment tags that can either render itself, or return context data:

{% get_tags template="custom/example.html" %}
{% get_tags as popular_tags %}

use:

from django.template import Library
from tag_parser.basetags import BaseAssignmentOrInclusionNode

register = Library()


@register.tag('get_tags')
class GetPopularTagsNode(BaseAssignmentOrInclusionNode):
    template_name = "myblog/templatetags/popular_tags.html"
    context_value_name = 'tags'
    allowed_kwargs = (
        'order', 'orderby', 'limit',
    )

    def get_value(self, context, *tag_args, **tag_kwargs):
        return query_tags(**tag_kwargs)   # Something that reads the tags.

Block tags

To have a “begin .. end” block, define end_tag_name in the class:

{% my_tag keyword1=foo %}
    Tag contents, possibly other tags.
{% end_my_tag %}

use:

from django.template import Library
from tag_parser.basetags import BaseAssignmentOrInclusionNode

register = Library()


@register.tag('my_tag')
class MyTagNode(BaseNode):
    max_args = 1
    allowed_kwargs = ('keyword1', 'keyword2',)
    end_tag_name = 'end_my_tag'

    def render_tag(self, context, *tag_args, **tag_kwargs):
        # Render contents inside
        return self.nodelist.render(context)

Custom parsing

With the standard Node class from Django, it’s easier to implement custom syntax. For example, to parse:

{% getfirstof val1 val2 as val3 %}

use:

from django.template import Library, Node, TemplateSyntaxError
from tag_parser import parse_token_kwargs, parse_as_var

register = Library()


@register.tag('getfirstof')
class GetFirstOfNode(Node):
    def __init__(self, options, as_var):
        self.options = options    # list of FilterExpression nodes.
        self.as_var = as_var

    @classmethod
    def parse(cls, parser, token):
        bits, as_var = parse_as_var(parser, token)
        tag_name, options, _ = parse_token_kwargs(parser, bits, allowed_kwargs=())

        if as_var is None or not choices:
            raise TemplateSyntaxError("Expected syntax: {{% {0} val1 val2 as val %}}".format(tag_name))

        return cls(options, as_var)

    def render(self, context):
        value = None
        for filterexpr in self.options:
            # The ignore_failures argument prevents that the value becomes TEMPLATE_STRING_IF_INVALID.
            value = filterexpr.resolve(context, ignore_failures=True)
            if value is not None:
                break

        context[self.as_var] = value
        return ''

Contributing

This module is designed to be generic. In case there is anything you didn’t like about it, or think it’s not flexible enough, please let us know. We’d love to improve it!

If you have any other valuable contribution, suggestion or idea, please let us know as well because we will look into it. Pull requests are welcome too. :-)

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django-tag-parser-3.2.tar.gz (15.2 kB view details)

Uploaded Source

Built Distribution

django_tag_parser-3.2-py2.py3-none-any.whl (17.4 kB view details)

Uploaded Python 2 Python 3

File details

Details for the file django-tag-parser-3.2.tar.gz.

File metadata

  • Download URL: django-tag-parser-3.2.tar.gz
  • Upload date:
  • Size: 15.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.13.0 pkginfo/1.5.0.1 requests/2.10.0 setuptools/41.4.0 requests-toolbelt/0.9.1 tqdm/4.32.2 CPython/2.7.17

File hashes

Hashes for django-tag-parser-3.2.tar.gz
Algorithm Hash digest
SHA256 903cd4bf3692c8c060770ca2b9d6e74e4ee52d6078308852acffe1ef4a49e0f7
MD5 4160554fe9beae29ebc86aea67107df0
BLAKE2b-256 305442caebce5b09aa8e630c088eed06e58e287f51845b08bef276cd618e4933

See more details on using hashes here.

File details

Details for the file django_tag_parser-3.2-py2.py3-none-any.whl.

File metadata

  • Download URL: django_tag_parser-3.2-py2.py3-none-any.whl
  • Upload date:
  • Size: 17.4 kB
  • Tags: Python 2, Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.13.0 pkginfo/1.5.0.1 requests/2.10.0 setuptools/41.4.0 requests-toolbelt/0.9.1 tqdm/4.32.2 CPython/2.7.17

File hashes

Hashes for django_tag_parser-3.2-py2.py3-none-any.whl
Algorithm Hash digest
SHA256 92e5da35d8774a6ab4e0ac0b3b2229b8a0b78a61c5a3b0fd159255f0a66c08d7
MD5 3f8e37ea8428d6758e40224417966c10
BLAKE2b-256 167ffedeec26fa0fa27d0d8cd3169894ea447e6bd7ff3a21154ee0d54804661c

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page