热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

DjangoWeb开发【6】使用Ajax增强用户体验

Ajax及其优点Ajax实际上就是指异步Javascript与XML,它包含以下技术:HTML与CSSJavascriptXMLHttpRequestXMLAjax技术让客户端与服

  Ajax及其优点

  Ajax实际上就是指异步Javascript与XML,它包含以下技术:

  • HTML与CSS
  • Javascript
  • XMLHttpRequest
  • XML

  Ajax技术让客户端与服务器实现在后端通信,而不需要每次发送请求的时候都重载整个页面。Ajax有以下优点:

  • 更好的用户体验
  • 更好的性能,只需重新加载部分页面,而不是整个页面。

  在Django中使用Ajax框架

  这一节我们将选择合适Ajax框架,这一步不是必须的,但是使用Ajax框架使运用ajax更加简单,下面是使用Ajax框架的优点:

  • Javascript在不同浏览器中的实现是不一样的,使用Ajax框架可以屏蔽这其中的差异,这样一样,开发者只需要专注于代码的实现而不用考虑浏览器的差异与限制。
  • Javascript的标准函数与类比较少,而ajax库提供了大量的函数与类,既然轮子已经准备好了,又何必重新发明轮子呢?

  现在网络中有很多种js框架,如prototype、jQuery等等,这里我们选用jQuery,因为jQuery是一个轻量库,而且拥有广大的用户群,以及众多插件。

  下载安装jQuery

  从http://jquery.com/ 中下载最新的jQuery。将jquery.js放到/static/中,然后在基础模板中引用jquery.js,编辑templates/base.html。

<head>
  <title>Django Bookmarks | 
  {% block title %}{% endblock %}title>
  <link rel="stylesheet" href="/site_media/style.css" type="text/css" />
  <script type="text/Javascript" src="/static/jquery.js">script>
head>

  在上面的html中添加如下代码(红色部分),以方便在以后的页面中引用自身创建的js文件。

<head>
  <title>Django Bookmarks | {% block title %}{% endblock %}title>
  <link rel="stylesheet" href="/site_media/style.css" type="text/css"/>
  <script type="text/Javascript" src="/site_media/jquery.js">
  script>
  {% block external %}{% endblock %}
head>

  关于jQuery的知识,这里就不再详述,请自己查阅相关文档。

  实现Bookmarks的实时搜索

  我们将使用ajax实现bookmarks的实时搜索功能,它的内部原理很简单:当用户输入要搜索的关键字时,脚本就在后端工作,发送请求并获取结果返回,然后展示在同一个页面中,搜索结果不会重新加载页面,因此能够节省流量,并提供更好的用户体验。

  在开始编码之前,要记得一个原则,那就是必须编写的代码应该能够在ajax不支持的情况下也能正常工作,当然现在一般浏览器都支持ajax,这个问题也不是那么重要了。

  实现搜索

  我们编写一个简单的例子,通过标题名查找bookmarks。首先,创建一个搜索表单,编辑bookmarks/forms.py,添加如下代码:

class SearchForm(forms.Form):
    query = forms.CharField(
    label=Enter a keyword to search for,
      widget=forms.TextInput(attrs={size: 32})
)

  搜索表单只有一个字段,查找用到的关键字。

  接下来,创建视图函数,编辑bookmarks/views.py,添加如下代码:

def search_page(request):
  form = SearchForm()
  bookmarks = []
  show_results = False
  if request.GET.has_key(query):
    show_results = True
    query = request.GET[query].strip()
    if query:
      form = SearchForm({query : query}) 
      bookmarks =         Bookmark.objects.filter (title__icontains=query)[:10]
  variables = RequestContext(request, { form: form,
    bookmarks: bookmarks,
    show_results: show_results,
    show_tags: True,
    show_user: True
  })
  return render_to_response(search.html, variables)

  这里使用GET方法而不是POST方法提交表单,因为这里我们只是简单的查询数据,而不是创建或者删除数据。

  使用filter方法获取结果,它相当于SQL中的SELECT语句,filter方法中的参数格式如下:

  field__operator

  注意field与operator中为双下划线,field是我们指我们要根据这个字段进行查询,operator是指查询的方法。下面是常见的使用方法:

  • exact  完全相当
  • contains 包含参数值
  • startswith 以参数值开头
  • lt  比参数值小
  • gt 比参数值大

  除此之外,还有不区分大小写的iexact,icontains以及istartswith。

   接下来在templates中创建search.html:

