## $Id: utils.py 8820 2012-06-27 07:15:21Z henrik $
##
## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
## This program is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation; either version 2 of the License, or
## (at your option) any later version.
##
## This program is distributed in the hope that it will be useful,
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
## GNU General Public License for more details.
##
## You should have received a copy of the GNU General Public License
## along with this program; if not, write to the Free Software
## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
##
"""General helper functions and utilities for the student section.
"""
import grok
from random import SystemRandom as r
from time import time
from datetime import datetime
from zope.i18n import translate
from zope.component import getUtility, createObject
from reportlab.pdfgen import canvas
from reportlab.lib import colors
from reportlab.lib.units import cm
from reportlab.lib.enums import TA_RIGHT
from reportlab.lib.pagesizes import A4
from reportlab.lib.styles import getSampleStyleSheet, ParagraphStyle
from reportlab.platypus import (Frame, Paragraph, Image, PageBreak, Table,
Spacer)
from reportlab.platypus.tables import TableStyle
from reportlab.platypus.flowables import PageBreak
from zope.component import getUtility
from zope.formlib.form import setUpEditWidgets
from waeup.kofa.interfaces import (
IExtFileStore, IKofaUtils, RETURNING, PAID, CLEARED)
from waeup.kofa.interfaces import MessageFactory as _
from waeup.kofa.students.interfaces import IStudentsUtils
from waeup.kofa.utils.helpers import now
SLIP_STYLE = [
('VALIGN',(0,0),(-1,-1),'TOP'),
#('FONT', (0,0), (-1,-1), 'Helvetica', 11),
]
CONTENT_STYLE = [
('VALIGN',(0,0),(-1,-1),'TOP'),
#('FONT', (0,0), (-1,-1), 'Helvetica', 8),
#('TEXTCOLOR',(0,0),(-1,0),colors.white),
('BACKGROUND',(0,0),(-1,0),colors.black),
]
FONT_SIZE = 10
FONT_COLOR = 'black'
def formatted_label(color=FONT_COLOR, size=FONT_SIZE):
tag1 ='' % (color, size)
return tag1 + '%s:'
def trans(text, lang):
# shortcut
return translate(text, 'waeup.kofa', target_language=lang)
def formatted_text(text, color=FONT_COLOR, size=FONT_SIZE):
"""Turn `text`, `color` and `size` into an HTML snippet.
The snippet is suitable for use with reportlab and generating PDFs.
Wraps the `text` into a ```` tag with passed attributes.
Also non-strings are converted. Raw strings are expected to be
utf-8 encoded (usually the case for widgets etc.).
Finally, a br tag is added if widgets contain div tags
which are not supported by reportlab.
The returned snippet is unicode type.
"""
try:
# In unit tests IKofaUtils has not been registered
portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
except:
portal_language = 'en'
if not isinstance(text, unicode):
if isinstance(text, basestring):
text = text.decode('utf-8')
else:
text = unicode(text)
# Mainly for boolean values we need our customized
# localisation of the zope domain
text = translate(text, 'zope', target_language=portal_language)
text = text.replace('', '
')
tag1 = u'' % (color, size)
return tag1 + u'%s' % text
def generate_student_id():
students = grok.getSite()['students']
new_id = students.unique_student_id
return new_id
def set_up_widgets(view, ignore_request=False):
view.adapters = {}
view.widgets = setUpEditWidgets(
view.form_fields, view.prefix, view.context, view.request,
adapters=view.adapters, for_display=True,
ignore_request=ignore_request
)
def render_student_data(studentview):
"""Render student table for an existing frame.
"""
width, height = A4
set_up_widgets(studentview, ignore_request=True)
data_left = []
data_right = []
style = getSampleStyleSheet()
img = getUtility(IExtFileStore).getFileByContext(
studentview.context, attr='passport.jpg')
if img is None:
from waeup.kofa.browser import DEFAULT_PASSPORT_IMAGE_PATH
img = open(DEFAULT_PASSPORT_IMAGE_PATH, 'rb')
doc_img = Image(img.name, width=4*cm, height=4*cm, kind='bound')
data_left.append([doc_img])
#data.append([Spacer(1, 12)])
portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
for widget in studentview.widgets:
if widget.name == 'form.adm_code':
continue
f_label = formatted_label(size=12) % translate(
widget.label.strip(), 'waeup.kofa',
target_language=portal_language)
f_label = Paragraph(f_label, style["Normal"])
f_text = formatted_text(widget(), size=12)
f_text = Paragraph(f_text, style["Normal"])
data_right.append([f_label,f_text])
table_left = Table(data_left,style=SLIP_STYLE)
table_right = Table(data_right,style=SLIP_STYLE, colWidths=[5*cm, 6*cm])
table = Table([[table_left, table_right],],style=SLIP_STYLE)
return table
def render_table_data(tableheader,tabledata):
"""Render children table for an existing frame.
"""
data = []
#data.append([Spacer(1, 12)])
line = []
style = getSampleStyleSheet()
for element in tableheader:
field = formatted_text(element[0], color='white')
field = Paragraph(field, style["Normal"])
line.append(field)
data.append(line)
for ticket in tabledata:
line = []
for element in tableheader:
field = formatted_text(getattr(ticket,element[1],u' '))
field = Paragraph(field, style["Normal"])
line.append(field)
data.append(line)
table = Table(data,colWidths=[
element[2]*cm for element in tableheader], style=CONTENT_STYLE)
return table
def docs_as_flowables(view, lang='en'):
"""Create reportlab flowables out of scanned docs.
"""
# XXX: fix circular import problem
from waeup.kofa.students.viewlets import FileManager
from waeup.kofa.browser import DEFAULT_IMAGE_PATH
from waeup.kofa.browser.pdf import NORMAL_STYLE, ENTRY1_STYLE
style = getSampleStyleSheet()
data = []
# Collect viewlets
fm = FileManager(view.context, view.request, view)
fm.update()
if fm.viewlets:
sc_translation = trans(_('Scanned Documents'), lang)
data.append(Paragraph(sc_translation, style["Heading3"]))
# Insert list of scanned documents
table_data = []
for viewlet in fm.viewlets:
f_label = Paragraph(trans(viewlet.label, lang), ENTRY1_STYLE)
img_path = getattr(getUtility(IExtFileStore).getFileByContext(
view.context, attr=viewlet.download_name), 'name', None)
f_text = Paragraph(trans(_('(not provided)'),lang), ENTRY1_STYLE)
if img_path is None:
pass
elif not img_path.endswith('.jpg'):
# reportlab requires jpg images, I think.
f_text = Paragraph('%s (Not displayable)' % (
viewlet.title,), ENTRY1_STYLE)
else:
f_text = Image(img_path, width=2*cm, height=1*cm, kind='bound')
table_data.append([f_label, f_text])
if table_data:
# safety belt; empty tables lead to problems.
data.append(Table(table_data, style=SLIP_STYLE))
return data
def insert_footer(pdf,width,style,text=None, number_of_pages=1):
"""Render the whole footer frame.
"""
story = []
frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
tz = getUtility(IKofaUtils).tzinfo
timestamp = now(tz).strftime("%d/%m/%Y %H:%M:%S %Z")
left_text = '%s' % timestamp
story.append(Paragraph(left_text, style["Normal"]))
frame_footer.addFromList(story,pdf)
story = []
frame_footer = Frame(1*cm,0,width-(2*cm),1*cm)
portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
right_text = translate(_('${a} Page ${b} of ${c}',
mapping = {'a':text, 'b':pdf.getPageNumber(), 'c':number_of_pages}),
'waeup.kofa', target_language=portal_language)
story.append(Paragraph(right_text, style["Right"]))
frame_footer.addFromList(story,pdf)
class StudentsUtils(grok.GlobalUtility):
"""A collection of methods subject to customization.
"""
grok.implements(IStudentsUtils)
def getReturningData(self, student):
""" This method defines what happens after school fee payment
depending on the student's senate verdict.
In the base configuration current level is always increased
by 100 no matter which verdict has been assigned.
"""
new_level = student['studycourse'].current_level + 100
new_session = student['studycourse'].current_session + 1
return new_session, new_level
def setReturningData(self, student):
""" This method defines what happens after school fee payment
depending on the student's senate verdict. It folllows
the same algorithm as getReturningData but it also sets the new
values
In the base configuration current level is always increased
by 100 no matter which verdict has been assigned.
"""
new_session, new_level = self.getReturningData(student)
student['studycourse'].current_level = new_level
student['studycourse'].current_session = new_session
verdict = student['studycourse'].current_verdict
student['studycourse'].current_verdict = '0'
student['studycourse'].previous_verdict = verdict
return
def setPaymentDetails(self, category, student):
"""Create Payment object and set the payment data of a student for
the payment category specified.
"""
details = {}
p_item = u''
amount = 0.0
error = u''
p_session = student['studycourse'].current_session
p_level = student['studycourse'].current_level
session = str(p_session)
try:
academic_session = grok.getSite()['configuration'][session]
except KeyError:
return _(u'Session configuration object is not available.'), None
if category == 'schoolfee':
try:
certificate = student['studycourse'].certificate
p_item = certificate.code
except (AttributeError, TypeError):
return _('Study course data are incomplete.'), None
if student.state == CLEARED:
amount = getattr(certificate, 'school_fee_1', 0.0)
elif student.state == RETURNING:
# In case of returning school fee payment the payment session
# and level contain the values of the session the student
# has paid for.
p_session, p_level = self.getReturningData(student)
amount = getattr(certificate, 'school_fee_2', 0.0)
elif student.is_postgrad and student.state == PAID:
# Returning postgraduate students also pay for the next session
# but their level always remains the same.
p_session += 1
amount = getattr(certificate, 'school_fee_2', 0.0)
elif category == 'clearance':
p_item = student['studycourse'].certificate.code
amount = academic_session.clearance_fee
elif category == 'bed_allocation':
p_item = self.getAccommodationDetails(student)['bt']
amount = academic_session.booking_fee
if amount in (0.0, None):
return _(u'Amount could not be determined.'), None
for key in student['payments'].keys():
ticket = student['payments'][key]
if ticket.p_state == 'paid' and\
ticket.p_category == category and \
ticket.p_item == p_item and \
ticket.p_session == p_session:
return _('This type of payment has already been made.'), None
payment = createObject(u'waeup.StudentOnlinePayment')
timestamp = "%d" % int(time()*1000)
payment.p_id = "p%s" % timestamp
payment.p_category = category
payment.p_item = p_item
payment.p_session = p_session
payment.p_level = p_level
payment.amount_auth = amount
return None, payment
def getAccommodationDetails(self, student):
"""Determine the accommodation dates of a student.
"""
d = {}
d['error'] = u''
hostels = grok.getSite()['hostels']
d['booking_session'] = hostels.accommodation_session
d['allowed_states'] = hostels.accommodation_states
d['startdate'] = hostels.startdate
d['enddate'] = hostels.enddate
d['expired'] = hostels.expired
# Determine bed type
studycourse = student['studycourse']
certificate = getattr(studycourse,'certificate',None)
entry_session = studycourse.entry_session
current_level = studycourse.current_level
if not (entry_session and current_level and certificate):
return
end_level = certificate.end_level
if entry_session == grok.getSite()['hostels'].accommodation_session:
bt = 'fr'
elif current_level >= end_level:
bt = 'fi'
else:
bt = 're'
if student.sex == 'f':
sex = 'female'
else:
sex = 'male'
special_handling = 'regular'
d['bt'] = u'%s_%s_%s' % (special_handling,sex,bt)
return d
def selectBed(self, available_beds):
"""Select a bed from a list of available beds.
In the base configuration we select the first bed found,
but can also randomize the selection if we like.
"""
return available_beds[0]
def renderPDF(self, view, filename='slip.pdf', student=None,
studentview=None, tableheader=None, tabledata=None,
note=None):
"""Render pdf slips for various pages.
"""
# XXX: we have to fix the import problems here.
from waeup.kofa.browser.interfaces import IPDFCreator
from waeup.kofa.browser.pdf import NORMAL_STYLE, ENTRY1_STYLE
style = getSampleStyleSheet()
creator = getUtility(IPDFCreator)
data = []
doc_title = view.label
author = '%s (%s)' % (view.request.principal.title,
view.request.principal.id)
footer_text = view.label
if getattr(student, 'student_id', None) is not None:
footer_text = "%s - %s - " % (student.student_id, footer_text)
# Insert student data table
portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
if student is not None:
bd_translation = trans(_('Base Data'), portal_language)
data.append(Paragraph(bd_translation, style["Heading3"]))
data.append(render_student_data(studentview))
# Insert widgets
data.append(Paragraph(view.title, style["Heading3"]))
portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
separators = getattr(self, 'SEPARATORS_DICT', {})
table = creator.getWidgetsTable(
view.form_fields, view.context, None, lang=portal_language,
separators=separators)
data.append(table)
# Insert scanned docs
data.extend(docs_as_flowables(view, portal_language))
# Insert content table (optionally on second page)
if tabledata and tableheader:
#data.append(PageBreak())
data.append(Spacer(1, 20))
data.append(Paragraph(view.content_title, style["Heading3"]))
contenttable = render_table_data(tableheader,tabledata)
data.append(contenttable)
view.response.setHeader(
'Content-Type', 'application/pdf')
try:
pdf_stream = creator.create_pdf(
data, None, doc_title, author=author, footer=footer_text,
note=note)
except IOError:
view.flash('Error in image file.')
return view.redirect(view.url(view.context))
return pdf_stream
VERDICTS_DICT = {
'0': _('(not yet)'),
'A': 'Successful student',
'B': 'Student with carryover courses',
'C': 'Student on probation',
}
SEPARATORS_DICT = {
}
#: A prefix used when generating new student ids. Each student id will
#: start with this string. The default is 'K' for ``Kofa``.
STUDENT_ID_PREFIX = u'K'