先说需求,有个项目,从excel导入数据库有个字段的格式为:建设银行+回车+(13002200),如果用Django后台的话,回车部分是录入不进去的,只能修改后台的规则,左下处理。
方法:
我的方法是:增加两个虚拟字段:“银行名”,“银行号”。输入完之后,再保存动作种,把这个两个字段的值付给数据库的字段——bank_name,并加上回车和括号。怎么做呢?用了我两天时间,有空就搜下,有空就搜下,终于... ...再查看了网友的贡献和官方文档之后,做出来了。也许你们早就知道了,给点面子,点个赞,哈哈。
首选就要增加两个虚拟的字段了,名字为name ,code。这里用到的是djanog.forms 的方法。在任意地方新建一个py文件,我直接在admin.py里写的,有洁癖的,不要这么干,很乱的。创建一个表单:
class BankcardsForm(forms.ModelForm):
name = forms.CharField(required=True,error_messages={'required':'银行名不能为空'},widget=forms.TextInput(attrs={'placeholder':u'输入银行名称'}))
code = forms.CharField(validators=[bankcode_validate,],widget=forms.TextInput(attrs={'placeholder':u'输入银行编号'}))
1:widget=forms.TextInput(attrs={'placeholder':u'输入银行编号'})这段代码是显示输入提示的。在输入框显示灰色的字。
2:required=True,error_messages={'required':'银行名不能为空'} 这个是输入验证,required=True表示非空,error_message,提示的信息。
3:validators=[bankcode_validate,] 自定义的判断方法。下面贴出完整校验方法。
def bankcode_validate(value):
try:
va = int(value)
except:
raise forms.ValidationError(u'请输入8位数字编码')
if len(value) != 8:
raise forms.ValidationError(u'输入长度位8')
其他的验证方式。
pwd = forms.CharField(required=True,min_length=6,max_length=10,
error_messages={'required':'密码不能为空',
'min_length':'最少输入6位',
'max_length':'最多输入10位'}
)
上面是其他的验证方式。
表单建立完成,验证也完成之后。
@admin.register(Bankcards)
class BankcardsAdmin(admin.ModelAdmin):
form = BankcardsForm #放在最上面,否则看不到自定义的表单
#改变表字段的排布--分层显示,单纯改变表的排布用fields = ()即可,注意这里不是list_display
fieldsets = (
('Bank_info',{
'fields':('name','code','bank_num')
}),
('Other',{
'fields':('bank_name','flag_long','flag_num','card_long','type','bank_lhh','shangf_code')
}),
)
readonly_fields = ('bank_num',)
def save_model(self, request, obj, form, change):
#print type(form.Meta.fields)
name = request.POST.get('name') #获取自定义表单提交的值
code = request.POST.get('code')#获取自定义表单提交的值
obj.bank_num = name+'\n'+'('+code+')'
obj.save()
#此处也可以用作其他模型对象保存,例如:
Bank.objects.create(name=name,code=code)#当然数据库要有这个模型对象啦
list_display = ('bank_num','bank_name','card_long','flag_long','flag_num','type','bank_lhh','shangf_code')
search_fields = ('bank_num','flag_num','flag_long')