Django用to_field值形成干净的外键

时间:2012-08-19 13:09:27

标签: django forms validation foreign-keys

class Advert(models.Model):
    location_relation = models.ForeignKey("administrative_division.Towns", to_field='name_unique')


class Towns(models.Model):
    name = models.CharField(max_length=128)
    slug = models.SlugField()

    voivodship = models.ForeignKey("Voivodships", to_field='voivodship_identifier')
    county = models.ForeignKey("Counties")

    name_unique=models.CharField(max_length=255, unique=True)
    town_unique=models.BooleanField()

    def __unicode__(self):
        return "%s. %s"% (self.id, self.name) 



class AdvertForm(ModelForm):

    category = CustomTreeNodeChoiceField(queryset=Category.objects.filter(parent__isnull=False), empty_label="Wybierz kategorię", label="Kategoria")

    class Meta:
        model = Advert
        exclude = ('ip', 'user')
        widgets = {
            'location_relation': TextInput
        }

    def __init__(self, *args, **kwargs):
        self.request = kwargs.pop('request', None)
        super(AdvertForm, self).__init__(*args, **kwargs)

    def clean_location_relation(self):
        from string import capitalize
        from django import forms
        from administrative_division.models import Towns

        cleaned_data = self.cleaned_data['location_relation']
        try:
            cleaned_data = Towns.objects.get(name_unique=capitalize(cleaned_data.name_unique))
        except:
            raise forms.ValidationError("Such city does not exist")

        return cleaned_data

我使用文本输入表单字段而不是选择字段。所以我需要手动提供唯一名称。

我需要的是将提供的字符串大写到名为location_relation的字段。 我尝试了不同的选项,比如在模型中使用干净的方法,现在我尝试以模型形式进行清理。

此刻我真的不知道怎么做。

请给我一些提示或解决这个问题:)

1 个答案:

答案 0 :(得分:0)

我刚刚做了这样的事情;你应该继承ModelChoiceField并修改它的prepare_valueclean方法来处理非整数输入(这是FK -ModelChoiceField所需的)。

就我而言,它是一个预先输入的外键文本字段,如果它尚不存在则会创建新条目(例如对标记很有用)。

class MyModelChoiceField(forms.ModelChoiceField):
    def __init__(self, *args, **kwargs):
        self.widget = kwargs.pop('widget', forms.TextInput)
        super(MyModelChoiceField, self).__init__(*args, **kwargs)

    def prepare_value(self, value):
        # return value from original value, a PK.
        value = super(MyModelChoiceField, self).prepare_value(value)
        if not value:
            return value

        try:
            return self.queryset.model.objects.get(id=value).name
        except self.queryset.model.DoesNotExist:
            return ''

    def clean(self, value):
        # return instance from text input string
        if not value and not self.required:
            return None
        elif not value:
            raise forms.ValidationError("%s is required" % (self.queryset.model._meta.verbose_name,))
        try:
            return self.queryset.model.objects.filter(my_field=value).get()
        except self.queryset.model.DoesNotExist:
            raise forms.ValidationError("%s is not a valid option" % (self.queryset.model._meta.verbose_name,))


class MyForm(forms.ModelForm):
    # ...
    my_field = MyModelChoiceField(queryset=Foo, etc.)

结合自定义窗口小部件,我现在可以将它用于任何带有几行代码的FK字段,并自动包含我的typeahead JS。我爱django!