X-Git-Url: http://git.ithinksw.org/philo.git/blobdiff_plain/5d30b71366b179449b54cd16b87ef7380338c064..91ff82666d936433802979e3051a363ae131ce13:/utils.py?ds=sidebyside diff --git a/utils.py b/utils.py index 6fc2cff..a968ac8 100644 --- a/utils.py +++ b/utils.py @@ -1,6 +1,8 @@ from django.db import models from django.contrib.contenttypes.models import ContentType from django.core.paginator import Paginator, EmptyPage +from django.template import Context +from django.template.loader_tags import ExtendsNode, ConstantIncludeNode class ContentTypeLimiter(object): @@ -103,3 +105,50 @@ def paginate(objects, per_page=None, page_number=1): objects = page.object_list return paginator, page, objects + + +LOADED_TEMPLATE_ATTR = '_philo_loaded_template' +BLANK_CONTEXT = Context() + + +def get_extended(self): + return self.get_parent(BLANK_CONTEXT) + + +def get_included(self): + return self.template + + +# We ignore the IncludeNode because it will never work in a blank context. +setattr(ExtendsNode, LOADED_TEMPLATE_ATTR, property(get_extended)) +setattr(ConstantIncludeNode, LOADED_TEMPLATE_ATTR, property(get_included)) + + +def default_callback(node, result_list): + result_list.append(node) + + if hasattr(node, 'child_nodelists'): + for nodelist_name in node.child_nodelists: + if hasattr(node, nodelist_name): + nodelist_crawl(getattr(node, nodelist_name), default_callback) + + # LOADED_TEMPLATE_ATTR contains the name of an attribute philo uses to declare a + # node as rendering an additional template. Philo monkeypatches the attribute onto + # the relevant default nodes and declares it on any native nodes. + if hasattr(node, LOADED_TEMPLATE_ATTR): + loaded_template = getattr(node, LOADED_TEMPLATE_ATTR) + if loaded_template: + nodelist_crawl(loaded_template.nodelist, default_callback) + + +def nodelist_crawl(nodelist, callback=default_callback, *args, **kwargs): + """This function crawls through a template's nodelist and the + nodelists of any included or extended templates, as determined by + the presence and value of on a node. Each + node will also be passed to a callback function for additional + processing, along with any additional args and kwargs.""" + for node in nodelist: + try: + callback(node, *args, **kwargs) + except: + raise # fail for this node \ No newline at end of file