在Django的CreateView中禁用表单字段

时间:2016-12-19 11:52:59

标签: django django-forms django-class-based-views django-generic-views

这个问题也可以表述为:“如何更改CreateView生成表单的属性?”

我正在使用“CreateView”在Django 1.10中生成视图和相关表单。这个想法是普通用户(教师)只能作为他/她自己创建模型的实例,但是特权用户可以创建实例并分配给任何教师。我想对两者使用相同的视图/表单。

模特:

class Set(models.Model):
    name = models.CharField(
        max_length=40,
    )
    matter = models.ForeignKey(
        Matter,
        on_delete=models.SET_NULL,
        null=True,
    )
    group = models.ForeignKey(
        Group,
        on_delete=models.SET_NULL,
        null=True,
    )
    teacher = models.ForeignKey(
        Teacher,
        on_delete=models.PROTECT,
    )

从技术上讲,它将包括填写当前“老师”登录的表单并禁用该字段,除非当前用户具有特权。我目前使用以下代码进行初始值设置,但我不知道如何阻止普通用户修改“教师”字段。

class SetCreate(LoginRequiredMixin, CreateView):
    model = Set
    fields = ('name', 'matter', 'group', 'teacher')

    def get_initial(self):
        return {'teacher': self.request.user.teacher.id}

    # code to limit 'teacher' field editing

我尝试的另一个选项是使用正确的“老师”创建实例,例如:

class SetCreate(LoginRequiredMixin, CreateView):
    model = Set
    fields = ('name', 'matter', 'group')

    def form_valid(self, form):
        form.instance.teacher = self.request.user.teacher
        return super(SetCreate, self).form_valid(form)

它工作得很好但我不能用它来编辑特权用户的“老师”字段。

我知道表格字段中存在'Field.disable'属性,但如果可能,我不知道如何更改'CreateView'生成表单的属性。

另一种选择是限制表单下拉列表中的选项,但我不知道如何操作。

欢迎任何建议,包括不同的观点。

1 个答案:

答案 0 :(得分:3)

查看代码,编辑视图会根据get_form_class属性的值在fields内生成表单。这被定义为类属性,但代码实际上通过self.fields引用它,因此没有理由不能在实例级别覆盖它。所以,你可以这样做:

class SetCreate(LoginRequiredMixin, CreateView):
    model = Set
    fields = ('name', 'matter', 'group')

    def get_form_class(self, *args, **kwargs):
        if self.request.user.has_the_right_permission():
            self.fields += ('teacher',)
        else:
            self.fields = self.fields
        return super(SetCreate, self).get_form_kwargs(*args, **kwargs)

(看似毫无意义的其他块是为了确保我们始终设置一个实例变量。)