{% extends "base.html" %}
{% block title %}Search Bookmarks{% endblock %}
{% block head %}Search Bookmarks{% endblock %}
{% block content %}
  <form id="search-form" method="get" action=".">
    {{ form.as_p }}
    <input type="submit" value="search" />
  form>
  <div id="search-results">
    {% if show_results %}
      {% include ‘bookmark_list.html‘ %}
    {% endif %}
  div>
{% endblock %}

  在urls.py中添加url:

urlpatterns = patterns(‘‘,
  # Browsing
  (r^$, main_page),
  (r^user/(\w+)/$, user_page),
  (r^tag/([^\s]+)/$, tag_page),
  (r^tag/$, tag_cloud_page),
  (r‘^search/$‘, search_page),
)

  在templates/base.html中添加如下代码,给导航菜单添加搜索链接:

<div id="nav">
  <a href="/">homea> |
  {% if user.is_authenticated %}
    <a href="/save/">submita> |
    <a href="/search/">searcha> |
    <a href="/user/{{ user.username }}/">
      {{ user.username }}a> |
    <a href="/logout/">logouta>
  {% else %}
    <a href="/login/">logina> |
    <a href="/register/">registera>
  {% endif %}
div>

  现在我们就拥有了搜索功能页面,接下来实现ajax获取数据,而不是通过重新加载页面。

  实现实时搜索

  为了实现实时搜索,需要做以下两件事:

  • 通过submit()方法拦截表单的默认提交方法,然后自定义处理表单的方法。
  • 使用Ajax在后台获取查询结果,然后插入页面中,这可以通过load()方法实现。

  jQuery提供了load()方法,可以从服务器加载指定页面,然后插入到指定元素中。它使用远端的页面URL作为参数。

  首先,我们对视图函数进行编辑,当request.GET字典中中包含ajax的键时,就返回bookmark_list.html。编辑bookmarks/views.py,修改search_page方法:

def search_page(request):
  [...]
  variables = RequestContext(request, {
    form: form,
    bookmarks: bookmarks,
    show_results: show_results,
    show_tags: True,
    show_user: True
  })
  if request.GET.has_key(ajax):
    return render_to_response(bookmark_list.html, variables)
  else:
    return render_to_response(search.html, variables)

  接下来,在static目录中创建search.js:

function search_submit() { 
  var query = $("#id_query").val(); 
  $("#search-results").load(
    "/search/?ajax&query=" + encodeURIComponent(query)
  );
  return false;
}

  返回false的作用是告诉浏览器在调用load方法之后,不再自动提交表单。

  然后在templates/search.html中添加如下链接:

{% extends "base.html" %}
{% block external %}
  <script type="text/Javascript" src="/site_media/search.js">
  script>
{% endblock %}
{% block title %}Search Bookmarks{% endblock %}
{% block head %}Search Bookmarks{% endblock %}
[...]

  最后在给search.js添加如下代码:

$(document).ready(function () {
    $("#search-form").submit(search_submit);
});

  这样就给表单添加了submit事件,这样就可以使用ajax提交表单了。

  实时编辑bookmarks

  编辑已经提交的内容在很多网站中都是很常见的任务,它通常通过在内容旁边提供一个编辑链接来实现,如果点击,这个链接就引导用户到一个可以编辑内容的页面,当用户提交表单之后,就重定向至内容页面。

  其实还有另外一种办法,不需要重定向至编辑页面,在当前页面就可编辑内容,所有的操作都发生在同一个页面,编辑表单与提交都是通过ajax实现的。

  上面的技术称之为实时编辑。

  实现编辑bookmarks功能

  回忆一下, 在bookmarks/views.py中,我们是这样实现bookmark_save_page视图函数的,如果用户尝试保存同一个URL,则bookmark只会进行更新,而不是再创建一个。

  实现编辑bookmarks功能还需要做以下两件事:

  • 将bookmark的url通过get方法传递给bookmark_save_page视图
  • 当bookmark_save_page接收到url参数时,就生成相应的编辑表单。

  在实现上面的代码之前,我们先来对bookmark_save_page进行简化,将保存bookmark的部分放置在另外一个函数中,这个函数名为_bookmark_save,函数前面的下划线告诉Python在导入views视图时,不导入这个函数。这个函数接收request与表单对象作为参数,编辑bookmarks/views.py,添加如下代码:

