Django表單外鍵選項初始化的問題及解決方法
問題描述
先說明一下問題的由來:
Django的模型中經(jīng)常會用ForeignKey來關(guān)聯(lián)其他表格數(shù)據(jù)
class MeasureTask(models.Model): taskname = models.CharField(max_length=LEN_FULLNAME, verbose_name="任務(wù)名稱") road = models.ForeignKey(Road, on_delete=models.CASCADE, verbose_name="設(shè)計路段") # 路面層,附加一個參數(shù) ,指定這個層的厚度,相對于底層的厚度 # road_level = models.ForeignKey(RoadLevel, on_delete=models.CASCADE, verbose_name="路面層") level_thick = models.IntegerField(default=0, verbose_name="層厚(mm)") # ...
使用Django的ModelForm轉(zhuǎn)化為表單代碼如下:
class MeasureTaskNewForm(forms.ModelForm): class Meta: model = MeasureTask fields = ('taskname', 'staff', 'start_mileage', 'end_mileage', 'road', 'level_thick', 'step', 'equip', 'comment')
如果不做進一步處理,在網(wǎng)頁中使用這個From時,關(guān)聯(lián)字段會自動轉(zhuǎn)化為一個select控件,里面包含了所有選項,如下圖:
實際應(yīng)用時,需要對關(guān)聯(lián)的字段做一些選擇過濾。期望的結(jié)果如下:
解決方式
在From類中設(shè)置一個初始化函數(shù):
class MeasureTaskNewForm(forms.ModelForm): class Meta: model = MeasureTask fields = ('taskname', 'staff', 'start_mileage', 'end_mileage', 'road', 'level_thick', 'step', 'equip', 'comment') # 對參數(shù)作初始化設(shè)置,導(dǎo)致返回之后的Form驗證失敗 def __init__(self, road_choices=None, *args, **kwargs): super(MeasureTaskNewForm, self).__init__(*args, **kwargs) if road_choices: self.fields['road'].choices = road_choices
應(yīng)用這個類的方式如下,注意傳入?yún)?shù)的數(shù)據(jù)類型,
# 對關(guān)聯(lián)數(shù)據(jù)過濾 roads = Road.objects.filter(project=p_item) # 生成值,分別對應(yīng)于 html 中 select->option 設(shè)置 choices = roads.values_list('id', 'name') dataform = MeasureTaskNewForm(road_choices=choices) # dataform = MeasureTaskNewFormShadow() return render(request, "mdata/html/measure_task_add.html", locals())
生成的html代碼:
<select name="road" id="id_road" class="form-control"> <option value="1">北四環(huán)主線</option> <option value="5">匝道A</option> </select>
到這里生成的表單頁面沒有問題了,但是表單提交返回時如果還是用這個From來接收Request數(shù)據(jù),則會出現(xiàn)數(shù)據(jù)校驗失敗的問題
if request.method == "POST": dataform = MeasureTaskNewForm(request.POST) # 這里將出現(xiàn)校驗失敗的問題 if dataform.is_valid(): dataform.save() return redirect('mdata:measure_task', pid=p_item.id)
為了解決這個問題,另外做了一個沒有初始化函數(shù)的表單類來接收數(shù)據(jù).
# 影子表單模型 class MeasureTaskNewFormShadow(forms.ModelForm): class Meta: model = MeasureTask fields = ('taskname', 'staff', 'start_mileage', 'end_mileage', 'road', 'level_thick', 'step', 'equip', 'comment')
if request.method == "POST": dataform = MeasureTaskNewFormShadow(request.POST) if dataform.is_valid(): dataform.save() return redirect('mdata:measure_task', pid=p_item.id)
感覺這里應(yīng)該有更好的方法,嘗試對Form的初始化函數(shù)做了一些修改,但是沒有成功。
參考資料
https://qastack.cn/programming/813418/django-set-field-value-after-a-form-is-initialized
http://hk.uwenku.com/question/p-vdjpsmjn-bes.html
https://www.itranslater.com/qa/details/2325790729974580224
到此這篇關(guān)于Django表單外鍵選項初始化的文章就介紹到這了,更多相關(guān)Django表單初始化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- django表單中的按鈕獲取數(shù)據(jù)的實例分析
- Django def clean()函數(shù)對表單中的數(shù)據(jù)進行驗證操作
- Django 構(gòu)建模板form表單的兩種方法
- Django form表單與請求的生命周期步驟詳解
- Django model.py表單設(shè)置默認值允許為空的操作
- Django表單提交后實現(xiàn)獲取相同name的不同value值
- Django框架獲取form表單數(shù)據(jù)方式總結(jié)
- django之從html頁面表單獲取輸入的數(shù)據(jù)實例
- 解決django中form表單設(shè)置action后無法回到原頁面的問題
- django-xadmin根據(jù)當前登錄用戶動態(tài)設(shè)置表單字段默認值方式
- Django給表單添加honeypot驗證增加安全性
相關(guān)文章
Python標準庫之隨機數(shù) (math包、random包)介紹
這篇文章主要介紹了Python標準庫之隨機數(shù) (math包、random包)介紹,本文講解了math包的常用函數(shù),同時給出了random包的使用例子,需要的朋友可以參考下2014-11-11Python 內(nèi)置變量和函數(shù)的查看及說明介紹
今天小編就為大家分享一篇Python 內(nèi)置變量和函數(shù)的查看及說明介紹,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-12-12Python編程實現(xiàn)tail-n查看日志文件的方法
這篇文章主要介紹了Python編程實現(xiàn)tail-n查看日志文件的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-07-07Pandas DataFrame轉(zhuǎn)換為字典的方法
實際開發(fā)中我們可能會遇到一類問題,如何將Pandas DataFrame轉(zhuǎn)換為字典,本文就來介紹一下,感興趣的可以了解一下2021-05-05不到40行代碼用Python實現(xiàn)一個簡單的推薦系統(tǒng)
這篇文章主要給大家介紹了如何利用不到40行python代碼實現(xiàn)一個簡單的推薦系統(tǒng),文中通過示例代碼介紹的非常詳細,對大家學習或者使用Python具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧2019-05-05如何使用VSCode愉快的寫Python于調(diào)試配置步驟
從我的使用經(jīng)驗出發(fā),可以說VSCode用來寫Python真的是再合適不過了,你將體驗到絲滑的編程體驗和無限擴展的可能。而且,如果你的項目是包含多種語言的,比如Web開發(fā),你不必再開多個編輯器和其他工具,因為這一切都可以在VSCode里完成了2018-04-04