form+ajax实现验证

form+ajax实现验证

#obj.errors返回的是ErrorDict,不是字典(虽然继承字典)
#obj.errors.as_json() 返回的字符串(前端要连续反解两次)
#obj.errors.as_data()  返回原生的字典 但是返回value 是ValidationError,不能直接序列化


1)通过errors.as_data()实现数据绑定和返回,errors.as_data()  返回原生的字典 但是返回value 是ValidationError,不能直接序列化,  通过json属性cls转换是ValidationError为字典返回。优点:前端jquery不需要JSON.parse转换两次。

2)通过errors.as_json()实现数据绑定和返回。as_json返回的是字符串。
    前端返回:需要两次JSON.parse转换(第一次:HttpResponse返回的是字符串;第二次:  返回的error字典中的value是字符串,需要在转成字典,才可以访问)

示例1:注册信息
#两种方式1:as_data(注意view中绑定方式和模板中jqeury调用方式)
forms.py
 1 class RegisterForm(forms.Form):
 2     username = fields.CharField(
 3             error_messages={"required": "用户名不能为空"},
 4             widget=widgets.Input(attrs={"class": "form-control"})
 5     )
 6     email = fields.EmailField(
 7            error_messages={'required':'邮箱不能为空','invalid':'邮箱格式错误'},
 8             widget=widgets.Input(attrs={"class": "form-control"})
 9     )
10     password = fields.CharField(
11             error_messages={"required": "密码不能为空"},
12             widget=widgets.Input(attrs={"class": "form-control"})
13     )
14     password2 = fields.CharField(error_messages={"required": "密码不能为空"},
15                                  widget=widgets.Input(attrs={"class": "form-control"}))
16     check_code = fields.CharField(error_messages={"required": "验证码不能为空"},
17                                   widget=widgets.Input(attrs={"class": "form-control"}))
18 
19     def clean_username(self):
20         """
21         验证用户存不存在
22         :return:
23         """
24         obj = models.UserInfo.objects.filter(username=self.cleaned_data['username'])
25         # 用户存在返回原来的值
26         if not obj:
27             return self.cleaned_data['username']
28         else:
29             raise ValidationError(message="用户已存在,请更换其他用户名", code="xxxx")
forms.py
 
#views.py

 1 from django.core.exceptions import ValidationError
 2 class JsonCunstomEncode(json.JSONEncoder):
 3     def default(self, field):
 4         if isinstance(field,ValidationError):
 5             return {"code":field.code,"message":field.message}
 6         else:
 7             return json.JSONEncoder.default(self,field) 
 8 
 9 def register(request):
10     """
11     注册
12     :param request:
13     :return:
14     """
15     ret={"status":False,"error":None,"data":None}
16     if request.method=="GET":
17         register_obj=forms.RegisterForm()
18         return render(request, 'register.html',{"register_obj":register_obj})
19     elif request.method=="POST":
20         register_obj=forms.RegisterForm(request.POST)
21         if register_obj.is_valid():
22             ret["status"]=True
23             ret["data"]=register_obj.cleaned_data
24         else:
25             ret["error"]=register_obj.errors.as_data()
26         result=json.dumps(ret,cls=JsonCunstomEncode)
27         #不能使用render,使用render返回数据,前端var data1=JSON.parse(arg)转换报错。可以使用HttpResponse直接返回数据
28         #return render(request, 'register.html',{"result":result})
29         return HttpResponse(result)
Views.py

模块实现:
 1 <div class="register">
 2     <div style="font-size: 25px; font-weight: bold;text-align: center;">
 3         用户注册
 4     </div>
 5     <form role="form" id="register_form">
 6         {% csrf_token %}
 7         <div class="form-group">
 8             <label for="username">用户名</label>
 9             <input type="text" class="form-control" id="username" placeholder="请输入用户名" name="username">
10             <p id="error_username"></p>
11         </div>
12         <div class="form-group">
13             <label for="email">邮箱</label>
14             <input type="email" class="form-control" id="email" placeholder="请输入邮箱" name="email">
15             <p id="error_email"></p>
16         </div>
17         <div class="form-group">
18             <label for="password">密码</label>
19             <input type="password" class="form-control" id="password" placeholder="请输入密码" name="password">
20             <p id="error_password"></p>
21         </div>
22         <div class="form-group">
23             <label for="confirm_password">确认密码</label>
24             <input type="password" class="form-control" id="password" placeholder="请输入密码" name="password2">
25             <p id="error_password2"></p>
26         </div>
27 
28         <div class="form-group">
29             <label for="password">验证码</label>
30 
31             <div class="row">
32                 <div class="col-xs-7">
33                     <input type="password" class="form-control" id="password" placeholder="请输入验证码" name="check_code">
34                     <p id="error_check_code"></p>
35                 </div>
36                 <div class="col-xs-5">
37                     <img src="/check_code.html">
38                 </div>
39             </div>
40             <p id="show_error"></p>
41         </div>
42         <a id="register_a" type="submit" class="btn btn-default" >下一步</a>
43     </form>
44     <script src="/static/js/jquery-1.12.4.js"></script>
45     <script src="/static/js/jquery.cookie.js"></script>
46     <script>
47 $("#register_a").click(
48         function(){
49             $.ajax({
50                 url:'{% url "register" %}',
51                 type:"POST",
52                 //headers: {"X-CSRFtoken":$.cookie("csrftoken")},如果是用serialize,他就csrftoken一起发送过去,所以不需要加headers
53                 data:$("#register_form").serialize(),
54                 success:function(arg){
55                     var data1=JSON.parse(arg);
56                     console.log(data1);
57                     //#如果返回的状态为false,输出出错的内容
58                     if (!data1.status){
59                         //清空所有error p标签的内容
60                         $("#register_form").find('p[id*=error_]').each(
61                                 function(){
62                                     $(this).text("")
63                                 }
64                         )
65                         //根据错误信息动态绑定到p标签
66                         for (var item in data1.error){
67                             var tmp='#error_'+item+''
68                             $(tmp).text(data1.error[item][0].message);
69                         }
70                         /*
71                         手动绑定标签,但是会存在error中没有的信息报错
72                         $("#error_username").text(data1.error["username"][0].message);
73                         $("#error_email").text(data1.error.email[0].message);
74                         $("#error_password").text(data1.error.password[0].message);
75                         $("#error_password2").text(data1.error.password2[0].message);
76                         $("#error_check_code").text(data1.error.check_code[0].message);
77                         */
78                     }
79                     else{
80                         //跳转到后台
81                         location.href='{% url "base_info" %}'
82                     }
83                 }
84 
85             })
86         }
87         )
88     </script>
前端页面实现
 
