source: main/waeup.kofa/trunk/src/waeup/kofa/students/studycourse.py @ 16257

Last change on this file since 16257 was 16040, checked in by Henrik Bettermann, 5 years ago

Do not show outstanding courses on transcript pages.

  • Property svn:keywords set to Id
File size: 6.4 KB
RevLine 
[7191]1## $Id: studycourse.py 16040 2020-03-13 17:25:31Z henrik $
2##
[6633]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 the student study courses
20and contains the (student) study level objects.
21"""
22import grok
23from zope.component.interfaces import IFactory
[8325]24from zope.component import createObject
[7256]25from zope.interface import implementedBy
[7811]26from waeup.kofa.students.interfaces import (
[15173]27    IStudentStudyCourse, IStudentNavigation, IStudentStudyLevel)
[7811]28from waeup.kofa.students.studylevel import CourseTicket
[8471]29from waeup.kofa.students.workflow import CLEARED, RETURNING, PAID
[7811]30from waeup.kofa.utils.helpers import attrs_to_fields
[6633]31
32class StudentStudyCourse(grok.Container):
33    """This is a container for study levels.
34    """
[15163]35    grok.implements(IStudentStudyCourse, IStudentNavigation)
[6633]36    grok.provides(IStudentStudyCourse)
37
[10155]38    is_special_postgrad = False
39
[6633]40    def __init__(self):
41        super(StudentStudyCourse, self).__init__()
42        return
43
[8736]44    @property
45    def student(self):
[6642]46        return self.__parent__
47
[8735]48    def writeLogMessage(self, view, message):
49        return self.__parent__.writeLogMessage(view, message)
50
[8323]51    @property
[8471]52    def next_session_allowed(self):
[8483]53        certificate = getattr(self, 'certificate', None)
54        if certificate == None:
55            return False
[8736]56        if self.student.state in (CLEARED, RETURNING):
[8471]57            return True
[8736]58        if self.student.state == PAID \
59            and self.student.is_postgrad:
[8471]60            return True
61        return False
[8323]62
[8483]63    @property
64    def is_postgrad(self):
[8972]65        if self.certificate is None:
66            return False
[8483]67        return self.certificate.study_mode.startswith('pg')
68
[9131]69    @property
70    def is_current(self):
[13002]71        if self.__name__ and '_' in self.__name__:
[9131]72            return False
73        return True
74
[10059]75    @property
76    def is_previous(self):
77        if self.__name__ == 'studycourse_2':
78            return True
79        if self.__name__ == 'studycourse_1' and  \
80            not self.__parent__.has_key('studycourse_2'):
81            return True
82        return False
83
[6782]84    def addStudentStudyLevel(self, cert, studylevel):
[6774]85        """Add a study level object.
86        """
87        if not IStudentStudyLevel.providedBy(studylevel):
88            raise TypeError(
89                'StudentStudyCourses contain only IStudentStudyLevel instances')
[6775]90        self[str(studylevel.level)] = studylevel
[9501]91        studylevel.addCertCourseTickets(cert)
[7661]92        # Collect carry-over courses in base levels (not in repeating levels)
[7664]93        try:
94            co_enabled = grok.getSite()['configuration'].carry_over
95        except TypeError:
96            # In tests we might not have a site object
97            co_enabled = True
[8337]98        if not co_enabled or studylevel.level % 100 != 0:
99            return
100        levels = sorted(self.keys())
101        index = levels.index(str(studylevel.level))
102        if index <= 0:
103            return
104        previous_level = self[levels[index-1]]
105        for key, val in previous_level.items():
[14134]106            if val.total_score >= val.passmark:
[8337]107                continue
108            if key in self[str(studylevel.level)]:
109                # Carry-over ticket exists
110                continue
111            co_ticket = createObject(u'waeup.CourseTicket')
112            for name in ['code', 'title', 'credits', 'passmark',
113                         'semester', 'mandatory', 'fcode', 'dcode']:
114                setattr(co_ticket, name, getattr(val, name))
115            co_ticket.automatic = True
116            co_ticket.carry_over = True
117            self[str(studylevel.level)][co_ticket.code] = co_ticket
[6774]118        return
119
[10178]120    def getTranscriptData(self):
121        """Get a sorted list of dicts with level and course ticket data.
122
123        This method is used for transcripts.
124        """
125        levels = []
[10260]126        cgpa = 0.0
127        total_credits_counted = 0
[10178]128        for level_key, level_val in self.items():
[15213]129            # Skip Level Zero
130            if level_key == '0':
131                continue
[10178]132            tickets_1 = []
133            tickets_2 = []
134            tickets_3 = []
[10250]135            tickets = sorted(level_val.values(), key=lambda ticket: ticket.code)
136            for ticket in tickets:
[16040]137                if ticket.outstanding:
138                    continue
[10178]139                if ticket.semester == 1:
140                    tickets_1.append(ticket)
141                elif ticket.semester == 2:
142                    tickets_2.append(ticket)
143                elif ticket.semester == 3:
144                    tickets_3.append(ticket)
[10479]145            gpa_params = level_val.gpa_params_rectified
[10276]146            sgpa = gpa_params[0]
147            total_credits_counted += gpa_params[1]
148            cgpa += gpa_params[2]
[10178]149            levels.append(
150                dict(level_key=level_key,
151                     level=level_val,
152                     tickets_1=tickets_1,
153                     tickets_2=tickets_2,
[10260]154                     tickets_3=tickets_3,
155                     sgpa=sgpa))
156        if total_credits_counted:
[14382]157            cgpa = cgpa/total_credits_counted
[14205]158        # Override cgpa if value has been imported
159        # (not implemented in base package)
160        imported_cgpa = getattr(self, 'imported_cgpa', None)
161        if imported_cgpa:
162            cgpa = imported_cgpa
[10260]163        return sorted(levels, key=lambda level: level['level_key']), cgpa
[10178]164
[6815]165StudentStudyCourse = attrs_to_fields(StudentStudyCourse)
166
[6822]167class StudentStudyCourseFactory(grok.GlobalUtility):
[7536]168    """A factory for student study courses.
[6822]169    """
170    grok.implements(IFactory)
171    grok.name(u'waeup.StudentStudyCourse')
172    title = u"Create a new student study course.",
173    description = u"This factory instantiates new student study course instances."
174
175    def __call__(self, *args, **kw):
176        return StudentStudyCourse()
177
178    def getInterfaces(self):
[7811]179        return implementedBy(StudentStudyCourse)
Note: See TracBrowser for help on using the repository browser.