def _bookmark_save(request, form):
  # Create or get link.
  link, dummy =     Link.objects.get_or_create(url=form.clean_data[url])
  # Create or get bookmark.
  bookmark, created = Bookmark.objects.get_or_create(
    user=request.user,
    link=link
  )
  # Update bookmark title.
  bookmark.title = form.cleaned_data[title]
  # If the bookmark is being updated, clear old tag list.
  if not created:
    bookmark.tag_set.clear()
  # Create new tag list.
    tag_names = form.cleaned_data[tags].split()
  for tag_name in tag_names:
    tag, dummy = Tag.objects.get_or_create(name=tag_name)
    bookmark.tag_set.add(tag)
  # Save bookmark to database and return it.
  bookmark.save()
  return bookmark

  继续编辑views.py,修改bookmark_save_page:

@login_required
def bookmark_save_page(request):
  if request.method == POST:
    form = BookmarkSaveForm(request.POST)
    if form.is_valid():
      bookmark = _bookmark_save(request, form)
      return HttpResponseRedirect(
        /user/%s/ % request.user.username
      )
  else:
    form = BookmarkSaveForm()
  variables = RequestContext(request, {
    form: form
  })
  return render_to_response(bookmark_save.html, variables)

  现在,bookmark_save_page视图的逻辑如下:

if there is POST data:
  Validate and save bookmark.
  Redirect to user page.
else:
  Create an empty form.
Render page.

  为了实现编辑功能,需要对这个逻辑进行如下修改:

if there is POST data:
  Validate and save bookmark.
  Redirect to user page.
else if there is a URL in GET data:
  Create a form an populate it with the URLs bookmark.
else:
  Create an empty form.
Render page.

  下面实现上述伪代码,编辑bookmark_save_page视图函数:

from django.core.exceptions import ObjectDoesNotExist
@login_required
def bookmark_save_page(request):
  if request.method == POST:
    form = BookmarkSaveForm(request.POST)
    if form.is_valid():
      bookmark = _bookmark_save(request, form)
      return HttpResponseRedirect(
        /user/%s/ % request.user.username
      )
  elif request.GET.has_key(url):
    url = request.GET[url]
    title = ‘‘
    tags = ‘‘
    try:
      link = Link.objects.get(url=url)
      bookmark = Bookmark.objects.get(
        link=link,
        user=request.user
      )
      title = bookmark.title
      tags =  .join(
        tag.name for tag in bookmark.tag_set.all()
      )
    except ObjectDoesNotExist:
      pass
    form = BookmarkSaveForm({
      url: url,
      title: title,
      tags: tags
    })
  else:
    form = BookmarkSaveForm()
      variables = RequestContext(request, {
      form: form
    })
  return render_to_response(bookmark_save.html, variables)

  接着编辑templates/bookmark_list.html,插入以下代码:

{% if bookmarks %}
  <ul class="bookmarks">
    {% for bookmark in bookmarks %}
      <li>
      <a href="{{ bookmark.link.url }}" class="title">
      {{ bookmark.title|escape }}a>
      {% if show_edit %}
        <a href="/save/?url={{ bookmark.link.url|urlencode }}"
        class="edit">[edit]a>
      {% endif %}
      <br />
      {% if show_tags %}
        Tags:
        {% if bookmark.tag_set.all %}
          <ul class="tags">
            {% for tag in bookmark.tag_set.all %}
              <li><a href="/tag/{{ tag.name|urlencode }}/">
                {{ tag.name|escape }}a>li>
            {% endfor %}
          ul>
      {% else %}
        None.
      {% endif %}
      <br />
