python16_day18【Django_Form表单、分页】
一、表单
Django的Form主要具有一下几大功能:
- 生成HTML标签
- 验证用户数据(显示错误信息)
- HTML Form提交保留上次提交数据
- 初始化页面显示内容
1.Form类
创建Form类时,主要涉及到 【字段】 和 【插件】,字段用于对用户请求数据的验证,插件用于自动生成HTML;
所有字段类型:
1 Field 2 required=True, 是否允许为空 3 widget=None, HTML插件 4 label=None, 用于生成Label标签或显示内容 5 initial=None, 初始值 6 help_text='', 帮助信息(在标签旁边显示) 7 error_messages=None, 错误信息 {'required': '不能为空', 'invalid': '格式错误'} 8 show_hidden_initial=False, 是否在当前插件后面再加一个隐藏的且具有默认值的插件(可用于检验两次输入是否一直) 9 validators=[], 自定义验证规则 10 localize=False, 是否支持本地化 11 disabled=False, 是否可以编辑 12 label_suffix=None Label内容后缀 13 14 15 CharField(Field) 16 max_length=None, 最大长度 17 min_length=None, 最小长度 18 strip=True 是否移除用户输入空白 19 20 IntegerField(Field) 21 max_value=None, 最大值 22 min_value=None, 最小值 23 24 FloatField(IntegerField) 25 ... 26 27 DecimalField(IntegerField) 28 max_value=None, 最大值 29 min_value=None, 最小值 30 max_digits=None, 总长度 31 decimal_places=None, 小数位长度 32 33 BaseTemporalField(Field) 34 input_formats=None 时间格式化 35 36 DateField(BaseTemporalField) 格式:2015-09-01 37 TimeField(BaseTemporalField) 格式:11:12 38 DateTimeField(BaseTemporalField)格式:2015-09-01 11:12 39 40 DurationField(Field) 时间间隔:%d %H:%M:%S.%f 41 ... 42 43 RegexField(CharField) 44 regex, 自定制正则表达式 45 max_length=None, 最大长度 46 min_length=None, 最小长度 47 error_message=None, 忽略,错误信息使用 error_messages={'invalid': '...'} 48 49 EmailField(CharField) 50 ... 51 52 FileField(Field) 53 allow_empty_file=False 是否允许空文件 54 55 ImageField(FileField) 56 ... 57 注:需要PIL模块,pip3 install Pillow 58 以上两个字典使用时,需要注意两点: 59 - form表单中 enctype="multipart/form-data" 60 - view函数中 obj = MyForm(request.POST, request.FILES) 61 62 URLField(Field) 63 ... 64 65 66 BooleanField(Field) 67 ... 68 69 NullBooleanField(BooleanField) 70 ... 71 72 ChoiceField(Field) 73 ... 74 choices=(), 选项,如:choices = ((0,'上海'),(1,'北京'),) 75 required=True, 是否必填 76 widget=None, 插件,默认select插件 77 label=None, Label内容 78 initial=None, 初始值 79 help_text='', 帮助提示 80 81 82 ModelChoiceField(ChoiceField) 83 ... django.forms.models.ModelChoiceField 84 queryset, # 查询数据库中的数据 85 empty_label="---------", # 默认空显示内容 86 to_field_name=None, # HTML中value的值对应的字段 87 limit_choices_to=None # ModelForm中对queryset二次筛选 88 89 ModelMultipleChoiceField(ModelChoiceField) 90 ... django.forms.models.ModelMultipleChoiceField 91 92 93 94 TypedChoiceField(ChoiceField) 95 coerce = lambda val: val 对选中的值进行一次转换 96 empty_value= '' 空值的默认值 97 98 MultipleChoiceField(ChoiceField) 99 ... 100 101 TypedMultipleChoiceField(MultipleChoiceField) 102 coerce = lambda val: val 对选中的每一个值进行一次转换 103 empty_value= '' 空值的默认值 104 105 ComboField(Field) 106 fields=() 使用多个验证,如下:即验证最大长度20,又验证邮箱格式 107 fields.ComboField(fields=[fields.CharField(max_length=20), fields.EmailField(),]) 108 109 MultiValueField(Field) 110 PS: 抽象类,子类中可以实现聚合多个字典去匹配一个值,要配合MultiWidget使用 111 112 SplitDateTimeField(MultiValueField) 113 input_date_formats=None, 格式列表:['%Y--%m--%d', '%m%d/%Y', '%m/%d/%y'] 114 input_time_formats=None 格式列表:['%H:%M:%S', '%H:%M:%S.%f', '%H:%M'] 115 116 FilePathField(ChoiceField) 文件选项,目录下文件显示在页面中 117 path, 文件夹路径 118 match=None, 正则匹配 119 recursive=False, 递归下面的文件夹 120 allow_files=True, 允许文件 121 allow_folders=False, 允许文件夹 122 required=True, 123 widget=None, 124 label=None, 125 initial=None, 126 help_text='' 127 128 GenericIPAddressField 129 protocol='both', both,ipv4,ipv6支持的IP格式 130 unpack_ipv4=False 解析ipv4地址,如果是::ffff:192.0.2.1时候,可解析为192.0.2.1, PS:protocol必须为both才能启用 131 132 SlugField(CharField) 数字,字母,下划线,减号(连字符) 133 ... 134 135 UUIDField(CharField) uuid类型 136 ...
常用字段:
CharField 字符串 EmailField 字符串(邮箱格式) IntegerField 字符串(数字格式) GenericIPAddressField 字符串(IP格式) FileField 文件对象 RegexField 字符串(自定义) ChoiceField 多选
所有插件:
TextInput(Input) NumberInput(TextInput) EmailInput(TextInput) URLInput(TextInput) PasswordInput(TextInput) HiddenInput(TextInput) Textarea(Widget) DateInput(DateTimeBaseInput) DateTimeInput(DateTimeBaseInput) TimeInput(DateTimeBaseInput) CheckboxInput Select NullBooleanSelect SelectMultiple RadioSelect CheckboxSelectMultiple FileInput ClearableFileInput MultipleHiddenInput SplitDateTimeWidget SplitHiddenDateTimeWidget SelectDateWidget
常用插件:
# 单radio,值为字符串 # user = fields.CharField( # initial=2, # widget=widgets.RadioSelect(choices=((1,'上海'),(2,'北京'),)) # ) # 单radio,值为字符串 # user = fields.ChoiceField( # choices=((1, '上海'), (2, '北京'),), # initial=2, # widget=widgets.RadioSelect # ) # 单select,值为字符串 # user = fields.CharField( # initial=2, # widget=widgets.Select(choices=((1,'上海'),(2,'北京'),)) # ) # 单select,值为字符串 # user = fields.ChoiceField( # choices=((1, '上海'), (2, '北京'),), # initial=2, # widget=widgets.Select # ) # 多选select,值为列表 # user = fields.MultipleChoiceField( # choices=((1,'上海'),(2,'北京'),), # initial=[1,], # widget=widgets.SelectMultiple # ) # 单checkbox # user = fields.CharField( # widget=widgets.CheckboxInput() # ) # 多选checkbox,值为列表 # user = fields.MultipleChoiceField( # initial=[2, ], # choices=((1, '上海'), (2, '北京'),), # widget=widgets.CheckboxSelectMultiple # )
二、分页
1 使用方法 2 3 all_count=Hosts.objects.all().count() 4 page_info = PageInfo(request.GET.get('p'), 20, all_count, request.path_info, page_range=15) 5 user_list = Hosts.objects.all().order_by('-id')[page_info.start():page_info.end()] 6 return render(request, 'index.html', locals()) 7 8 类方法 9 class PageInfo(object): 10 def __init__(self, current_page, per_age_num, all_count, base_url, page_range=11): 11 """ 12 :param current_page: 当前页 13 :param per_age_num: 每页显示数据条数 14 :param all_count: 数据库总个数 15 :param base_url: 页码标签的前缀 16 :param page_range: 页码个数 17 :return: 列表-->str 18 """ 19 try: 20 current_page = int(current_page) 21 except Exception as e: 22 current_page = 1 23 self.current_page = current_page 24 self.per_page_num = per_age_num 25 self.all_count = all_count 26 a, b = divmod(all_count, per_age_num) 27 if b != 0: 28 self.all_page = a + 1 29 else: 30 self.all_page = a 31 32 self.base_url = base_url 33 self.page_range = page_range 34 35 def start(self): 36 return (self.current_page - 1) * self.per_page_num 37 38 def end(self): 39 return self.current_page * self.per_page_num 40 41 def page_str(self): 42 """ 43 :return: list --> str 44 """ 45 page_list = [] 46 47 first_page = "<li><a href='%s?p=%s'>首页</a></li>" % (self.base_url, 1) 48 page_list.append(first_page) 49 50 if self.current_page <= 1: 51 prev = "<li><a href='#'>上一页</a></li>" 52 else: 53 prev = "<li><a href='%s?p=%s'>上一页</a></li>" % (self.base_url, self.current_page - 1) 54 page_list.append(prev) 55 56 # 只有 8页 57 if self.all_page <= self.page_range: 58 start = 1 59 end = self.all_page + 1 60 else: 61 # 页数 18 62 if self.current_page > int(self.page_range / 2): 63 # 当前页: 100,101,102 64 if (self.current_page + int(self.page_range / 2)) > self.all_page: 65 start = self.all_page - self.page_range + 1 66 end = self.all_page + 1 67 # 当前页: 6,7,8,9,10 68 else: 69 start = self.current_page - int(self.page_range / 2) 70 end = self.current_page + int(self.page_range / 2) + 1 71 else: 72 # 当前页: 1,2,3,4,5, 73 start = 1 74 end = self.page_range + 1 75 76 for i in range(start, end): 77 if self.current_page == i: 78 temp = '<li class="active"><a href="%s?p=%s">%s</a></li>' % ( 79 self.base_url, i, i,) 80 else: 81 temp = '<li><a href="%s?p=%s">%s</a></li>' % ( 82 self.base_url, i, i,) 83 page_list.append(temp) 84 85 if self.current_page >= self.all_page: 86 nex = "<li><a href='#'>下一页</a></li>" 87 else: 88 nex = "<li><a href='%s?p=%s'>下一页</a></li>" % (self.base_url, self.current_page + 1) 89 page_list.append(nex) 90 91 last_page = "<li><a href='%s?p=%s'>尾页</a></li>" % (self.base_url, self.all_page) 92 page_list.append(last_page) 93 return "".join(page_list)