source: main/waeup.kofa/trunk/src/waeup/kofa/students/studylevel.py @ 12880

Last change on this file since 12880 was 12873, checked in by Henrik Bettermann, 10 years ago

Convert level into a schema field to be consistent with the documentation.

  • Property svn:keywords set to Id
File size: 11.5 KB
RevLine 
[7191]1## $Id: studylevel.py 12873 2015-04-23 19:27:29Z henrik $
2##
[6775]3## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
4## This program is free software; you can redistribute it and/or modify
5## it under the terms of the GNU General Public License as published by
6## the Free Software Foundation; either version 2 of the License, or
7## (at your option) any later version.
8##
9## This program is distributed in the hope that it will be useful,
10## but WITHOUT ANY WARRANTY; without even the implied warranty of
11## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12## GNU General Public License for more details.
13##
14## You should have received a copy of the GNU General Public License
15## along with this program; if not, write to the Free Software
16## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17##
18"""
19Container which holds the data of a student study level
20and contains the course tickets.
21"""
22import grok
[7536]23from zope.component.interfaces import IFactory
[10539]24from zope.catalog.interfaces import ICatalog
25from zope.component import createObject, queryUtility
[8330]26from zope.interface import implementedBy
[10631]27from waeup.kofa.interfaces import academic_sessions_vocab, VALIDATED
[7811]28from waeup.kofa.students.interfaces import (
[6781]29    IStudentStudyLevel, IStudentNavigation, ICourseTicket)
[7811]30from waeup.kofa.utils.helpers import attrs_to_fields
31from waeup.kofa.students.vocabularies import StudyLevelSource
[6775]32
[10276]33def find_carry_over(ticket):
[10277]34    studylevel = ticket.__parent__
35    studycourse = ticket.__parent__.__parent__
36    levels = sorted(studycourse.keys())
37    index = levels.index(str(studylevel.level))
38    try:
39        next_level = levels[index+1]
40    except IndexError:
41        return None
42    next_studylevel = studycourse[next_level]
43    co_ticket = next_studylevel.get(ticket.code, None)
44    return co_ticket
[10276]45
[9684]46def getGradeWeightFromScore(score):
47    if score is None:
48        return (None, None)
49    if score >= 70:
50        return ('A',5)
51    if score >= 60:
52        return ('B',4)
53    if score >= 50:
54        return ('C',3)
55    if score >= 45:
56        return ('D',2)
57    if score >= 40:
58        return ('E',1)
59    return ('F',0)
60
[6775]61class StudentStudyLevel(grok.Container):
62    """This is a container for course tickets.
63    """
64    grok.implements(IStudentStudyLevel, IStudentNavigation)
65    grok.provides(IStudentStudyLevel)
66
67    def __init__(self):
68        super(StudentStudyLevel, self).__init__()
69        return
70
[8736]71    @property
72    def student(self):
73        try:
74            return self.__parent__.__parent__
75        except AttributeError:
76            return None
[6775]77
[9235]78    @property
[9253]79    def certcode(self):
80        try:
81            return self.__parent__.certificate.code
82        except AttributeError:
83            return None
84
85    @property
[9235]86    def number_of_tickets(self):
87        return len(self)
88
[9257]89    @property
[9532]90    def total_credits(self):
91        total = 0
92        for ticket in self.values():
93            total += ticket.credits
94        return total
95
96    @property
[9912]97    def getSessionString(self):
98        return academic_sessions_vocab.getTerm(
99            self.level_session).title
100
101    @property
[10479]102    def gpa_params_rectified(self):
[10598]103        """Calculate corrected corrected level (sesional) gpa parameters.
104
105        The corrected gpa is displayed on transcripts only.
106        """
107        credits_weighted = 0.0
[9687]108        credits_counted = 0
[10276]109        level_gpa = 0.0
[9687]110        for ticket in self.values():
[10276]111            if ticket.carry_over is False and ticket.score:
112                if ticket.score < ticket.passmark:
113                    co_ticket = find_carry_over(ticket)
[10313]114                    if co_ticket is not None and co_ticket.weight is not None:
[10276]115                        credits_counted += co_ticket.credits
[10598]116                        credits_weighted += co_ticket.credits * co_ticket.weight
[10276]117                    continue
[9687]118                credits_counted += ticket.credits
[10598]119                credits_weighted += ticket.credits * ticket.weight
[9687]120        if credits_counted:
[10598]121            level_gpa = round(credits_weighted/credits_counted, 3)
122        return level_gpa, credits_counted, credits_weighted
[9687]123
124    @property
[10479]125    def gpa_params(self):
[10598]126        """Calculate gpa parameters for this level.
127        """
128        credits_weighted = 0.0
[10479]129        credits_counted = 0
130        level_gpa = 0.0
131        for ticket in self.values():
[10582]132            if ticket.score is not None:
[10479]133                credits_counted += ticket.credits
[10598]134                credits_weighted += ticket.credits * ticket.weight
[10479]135        if credits_counted:
[10598]136            level_gpa = round(credits_weighted/credits_counted, 3)
137        return level_gpa, credits_counted, credits_weighted
[10479]138
139    @property
140    def gpa(self):
141        return self.gpa_params[0]
142
143    @property
[10553]144    def passed_params(self):
[10598]145        """Determine the number and credits of passed and failed courses.
146
147        This method is used for level reports.
148        """
[10553]149        passed = failed = 0
[10616]150        courses_failed = []
151        credits_failed = 0
152        credits_passed = 0
[10553]153        for ticket in self.values():
154            if ticket.score is not None:
155                if ticket.score < ticket.passmark:
156                    failed += 1
[10616]157                    credits_failed += ticket.credits
158                    courses_failed.append(ticket.code)
[10553]159                else:
160                    passed += 1
[10616]161                    credits_passed += ticket.credits
162        return passed, failed, credits_passed, credits_failed, courses_failed
[10553]163
164    @property
[10598]165    def cumulative_params(self):
166        """Calculate the cumulative gpa and other cumulative parameters
167        for this level.
168
169        All levels below this level are taken under consideration
170        (including repeating levels). This method is used for level reports.
171        """
[10616]172        credits_passed = 0
[10598]173        total_credits = 0
174        total_credits_counted = 0
175        total_credits_weighted = 0
[10618]176        cgpa = 0.0
[10598]177        for level in self.__parent__.values():
178            if level.level > self.level:
179                continue
[10616]180            credits_passed += level.passed_params[2]
[10598]181            total_credits += level.total_credits
182            gpa_params = level.gpa_params
183            total_credits_counted += gpa_params[1]
184            total_credits_weighted += gpa_params[2]
185        if total_credits_counted:
186            cgpa = round(total_credits_weighted / total_credits_counted, 3)
187        return (cgpa, total_credits_counted, total_credits_weighted,
[10616]188               total_credits, credits_passed)
[10598]189
190    @property
[9257]191    def is_current_level(self):
192        try:
193            return self.__parent__.current_level == self.level
194        except AttributeError:
195            return False
196
[8735]197    def writeLogMessage(self, view, message):
198        return self.__parent__.__parent__.writeLogMessage(view, message)
199
[6775]200    @property
201    def level_title(self):
202        studylevelsource = StudyLevelSource()
203        return studylevelsource.factory.getTitle(self.__parent__, self.level)
204
[8920]205    def addCourseTicket(self, ticket, course):
[6781]206        """Add a course ticket object.
207        """
[8920]208        if not ICourseTicket.providedBy(ticket):
[6781]209            raise TypeError(
210                'StudentStudyLeves contain only ICourseTicket instances')
[8920]211        ticket.code = course.code
212        ticket.title = course.title
213        ticket.fcode = course.__parent__.__parent__.__parent__.code
214        ticket.dcode = course.__parent__.__parent__.code
215        ticket.credits = course.credits
216        ticket.passmark = course.passmark
217        ticket.semester = course.semester
218        self[ticket.code] = ticket
[6781]219        return
220
[9501]221    def addCertCourseTickets(self, cert):
222        """Collect all certificate courses and create course
223        tickets automatically.
224        """
225        if cert is not None:
226            for key, val in cert.items():
227                if val.level != self.level:
228                    continue
229                ticket = createObject(u'waeup.CourseTicket')
230                ticket.automatic = True
231                ticket.mandatory = val.mandatory
232                ticket.carry_over = False
233                self.addCourseTicket(ticket, val.course)
234        return
235
[9690]236StudentStudyLevel = attrs_to_fields(
[10479]237    StudentStudyLevel, omit=['total_credits', 'gpa'])
[6781]238
[7536]239class StudentStudyLevelFactory(grok.GlobalUtility):
240    """A factory for student study levels.
241    """
242    grok.implements(IFactory)
243    grok.name(u'waeup.StudentStudyLevel')
244    title = u"Create a new student study level.",
245    description = u"This factory instantiates new student study level instances."
246
247    def __call__(self, *args, **kw):
248        return StudentStudyLevel()
249
250    def getInterfaces(self):
251        return implementedBy(StudentStudyLevel)
252
[6781]253class CourseTicket(grok.Model):
254    """This is a course ticket which allows the
255    student to attend the course. Lecturers will enter scores and more at
256    the end of the term.
[6783]257
258    A course ticket contains a copy of the original course and
[8920]259    certificate course data. If the courses and/or the referrin certificate
260    courses are removed, the corresponding tickets remain unchanged.
261    So we do not need any event
[6783]262    triggered actions on course tickets.
[6781]263    """
264    grok.implements(ICourseTicket, IStudentNavigation)
265    grok.provides(ICourseTicket)
266
[6795]267    def __init__(self):
[6781]268        super(CourseTicket, self).__init__()
[6795]269        self.code = None
[6781]270        return
271
[8736]272    @property
273    def student(self):
[8338]274        """Get the associated student object.
275        """
276        try:
277            return self.__parent__.__parent__.__parent__
278        except AttributeError:
279            return None
[6781]280
[9253]281    @property
282    def certcode(self):
283        try:
284            return self.__parent__.__parent__.certificate.code
285        except AttributeError:
286            return None
287
[9698]288    @property
289    def removable_by_student(self):
290        return not self.mandatory
291
[10631]292    @property
293    def editable_by_lecturer(self):
294        cas = grok.getSite()['configuration'].current_academic_session
295        if self.student.state == VALIDATED and self.student.current_session == cas:
296            return True
297        return False
298
[8735]299    def writeLogMessage(self, view, message):
300        return self.__parent__.__parent__.__parent__.writeLogMessage(view, message)
301
[9925]302    @property
303    def level(self):
[7633]304        """Returns the id of the level the ticket has been added to.
305        """
[8338]306        try:
307            return self.__parent__.level
308        except AttributeError:
309            return None
[7633]310
[9925]311    @property
312    def level_session(self):
[7633]313        """Returns the session of the level the ticket has been added to.
314        """
[8338]315        try:
316            return self.__parent__.level_session
317        except AttributeError:
318            return None
[7633]319
[9684]320    @property
321    def grade(self):
322        """Returns the grade calculated from score.
323        """
324        return getGradeWeightFromScore(self.score)[0]
[7633]325
[9684]326    @property
327    def weight(self):
328        """Returns the weight calculated from score.
329        """
330        return getGradeWeightFromScore(self.score)[1]
331
[10539]332    @property
333    def course(self):
334        """Returns the course the ticket is referring to. Returns
335        None if the course has been removed.
336
337        This method is not used in Kofa anymore.
338        """
339        cat = queryUtility(ICatalog, name='courses_catalog')
340        result = cat.searchResults(code=(self.code, self.code))
341        if len(result) != 1:
342            return None
343        return list(result)[0]
344
[6782]345CourseTicket = attrs_to_fields(CourseTicket)
[7548]346
347class CourseTicketFactory(grok.GlobalUtility):
348    """A factory for student study levels.
349    """
350    grok.implements(IFactory)
351    grok.name(u'waeup.CourseTicket')
352    title = u"Create a new course ticket.",
353    description = u"This factory instantiates new course ticket instances."
354
355    def __call__(self, *args, **kw):
356        return CourseTicket()
357
358    def getInterfaces(self):
359        return implementedBy(CourseTicket)
Note: See TracBrowser for help on using the repository browser.