I need to make a form, which have 1 select and 1 text input. Select must be taken from database. model looks like this:
class Province(models.Model):
    name = models.CharField(max_length=30)
    slug = models.SlugField(max_length=30)
    def __unicode__(self):
        return self.name
It's rows to this are added only by admin, but all users can see it in forms. I want to make a ModelForm from that. I made something like this:
class ProvinceForm(ModelForm):
    class Meta:
        CHOICES = Province.objects.all()
        model = Province
        fields = ('name',)
        widgets = {
            'name': Select(choices=CHOICES),
        }
but it doesn't work. The select tag is not displayed in html. What did I wrong?
UPDATE:
This solution works as I wanto it to work:
class ProvinceForm(ModelForm):
    def __init__(self, *args, **kwargs):
        super(ProvinceForm, self).__init__(*args, **kwargs)
        user_provinces = UserProvince.objects.select_related().filter(user__exact=self.instance.id).values_list('province')
        self.fields['name'].queryset = Province.objects.exclude(id__in=user_provinces).only('id', 'name')
    name = forms.ModelChoiceField(queryset=None, empty_label=None)
    class Meta:
        model = Province
        fields = ('name',)
A QuerySet is a collection of data from a database. A QuerySet is built up as a list of objects. QuerySets makes it easier to get the data you actually need, by allowing you to filter and order the data.
You get a QuerySet by using your model's Manager . Each model has at least one Manager , and it's called objects by default. Access it directly via the model class, like so: >>> Blog.objects <django.db.models.manager.Manager object at ...> >>> b = Blog(name='Foo', tagline='Bar') >>> b.objects Traceback: ...
Appending the annotate() clause onto a QuerySet lets you add an attribute to each item in the QuerySet, like if you wanted to count the amount of articles in each category. However, sometimes you only want to count objects that match a certain condition, for example only counting articles that are published.
Choices limits the input from the user to the particular values specified in models.py . If choices are given, they're enforced by model validation and the default form widget will be a select box with these choices instead of the standard text field.
Read Maersu's answer for the method that just "works".
If you want to customize, know that choices takes a list of tuples, ie     (('val','display_val'), (...), ...)
Choices doc:
An iterable (e.g., a list or tuple) of 2-tuples to use as choices for this field.
from django.forms.widgets import Select
class ProvinceForm(ModelForm):
    class Meta:
        CHOICES = Province.objects.all()
        model = Province
        fields = ('name',)
        widgets = {
            'name': Select(choices=( (x.id, x.name) for x in CHOICES )),
        }
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With