Django中自定义标签的所有步骤

编写编译函数

当遇到一个模板标签(template tag)时,模板解析器就会把标签包含的内容,以及模板解析器自己作为参数调用一个python函数。 这个函数负责返回一个和当前模板标签内容相对应的节点(Node)的实例。

例如,写一个显示当前日期的模板标签:{% current_time %}。该标签会根据参数指定的 strftime 格式(参见:http://www.djangoproject.com/r/python/strftime/)显示当前时间。首先确定标签的语法是个好主意。 在这个例子里,标签应该这样使用:

<p>The time is {% current_time "%Y-%m-%d %I:%M %p" %}.</p> 

注意

没错, 这个模板标签是多余的,Django默认的 {% now %} 用更简单的语法完成了同样的工作。 这个模板标签在这里只是作为一个例子。

这个函数的分析器会获取参数并创建一个 Node 对象:

from django import template  
register = template.Library()  
def do_current_time(parser, token):     
 try:         # split_contents() knows not to split quoted strings.         
 tag_name, format_string = token.split_contents()     
 except ValueError:         
 msg = '%r tag requires a single argument' % token.split_contents()[0]         
 raise template.TemplateSyntaxError(msg)     
 return CurrentTimeNode(format_string[1:-1]) 

这里需要说明的地方很多:

  • 每个标签编译函数有两个参数,parsertokenparser是模板解析器对象。 我们在这个例子中并不使用它。 token是正在被解析的语句。

  • token.contents 是包含有标签原始内容的字符串。 在我们的例子中,它是 'current_time "%Y-%m-%d %I:%M %p"'

  • token.split_contents() 方法按空格拆分参数同时保证引号中的字符串不拆分。 应该避免使用 token.contents.split() (仅使用Python的标准字符串拆分)。 它不够健壮,因为它只是简单的按照所有空格进行拆分,包括那些引号引起来的字符串中的空格。

  • 这个函数可以抛出 django.template.TemplateSyntaxError ,这个异常提供所有语法错误的有用信息。

  • 不要把标签名称硬编码在你的错误信息中,因为这样会把标签名称和你的函数耦合在一起。 token.split_contents()[0]总是记录标签的名字,就算标签没有任何参数。

  • 这个函数返回一个 CurrentTimeNode (稍后我们将创建它),它包含了节点需要知道的关于这个标签的全部信息。 在这个例子中,它只是传递了参数 "%Y-%m-%d %I:%M %p" 。模板标签开头和结尾的引号使用 format_string[1:-1] 除去。

  • 模板标签编译函数 必须 返回一个 Node 子类,返回其它值都是错的。

编写模板节点

编写自定义标签的第二步就是定义一个拥有 render() 方法的 Node 子类。 继续前面的例子,我们需要定义 CurrentTimeNode

import datetime 
 
class CurrentTimeNode(template.Node):     
 def __init__(self, format_string):         
 self.format_string = str(format_string)  
    
 def render(self, context):         
 now = datetime.datetime.now()         
 return now.strftime(self.format_string) 

这两个函数( __init__()render() )与模板处理中的两步(编译与渲染)直接对应。 这样,初始化函数仅仅需要存储后面要用到的格式字符串,而 render() 函数才做真正的工作。

与模板过滤器一样,这些渲染函数应该静静地捕获错误,而不是抛出错误。 模板标签只允许在编译的时候抛出错误。

注册标签

最后,你需要用你模块的Library 实例注册这个标签。 注册自定义标签与注册自定义过滤器非常类似(如前文所述)。 只需实例化一个 template.Library 实例然后调用它的 tag() 方法。 例如:

register.tag('current_time', do_current_time) 

tag() 方法需要两个参数:

  • 模板标签的名字(字符串)。

  • 编译函数。

和注册过滤器类似,也可以在Python2.4及其以上版本中使用 register.tag装饰器:

@register.tag(name="current_time") def do_current_time(parser, token):     
# ...  

@register.tag def shout(parser, token):     # ... 

如果你像在第二个例子中那样忽略 name 参数的话,Django会使用函数名称作为标签名称。

在上下文中设置变量

前一节的例子只是简单的返回一个值。 很多时候设置一个模板变量而非返回值也很有用。 那样,模板作者就只能使用你的模板标签所设置的变量。

要在上下文中设置变量,在 render() 函数的context对象上使用字典赋值。 这里是一个修改过的 CurrentTimeNode ,其中设定了一个模板变量 current_time ,并没有返回它:

class CurrentTimeNode2(template.Node):     
 def __init__(self, format_string):         
 self.format_string = str(format_string) 
     
 def render(self, context):         
 now = datetime.datetime.now()         
 context['current_time'] = now.strftime(self.format_string)         
 return '' 

(我们把创建函数do_current_time2和注册给current_time2模板标签的工作留作读者练习。)

注意 render() 返回了一个空字符串。 render() 应当总是返回一个字符串,所以如果模板标签只是要设置变量, render() 就应该返回一个空字符串。

你应该这样使用这个新版本的标签:

{% current_time2 "%Y-%M-%d %I:%M %p" %} <p>The time is {{ current_time }}.</p> 

但是 CurrentTimeNode2 有一个问题: 变量名 current_time 是硬编码的。 这意味着你必须确定你的模板在其它任何地方都不使用 {{ current_time }} ,因为 {% current_time2 %} 会盲目的覆盖该变量的值。

一种更简洁的方案是由模板标签来指定需要设定的变量的名称,就像这样:

{% get_current_time "%Y-%M-%d %I:%M %p" as my_current_time %} 
<p>The current time is {{ my_current_time }}.</p> 

为此,你需要重构编译函数和 Node 类,如下所示:

import re  

class CurrentTimeNode3(template.Node):     
 def __init__(self, format_string, var_name):         
 self.format_string = str(format_string)         
 self.var_name = var_name    
  
 def render(self, context):         
 now = datetime.datetime.now()         
 context[self.var_name] = now.strftime(self.format_string)         
 return '' 
 
 def do_current_time(parser, token): # This version uses a regular expression to parse tag contents.     
 try: # Splitting by None == splitting by spaces.         
 tag_name, arg = token.contents.split(None, 1)     
 except ValueError:         
 msg = '%r tag requires arguments' % token.contents[0]         
 raise template.TemplateSyntaxError(msg) 
     
 m = re.search(r'(.*?) as (\w+)', arg)     
 if m:         
 fmt, var_name = m.groups()     
 else:         
 msg = '%r tag had invalid arguments' % tag_name         
 raise template.TemplateSyntaxError(msg)      
 if not (fmt[0] == fmt[-1] and fmt[0] in ('"', "'")):         
 msg = "%r tag's argument should be in quotes" % tag_name         
 raise template.TemplateSyntaxError(msg)      
 return CurrentTimeNode3(fmt[1:-1], var_name) 

现在 do_current_time() 把格式字符串和变量名传递给 CurrentTimeNode3

分析直至另一个模板标签

模板标签可以像包含其它标签的块一样工作(想想 {% if %} {% for %} 等)。 要创建一个这样的模板标签,在你的编译函数中使用 parser.parse()

标准的 {% comment %} 标签是这样实现的:

def do_comment(parser, token):     
 nodelist = parser.parse(('endcomment',))     
 parser.delete_first_token()     
 return CommentNode()  

class CommentNode(template.Node):     
 def render(self, context):         
 return '' 

parser.parse() 接收一个包含了需要分析的模板标签名的元组作为参数。 它返回一个django.template.NodeList实例,它是一个包含了所有Node对象的列表,这些对象是解析器在解析到任一元组中指定的标签之前遇到的内容.

因此在前面的例子中, nodelist 是在 {% comment %} {% endcomment %} 之间所有节点的列表,不包括 {% comment %} {% endcomment %} 自身。

parser.parse() 被调用之后,分析器还没有清除 {% endcomment %} 标签,因此代码需要显式地调用 parser.delete_first_token() 来防止该标签被处理两次。

之后 CommentNode.render() 只是简单地返回一个空字符串。 在 {% comment %} {% endcomment %} 之间的所有内容都被忽略。

分析直至另外一个模板标签并保存内容

在前一个例子中, do_comment() 抛弃了{% comment %} {% endcomment %} 之间的所有内容。当然也可以修改和利用下标签之间的这些内容。

例如,这个自定义模板标签{% upper %},它会把它自己和{% endupper %}之间的内容变成大写:

{% upper %}     This will appear in uppercase, {{ user_name }}. {% endupper %} 

就像前面的例子一样,我们将使用 parser.parse() 。这次,我们将产生的 nodelist 传递给 Node

def do_upper(parser, token):     
 nodelist = parser.parse(('endupper',))     
 parser.delete_first_token()     
 return UpperNode(nodelist)  

class UpperNode(template.Node):     
 def __init__(self, nodelist):         
 self.nodelist = nodelist      

 def render(self, context):         
 output = self.nodelist.render(context)         
 return output.upper() 

这里唯一的一个新概念是 UpperNode.render() 中的 self.nodelist.render(context) 。它对节点列表中的每个 Node 简单的调用 render()

更多的复杂渲染示例请查看 django/template/defaulttags.py 中的 {% if %} {% for %} {% ifequal %}{% ifchanged %} 的代码。

简单标签的快捷方式

许多模板标签接收单一的字符串参数或者一个模板变量引用,然后独立地根据输入变量和一些其它外部信息进行处理并返回一个字符串。 例如,我们先前写的current_time标签就是这样一个例子。 我们给定了一个格式化字符串,然后它返回一个字符串形式的时间。

为了简化这类标签,Django提供了一个帮助函数simple_tag。这个函数是django.template.Library的一个方法,它接受一个只有一个参数的函数作参数,把它包装在render函数和之前提及过的其他的必要单位中,然后通过模板系统注册标签。

我们之前的的 current_time 函数于是可以写成这样:

def current_time(format_string):     
 try:         
 return datetime.datetime.now().strftime(str(format_string))     
 except UnicodeEncodeError:         
 return ''  register.simple_tag(current_time) 

在Python 2.4中,也可以使用装饰器语法:

@register.simple_tag def current_time(token):     # ... 

有关 simple_tag 辅助函数,需要注意下面一些事情:

  • 传递给我们的函数的只有(单个)参数。

  • 在我们的函数被调用的时候,检查必需参数个数的工作已经完成了,所以我们不需要再做这个工作。

  • 参数两边的引号(如果有的话)已经被截掉了,所以我们会接收到一个普通Unicode字符串。

包含标签

另外一类常用的模板标签是通过渲染 其他 模板显示数据的。 比如说,Django的后台管理界面,它使用了自定义的模板标签来显示新增/编辑表单页面下部的按钮。 那些按钮看起来总是一样的,但是链接却随着所编辑的对象的不同而改变。 这就是一个使用小模板很好的例子,这些小模板就是当前对象的详细信息。

这些排序标签被称为 包含标签 。如何写包含标签最好通过举例来说明。 让我们来写一个能够产生指定作者对象的书籍清单的标签。 我们将这样利用标签:

{% books_for_author author %} 

结果将会像下面这样:

<ul>     
 <li>The Cat In The Hat</li>     
 <li>Hop On Pop</li>     
 <li>Green Eggs And Ham</li> 
</ul> 

首先,我们定义一个函数,通过给定的参数生成一个字典形式的结果。 需要注意的是,我们只需要返回字典类型的结果就行了,不需要返回更复杂的东西。 这将被用来作为模板片段的内容:

def books_for_author(author):     
 books = Book.objects.filter(authors__id=author.id)     
 return {'books': books} 

接下来,我们创建用于渲染标签输出的模板。 在我们的例子中,模板很简单:

<ul> {% for book in books %}     
 <li>{{ book.title }}</li> 
{% endfor %} </ul> 

最后,我们通过对一个 Library 对象使用 inclusion_tag() 方法来创建并注册这个包含标签。

在我们的例子中,如果先前的模板在 polls/result_snippet.html 文件中,那么我们这样注册标签:

register.inclusion_tag('book_snippet.html')(books_for_author) 

Python 2.4装饰器语法也能正常工作,所以我们可以这样写:

@register.inclusion_tag('book_snippet.html') def books_for_author(author):     # ... 

有时候,你的包含标签需要访问父模板的context。 为了解决这个问题,Django为包含标签提供了一个 takes_context 选项。 如果你在创建模板标签时,指明了这个选项,这个标签就不需要参数,并且下面的Python函数会带一个参数: 就是当这个标签被调用时的模板context。

例如,你正在写一个包含标签,该标签包含有指向主页的 home_link home_title 变量。 Python函数会像这样:

@register.inclusion_tag('link.html', takes_context=True) 
def jump_link(context):     
 return {'link': context['home_link'], 'title': context['home_title'],} 

(注意函数的第一个参数 必须 context 。)

模板 link.html 可能包含下面的东西:

Jump directly to <a href="{{ link }}">{{ title }}</a>. 

然后您想使用自定义标签时,就可以加载它的库,然后不带参数地调用它,就像这样:

{% jump_link %} 

你可能感兴趣的:(django)