方式2:as_json

views.py

 1 def register(request):
 2     """
 3     注册
 4     :param request:
 5     :return:
 6     """
 7     ret={"status":False,"error":None,"data":None}
 8     if request.method=="GET":
 9         register_obj=forms.RegisterForm()
10         return render(request, 'register.html',{"register_obj":register_obj})
11     elif request.method=="POST":
12         register_obj=forms.RegisterForm(request.POST)
13         if register_obj.is_valid():
14             ret["status"]=True
15             ret["data"]=register_obj.cleaned_data
16         else:
17             ret["error"]=register_obj.errors.as_json()
18         result=json.dumps(ret)
19         #不能使用render,使用render返回数据,前端var data1=JSON.parse(arg)转换报错。可以使用HttpResponse直接返回数据
20         #return render(request, 'register.html',{"result":result})
21         return HttpResponse(result)
views.py

模板:
 1 <div class="register">
 2 <div style="font-size: 25px; font-weight: bold;text-align: center;">
 3     用户注册
 4 </div>
 5 <form role="form" id="register_form">
 6     {% csrf_token %}
 7     <div class="form-group">
 8         <label for="username">用户名</label>
 9         <input type="text" class="form-control" id="username" placeholder="请输入用户名" name="username">
10         <p id="error_username"></p>
11     </div>
12     <div class="form-group">
13         <label for="email">邮箱</label>
14         <input type="email" class="form-control" id="email" placeholder="请输入邮箱" name="email">
15         <p id="error_email"></p>
16     </div>
17     <div class="form-group">
18         <label for="password">密码</label>
19         <input type="password" class="form-control" id="password" placeholder="请输入密码" name="password">
20         <p id="error_password"></p>
21     </div>
22     <div class="form-group">
23         <label for="confirm_password">确认密码</label>
24         <input type="password" class="form-control" id="password" placeholder="请输入密码" name="password2">
25         <p id="error_password2"></p>
26     </div>
27 
28     <div class="form-group">
29         <label for="password">验证码</label>
30 
31         <div class="row">
32             <div class="col-xs-7">
33                 <input type="password" class="form-control" id="password" placeholder="请输入验证码" name="check_code">
34                 <p id="error_check_code"></p>
35             </div>
36             <div class="col-xs-5">
37                 <img src="/check_code.html">
38             </div>
39         </div>
40         <p id="show_error"></p>
41     </div>
42     <a id="register_a" type="submit" class="btn btn-default" >下一步</a>
43 </form>
44 <script src="/static/js/jquery-1.12.4.js"></script>
45 <script src="/static/js/jquery.cookie.js"></script>
46 <script>
47     $("#register_a").click(
48             function(){
49                 $.ajax({
50                     url:'{% url "register" %}',
51                     type:"POST",
52                     //headers: {"X-CSRFtoken":$.cookie("csrftoken")},如果是用serialize,他就csrftoken一起发送过去,所以不需要加headers
53                     data:$("#register_form").serialize(),
54                     success:function(arg){
55                         //第一次转换
56                         var data1=JSON.parse(arg);
57                         console.log(data1);
58                         //#如果返回的状态为false,输出出错的内容
59                         if (!data1.status){
60                             //清空所有error p标签的内容
61                             $("#register_form").find('p[id*=error_]').each(
62                                     function(){
63                                         $(this).text("")
64                                     }
65                             )
66                             //第二次转换var error_list=JSON.parse(data1.error)
67                             //根据错误信息动态绑定到p标签for (var item in error_list){
68                                 var tmp='#error_'+item+''
69                                 $(tmp).text(error_list[item][0].message);
70                             }
71                             /*
72                             手动绑定标签,但是会存在error中没有的信息报错
73                             $("#error_username").text(data1.error["username"][0].message);
74                             $("#error_email").text(data1.error.email[0].message);
75                             $("#error_password").text(data1.error.password[0].message);
76                             $("#error_password2").text(data1.error.password2[0].message);
77                             $("#error_check_code").text(data1.error.check_code[0].message);
78                             */
79                         }
80                         else{
81                             //跳转到后台
82                             location.href='{% url "base_info" %}'
83                         }
84                     }
85 
86 
87 
88 
89                 })
90             }
91     )
92 </script>
前端页面实现


 
原文地址:https://www.cnblogs.com/hongdoudou/p/12630727.html