Add a grade field to the Grading Project Survey form.
authorDaniel Diniz <ajaksu@gmail.com>
Sat, 04 Jul 2009 14:15:11 +0200
changeset 2527 ce657149b90a
parent 2526 8f29bfb9eb52
child 2528 9de8348890d8
Add a grade field to the Grading Project Survey form. This also takes care of validating and transforming the data into True(pass) or False(fail). GradingProjectSurveys are hereby working for developers. Reviewed by: Lennard de Rijk
app/soc/views/helper/surveys.py
app/soc/views/models/grading_project_survey.py
--- a/app/soc/views/helper/surveys.py	Sat Jul 04 13:53:17 2009 +0200
+++ b/app/soc/views/helper/surveys.py	Sat Jul 04 14:15:11 2009 +0200
@@ -87,6 +87,7 @@
       survey_logic: instance of SurveyLogic.
       survey_record: a SurveyRecord entity.
       read_only: controls whether the survey taking UI allows data entry.
+      data: dictionary mapping fields to data for validation.
     """
 
     self.kwargs = kwargs
@@ -95,6 +96,7 @@
     self.survey_logic = self.kwargs.pop('survey_logic', None)
     self.survey_record = self.kwargs.pop('survey_record', None)
     self.read_only = self.kwargs.pop('read_only', None)
+    data = self.kwargs.pop('data', {})
 
     self.fields_map = dict(
         long_answer=self.addLongField,
@@ -104,7 +106,7 @@
         pick_quant=self.addQuantField,
         )
 
-    self.kwargs['data'] = {}
+    self.kwargs['data'] = data
     super(SurveyTakeForm, self).__init__(*args, **self.kwargs)
 
   def getFields(self, post_dict=None):
--- a/app/soc/views/models/grading_project_survey.py	Sat Jul 04 13:53:17 2009 +0200
+++ b/app/soc/views/models/grading_project_survey.py	Sat Jul 04 14:15:11 2009 +0200
@@ -22,12 +22,14 @@
   '"Lennard de Rijk" <ljvderijk@gmail.com>',
   ]
 
+from django import forms
 
 from soc.logic import dicts
 from soc.logic.models.survey import GRADES
 from soc.logic.models.survey import grading_logic as grading_survey_logic
 from soc.views.helper import access
 from soc.views.helper import decorators
+from soc.views.helper import surveys
 from soc.views.models import project_survey
 
 
@@ -71,7 +73,6 @@
         '?activate=1')
     return http.HttpResponseRedirect(redirect_path)
 
-
   def activateGrades(self, request, **kwargs):
     """Updates SurveyRecord's grades for a given Survey.
     """
@@ -80,57 +81,102 @@
     survey_logic.activateGrades(survey)
     return
 
-  def _takeGet(self, request, template, context, params, entity, record,
-              **kwargs):
-    """Hook for the GET request for the Survey's take page.
-
-    This method is called just before the GET page is shown.
+  def _getSurveyTakeForm(self, survey, record, params):
+    """Returns the specific SurveyTakeForm needed for the take view.
 
     Args:
-        template: the template used for this view
-        entity: the Survey entity
-        record: a SurveyRecord entity
-        rest: see base.View.public()
+        survey: a Survey entity
+        record: a SurveyRecord instance if any exist
+        params: the params dict for the requesting View
+
+    Returns:
+        An instance of GradseSurveyTakeForm.
     """
 
-    gradeField = self.addGradeField(entity, record)
-    field_count = len(eval(entity.survey_content.schema).items())
-    context['survey_form'].fields.insert(field_count + 1, 'grade', gradeField)
+    grade_choices = (('pass', 'Pass'), ('fail', 'Fail'))
+    survey_form = GradeSurveyTakeForm(survey_content=survey.survey_content,
+                                      survey_logic=params['logic'],
+                                      grade_choices=grade_choices)
+
+    return survey_form
+
 
-    return super(View, self)._takeGet(request, template, context,
-                                      params, entity, record, **kwargs)
+class GradeSurveyTakeForm(surveys.SurveyTakeForm):
+  """Extends SurveyTakeForm by adding a grade field.
 
-  def addGradeField(self, survey, survey_record):
-    """Adds a Grade Field to Survey.
+  The grade field logic is dependent on the kwarg 'grade_choices' (behavior
+  should be the same as the base class's if this argument is missing).
+  """
 
-    Used for mentor evaluations.
+  def __init__(self, *args, **kwargs):
+    """Store grade choices and initialize the form.
 
     params:
-      survey: the survey being taken
-      survey_record: an existing survey record for a user-project-survey combo,
-        or None
+      grade_choices: pair of tuples representing the grading choices
+    """
+
+    self.grade_choices = kwargs.pop('grade_choices', None)
+
+    if self.grade_choices:
+      # add grade field to self.data, respecting the data kwarg if present
+      data = kwargs.pop('data', {})
+      data['grade'] = None
+      kwargs['data'] = data
+
+    super(GradeSurveyTakeForm, self).__init__(*args, **kwargs)
+
+  def clean_grade(self):
+    """Validate the grade field.
+    """
+
+    grade = self.cleaned_data['grade']
+
+    # map to bool
+    grade_vals = {'pass': True, 'fail': False, '': ''}
+
+    return grade_vals.get(grade, None)
+
+  def getFields(self, post_dict=None):
+    """Add the extra grade field's value from POST or survey_record.
+
+    Args:
+        post_dict: dict with POST data if exists
     """
 
-    # Add a grade field determining if student passes or fails.
-    # Activate grades handler should determine whether new status
-    # is midterm_passed, final_passed, etc.
-    grade_choices = (('pass', 'Pass'), ('fail', 'Fail'))
-    grade_vals = { 'pass': True, 'fail': False }
-    from django import forms
-    gradeField = forms.fields.ChoiceField(choices=grade_choices,
-                                           required=True,
-                                           widget=forms.Select())
+    # fetch value from post_dict if it's there
+    post_dict = post_dict or {}
+    grade = post_dict.get('grade', None)
+
+    # otherwise, try to fetch from survey_record
+    if not grade and hasattr(self.survey_record, 'grade'):
+      grade = self.survey_record.grade
+
+    # remap bool to string values as the ChoiceField validates on 'choices'.
+    vals_grade = {True: 'pass', False: 'fail'}
+
+    if self.grade_choices:
+      self.data['grade'] = vals_grade.get(grade, None) or grade
+
+    super(GradeSurveyTakeForm, self).getFields(post_dict)
 
-    gradeField.choices.insert(0, (None, "Choose a Grade")  )
-    if survey_record:
-      for g in grade_choices:
-        if grade_vals[g[0]] == survey_record.grade:
-          gradeField.choices.insert(0, (g[0],g[1] + " (Saved)")   )
-          gradeField.choices.remove(g)
-          break;
-      gradeField.show_hidden_initial = True
+  def insertFields(self):
+    """Add ordered fields to self.fields, add grade field with grade choices.
+    """
+
+    # add common survey fields
+    super(GradeSurveyTakeForm, self).insertFields()
 
-    return gradeField
+    if self.grade_choices:
+      # add empty option to choices
+      grade_choices = (('', "Choose a Grade"),) + tuple(self.grade_choices)
+
+      gradeField = forms.fields.ChoiceField(choices=grade_choices,
+                                            required=True,
+                                            widget=forms.Select())
+      # add the grade field at the form's bottom
+      self.fields.insert(len(self.fields) + 1, 'grade', gradeField)
+
+    return self.fields
 
 
 view = View()