4

我正在尝试将某些字段更新为 ModelForm,这些字段不固定。(我只有tutor视图自动填充的那个)

模型:

class Session(models.Model):
  tutor = models.ForeignKey(User)
  start_time = models.DateTimeField()
  end_time = models.DateTimeField()
  status = models.CharField(max_length=1)

形式:

class SessionForm(forms.ModelForm):
  class Meta:
    model = Session
    exclude = ['tutor']

对于给定的会话,有时我只需要更新end_time,有时只需要start_time& end_time

我怎样才能在视图中做到这一点?


编辑

我已经给出了示例,但不限于这些示例,我需要更新的字段不是预定义的,我需要能够更新任何字段

4

4 回答 4

10

我以前也做过类似的事情,虽然不是很漂亮,但很有效。它涉及在运行时动态创建一个类型,并使用该类型。对于一些文档,您可以查看django 的 DynamicModels。

我们来了……你的要求。

  • 您希望能够使用表单更新模型
  • 您希望有选择地指定要在运行时更新的字段

所以,一些代码:

def create_form(model, field_names):
    # the inner class is the only useful bit of your ModelForm
    class Meta:
        pass
    setattr(Meta, 'model', model)
    setattr(Meta, 'include', field_names)
    attrs = {'Meta': Meta}

    name = 'DynamicForm'
    baseclasses = (forms.ModelForm,)
    form = type('DynamicForm', baseclasses, attrs)
    return form

def my_awesome_view(request):
    fields = ['start_time', 'end_time']
    form = create_form(Session, fields)
    # work with your form!
于 2012-04-19T14:19:13.713 回答
1

将您的表单字段设置为可为空,并使用“干净”方法向字段添加逻辑,例如:

class SessionForm(forms.ModelForm):
    def clean_end_date(self):
        cd = self.cleaned_data
        if (cd["start_date"] and cd["end_date"]) and cd["end_date"] < cd["start_date"]:
            raise forms.ValidationError("WTF?!")
        if not (cd["start_date"] or cd["end_date"]):
            raise forms.ValidationError("need one date")
        return cd['end_date']

如果要更改值,请对 return 语句使用不同的值。

这是您可能需要进行验证的内容。

如果需要,您可以在视图中复制 GET 字典并在实例化表单之前更新值

def my_view(request):
    r_data = request.GET.copy()
    r_data.merge(request.POST)

    data = dict([(key, my_function(key, value)) for key, value in r_data.iteritems() if key in MyForm.fields])
    form = MyForm(data=data)
    [...]

希望它有所帮助。

于 2012-04-19T08:22:03.573 回答
0

您可以像这样调用表单的构造函数:

class SessionForm(forms.ModelForm):
    class Meta:
        model = Session
        exclude = ['tutor', 'start_time']

    def __init__(self, your_extra_args, *args, **kwargs):
        super(SessionForm, self).__init__(*args, **kwargs)
        if need_end_time_start_time(your_extra_args):
            self.fields['start_time'] = forms.DateTimeField()     

要使用此类,您必须将参数“your_extra_args”传递给您的表单:

session_form = SessionForm('foo')
于 2012-04-19T09:16:54.843 回答
0

您的问题表明您有一个使用相同模型但不同字段隐藏在 ModelForm 中的要求,根据您的需要。

如果我理解您的问题,那么您需要创建两个不同的基于 ModelForm 的类,这两个类在 Meta 类中具有相同的模型值。排除和包括您需要的字段。所以现在你有不同的 ModelForm 类,你可以在你的视图逻辑中调用它们。

有了这个,您将拥有两种不同的表单,它们依赖于相同的模型,具有不同的默认字段。

通过巧妙的 url 和视图设计,您将能够使用上述想法为同一模型的多个表单提供服务。

于 2014-02-03T13:11:04.683 回答