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。

  1. <head>
  2.   <title>Django Bookmarks |
  3.   {% block title %}{% endblock %}</title>
  4.   <link rel="stylesheet" href="/site_media/style.css" type="text/css" />
  5.   <script type="text/javascript" src="/static/jquery.js"></script>
  6. </head>

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

  1. <head>
  2.   <title>Django Bookmarks | {% block title %}{% endblock %}</title>
  3.   <link rel="stylesheet" href="/site_media/style.css" type="text/css"/>
  4.   <script type="text/javascript" src="/site_media/jquery.js">
  5.   </script>
  6.   {% block external %}{% endblock %}
  7. </head>

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

  实现Bookmarks的实时搜索

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

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

  实现搜索

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

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

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

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

  1. def search_page(request):
  2.   form = SearchForm()
  3.   bookmarks = []
  4.   show_results = False
  5.   if request.GET.has_key('query'):
  6.     show_results = True
  7.     query = request.GET['query'].strip()
  8.     if query:
  9.       form = SearchForm({'query' : query})
  10.       bookmarks = \
  11.         Bookmark.objects.filter (title__icontains=query)[:10]
  12.   variables = RequestContext(request, { 'form': form,
  13.     'bookmarks': bookmarks,
  14.     'show_results': show_results,
  15.     'show_tags': True,
  16.     'show_user': True
  17.   })
  18.   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:

  1. {% extends "base.html" %}
  2. {% block title %}Search Bookmarks{% endblock %}
  3. {% block head %}Search Bookmarks{% endblock %}
  4. {% block content %}
  5.   <form id="search-form" method="get" action=".">
  6.     {{ form.as_p }}
  7.     <input type="submit" value="search" />
  8.   </form>
  9.   <div id="search-results">
  10.     {% if show_results %}
  11.       {% include 'bookmark_list.html' %}
  12.     {% endif %}
  13.   </div>
  14. {% endblock %}

  在urls.py中添加url:

  1. urlpatterns = patterns('',
  2.   # Browsing
  3.   (r'^$', main_page),
  4.   (r'^user/(\w+)/$', user_page),
  5.   (r'^tag/([^\s]+)/$', tag_page),
  6.   (r'^tag/$', tag_cloud_page),
  7.   (r'^search/$', search_page),
  8. )

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

  1. <div id="nav">
  2.   <a href="/">home</a> |
  3.   {% if user.is_authenticated %}
  4.     <a href="/save/">submit</a> |
  5.     <a href="/search/">search</a> |
  6.     <a href="/user/{{ user.username }}/">
  7.       {{ user.username }}</a> |
  8.     <a href="/logout/">logout</a>
  9.   {% else %}
  10.     <a href="/login/">login</a> |
  11.     <a href="/register/">register</a>
  12.   {% endif %}
  13. </div>

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

  实现实时搜索

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

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

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

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

  1. def search_page(request):
  2.   [...]
  3.   variables = RequestContext(request, {
  4.     'form': form,
  5.     'bookmarks': bookmarks,
  6.     'show_results': show_results,
  7.     'show_tags': True,
  8.     'show_user': True
  9.   })
  10.   if request.GET.has_key('ajax'):
  11.     return render_to_response('bookmark_list.html', variables)
  12.   else:
  13.     return render_to_response('search.html', variables)

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

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

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

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

  1. {% extends "base.html" %}
  2. {% block external %}
  3.   <script type="text/javascript" src="/site_media/search.js">
  4.   </script>
  5. {% endblock %}
  6. {% block title %}Search Bookmarks{% endblock %}
  7. {% block head %}Search Bookmarks{% endblock %}
  8. [...]

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

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

  这样就给表单添加了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,添加如下代码:

  1. def _bookmark_save(request, form):
  2.   # Create or get link.
  3.   link, dummy = \
  4.     Link.objects.get_or_create(url=form.clean_data['url'])
  5.   # Create or get bookmark.
  6.   bookmark, created = Bookmark.objects.get_or_create(
  7.     user=request.user,
  8.     link=link
  9.   )
  10.   # Update bookmark title.
  11.   bookmark.title = form.cleaned_data['title']
  12.   # If the bookmark is being updated, clear old tag list.
  13.   if not created:
  14.     bookmark.tag_set.clear()
  15.   # Create new tag list.
  16.     tag_names = form.cleaned_data['tags'].split()
  17.   for tag_name in tag_names:
  18.     tag, dummy = Tag.objects.get_or_create(name=tag_name)
  19.     bookmark.tag_set.add(tag)
  20.   # Save bookmark to database and return it.
  21.   bookmark.save()
  22.   return bookmark

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

  1. @login_required
  2. def bookmark_save_page(request):
  3.   if request.method == 'POST':
  4.     form = BookmarkSaveForm(request.POST)
  5.     if form.is_valid():
  6.       bookmark = _bookmark_save(request, form)
  7.       return HttpResponseRedirect(
  8.         '/user/%s/' % request.user.username
  9.       )
  10.   else:
  11.     form = BookmarkSaveForm()
  12.   variables = RequestContext(request, {
  13.     'form': form
  14.   })
  15.   return render_to_response('bookmark_save.html', variables)

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

  1. if there is POST data:
  2.   Validate and save bookmark.
  3.   Redirect to user page.
  4. else:
  5.   Create an empty form.
  6. Render page.

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

  1. if there is POST data:
  2.   Validate and save bookmark.
  3.   Redirect to user page.
  4. else if there is a URL in GET data:
  5.   Create a form an populate it with the URL's bookmark.
  6. else:
  7.   Create an empty form.
  8. Render page.

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

  1. from django.core.exceptions import ObjectDoesNotExist
  2. @login_required
  3. def bookmark_save_page(request):
  4.   if request.method == 'POST':
  5.     form = BookmarkSaveForm(request.POST)
  6.     if form.is_valid():
  7.       bookmark = _bookmark_save(request, form)
  8.       return HttpResponseRedirect(
  9.         '/user/%s/' % request.user.username
  10.       )
  11.   elif request.GET.has_key('url'):
  12.     url = request.GET['url']
  13.     title = ''
  14.     tags = ''
  15.     try:
  16.       link = Link.objects.get(url=url)
  17.       bookmark = Bookmark.objects.get(
  18.         link=link,
  19.         user=request.user
  20.       )
  21.       title = bookmark.title
  22.       tags = ' '.join(
  23.         tag.name for tag in bookmark.tag_set.all()
  24.       )
  25.     except ObjectDoesNotExist:
  26.       pass
  27.     form = BookmarkSaveForm({
  28.       'url': url,
  29.       'title': title,
  30.       'tags': tags
  31.     })
  32.   else:
  33.     form = BookmarkSaveForm()
  34.       variables = RequestContext(request, {
  35.       'form': form
  36.     })
  37.   return render_to_response('bookmark_save.html', variables)

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

  1. {% if bookmarks %}
  2.   <ul class="bookmarks">
  3.     {% for bookmark in bookmarks %}
  4.       <li>
  5.       <a href="{{ bookmark.link.url }}" class="title">
  6.       {{ bookmark.title|escape }}</a>
  7.       {% if show_edit %}
  8.         <a href="/save/?url={{ bookmark.link.url|urlencode }}"
  9.         class="edit">[edit]</a>
  10.       {% endif %}
  11.       <br />
  12.       {% if show_tags %}
  13.         Tags:
  14.         {% if bookmark.tag_set.all %}
  15.           <ul class="tags">
  16.             {% for tag in bookmark.tag_set.all %}
  17.               <li><a href="/tag/{{ tag.name|urlencode }}/">
  18.                 {{ tag.name|escape }}</a></li>
  19.             {% endfor %}
  20.           </ul>
  21.       {% else %}
  22.         None.
  23.       {% endif %}
  24.       <br />
  25. [...]

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

  1. def user_page(request, username):
  2.   user = get_object_or_404(User, username=username)
  3.   bookmarks = user.bookmark_set.order_by('-id')
  4.   variables = RequestContext(request, {
  5.     'bookmarks': bookmarks,
  6.     'username': username,
  7.     'show_tags': True,
  8.     'show_edit': username == request.user.username,
  9.   })
  10.   return render_to_response('user_page.html', variables)

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

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

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

  实现实时编辑功能

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

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

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

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

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

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

  1. {% extends "base.html" %}
  2. {% block title %}Save Bookmark{% endblock %}
  3. {% block head %}Save Bookmark{% endblock %}
  4. {% block content %}
  5. {% include 'bookmark_save_form.html' %}
  6. {% endblock %}

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

  1. from django.views.decorators.csrf import csrf_exempt
  2.  
  3. @csrf_exempt
  4. def bookmark_save_page(request):
  5.   ajax = request.GET.has_key('ajax')
  6.   if request.method == 'POST':
  7.     form = BookmarkSaveForm(request.POST)
  8.     if form.is_valid():
  9.       bookmark = _bookmark_save(form)
  10.       if ajax:
  11.         variables = RequestContext(request, {
  12.           'bookmarks': [bookmark],
  13.           'show_edit': True,
  14.           'show_tags': True
  15.         })
  16.         return render_to_response('bookmark_list.html', variables)
  17.       else:
  18.         return HttpResponseRedirect(
  19.               '/user/%s/' % request.user.username
  20.             )
  21.     else:
  22.       if ajax:
  23.         return HttpResponse('failure')
  24.   elif request.GET.has_key('url'):
  25.     url = request.GET['url']
  26.     title = ''
  27.     tags = ''
  28.     try:
  29.       link = Link.objects.get(url=url)
  30.       bookmark = Bookmark.objects.get(link=link, user=request.user)
  31.       title = bookmark.title
  32.       tags = ' '.join(tag.name for tag in bookmark.tag_set.all())
  33.     except:
  34.       pass
  35.     form = BookmarkSaveForm({
  36.       'url': url,
  37.       'title': title,
  38.       'tags': tags
  39.     })
  40.   else:
  41.     form = BookmarkSaveForm()
  42.     variables = RequestContext(request, {
  43.       'form': form
  44.     })
  45.   if ajax:
  46.     return render_to_response(
  47.           'bookmark_save_form.html',
  48.           variables
  49.         )
  50.   else:
  51.     return render_to_response(
  52.           'bookmark_save.html',
  53.           variables
  54.         )

  编辑templates/user_page.html:

  1. {% extends "base.html" %}
  2. {% block external %}
  3.   <script type="text/javascript" src="/static/bookmark_edit.js">
  4.   </script>
  5. {% endblock %}
  6. {% block title %}{{ username }}{% endblock %}
  7. {% block head %}Bookmarks for {{ username }}{% endblock %}
  8. {% block content %}
  9.   {% include 'bookmark_list.html' %}
  10. {% endblock %}

  创建bookmark_edit.js:

  1. function bookmark_edit() {
  2.   var item = $(this).parent();
  3.   var url = item.find(".title").attr("href");
  4.   item.load("/save/?ajax&url=" + escape(url), null, function () {
  5.     $("#save-form").submit(bookmark_save);
  6.   });
  7.   return false;
  8. }
  9. function bookmark_save() {
  10.   var item = $(this).parent();
  11.   var data = {
  12.     url: item.find("#id_url").val(),
  13.     title: item.find("#id_title").val(),
  14.     tags: item.find("#id_tags").val()
  15.   };
  16.   $.ajax({
  17.     url:"/save/?ajax",
  18.     type:"POST",
  19.     data:data,
  20.     success:function (result) {
  21.       if (result != "failure") {
  22.         item.before($("li", result).get(0));
  23.         item.remove();
  24.         $("ul.bookmarks .edit").click(bookmark_edit);
  25.       }
  26.       else {
  27.         alert("Failed to validate bookmark before saving.");
  28.       }
  29.     }
  30.   })
  31.   return false;
  32. }
  33. $(document).ready(function () {
  34.   $("ul.bookmarks .edit").click(bookmark_edit);
  35. });

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

  