[...]

  编辑bookmarks/views.py,在user_page视图中给模板传递show_edit变量:

def user_page(request, username):
  user = get_object_or_404(User, username=username)
  bookmarks = user.bookmark_set.order_by(-id)
  variables = RequestContext(request, {
    bookmarks: bookmarks,
    username: username,
    show_tags: True,
    show_edit: username == request.user.username,
  })
  return render_to_response(user_page.html, variables)

  当用户查看自身用户视图时,username==request.user.username才返回True。

  最后,将页面中edit链接的字体改小点,编辑static/style.css:

ul.bookmarks .edit {
    font-size: 70%;
}

  实现实时编辑功能

  显示实时编辑还需要做以下操作:

  • 拦截单击编辑链接引发的事件,使用ajax从服务器获取修改表单,然后将页面中的bookmark替换成编辑表单
  • 拦截用户提交表单的事件,然后使用ajax提交更新后的bookmark到服务器。

  首先创建templates/bookmark_save_form.html,将bookmark_save.html中的保存表单移动这个文件中。

<form id="save-form" method="post" action="/save/">
  {{ form.as_p }}
  {% csrf_token %}   
<input type="submit" value="save" /> form>

  注意这里我们给表单赋予了ID属性与action属性,这是为了让它能够在用户页面与bookmark提交页面都能正常工作。

  接下来在bookmark_save.html中包含上面的html。

{% extends "base.html" %}
{% block title %}Save Bookmark{% endblock %}
{% block head %}Save Bookmark{% endblock %}
{% block content %}
{% include ‘bookmark_save_form.html‘ %}
{% endblock %}

  打开bookmarks/views.py,进行编辑:

from django.views.decorators.csrf import csrf_exempt

@csrf_exempt
def bookmark_save_page(request):
  ajax = request.GET.has_key(ajax)
  if request.method == POST:
    form = BookmarkSaveForm(request.POST)
    if form.is_valid():
      bookmark = _bookmark_save(form)
      if ajax:
        variables = RequestContext(request, {
          bookmarks: [bookmark],
          show_edit: True,
          show_tags: True
        })
        return render_to_response(bookmark_list.html, variables)
      else: 
        return HttpResponseRedirect(
              /user/%s/ % request.user.username
            )
    else:
      if ajax:
        return HttpResponse(failure)
  elif request.GET.has_key(url):
    url = request.GET[url]
    title = ‘‘
    tags = ‘‘
    try:
      link = Link.objects.get(url=url)
      bookmark = Bookmark.objects.get(link=link, user=request.user)
      title = bookmark.title
      tags =  .join(tag.name for tag in bookmark.tag_set.all())
    except:
      pass
    form = BookmarkSaveForm({
      url: url,
      title: title,
      tags: tags
    })
  else:
    form = BookmarkSaveForm()
    variables = RequestContext(request, {
      form: form
    })
  if ajax: 
    return render_to_response(
          bookmark_save_form.html,
          variables
        )
  else: 
    return render_to_response(
          bookmark_save.html,
          variables
        )

  编辑templates/user_page.html:

{% extends "base.html" %}
{% block external %}
  <script type="text/Javascript" src="/static/bookmark_edit.js">
  script>
{% endblock %}
{% block title %}{{ username }}{% endblock %}
{% block head %}Bookmarks for {{ username }}{% endblock %}
{% block content %}
  {% include ‘bookmark_list.html‘ %}
{% endblock %}

  创建bookmark_edit.js:

function bookmark_edit() {
  var item = $(this).parent();
  var url = item.find(".title").attr("href");
  item.load("/save/?ajax&url=" + escape(url), null, function () {
    $("#save-form").submit(bookmark_save);
  });
  return false;
}
function bookmark_save() {
  var item = $(this).parent();
  var data = {
    url: item.find("#id_url").val(),
    title: item.find("#id_title").val(),
    tags: item.find("#id_tags").val()
  };
  $.ajax({
    url:"/save/?ajax",
    type:"POST",
    data:data,
    success:function (result) {
      if (result != "failure") {
        item.before($("li", result).get(0));
        item.remove();
        $("ul.bookmarks .edit").click(bookmark_edit);
      }
      else {
        alert("Failed to validate bookmark before saving.");
      }
    }
  })
  return false;
}
$(document).ready(function () {
  $("ul.bookmarks .edit").click(bookmark_edit);
});

  这样就实现了实时编辑的功能。

Django Web开发【6】使用Ajax增强用户体验


推荐阅读
  • 本文介绍了django中视图函数的使用方法,包括如何接收Web请求并返回Web响应,以及如何处理GET请求和POST请求。同时还介绍了urls.py和views.py文件的配置方式。 ... [详细]
  • 本文介绍了使用AJAX的POST请求实现数据修改功能的方法。通过ajax-post技术,可以实现在输入某个id后,通过ajax技术调用post.jsp修改具有该id记录的姓名的值。文章还提到了AJAX的概念和作用,以及使用async参数和open()方法的注意事项。同时强调了不推荐使用async=false的情况,并解释了JavaScript等待服务器响应的机制。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文介绍了通过ABAP开发往外网发邮件的需求,并提供了配置和代码整理的资料。其中包括了配置SAP邮件服务器的步骤和ABAP写发送邮件代码的过程。通过RZ10配置参数和icm/server_port_1的设定,可以实现向Sap User和外部邮件发送邮件的功能。希望对需要的开发人员有帮助。摘要长度:184字。 ... [详细]
  • WebSocket与Socket.io的理解
    WebSocketprotocol是HTML5一种新的协议。它的最大特点就是,服务器可以主动向客户端推送信息,客户端也可以主动向服务器发送信息,是真正的双向平等对话,属于服务器推送 ... [详细]
  • 延迟注入工具(python)的SQL脚本
    本文介绍了一个延迟注入工具(python)的SQL脚本,包括使用urllib2、time、socket、threading、requests等模块实现延迟注入的方法。该工具可以通过构造特定的URL来进行注入测试,并通过延迟时间来判断注入是否成功。 ... [详细]
  • JavaWeb中读取文件资源的路径问题及解决方法
    在JavaWeb开发中,读取文件资源的路径是一个常见的问题。本文介绍了使用绝对路径和相对路径两种方法来解决这个问题,并给出了相应的代码示例。同时,还讨论了使用绝对路径的优缺点,以及如何正确使用相对路径来读取文件。通过本文的学习,读者可以掌握在JavaWeb中正确找到和读取文件资源的方法。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • http:my.oschina.netleejun2005blog136820刚看到群里又有同学在说HTTP协议下的Get请求参数长度是有大小限制的,最大不能超过XX ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • 在重复造轮子的情况下用ProxyServlet反向代理来减少工作量
    像不少公司内部不同团队都会自己研发自己工具产品,当各个产品逐渐成熟,到达了一定的发展瓶颈,同时每个产品都有着自己的入口,用户 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • CentOS 7部署KVM虚拟化环境之一架构介绍
    本文介绍了CentOS 7部署KVM虚拟化环境的架构,详细解释了虚拟化技术的概念和原理,包括全虚拟化和半虚拟化。同时介绍了虚拟机的概念和虚拟化软件的作用。 ... [详细]
  • 在编写业务代码时,常常会遇到复杂的业务逻辑导致代码冗长混乱的情况。为了解决这个问题,可以利用中间件模式来简化代码逻辑。中间件模式可以帮助我们更好地设计架构和代码,提高代码质量。本文介绍了中间件模式的基本概念和用法。 ... [详细]
  • 网络请求模块选择——axios框架的基本使用和封装
    本文介绍了选择网络请求模块axios的原因,以及axios框架的基本使用和封装方法。包括发送并发请求的演示,全局配置的设置,创建axios实例的方法,拦截器的使用,以及如何封装和请求响应劫持等内容。 ... [详细]
author-avatar
小阳
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有