django實(shí)現(xiàn)模型字段動(dòng)態(tài)choice的操作
需求是根據(jù)當(dāng)前登錄用戶來(lái)顯示某個(gè)choice字段不同的選擇項(xiàng)。
先放現(xiàn)在的實(shí)現(xiàn)版本。
1、重寫(xiě)PushRuleForm的__init__方法,
讓每次實(shí)例化PushRuleForm時(shí),test_mode字段的choices根據(jù)用戶重新賦值
class PushRuleForm(forms.ModelForm): def __init__(self, *args, **kwargs): if self.request.user.username in Const.TEST_USER_LIST: # 如果進(jìn)入都是add添加新項(xiàng)的頁(yè)面 if not kwargs.get(’instance’):# self.fields[’test_mode’].initial = 1self.fields[’test_mode’].choices = [(1,’Test’)] # else: # self.fields[’test_mode’].choices = [choice for choice in [(0,’OnLine’),(1,’Test’)] if self.instance.test_mode in choice]
2、重寫(xiě)PushRuleAdmin的changeform_view方法,進(jìn)入add和change頁(yè)面都會(huì)調(diào)用changeform_view方法,都能讓form獲取request屬性,所以重寫(xiě)這個(gè)方法比較好,PushRuleForm獲取request屬性后,form表單處理是就能通request.user.username取用戶名
class PushRuleAdmin: form = PushRuleForm def changeform_view(self, request, object_id=None, form_url=’’, extra_context=None): self.form.request = request return super(PushRuleAdmin, self).changeform_view(request, object_id, extra_context=extra_context)
mode.py對(duì)應(yīng)的代碼如下:
class PushRule(models.Model): test_mode = models.IntegerField(verbose_name=’TestMode’, default=0, choices=[(0,’OnLine’),(1,’Test’)])
實(shí)現(xiàn)方式2:
,重寫(xiě)PushRuleAdmin的render_change_form方法,傳入test_user_list上下文,通過(guò)js來(lái)判斷當(dāng)前用戶是否是測(cè)試用戶。
class PushRuleAdmin: def render_change_form(self, request, context, add=False, change=False, form_url=’’, obj=None): context[’test_user_list’]=Const.TEST_USER_LIST return super(PushRuleAdmin, self).render_change_form(request, context, add=False, change=False, form_url=’’, obj=None)
js代碼:
if (test_user_list.includes(username)) { $('#id_test_mode > option[value=’0’]').remove(); }
html模板代碼:
為了讓js獲取django模板變量,先定義一個(gè)username和test_user_list變量
<script> var username='{{ user.username }}', test_user_list='{{ test_user_list }}'</script>
不過(guò)這么的壞處是用戶列表信息直接暴露在前端代碼里了,跟直接在js里維護(hù)一個(gè)測(cè)試用戶列表一樣的效果,遂放棄這種做法
實(shí)現(xiàn)方式3:
后端寫(xiě)一個(gè)視圖接口,返回對(duì)應(yīng)的test_user_list,js里寫(xiě)一個(gè)ajax請(qǐng)求,來(lái)請(qǐng)求這個(gè)視圖獲取test_user_list
實(shí)測(cè)沒(méi)有問(wèn)題。
實(shí)現(xiàn)方式4:
類似方法2,只不過(guò)不通過(guò)js來(lái)處理,直接通過(guò)django模板來(lái)處理,主要是重寫(xiě)django/contrib/admin/templates/admin/includes/fieldset.html這個(gè)模板文件,對(duì)django模板語(yǔ)法不太熟,遂放棄。
未實(shí)現(xiàn)的思路,想在PushRuleAdmin中直接修改model的test_mode字段的chioce選項(xiàng),不過(guò)沒(méi)實(shí)現(xiàn),
想修改model的fields,不過(guò)發(fā)現(xiàn)他是一個(gè)ImmutableList類型,修改會(huì)報(bào)錯(cuò)。
不過(guò)stackoverflow上的給出的這個(gè)方法不錯(cuò),可以參考,就是缺一個(gè)獲取用戶名的地方,哪天再看一下
補(bǔ)充知識(shí):django 中優(yōu)雅的使用 choice 字段
問(wèn)題
django中如何比較優(yōu)雅的對(duì)元組進(jìn)行標(biāo)記分類??墒褂胏hoice字段
choice字段
# models.pyclass BookTagNum(object): OTHER = 1 SCIENCE = 2 SOCIAL_SCIENCES = 3 ECONOMIC = 4 COMPUTER = 5class BOOK(models.Model): TAG_NUM_CHOICE = ( (BookTagNum.OTHER, ’其它’), (BookTagNum.SCIENCE, ’科學(xué)類’), (BookTagNum.SOCIAL_SCIENCES, ’社科類’), (BookTagNum.ECONOMIC, ’經(jīng)濟(jì)類’), (BookTagNum.COMPUTER, ’計(jì)算機(jī)類’), ) tag = models.IntegerField(choices=TAG_NUM_CHOICE)
在代碼中盡量不要出現(xiàn)固定的硬編碼,比如某個(gè)判斷條件,判斷書(shū)的分類為:
# view.pydef get(self, request): book = Book.obejects.filter(tag = BookTagNum.COMPUTER)
以上這篇django實(shí)現(xiàn)模型字段動(dòng)態(tài)choice的操作就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. Python如何批量生成和調(diào)用變量2. 通過(guò)CSS數(shù)學(xué)函數(shù)實(shí)現(xiàn)動(dòng)畫(huà)特效3. .net如何優(yōu)雅的使用EFCore實(shí)例詳解4. Python獲取B站粉絲數(shù)的示例代碼5. ASP.Net Core對(duì)USB攝像頭進(jìn)行截圖6. python基礎(chǔ)之匿名函數(shù)詳解7. ASP.NET MVC實(shí)現(xiàn)橫向展示購(gòu)物車8. ajax動(dòng)態(tài)加載json數(shù)據(jù)并詳細(xì)解析9. python b站視頻下載的五種版本10. ASP.Net Core(C#)創(chuàng)建Web站點(diǎn)的實(shí)現(xiàn)
