Django的模板层

  将页面的设计和Python的代码分离开会更干净简洁更容易维护。 我们可以使用 Django的 模板系统 (Template System)来实现这种模式,这就是本章要具体讨论的问题。

  django的模板:HTML代码+模板语法:

def timer(request):
    import time
    now_time = time.time()
    # html = "<html><body>现在时刻:<h1>%s.</h1></body></html>" % now_time
    # return HttpResponse(html)
    return render(request,'timer.html',{'timer':now_time})

模板语法之变量:

  在django模板中遍历复杂数据结构的关键是句点字符。

def index(request):
    import datetime
    now_time = datetime.datetime.now()
    l = [123,456,789]
    dic = {'name':'qingqiu','age':17}
    class Person:
        def __init__(self,name,age):
            self.name = name
            self.age = age
        def sing(self):
            return '唱歌...'
    p1 = Person('qingqiu',17)
    p2 = Person('xiaobai',17)
    person_list = [p1,p2]
    return render(request,'index.html',{'timer':now_time,'l':l,'dic':dic,'person_list':person_list})
views.py
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<h3>hello</h3>
<p>{{ timer }}</p>
<p>{{ timer.strptime }}</p>
<p>{{ l }}</p>
<p>{{ l.0 }}</p>
<p>{{ l.1 }}</p>
<p>{{ l.2 }}</p>
<p>{{ dic }}</p>
<p>{{ dic.name }}</p>
<p>{{ dic.age }}</p>
<p>{{ person_list }}</p>
<p>{{ person_list.0.name }}</p>
<p>{{ person_list.0.age }}</p>
<p>{{ person_list.1.name }}</p>
<p>{{ person_list.1.age }}</p>
<p>{{ person_list.1.sing }}</p>


</body>
</html>
template.index

模板之过滤器:

  语法:{{ obj | filter_name:param }}

  1,default: 

  {{ obj|default:'没有参数' }}

  如果一个变量是false或者为空,则使用给定的默认值,否则,使用变量的值。

  2,length

  {{ obj|length }}

  返回值的长度,它对字符串和列表都起作用。

  3,filesizeformat:

  将值(整形)格式化为一个文件大小。

  4,date

  {{ value|date:"Y-m-d" }} 

  如果 value = datetime.datetime.now()

  5,slice

  {{ value | slice : 起始位置:结束位置:步长 }}  

  6,truncatechars

  {{ value|truncatechars:6 }}  

  数字是加上3(省略号)的总值。

  如果字符串字符多于指定的字符串数量,那么会被截断,阶段的字符串将以可翻译的省略号序列(‘...’)结尾。

  7,safe

  Django的模板中会对HTML标签和JS等语法标签进行自动转义,原因显而易见,这样是为了安全。但是有的时候我们可能不希望这些HTML元素被转义,比如我们做一个内容管理系统,后台添加的文章中是经过修饰的,这些修饰可能是通过一个类似于FCKeditor编辑加注了HTML修饰符的文本,如果自动转义的话显示的就是保护HTML标签的源文件。为了在Django中关闭HTML的自动转义有两种方式,如果是一个单独的变量我们可以通过过滤器“|safe”的方式告诉Django这段代码是安全的不必转义。比如:

  value = "<a href = 'http://www.baidu.com'>click</a>"

  {{ value|safe}}

  最好在href中写上绝对地址,否则它会在你原有的地址后面加上你的href连接。

  8,add

  {{ value|add:100}}

模板之标签:

  for标签:

  遍历每一个元素:

{% for person in person_list %}
    <p>{{ person.name }}</p>
{% endfor %}

  可以利用{% for obj in list reversed %}反向完成循环。

  注:循环序号可以通过{{forloop}}显示

forloop.counter        从1开始排序

forloop.counter0        从0开始排序

forloop.revcounter        倒序,最后一个为1

forloop.revcounter0        倒序,最后一个为0

forloop.first        如果是第一次遍历返回true,否则返回false

forloop.last       如果是最后一次遍历返回true,否则返回false

  for ... empty

  for 标签带有一个可选的{{ % empty %}} 从句,以便在给出的组是空的或者没有被找到时,可以有所操作。

{% for person in person_list %}
    <p>{{ person.name }}</p>

{% empty %}
    <p>sorry,no person here</p>
{% endfor %}

  if 标签:

  {% if 条件 %} 会对一个变量求值,如果它的值是True,(存在,不为空,且不是boolean类型的false值),对应的内容会输出。
  

{% if num > 100 %}
    <p>大于100</p>
{% elif num < 100 %}
    <p>小于100</p>
{% else %}
    <p>等于100</p>
{% endif %}

  for 和 if 搭配使用:

{% endfor %}
<hr>
{% for person in person_list %}
    <span>{{ person.name }}</span>
    {% if not forloop.last %}
        ,
    {% endif %}

{% endfor %}


# 清秋,小白   以逗号隔开

  with 标签:

  使用一个简单的名字缓存一个复杂的变量,当你需要使用一个繁琐的方法(比如访问数据库)很多次的时候是非常方便的。

  简而言之就是起一个别名,方便调用。

例如:
{% with p1=person_list.0 %}
    <p>{{ p1.name }},{{ p1.age }}</p>
{% endwith %}

  第二种方法:

{% with person_list.0 as p1  %}
    <p>{{ p1.name }},{{ p1.age }}</p>
{% endwith %}

  crsf_taken

  这个标签用于跨站请求伪造保护。

自定义标签和过滤器:

  1,在settings中的INSTALLED_APPS配置当前app,不然django无法找到自定义的simple_tag。

  2,在app中创建templatetags模块(模块名只能是templatetags)

  3,创建任意 .py文件, 如 my_tags.py。

from django import template
from django.utils.safestring import mark_safe
register = template.Library()

#  以上三行固定写法,不能更改,变量名也不行。

@register.filter   # 这样在一个函数上加一个装饰器,就可以使函数成为了过滤器。
def mult_filter(x,y):
    return x*y

@register.simple_tag    # 这样在函数上面加这个装饰器,相当于自定义了一个标签
def multi_tag(x,y,z):
    return x*y*z

@register.filter
def link_tag(href):
    return mark_safe("<a href=%s>click</a>"%href)
    # mark_safe是在python中通过django的安全保护机制。需要导入模块。
templatetags.my_filter_tags
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<h1>自定义过滤器,标签</h1>
{% load my_filter %}

{# 一定要通过load 模板 将 自定义过滤器或标签的模板导入,这样才能使用#}
<p>{{ num|mult_filter:2 }}</p>
{# num 为第一个参数,冒号后的数值为第二个变量,且只能接收两个变量。 #}

{#{% multi_tag 12 12 %}#}
{% multi_tag 2 2 2 %}
{# 标签可以传入多个参数,用空格隔开,在这里中传入参数,一定要与python文件中传入的形参数量一致。#}

<p>{{ "http://www.baidu.com"|link_tag }}</p>>
{# 网址作为参数传入到过滤器函数中#}
</body>
</html>
index.html

模板继承:

  1,include:语法:{% include '模板名' %}

{% include 'advertise.html' %}

  相当于直接把一个模板直接搬了过来。

  Django模板引擎中最强大也是最复杂的部分就是模板继承了,模板继承可以让您创建一个基本的“骨架”模板,它包含您站点中全部元素,并且可以定义能够被子模板覆盖的blocks.

  2,extend

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    {% block title %}
    <title>title</title>
    {% endblock %}

    <link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css"
          integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
    <style>
        * {
            margin: 0;
            padding: 0;
        }

        .header{
            width: 100%;
            height: 60px;
            background-color: #369;
        }

        .title{
            line-height: 60px;
            color: white;
            font-weight: 100;
            margin-left: 20px;
            font-size: 20px;
        }

        .container{
            margin-top: 20px;
        }
    </style>
</head>
<body>

<div class="header">
    <p class="title">
        路飞学诚
    </p>
</div>

<div class="container">
    <div class="row">
        <div class="col-md-3">
            <div class="panel panel-danger">
                <div class="panel-heading"><a href="http://127.0.0.1:8000/index/">首页</a></div>
                <div class="panel-body">
                    Panel content
                </div>
            </div>
            <div class="panel panel-success">
                <div class="panel-heading"><a href="http://127.0.0.1:8000/authors/">作者</a></div>
                <div class="panel-body">
                    Panel content
                </div>
            </div>
            <div class="panel panel-warning">
                <div class="panel-heading"><a href="http://127.0.0.1:8000/articles/">文章管理</a></div>
                <div class="panel-body">
                    Panel content
                </div>
            </div>
        </div>
        <div class="col-md-9">
            {% block content %}
                <h3>详细内容</h3>
            {% endblock %}
        </div>
    </div>
</div>

</body>
</html>

  上面这个模板,我们把它叫做base.html,它定义了一个可以用于两列排版页面的简单HTML骨架。“子模板”的工作是用它们的内容填充空的blocks。

  子模板 index.html代码:

{% extends 'base.html' %}

{% block title %}
    <title>首页</title>
{% endblock %}

{% block content %}
    <div class="jumbotron">
        <h1>hello,index!</h1>
        <p><a  class= "btn btn_danger btn-lg" href="#" role="button"> learn more</a></p>
    </div>
{% endblock %}

  extends 标签是这里的关键,它告诉模板引擎,这个模板继承了‘base.html’模板。当处理这个模板时,会先继承。所以继承最好要写在首行,不能写在下面。

  模板引擎将注意到base.html模板中的两个block标签,并用自模板中的内容覆盖这些block内的内容。

  注意:如果子模板中并没有定义 sidebar block , 系统会使用父模板中的值。父模板中的{% block %}  标签中的内容总是被用作备选内容。如果子模版中有内容,就会把夫模板中的内容覆盖。

  拥有base.html的优点:是解决了代码的复用。

  提示:

    1,如果你在模板中使用{% extends %}标签,它必须是模板中的第一个标签,其他任何情况下,模板继承都将无法工作。

    2,在base模板中设置越多的{% block %}标签越好。

    3,如果你发现自己在大量的模板中复制内容,那可能意味着你应该把内容移动到父模板中的一个{% block %} 中。

    4,为了更好的可读性,可以给{% block %} 标签加一个 {% endblock %}标签 一个名字。如:

{ % block content % }
...
{ % endblock content% }

    5,不能在一个模板中定义多个相同的名字的block标签。

原文地址:https://www.cnblogs.com/stfei/p/9226518.html