Django Web开发【6】使用Ajax增强用户体验的更多相关文章

  1. Web开发中设置快捷键来增强用户体验

    从事对日外包一年多以来,发现日本的无论是WinForm项目还是Web项目都注重快捷键的使用,日本人操作的时候都喜欢用键盘而不是用鼠标去点,用他们的话来说"键盘永远比鼠标来的快",所 ...

  2. Django web 开发指南 no such table:

    在学习django web开发指南时,发布新博客点击save后会有error提示:no such table balabalabala... 百度了一下说重新运行manage.py syncdb 就可 ...

  3. Django Web开发学习笔记(1)

    一.Python的标准类型 (1)bool型 >>> bool("") False >>> bool(None) False >>& ...

  4. Django Web开发指南笔记

    Django Web开发指南笔记 语句VS表达式 python代码由表达式和语句组成,由解释器负责执行. 主要区别:表达式是一个值,它的结果一定是一个python对象:如:12,1+2,int('12 ...

  5. 基于gin的golang web开发:永远不要相信用户的输入

    作为后端开发者我们要记住一句话:"永远不要相信用户的输入",这里所说的用户可能是人,也可能是另一个应用程序."永远不要相信用户的输入"是安全编码的准则,也就是说 ...

  6. Django web开发【5】 实现标签功能

    标签tag在很多web2.0应用中都很常见,标签其实就是关联某些信息的一个关键字.打标签实际上就是给内容分配标签的过程,它通常由作者或者用户实现.标签之所有这么流行是因为它允许用户对自己创建的博客.图 ...

  7. Django Web开发【4】 用户注册与管理

    几乎所有的网站都提供了用户注册与管理功能,这一节,我们将讲解如何利用Django自身提供的用户认证系统实现用户注册与管理功能. 会话认证 在上一节中,我们学习了User数据模型,并用它来保存用户信息, ...

  8. [python] python django web 开发 —— 15分钟送到会用(只能送你到这了)

    1.安装python环境 1.1 安装python包管理器: wget https://bootstrap.pypa.io/get-pip.py sudo python get-pip.py   1. ...

  9. Android开发学习之路-提升用户体验小技巧

    记得之前看谷歌的一个视频提到这个用户体验的问题,今天想起来了就写了个Demo来记录下. 当一个事件发生之后,用户需要一段时间才能知道结果,那么这段时间究竟应该让用户干什么?这个问题很常见,比如我们的软 ...

随机推荐

  1. protobuf使用错误总结

    1>HelloWorldScene.obj : error LNK2019: 无法解析的外部符号 "public: virtual __thiscall LoginReqMessage ...

  2. ActionBar隐藏修改图标和标题

    有时候在一些子页面或者内容页面,不需要显示ActionBar的标题栏图标.可用如下方式进行设置. 首先获取到ActionBar对象 ActionBar actionBar=getActionBar() ...

  3. web基础-web工作原理,http协议,浏览器缓存

    1,web工作原理 2,http协议 3,浏览器缓存 4,cookie和session -------------------------------------------------------- ...

  4. i = i++ 在java字节码层面的分析

    有这么一段代码: package zl.test; public class PcodeTest { /** * @param args */ public static void main(Stri ...

  5. Spring-----Spring Jar包

    转载自:http://blog.csdn.net/hekewangzi/article/details/51713110

  6. 加密传输SSL协议5_Hash Function

    怎么对一个大的文件进行签名,因为文件比较大,非对称签名很慢.那么想,我能把这个大的文件通过一种函数变换,变成一个和源文件唯一对应的的小的文件吗?答案是可以的. Hash Function 这里任何的文 ...

  7. Android RelativeLayout常用属性介绍

    下面介绍一下RelativeLayout用到的一些重要的属性: 第一类:属性值为true或false android:layout_centerHrizontal 水平居中 android:layou ...

  8. Android Studio常用插件续

    这个月因为各种事情在忙,包括赶项目,回老家,还有准备旅游的事,所以应该写不了四篇博客了.今天介绍一下关于Android Studio 的几个好用的插件,都是我在用的,它们或能帮你节省时间,或者让你心情 ...

  9. 方形图片转动并转换成圆形CSS特效

    <style> img { transition:all 0.8s ease 0s;} img:hover { border-radius:50%; transform:rotate(72 ...

  10. codeforces 659C . Tanya and Toys 二分

    题目链接 将给出的已经有了的排序, 在前面加上0, 后面加上1e9+1. 然后对相邻的两项判断. 如果相邻两项之间的数的和小于m, 那么全都选上, m减去相应的值. 如果大于m, 那么二分判断最多能选 ...