source: main/waeup.sirp/trunk/src/waeup/sirp/students/utils.py @ 7487

Last change on this file since 7487 was 7369, checked in by Henrik Bettermann, 13 years ago

Implement ChangePasswordRequestPage?.

Catch traceback, if student data are not complete and a test student presses the 'Book accommodation' button (regression test needed).

  • Property svn:keywords set to Id
File size: 12.6 KB
RevLine 
[7191]1## $Id: utils.py 7369 2011-12-18 08:24:04Z 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
[6662]21from random import SystemRandom as r
[7256]22from datetime import datetime
[7019]23from reportlab.pdfgen import canvas
[7318]24from reportlab.lib import colors
[7019]25from reportlab.lib.units import cm
[7310]26from reportlab.lib.enums import TA_RIGHT
[7019]27from reportlab.lib.pagesizes import A4
[7310]28from reportlab.lib.styles import getSampleStyleSheet, ParagraphStyle
[7019]29from reportlab.platypus import (Frame, Paragraph, Image,
30    Table, Spacer)
31from reportlab.platypus.tables import TableStyle
[7310]32from reportlab.platypus.flowables import PageBreak
[7280]33from zope.component import getUtility
[7019]34from zope.formlib.form import setUpEditWidgets
[7280]35from waeup.sirp.interfaces import IExtFileStore
[7150]36from waeup.sirp.students.interfaces import IStudentsUtils
[6651]37
[7318]38SLIP_STYLE = [
39    ('VALIGN',(0,0),(-1,-1),'TOP'),
40    #('FONT', (0,0), (-1,-1), 'Helvetica', 11),
41    ]
[7019]42
[7318]43CONTENT_STYLE = [
44    ('VALIGN',(0,0),(-1,-1),'TOP'),
45    #('FONT', (0,0), (-1,-1), 'Helvetica', 8),
46    #('TEXTCOLOR',(0,0),(-1,0),colors.white),
47    ('BACKGROUND',(0,0),(-1,0),colors.black),
48    ]
[7304]49
[7318]50FONT_SIZE = 10
51FONT_COLOR = 'black'
52
[7319]53def formatted_label(color=FONT_COLOR, size=FONT_SIZE):
[7318]54    tag1 ='<font color=%s size=%d>' % (color, size)
[7319]55    return tag1 + '%s:</font>'
56
57def formatted_text(color=FONT_COLOR, size=FONT_SIZE):
58    tag1 ='<font color=%s size=%d>' % (color, size)
[7318]59    return tag1 + '%s</font>'
60
[6749]61def generate_student_id(students,letter):
[6651]62    if letter == '?':
[6664]63        letter= r().choice('ABCDEFGHKLMNPQRSTUVWXY')
64    sid = u"%c%d" % (letter,r().randint(99999,1000000))
[6749]65    while sid in students.keys():
[6664]66        sid = u"%c%d" % (letter,r().randint(99999,1000000))
[6662]67    return sid
[6742]68
69def set_returning_data(student):
70    student['studycourse'].current_level += 100
71    student['studycourse'].current_session += 1
72    verdict = student['studycourse'].current_verdict
[6804]73    student['studycourse'].current_verdict = '0'
[6742]74    student['studycourse'].previous_verdict = verdict
75    return
[6876]76
[7186]77def set_up_widgets(view, ignore_request=False):
[7019]78    view.adapters = {}
79    view.widgets = setUpEditWidgets(
80        view.form_fields, view.prefix, view.context, view.request,
81        adapters=view.adapters, for_display=True,
82        ignore_request=ignore_request
83        )
84
[7310]85def render_student_data(studentview):
[7318]86    """Render student table for an existing frame.
87    """
88    width, height = A4
[7186]89    set_up_widgets(studentview, ignore_request=True)
[7318]90    data_left = []
91    data_right = []
[7019]92    style = getSampleStyleSheet()
[7280]93    img = getUtility(IExtFileStore).getFileByContext(
94        studentview.context, attr='passport.jpg')
95    if img is None:
96        from waeup.sirp.browser import DEFAULT_PASSPORT_IMAGE_PATH
97        img = open(DEFAULT_PASSPORT_IMAGE_PATH, 'rb')
[7318]98    doc_img = Image(img.name, width=4*cm, height=4*cm, kind='bound')
99    data_left.append([doc_img])
100    #data.append([Spacer(1, 12)])
[7019]101    for widget in studentview.widgets:
102        if widget.name == 'form.adm_code':
103            continue
[7319]104        f_label = formatted_label() % widget.label.strip()
[7019]105        f_label = Paragraph(f_label, style["Normal"])
[7319]106        f_text = formatted_text() % widget()
[7019]107        f_text = Paragraph(f_text, style["Normal"])
[7318]108        data_right.append([f_label,f_text])
109    table_left = Table(data_left,style=SLIP_STYLE)
110    table_right = Table(data_right,style=SLIP_STYLE, colWidths=[4*cm, 10*cm])
111    table = Table([[table_left, table_right],],style=SLIP_STYLE)
[7019]112    return table
113
[7304]114def render_table_data(tableheader,tabledata):
[7318]115    """Render children table for an existing frame.
116    """
[7304]117    data = []
[7318]118    #data.append([Spacer(1, 12)])
[7304]119    line = []
120    style = getSampleStyleSheet()
121    for element in tableheader:
[7319]122        field = formatted_text(color='white') % element[0]
[7310]123        field = Paragraph(field, style["Normal"])
[7304]124        line.append(field)
125    data.append(line)
126    for ticket in tabledata:
127        line = []
128        for element in tableheader:
[7319]129              field = formatted_text() % str(getattr(ticket,element[1],u' '))
[7318]130              field = Paragraph(field, style["Normal"])
[7304]131              line.append(field)
132        data.append(line)
[7310]133    table = Table(data,colWidths=[
[7318]134        element[2]*cm for element in tableheader], style=CONTENT_STYLE)
[7304]135    return table
136
[7318]137
138def insert_footer(pdf,width,style,text=None, number_of_pages=1):
139      """Render the whole footer frame.
140      """
141      story = []
142      frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
143      timestamp = datetime.now().strftime("%d/%m/%Y %H:%M:%S")
144      left_text = '<font size=10>%s</font>' % timestamp
145      story.append(Paragraph(left_text, style["Normal"]))
146      frame_footer.addFromList(story,pdf)
147      story = []
148      frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
149      right_text = '<font size=10>%s Page %s of %s</font>' % (
150          text, pdf.getPageNumber(), number_of_pages)
151      story.append(Paragraph(right_text, style["Right"]))
152      frame_footer.addFromList(story,pdf)
153
[7150]154class StudentsUtils(grok.GlobalUtility):
155    """A collection of methods subject to customization.
156    """
157    grok.implements(IStudentsUtils)
[7019]158
[7186]159    def getPaymentDetails(self,category, student):
[7150]160        d = {}
161        d['p_item'] = u''
162        d['amount'] = 0
163        d['error'] = u''
164        d['p_session'] = student['studycourse'].current_session
165        session = str(d['p_session'])
166        try:
167            academic_session = grok.getSite()['configuration'][session]
168        except KeyError:
169            d['error'] = u'Session configuration object is not available.'
170            return d
171        d['surcharge_1'] = academic_session.surcharge_1
172        d['surcharge_2'] = academic_session.surcharge_2
173        d['surcharge_3'] = academic_session.surcharge_3
174        if category == 'schoolfee':
175            d['amount'] = academic_session.school_fee_base
176            d['p_item'] = student['studycourse'].certificate.code
177        elif category == 'clearance':
178            d['p_item'] = student['studycourse'].certificate.code
179            d['amount'] = academic_session.clearance_fee
180        elif category == 'bed_allocation':
[7186]181            d['p_item'] = self.getAccommodationDetails(student)['bt']
[7150]182            d['amount'] = academic_session.booking_fee
183        return d
[7019]184
[7186]185    def getAccommodationDetails(self, student):
[7150]186        d = {}
187        d['error'] = u''
[7365]188        site_configuration = grok.getSite()['configuration']
189        d['booking_session'] = site_configuration.accommodation_session
190        d['allowed_states'] = site_configuration.accommodation_states
[7150]191        # Determine bed type
192        studycourse = student['studycourse']
[7369]193        certificate = getattr(studycourse,'certificate',None)
[7150]194        entry_session = studycourse.entry_session
195        current_level = studycourse.current_level
[7369]196        if not (entry_session and current_level and certificate):
197            return
198        end_level = certificate.end_level
[7150]199        if entry_session == grok.getSite()['configuration'].accommodation_session:
200            bt = 'fr'
201        elif current_level >= end_level:
202            bt = 'fi'
203        else:
204            bt = 're'
205        if student.sex == 'f':
206            sex = 'female'
207        else:
208            sex = 'male'
209        special_handling = 'regular'
210        d['bt'] = u'%s_%s_%s' % (special_handling,sex,bt)
211        return d
[7019]212
[7150]213    # In the standard configuration we select the first bed found,
214    # but can also randomize the selection if we like.
[7186]215    def selectBed(self, available_beds):
[7150]216        return available_beds[0]
217
[7318]218    def renderPDF(self, view, filename='slip.pdf',
[7304]219        student=None, studentview=None, tableheader=None, tabledata=None):
[7150]220        # (0,0),(-1,-1) = whole table
221        # (0,0),(0,-1) = first column
222        # (-1,0),(-1,-1) = last column
223        # (0,0),(-1,0) = first row
224        # (0,-1),(-1,-1) = last row
225
226        pdf = canvas.Canvas(filename,pagesize=A4)
227        pdf.setTitle(view.label)
[7318]228        pdf.setSubject(view.title)
[7150]229        pdf.setAuthor('%s (%s)' % (view.request.principal.title,
230            view.request.principal.id))
231        pdf.setCreator('WAeUP SIRP')
232        width, height = A4
[7310]233        footer_text = view.label
234        if student is not None:
235            footer_text = "%s - %s - " % (student.student_id, footer_text)
[7150]236        style = getSampleStyleSheet()
[7310]237        style.add(ParagraphStyle(name='Right', alignment=TA_RIGHT))
[7150]238        pdf.line(1*cm,height-(1.8*cm),width-(1*cm),height-(1.8*cm))
239
240        story = []
241        frame_header = Frame(1*cm,1*cm,width-(1.7*cm),height-(1.7*cm))
242        header_title = getattr(grok.getSite(), 'name', u'Sample University')
243        story.append(Paragraph(header_title, style["Heading1"]))
244        frame_header.addFromList(story,pdf)
245
[7318]246        # Insert student data table
[7150]247        story = []
248        frame_body = Frame(1*cm,1*cm,width-(2*cm),height-(3.5*cm))
249        story.append(Paragraph(view.label, style["Heading2"]))
[7310]250        if student is not None:
[7318]251            #story.append(Spacer(1, 12))
252            story.append(Paragraph('Student Base Data', style["Heading3"]))
[7310]253            studenttable = render_student_data(studentview)
[7150]254            story.append(studenttable)
[7304]255
[7318]256        # Insert widgets
257        story.append(Paragraph(view.title, style["Heading3"]))
[7186]258        set_up_widgets(view)
[7150]259        data = []
260        for widget in view.widgets:
[7319]261            f_label = formatted_label() % widget.label.strip()
[7150]262            f_label = Paragraph(f_label, style["Normal"])
[7319]263            f_text = formatted_text() % str(widget())
[7150]264            f_text = Paragraph(f_text, style["Normal"])
265            data.append([f_label,f_text])
[7318]266        table = Table(data,style=SLIP_STYLE, colWidths=[5*cm, 13.5*cm])
267        story.append(table)
268
269        # Import browser components locally
[7280]270        # to avoid circular import
271        from waeup.sirp.students.viewlets import FileManager
272        from waeup.sirp.browser import DEFAULT_IMAGE_PATH
[7318]273        data = []
274        # Collect viewlets
[7280]275        fm = FileManager(view.context, view.request, view)
276        fm.update()
[7318]277        if fm.viewlets:
278            story.append(Paragraph('Scanned Documents', style["Heading3"]))
279            # Insert list of scanned documents
280            for viewlet in fm.viewlets:
[7319]281                f_label = formatted_label() % viewlet.label
[7318]282                f_label = Paragraph(f_label, style["Normal"])
283                if viewlet.template.__grok_name__ == 'imagedisplay':
284                    # In case we define FileDisplay viewlets with
285                    # an imagedisplay template in the customization package
286                    img = getUtility(IExtFileStore).getFileByContext(
287                        view.context, attr=viewlet.download_name)
288                    if img is None:
289                        img = open(DEFAULT_IMAGE_PATH, 'rb')
290                    doc_img = Image(img.name, width=2*cm, height=1*cm, kind='bound')
291                    data.append([f_label,doc_img])
292                else:
[7319]293                    f_text = formatted_text() % viewlet.title
[7318]294                    f_text = Paragraph(f_text, style["Normal"])
295                    data.append([f_label,f_text])
296            table = Table(data,style=SLIP_STYLE, colWidths=[5*cm, 13.5*cm])
297            story.append(table)
298
[7280]299        try:
300            frame_body.addFromList(story,pdf)
301        except IOError:
302            view.flash('Error in image file.')
303            return view.redirect(view.url(view.context))
[7318]304
[7304]305        if tabledata and tableheader:
[7318]306            insert_footer(pdf,width,style,footer_text,number_of_pages=2)
307        else:
308            insert_footer(pdf,width,style,footer_text)
309
310        # Insert content table on second page
311        if tabledata and tableheader:
312            pdf.showPage()
313            frame_body = Frame(1*cm,1*cm,width-(2*cm),height-(2*cm))
[7304]314            story = []
[7318]315            story.append(Paragraph(view.content_title, style["Heading3"]))
[7310]316            #story.append(Spacer(1, 12))
[7304]317            contenttable = render_table_data(tableheader,tabledata)
318            story.append(contenttable)
319            frame_body.addFromList(story,pdf)
[7318]320            insert_footer(pdf,width,style,footer_text,number_of_pages=2)
321
[7150]322        view.response.setHeader(
323            'Content-Type', 'application/pdf')
324        return pdf.getpdfdata()
Note: See TracBrowser for help on using the repository browser.