Автоматически генерировать поля формы для формы в Django

У меня есть несколько моделей, и я хочу создать форму множественного выбора на основе этих данных. Таким образом, форма будет содержать запись для каждой категории, и выбор будет навыков в этой категории.

models.py

class SkillCategory(models.Model):
    name = models.CharField(max_length=50)

class Skill(models.Model):
    name = models.CharField(max_length=50)
    category = models.ForeignKey(SkillCategory)

Есть ли способ автоматически генерировать поля формы? Я знаю, что могу добавитьSkillCategory» укажите в форме для каждой категории SkillCategory, но причина для ее использования в качестве модели заключается в том, что категории навыков и навыков можно редактировать свободно.

Я хочу сделать что-то вроде этого: (Я пытался это, но нене заставить его работать, нене помню точную ошибку ...)

forms.py

class SkillSelectionForm(forms.Form):
    def __init__(*args, **kwargs):
        super(SkillSelectionForm, self).__init__(*args, **kwargs)
        for c in SkillCategory.objects.all():
            category_skills = [(pk, s.name) for s in c.skill_set.all()]
            setattr(self, c.name, forms.MultipleChoiceField(choices=category_skills, widget=forms.CheckboxSelectMultiple))

РЕШЕНИЕ

Это создает запись поля формы с использованием SkillCategory.name и назначает варианты, как в Skill. field_name / display_name используются, чтобы избежать проблем с не-asciiназвания категорий.

forms.py

def get_categorized_skills():
    skills = {}
    for s in Skill.objects.values('pk', 'name', 'category__name').order_by('category__name'):
        if s['category__name'] not in skills.keys():
            skills[s['category__name']] = []
        skills[s['category__name']].append((s['pk'], s['name']))
    return skills

class SkillSelectionForm(forms.Form): 
    def __init__(self, *args, **kwargs):
        super(SkillSelectionForm, self).__init__(*args, **kwargs)
        skills = get_categorized_skills()
        for idx, cat in enumerate(skills.keys()):
            field_name = u'category-{0}'.format(idx)
            display_name = cat
            self.fields[field_name] = forms.MultipleChoiceField(choices=skills[cat], widget=forms.CheckboxSelectMultiple, label=display_name)

Ответы на вопрос(3)

Ваш ответ на вопрос