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

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

Insert a br tag if widgets contain div tags which are not supported by reportlab.

  • Property svn:keywords set to Id
File size: 15.6 KB
RevLine 
[7191]1## $Id: utils.py 7804 2012-03-08 16:30:34Z 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
[7714]23from zope.i18n import translate
24from zope.component import getUtility
[7019]25from reportlab.pdfgen import canvas
[7318]26from reportlab.lib import colors
[7019]27from reportlab.lib.units import cm
[7310]28from reportlab.lib.enums import TA_RIGHT
[7019]29from reportlab.lib.pagesizes import A4
[7310]30from reportlab.lib.styles import getSampleStyleSheet, ParagraphStyle
[7019]31from reportlab.platypus import (Frame, Paragraph, Image,
32    Table, Spacer)
33from reportlab.platypus.tables import TableStyle
[7310]34from reportlab.platypus.flowables import PageBreak
[7280]35from zope.component import getUtility
[7019]36from zope.formlib.form import setUpEditWidgets
[7714]37from waeup.sirp.interfaces import IExtFileStore, ISIRPUtils
38from waeup.sirp.interfaces import MessageFactory as _
[7150]39from waeup.sirp.students.interfaces import IStudentsUtils
[6651]40
[7318]41SLIP_STYLE = [
42    ('VALIGN',(0,0),(-1,-1),'TOP'),
43    #('FONT', (0,0), (-1,-1), 'Helvetica', 11),
44    ]
[7019]45
[7318]46CONTENT_STYLE = [
47    ('VALIGN',(0,0),(-1,-1),'TOP'),
48    #('FONT', (0,0), (-1,-1), 'Helvetica', 8),
49    #('TEXTCOLOR',(0,0),(-1,0),colors.white),
50    ('BACKGROUND',(0,0),(-1,0),colors.black),
51    ]
[7304]52
[7318]53FONT_SIZE = 10
54FONT_COLOR = 'black'
55
[7319]56def formatted_label(color=FONT_COLOR, size=FONT_SIZE):
[7318]57    tag1 ='<font color=%s size=%d>' % (color, size)
[7319]58    return tag1 + '%s:</font>'
59
[7511]60def formatted_text(text, color=FONT_COLOR, size=FONT_SIZE):
61    """Turn `text`, `color` and `size` into an HTML snippet.
[7318]62
[7511]63    The snippet is suitable for use with reportlab and generating PDFs.
64    Wraps the `text` into a ``<font>`` tag with passed attributes.
65
66    Also non-strings are converted. Raw strings are expected to be
67    utf-8 encoded (usually the case for widgets etc.).
68
[7804]69    Finally, a br tag is added if widgets contain div tags
70    which are not supported by reportlab.
71
[7511]72    The returned snippet is unicode type.
73    """
74    if not isinstance(text, unicode):
75        if isinstance(text, basestring):
76            text = text.decode('utf-8')
77        else:
78            text = unicode(text)
[7804]79    text = text.replace('</div>', '<br /></div>')
[7511]80    tag1 = u'<font color="%s" size="%d">' % (color, size)
81    return tag1 + u'%s</font>' % text
82
[6749]83def generate_student_id(students,letter):
[6651]84    if letter == '?':
[6664]85        letter= r().choice('ABCDEFGHKLMNPQRSTUVWXY')
86    sid = u"%c%d" % (letter,r().randint(99999,1000000))
[6749]87    while sid in students.keys():
[6664]88        sid = u"%c%d" % (letter,r().randint(99999,1000000))
[6662]89    return sid
[6742]90
[7186]91def set_up_widgets(view, ignore_request=False):
[7019]92    view.adapters = {}
93    view.widgets = setUpEditWidgets(
94        view.form_fields, view.prefix, view.context, view.request,
95        adapters=view.adapters, for_display=True,
96        ignore_request=ignore_request
97        )
98
[7310]99def render_student_data(studentview):
[7318]100    """Render student table for an existing frame.
101    """
102    width, height = A4
[7186]103    set_up_widgets(studentview, ignore_request=True)
[7318]104    data_left = []
105    data_right = []
[7019]106    style = getSampleStyleSheet()
[7280]107    img = getUtility(IExtFileStore).getFileByContext(
108        studentview.context, attr='passport.jpg')
109    if img is None:
110        from waeup.sirp.browser import DEFAULT_PASSPORT_IMAGE_PATH
111        img = open(DEFAULT_PASSPORT_IMAGE_PATH, 'rb')
[7318]112    doc_img = Image(img.name, width=4*cm, height=4*cm, kind='bound')
113    data_left.append([doc_img])
114    #data.append([Spacer(1, 12)])
[7714]115    portal_language = getUtility(ISIRPUtils).PORTAL_LANGUAGE
[7019]116    for widget in studentview.widgets:
117        if widget.name == 'form.adm_code':
118            continue
[7714]119        f_label = formatted_label(size=12) % translate(
120            widget.label.strip(), 'waeup.sirp',
121            target_language=portal_language)
[7019]122        f_label = Paragraph(f_label, style["Normal"])
[7714]123        f_text = formatted_text(widget(), size=12)
[7019]124        f_text = Paragraph(f_text, style["Normal"])
[7318]125        data_right.append([f_label,f_text])
126    table_left = Table(data_left,style=SLIP_STYLE)
[7741]127    table_right = Table(data_right,style=SLIP_STYLE, colWidths=[6*cm, 8*cm])
[7318]128    table = Table([[table_left, table_right],],style=SLIP_STYLE)
[7019]129    return table
130
[7304]131def render_table_data(tableheader,tabledata):
[7318]132    """Render children table for an existing frame.
133    """
[7304]134    data = []
[7318]135    #data.append([Spacer(1, 12)])
[7304]136    line = []
137    style = getSampleStyleSheet()
138    for element in tableheader:
[7511]139        field = formatted_text(element[0], color='white')
[7310]140        field = Paragraph(field, style["Normal"])
[7304]141        line.append(field)
142    data.append(line)
143    for ticket in tabledata:
144        line = []
145        for element in tableheader:
[7511]146              field = formatted_text(getattr(ticket,element[1],u' '))
[7318]147              field = Paragraph(field, style["Normal"])
[7304]148              line.append(field)
149        data.append(line)
[7310]150    table = Table(data,colWidths=[
[7318]151        element[2]*cm for element in tableheader], style=CONTENT_STYLE)
[7304]152    return table
153
[7318]154
155def insert_footer(pdf,width,style,text=None, number_of_pages=1):
156      """Render the whole footer frame.
157      """
158      story = []
159      frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
160      timestamp = datetime.now().strftime("%d/%m/%Y %H:%M:%S")
161      left_text = '<font size=10>%s</font>' % timestamp
162      story.append(Paragraph(left_text, style["Normal"]))
163      frame_footer.addFromList(story,pdf)
164      story = []
165      frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
[7714]166      portal_language = getUtility(ISIRPUtils).PORTAL_LANGUAGE
167      right_text = translate(_('<font size=10>${a} Page ${b} of ${c}</font>',
168          mapping = {'a':text, 'b':pdf.getPageNumber(), 'c':number_of_pages}),
169          'waeup.sirp', target_language=portal_language)
[7318]170      story.append(Paragraph(right_text, style["Right"]))
171      frame_footer.addFromList(story,pdf)
172
[7150]173class StudentsUtils(grok.GlobalUtility):
174    """A collection of methods subject to customization.
175    """
176    grok.implements(IStudentsUtils)
[7019]177
[7615]178    def setReturningData(self, student):
179        student['studycourse'].current_level += 100
180        student['studycourse'].current_session += 1
181        verdict = student['studycourse'].current_verdict
182        student['studycourse'].current_verdict = '0'
183        student['studycourse'].previous_verdict = verdict
184        return
185
186    def getPaymentDetails(self, category, student):
[7150]187        d = {}
188        d['p_item'] = u''
189        d['amount'] = 0
190        d['error'] = u''
191        d['p_session'] = student['studycourse'].current_session
192        session = str(d['p_session'])
193        try:
194            academic_session = grok.getSite()['configuration'][session]
195        except KeyError:
196            d['error'] = u'Session configuration object is not available.'
197            return d
198        d['surcharge_1'] = academic_session.surcharge_1
199        d['surcharge_2'] = academic_session.surcharge_2
200        d['surcharge_3'] = academic_session.surcharge_3
201        if category == 'schoolfee':
202            d['amount'] = academic_session.school_fee_base
203            d['p_item'] = student['studycourse'].certificate.code
204        elif category == 'clearance':
205            d['p_item'] = student['studycourse'].certificate.code
206            d['amount'] = academic_session.clearance_fee
207        elif category == 'bed_allocation':
[7186]208            d['p_item'] = self.getAccommodationDetails(student)['bt']
[7150]209            d['amount'] = academic_session.booking_fee
210        return d
[7019]211
[7186]212    def getAccommodationDetails(self, student):
[7150]213        d = {}
214        d['error'] = u''
[7365]215        site_configuration = grok.getSite()['configuration']
216        d['booking_session'] = site_configuration.accommodation_session
217        d['allowed_states'] = site_configuration.accommodation_states
[7150]218        # Determine bed type
219        studycourse = student['studycourse']
[7369]220        certificate = getattr(studycourse,'certificate',None)
[7150]221        entry_session = studycourse.entry_session
222        current_level = studycourse.current_level
[7369]223        if not (entry_session and current_level and certificate):
224            return
225        end_level = certificate.end_level
[7150]226        if entry_session == grok.getSite()['configuration'].accommodation_session:
227            bt = 'fr'
228        elif current_level >= end_level:
229            bt = 'fi'
230        else:
231            bt = 're'
232        if student.sex == 'f':
233            sex = 'female'
234        else:
235            sex = 'male'
236        special_handling = 'regular'
237        d['bt'] = u'%s_%s_%s' % (special_handling,sex,bt)
238        return d
[7019]239
[7150]240    # In the standard configuration we select the first bed found,
241    # but can also randomize the selection if we like.
[7186]242    def selectBed(self, available_beds):
[7150]243        return available_beds[0]
244
[7318]245    def renderPDF(self, view, filename='slip.pdf',
[7304]246        student=None, studentview=None, tableheader=None, tabledata=None):
[7150]247        # (0,0),(-1,-1) = whole table
248        # (0,0),(0,-1) = first column
249        # (-1,0),(-1,-1) = last column
250        # (0,0),(-1,0) = first row
251        # (0,-1),(-1,-1) = last row
252
253        pdf = canvas.Canvas(filename,pagesize=A4)
254        pdf.setTitle(view.label)
[7318]255        pdf.setSubject(view.title)
[7150]256        pdf.setAuthor('%s (%s)' % (view.request.principal.title,
257            view.request.principal.id))
258        pdf.setCreator('WAeUP SIRP')
259        width, height = A4
[7310]260        footer_text = view.label
[7714]261        if getattr(student, 'student_id', None) is not None:
[7310]262            footer_text = "%s - %s - " % (student.student_id, footer_text)
[7150]263        style = getSampleStyleSheet()
[7310]264        style.add(ParagraphStyle(name='Right', alignment=TA_RIGHT))
[7150]265        pdf.line(1*cm,height-(1.8*cm),width-(1*cm),height-(1.8*cm))
266
267        story = []
268        frame_header = Frame(1*cm,1*cm,width-(1.7*cm),height-(1.7*cm))
269        header_title = getattr(grok.getSite(), 'name', u'Sample University')
270        story.append(Paragraph(header_title, style["Heading1"]))
271        frame_header.addFromList(story,pdf)
272
[7318]273        # Insert student data table
[7150]274        story = []
275        frame_body = Frame(1*cm,1*cm,width-(2*cm),height-(3.5*cm))
276        story.append(Paragraph(view.label, style["Heading2"]))
[7714]277        portal_language = getUtility(ISIRPUtils).PORTAL_LANGUAGE
[7310]278        if student is not None:
[7318]279            #story.append(Spacer(1, 12))
[7714]280            bd_translation = translate(_('Base Data'),
281                'waeup.sirp', target_language=portal_language)
282            story.append(Paragraph(bd_translation, style["Heading3"]))
[7310]283            studenttable = render_student_data(studentview)
[7150]284            story.append(studenttable)
[7304]285
[7318]286        # Insert widgets
287        story.append(Paragraph(view.title, style["Heading3"]))
[7186]288        set_up_widgets(view)
[7150]289        data = []
[7714]290        portal_language = getUtility(ISIRPUtils).PORTAL_LANGUAGE
[7150]291        for widget in view.widgets:
[7714]292            f_label = '<font size=12>%s</font>:' % translate(
293                widget.label.strip(), 'waeup.sirp',
294                target_language=portal_language)
[7150]295            f_label = Paragraph(f_label, style["Normal"])
[7714]296            f_text = formatted_text(widget(), size=12)
[7150]297            f_text = Paragraph(f_text, style["Normal"])
298            data.append([f_label,f_text])
[7318]299        table = Table(data,style=SLIP_STYLE, colWidths=[5*cm, 13.5*cm])
300        story.append(table)
301
302        # Import browser components locally
[7280]303        # to avoid circular import
304        from waeup.sirp.students.viewlets import FileManager
305        from waeup.sirp.browser import DEFAULT_IMAGE_PATH
[7318]306        data = []
307        # Collect viewlets
[7280]308        fm = FileManager(view.context, view.request, view)
309        fm.update()
[7318]310        if fm.viewlets:
[7741]311            sc_translation = translate(_('Scanned Documents'), 'waeup.sirp',
312                target_language=portal_language)
313            story.append(Paragraph(sc_translation, style["Heading3"]))
[7318]314            # Insert list of scanned documents
315            for viewlet in fm.viewlets:
[7741]316                f_label = formatted_label(size=12) % translate(
317                viewlet.label, 'waeup.sirp',
318                target_language=portal_language)
[7318]319                f_label = Paragraph(f_label, style["Normal"])
320                if viewlet.template.__grok_name__ == 'imagedisplay':
321                    # In case we define FileDisplay viewlets with
322                    # an imagedisplay template in the customization package
323                    img = getUtility(IExtFileStore).getFileByContext(
324                        view.context, attr=viewlet.download_name)
325                    if img is None:
326                        img = open(DEFAULT_IMAGE_PATH, 'rb')
327                    doc_img = Image(img.name, width=2*cm, height=1*cm, kind='bound')
328                    data.append([f_label,doc_img])
329                else:
[7714]330                    f_text = formatted_text(viewlet.title, size=12)
[7318]331                    f_text = Paragraph(f_text, style["Normal"])
332                    data.append([f_label,f_text])
333            table = Table(data,style=SLIP_STYLE, colWidths=[5*cm, 13.5*cm])
334            story.append(table)
335
[7280]336        try:
337            frame_body.addFromList(story,pdf)
338        except IOError:
339            view.flash('Error in image file.')
340            return view.redirect(view.url(view.context))
[7318]341
[7304]342        if tabledata and tableheader:
[7318]343            insert_footer(pdf,width,style,footer_text,number_of_pages=2)
344        else:
345            insert_footer(pdf,width,style,footer_text)
346
347        # Insert content table on second page
348        if tabledata and tableheader:
349            pdf.showPage()
350            frame_body = Frame(1*cm,1*cm,width-(2*cm),height-(2*cm))
[7304]351            story = []
[7318]352            story.append(Paragraph(view.content_title, style["Heading3"]))
[7310]353            #story.append(Spacer(1, 12))
[7304]354            contenttable = render_table_data(tableheader,tabledata)
355            story.append(contenttable)
356            frame_body.addFromList(story,pdf)
[7318]357            insert_footer(pdf,width,style,footer_text,number_of_pages=2)
358
[7150]359        view.response.setHeader(
360            'Content-Type', 'application/pdf')
361        return pdf.getpdfdata()
[7620]362
363    def getVerdictsDict(self):
364        """Provide a dictionary of verdicts.
365        """
366        return {
367            '0': 'not yet',
368            'A': 'Successful student',
[7624]369            'B': 'Student with carryover courses',
[7620]370            'C': 'Student on probation',
[7624]371            'D': 'Withdrawn from the faculty',
[7620]372            'E': 'Student who were previously on probation',
[7624]373            'F': 'Medical case',
[7620]374            'G': 'Absent from examination',
[7624]375            'H': 'Withheld results',
[7620]376            'I': 'Expelled/rusticated/suspended student',
[7624]377            'J': 'Temporary withdrawn from the university',
[7620]378            'K': 'Unregistered student',
[7624]379            'L': 'Referred student',
[7620]380            'M': 'Reinstatement',
[7624]381            'N': 'Student on transfer',
[7620]382            'O': 'NCE-III repeater',
383            'Y': 'No previous verdict',
384            'X': 'New 300 level student',
[7626]385            'Z': 'Successful student (provisional)',
[7624]386            'A1': 'First Class',
387            'A2': 'Second Class Upper',
388            'A3': 'Second Class Lower',
389            'A4': 'Third Class',
390            'A5': 'Pass',
391            'A6': 'Distinction',
392            'A7': 'Credit',
393            'A8': 'Merit',
394            }
Note: See TracBrowser for help on using the repository browser.