与Jinja一起浏览2D列表

时间:2018-06-29 14:22:13

标签: python ajax flask jinja2

为简单起见,我将做一个假设,该假设与我目前的情况相同。我正在使用Flask作为后端来渲染前端的Jinja模板。假设我有一个班级列表,每个班级都是学生列表(具有属性和全部的Python类/节点)。我想一次渲染一个“学生班”,并有一个按钮可以循环到下一组学生。这是一个外观的示例:

app.py

@app.route('/', methods=['GET', 'POST'])
def get_students():
    groups = None
    # some calculations and algorithms
    groups = [['foo', 'bar'], ['baz', 'boo']]
    return render_template('index.html', groups=groups, index=0)

index.html

{% if groups %}
{% set display_group = groups[index] %}
    <ul id="students" class="">
        {% for student in display_group %}
            <li class="student">{{ student.name }}</li>
        {% endfor %}
    </ul>
{% endif %}
<button onclick="next_group()">Next Group</button>
<button onclick="prev_group()">Previous Group</button>

我希望这些按钮重新呈现此列表,就像索引分别增加/减少一样。我不希望通过网址参数(例如page_url / number)来完成此操作。如何在没有页面刷新的情况下实现这种效果?

2 个答案:

答案 0 :(得分:1)

这类似于分页。首先,您可以创建一个简单的分页对象,以更好地帮助您为新索引进行正确的页面查找,同时还可以控制下一页和上一页的索引:

import typing

class StudentList(typing.NamedTuple):
  name:str

class Pagination:
  def __init__(self, _num = 1):
     self.num = _num
     self.data = [['foo', 'bar'], ['baz', 'boo'], ['first', 'last']]
  @property
  def has_next(self):
     return self.num < len(self.data)
  @property
  def has_previous(self):
     return self.num > 0
  @property
  def next(self):
     return self.num + 1
  @property
  def previous(self):
     return self.num - 1
  def __iter__(self):
     for i in self.data[self.num-1]:
        yield StudentList(i)

接下来,为了创建动态查找,需要两段html:首页,使用javascript来控制按钮单击并与后端通信,而html返回为ajax向后端查询的一部分。首先,创建查询html

students_and_classes.html

<div class='student_class'>
 {%for student in lecture%} 
   <span class='student'>Name: {{student.name}}</span>
 {%endfor%}
 {%if lecture.has_previous%}
   <button id='previous_{{lecture.previous}}'>Previous</button>
 {%endif%}
 {%if lecture.has_next%}
   <button id='next_{{lecture.next}}'>Next</button>
 {%endif%}
 <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
 <script>
 $(document).ready(function() {
  $('button').click(function(event) {
   var result = event.target.id;
   $.ajax({
    url: "/update_page",
    type: "get",
    data: {results: result},
    success: function(response) {
    $("#pagination_results").html(response.html);
    },
    error: function(xhr) {
     $("#pagination_results").html('<p>Opps, something when wrong</p>');
    }
   });
  });
 });
</script>
</div>

第二,显示完整学生分页以及jqueryajax的页面:

main.html

<html>
  <body>
    <div id='pagination_results'>
      <div class='student_class'>
      {%for student in lecture%} 
        <span class='student'>Name: {{student.name}}</span>
      {%endfor%}
      {%if lecture.has_previous%}
        <button id='previous_{{lecture.previous}}'>Previous</button>
      {%endif%}
      {%if lecture.has_next%}
        <button id='next_{{lecture.next}}'>Next</button>
     {%endif%}
    </div>
    </div>
  </body>
  <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
  <script>
  $(document).ready(function() {
   $('button').click(function(event) {
    var result = event.target.id;
    $.ajax({
      url: "/update_page",
      type: "get",
      data: {results: result},
      success: function(response) {
      $("#pagination_results").html(response.html);
     },
     error: function(xhr) {
      $("#pagination_results").html('<p>Opps, something when wrong</p>');
    }
   });
 });
});
</script>
</html>

最后,在所需的路由(在本例中为'/')中,可以创建服务main.html的路由:

@app.route('/', methods = ['GET'])
def home():
  return flask.render_template('main.html', lecture=Pagination())

然后,需要创建从ajax GET方法接收数据的路由:

import re
@app.route('/update_page')
def update_page():
  _r = flask.request.args.get('results')
  _l = Pagination(int(re.findall('\d+$', _r)[0]))
  return flask.jsonify({'html':flask.render_template('students_and_classes.html', lecture=_l)})

注意:

    在您的项目中,self.data中的
  1. Pagination可以用数据库查询等代替
  2. StudentList用于将所有列表值都作为基本数据类型的模板的更清晰呈现。在您的示例中,没有必要,因为您提到列表已经存储了自定义类对象,并且可以用yield i来代替yield StudentList(i)

答案 1 :(得分:0)

这是JavaScript的工作,而不是jinja模板系统,当您将模板发送给客户端时,模板是根据组成模板的变量预先格式化的。您应该有一个端点,可以通过java脚本中的ajax调用来调用,而不是返回render_template而是返回组的json,并通过JS处理DOM,jinja将无法做到这一点,尽管您可以发送javascript通过Jinja模板系统构建。