Limit ManyToMany fields in forms

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
from django import forms


class LimitManyToManyFields(object):
    """
    Limit ManyToMany fields in forms. Hide the field, if only one item can be selected.
    
    e.g. limit sites choices only to accessible sites:
    --------------------------------------------------------------------------
    class MyModel(models.Model):
        sites = models.ManyToManyField(Site)
        ...
        
    class UserProfile(models.Model):
        sites = models.ManyToManyField(Site)
        ...
        
    class MyForm(LimitManyToManyFields, forms.ModelForm): # <- Order is important !
        class Meta:
            model = MyModel
            
    def my_view(request):
        user_profile = request.user.get_profile()

        m2m_limit = {"sites": user_profile.sites.values_list("id", "name")}
        
        if request.method == "POST":
            form = MyForm(m2m_limit, request.POST)
            if form.is_valid():
                ...
        else:
            # Preselect all existing sites
            form = MyForm(m2m_limit)
        ...
    --------------------------------------------------------------------------
    """
    def __init__(self, m2m_limit, *args, **kwargs):
        """
        preselect site select options. If user can only access one site or there exist only one site
        we remove the site field and insert the site info in save() method.
        """
        assert isinstance(m2m_limit, dict), \
            "%s error: first argument must be the m2m limit dict!" % self.__class__.__name__

        super(LimitManyToManyFields, self).__init__(*args, **kwargs)

        for field_name, limits in m2m_limit.iteritems():
            if len(limits) == 1:
                value = limits[0][0]
                # Only one item can be selected. Hide the ManyToMany field. To hide the field and
                # for validation, we changed the MultipleChoiceField to a IntegerField.
                self.fields[field_name] = forms.IntegerField(
                    max_value=value, min_value=value, initial=value
                )
                self.fields[field_name].widget.input_type = 'hidden'
                self.fields[field_name].widget.is_hidden = True
            else:
                # Limit the ManyToMany field choices
                self.fields[field_name].choices = limits

More like this

  1. manually models unique_together check via signals by jedie 3 years, 11 months ago
  2. Limit queryset to objects related to parent in ManyToMany fields within admin inlines by DrMeers 3 years ago
  3. superuser only decorator by jedie 4 years ago
  4. ManyToMany field with newforms by lawgon 5 years, 6 months ago
  5. jQuery Double Click Edit ManyToMany in Admin by justhamade 3 years, 4 months ago

Comments

(Forgotten your password?)