source: main/waeup.kofa/trunk/src/waeup/kofa/students/utils.py @ 11599

Last change on this file since 11599 was 11595, checked in by Henrik Bettermann, 11 years ago

constructMatricNumber must not be private. We will be calling it in custom packages.

  • Property svn:keywords set to Id
File size: 37.9 KB
RevLine 
[7191]1## $Id: utils.py 11595 2014-04-23 08:26:03Z henrik $
2##
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##
[7358]18"""General helper functions and utilities for the student section.
[6651]19"""
[7150]20import grok
[8595]21from time import time
[7318]22from reportlab.lib import colors
[7019]23from reportlab.lib.units import cm
24from reportlab.lib.pagesizes import A4
[9015]25from reportlab.lib.styles import getSampleStyleSheet
26from reportlab.platypus import Paragraph, Image, Table, Spacer
[11589]27from zope.event import notify
[9922]28from zope.schema.interfaces import ConstraintNotSatisfied
[9015]29from zope.component import getUtility, createObject
[7019]30from zope.formlib.form import setUpEditWidgets
[9015]31from zope.i18n import translate
[8596]32from waeup.kofa.interfaces import (
[9762]33    IExtFileStore, IKofaUtils, RETURNING, PAID, CLEARED,
34    academic_sessions_vocab)
[7811]35from waeup.kofa.interfaces import MessageFactory as _
36from waeup.kofa.students.interfaces import IStudentsUtils
[10706]37from waeup.kofa.students.workflow import ADMITTED
[11589]38from waeup.kofa.students.vocabularies import StudyLevelSource, MatNumNotInSource
[9910]39from waeup.kofa.browser.pdf import (
[9965]40    ENTRY1_STYLE, format_html, NOTE_STYLE, HEADING_STYLE,
[11550]41    get_signature_tables, get_qrcode)
[9910]42from waeup.kofa.browser.interfaces import IPDFCreator
[10256]43from waeup.kofa.utils.helpers import to_timezone
[6651]44
[7318]45SLIP_STYLE = [
46    ('VALIGN',(0,0),(-1,-1),'TOP'),
47    #('FONT', (0,0), (-1,-1), 'Helvetica', 11),
48    ]
[7019]49
[7318]50CONTENT_STYLE = [
51    ('VALIGN',(0,0),(-1,-1),'TOP'),
52    #('FONT', (0,0), (-1,-1), 'Helvetica', 8),
53    #('TEXTCOLOR',(0,0),(-1,0),colors.white),
[9906]54    #('BACKGROUND',(0,0),(-1,0),colors.black),
55    ('INNERGRID', (0,0), (-1,-1), 0.25, colors.black),
56    ('BOX', (0,0), (-1,-1), 1, colors.black),
57
[7318]58    ]
[7304]59
[7318]60FONT_SIZE = 10
61FONT_COLOR = 'black'
62
[8112]63def trans(text, lang):
64    # shortcut
65    return translate(text, 'waeup.kofa', target_language=lang)
66
[10261]67def formatted_text(text, color=FONT_COLOR, lang='en'):
[7511]68    """Turn `text`, `color` and `size` into an HTML snippet.
[7318]69
[7511]70    The snippet is suitable for use with reportlab and generating PDFs.
71    Wraps the `text` into a ``<font>`` tag with passed attributes.
72
73    Also non-strings are converted. Raw strings are expected to be
74    utf-8 encoded (usually the case for widgets etc.).
75
[7804]76    Finally, a br tag is added if widgets contain div tags
77    which are not supported by reportlab.
78
[7511]79    The returned snippet is unicode type.
80    """
81    if not isinstance(text, unicode):
82        if isinstance(text, basestring):
83            text = text.decode('utf-8')
84        else:
85            text = unicode(text)
[9717]86    if text == 'None':
87        text = ''
[8141]88    # Mainly for boolean values we need our customized
89    # localisation of the zope domain
[10261]90    text = translate(text, 'zope', target_language=lang)
[7804]91    text = text.replace('</div>', '<br /></div>')
[9910]92    tag1 = u'<font color="%s">' % (color)
[7511]93    return tag1 + u'%s</font>' % text
94
[8481]95def generate_student_id():
[8410]96    students = grok.getSite()['students']
97    new_id = students.unique_student_id
98    return new_id
[6742]99
[7186]100def set_up_widgets(view, ignore_request=False):
[7019]101    view.adapters = {}
102    view.widgets = setUpEditWidgets(
103        view.form_fields, view.prefix, view.context, view.request,
104        adapters=view.adapters, for_display=True,
105        ignore_request=ignore_request
106        )
107
[11550]108def render_student_data(studentview, context, omit_fields=(),
109                        lang='en', slipname=None):
[7318]110    """Render student table for an existing frame.
111    """
112    width, height = A4
[7186]113    set_up_widgets(studentview, ignore_request=True)
[7318]114    data_left = []
[11550]115    data_middle = []
[7019]116    style = getSampleStyleSheet()
[7280]117    img = getUtility(IExtFileStore).getFileByContext(
118        studentview.context, attr='passport.jpg')
119    if img is None:
[7811]120        from waeup.kofa.browser import DEFAULT_PASSPORT_IMAGE_PATH
[7280]121        img = open(DEFAULT_PASSPORT_IMAGE_PATH, 'rb')
[7318]122    doc_img = Image(img.name, width=4*cm, height=4*cm, kind='bound')
123    data_left.append([doc_img])
124    #data.append([Spacer(1, 12)])
[9141]125
[10261]126    f_label = trans(_('Name:'), lang)
[9910]127    f_label = Paragraph(f_label, ENTRY1_STYLE)
[9911]128    f_text = formatted_text(studentview.context.display_fullname)
[9910]129    f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]130    data_middle.append([f_label,f_text])
[9141]131
[7019]132    for widget in studentview.widgets:
[9141]133        if 'name' in widget.name:
[7019]134            continue
[9911]135        f_label = translate(
[7811]136            widget.label.strip(), 'waeup.kofa',
[10261]137            target_language=lang)
[9911]138        f_label = Paragraph('%s:' % f_label, ENTRY1_STYLE)
[10261]139        f_text = formatted_text(widget(), lang=lang)
[9910]140        f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]141        data_middle.append([f_label,f_text])
[9141]142
[9452]143    if getattr(studentview.context, 'certcode', None):
[10250]144        if not 'certificate' in omit_fields:
[10261]145            f_label = trans(_('Study Course:'), lang)
[10250]146            f_label = Paragraph(f_label, ENTRY1_STYLE)
147            f_text = formatted_text(
[10650]148                studentview.context['studycourse'].certificate.longtitle)
[10250]149            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]150            data_middle.append([f_label,f_text])
[10250]151        if not 'department' in omit_fields:
[10261]152            f_label = trans(_('Department:'), lang)
[10250]153            f_label = Paragraph(f_label, ENTRY1_STYLE)
154            f_text = formatted_text(
155                studentview.context[
[10650]156                'studycourse'].certificate.__parent__.__parent__.longtitle,
[10250]157                )
158            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]159            data_middle.append([f_label,f_text])
[10250]160        if not 'faculty' in omit_fields:
[10261]161            f_label = trans(_('Faculty:'), lang)
[10250]162            f_label = Paragraph(f_label, ENTRY1_STYLE)
163            f_text = formatted_text(
164                studentview.context[
[10650]165                'studycourse'].certificate.__parent__.__parent__.__parent__.longtitle,
[10250]166                )
167            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]168            data_middle.append([f_label,f_text])
[10688]169        if not 'current_mode' in omit_fields:
170            studymodes_dict = getUtility(IKofaUtils).STUDY_MODES_DICT
[11535]171            sm = studymodes_dict[studentview.context.current_mode]
[10688]172            f_label = trans(_('Study Mode:'), lang)
173            f_label = Paragraph(f_label, ENTRY1_STYLE)
174            f_text = formatted_text(sm)
175            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]176            data_middle.append([f_label,f_text])
[10250]177        if not 'entry_session' in omit_fields:
[10261]178            f_label = trans(_('Entry Session:'), lang)
[10250]179            f_label = Paragraph(f_label, ENTRY1_STYLE)
[11535]180            entry_session = studentview.context.entry_session
[10250]181            entry_session = academic_sessions_vocab.getTerm(entry_session).title
182            f_text = formatted_text(entry_session)
183            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]184            data_middle.append([f_label,f_text])
[11535]185        # Requested by Uniben, does not really make sense
186        if not 'current_level' in omit_fields:
187            f_label = trans(_('Current Level:'), lang)
188            f_label = Paragraph(f_label, ENTRY1_STYLE)
189            current_level = studentview.context['studycourse'].current_level
190            studylevelsource = StudyLevelSource().factory
191            current_level = studylevelsource.getTitle(
192                studentview.context, current_level)
193            f_text = formatted_text(current_level)
194            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]195            data_middle.append([f_label,f_text])
[10256]196        if not 'date_of_birth' in omit_fields:
[10261]197            f_label = trans(_('Date of Birth:'), lang)
[10256]198            f_label = Paragraph(f_label, ENTRY1_STYLE)
199            date_of_birth = studentview.context.date_of_birth
200            tz = getUtility(IKofaUtils).tzinfo
201            date_of_birth = to_timezone(date_of_birth, tz)
202            if date_of_birth is not None:
203                date_of_birth = date_of_birth.strftime("%d/%m/%Y")
204            f_text = formatted_text(date_of_birth)
205            f_text = Paragraph(f_text, ENTRY1_STYLE)
[11550]206            data_middle.append([f_label,f_text])
[9141]207
[11550]208    # append QR code to the right
209    if slipname:
210        url = studentview.url(context, slipname)
211        data_right = [[get_qrcode(url, width=70.0)]]
212        table_right = Table(data_right,style=SLIP_STYLE)
213    else:
214        table_right = None
215
[7318]216    table_left = Table(data_left,style=SLIP_STYLE)
[11550]217    table_middle = Table(data_middle,style=SLIP_STYLE, colWidths=[5*cm, 5*cm])
218    table = Table([[table_left, table_middle, table_right],],style=SLIP_STYLE)
[7019]219    return table
220
[10261]221def render_table_data(tableheader, tabledata, lang='en'):
[7318]222    """Render children table for an existing frame.
223    """
[7304]224    data = []
[7318]225    #data.append([Spacer(1, 12)])
[7304]226    line = []
227    style = getSampleStyleSheet()
228    for element in tableheader:
[10261]229        field = '<strong>%s</strong>' % formatted_text(element[0], lang=lang)
[7310]230        field = Paragraph(field, style["Normal"])
[7304]231        line.append(field)
232    data.append(line)
233    for ticket in tabledata:
234        line = []
235        for element in tableheader:
[7511]236              field = formatted_text(getattr(ticket,element[1],u' '))
[7318]237              field = Paragraph(field, style["Normal"])
[7304]238              line.append(field)
239        data.append(line)
[7310]240    table = Table(data,colWidths=[
[7318]241        element[2]*cm for element in tableheader], style=CONTENT_STYLE)
[7304]242    return table
243
[10261]244def render_transcript_data(view, tableheader, levels_data, lang='en'):
[10250]245    """Render children table for an existing frame.
246    """
247    data = []
248    style = getSampleStyleSheet()
249    for level in levels_data:
250        level_obj = level['level']
[10251]251        tickets = level['tickets_1'] + level['tickets_2'] + level['tickets_3']
252        headerline = []
253        tabledata = []
[10261]254        subheader = '%s %s, %s %s' % (
255            trans(_('Session'), lang),
[10250]256            view.session_dict[level_obj.level_session],
[10261]257            trans(_('Level'), lang),
[10266]258            view.level_dict[level_obj.level])
[10250]259        data.append(Paragraph(subheader, HEADING_STYLE))
260        for element in tableheader:
261            field = '<strong>%s</strong>' % formatted_text(element[0])
262            field = Paragraph(field, style["Normal"])
[10251]263            headerline.append(field)
264        tabledata.append(headerline)
[10250]265        for ticket in tickets:
[10251]266            ticketline = []
[10250]267            for element in tableheader:
268                  field = formatted_text(getattr(ticket,element[1],u' '))
269                  field = Paragraph(field, style["Normal"])
[10251]270                  ticketline.append(field)
271            tabledata.append(ticketline)
[10250]272        table = Table(tabledata,colWidths=[
273            element[2]*cm for element in tableheader], style=CONTENT_STYLE)
274        data.append(table)
[10479]275        sgpa = '%s: %s' % (trans('Sessional GPA (rectified)', lang), level['sgpa'])
[10261]276        data.append(Paragraph(sgpa, style["Normal"]))
[10250]277    return data
278
[8112]279def docs_as_flowables(view, lang='en'):
280    """Create reportlab flowables out of scanned docs.
281    """
282    # XXX: fix circular import problem
283    from waeup.kofa.students.viewlets import FileManager
284    from waeup.kofa.browser import DEFAULT_IMAGE_PATH
285    style = getSampleStyleSheet()
286    data = []
[7318]287
[8112]288    # Collect viewlets
289    fm = FileManager(view.context, view.request, view)
290    fm.update()
291    if fm.viewlets:
292        sc_translation = trans(_('Scanned Documents'), lang)
[9910]293        data.append(Paragraph(sc_translation, HEADING_STYLE))
[8112]294        # Insert list of scanned documents
295        table_data = []
296        for viewlet in fm.viewlets:
[10020]297            if viewlet.file_exists:
298                # Show viewlet only if file exists
299                f_label = Paragraph(trans(viewlet.label, lang), ENTRY1_STYLE)
300                img_path = getattr(getUtility(IExtFileStore).getFileByContext(
301                    view.context, attr=viewlet.download_name), 'name', None)
302                #f_text = Paragraph(trans(_('(not provided)'),lang), ENTRY1_STYLE)
303                if img_path is None:
304                    pass
305                elif not img_path[-4:] in ('.jpg', '.JPG'):
306                    # reportlab requires jpg images, I think.
307                    f_text = Paragraph('%s (not displayable)' % (
308                        viewlet.title,), ENTRY1_STYLE)
309                else:
310                    f_text = Image(img_path, width=2*cm, height=1*cm, kind='bound')
311                table_data.append([f_label, f_text])
[8112]312        if table_data:
313            # safety belt; empty tables lead to problems.
314            data.append(Table(table_data, style=SLIP_STYLE))
315    return data
316
[7150]317class StudentsUtils(grok.GlobalUtility):
318    """A collection of methods subject to customization.
319    """
320    grok.implements(IStudentsUtils)
[7019]321
[8268]322    def getReturningData(self, student):
[9005]323        """ Define what happens after school fee payment
[7841]324        depending on the student's senate verdict.
325
326        In the base configuration current level is always increased
327        by 100 no matter which verdict has been assigned.
328        """
[8268]329        new_level = student['studycourse'].current_level + 100
330        new_session = student['studycourse'].current_session + 1
331        return new_session, new_level
332
333    def setReturningData(self, student):
[9005]334        """ Define what happens after school fee payment
335        depending on the student's senate verdict.
[8268]336
[9005]337        This method folllows the same algorithm as getReturningData but
338        it also sets the new values.
[8268]339        """
340        new_session, new_level = self.getReturningData(student)
[9922]341        try:
342            student['studycourse'].current_level = new_level
343        except ConstraintNotSatisfied:
344            # Do not change level if level exceeds the
345            # certificate's end_level.
346            pass
[8268]347        student['studycourse'].current_session = new_session
[7615]348        verdict = student['studycourse'].current_verdict
[8820]349        student['studycourse'].current_verdict = '0'
[7615]350        student['studycourse'].previous_verdict = verdict
351        return
352
[9519]353    def _getSessionConfiguration(self, session):
354        try:
355            return grok.getSite()['configuration'][str(session)]
356        except KeyError:
357            return None
358
[11451]359    def _isPaymentDisabled(self, p_session, category, student):
360        academic_session = self._getSessionConfiguration(p_session)
[11452]361        if category == 'schoolfee' and \
362            'sf_all' in academic_session.payment_disabled:
[11451]363            return True
364        return False
365
[9148]366    def setPaymentDetails(self, category, student,
[9151]367            previous_session, previous_level):
[8595]368        """Create Payment object and set the payment data of a student for
369        the payment category specified.
370
[7841]371        """
[8595]372        p_item = u''
373        amount = 0.0
[9148]374        if previous_session:
[9517]375            if previous_session < student['studycourse'].entry_session:
376                return _('The previous session must not fall below '
377                         'your entry session.'), None
378            if category == 'schoolfee':
379                # School fee is always paid for the following session
380                if previous_session > student['studycourse'].current_session:
381                    return _('This is not a previous session.'), None
382            else:
383                if previous_session > student['studycourse'].current_session - 1:
384                    return _('This is not a previous session.'), None
[9148]385            p_session = previous_session
386            p_level = previous_level
387            p_current = False
388        else:
389            p_session = student['studycourse'].current_session
390            p_level = student['studycourse'].current_level
391            p_current = True
[9519]392        academic_session = self._getSessionConfiguration(p_session)
393        if academic_session == None:
[8595]394            return _(u'Session configuration object is not available.'), None
[9521]395        # Determine fee.
[7150]396        if category == 'schoolfee':
[8595]397            try:
[8596]398                certificate = student['studycourse'].certificate
399                p_item = certificate.code
[8595]400            except (AttributeError, TypeError):
401                return _('Study course data are incomplete.'), None
[9148]402            if previous_session:
[9916]403                # Students can pay for previous sessions in all
404                # workflow states.  Fresh students are excluded by the
405                # update method of the PreviousPaymentAddFormPage.
[9148]406                if previous_level == 100:
407                    amount = getattr(certificate, 'school_fee_1', 0.0)
408                else:
409                    amount = getattr(certificate, 'school_fee_2', 0.0)
410            else:
411                if student.state == CLEARED:
412                    amount = getattr(certificate, 'school_fee_1', 0.0)
413                elif student.state == RETURNING:
[9916]414                    # In case of returning school fee payment the
415                    # payment session and level contain the values of
416                    # the session the student has paid for. Payment
417                    # session is always next session.
[9148]418                    p_session, p_level = self.getReturningData(student)
[9519]419                    academic_session = self._getSessionConfiguration(p_session)
420                    if academic_session == None:
[9916]421                        return _(
422                            u'Session configuration object is not available.'
423                            ), None
[9148]424                    amount = getattr(certificate, 'school_fee_2', 0.0)
425                elif student.is_postgrad and student.state == PAID:
[9916]426                    # Returning postgraduate students also pay for the
427                    # next session but their level always remains the
428                    # same.
[9148]429                    p_session += 1
[9519]430                    academic_session = self._getSessionConfiguration(p_session)
431                    if academic_session == None:
[9916]432                        return _(
433                            u'Session configuration object is not available.'
434                            ), None
[9148]435                    amount = getattr(certificate, 'school_fee_2', 0.0)
[7150]436        elif category == 'clearance':
[9178]437            try:
438                p_item = student['studycourse'].certificate.code
439            except (AttributeError, TypeError):
440                return _('Study course data are incomplete.'), None
[8595]441            amount = academic_session.clearance_fee
[7150]442        elif category == 'bed_allocation':
[8595]443            p_item = self.getAccommodationDetails(student)['bt']
444            amount = academic_session.booking_fee
[9423]445        elif category == 'hostel_maintenance':
[10681]446            amount = 0.0
[9429]447            bedticket = student['accommodation'].get(
448                str(student.current_session), None)
449            if bedticket:
450                p_item = bedticket.bed_coordinates
[10681]451                if bedticket.bed.__parent__.maint_fee > 0:
452                    amount = bedticket.bed.__parent__.maint_fee
453                else:
454                    # fallback
455                    amount = academic_session.maint_fee
[9429]456            else:
457                # Should not happen because this is already checked
458                # in the browser module, but anyway ...
459                portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
460                p_item = trans(_('no bed allocated'), portal_language)
[10449]461        elif category == 'transcript':
462            amount = academic_session.transcript_fee
[8595]463        if amount in (0.0, None):
[9517]464            return _('Amount could not be determined.'), None
[8595]465        for key in student['payments'].keys():
466            ticket = student['payments'][key]
467            if ticket.p_state == 'paid' and\
468               ticket.p_category == category and \
469               ticket.p_item == p_item and \
470               ticket.p_session == p_session:
[9517]471                  return _('This type of payment has already been made.'), None
[11451]472        if self._isPaymentDisabled(p_session, category, student):
473            return _('Payment temporarily disabled.'), None
[8708]474        payment = createObject(u'waeup.StudentOnlinePayment')
[8951]475        timestamp = ("%d" % int(time()*10000))[1:]
[8595]476        payment.p_id = "p%s" % timestamp
477        payment.p_category = category
478        payment.p_item = p_item
479        payment.p_session = p_session
480        payment.p_level = p_level
[9148]481        payment.p_current = p_current
[8595]482        payment.amount_auth = amount
483        return None, payment
[7019]484
[9868]485    def setBalanceDetails(self, category, student,
[9864]486            balance_session, balance_level, balance_amount):
487        """Create Payment object and set the payment data of a student for.
488
489        """
[9868]490        p_item = u'Balance'
[9864]491        p_session = balance_session
492        p_level = balance_level
493        p_current = False
494        amount = balance_amount
495        academic_session = self._getSessionConfiguration(p_session)
496        if academic_session == None:
497            return _(u'Session configuration object is not available.'), None
[9874]498        if amount in (0.0, None) or amount < 0:
499            return _('Amount must be greater than 0.'), None
[9864]500        for key in student['payments'].keys():
501            ticket = student['payments'][key]
502            if ticket.p_state == 'paid' and\
503               ticket.p_category == 'balance' and \
504               ticket.p_item == p_item and \
505               ticket.p_session == p_session:
506                  return _('This type of payment has already been made.'), None
507        payment = createObject(u'waeup.StudentOnlinePayment')
508        timestamp = ("%d" % int(time()*10000))[1:]
509        payment.p_id = "p%s" % timestamp
[9868]510        payment.p_category = category
[9864]511        payment.p_item = p_item
512        payment.p_session = p_session
513        payment.p_level = p_level
514        payment.p_current = p_current
515        payment.amount_auth = amount
516        return None, payment
517
[11595]518    def constructMatricNumber(self, student):
519        next_integer = grok.getSite()['configuration'].next_matric_integer
520        if next_integer == 0:
521            return None
[11589]522        return unicode(next_integer)
523
524    def setMatricNumber(self, student):
525        """Set matriculation number of student.
526
527        If the student's matric number is unset a new matric number is
528        constructed using the next_matric_integer attribute of
529        the site configuration container and according to the
530        matriculation number construction rules defined in the
[11595]531        constructMatricNumber method. The new matric number is set,
[11589]532        the students catalog updated and next_matric_integer
533        increased by one.
534
535        This method is tested but not used in the base package. It can
536        be used in custom packages by adding respective views
537        and by customizing _composeMatricNumber according to the
538        university's matriculation number construction rules.
539
540        The method can be disabled by setting next_matric_integer to zero.
541        """
542        if student.matric_number is not None:
543            return _('Matriculation number already set.'), None
[11590]544        if student.certcode is None:
545            return _('No certificate assigned.'), None
[11595]546        matric_number = self.constructMatricNumber(student)
[11592]547        if not matric_number:
548            return _('Matriculation number cannot be set.'), None
[11589]549        try:
[11592]550            student.matric_number = matric_number
[11589]551        except MatNumNotInSource:
552            return _('Matriculation number exists.'), None
553        notify(grok.ObjectModifiedEvent(student))
554        grok.getSite()['configuration'].next_matric_integer += 1
[11595]555        return None, matric_number
[11589]556
[7186]557    def getAccommodationDetails(self, student):
[9219]558        """Determine the accommodation data of a student.
[7841]559        """
[7150]560        d = {}
561        d['error'] = u''
[8685]562        hostels = grok.getSite()['hostels']
563        d['booking_session'] = hostels.accommodation_session
564        d['allowed_states'] = hostels.accommodation_states
[8688]565        d['startdate'] = hostels.startdate
566        d['enddate'] = hostels.enddate
567        d['expired'] = hostels.expired
[7150]568        # Determine bed type
569        studycourse = student['studycourse']
[7369]570        certificate = getattr(studycourse,'certificate',None)
[7150]571        entry_session = studycourse.entry_session
572        current_level = studycourse.current_level
[9187]573        if None in (entry_session, current_level, certificate):
574            return d
[7369]575        end_level = certificate.end_level
[9148]576        if current_level == 10:
577            bt = 'pr'
578        elif entry_session == grok.getSite()['hostels'].accommodation_session:
[7150]579            bt = 'fr'
580        elif current_level >= end_level:
581            bt = 'fi'
582        else:
583            bt = 're'
584        if student.sex == 'f':
585            sex = 'female'
586        else:
587            sex = 'male'
588        special_handling = 'regular'
589        d['bt'] = u'%s_%s_%s' % (special_handling,sex,bt)
590        return d
[7019]591
[7186]592    def selectBed(self, available_beds):
[7841]593        """Select a bed from a list of available beds.
594
595        In the base configuration we select the first bed found,
596        but can also randomize the selection if we like.
597        """
[7150]598        return available_beds[0]
599
[9981]600    def _admissionText(self, student, portal_language):
[9979]601        inst_name = grok.getSite()['configuration'].name
602        text = trans(_(
603            'This is to inform you that you have been provisionally'
604            ' admitted into ${a} as follows:', mapping = {'a': inst_name}),
605            portal_language)
606        return text
607
[10686]608    def renderPDFAdmissionLetter(self, view, student=None, omit_fields=(),
609                                 pre_text=None, post_text=None,):
[9191]610        """Render pdf admission letter.
611        """
612        if student is None:
613            return
614        style = getSampleStyleSheet()
[9949]615        creator = self.getPDFCreator(student)
[9979]616        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[9191]617        data = []
618        doc_title = view.label
619        author = '%s (%s)' % (view.request.principal.title,
620                              view.request.principal.id)
[9944]621        footer_text = view.label.split('\n')
622        if len(footer_text) > 1:
623            # We can add a department in first line
624            footer_text = footer_text[1]
625        else:
626            # Only the first line is used for the footer
627            footer_text = footer_text[0]
[9191]628        if getattr(student, 'student_id', None) is not None:
629            footer_text = "%s - %s - " % (student.student_id, footer_text)
630
[10702]631        # Text before student data
[10686]632        if pre_text is None:
633            html = format_html(self._admissionText(student, portal_language))
634        else:
635            html = format_html(pre_text)
[9191]636        data.append(Paragraph(html, NOTE_STYLE))
637        data.append(Spacer(1, 20))
638
639        # Student data
[11550]640        data.append(render_student_data(view, student,
641                    omit_fields, lang=portal_language,
642                    slipname='admission_slip.pdf'))
[9191]643
[10702]644        # Text after student data
[9191]645        data.append(Spacer(1, 20))
[10686]646        if post_text is None:
647            datelist = student.history.messages[0].split()[0].split('-')
648            creation_date = u'%s/%s/%s' % (datelist[2], datelist[1], datelist[0])
[10702]649            post_text = trans(_(
[10686]650                'Your Kofa student record was created on ${a}.',
651                mapping = {'a': creation_date}),
652                portal_language)
[10702]653        #html = format_html(post_text)
654        #data.append(Paragraph(html, NOTE_STYLE))
[9191]655
656        # Create pdf stream
657        view.response.setHeader(
658            'Content-Type', 'application/pdf')
659        pdf_stream = creator.create_pdf(
660            data, None, doc_title, author=author, footer=footer_text,
[10702]661            note=post_text)
[9191]662        return pdf_stream
663
[9949]664    def getPDFCreator(self, context):
665        """Get a pdf creator suitable for `context`.
666
667        The default implementation always returns the default creator.
668        """
669        return getUtility(IPDFCreator)
670
[8257]671    def renderPDF(self, view, filename='slip.pdf', student=None,
[9906]672                  studentview=None,
[10439]673                  tableheader=[], tabledata=[],
[9555]674                  note=None, signatures=None, sigs_in_footer=(),
[10250]675                  show_scans=True, topMargin=1.5,
676                  omit_fields=()):
[7841]677        """Render pdf slips for various pages.
678        """
[10261]679        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[9916]680        # XXX: tell what the different parameters mean
[8112]681        style = getSampleStyleSheet()
[9949]682        creator = self.getPDFCreator(student)
[8112]683        data = []
684        doc_title = view.label
685        author = '%s (%s)' % (view.request.principal.title,
686                              view.request.principal.id)
[9913]687        footer_text = view.label.split('\n')
688        if len(footer_text) > 2:
689            # We can add a department in first line
690            footer_text = footer_text[1]
691        else:
[9917]692            # Only the first line is used for the footer
[9913]693            footer_text = footer_text[0]
[7714]694        if getattr(student, 'student_id', None) is not None:
[7310]695            footer_text = "%s - %s - " % (student.student_id, footer_text)
[7150]696
[7318]697        # Insert student data table
[7310]698        if student is not None:
[8112]699            bd_translation = trans(_('Base Data'), portal_language)
[9910]700            data.append(Paragraph(bd_translation, HEADING_STYLE))
[10261]701            data.append(render_student_data(
[11550]702                studentview, view.context, omit_fields, lang=portal_language,
703                slipname=filename))
[7304]704
[7318]705        # Insert widgets
[9191]706        if view.form_fields:
[9910]707            data.append(Paragraph(view.title, HEADING_STYLE))
[9191]708            separators = getattr(self, 'SEPARATORS_DICT', {})
709            table = creator.getWidgetsTable(
710                view.form_fields, view.context, None, lang=portal_language,
711                separators=separators)
712            data.append(table)
[7318]713
[8112]714        # Insert scanned docs
[9550]715        if show_scans:
716            data.extend(docs_as_flowables(view, portal_language))
[7318]717
[9452]718        # Insert history
[9910]719        if filename.startswith('clearance'):
[9452]720            hist_translation = trans(_('Workflow History'), portal_language)
[9910]721            data.append(Paragraph(hist_translation, HEADING_STYLE))
[9452]722            data.extend(creator.fromStringList(student.history.messages))
723
[10438]724        # Insert content tables (optionally on second page)
[10439]725        if hasattr(view, 'tabletitle'):
726            for i in range(len(view.tabletitle)):
727                if tabledata[i] and tableheader[i]:
728                    #data.append(PageBreak())
729                    #data.append(Spacer(1, 20))
730                    data.append(Paragraph(view.tabletitle[i], HEADING_STYLE))
731                    data.append(Spacer(1, 8))
732                    contenttable = render_table_data(tableheader[i],tabledata[i])
733                    data.append(contenttable)
[7318]734
[9010]735        # Insert signatures
[9965]736        # XXX: We are using only sigs_in_footer in waeup.kofa, so we
737        # do not have a test for the following lines.
[9555]738        if signatures and not sigs_in_footer:
[9010]739            data.append(Spacer(1, 20))
[9966]740            # Render one signature table per signature to
741            # get date and signature in line.
742            for signature in signatures:
743                signaturetables = get_signature_tables(signature)
744                data.append(signaturetables[0])
[9010]745
[7150]746        view.response.setHeader(
747            'Content-Type', 'application/pdf')
[8112]748        try:
749            pdf_stream = creator.create_pdf(
[8257]750                data, None, doc_title, author=author, footer=footer_text,
[9948]751                note=note, sigs_in_footer=sigs_in_footer, topMargin=topMargin)
[8112]752        except IOError:
753            view.flash('Error in image file.')
754            return view.redirect(view.url(view.context))
755        return pdf_stream
[7620]756
[10578]757    gpa_boundaries = ((1, 'Fail'),
758                      (1.5, 'Pass'),
759                      (2.4, '3rd Class'),
760                      (3.5, '2nd Class Lower'),
761                      (4.5, '2nd Class Upper'),
762                      (5, '1st Class'))
[10576]763
[10445]764    def getClassFromCGPA(self, gpa):
[10578]765        if gpa < self.gpa_boundaries[0][0]:
766            return 0, self.gpa_boundaries[0][1]
767        if gpa < self.gpa_boundaries[1][0]:
768            return 1, self.gpa_boundaries[1][1]
769        if gpa < self.gpa_boundaries[2][0]:
770            return 2, self.gpa_boundaries[2][1]
771        if gpa < self.gpa_boundaries[3][0]:
772            return 3, self.gpa_boundaries[3][1]
773        if gpa < self.gpa_boundaries[4][0]:
774            return 4, self.gpa_boundaries[4][1]
775        if gpa <= self.gpa_boundaries[5][0]:
776            return 5, self.gpa_boundaries[5][1]
777        return 'N/A'
[10445]778
[10250]779    def renderPDFTranscript(self, view, filename='transcript.pdf',
780                  student=None,
781                  studentview=None,
782                  note=None, signatures=None, sigs_in_footer=(),
783                  show_scans=True, topMargin=1.5,
784                  omit_fields=(),
785                  tableheader=None):
786        """Render pdf slips for transcript.
787        """
[10261]788        portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
[10250]789        # XXX: tell what the different parameters mean
790        style = getSampleStyleSheet()
791        creator = self.getPDFCreator(student)
792        data = []
793        doc_title = view.label
794        author = '%s (%s)' % (view.request.principal.title,
795                              view.request.principal.id)
796        footer_text = view.label.split('\n')
797        if len(footer_text) > 2:
798            # We can add a department in first line
799            footer_text = footer_text[1]
800        else:
801            # Only the first line is used for the footer
802            footer_text = footer_text[0]
803        if getattr(student, 'student_id', None) is not None:
804            footer_text = "%s - %s - " % (student.student_id, footer_text)
805
806        # Insert student data table
807        if student is not None:
808            #bd_translation = trans(_('Base Data'), portal_language)
809            #data.append(Paragraph(bd_translation, HEADING_STYLE))
[10261]810            data.append(render_student_data(
[11550]811                studentview, view.context,
812                omit_fields, lang=portal_language,
813                slipname=filename))
[10250]814
815        transcript_data = view.context.getTranscriptData()
816        levels_data = transcript_data[0]
817        gpa = transcript_data[1]
818
819        contextdata = []
[10261]820        f_label = trans(_('Course of Study:'), portal_language)
[10250]821        f_label = Paragraph(f_label, ENTRY1_STYLE)
[10650]822        f_text = formatted_text(view.context.certificate.longtitle)
[10250]823        f_text = Paragraph(f_text, ENTRY1_STYLE)
824        contextdata.append([f_label,f_text])
825
[10261]826        f_label = trans(_('Faculty:'), portal_language)
[10250]827        f_label = Paragraph(f_label, ENTRY1_STYLE)
828        f_text = formatted_text(
[10650]829            view.context.certificate.__parent__.__parent__.__parent__.longtitle)
[10250]830        f_text = Paragraph(f_text, ENTRY1_STYLE)
831        contextdata.append([f_label,f_text])
832
[10261]833        f_label = trans(_('Department:'), portal_language)
[10250]834        f_label = Paragraph(f_label, ENTRY1_STYLE)
835        f_text = formatted_text(
[10650]836            view.context.certificate.__parent__.__parent__.longtitle)
[10250]837        f_text = Paragraph(f_text, ENTRY1_STYLE)
838        contextdata.append([f_label,f_text])
839
[10261]840        f_label = trans(_('Entry Session:'), portal_language)
[10250]841        f_label = Paragraph(f_label, ENTRY1_STYLE)
[10256]842        f_text = formatted_text(
843            view.session_dict.get(view.context.entry_session))
[10250]844        f_text = Paragraph(f_text, ENTRY1_STYLE)
845        contextdata.append([f_label,f_text])
846
[10261]847        f_label = trans(_('Entry Mode:'), portal_language)
[10250]848        f_label = Paragraph(f_label, ENTRY1_STYLE)
[10256]849        f_text = formatted_text(view.studymode_dict.get(
850            view.context.entry_mode))
[10250]851        f_text = Paragraph(f_text, ENTRY1_STYLE)
852        contextdata.append([f_label,f_text])
853
[10262]854        f_label = trans(_('Cumulative GPA:'), portal_language)
[10250]855        f_label = Paragraph(f_label, ENTRY1_STYLE)
[10576]856        f_text = formatted_text('%s (%s)' % (gpa, self.getClassFromCGPA(gpa)[1]))
[10250]857        f_text = Paragraph(f_text, ENTRY1_STYLE)
858        contextdata.append([f_label,f_text])
859
860        contexttable = Table(contextdata,style=SLIP_STYLE)
861        data.append(contexttable)
862
863        transcripttables = render_transcript_data(
[10261]864            view, tableheader, levels_data, lang=portal_language)
[10250]865        data.extend(transcripttables)
866
867        # Insert signatures
868        # XXX: We are using only sigs_in_footer in waeup.kofa, so we
869        # do not have a test for the following lines.
870        if signatures and not sigs_in_footer:
871            data.append(Spacer(1, 20))
872            # Render one signature table per signature to
873            # get date and signature in line.
874            for signature in signatures:
875                signaturetables = get_signature_tables(signature)
876                data.append(signaturetables[0])
877
878        view.response.setHeader(
879            'Content-Type', 'application/pdf')
880        try:
881            pdf_stream = creator.create_pdf(
882                data, None, doc_title, author=author, footer=footer_text,
883                note=note, sigs_in_footer=sigs_in_footer, topMargin=topMargin)
884        except IOError:
[10261]885            view.flash(_('Error in image file.'))
[10250]886            return view.redirect(view.url(view.context))
887        return pdf_stream
888
[9830]889    def maxCredits(self, studylevel):
890        """Return maximum credits.
[9532]891
[9830]892        In some universities maximum credits is not constant, it
893        depends on the student's study level.
894        """
895        return 50
896
[9532]897    def maxCreditsExceeded(self, studylevel, course):
[9830]898        max_credits = self.maxCredits(studylevel)
899        if max_credits and \
900            studylevel.total_credits + course.credits > max_credits:
901            return max_credits
[9532]902        return 0
903
[9987]904    def getBedCoordinates(self, bedticket):
905        """Return bed coordinates.
906
907        This method can be used to customize the display_coordinates
908        property method.
909        """
910        return bedticket.bed_coordinates
911
[7841]912    VERDICTS_DICT = {
[8820]913        '0': _('(not yet)'),
[7841]914        'A': 'Successful student',
915        'B': 'Student with carryover courses',
916        'C': 'Student on probation',
917        }
[8099]918
919    SEPARATORS_DICT = {
920        }
[8410]921
[10021]922    SKIP_UPLOAD_VIEWLETS = ()
923
[10803]924    PWCHANGE_STATES = (ADMITTED,)
[10706]925
[8410]926    #: A prefix used when generating new student ids. Each student id will
927    #: start with this string. The default is 'K' for ``Kofa``.
928    STUDENT_ID_PREFIX = u'K'
Note: See TracBrowser for help on using the repository browser.