2010-04-12 81 views
5

表单将吐出未知数量的待回答问题。每个问题都包含提示,值字段和单位字段。该表单是在运行时在formclass的init方法中构建的。django - 动态表单字段集

编辑:每个问题都会收到一个独特提示作为标签,以及select元素的唯一单位列表。

对于可迭代的表单字段集来说,这似乎是一个完美的选择,它可以很容易地设计。但由于字段集 - 例如django-form-utils中的字段集被定义为元组,所以它们是不可变的......我找不到在运行时定义它们的方法。这是可能的,或者可能是另一种解决方案?

编辑:

表单集与initial_data不是答案 - initial_data只是使默认值设置为一个formset表单字段。项目列表不能通过initial_data发送到choicefield构造函数。

......除非我错了。

+0

那岂不是更好地使用[表单集(HTTP://docs.djangoproject .com/en/dev/topics/forms/formsets /)而不是字段集?一个问题的自定义Form类(带有'prompt'属性),然后使用['initial'关键字参数]加载问题数据(http://docs.djangoproject.com/en/dev/topics/forms/formssets /#使用-初始数据上带有一个-表单集)? – 2010-04-12 21:34:14

+0

表单集不会诀窍。至少不是给定的formset_factory。我必须能够为formset中的每个表单的实际字段构造函数提供一些参数 - 值字段的标签/提示以及choicefield的单元列表。 – 2010-04-20 04:49:02

回答

2

结账formsets。您应该能够将N个问题中的每一个的数据传递为initial data。沿着这些线:

question_data = [] 
for question in your_question_list: 
    question_data.append({'prompt': question.prompt, 
          'value': question.value, 
          'units': question.units}) 
QuestionFormSet = formset_factory(QuestionForm, extra=2) 
formset = QuestionFormSet(initial=question_data) 
+0

初始数据用于为表单元素提供默认值,而不是为formset中的表单构造提供数据。 – 2010-04-20 04:25:59

0

我用下面的技巧来创建一个动态的formset。从视图中调用create_dynamic_formset()函数。

def create_dynamic_formset(name_filter): 

    """ 
    -Need to create the classess dynamically since there is no other way to filter 
    """ 
    class FormWithFilteredField(forms.ModelForm): 
     type = forms.ModelChoiceField(queryset=SomeType.objects.filter(name__icontains=name_filter)) 

     class Meta: 
      model=SomeModelClass 

    return modelformset_factory(SomeModelClass, form=FormWithFilteredField) 
1

老问题,但我遇到了类似的问题。到目前为止,我发现的最接近的是这个片段,它基于几年前Malcom做过的一篇文章。 http://djangosnippets.org/snippets/1955/

原始代码段没有解决模板方面的问题,并将它们拆分为字段集,但将每个表单添加到其自己的字段集应该可以实现这一点。

forms.py

from django.forms.formsets import Form, BaseFormSet, formset_factory, \ 
      ValidationError 


    class QuestionForm(Form): 
     """Form for a single question on a quiz""" 
     def __init__(self, *args, **kwargs): 
      # CODE TRICK #1 
      # pass in a question from the formset 
      # use the question to build the form 
      # pop removes from dict, so we don't pass to the parent 
      self.question = kwargs.pop('question') 
      super(QuestionForm, self).__init__(*args, **kwargs) 

      # CODE TRICK #2 
      # add a non-declared field to fields 
      # use an order_by clause if you care about order 
      self.answers = self.question.answer_set.all(
        ).order_by('id') 
      self.fields['answers'] = forms.ModelChoiceField(
        queryset=self.answers()) 


    class BaseQuizFormSet(BaseFormSet): 
     def __init__(self, *args, **kwargs): 
      # CODE TRICK #3 - same as #1: 
      # pass in a valid quiz object from the view 
      # pop removes arg, so we don't pass to the parent 
      self.quiz = kwargs.pop('quiz') 

      # CODE TRICK #4 
      # set length of extras based on query 
      # each question will fill one 'extra' slot 
      # use an order_by clause if you care about order 
      self.questions = self.quiz.question_set.all().order_by('id') 
      self.extra = len(self.questions) 
      if not self.extra: 
       raise Http404('Badly configured quiz has no questions.') 

      # call the parent constructor to finish __init__    
      super(BaseQuizFormSet, self).__init__(*args, **kwargs) 

     def _construct_form(self, index, **kwargs): 
      # CODE TRICK #5 
      # know that _construct_form is where forms get added 
      # we can take advantage of this fact to add our forms 
      # add custom kwargs, using the index to retrieve a question 
      # kwargs will be passed to our form class 
      kwargs['question'] = self.questions[index] 
      return super(BaseQuizFormSet, self)._construct_form(index, **kwargs) 


    QuizFormSet = formset_factory(
     QuestionForm, formset=BaseQuizDynamicFormSet) 

views.py

from django.http import Http404 


    def quiz_form(request, quiz_id): 
     try: 
      quiz = Quiz.objects.get(pk=quiz_id) 
     except Quiz.DoesNotExist: 
      return Http404('Invalid quiz id.') 
     if request.method == 'POST': 
      formset = QuizFormSet(quiz=quiz, data=request.POST) 
      answers = [] 
      if formset.is_valid(): 
       for form in formset.forms: 
        answers.append(str(int(form.is_correct()))) 
       return HttpResponseRedirect('%s?a=%s' 
         % (reverse('result-display',args=[quiz_id]), ''.join(answers))) 
     else: 
      formset = QuizFormSet(quiz=quiz) 

     return render_to_response('quiz.html', locals()) 

模板

{% for form in formset.forms %} 
<fieldset>{{ form }}</fieldset> 
{% endfor %} 
0

以下是我用于类似的情况(变量集字段集,每一个包含一组可变字段)。

我用type()函数来建立我的Form类,BetterBaseForm类从django-form-utils

def makeFurnitureForm(): 
    """makeFurnitureForm() function will generate a form with 
    QuantityFurnitureFields.""" 

    furnitures = Furniture.objects.all() 
    fieldsets = {} 
    fields = {} 

    for obj in furnitures: 
     # I used a custom Form Field, but you can use whatever you want. 
     field = QuantityFurnitureField(name = obj.name) 

     fields[obj.name] = field 
     if not obj.room in fieldsets.keys(): 
      fieldsets[obj.room] = [field,] 
     else: 
      fieldsets[obj.room].append(field) 

    # Here I use a double list comprehension to define my fieldsets 
    # and the fields within. 
    # First item of each tuple is the fieldset name. 
    # Second item of each tuple is a dictionnary containing : 
    # -The names of the fields. (I used a list comprehension for this) 
    # -The legend of the fieldset. 
    # You also can add other meta attributes, like "description" or "classes", 
    # see the documentation for further informations. 
    # I added an example of output to show what the dic variable 
    # I create may look like. 
    dic = [(name, {"fields": [field.name for field in fieldsets[name]], "legend" : name}) 
      for name in fieldsets.keys()] 
    print(dic) 
    # Here I return a class object that is my form class. 
    # It inherits from both forms.BaseForm and forms_utils.forms.BetterBaseForm. 
    return (type("FurnitureForm", 
       (forms.BaseForm, form_utils.forms.BetterBaseForm,), 
       {"_fieldsets" : dic, "base_fields" : fields, 
        "_fieldset_collection" : None, '_row_attrs' : {}})) 

下面是如何dic可能看起来像一个例子:

[('fieldset name 1', 
    {'legend': 'fieldset legend 2', 
    'fields' ['field name 1-1']}), 
('fieldset name 2', 
    {'legend': 'fieldset legend 2', 
    'fields' : ['field 1-1', 'field 1-2']})] 

我用BetterBaseForm而非BetterForm出于同样的原因this article建议使用BaseForm而非Form

这篇文章很有意思,即使它很老,也解释了如何做动态表单(带有可变域设置)。它还提供了其他方法来实现动态表单。

虽然它没有解释如何使用字段集,但它激励我找到如何去做,并且原理保持不变。

在视图中使用它很简单:

return (render(request,'main/form-template.html', {"form" : (makeFurnitureForm())()})) 

,并在模板:

<form method="POST" name="myform" action="."> 
     {% csrf_token %} 
     <div> 
     {% for fieldset in form.fieldsets %} 
     <fieldset> 
      <legend>{{ fieldset.legend }}</legend> 
      {% for field in fieldset %} 
      <div> 
      {% include "main/furniturefieldtemplate.html" with field=field %} 
      </div> 
      {% endfor %} 
     </fieldset> 
     {% endfor %} 
     </div> 
     <input type="submit" value="Submit"/> 
    </form>