Джанго Формы: скрытое модельное поле? - PullRequest
18 голосов
/ 14 января 2011

У меня есть Form.Я хочу включить скрытое поле, которое возвращает модель.Я установлю это значение в представлении;Мне просто нужно, чтобы оно было опубликовано на следующей странице.

Какое поле я должен использовать в классе формы?

Ответы [ 2 ]

29 голосов
/ 14 января 2011

Скрытое поле, которое возвращает модель?Итак, идентификатор экземпляра модели?

Виджет forms.HiddenInput должен помочь, будь то поле FK или CharField, в которое вы поместили идентификатор экземпляра модели.

class MyForm(forms.Form):
    hidden_2 = forms.CharField(widget=forms.HiddenInput())
    hidden_css = forms.CharField(widget=forms.MostWidgets(attrs={'style': 'display:none;'}))

Полагаю, самый быстрыйспособ получить эту работу -

class MyForm(forms.Form):
    model_instance = forms.ModelChoiceField(queryset=MyModel.objects.all(), widget=forms.HiddenInput())

form = MyForm({'model_instance': '1'})
form.cleaned_data['model_instance']

Но мне не нравится идея предоставления MyModel.objects.all (), если вы все равно собираетесь указать один элемент.

Кажется, чтобы избежать такого поведения, вам придется переопределить форму __init__ с помощью меньшего QuerySet.

Мне кажется, я предпочитаю старомодный способ:

class MyForm(forms.Form):
    model_instance = forms.CharField(widget=forms.HiddenInput())

    def clean_model_instance(self):
        data = self.cleaned_data['model_instance']
        if not data:
            raise forms.ValidationError()
        try:
            instance = MyModel.objects.get(id=data)
        except MyModel.DoesNotExist:
            raise forms.ValidationError()
        return instance
2 голосов
/ 04 сентября 2014

Подход в ответе Юджи использует метод clean_model_instance в форме, что хорошо, если вы когда-либо делаете это только один раз в своей базе кода. Если вы будете делать это чаще, вам может быть полезно реализовать поле пользовательской модели.

У меня есть код:

from django import forms

class ModelField(forms.Field):

    Model = None

    def prepare_value(self, value):
        """Inject entities' id value into the form's html data"""
        if isinstance(value, self.Model):
            return value.id
        return value

    def to_python(self, value):
        """More or less stolen from ModelChoiceField.to_python"""

        if value in self.empty_values:
            return None

        try:
            value = self.Model.objects.get(id=value)
        except (ValueError, self.Model.DoesNotExist):
            raise forms.ValidationError('%s does not exist'
                                         % self.Model.__class__.__name__.capitalize())

        return value

Если вы используете его в качестве базового класса, а затем специализируете его на своих собственных моделях, тогда он становится полезным на основе. Например:

# In app/fields.py
from .models import CustomModel

class CustomModelField(ModelField):

    Model = CustomModel

Затем вы можете связать это с любым виджетом, который вам нужен в данный момент:

# in app/forms.py
class MyForm(forms.Form):

    hidden_custom_model_field = CustomModelField(widget=forms.HiddenInput())

    other_widget_custom_model_field = CustomModelField(widget=MyCustomWidget())
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...