1

我有以下型号:

class Recipe(models.Model):
    fields...

class Ingredient(models.Model):
    fields...

class UsesIngredient(models.Model):
    recipe = models.ForeignKey(Recipe)
    ingredient = models.ForeignKey(Ingredient)
    amount = models.FloatField()
    group = models.CharField()

我有一个视图,它允许用户通过动态表单集为某个配方添加任意数量的“UsesIngredient”模型。组属性自动填充为对用户隐藏。

问题是当用户在表单集中添加一个新表单但不填写任何字段时,我不希望保存该表单。但是,django 仍然尝试保存表单,因为“组”属性已“更改”(因为它已在创建额外表单时自动填写)。

有没有办法解决这个问题?

谢谢!

4

2 回答 2

2

好吧,我仍然对蒂姆埃德加的解决方案感到不满意,所以我一直在寻找。我想我找到了我要找的东西。'Form' 类有两个在这种情况下有用的未记录方法:'has_changed()' 和 '_get_changed_data'。

在 ModelFormSet 验证期间,每个表单都会检查“has_changed()”。如果表单未更改,则跳过验证并假定表单正确。同样,在 ModelFormSet 保存期间,save_new_objects 检查每个表单以查看它是否已更改。如果它没有更改,则不会保存表单。

所以我的解决方案是重写 has_changed() 方法,如果只有 'group' 属性发生了变化,并且所有其他字段都为空,则返回 False。这是我的实现:

class UsesIngredientForm(forms.ModelForm):    
    class Meta:
        model = UsesIngredient

    def has_changed(self, *args, **kwargs):
        self._get_changed_data(*args, **kwargs)
        # If group is in changed_data, but no other fields are filled in, remove group so
        # the form will not be validated or saved
        if 'group' in self._changed_data and len(self._changed_data) == 1:
            contains_data = False
            for name in ['ingredient', 'amount', 'unit']:
                field = self.fields[name]
                prefixed_name = self.add_prefix(name)
                data_value = field.widget.value_from_datadict(self.data, self.files, prefixed_name)
                if data_value:
                    contains_data = True
                    break
            if not contains_data:
                self._changed_data.remove('group')
        return bool(self._changed_data)

希望这对将来的任何人都有帮助!

编辑:我编辑了这个答案以反映蒂姆埃德加斯的评论。我意识到这个实现仍然使用“私有”方法,但我还没有找到只使用公开记录的方法的更干净的实现。但也许这只是我自己的无能:)。

于 2013-11-18T20:14:04.513 回答
0

您可以尝试通过设置使所有字段都需要一个值blank=False在这里查看更多。它应该要求验证您关心的值没有留空。

如果这不起作用,您可以尝试创建自己的自定义save方法来执行您关心的验证。

def save(self, *args, **kwargs):
   # Do your checks on the properties such as self.group, self.amount, etc
   # If it is fine then call
   super(UsesIngredient, self).save(*args, **kwargs)
于 2013-10-20T17:57:54.407 回答